【问题标题】:Size issues with JTabbedPane and fixed height contentJTabbedPane 和固定高度内容的大小问题
【发布时间】:2016-04-28 22:47:33
【问题描述】:

在写这个问题时,我已经能够找到一种方法让它按照我想要的方式运行。基于this,我仍在发布问题,因为其他人可能面临类似问题。

对于用于水平拉伸但具有固定高度的内容的 JTabbedPane 的大小,我有以下问题。 setTabLayoutPolicy() 的两个选项似乎都会改变内容的高度,并且不会始终以首选或最小高度显示它。

使用默认WRAP_TAB_LAYOUT,选项卡窗格的首选大小不考虑选项卡当前是否实际堆叠或显示彼此相邻,如herehere 和@ 987654324@。如果选项卡式窗格基于堆叠的选项卡,则当存在足够的空间以彼此相邻显示时,内容的高度增加约20个像素(一个标签的高度)。如果基于彼此旁边的标签布置选项卡式窗格,则当必须堆叠选项卡时,内容高度会降低。

当策略设置为SCROLL_TAB_LAYOUT 时,标签栏的高度是固定的,并且布局大部分是正确的。但是,根据外观,选项卡内容的大小会减少几个像素。我已经能够发现这是由于 L&F 定义的选项卡区域的插图,它们没有被考虑到选项卡窗格的首选大小计算中(请参阅this bug report)。设置 UIManager.getDefaults().put("TabbedPane.tabAreaInsets", new Insets(0,0,0,0)) 适用于某些 L&F(例如 Metal),但不适用于其他(例如 Nimbus)。

好像只有以下几个选项:

  • 使用堆叠的标签并为内容增加额外的高度
  • 使用堆叠标签并在空间不足时覆盖内容
  • 使用滚动标签并在标签内容的最小/首选大小上添加几个像素,使其看起来与每个 L&F 有点不同(但至少内容不应被截断)
  • 使用滚动选项卡并将选项卡式窗格的 UI 设置为新的 BasicTabbedPaneUI,这看起来不太好

是否有一种简洁的方法来强制选项卡式窗格的内容始终以固定高度显示?

以下代码和截图说明问题

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JTabbedPane;
import javax.swing.UIManager;

public class TabbedPaneTest extends JFrame {

    TabbedPaneTest() {

        JPanel mainPanel = new JPanel();
        mainPanel.setBackground(Color.white);

        JTabbedPane tabs = new JTabbedPane();
        //tabs.setTabLayoutPolicy(JTabbedPane.SCROLL_TAB_LAYOUT); // content cut off by a few pixels
        tabs.setTabPlacement(JTabbedPane.BOTTOM);

        Dimension min  = new Dimension(100,200);
        Dimension max = new Dimension(Short.MAX_VALUE,Short.MAX_VALUE);
        //Dimension pref = new Dimension(Short.MAX_VALUE,200); // content cut off when small
        Dimension pref = new Dimension(0,200); // content gets extra space when large

        int tabCount = 3;
        for (int i = 0; i < tabCount; i++) {
            JLabel content = new JLabel();
            content.setMinimumSize(min);
            content.setMaximumSize(max);
            content.setPreferredSize(pref);
            tabs.addTab("lorem ipsum dolor sit amet", content);
        }

        // set up and render window
        getContentPane().add(mainPanel, BorderLayout.CENTER);
        getContentPane().add(tabs, BorderLayout.PAGE_END);
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        setTitle("JScrollPane Test");
        pack();
        setSize(700,400);

        tabs.addComponentListener(new ComponentAdapter() {
            @Override
            public void componentResized(ComponentEvent e) {
                JTabbedPane tabbedPane = (JTabbedPane) e.getComponent();
                int tabCount = tabbedPane.getTabCount();
                for (int i = 0; i < tabCount; i++) {
                    Component c = tabbedPane.getComponentAt(i);
                    ((JLabel) c).setText("<html>"
                            + getSizes(c, "content")
                            + getSizes(tabbedPane, "tabs")
                            + "</html>");
                }
            }
        });
    }

    private static String getSizes(Component c, String name) {
        return "<p>" + name + " - "
                + "  minimum:" + Integer.toString(c.getMinimumSize().width)
                         + "x" + Integer.toString(c.getMinimumSize().height)
                + "  maximum:" + Integer.toString(c.getMaximumSize().width)
                         + "x" + Integer.toString(c.getMaximumSize().height)
                + "  preferred:" + Integer.toString(c.getPreferredSize().width)
                         + "x" + Integer.toString(c.getPreferredSize().height)
                + "  actual:" + Integer.toString(c.getSize().width)
                         + "x" + Integer.toString(c.getSize().height)
                + "</p>";
    }

    public static void main(String args[]) {

        /* Set the Nimbus look and feel */
        //<editor-fold defaultstate="collapsed" desc=" Look and feel setting code (optional) ">
        /* If Nimbus (introduced in Java SE 6) is not available, stay with the default look and feel.
         * For details see http://download.oracle.com/javase/tutorial/uiswing/lookandfeel/plaf.html
         */
        try {
            for (javax.swing.UIManager.LookAndFeelInfo info : javax.swing.UIManager.getInstalledLookAndFeels()) {
                if ("Nimbus".equals(info.getName())) {
                    javax.swing.UIManager.setLookAndFeel(info.getClassName());
                    break;
                }
            }
        } catch (ClassNotFoundException | InstantiationException | IllegalAccessException | javax.swing.UnsupportedLookAndFeelException ex) {
            java.util.logging.Logger.getLogger(TabbedPaneTest.class
                    .getName()).log(java.util.logging.Level.SEVERE, null, ex);
        }
        //</editor-fold>

        /* Create and display the form */
        java.awt.EventQueue.invokeLater(new Runnable() {
            @Override public void run() {
                new TabbedPaneTest().setVisible(true);
            }
        });
    }

}

使用滚动标签时,内容会被截断几个像素(这里是 193 像素而不是 200 像素):

使用堆叠的选项卡和宽内容,当窗口很小(这里是 160 像素而不是 200 像素)时,内容会被截断:

使用堆叠的选项卡和狭窄的内容,当窗口很大时,内容会变大(这里是 240 像素而不是 200 像素):

【问题讨论】:

    标签: java swing jtabbedpane preferredsize


    【解决方案1】:

    在阅读了有关计算选项卡式窗格首选大小方式的更多信息后,我已经能够针对WRAP_TAB_LAYOUT 案例提出以下解决方案。

    问题在于,对于选项卡式窗格,首选的高度和宽度是耦合的。首选高度是针对首选宽度计算的,而不是针对实际的当前宽度。如果父级的布局管理器尊重首选高度但不尊重首选宽度,则这是有问题的。

    我想出的解决方案是设置一个侦听器,将每个选项卡内容的首选宽度设置为其当前宽度。通过这种方式,选项卡式窗格可以正确计算其首选高度,并且可以进行布局,例如带有边框布局。

    tabs.addComponentListener(new ComponentAdapter() {
        @Override
        public void componentResized(ComponentEvent e) {
            JTabbedPane tabbedPane = (JTabbedPane) e.getComponent();
            int tabCount = tabbedPane.getTabCount();
            for (int i = 0; i < tabCount; i++) {
                Component c = tabbedPane.getComponentAt(i);
                c.setPreferredSize(new Dimension(c.getSize().width, c.getPreferredSize().height));
            }
        }
    });
    

    我还没有为SCROLL_TAB_LAYOUT 案例找到令人满意的解决方案,所以如果有人有想法,我们将不胜感激。

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2015-12-22
      • 1970-01-01
      • 2014-07-25
      相关资源
      最近更新 更多