Summary

JMenuBarのレイアウトマネージャーを変更して、JMenuを折り返して表示します。

Source Code Examples

JMenuBar menuBar = new JMenuBar();
menuBar.setLayout(new FlowLayout(FlowLayout.LEFT, 2, 2) {
  @Override public Dimension preferredLayoutSize(Container target) {
    synchronized (target.getTreeLock()) {
      int targetWidth = target.getSize().width;
      targetWidth = targetWidth == 0 ? Integer.MAX_VALUE : targetWidth;
      Insets insets = target.getInsets();
      int hgap = getHgap();
      int vgap = getVgap();
      int maxWidth = targetWidth - (insets.left + insets.right);
      int height   = vgap;
      int rowWidth = hgap, rowHeight = 0;
      int nmembers = target.getComponentCount();
      for (int i = 0; i < nmembers; i++) {
        Component m = target.getComponent(i);
        if (m.isVisible()) {
          Dimension d = m.getPreferredSize();
          if (rowWidth + d.width > maxWidth) {
            height += rowHeight;
            rowWidth = hgap;
            rowHeight = 0;
          }
          rowWidth += d.width + hgap;
          rowHeight = Math.max(rowHeight, d.height + vgap);
        }
      }
      height += rowHeight + insets.top  + insets.bottom;
      return new Dimension(targetWidth, height);
    }
  }
});
View in GitHub: Java, Kotlin

Explanation

上記のサンプルでは、JMenuBarFlowLayoutを継承して折り返しを行うLayoutManagerを設定して(JMenuBarのデフォルトLayoutManagerBoxLayout)、内部のJMenuなどがフレームの幅に収まらない場合は折り返して表示しています。

  • BorderLayoutを設定したJPanel#add(menubar, BorderLayout.NORTH)としてJMenuBarを追加してJFrame#setJMenuBar(...)メソッドを使用した場合、以下のような不具合が存在する?
    • JFrameの最大化、最小化で折り返しが更新されない
    • 以下のようなWindowStateListenerJFrameに追加し、ContentPanerevalidate()して回避
      frame.addWindowStateListener(new WindowStateListener() {
        @Override public void windowStateChanged(final WindowEvent e) {
          EventQueue.invokeLater(new Runnable() {
            @Override public void run() {
              System.out.println("windowStateChanged");
              JFrame f = (JFrame) e.getWindow();
              ((JComponent) f.getContentPane()).revalidate();
            }
          });
        }
      });
      // frame.getContentPane().addComponentListener(new ComponentAdapter() {
      //   @Override public void componentResized(ComponentEvent e) {
      //     ((JComponent) e.getSource()).revalidate();
      //   }
      // });
      
    • または、以下のようにFlowLayout#layoutContainerをオーバーライドすることで回避
      // https://tips4java.wordpress.com/2008/11/06/wrap-layout/
      // WrapLayout.java
      // Rob Camick on November 6, 2008
      private Dimension preferredLayoutSize;
      @Override public void layoutContainer(Container target) {
        Dimension size = preferredLayoutSize(target);
        if (size.equals(preferredLayoutSize)) {
          super.layoutContainer(target);
        } else {
          preferredLayoutSize = size;
          Container top = target;
          while (!(top instanceof Window) && top.getParent() != null) {
            top = top.getParent();
          }
          top.validate();
        }
      }
      
    • JFrame#pack()してもJFrameのサイズが変更されない
      • JFrame#setSize(...)に変更することで回避

Reference

Comment