以下は、私のライブアプリから取得した完全に機能する回答です。
マウスアウト時にメニューを閉じる方法は、コンストラクターの上部でブール変数「isMouseOut」を実行して追跡し、複数の MouseIn-MouseOut イベントを追跡するために、よりオブジェクト指向に適した方法で MouseListener を割り当てることでした。ユーザーがメニューを操作するとき。これは、ブール値「isMouseOut」の状態に基づいて動作する別の menuClear メソッドを呼び出します。クラスは MouseListener を実装します。これがそのやり方です。
最初にすべてのメニュー項目をこの配列に追加する ArrayList を作成します。そのようです:
Font menuFont = new Font("Arial", Font.PLAIN, 12);
JMenuBar menuBar = new JMenuBar();
getContentPane().add(menuBar, BorderLayout.NORTH);
// Array of MenuItems
ArrayList<JMenuItem> aMenuItms = new ArrayList<JMenuItem>();
JMenuItem mntmRefresh = new JMenuItem("Refresh");
JMenuItem mntmNew = new JMenuItem("New");
JMenuItem mntmNormal = new JMenuItem("Normal");
JMenuItem mntmMax = new JMenuItem("Max");
JMenuItem mntmStatus = new JMenuItem("Status");
JMenuItem mntmFeedback = new JMenuItem("Send Feedback");
JMenuItem mntmEtsyTWebsite = new JMenuItem("EtsyT website");
JMenuItem mntmAbout = new JMenuItem("About");
aMenuItms.add(mntmRefresh);
aMenuItms.add(mntmNew);
aMenuItms.add(mntmNormal);
aMenuItms.add(mntmMax);
aMenuItms.add(mntmStatus);
aMenuItms.add(mntmFeedback);
aMenuItms.add(mntmEtsyTWebsite);
aMenuItms.add(mntmAbout);
次に、この段階で arrayList を反復処理し、for() ループを使用して MouseListener を追加します。
for (Component c : aMenuItms) {
if (c instanceof JMenuItem) {
c.addMouseListener(ml);
}
}
MenuBar の JMenu 親を設定します。
// Now set JMenu parents on MenuBar
final JMenu mnFile = new JMenu("File");
menuBar.add(mnFile).setFont(menuFont);
final JMenu mnView = new JMenu("View");
menuBar.add(mnView).setFont(menuFont);
final JMenu mnHelp = new JMenu("Help");
menuBar.add(mnHelp).setFont(menuFont);
次に、ドロップダウン menuItems の子を JMenu の親に追加します。
// Now set menuItems as children of JMenu parents
mnFile.add(mntmRefresh).setFont(menuFont);
mnFile.add(mntmNew).setFont(menuFont);
mnView.add(mntmNormal).setFont(menuFont);
mnView.add(mntmMax).setFont(menuFont);
mnHelp.add(mntmStatus).setFont(menuFont);
mnHelp.add(mntmFeedback).setFont(menuFont);
mnHelp.add(mntmEtsyTWebsite).setFont(menuFont);
mnHelp.add(mntmAbout).setFont(menuFont);
別の手順として、mouseListeners を JMenu の親に追加します。
for (Component c : menuBar.getComponents()) {
if (c instanceof JMenu) {
c.addMouseListener(ml);
}
}
子の menuItem 要素はすべて、親の JMenu 要素および MenuBar 自体とは別の独自のリスナーを持っているため、MouseListener() インスタンス化内でオブジェクト タイプを識別して、マウスオーバー時にメニューが自動的に開くようにすることが重要です (この例では 3x JMenu の親) しかし、子の例外エラーを回避し、マウスの位置がどこにあるかを監視しようとせずに、メニュー構造の mouseOUT を明確に識別できるようにします。MouseListener は次のとおりです。
MouseListener ml = new MouseListener() {
public void mouseClicked(MouseEvent e) {
}
public void mousePressed(MouseEvent e) {
}
public void mouseReleased(MouseEvent e) {
}
public void mouseExited(MouseEvent e) {
isMouseOut = true;
timerMenuClear();
}
public void mouseEntered(MouseEvent e) {
isMouseOut = false;
Object eSource = e.getSource();
if(eSource == mnHelp || eSource == mnView || eSource == mnFile){
((JMenu) eSource).doClick();
}
}
};
上記は、子メニューのドロップダウンのトリガーであるため、JMenu の「親」(この例では 3 回) へのマウス クリックのみをシミュレートします。timerMenuClear() メソッドは、MenuSelectionManager を呼び出して、実際の mouseOUT の時点でアクティブだった selectedpath ポイントを空にします。
public void timerMenuClear(){
ActionListener task = new ActionListener() {
public void actionPerformed(ActionEvent e) {
if(isMouseOut == true){
System.out.println("Timer");
MenuSelectionManager.defaultManager().clearSelectedPath();
}
}
};
//Delay timer half a second to ensure real mouseOUT
Timer timer = new Timer(1000, task);
timer.setInitialDelay(500);
timer.setRepeats(false);
timer.start();
}
開発中にJVM内でアクセスできる値を監視して、少しテストが必要でしたが、うまくいきました!ネストされたメニューでも :) 多くの人がこの完全な例が非常に役立つことを願っています。