如何在弹出窗口中直接显示动态添加的项目,而不是双击?

时间:2013-05-17 15:43:16

标签: java swing jcombobox jpopupmenu jpopup

我有这个PopupMenuListener

public class MyPopupMenuListener implements PopupMenuListener {

protected JTable _table;

public MyPopupMenuListener(JTable table) {
    _table = table;
}

@Override
public void popupMenuWillBecomeVisible(PopupMenuEvent e) {
    JComboBox comboBox = null;

    comboBox = (JComboBox) e.getSource();
    final int selectedRow = _table.rowAtPoint(comboBox.getLocation());
    final int selectedColumn = _table.columnAtPoint(comboBox.getLocation());

    if (_table.getValueAt(selectedRow, selectedColumn)
            .equals(MyEnum.BOTH.getDescription())) {
        comboBox.getModel().setSelectedItem(MyEnum.BOTH.getDescription());
    }
}

@Override
public void popupMenuWillBecomeInvisible(PopupMenuEvent e) {
}

@Override
public void popupMenuCanceled(PopupMenuEvent e) {
}
}

当我点击这个组合框时,项目被添加,但我必须再次点击以查看带有新项目和原始项目的弹出菜单。我想添加项目并显示所有四个项目而无需双击。

更新:这是一个SSCCE。在示例中,弹出菜单是直接可见的,在我的代码库中,当我第一次单击带有添加项目的组合框时,我没有看到其他的,我必须再次单击组合框以查看所有这些。

public class Example extends JFrame {
private JTable _table;

/**
 * Launch the application.
 */
public static void main(String[] args) {
    EventQueue.invokeLater(new Runnable() {
        public void run() {
            try {
                Example frame = new Example();
                frame.setVisible(true);
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    });
}

/**
 * Create the frame.
 */
public Example() {
    setBounds(100, 100, 450, 300);
    setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);

    _table = new JTable();
    _table.setModel(new DefaultTableModel(
        new Object[][] {
            {MyEnum.ONE.getDescription()},
            {MyEnum.ONE.getDescription()},
            {MyEnum.ONE.getDescription()},
        },
        new String[] {
            "Selection"
        }
    ) {
        Class[] columnTypes = new Class[] {
            String.class
        };
        public Class getColumnClass(int columnIndex) {
            return columnTypes[columnIndex];
        }
    });
    getContentPane().add(_table, BorderLayout.CENTER);
    _table.getColumnModel()
            .getColumn(0)
            .setCellEditor(new MyCellEditor(new MySwitchComboBox(new String[] {
                    MyEnum.ONE.getDescription(),
                    MyEnum.TWO.getDescription(),
            }), _table));

}

@SuppressWarnings("serial")
public class MyCellEditor extends DefaultCellEditor {

    private JComboBox _comboBox;

    public MyCellEditor(JComboBox comboBox, JTable table) {
        super(comboBox);
        _comboBox = comboBox;
        _table = table;
    }

    @Override
    public Component getTableCellEditorComponent(JTable table,
                                                 Object value,
                                                 boolean isSelected,
                                                 int row,
                                                 int column) {

        Rectangle r = _table.getCellRect(row, column, true);
        _comboBox.setLocation(new Point(r.x, r.y + r.height));
        return super.getTableCellEditorComponent(table, value, isSelected, row, column);
    }

}

@SuppressWarnings("serial")
public class MySwitchComboBox extends JComboBox {

    public MySwitchComboBox(String[] items) {
        super(items);
        addPopupMenuListener(new PopupMenuListener() {
            private boolean _firedPopupMenuWillBecomeVisible;

            @Override
            public void popupMenuWillBecomeVisible(PopupMenuEvent e) {
                JComboBox comboBox = null;

                comboBox = (JComboBox) e.getSource();
                final int selectedRow =
                        _table.rowAtPoint(comboBox.getLocation());
                final int selectedColumn =
                        _table.columnAtPoint(comboBox.getLocation());

                if (_table.getValueAt(selectedRow, selectedColumn)
                        .equals(MyEnum.BOTH.getDescription())) {
                    comboBox.getModel().setSelectedItem(MyEnum.BOTH.getDescription());

                    if (_firedPopupMenuWillBecomeVisible) {
                        _firedpopupMenuWillBecomeVisible = false;
                    } else {
                       _firedPopupMenuWillBecomeVisible = true;
                       comboBox.firePopupMenuWillBecomeVisible();
                    }
                }
            }

            @Override
            public void popupMenuWillBecomeInvisible(PopupMenuEvent e) {
            }

            @Override
            public void popupMenuCanceled(PopupMenuEvent e) {
            }
        });

        addItemListener(new ItemListener() {

            @Override
            public void itemStateChanged(ItemEvent e) {
                JComboBox comboBox = null;

                MyEnum selectedConstant = MyEnum.getEnum(e.getItem().toString());
                if (e.getStateChange() == ItemEvent.SELECTED && selectedConstant != MyEnum.BOTH) {
                    comboBox = (JComboBox) e.getSource();
                    final int selectedRow = _table.rowAtPoint(comboBox.getLocation());
                    final int selectedColumn = _table.columnAtPoint(comboBox.getLocation());


                    if (selectedRow != 0) {
                        boolean allEqual = false;
                        boolean allNotEqual = false;
                        for (int i = 0; i < 3; i++) {
                            if (0 == i) {
                                continue;
                            }

                            if (selectedRow == i) {
                                allEqual = true;
                            } else if (selectedConstant == MyEnum.getEnum(_table.getValueAt(i,
                                                                                            selectedColumn)
                                    .toString())) {
                                allEqual = true;
                            } else {
                                allNotEqual = true;
                            }
                        }

                        if (allEqual && !allNotEqual) {
                            setHeaderValue(selectedConstant.getDescription(),
                                           0,
                                           selectedColumn);
                        } else {
                            setHeaderValue(MyEnum.BOTH.getDescription(),
                                           0,
                                           selectedColumn);
                        }
                        // do something
                    } else {
                        for (int i = 0; i < 3; i++) {
                            if (selectedRow != i) {
                                _table.setValueAt(selectedConstant.getDescription(),
                                                  i,
                                                  selectedColumn);
                                // do something
                            }
                        }
                    }
                }
            }

            protected void setHeaderValue(final String value,
                                          final Integer row,
                                          final int column) {
                _table.setValueAt(value, row, column);
            }

        });
    }
}

public enum MyEnum {
    BOTH("both"),
    ONE("one"),
    TWO("two");

    private String _description;

    private MyEnum(String description) {
        _description = description;
    }

    public synchronized final String getDescription() {
        return _description;
    }

    public static MyEnum getEnum(final String description) {
        for (final MyEnum element : MyEnum.values()) {
            if (element.getDescription().equals(description)) {
                return element;
            }
        }

        throw new IllegalArgumentException(
                "No enum in const class " + MyEnum.class.getCanonicalName()
                        + " contains description '" + description + "'.");
    }

}
}

2 个答案:

答案 0 :(得分:1)

我无法对您的代码进行测试或建议修复,因为它不是SSCCE,因此我现在只能提供建议。

而不是PopupMenuListener让您的组合检测表格中的所选项目,如何在单独的JButton的ActionListener中执行此操作?

或者,如果您尝试捕获用户点击的表格中的任何单元格,您可以为该列添加TableColumnModelListener,为该行添加ListSelectionListener以实现相同。

答案 1 :(得分:1)

    在Swing中
  1. 不能同时在show()两个弹出窗口中进行,

  2. JPopupMenu will (immediatelly) hide() by click to JComboboxes Arrow Button

  3. 可以将JPopup更改为JWindow,更好的解开JDialog

  4. example how to add / remove / modify JMenuItems之前的JPopupMenu可见,在{{1}中添加所需的更改在代码行mousePressed执行之前,mouseReleased更好,