Swing/DragRowsAnotherTable のバックアップ(No.23)
- バックアップ一覧
 - 差分 を表示
 - 現在との差分 を表示
 - 現在との差分 - Visual を表示
 - ソース を表示
 - Swing/DragRowsAnotherTable へ行く。
  
- 1 (2009-09-28 (月) 14:28:36)
 - 2 (2009-09-30 (水) 12:43:21)
 - 3 (2009-11-16 (月) 10:51:53)
 - 4 (2010-12-20 (月) 22:02:47)
 - 5 (2013-01-05 (土) 19:55:46)
 - 6 (2014-02-20 (木) 19:52:46)
 - 7 (2014-03-18 (火) 18:52:24)
 - 8 (2014-05-02 (金) 12:45:50)
 - 9 (2014-09-30 (火) 00:50:15)
 - 10 (2014-11-30 (日) 00:54:45)
 - 11 (2015-01-03 (土) 06:19:07)
 - 12 (2015-03-18 (水) 18:48:37)
 - 13 (2015-12-15 (火) 17:06:08)
 - 14 (2017-06-02 (金) 15:28:43)
 - 15 (2017-08-15 (火) 14:46:43)
 - 16 (2017-11-17 (金) 17:46:30)
 - 17 (2018-02-24 (土) 19:51:30)
 - 18 (2019-05-11 (土) 15:54:34)
 - 19 (2019-05-12 (日) 18:45:55)
 - 20 (2019-09-26 (木) 20:50:29)
 - 21 (2019-10-15 (火) 17:34:55)
 - 22 (2019-10-23 (水) 21:26:55)
 - 23 (2021-05-23 (日) 12:42:25)
 - 24 (2025-01-03 (金) 08:57:02)
 - 25 (2025-01-03 (金) 09:01:23)
 - 26 (2025-01-03 (金) 09:02:38)
 - 27 (2025-01-03 (金) 09:03:21)
 - 28 (2025-01-03 (金) 09:04:02)
 - 29 (2025-06-19 (木) 12:41:37)
 - 30 (2025-06-19 (木) 12:43:47)
 
 
- category: swing
folder: DragRowsAnotherTable
title: JTableの行を別のJTableにドラッグして移動
tags: [JTable, DragAndDrop, TransferHandler, Cursor]
author: aterai
pubdate: 2009-09-28T14:28:36+09:00
description: JTableの行を別のJTableにDrag&Dropで移動します。
image: 
hreflang:
href: https://java-swing-tips.blogspot.com/2009/09/drag-rows-from-one-jtable-to-another.html lang: en
 
概要
JTableの行を別のJTableにDrag & Dropで移動します。
Screenshot

Advertisement
サンプルコード
class TableRowTransferHandler extends TransferHandler {
  private final DataFlavor localObjectFlavor;
  private int[] indices;
  private int addIndex = -1; //Location where items were added
  private int addCount; //Number of items added.
  private JComponent source;
  public TableRowTransferHandler() {
    super();
    localObjectFlavor = new DataFlavor(Object[].class, "Array of items");
  }
  @Override protected Transferable createTransferable(JComponent c) {
    source = c;
    JTable table = (JTable) c;
    DefaultTableModel model = (DefaultTableModel) table.getModel();
    List<Object> list = new ArrayList<>();
    indices = table.getSelectedRows();
    for (int i : indices) {
      list.add(model.getDataVector().elementAt(i));
    }
    Object[] transferredObjects = list.toArray();
    return new Transferable() {
      @Override public DataFlavor[] getTransferDataFlavors() {
        return new DataFlavor[] {FLAVOR};
      }
      @Override public boolean isDataFlavorSupported(DataFlavor flavor) {
        return Objects.equals(FLAVOR, flavor);
      }
      @Override public Object getTransferData(DataFlavor flavor)
            throws UnsupportedFlavorException, IOException {
        if (isDataFlavorSupported(flavor)) {
          return nodes;
        } else {
          throw new UnsupportedFlavorException(flavor);
        }
      }
    };
  }
  @Override public boolean canImport(TransferSupport info) {
    JTable table = (JTable) info.getComponent();
    boolean isDroppable = info.isDrop()
      && info.isDataFlavorSupported(localObjectFlavor);
    //XXX bug?
    table.setCursor(isDroppable ? DragSource.DefaultMoveDrop
                                : DragSource.DefaultMoveNoDrop);
    return isDroppable;
  }
  @Override public int getSourceActions(JComponent c) {
    return TransferHandler.MOVE; //TransferHandler.COPY_OR_MOVE;
  }
  @Override public boolean importData(TransferSupport info) {
    TransferHandler.DropLocation tdl = info.getDropLocation();
    if (!(tdl instanceof JTable.DropLocation)) {
      return false;
    }
    JTable.DropLocation dl = (JTable.DropLocation) tdl;
    JTable target = (JTable) info.getComponent();
    DefaultTableModel model = (DefaultTableModel) target.getModel();
    int index = dl.getRow();
    //boolean insert = dl.isInsert();
    int max = model.getRowCount();
    if (index < 0 || index > max) {
      index = max;
    }
    addIndex = index;
    target.setCursor(Cursor.getDefaultCursor());
    try {
      Object[] values =
        (Object[]) info.getTransferable().getTransferData(localObjectFlavor);
      if (Objects.equals(source, target)) {
        addCount = values.length;
      }
      for (int i = 0; i < values.length; i++) {
        int idx = index++;
        // model.insertRow(idx, (Vector<?>) values[i]);
        model.insertRow(i, ((List<?>) o).toArray(new Object[0]));
        target.getSelectionModel().addSelectionInterval(idx, idx);
      }
      return true;
    } catch (UnsupportedFlavorException | IOException ex) {
      ex.printStackTrace();
    }
    return false;
  }
  @Override protected void exportDone(
      JComponent c, Transferable data, int action) {
    cleanup(c, action == MOVE);
  }
  private void cleanup(JComponent c, boolean remove) {
    if (remove && indices != null) {
      c.setCursor(Cursor.getDefaultCursor());
      DefaultTableModel model = (DefaultTableModel) ((JTable) c).getModel();
      if (addCount > 0) {
        for (int i = 0; i < indices.length; i++) {
          if (indices[i] >= addIndex) {
            indices[i] += addCount;
          }
        }
      }
      for (int i = indices.length - 1; i >= 0; i--) {
        model.removeRow(indices[i]);
      }
    }
    indices = null;
    addCount = 0;
    addIndex = -1;
  }
}
View in GitHub: Java, Kotlin解説
上記のサンプルでは、1つのJTable内で行の並べ替えを行うTransferHandlerを使ってJTableの行をドラッグ&ドロップ、並べ替えを元に、複数JTable間で行移動が可能になるようTableRowTransferHandlerを拡張しています。
以下のように、JTable#setFillsViewportHeight(true)でJTable自体の高さを拡張しておかないと、JTableが空の状態でドロップが不可になります。
TransferHandler handler = new TableRowTransferHandler();
table.getSelectionModel().setSelectionMode(
    ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
table.setTransferHandler(handler);
table.setDropMode(DropMode.INSERT_ROWS);
table.setDragEnabled(true);
table.setFillsViewportHeight(true);
WindowsLookAndFeelでカーソルのチラつき防止のために、TransferHandler#canImport(...)内でJTable#setCursor(...)をしているため、デスクトップなどからファイルをドラッグしてドロップ不可カーソルが表示されると、その後マウスをリリースしてもカーソルが表示されたままになるバグがあります。
また、このサンプルでは、各JTableにTableRowSorterなどが設定され、ソートされた状態での並べ替えは想定していません。