8

我的程序中有一个 JTextField 和一个 JList。JList 包含用户的联系人。我想根据 JTextField 上的文本过滤 JList。例如,如果我输入“Mike”,它只会显示包括“Mike”在内的联系人。当用户清除 JTextField 时,它将重置过滤器。

我知道我可以通过拥有两个数组来手动执行此操作。一个用于原始联系人,一个用于过滤后的联系人。当用户更改 JTextField 的值时,我会查看原始列表,更新临时列表并更新 JList。我只是想知道是否有一些内置功能可以避免体力劳动。

4

3 回答 3

18

做这样的事情的最好方法是有一个过滤其内容的 ListModel 实现。我不知道任何默认的过滤 ListModel 实现,但应该不难做到。这是一个快速而肮脏的解决方案,只是为了给您一个想法。您可能想为它添加更多的花里胡哨。

package test;

import java.util.ArrayList;

import javax.swing.AbstractListModel;
import javax.swing.ListModel;
import javax.swing.event.ListDataEvent;
import javax.swing.event.ListDataListener;

public class FilteredListModel extends AbstractListModel {
    public static interface Filter {
        boolean accept(Object element);
    }

    private final ListModel _source;
    private Filter _filter;
    private final ArrayList<Integer> _indices = new ArrayList<Integer>();

    public FilteredListModel(ListModel source) {
        if (source == null)
            throw new IllegalArgumentException("Source is null");
        _source = source;
        _source.addListDataListener(new ListDataListener() {
            public void intervalRemoved(ListDataEvent e) {
                doFilter();
            }

            public void intervalAdded(ListDataEvent e) {
                doFilter();
            }

            public void contentsChanged(ListDataEvent e) {
                doFilter();
            }
        });
    }

    public void setFilter(Filter f) {
        _filter = f;
        doFilter();
    }

    private void doFilter() {
        _indices.clear();

        Filter f = _filter;
        if (f != null) {
            int count = _source.getSize();
            for (int i = 0; i < count; i++) {
                Object element = _source.getElementAt(i);
                if (f.accept(element)) {
                    _indices.add(i);
                }
            }
            fireContentsChanged(this, 0, getSize() - 1);
        }
    }

    public int getSize() {
        return (_filter != null) ? _indices.size() : _source.getSize();
    }

    public Object getElementAt(int index) {
        return (_filter != null) ? _source.getElementAt(_indices.get(index)) : _source.getElementAt(index);
    }
}

为了使用它,您需要将其设置为您的 JList,然后根据需要调用 setFilter()。这是一个例子:

    ListModel source = new DefaultListModel(); // use a model of your choice here;
    FilteredListModel filteredListModel = new FilteredListModel(source);
    JList list = new JList(filteredListModel);
    filteredListModel.setFilter(new FilteredListModel.Filter() {
        public boolean accept(Object element) {
            return false; // put your filtering logic here.
        }
    });

一旦调用 setFilter() 方法,屏幕上的 JList 就会相应地更改其内容。

或者,您可能希望为您的过滤器实现一个观察者/可观察模式,因此您可以重新过滤列表而无需调用方法 setFilter()。您可以稍后进行试验。对于第一次迭代,只要每次用户在 JTextField 中键入内容时调用方法 setFilter 就足够了。

于 2013-02-07T19:10:00.053 回答
3

一个更简单的解决方案可能是使用,它确实具有过滤和排序 ( )JTable的内置功能。RowSorter单列表与列表没有太大区别。

于 2013-02-07T18:49:40.890 回答
0

如果你对外部库没问题,我会推荐 Jide 的 QuickListFilterField/QuickTreeFilterField。只需几行代码,您就可以获得可视觉过滤的 JList/JTree、区分大小写/不区分大小写的搜索、通配符/正则表达式匹配等……非常易于使用!

于 2015-06-30T08:24:02.073 回答