UIManager SetLookAndFeel 的 Java 鼠标侦听器错误

问题描述

下面的代码显示一个 JFrame,带有一个 JMenu一个 [自定义] JPanel

我通过调用方法 UIManager.setLookAndFeel

将外观 (LAF) 设置为 Windows LAF

我用鼠标单击 JMenu,然后将鼠标移到 [custom] JPanel 上 - 菜单仍然可见。然后我再次单击鼠标,即使 mousepressed 方法调用了,我的 [custom] MouseListener 上的 JPanelmouseReleased 方法没有被调用

如果我没有明确设置 LAF,即如果我使用认的 LAF,它工作正常。

我该如何解决这个问题?

下面是两个类:MListenPanel。类 MListen 声明了一个 main 方法,因此您可以启动该类。然后您可以执行上述操作来重现该行为。 Panel 类是我的自定义 JPanel

如果鼠标在 mousepressed 上,我希望在每次按下鼠标时调用MouseListener 中的 Panel 方法。>

  • MListen.java
Panel
  • Panel.java
package drawer.Main;
import javax.swing.*;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.event.*;

public class MListen extends JFrame{

    public MListen(){
        
        setDefaultCloSEOperation(JFrame.EXIT_ON_CLOSE);
        
        setSize(800,600);
        //setExtendedState(MAXIMIZED_BOTH);
        setLayout(new BorderLayout());
        getContentPane().setBackground(Color.WHITE);
        
        
        Panel panel = new Panel();
        JMenuBar bar = new JMenuBar();
        JMenu file = new JMenu("file");
        JMenuItem open = new JMenuItem("open");
        JButton b = new JButton("AB");
        file.add(b);
        
        bar.add(file);
        file.add(open);
        setJMenuBar(bar);
        
        add(panel,BorderLayout.CENTER);
        
        setVisible(true);
        
    }
    
    public static void main(String[] args){
        javax.swing.SwingUtilities.invokelater(new Runnable() {  
              
            public void run() {
                try {
                    UIManager.setLookAndFeel("com.sun.java.swing.plaf.windows.WindowsLookAndFeel");
                } catch (ClassNotFoundException | InstantiationException | illegalaccessexception
                        | UnsupportedLookAndFeelException e) {
                    e.printstacktrace();
                }
                new MListen();  
            }  
        });  
    }

}

点击红色的 package drawer.Main; import java.awt.*; import java.awt.event.*; import javax.swing.JPanel; public class Panel extends JPanel implements MouseListener,MouseMotionListener{ public float size; public Color color; public boolean DoDraw; public Panel(){ addMouseListener(this); setFocusable(true); addMouseMotionListener(this); setBackground(Color.RED); } @Override public void mouseDragged(MouseEvent arg0) { //System.out.println("Drag"); } @Override public void mouseMoved(MouseEvent arg0) { // Todo Auto-generated method stub } @Override public void mouseClicked(MouseEvent arg0) { // Todo Auto-generated method stub } @Override public void mouseEntered(MouseEvent arg0) { // Todo Auto-generated method stub } @Override public void mouseExited(MouseEvent arg0) { // Todo Auto-generated method stub } @Override public void mousepressed(MouseEvent arg0) { System.out.println("Press"); // Todo Auto-generated method stub } @Override public void mouseReleased(MouseEvent arg0) { System.out.println("relese"); // Todo Auto-generated method stub } } 尝试运行上述代码。控制台将显示“按下”。现在点击 JPanel,然后点击 JMenu,“Press”将显示在控制台中。

谢谢!

  • 编辑 我需要 UIManager.setLookAndFeel("com.sun.java.swing.plaf.windows.WindowsLookAndFeel"); 我可以在不删除它的情况下解决这个问题吗? 另外我第一次需要 mousepressed

解决方法

这是一个糟糕的答案,但它应该有效。您可以在委派之前捕获所有事件。

这可以添加到您的 JPanel 构造函数中。

long eventMask = AWTEvent.MOUSE_EVENT_MASK;
Toolkit.getDefaultToolkit().addAWTEventListener( new AWTEventListener() {
    public void eventDispatched(AWTEvent e){
                MouseEvent evt = (MouseEvent)e;
                if( evt.getSource() == this && evt.getId()==MouseEvent.MOUSE_PRESSED ){
                    this.mousePressed(evt);
                    evt.consume();
                }
    }
},eventMask );

这会捕获所有鼠标事件,检查它们是否来自正确的面板,检查它是否是 mousePressed 事件并调用该方法。

这是用于调试的好代码,但不好在您的项目中找到。