假设我有一个包含一些JTextField的JPanel。我想为这个JPanel及其孩子执行相同的放下操作。换句话说,我希望对孩子们的放置动作与对JPanel的放置动作一样对待。
除了为JPanel及其子代设置相同的DropTargetListener之外,还有其他方法吗?
我知道如果我将这些JTextFields的TransferHandler设置为null,JPanel将收到拖放事件。但是,这会破坏文本字段的复制和粘贴功能。
我知道我可以用JLayer拦截鼠标事件。拖动事件有这样的东西吗?
最后,我将监听器分别添加到子组件中。因为我也需要相对于父级的放置位置,所以我在子组件上分别使用了SwingUtilities.convertToPoint()。这意味着使用了许多不同的侦听器 - 更多的内存使用。但似乎是目前最好的方式。
答案 0 :(得分:1)
不确定这是否是您的想法,但是,我基本上为所有组件添加了相同的DropTargetListener
,这意味着我拖动/删除传入请求的位置并不重要,所有组件触发了相同的事件......
import java.awt.BorderLayout;
import java.awt.EventQueue;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.dnd.DnDConstants;
import java.awt.dnd.DropTarget;
import java.awt.dnd.DropTargetDragEvent;
import java.awt.dnd.DropTargetDropEvent;
import java.awt.dnd.DropTargetEvent;
import java.awt.dnd.DropTargetListener;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JTextField;
import javax.swing.UIManager;
import javax.swing.UnsupportedLookAndFeelException;
public class TestDragNDrop100 {
public static void main(String[] args) {
new TestDragNDrop100();
}
public TestDragNDrop100() {
EventQueue.invokeLater(new Runnable() {
@Override
public void run() {
try {
UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
} catch (ClassNotFoundException | InstantiationException | IllegalAccessException | UnsupportedLookAndFeelException ex) {
}
JFrame frame = new JFrame("Testing");
frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
frame.setLayout(new BorderLayout());
frame.add(new TestPane());
frame.pack();
frame.setLocationRelativeTo(null);
frame.setVisible(true);
}
});
}
public class TestPane extends JPanel implements DropTargetListener {
public TestPane() {
DropTarget dt = new DropTarget(this, DnDConstants.ACTION_COPY_OR_MOVE, this, true);
setLayout(new GridBagLayout());
GridBagConstraints gbc = new GridBagConstraints();
gbc.fill = GridBagConstraints.HORIZONTAL;
for (int y = 0; y < 4; y++) {
gbc.gridy = y;
for (int x = 0; x < 4; x++) {
gbc.gridx = x;
JTextField field = new JTextField(10);
DropTarget child = new DropTarget(field, DnDConstants.ACTION_COPY_OR_MOVE, this, true);
add(field, gbc);
}
}
}
@Override
public void dragEnter(DropTargetDragEvent dtde) {
System.out.println("DragEnter - " + dtde.getDropTargetContext().getComponent());
}
@Override
public void dragOver(DropTargetDragEvent dtde) {
System.out.println("DragOver - " + dtde.getDropTargetContext().getComponent());
}
@Override
public void dropActionChanged(DropTargetDragEvent dtde) {
System.out.println("dropActionChanged" + dtde.getDropTargetContext().getComponent());
}
@Override
public void dragExit(DropTargetEvent dte) {
System.out.println("dragExit" + dte.getDropTargetContext().getComponent());
}
@Override
public void drop(DropTargetDropEvent dtde) {
System.out.println("drop" + dtde.getDropTargetContext().getComponent());
}
}
}
我也应该注意。我测试了字段剪切/复制/粘贴功能,没有问题。
答案 1 :(得分:1)
在容器上实现一个自定义TransferHandler,该容器会根据需要委托给它的子容器:
for (int i = 0; i < 5; i++) {
parent.add(new JTextField("item" + i, 20));
};
TransferHandler handler = new TransferHandler() {
@Override
public boolean canImport(TransferSupport support) {
TransferHandler childHandler = getTargetHandler();
return childHandler.canImport(
getTargetSupport(support));
}
protected TransferSupport getTargetSupport(TransferSupport support) {
return new TransferSupport(getTarget(), support.getTransferable());
}
protected TransferHandler getTargetHandler() {
return getTarget().getTransferHandler();
}
protected JComponent getTarget() {
return (JComponent) parent.getComponent(0);
}
@Override
public boolean importData(TransferSupport support) {
return getTargetHandler().importData(getTargetSupport(support));
}
};
parent.setTransferHandler(handler);
答案 2 :(得分:1)
如果您希望能够删除面板上的项目,但希望忽略面板上的任何组件以进行删除,则可以停用添加到面板的每个组件上的放置目标。你仍然可以在它们中剪切和粘贴,甚至有一种方法可以从它们发起拖动,但是你不能放弃任何东西 - 掉落事件直接通过它们到掉落与小组相关的目标。
要执行此操作,只需致电
component.getDropTarget().setActive(false);
面板上的每个组件。
我在构建日历面板时发现这很有用,我希望能够在周围拖动约会但是将它们放在面板上,即使它已被(部分或全部)覆盖在其他约会中。