2

如果有问题的列使用组合作为编辑器,我需要使用JTablea呈现选定/聚焦的单元格。JComboBox这样做的目的是向用户提示该列中的单元格是使用组合而不是JTextField.

我的问题是,LookAndFeels除了金属之外,不可能将表格的选择背景设置为组合框。如果单击或使用键盘导航到“组合列”,则所选单元格将由具有默认背景的组合呈现。

Metal L&F 可以按我的意愿工作,但这不是一个选择,因为我们的客户使用 Substance/Nimbus/Windows L&F 并且对他们中的任何一个都不起作用。

希望我足够清楚,任何帮助表示赞赏!

选择此 SSCCE 的“组合列”中的单元格以了解我的意思:

import java.awt.Component;

import javax.swing.DefaultCellEditor;
import javax.swing.DefaultComboBoxModel;
import javax.swing.JComboBox;
import javax.swing.JFrame;
import javax.swing.JScrollPane;
import javax.swing.JTable;
import javax.swing.ListSelectionModel;
import javax.swing.SwingUtilities;
import javax.swing.UIManager;
import javax.swing.table.DefaultTableCellRenderer;
import javax.swing.table.DefaultTableModel;
import javax.swing.table.TableCellRenderer;
import javax.swing.table.TableModel;

import com.sun.java.swing.plaf.nimbus.NimbusLookAndFeel;


public class RenderSelectedCellWithComboSSCCE extends JFrame
{   
    RenderSelectedCellWithComboSSCCE()
    {
        setDefaultCloseOperation(DISPOSE_ON_CLOSE);
        add(new JScrollPane(createTable()));
        pack();
    }

    private JTable createTable()
    {
        TableModel model = new DefaultTableModel(
                new Object [][] {{"A", "Item 0"},
                                 {"B", "Item 1"},
                                 {"C", "Item 2"},
                                 {"D", "Item 3"},
                                 {"E", "Item 4"}},
                new String [] {"TextField", "Combo"});
        JTable table = new JTable(model);
        table.setSelectionMode(ListSelectionModel.SINGLE_SELECTION);
        DefaultCellEditor editor = new DefaultCellEditor(new JComboBox(new Object[]{"Item 0", "Item 1", "Item 2", "Item 3", "Item 4"}));
        editor.setClickCountToStart(2);
        table.getColumnModel().getColumn(1).setCellEditor(editor);
        table.getColumnModel().getColumn(1).setCellRenderer(new ComboCellRenderer());
        return table;
    }

    /**
     * Renderer that demonstrates problem rendering selected cells with a combo with selection background.
     */
    private static class ComboCellRenderer implements TableCellRenderer
    {
        final TableCellRenderer defaultRenderer;
        final JComboBox combo;

        ComboCellRenderer()
        {
            defaultRenderer = new DefaultTableCellRenderer();
            combo = new JComboBox();
        }

        @Override
        public Component getTableCellRendererComponent(JTable table, Object value, boolean isSelected, boolean hasFocus, int row, int column)
        {
            if (hasFocus && table != null && table.isCellEditable(row, column))
            {
                combo.setModel(new DefaultComboBoxModel(new Object[]{value}));
                combo.setSelectedItem(value);
                combo.setBackground(table.getSelectionBackground());    // This only have desired effect for Metal L&F
                return combo;
            }
            else
            {
                return defaultRenderer.getTableCellRendererComponent(table, value, isSelected, hasFocus, row, column);
            }
        }
    }

    public static void main(String[] args)
    {
        SwingUtilities.invokeLater(new Runnable()
        {
            @Override
            public void run()
            {
                setLookAndFeel();
                new RenderSelectedCellWithComboSSCCE().setVisible(true);
            }

            private void setLookAndFeel()
            {
                try
                {
                    //UIManager.setLookAndFeel(UIManager.getCrossPlatformLookAndFeelClassName());       // Metal works fine
                    UIManager.setLookAndFeel(new NimbusLookAndFeel());
                    //UIManager.setLookAndFeel(new WindowsLookAndFeel());
                    //UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
                    //UIManager.setLookAndFeel("org.jvnet.substance.SubstanceLookAndFeel");
                }
                catch (Exception e)
                {
                    throw new RuntimeException("Failed to set LookAndFeel", e);
                }
            }
        });
    }
}
4

2 回答 2

3

ComboBox 渲染确实会带来问题,通常 LAF 对如何显示组合本身有非常特殊的想法,许多都是有样式的,有圆角……它不可能对所有人都正确。在玩了一下斯坦尼的建议时,我遇到了所有令人讨厌的(很快就忘记了:-)细节,比如

  • Windows 在渲染组合时关闭不透明度,因此在这里将组合的渲染器包装成一个在调用 super 后将不透明度设置为 true 的渲染器会有所帮助
  • Nimbus 会忽略自己设置的组件背景,无论不透明度如何:如果您设置与默认值不同的选择颜色(在表格上),您的渲染器将按预期工作(好吧,几乎,模数意外的颜色变化:-)
  • ...

另一方面,您不需要功能齐全的 JComboBox,您所需要的只是一个可以打开的视觉线索。因此,您可能会使用自定义组件 - 最简单的是带有一些箭头的 JLabel - 模拟 JComboBox,例如:

public static class MockCombo extends JLabel {

    private JButton arrow;

    public MockCombo() {
        JComboBox box = new JComboBox();
        box.setEditable(true);
        arrow = (JButton) box.getComponent(0);
        setLayout(new BorderLayout());
        add(arrow, BorderLayout.LINE_END);
        setOpaque(true);
    }

}

仍然需要由 LAF 进行微调,但比调整 JComboBox 容易得多

于 2011-08-08T12:07:15.690 回答
2

You can specify the combo renderer and use desired color there. Kind of renderer for rendered component.

于 2011-08-08T08:56:49.937 回答