Summary

JToolBar内にJMenuBarを配置することで分離・移動可能なメニューを作成します。

Source Code Examples

JMenuBar menuBar = new JMenuBar();
menuBar.add(makeMenu("JMenu 1"));

JMenu menu2 = makeMenu("JMenu 2");
menu2.addMenuListener(new MenuListener() {
  private boolean isFloating(Component c) {
    Container p = SwingUtilities.getAncestorOfClass(JToolBar.class, c);
    return p instanceof JToolBar &&
        ((BasicToolBarUI) ((JToolBar) p).getUI()).isFloating();
  }

  @Override public void menuSelected(MenuEvent e) {
    JMenu menu = (JMenu) e.getSource();
    if (menu.isTopLevelMenu() && !isFloating(menu)) {
      Dimension d = menu.getPreferredSize();
      Component cp = menu.getRootPane().getContentPane();
      Point pt = SwingUtilities.convertPoint(
          menu.getParent(), menu.getLocation(), cp);
      pt.y += d.height * 2;
      if (!cp.getBounds().contains(pt)) {
        EventQueue.invokeLater(() -> {
          JPopupMenu popup = menu.getPopupMenu();
          Rectangle bounds = popup.getBounds();
          Point loc = menu.getLocationOnScreen();
          loc.y -= bounds.height + UIManager.getInt("Menu.menuPopupOffsetY");
          popup.setLocation(loc);
        });
      }
    }
  }

  @Override public void menuDeselected(MenuEvent e) {
    // Do nothing
  }

  @Override public void menuCanceled(MenuEvent e) {
    // Do nothing
  }
});
menuBar.add(menu2);

JToolBar toolBar = new JToolBar() {
  @Override public void updateUI() {
    super.updateUI();
    setLayout(new BorderLayout());
  }
};
toolBar.add(menuBar);

add(toolBar, BorderLayout.NORTH);
View in GitHub: Java, Kotlin

Explanation

  • JToolBarのレイアウトをBorderLayoutに変更してJMenuBarを追加
    • JToolBarのデフォルトレイアウトは配置で水平垂直を切り替えるBoxLayout
    • このBoxLayoutのままだとJMenuBarの幅がJToolBarの幅まで拡張されない
    • JMenuBar#setMaximumSize(...)で十分大きな最大サイズを設定すればBoxLayoutのままでも回避可能
    • このサンプルでは垂直にしてもレイアウトの切り替えは実行しないのでパネルの左右にJToolBarを配置不可に設定
  • MenuListenerJMenuに追加してBorderLayout.SOUTHJToolBarが配置されている場合はJMenuの上方にJPopupMenuを開くようその位置を変更
    • JMenuがトップレベル(親がJPopupMenuではなくJMenuBar)、かつJMenuBarがフローティング状態ではない場合のみ、JPopupMenuを開く位置を変更するかどうかチェックしている

Reference

Comment