Swing/DragRowsAnotherTable のバックアップの現在との差分(No.11)
-
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 href: https://java-swing-tips.blogspot.com/2009/09/drag-rows-from-one-jtable-to-another.html lang: en
概要
JTable
の行を別のJTable
にDrag&Drop
で移動します。
概要
JTable
の行を別のJTable
にDrag & Drop
で移動します。
Screenshot
Advertisement
サンプルコード
サンプルコード
class TableRowTransferHandler extends TransferHandler {
private int[] rows = null;
private int addIndex = -1; //Location where items were added
private int addCount = 0; //Number of items added.
private final DataFlavor localObjectFlavor;
private Object[] transferedObjects = null;
private JComponent source = null;
private int[] indices;
private int addIndex = -1; // Location where items were added
private int addCount; // Number of items added.
private JComponent source;
#spanadd
#spanend
public TableRowTransferHandler() {
localObjectFlavor = new ActivationDataFlavor(
Object[].class, DataFlavor.javaJVMLocalObjectMimeType, "Array of items");
super();
localObjectFlavor = new DataFlavor(Object[].class, "Array of items");
}
#spanadd
#spanend
@Override protected Transferable createTransferable(JComponent c) {
source = c;
JTable table = (JTable) c;
DefaultTableModel model = (DefaultTableModel)table.getModel();
DefaultTableModel model = (DefaultTableModel) table.getModel();
List<Object> list = new ArrayList<>();
for(int i: rows = table.getSelectedRows())
indices = table.getSelectedRows();
for (int i : indices) {
list.add(model.getDataVector().elementAt(i));
transferedObjects = list.toArray();
return new DataHandler(transferedObjects,localObjectFlavor.getMimeType());
}
Object[] transferredObjects = list.toArray();
return new Transferable() {
@Override public DataFlavor[] getTransferDataFlavors() {
return new DataFlavor[] {FLAVOR};
}
#spanadd
#spanend
@Override public boolean isDataFlavorSupported(DataFlavor flavor) {
return Objects.equals(FLAVOR, flavor);
}
#spanadd
#spanend
@Override public Object getTransferData(DataFlavor flavor)
throws UnsupportedFlavorException, IOException {
if (isDataFlavorSupported(flavor)) {
return nodes;
} else {
throw new UnsupportedFlavorException(flavor);
}
}
};
}
#spanadd
#spanend
@Override public boolean canImport(TransferSupport info) {
JTable t = (JTable)info.getComponent();
boolean b = info.isDrop()&&info.isDataFlavorSupported(localObjectFlavor);
//XXX bug?
t.setCursor(b?DragSource.DefaultMoveDrop:DragSource.DefaultMoveNoDrop);
return b;
JTable table = (JTable) info.getComponent();
boolean isDroppable = info.isDrop()
&& info.isDataFlavorSupported(localObjectFlavor);
// XXX bug?
table.setCursor(isDroppable ? DragSource.DefaultMoveDrop
: DragSource.DefaultMoveNoDrop);
return isDroppable;
}
#spanadd
#spanend
@Override public int getSourceActions(JComponent c) {
return COPY_OR_MOVE;
return TransferHandler.MOVE; //TransferHandler.COPY_OR_MOVE;
}
#spanadd
#spanend
@Override public boolean importData(TransferSupport info) {
JTable target = (JTable)info.getComponent();
JTable.DropLocation dl = (JTable.DropLocation)info.getDropLocation();
DefaultTableModel model = (DefaultTableModel)target.getModel();
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;
if (index < 0 || index > max) {
index = max;
}
addIndex = index;
target.setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR));
target.setCursor(Cursor.getDefaultCursor());
try {
Object[] values =
(Object[])info.getTransferable().getTransferData(localObjectFlavor);
if(source==target) addCount = values.length;
for(int i=0;i<values.length;i++) {
(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(idx, (Vector<?>) values[i]);
model.insertRow(i, ((List<?>) o).toArray(new Object[0]));
target.getSelectionModel().addSelectionInterval(idx, idx);
}
return true;
}catch(Exception ufe) { ufe.printStackTrace(); }
} catch (UnsupportedFlavorException | IOException ex) {
ex.printStackTrace();
}
return false;
}
@Override protected void exportDone(JComponent c, Transferable t, int act) {
cleanup(c, act == MOVE);
#spanadd
#spanend
@Override protected void exportDone(
JComponent c, Transferable data, int action) {
cleanup(c, action == MOVE);
}
private void cleanup(JComponent src, boolean remove) {
if(remove && rows != null) {
JTable table = (JTable)src;
src.setCursor(Cursor.getPredefinedCursor(Cursor.DEFAULT_CURSOR));
DefaultTableModel model = (DefaultTableModel)table.getModel();
if(addCount > 0) {
for(int i=0;i<rows.length;i++) {
if(rows[i]>=addIndex) { rows[i] += addCount; }
#spanadd
#spanend
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=rows.length-1;i>=0;i--) model.removeRow(rows[i]);
for (int i = indices.length - 1; i >= 0; i--) {
model.removeRow(indices[i]);
}
}
rows = null;
indices = null;
addCount = 0;
addIndex = -1;
}
}
View in GitHub: Java, Kotlin解説
上記のサンプルでは、一つのJTable
内での行の並べ替えを行うTransferHandlerを使ってJTableの行をドラッグ&ドロップ、並べ替えを元にTableRowTransferHandler
を作成し、JTable
間での行移動もできるようになっています。
解説
上記のサンプルでは、1
つのJTable
内で行の並べ替えを行うTransferHandlerを使ってJTableの行をドラッグ&ドロップ、並べ替えを元に、複数JTable
間で行移動が可能になるようTableRowTransferHandler
を拡張しています。
- -
以下のように、
JTable#setFillsViewportHeight(true)
で、JTable自体の高さを拡張しておかないと、行が一つもない状態でドロップができなくなります。
- 以下のように、
JTable#setFillsViewportHeight(true)
でJTable自体の高さを拡張しておかないと、JTable
が空の状態でドロップが不可になる
TransferHandler handler = new TableRowTransferHandler();
table.getSelectionModel().setSelectionMode(
ListSelectionModel.MULTIPLE_INTERVAL_SELECTION);
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
などが設定され、ソートされた状態での並べ替えは想定していません。