java - 使用 JComboBox 作为搜索

标签 java database swing search jcombobox

我想知道如何让 JComboBox 在我们输入组合框时自动建议元素。例如,如果我使用 MySQL [Amila, Nimila, Nikila, Sunil] 加载了一个包含这些名称的 JComboBox,当我在组合框中键入“N”时,它会下拉列表,仅显示 Nimila & Nikila 作为建议。我正在尝试在 NetBeans 上执行此操作。谢谢。

最佳答案

可能重复 JComboBox autocomplete . 但是,如果您想使用基于纯 swing 的解决方案,请遵循本文。很全面

http://www.orbital-computer.de/JComboBox/

如果您想要快速简单的解决方案,只需将此类添加到您的项目中即可;

import java.awt.event.*;
import java.beans.PropertyChangeEvent;
import java.beans.PropertyChangeListener;
import javax.swing.*;
import javax.swing.text.*;

/* This work is hereby released into the Public Domain.
 * To view a copy of the public domain dedication, visit
 * http://creativecommons.org/licenses/publicdomain/
 */
public class AutoCompletion extends PlainDocument {
JComboBox comboBox;
ComboBoxModel model;
JTextComponent editor;
// flag to indicate if setSelectedItem has been called
// subsequent calls to remove/insertString should be ignored
boolean selecting=false;
boolean hidePopupOnFocusLoss;
boolean hitBackspace=false;
boolean hitBackspaceOnSelection;

KeyListener editorKeyListener;
FocusListener editorFocusListener;

public AutoCompletion(final JComboBox comboBox) {
    this.comboBox = comboBox;
    model = comboBox.getModel();
    comboBox.addActionListener(new ActionListener() {
        public void actionPerformed(ActionEvent e) {
            if (!selecting) highlightCompletedText(0);
        }
    });
    comboBox.addPropertyChangeListener(new PropertyChangeListener() {
        public void propertyChange(PropertyChangeEvent e) {
            if (e.getPropertyName().equals("editor")) configureEditor((ComboBoxEditor) e.getNewValue());
            if (e.getPropertyName().equals("model")) model = (ComboBoxModel) e.getNewValue();
        }
    });
    editorKeyListener = new KeyAdapter() {
        public void keyPressed(KeyEvent e) {
            if (comboBox.isDisplayable()) comboBox.setPopupVisible(true);
            hitBackspace=false;
            switch (e.getKeyCode()) {
                // determine if the pressed key is backspace (needed by the remove method)
                case KeyEvent.VK_BACK_SPACE : hitBackspace=true;
                hitBackspaceOnSelection=editor.getSelectionStart()!=editor.getSelectionEnd();
                break;
                // ignore delete key
                case KeyEvent.VK_DELETE : e.consume();
                comboBox.getToolkit().beep();
                break;
            }
        }
    };
    // Bug 5100422 on Java 1.5: Editable JComboBox won't hide popup when tabbing out
    hidePopupOnFocusLoss=System.getProperty("java.version").startsWith("1.5");
    // Highlight whole text when gaining focus
    editorFocusListener = new FocusAdapter() {
        public void focusGained(FocusEvent e) {
            highlightCompletedText(0);
        }
        public void focusLost(FocusEvent e) {
            // Workaround for Bug 5100422 - Hide Popup on focus loss
            if (hidePopupOnFocusLoss) comboBox.setPopupVisible(false);
        }
    };
    configureEditor(comboBox.getEditor());
    // Handle initially selected object
    Object selected = comboBox.getSelectedItem();
    if (selected!=null) setText(selected.toString());
    highlightCompletedText(0);
}

public static void enable(JComboBox comboBox) {
    // has to be editable
    comboBox.setEditable(true);
    // change the editor's document
    new AutoCompletion(comboBox);
}

void configureEditor(ComboBoxEditor newEditor) {
    if (editor != null) {
        editor.removeKeyListener(editorKeyListener);
        editor.removeFocusListener(editorFocusListener);
    }

    if (newEditor != null) {
        editor = (JTextComponent) newEditor.getEditorComponent();
        editor.addKeyListener(editorKeyListener);
        editor.addFocusListener(editorFocusListener);
        editor.setDocument(this);
    }
}

public void remove(int offs, int len) throws BadLocationException {
    // return immediately when selecting an item
    if (selecting) return;
    if (hitBackspace) {
        // user hit backspace => move the selection backwards
        // old item keeps being selected
        if (offs>0) {
            if (hitBackspaceOnSelection) offs--;
        } else {
            // User hit backspace with the cursor positioned on the start => beep
            comboBox.getToolkit().beep(); // when available use: UIManager.getLookAndFeel().provideErrorFeedback(comboBox);
        }
        highlightCompletedText(offs);
    } else {
        super.remove(offs, len);
    }
}

public void insertString(int offs, String str, AttributeSet a) throws BadLocationException {
    // return immediately when selecting an item
    if (selecting) return;
    // insert the string into the document
    super.insertString(offs, str, a);
    // lookup and select a matching item
    Object item = lookupItem(getText(0, getLength()));
    if (item != null) {
        setSelectedItem(item);
    } else {
        // keep old item selected if there is no match
        item = comboBox.getSelectedItem();
        // imitate no insert (later on offs will be incremented by str.length(): selection won't move forward)
        offs = offs-str.length();
        // provide feedback to the user that his input has been received but can not be accepted
        comboBox.getToolkit().beep(); // when available use: UIManager.getLookAndFeel().provideErrorFeedback(comboBox);
    }
    setText(item.toString());
    // select the completed part
    highlightCompletedText(offs+str.length());
}

private void setText(String text) {
    try {
        // remove all text and insert the completed string
        super.remove(0, getLength());
        super.insertString(0, text, null);
    } catch (BadLocationException e) {
        throw new RuntimeException(e.toString());
    }
}

private void highlightCompletedText(int start) {
    editor.setCaretPosition(getLength());
    editor.moveCaretPosition(start);
}

private void setSelectedItem(Object item) {
    selecting = true;
    model.setSelectedItem(item);
    selecting = false;
}

private Object lookupItem(String pattern) {
    Object selectedItem = model.getSelectedItem();
    // only search for a different item if the currently selected does not match
    if (selectedItem != null && startsWithIgnoreCase(selectedItem.toString(), pattern)) {
        return selectedItem;
    } else {
        // iterate over all items
        for (int i=0, n=model.getSize(); i < n; i++) {
            Object currentItem = model.getElementAt(i);
            // current item starts with the pattern?
            if (currentItem != null && startsWithIgnoreCase(currentItem.toString(), pattern)) {
                return currentItem;
            }
        }
    }
    // no item starts with the pattern => return null
    return null;
}

// checks if str1 starts with str2 - ignores case
private boolean startsWithIgnoreCase(String str1, String str2) {
    return str1.toUpperCase().startsWith(str2.toUpperCase());
}

private static void createAndShowGUI() {
    // the combo box (add/modify items if you like to)
    final JComboBox comboBox = new JComboBox(new Object[] {"Ester", "Jordi", "Jordina", "Jorge", "Sergi"});
    enable(comboBox);

    // create and show a window containing the combo box
    final JFrame frame = new JFrame();
    frame.setDefaultCloseOperation(3);
    frame.getContentPane().add(comboBox);
    frame.pack(); frame.setVisible(true);
}


public static void main(String[] args) {
    javax.swing.SwingUtilities.invokeLater(new Runnable() {
        public void run() {
            createAndShowGUI();
        }
    });
}
}

然后,要将自动完成功能添加到 JComboBox,只需调用即可;

AutoCompletion.enable(yourComboBox);

关于java - 使用 JComboBox 作为搜索,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16418925/

相关文章:

java - View 类监听并响应自己的组件是否是一种不好的形式

java - 为什么 TableRowSorter 弄乱了我的结果?

java Tigase 性能

java - 使用 Provider 时如何将可配置参数传递给构造函数?

database - 指定在查询中开始排序的值...Access 2016

java - 从 Mysql 获取唯一元素的数组

java - 带有 Liberty Profile 的 JSR 352 - 当 ItemReader 执行数据库查询时如何实现检查点

java - 变量以某种方式失去了值(value)?

mysql - 如何在 mysql 中拆分/反透视 group_concat?

java - 将 Object 类型转换为 JButton 类型?