實現 Swing 的 JTables 和 Excel 間的複製和粘貼功能

實現 Swing 的 JTables 和 Excel 間的複製和粘貼功能 

當今很多業務應用程序都是使用 Java 開發的,並且以後這種用 Java 開發的應用會更多。而在許多應用程序都使用了 Swing 的 JTable 組件,以類似電子表格的格式顯示數據。如果業務應用程序可以將數據導入 Microsoft Excel 和從 Microsoft Excel 中導出數據,則會爲用戶帶來方便,使用戶可以使用無處不在的該電子表格程序的強大功能。此 Java 技巧將幫助您理解系統剪貼板,並使您的 JTable 能夠與 Excel 交互和互操作。您將看到,通過在當前應用程序中僅僅添加另外的一行代碼即可實現這一有用功能的添加。

要實現這一目標,需要做的只是複製這裏給出的文件 ExcelAdapter.java,對其進行編譯,並確保您的應用程序可以找到 ExcelAdapter.class 文件;做完後,JTable 就可以與 Excel 進行通話了!我們將向您展示僅通過這一行代碼,如何實現到 Excel 的複製 (Ctrl+C) 和粘貼 (Ctrl+V) 以及從 Excel 進行的複製和粘貼。另外還提供了一個使用 ExcelAdapter 的示例應用程序。

代碼
下面是實際完成此任務的適配器代碼(稱爲 ExcelAdapter.java):

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import java.awt.datatransfer.*;
import java.util.*;

/**
 * ExcelAdapter 實現 JTables 中的複製粘貼
 * 剪貼板功能。 適配器所用的剪貼板數據格式
 * 與 Excel 所用的剪貼板格式兼容。這提供了
 * 支持的 JTables 和 Excel 間的互操作。
 */
public class ExcelAdapter implements ActionListener
   {
   private String rowstring,value;
   private Clipboard system;
   private StringSelection stsel;
   private JTable jTable1 ;


   /**   
 * Excel 適配器由 JTable 構成,
 * 它實現了 JTable 上的複製粘貼
 * 功能,並充當剪貼板監聽程序。
    */


public ExcelAdapter(JTable myJTable)
   {
      jTable1 = myJTable;
      KeyStroke copy = KeyStroke.getKeyStroke(KeyEvent.VK_C,ActionEvent.CTRL_MASK,false);

      // 確定複製按鍵用戶可以對其進行修改
      // 以實現其它按鍵組合的複製功能。
      KeyStroke paste = KeyStroke.getKeyStroke(KeyEvent.VK_V,ActionEvent.CTRL_MASK,false);

      // 確定粘貼按鍵用戶可以對其進行修改
      // 以實現其它按鍵組合的複製功能。

jTable1.registerKeyboardAction(this,"Copy",copy,JComponent.WHEN_FOCUSED);


jTable1.registerKeyboardAction(this,"Paste",paste,JComponent.WHEN_FOCUSED);

      system = Toolkit.getDefaultToolkit().getSystemClipboard();
   }

   /**
    * 此適配器運行圖表的公共讀方法。
    */
public JTable getJTable() {return jTable1;}

public void setJTable(JTable jTable1) {this.jTable1=jTable1;}

   /**
 * 在我們監聽此實現的按鍵上激活這種方法。
 * 此處,它監聽複製和粘貼 ActionCommands。
 * 包含不相鄰單元格的選擇導致選擇無效,
 * 而且此後複製動作無法執行。
 * 粘貼的方法是將選定內容的左上角與
        * JTable 的當前選定內容的第一個元素對齊。
    */
public void actionPerformed(ActionEvent e)
   {
      if (e.getActionCommand().compareTo("Copy")==0)
      {
         StringBuffer sbf=new StringBuffer();

         // 檢查以確保我們僅選擇了單元格的
         // 相鄰塊
         int numcols=jTable1.getSelectedColumnCount();
         int numrows=jTable1.getSelectedRowCount();
         int[] rowsselected=jTable1.getSelectedRows();
         int[] colsselected=jTable1.getSelectedColumns();

         if (!((numrows-1==rowsselected[rowsselected.length-1]-rowsselected[0] &&
                numrows==rowsselected.length) &&

(numcols-1==colsselected[colsselected.length-1]-colsselected[0] &&
                numcols==colsselected.length)))
         {
            JOptionPane.showMessageDialog(null, "Invalid Copy Selection",
                                          "Invalid Copy Selection",
                                          JOptionPane.ERROR_MESSAGE);

            return;
         }

         for (int i=0;i<numrows;i++)
         {
            for (int j=0;j<numcols;j++)
            {

sbf.append(jTable1.getValueAt(rowsselected[i],colsselected[j]));
               if (j<numcols-1) sbf.append("/t");
            }
            sbf.append("/n");
         }

         stsel  = new StringSelection(sbf.toString());
         system = Toolkit.getDefaultToolkit().getSystemClipboard();
         system.setContents(stsel,stsel);
      }


      if (e.getActionCommand().compareTo("Paste")==0)
      {
          System.out.println("Trying to Paste");
          int startRow=(jTable1.getSelectedRows())[0];
          int startCol=(jTable1.getSelectedColumns())[0];
          try
          {
             String trstring= (String)(system.getContents(this).getTransferData(DataFlavor.stringFlavor));

             System.out.println("String is:"+trstring);
             StringTokenizer st1=new StringTokenizer(trstring,"/n");
             for(int i=0;st1.hasMoreTokens();i++)
             {
                rowstring=st1.nextToken();
                StringTokenizer st2=new StringTokenizer(rowstring,"/t");

                for(int j=0;st2.hasMoreTokens();j++)
                {
                   value=(String)st2.nextToken();
                   if (startRow+i< jTable1.getRowCount()  &&
                       startCol+j< jTable1.getColumnCount())
                      jTable1.setValueAt(value,startRow+i,startCol+j);
                   System.out.println("Putting "+ value+"at
row="+startRow+i+"column="+startCol+j);
               }
            }
         }
         catch(Exception ex){ex.printStackTrace();}

      }
   }
}

示例應用程序
下面是示例應用程序 Frame1.java,它使用 ExcelAdapter 實現 JTable 與 Excel 的兼容。

import java.awt.*;
import javax.swing.*;

public class Frame1 extends Frame
   {
   BorderLayout borderLayout1 = new BorderLayout();
   JTable jTable1 ;
   Object[][] data=new Object[4][4];
   Object header[]= {"Jan","Feb","Mar","Apr"};

public static void main(String args[])
   {
   Frame1 myframe=new Frame1();
   myframe.setSize(new Dimension(250,250));
   myframe.setVisible(true);
   }


public Frame1()
   {
      super();
      try
      {
         jbInit();
      }
      catch (Exception e)
      {
         e.printStackTrace();
      }
   }

private void jbInit() throws Exception
   {
   for (int i=0;i<4;i++)
      for (int j=0;j<4;j++)
         data[i][j]=new Integer(i*10+j);

   System.out.println("Header length="+header[1]);
   jTable1=new JTable(data,header);
   jTable1.setCellSelectionEnabled(true);
   this.setTitle("Excel Lent JTABLE");
   jTable1.setBackground(Color.pink);
   this.setLayout(borderLayout1);
   this.setSize(new Dimension(400, 300));

   this.setBackground(Color.white);
   this.add(jTable1, BorderLayout.CENTER);

   // 這就是添加複製和粘貼功能的那一行!
   ExcelAdapter myAd = new ExcelAdapter(jTable1);
   }
}

Excel 的剪貼板格式
Excel 的剪貼板格式非常簡單。它採用製表符分隔同一行上的元素,並用換行符分隔行。這樣,當您複製一組連續的和/或相鄰的單元格時,Excel 只將電子表格數據標記到一個長字符串中,各個單元格值由該字符串內的製表符和換行符分隔。如果所選的單元格不相鄰時怎麼辦?很簡單:Excel 不會讓您將所選內容複製到剪貼板。這種行爲由本文所說明的適配器模仿,如果所選擇的單元格不相鄰,也不會使您複製數據。在 Excel 中,會彈出一個對話框告訴我們不允許複製;這一行爲由適配器再次模仿。

代碼簡單解釋
要使用這一功能,您需要下載 ExcelAdapter.java 文件,對其進行編譯,將示例應用程序中的最後一行添加到您代碼中的某個位置,以激活 JTable 上的適配器。

在適配器中,複製和粘貼功能的激活按鍵都經過了註冊。其後,無論何時鍵入激活的按鍵, actionPerformed 方法就會被調用。如果是複製操作,則所選擇的單元格數據都會爲 Excel 進行適當的標記,並寫入系統剪貼板中。如果是粘貼操作,則將系統剪貼板中的數據轉換爲字符串形式,並經過分析,根據所選擇的單元格植入 JTables 單元格中。

結論
使用用 Java 編寫的表格的用戶常常希望 Jtables 中的數據與 Excel 中的數據處理方法相似。它並不是編寫代碼來實現每個 JTable 上的複製-粘貼功能,而是更爲簡單地編寫通用的適配器,只需添加一行代碼即可將所需功能添加到每個 Jtable 中。此 Java Tip 中提供的適配器正好做到這一點。



Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=579643

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章