为了账号安全,请及时绑定邮箱和手机立即绑定

JavaFX中的自动完成组合框

JavaFX中的自动完成组合框

MYYA 2019-11-25 15:54:57
我正在寻找一种将自动完成功能添加到JavaFX的方法ComboBox。经过大量搜索之后,该在这里提问了。这AutoFillBox是已知的,但不是我要搜索的内容。我想要的是一个可编辑的组合框,键入时应过滤列表。但是我也想打开列表而不输入并查看整个项目。任何的想法?
查看完整描述

3 回答

?
拉丁的传说

TA贡献1789条经验 获得超8个赞

首先,您必须在项目中创建此类:


import javafx.collections.FXCollections;

import javafx.collections.ObservableList;

import javafx.event.EventHandler;

import javafx.scene.control.ComboBox;

import javafx.scene.input.KeyCode;

import javafx.scene.input.KeyEvent;


public class FxUtilTest {


    public interface AutoCompleteComparator<T> {

        boolean matches(String typedText, T objectToCompare);

    }


    public static<T> void autoCompleteComboBoxPlus(ComboBox<T> comboBox, AutoCompleteComparator<T> comparatorMethod) {

        ObservableList<T> data = comboBox.getItems();


        comboBox.setEditable(true);

        comboBox.getEditor().focusedProperty().addListener(observable -> {

            if (comboBox.getSelectionModel().getSelectedIndex() < 0) {

                comboBox.getEditor().setText(null);

            }

        });

        comboBox.addEventHandler(KeyEvent.KEY_PRESSED, t -> comboBox.hide());

        comboBox.addEventHandler(KeyEvent.KEY_RELEASED, new EventHandler<KeyEvent>() {


            private boolean moveCaretToPos = false;

            private int caretPos;


            @Override

            public void handle(KeyEvent event) {

                if (event.getCode() == KeyCode.UP) {

                    caretPos = -1;

                    if (comboBox.getEditor().getText() != null) {

                        moveCaret(comboBox.getEditor().getText().length());

                    }

                    return;

                } else if (event.getCode() == KeyCode.DOWN) {

                    if (!comboBox.isShowing()) {

                        comboBox.show();

                    }

                    caretPos = -1;

                    if (comboBox.getEditor().getText() != null) {

                        moveCaret(comboBox.getEditor().getText().length());

                    }

                    return;

                } else if (event.getCode() == KeyCode.BACK_SPACE) {

                    if (comboBox.getEditor().getText() != null) {

                        moveCaretToPos = true;

                        caretPos = comboBox.getEditor().getCaretPosition();

                    }

                } else if (event.getCode() == KeyCode.DELETE) {

                    if (comboBox.getEditor().getText() != null) {

                        moveCaretToPos = true;

                        caretPos = comboBox.getEditor().getCaretPosition();

                    }

                } else if (event.getCode() == KeyCode.ENTER) {

                    return;

                }


                if (event.getCode() == KeyCode.RIGHT || event.getCode() == KeyCode.LEFT || event.getCode().equals(KeyCode.SHIFT) || event.getCode().equals(KeyCode.CONTROL)

                        || event.isControlDown() || event.getCode() == KeyCode.HOME

                        || event.getCode() == KeyCode.END || event.getCode() == KeyCode.TAB) {

                    return;

                }


                ObservableList<T> list = FXCollections.observableArrayList();

                for (T aData : data) {

                    if (aData != null && comboBox.getEditor().getText() != null && comparatorMethod.matches(comboBox.getEditor().getText(), aData)) {

                        list.add(aData);

                    }

                }

                String t = "";

                if (comboBox.getEditor().getText() != null) {

                    t = comboBox.getEditor().getText();

                }


                comboBox.setItems(list);

                comboBox.getEditor().setText(t);

                if (!moveCaretToPos) {

                    caretPos = -1;

                }

                moveCaret(t.length());

                if (!list.isEmpty()) {

                    comboBox.show();

                }

            }


            private void moveCaret(int textLength) {

                if (caretPos == -1) {

                    comboBox.getEditor().positionCaret(textLength);

                } else {

                    comboBox.getEditor().positionCaret(caretPos);

                }

                moveCaretToPos = false;

            }

        });

    }


    public static<T> T getComboBoxValue(ComboBox<T> comboBox){

        if (comboBox.getSelectionModel().getSelectedIndex() < 0) {

            return null;

        } else {

            return comboBox.getItems().get(comboBox.getSelectionModel().getSelectedIndex());

        }

    }


}

要使组合框自动完成,请按以下方式使用它:


FxUtilTest.autoCompleteComboBoxPlus(myComboBox, (typedText, itemToCompare) -> itemToCompare.getName().toLowerCase().contains(typedText.toLowerCase()) || itemToCompare.getAge().toString().equals(typedText));

当需要从组合框中获取选定的值时,也请务必使用此方法,否则您可能会遇到一些异常,例如“类强制转换异常”:


FxUtilTest.getComboBoxValue(myComboBox);

PS:在jre 8.51和8.65之间的版本中,此方法存在一些问题,这导致了一些奇怪的行为,现在这些问题似乎不再发生了。如果您遇到问题,则可以查看对此答案所做的修改,并获得可以解决该问题的旧版本。此方法必须可以正常工作,如果您遇到任何问题,请告诉我。


查看完整回答
反对 回复 2019-11-25
  • 3 回答
  • 0 关注
  • 598 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信