• title: JMenuBarのJMenuを折り返し tags: [JMenuBar, JMenu, LayoutManager, FlowLayout] author: aterai pubdate: 2010-12-27T11:25:50+09:00 description: JMenuBarのレイアウトマネージャーを変更して、JMenuを折り返して表示します。

概要

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

サンプルコード

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;
      if (targetWidth == 0) targetWidth = Integer.MAX_VALUE;
      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

解説

上記のサンプルでは、JMenuBar(デフォルトのLayoutManagerBoxLayout)に、FlowLayoutを継承して折り返しを行うLayoutManagerを設定して、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をオーバーライドすることで回避
    //http://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(...)に変更することで回避

参考リンク

コメント