9

我需要帮助来理解 Swing 中的事件传播。我知道每个事件仅由一个组件处理。因此,当我有一个outside带有一些子面板的面板inside并向它们添加 mouseListeners 时,inside将调用其中的一个。这很好,这是预期的行为。

但是我不明白以下情况下的行为: inside注册一个MouseMotionListener并outside注册一个MouseListener。我希望inside消耗所有 MouseMotionEvents 并outside接收 MouseEvents,因为在inside. 但事实并非如此,inside以某种方式消耗所有 MouseEvents,而不仅仅是 MouseMotionEvents。

下面的代码说明了这个问题:

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

public class EventTest {
public static void main(String... args) {
    SwingUtilities.invokeLater(new Runnable(){
        @Override
        public void run() {
            JComponent inside = new JPanel(); 
            inside.setBackground(Color.red);
            inside.setPreferredSize(new Dimension(200,200));
            MouseMotionListener mm = new MouseMotionListener() {
                @Override
                public void mouseDragged(MouseEvent arg0) {
                    System.err.println("dragged");                      
                }
                @Override
                public void mouseMoved(MouseEvent arg0) {
                    System.err.println("moved");
                }
            };
            // next line disables handling of mouse clicked events in outside 
            inside.addMouseMotionListener(mm); 

            JComponent outside = new JPanel();
            outside.add(inside);
            outside.setPreferredSize(new Dimension(300,300));
            outside.addMouseListener( new MouseAdapter() {
                public void mouseClicked(MouseEvent e) {
                    System.err.println("clicked");
                }
            });

            JFrame frame = new JFrame();
            frame.add(outside);
            frame.pack();
            frame.setVisible(true);
        }
    });
    }
}

inside我可以通过为父组件可能感兴趣的所有事件注册一个侦听器来解决此问题,然后调用 dispatchEvent 将事件转发给父组件。

a) 有人可以指点我一些文档,其中描述了这种行为吗?MouseEvent 的 javadocs 让我觉得我的期望是正确的。所以,我需要一个不同的描述来理解它。

b) 有没有比上面概述的更好的解决方案?

谢谢,凯瑟琳

编辑:目前还不清楚为什么 Swing 会这样。但看起来,让这些东西正常工作的唯一方法是手动转发事件,我会这样做。

4

3 回答 3

8

a) 按照设计,Java 鼠标事件只有在子组件上没有鼠标侦听器时才会“冒泡”。

b) 您可以事件转发到另一个组件,如下所示

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

public class EventTest {

    public static void main(String... args) {
        SwingUtilities.invokeLater(new Runnable() {

            @Override
            public void run() {
                final JComponent outside = new JPanel();
                JComponent inside = new JPanel();
                inside.setBackground(Color.red);
                inside.setPreferredSize(new Dimension(200, 200));
                inside.addMouseMotionListener(new MouseAdapter() {

                    @Override
                    public void mouseDragged(MouseEvent e) {
                        System.err.println("dragged");
                    }

                    @Override
                    public void mouseMoved(MouseEvent e) {
                        System.err.println("moved inside");
                        outside.dispatchEvent(e);
                    }
                });

                outside.add(inside);
                outside.setPreferredSize(new Dimension(300, 300));
                outside.addMouseMotionListener(new MouseAdapter() {

                    @Override
                    public void mouseMoved(MouseEvent arg0) {
                        System.err.println("moved outside");
                    }
                });

                JFrame frame = new JFrame();
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.add(outside);
                frame.pack();
                frame.setVisible(true);
            }
        });
    }
}
于 2012-08-06T17:58:54.460 回答
1

与trashgod 的回答非常相似 - 您可以使用 aMouseAdapter作为您的运动侦听器,并覆盖它以转发您希望由父级处理的任何事件。这应该只会为您的代码添加最少的数量。

        MouseAdapter mm = new MouseAdapter() {
            @Override
            public void mouseDragged(MouseEvent arg0) {
                System.err.println("dragged");                      
            }
            @Override
            public void mouseMoved(MouseEvent arg0) {
                System.err.println("moved");
            }
            @Override
            public void mouseClicked(MouseEvent e) {
                outside.dispatchEvent(e);
            }
        };
        // For forwarding events
        inside.addMouseListener(mm); 
        // For consuming events you care about
        inside.addMouseMotionListener(mm);

我也找不到使用该方法的任何dispatchEvent(e)方法。我想你被那条路线困住了。

于 2012-08-06T18:13:57.040 回答
0

这对我有用:

  Ellipse2D mCircle = new Ellipse2D.Double(x,y,size,size);

  void PassMouseEvent(MouseEvent e) {
    getParent().dispatchEvent(e);
  }

  public void mousePressed(MouseEvent arg0) {
    if(mCircle.contains(arg0.getX(), arg0.getY())) {
      // Do stuff if we click on this object
    } else {
      // Pass to the underlying object to deal with the mouse event
      PassMouseEvent(arg0);
    }
  }
于 2013-10-30T02:37:30.657 回答