MouseEvent 在 JScrollPane 中丢失

Nik*_*sov 4 java swing jpanel jframe jscrollpane

这是我用来显示我在另一个项目中面临的问题的代码。

如果我使用 JScrollPane 作为 panel2 的包装器,我不会得到任何这样的行。为什么? 我想点击 JscrollPane 并打印如下事件。

java.awt.event.MouseEvent[MOUSE_CLICKED,(800,469),absolute(808,499),button=1,modifiers=Button1,clickCount=1] on javax.swing.JPanel[,0,0,934x612,layout=java.awt.FlowLayout,alignmentX=0.0,alignmentY=0.0,border=javax.swing.border.LineBorder@cc0e01,flags=9,maximumSize=,minimumSize=,preferredSize=java.awt.Dimension[width=880,height=630]]
Run Code Online (Sandbox Code Playgroud)

如果现在我改变

panel1.add(pane);
Run Code Online (Sandbox Code Playgroud)

panel1.add(panel2);
Run Code Online (Sandbox Code Playgroud)

然后上面的消息被打印出来。

public class LostMouseEvent {

public static void main(String[] args) {
    new LostMouseEvent();
}

public LostMouseEvent() {
    EventQueue.invokeLater(new Runnable() {
        @Override
        public void run() {
            JFrame frame = new JFrame();
            frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
            frame.setLayout(new BorderLayout());

            JPanel panel1 = new JPanel();
            JPanel panel2 = new JPanel();
            JScrollPane pane = new JScrollPane(panel2);

            panel1.setPreferredSize(new Dimension(880, 630));
            panel1.setBorder(BorderFactory.createLineBorder(Color.blue));
            panel2.setPreferredSize(new Dimension(840, 610));
            panel2.setBorder(BorderFactory.createLineBorder(Color.green));


            panel1.add(pane);
            frame.add(panel1);

            frame.pack();
            frame.setVisible(true);
            frame.setSize(950, 650);

            panel1.addMouseListener(new MyMouseListener());
        }
    });
}

private class MyMouseListener extends MouseAdapter {
    @Override
    public void mouseClicked (MouseEvent me) {
        System.out.println(me);
    }
}
}
Run Code Online (Sandbox Code Playgroud)

UPD:实际上在我的项目中不止一个 panel2。最初,我里面有 panel1 和许多 panel2。然后我想用 JScrollPane 包裹每个 panel2 并开始面对这个问题。

我只需要一个 MouseListener 来最小化对代码的更改。

Dav*_*amp 5

  • 使用EDT创建和操作 Swing 组件
  • 在设置可见之前不要调用setSize()而是调用。pack()JFrame
  • 不要调用setPrefferedSize()而是覆盖getPrefferedSize()

您的代码按预期工作,它只会在panel1单击时打印消息,注释panel1在后面JScrollPane,因此绿色边框外的任何内容都是panel1。要使其同时适用于JScrollpane/panel2JPanel/ ,panel1只需将BOTH添加MouseListener到所需组件中:

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Dimension;
import java.awt.EventQueue;
import java.awt.event.MouseAdapter;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import javax.swing.BorderFactory;
import javax.swing.JFrame;
import javax.swing.JPanel;
import javax.swing.JScrollPane;
import javax.swing.SwingUtilities;

public class LostMouseEvent {

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

            @Override
            public void run() {
        new LostMouseEvent();
            }
        });
    }

    public LostMouseEvent() {
        EventQueue.invokeLater(new Runnable() {
            @Override
            public void run() {
                JFrame frame = new JFrame();
                frame.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
                frame.setLayout(new BorderLayout());

                JPanel panel1 = new JPanel() {

                    @Override
                    public Dimension getPreferredSize() {
                        return new Dimension(880, 630);
                    }

                };
                JPanel panel2 = new JPanel() {

                    @Override
                    public Dimension getPreferredSize() {
                        return new Dimension(840, 610);
                    }
                };
                JScrollPane pane = new JScrollPane(panel2);

                panel1.setBorder(BorderFactory.createLineBorder(Color.blue));
                panel2.setBorder(BorderFactory.createLineBorder(Color.green));


                panel1.add(pane);
                frame.add(panel1);

                MouseListener ml=new MyMouseListener();

                //add mouse listener to panel1 and panel2
                panel1.addMouseListener(ml);
                panel2.addMouseListener(ml);

                //alternatively add to pane
                //pane.addMouseListener(ml);

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

            }
        });
    }

    private class MyMouseListener extends MouseAdapter {

        @Override
        public void mouseClicked(MouseEvent me) {
            System.out.println(me);
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

编辑:

我个人不会推荐这个,但是,

将单个侦听器添加到JFrame将捕获所有MouseEvents 使用Toolkit类并addAWTEventListener像这样调用:

Toolkit.getDefaultToolkit().addAWTEventListener(new AWTEventListener() {
    @Override
    public void eventDispatched(AWTEvent awte) {//all mouse events will be processed here you will have to check for the mouse events you are interested in
    System.out.println(awte);
    }
}, AWTEvent.MOUSE_EVENT_MASK);//for Mouse events only
Run Code Online (Sandbox Code Playgroud)

更新1:

您还可以通过不要忘记在设置可见后将玻璃面板设置为可见来将其添加MouseListener到您JFrame玻璃面板中。这将允许您只需要添加一个. 看这里:JFrame.getGlassPane().addMouseListener(ml)JFrameListener

...

MouseListener ml = new MyMouseListener();

//add mouse listener to panel1 and panel2
//panel1.addMouseListener(ml);
//panel2.addMouseListener(ml);

//alternatively add to pane
//pane.addMouseListener(ml);

frame.getGlassPane().addMouseListener(ml);

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

frame.getGlassPane().setVisible(true);

...
Run Code Online (Sandbox Code Playgroud)

更新 2:

MouseEvent遇到迷路问题的主要原因JScrollPane是因为它是一个错误。见这里

显示的解决方法是:

public Test()
  {
    setUI(new javax.swing.plaf.metal.MetalScrollPaneUI(){
      public void installListeners(JScrollPane scrollPane){}
    });
    JPanel canvas = new JPanel();
    canvas.add( new JLabel("Test") );

    setViewportView( canvas );
    setVisible(true);
  }
Run Code Online (Sandbox Code Playgroud)