Swing:创建一个可拖动的组件...?

2022-09-01 20:38:45

我在 Web 上搜索了可拖动 Swing 组件的示例,但我发现示例不完整或无法正常工作。

我需要的是一个 Swing 组件,它可以通过鼠标拖动到其他组件中。在被拖拽时,它应该已经改变了它的位置,而不仅仅是“跳”到它的目的地。

我很欣赏没有非标准API的示例。

谢谢。


答案 1

我提出了一个简单但行之有效的解决方案,由我自己发现;)

我该怎么办?

  • 当按下鼠标时,我记录光标在屏幕上的位置以及组件的位置。
  • 当拖动鼠标时,我计算新旧光标在屏幕上的位置之间的差异,并按此差异移动组件。

使用最新的JDK 6 unter Linux(OpenSuse,KDE3)
进行测试,但是嘿,它是Java Swing,应该在任何地方都能正常工作。

代码如下:

import java.awt.*;
import java.awt.event.*;
import javax.swing.*;
import javax.swing.border.*;

public class MyDraggableComponent
    extends JComponent {

  private volatile int screenX = 0;
  private volatile int screenY = 0;
  private volatile int myX = 0;
  private volatile int myY = 0;

  public MyDraggableComponent() {
    setBorder(new LineBorder(Color.BLUE, 3));
    setBackground(Color.WHITE);
    setBounds(0, 0, 100, 100);
    setOpaque(false);

    addMouseListener(new MouseListener() {

      @Override
      public void mouseClicked(MouseEvent e) { }

      @Override
      public void mousePressed(MouseEvent e) {
        screenX = e.getXOnScreen();
        screenY = e.getYOnScreen();

        myX = getX();
        myY = getY();
      }

      @Override
      public void mouseReleased(MouseEvent e) { }

      @Override
      public void mouseEntered(MouseEvent e) { }

      @Override
      public void mouseExited(MouseEvent e) { }

    });
    addMouseMotionListener(new MouseMotionListener() {

      @Override
      public void mouseDragged(MouseEvent e) {
        int deltaX = e.getXOnScreen() - screenX;
        int deltaY = e.getYOnScreen() - screenY;

        setLocation(myX + deltaX, myY + deltaY);
      }

      @Override
      public void mouseMoved(MouseEvent e) { }

    });
  }

}

public class Main {

  public static void main(String[] args) {
    JFrame f = new JFrame("Swing Hello World");

    // by doing this, we prevent Swing from resizing
    // our nice component
    f.setLayout(null);

    MyDraggableComponent mc = new MyDraggableComponent();
    f.add(mc);

    f.setSize(500, 500);

    f.setDefaultCloseOperation(JFrame.DISPOSE_ON_CLOSE);
    f.setVisible(true);
  }

}

答案 2

另外,我发现可以在JFrame中创建一个JInternalFrame,但问题是:你总是得到一个烦人的窗口标题栏

可悲的是,要禁用标题栏,需要一个肮脏的解决方法

public class MyDraggableComponent extends JInternalFrame {

  public MyDraggableComponent() {
    InternalFrameUI thisUI = getUI();
    if (thisUI instanceof BasicInternalFrameUI) {
      ((BasicInternalFrameUI) thisUI).setNorthPane(null);
    }

}

我真的很想念像“someInternalFrame.setWindowTitleBar(false)”这样的方法......
:'(