Swing/DragRowsAnotherTable のバックアップ(No.17)
- バックアップ一覧
- 差分 を表示
- 現在との差分 を表示
- 現在との差分 - 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)
- 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: http://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[] transferedObjects = 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) {
if (!canImport(info)) {
return false;
}
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.getPredefinedCursor(Cursor.DEFAULT_CURSOR));
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]);
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.getPredefinedCursor(Cursor.DEFAULT_CURSOR));
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
などが設定され、ソートされた状態での並べ替えは想定していません。