Swing/DragRowsAnotherTable の変更点
- 追加された行はこの色です。
- 削除された行はこの色です。
- Swing/DragRowsAnotherTable へ行く。
- Swing/DragRowsAnotherTable の差分を削除
--- 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: https://lh3.googleusercontent.com/_9Z4BYR88imo/TQTL05H70DI/AAAAAAAAAYM/YtTZHzrA2HU/s800/DragRowsAnotherTable.png hreflang: href: https://java-swing-tips.blogspot.com/2009/09/drag-rows-from-one-jtable-to-another.html lang: en --- * 概要 [#summary] `JTable`の行を別の`JTable`に`Drag & Drop`で移動します。 #download(https://lh3.googleusercontent.com/_9Z4BYR88imo/TQTL05H70DI/AAAAAAAAAYM/YtTZHzrA2HU/s800/DragRowsAnotherTable.png) * サンプルコード [#sourcecode] #code(link){{ 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; } } }} * 解説 [#explanation] 上記のサンプルでは、`1`つの`JTable`内で行の並べ替えを行う[[TransferHandlerを使ってJTableの行をドラッグ&ドロップ、並べ替え>Swing/DnDReorderTable]]を元に、複数`JTable`間で行移動が可能になるよう`TableRowTransferHandler`を拡張しています。 ---- 以下のように、`JTable#setFillsViewportHeight(true)`で[[JTable自体の高さを拡張>Swing/FillsViewportHeight]]しておかないと、`JTable`が空の状態でドロップが不可になります。 - 以下のように、`JTable#setFillsViewportHeight(true)`で[[JTable自体の高さを拡張>Swing/FillsViewportHeight]]しておかないと、`JTable`が空の状態でドロップが不可になる #code{{ 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(...)`をしているため、デスクトップなどからファイルをドラッグしてドロップ不可カーソルが表示されると、その後マウスをリリースしてもカーソルが表示されたままになるバグがあります。 - 制限: -- `WindowsLookAndFeel`でのカーソルのチラつき防止のために、`TransferHandler#canImport(...)`内で`JTable#setCursor(...)`をしているため、デスクトップなどからファイルをドラッグしてドロップ不可カーソルが表示されると、その後マウスをリリースしてもカーソルが表示されたままになるバグがある -- このサンプルでは、各`JTable`に`TableRowSorter`などが設定され、ソートされた状態での並べ替えは想定していない また、このサンプルでは、各`JTable`に`TableRowSorter`などが設定され、ソートされた状態での並べ替えは想定していません。 * 参考リンク [#reference] - [[JTableの行をドラッグ&ドロップ>Swing/DnDTable]] - [[TransferHandlerを使ってJTableの行をドラッグ&ドロップ、並べ替え>Swing/DnDReorderTable]] * コメント [#comment] #comment #comment