Java实现鼠标拖放功能的方法
本文实例讲述了Java利用鼠标的拖放来实现交换程序数据的方法,即所谓的鼠标拖放功能。鼠标的拖放功能在图形化系统中非常常用,Java提供了java.awt.dnd和java.awt.datatransfer包来支持该功能。本例演示如何在程序中实现拖放的实现方法,当在窗口上部的“HelloWorld!”标签点下鼠标,并拖至窗口下部的文本框放开,则在文本框中将添加“HelloWorld!”文本;继续上述过程,将继续添加该文本。
该程序功能具体的实现思路和方法为:在鼠标拖放的实现中,两个最重要的概念是拖拽源和放置目标,即dragsource和droptarget。拖拽源和放置目标都是与可视化的组件相关联的(如果不可视,还怎么拖呢?!)。拖放技术的实质就是将拖拽源组件上的数据传递到放置目标组件上,因此从底层看,拖放和上例中的剪贴板技术很接近。
拖拽源的实现:拖拽源类必须先创建一个DragGestureRecognizer实例,表明该类是拖拽源组件类或包含拖拽源组件。可以通过调用DataSource对象的createDefaultDragGestureRecognizer()方法实现。具体的实现如下:
intaction=DnDConstants.ACTION_COPY_OR_MOVE;//拖放的类型 ds.createDefaultDragGestureRecognizer(this,action,this);
上面的语句表明,拖拽源组件是本类自身的实例对象,要完成的拖放的种类是DnDConstants.ACTION_COPY_OR_MOVE型的,实现DragGestureListener接口的类是本类。拖拽源一般实现DragGestureListener接口,该接口中定义了一个dragGestureRecognized()方法,当开始拖拽是,DragGestureListener监听到事件,随即转入dragGestureRecognized()方法处理事件,如将拖拽源的数据发送出去。具体代码:
publicvoiddragGestureRecognized(DragGestureEventdge){
//thrownewjava.lang.UnsupportedOperationException("MethoddragGestureRecognized()notyetimplemented.");
try{
Transferabletr=newStringSelection(this.getText());//将标签的文本作为数据,由Transferable对象包装
//开始拖拽,设置光标为DragSource.DefaultCopyNoDrop形,拖放的数据是tr对象,DragSourceListener是本类
dge.startDrag(DragSource.DefaultCopyNoDrop,tr,this);
}catch(Exceptionerr){
err.printStackTrace();
}
}
拖拽源还得实现DragSourceListener接口,该接口定义了拖拽相关的各状态的事件处理方法。如dragEnter、dragOver、dropActionChanged、dragExit等方法。本例中,将实现dragEnter()方法来设置拖拽时的光标形状,其他的方法为空方法。具体实现代码如下:
publicvoiddragEnter(DragSourceDragEventdsde){
//thrownewjava.lang.UnsupportedOperationException("MethoddragEnter()notyetimplemented.");
DragSourceContextdsc=dsde.getDragSourceContext();//得到拖拽源的上下文引用
//设置拖拽时的光标形状
intaction=dsde.getDropAction();
if((action&DnDConstants.ACTION_COPY)!=0)
dsc.setCursor(DragSource.DefaultCopyDrop);
else
dsc.setCursor(DragSource.DefaultCopyNoDrop);
}
放置目标的实现:放置目标的类中必须先创建一个DragTarget实例,来表明本类是放置目标组件类或包含放置目标组件,实现如下:
newDropTarget(this.jTextField1,DnDConstants.ACTION_COPY_OR_MOVE,this);
上面的语句表明,放置目标是this.jTextField1对象,拖放操作是DnDConstants.ACTION_COPY_OR_MOVE型的,而实现DropTargetListener接口的类是本类。与DrafSourceListener相对应,放置目标或其所在类一般实现DropTargetListener接口。该接口也定义了很多的方法,如dragEnter、dragOver等,用于处理当拖放过程进入不同阶段时的事件。本例只关心drop()方法,即当在放置目标组件上放开鼠标时的事件处理,一般用来处理传递到的数据,如本例中将在JTextField组件上显示传递来的文本数据,其他方法为空方法,具体代码如下:
publicvoiddrop(DropTargetDropEventdtde){
//thrownewjava.lang.UnsupportedOperationException("Methoddrop()notyetimplemented.");
try{
Transferabletr=dtde.getTransferable();//得到传递来的数据对象
//处理数据对象,得到其中的文本信息
if(dtde.isDataFlavorSupported(DataFlavor.stringFlavor)){
dtde.acceptDrop(dtde.getDropAction());
Strings=(String)tr.getTransferData(DataFlavor.stringFlavor);
this.jTextField1.setText(this.jTextField1.getText()+s);//在放置目标上显示从拖拽源传递来的文本信息
dtde.dropComplete(true);
}else{
dtde.rejectDrop();
}
}catch(Exceptionerr){
err.printStackTrace();
}
}
程序代码:
1.新建一个Project,取名为JDragAndDropDemo。
2.新建一个Application,取名为JDragAndDropDemo;主窗口取名为MainFrame,标题为JDragAndDropDemo。
3.新建一个Class,取名为DragJLabel,继承JLabel类。
4.利用wizards|implementsinterface使DragJLabel类实现DragGestureListener、DragSourceListener接口。
5.在类DragJLabel中添加新的属性DragSourceds,代码如下:
classDragJLabelextendsJLabelimplementsDragGestureListener,DragSourceListener{
DragSourceds=DragSource.getDefaultDragSource();//创建DragSource实例
……
}
6.编写DragJLabel类的构造方法。
publicDragJLabel(Stringtitle,intalignment){
super(title,alignment);//使用父类的方法
intaction=DnDConstants.ACTION_COPY_OR_MOVE;
ds.createDefaultDragGestureRecognizer(this,action,this);//创建
}
7.实现DragJLabel类中的dragGestureRecognized()方法,包装并发送数据。
publicvoiddragGestureRecognized(DragGestureEventdge){
//thrownewjava.lang.UnsupportedOperationException("MethoddragGestureRecognized()notyetimplemented.");
try{
Transferabletr=newStringSelection(this.getText());
dge.startDrag(DragSource.DefaultCopyNoDrop,tr,this);
}catch(Exceptionerr){
err.printStackTrace();
}
}
8.实现DragJLabel类中的dragEnter()方法,设置光标的形状。
publicvoiddragEnter(DragSourceDragEventdsde){
//thrownewjava.lang.UnsupportedOperationException("MethoddragEnter()notyetimplemented.");
DragSourceContextdsc=dsde.getDragSourceContext();
intaction=dsde.getDropAction();
if((action&DnDConstants.ACTION_COPY)!=0)
dsc.setCursor(DragSource.DefaultCopyDrop);
else
dsc.setCursor(DragSource.DefaultCopyNoDrop);
}
9.在MainFrame类的设计窗口中下部添加一个JTextField组件,并在类中创建一个DragJLabel实例,具体代码如下:
publicclassMainFrameextendsJFrameimplementsDropTargetListener{
privateJPanelcontentPane;
privateBorderLayoutborderLayout1=newBorderLayout();
privateJTextFieldjTextField1=newJTextField();
DragJLabellabel=newDragJLabel("HelloWorld!",SwingConstants.CENTER);
……
}
10.编写MainFrame类的初始化方法jbInit(),设置组件的初始属性,并创建一个新的DropTarget实例,代码如下:
privatevoidjbInit()throwsException{
//setIconImage(Toolkit.getDefaultToolkit().createImage(MainFrame.class.getResource("[YourIcon]")));
contentPane=(JPanel)this.getContentPane();
contentPane.setLayout(borderLayout1);
this.setSize(newDimension(410,114));
this.setTitle("JDragAndDropDemo");
jTextField1.setFont(newjava.awt.Font("Dialog",0,14));
contentPane.add(jTextField1,BorderLayout.SOUTH);
contentPane.add(this.label,BorderLayout.NORTH);
newDropTarget(this.jTextField1,DnDConstants.ACTION_COPY_OR_MOVE,this);
}
11.利用wizards|implementsinterface使MainFrame类实现DropTargetListener接口。
12.实现继承自DropTargetListener接口的方法drop(),处理传递来的数据,具体代码如下:
publicvoiddrop(DropTargetDropEventdtde){
//thrownewjava.lang.UnsupportedOperationException("Methoddrop()notyetimplemented.");
try{
Transferabletr=dtde.getTransferable();
if(dtde.isDataFlavorSupported(DataFlavor.stringFlavor)){
dtde.acceptDrop(dtde.getDropAction());
Strings=(String)tr.getTransferData(DataFlavor.stringFlavor);
this.jTextField1.setText(this.jTextField1.getText()+s);
dtde.dropComplete(true);
}else{
dtde.rejectDrop();
}
}catch(Exceptionerr){
err.printStackTrace();
}
}