11

我有一个JTextField我希望建议结果与用户输入相匹配的结果。我将这些建议显示JListJPopupMenu.

但是,当通过编程方式打开弹出菜单时show(Component invoker, int x, int y),焦点从JTextField.

奇怪的是,如果我setVisible(true)改为调用,焦点并没有被盗;但是JPopupMenu没有附加到任何面板上,并且当在盒子打开时最小化应用程序时,它会停留在窗口上。

我也尝试将焦点重置为JTextFieldusing requestFocus(),但随后我必须恢复插入符号的位置 using SwingUtilities.invokeLater(),并且稍后的调用给用户一点余地来处理现有内容/覆盖它/或做其他不可预测的事情。

我得到的代码实际上是:

JTextField field = new JTextField();
JPopupMenu menu = new JPopupMenu();

field.addKeyListener(new KeyAdapter() {
    public void keyTyped(KeyEvent e) {
        JList list = getAListOfResults();

        menu.add(list);
        menu.show(field, 0, field.getHeight());
    }
});

任何人都可以建议最好的途径来以JPopupMenu编程方式显示,同时保持对 的关注JTextField

4

3 回答 3

14

技术答案是将弹出窗口的可聚焦属性设置为 false:

popup.setFocusable(false);

这意味着 textField 必须接管通常由列表本身处理的所有键盘和鼠标触发的操作,例如:

final JList list = new JList(Locale.getAvailableLocales());
final JPopupMenu popup = new JPopupMenu();
popup.add(new JScrollPane(list));
popup.setFocusable(false);
final JTextField field = new JTextField(20);
Action down = new AbstractAction("nextElement") {

    @Override
    public void actionPerformed(ActionEvent e) {
       int next = Math.min(list.getSelectedIndex() + 1,
               list.getModel().getSize() - 1);
       list.setSelectedIndex(next);
       list.ensureIndexIsVisible(next);
    }
};
field.getActionMap().put("nextElement", down);
field.getInputMap().put(
        KeyStroke.getKeyStroke("DOWN"), "nextElement");

由于您的上下文与 JComboBox 非常相似,您可能会考虑查看 BasicComboBoxUI 和 BasicComboPopup 的源代码。

编辑

只是为了好玩,以下不是回答焦点问题 :-) 相反,它演示了如何使用可排序/可过滤的 JXList 仅显示下拉列表中与键入的文本相对应的选项(此处带有开头规则)

// instantiate a sortable JXList
final JXList list = new JXList(Locale.getAvailableLocales(), true);
list.setSortOrder(SortOrder.ASCENDING);

final JPopupMenu popup = new JPopupMenu();
popup.add(new JScrollPane(list));
popup.setFocusable(false);
final JTextField field = new JTextField(20);

// instantiate a PatternModel to map text --> pattern 
final PatternModel model = new PatternModel();
model.setMatchRule(PatternModel.MATCH_RULE_STARTSWITH);
// listener which to update the list's RowFilter on changes to the model's pattern property  
PropertyChangeListener modelListener = new PropertyChangeListener() {

    @Override
    public void propertyChange(PropertyChangeEvent evt) {
        if ("pattern".equals(evt.getPropertyName())) {
            updateFilter((Pattern) evt.getNewValue());
        }
    }

    private void updateFilter(Pattern newValue) {
        RowFilter<Object, Integer> filter = null;
        if (newValue != null) {
            filter = RowFilters.regexFilter(newValue);
        }
        list.setRowFilter(filter);
    }
};
model.addPropertyChangeListener(modelListener);

// DocumentListener to update the model's rawtext property on changes to the field
DocumentListener documentListener = new DocumentListener() {

    @Override
    public void removeUpdate(DocumentEvent e) {
        updateAfterDocumentChange();
    }

    @Override
    public void insertUpdate(DocumentEvent e) {
        updateAfterDocumentChange();
    }

    private void updateAfterDocumentChange() {
        if (!popup.isVisible()) {
            popup.show(field, 0, field.getHeight());
        } 
        model.setRawText(field.getText());
    }

    @Override
    public void changedUpdate(DocumentEvent e) {
    }
};
field.getDocument().addDocumentListener(documentListener);
于 2012-04-06T11:38:20.800 回答
2

它对我来说是直截了当的。添加以下内容

field.requestFocus();

 menu.add(list);
 menu.show(field, 0, field.getHeight());

当然,您必须根据JTextField.

IE;

 menu.show(field, field.getX(), field.getY()+field.getHeight());
 menu.setVisible(true);
 field.requestFocus();
于 2012-04-05T17:34:31.327 回答
1

您可以查看 JXSearchField,它是xswingx的一部分

于 2012-08-10T12:39:07.327 回答