微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

组件添加到窗口时停止触发鼠标拖动事件

如何解决组件添加到窗口时停止触发鼠标拖动事件

我有一个可以在其中拖动组件的容器。我遇到的问题是,每当我拿起一个组件并将其自动添加到窗口中时,即使鼠标仍在手柄上拖动,该组件也会停止触发拖动事件。然后,下次我单击并拖动手柄(现在在浮动窗口中)时,它将继续拖动。这是一些基本代码

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.Container;
import java.awt.Dimension;
import java.awt.GridBagConstraints;
import java.awt.GridBagLayout;
import java.awt.Point;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;

import javax.swing.BorderFactory;
import javax.swing.Box;
import javax.swing.Box.Filler;
import javax.swing.JComponent;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JWindow;
import javax.swing.SwingUtilities;
import javax.swing.border.Border;
import javax.swing.border.LineBorder;

/**
 *
 */
@SuppressWarnings("serial")
public class DraggableDemo extends JPanel {

    private class DraggablePanel extends JPanel {

        private class MyMouseAdapter extends MouseAdapter {
            private DraggablePanel floater;
            private Point dragOffset;

            @Override
            public void mouseDragged(MouseEvent e) {
                if (floater == null) {
                    startDragging(e);
                }

                Point transformedEventPoint = SwingUtilities.convertPoint(e.getComponent(),e.getPoint(),owner);
                updateWindowLocation(transformedEventPoint);

            }

            private void startDragging(MouseEvent e) {
                Component component = e.getComponent();
                floater = (DraggablePanel) SwingUtilities.getAncestorOfClass(DraggablePanel.class,component);
                Point floaterabsoluteLocation = SwingUtilities.convertPoint(floater.getParent(),floater.getLocation(),owner);
                Point transformedEventPoint = SwingUtilities.convertPoint(component,owner);
                // designate a drag offset so the component's corner doesn't teleport to mouse location
                dragOffset = new Point(transformedEventPoint.x - floaterabsoluteLocation.x,transformedEventPoint.y - floaterabsoluteLocation.y);

                swapComponents(getFiller(),floater);

                // place the floating component in a window
                window.add(floater);
                window.pack();
                floater.setBorder(new LineBorder(Color.YELLOW));
                updateWindowLocation(transformedEventPoint);
                window.setVisible(true);
            }

            private void updateWindowLocation(Point point) {
                Point p = new Point(point.x - dragOffset.x,point.y - dragOffset.y);
                SwingUtilities.convertPointToScreen(p,owner);
                window.setLocation(p);
            }

            private JComponent getFiller() {
                Dimension dimension = floater.getSize();
                JComponent filler = (JComponent) Box.createRigidArea(dimension);

                // border
                Border dashedBorder = BorderFactory.createDashedBorder(Color.gray,3f,2f,true);
                filler.setBorder(dashedBorder);
                filler.setopaque(true);

                return filler;
            }

            @Override
            public void mouseReleased(MouseEvent e) {
                Point transformedEventPoint = SwingUtilities.convertPoint(e.getComponent(),owner);
                Component compDroppedOn = SwingUtilities.getDeepestComponentAt(owner,transformedEventPoint.x,transformedEventPoint.y);
                if (compDroppedOn instanceof Filler) {
                    window.remove(floater);
                    swapComponents(floater,compDroppedOn);
                    window.setVisible(false);
                    floater = null;
                }
            }
        }

        private JWindow window;
        private MyMouseAdapter mouseAdapter;
        private DraggableDemo owner;

        DraggablePanel(DraggableDemo owner) {
            this.owner = owner;
            JPanel smallPanel = new JPanel();
            smallPanel.setPreferredSize(new Dimension(200,100));
            smallPanel.setBackground(Color.green);

            JPanel bigPanel = new JPanel();
            bigPanel.setPreferredSize(new Dimension(200,400));
            bigPanel.setBackground(Color.red);

            setLayout(new BorderLayout());
            add(smallPanel,BorderLayout.norTH);
            add(bigPanel,BorderLayout.CENTER);

            setBorder(new LineBorder(Color.blue));

            mouseAdapter = new MyMouseAdapter();
            smallPanel.addMouseListener(mouseAdapter);
            smallPanel.addMouseMotionListener(mouseAdapter);
            owner.addMouseListener(mouseAdapter);
            owner.addMouseMotionListener(mouseAdapter);

            window = new JWindow();
            window.setAlwaysOnTop(true);
        }
    }

    private GridBagConstraints gbc;

    /**
     * 
     */
    public DraggableDemo() {
        setLayout(new GridBagLayout());
        gbc = new GridBagConstraints();
        gbc.fill = GridBagConstraints.HORIZONTAL;
        gbc.weightx = 1;

        for (int i = 0; i < 2; i++) {
            add(new DraggablePanel(this),gbc);
        }
    }

    private static void createAndShowGUI() {
        JFrame frame = new JFrame("Demo");
        frame.setDefaultCloSEOperation(JFrame.EXIT_ON_CLOSE);

        DraggableDemo newContentPane = new DraggableDemo();
        newContentPane.setopaque(true);
        frame.setContentPane(newContentPane);

        frame.pack();
        frame.setVisible(true);
    }

    public static void main(String[] args) {
        SwingUtilities.invokelater(new Runnable() {
            @Override
            public void run() {
                createAndShowGUI();
            }
        });
    }

    /**
     */
    private void swapComponents(Component toAdd,Component toRemove) {
        Container parent = toRemove.getParent();
        int index = parent.getComponentZOrder(toRemove);

        parent.remove(toRemove);
        parent.add(toAdd,gbc,index);

        revalidate();
        repaint();

    }

}

我尝试过的事情:

原始容器每列包含几个组件,每个组件触发不同的鼠标事件。

最初,我将鼠标侦听器注册到主容器,并根据坐标获取并移动了组件,但这不满足鼠标本身进入/退出较小组件触发事件的条件。

此后,我尝试注册多个鼠标侦听器,每个侦听器都在做自己的事情,但是我了解到它们会吃掉层次结构中发生的事件。

最后,我决定为需要注册的每个组件注册一个侦听器,并根据事件返回的组件实例,将这些事件委托给适当的鼠标“适配器”。

如何修复组件以便在拾取组件时正确拖动?就像我说过的那样,我不能只注册到主容器,因为那样我将无法访问由较小组件触发的事件,并且我不能注册多个侦听器,因为事件将被第一个事件吞噬一个被解雇了。

解决方法

每当我拿起一个组件并将其自动添加到窗口中时,该组件都会停止触发拖动事件

不仅仅是停止触发拖动事件的组件。似乎没有为任何组件生成拖动事件:

在您的createAndShowGUI()方法中,框架可见后,我添加了:

long eventMask = AWTEvent.MOUSE_MOTION_EVENT_MASK + AWTEvent.MOUSE_EVENT_MASK;

Toolkit.getDefaultToolkit().addAWTEventListener( new AWTEventListener()
{
    public void eventDispatched(AWTEvent e)
    {
        System.out.println(e.getID());
    }
},eventMask);

应该显示为任何组件生成的所有鼠标事件。但是,一旦显示窗口,就不会再为任何组件生成任何事件,从而确认您的问题。

接下来,我删除了上面的代码,并将其替换为自定义EventQueue,以简单地显示所生成的每个事件:

EventQueue queue = new EventQueue()
{
    protected void dispatchEvent(AWTEvent event)
    {
        System.out.println(event);
        super.dispatchEvent(event);
    }
};

Toolkit.getDefaultToolkit().getSystemEventQueue().push(queue);

现在我确实看到了所有的鼠标拖动事件。

那么也许您可以创建一个自定义EventQueue来处理在DraggablePanel上生成的鼠标事件?

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。