【问题标题】:The clear() in custom listview adapter also updates the list data自定义列表视图适配器中的 clear() 也会更新列表数据
【发布时间】:2014-09-06 15:36:20
【问题描述】:

我正在研究一个列表视图过滤器,也就是说,有一个编辑框,如果我输入一些关键字,列表视图数据会相应地改变。首先,有一个数据集。

private ArrayList<PostItem> postList;

假设列表中已经填满了数据,这里是带有过滤的适配器:

private class PostAdapter extends ArrayAdapter<PostItem> implements
        Filterable {

    public PostAdapter(Context context, int resource, List<PostItem> items) {
        super(context, resource, items);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        View v = convertView;

        if (v == null) {
            LayoutInflater vi;
            vi = LayoutInflater.from(getContext());
            v = vi.inflate(R.layout.post_item, parent, false);
        }

        PostItem post = getItem(position);

        if (post != null) {
            TextView title = (TextView) v.findViewById(R.id.postTitle);
            TextView date = (TextView) v.findViewById(R.id.postDate);

            v.setTag(post.id);
            title.setText(Html.fromHtml(post.title));
            date.setText(post.name + " "
                    + getResources().getString(R.string.post_on) + " "
                    + post.date);
        }

        return v;

    }

    @Override
    public Filter getFilter() {

        Filter filter = new Filter() {

            @SuppressWarnings("unchecked")
            @Override
            protected void publishResults(CharSequence constraint,
                    FilterResults results) {

                ArrayList<PostItem> filteredList = (ArrayList<PostItem>)results.values;

                clear();

                for (PostItem item :filteredList) {
                    add(item);
                }
            }

            @Override
            protected FilterResults performFiltering(CharSequence constraint) {
                FilterResults results = new FilterResults();
                ArrayList<PostItem> filteredArray = new ArrayList<PostItem>();

                constraint = constraint.toString().toLowerCase();

                for (PostItem item : postList) {
                    if (item.title.toLowerCase().contains(constraint)) {
                        filteredArray.add(item);
                    }
                }

                results.count = filteredArray.size();
                results.values = filteredArray;
                return results;
            }
        };

        return filter;
    }

}

问题是,clear() 函数还会删除 postList 中包含的所有项目,并插入所有最新的过滤结果。如何解决?我试图有一个复制的变量,例如copyPostList = postList 但还是一样。感谢您的帮助

更新:更多解释:

  1. 首先,postList 有项目“abc”;

  2. 然后,输入“ab”,列表显示“abc”,正确

  3. 然后,输入“xyz”,列表什么都不显示,这是正确的

  4. 最后输入“abc”,列表不显示,不正确

【问题讨论】:

标签: android listview android-listview


【解决方案1】:
public class AutoCompleteAdapter extends ArrayAdapter<String> implements Filterable {

    private ArrayList<String> fullList;
    private ArrayList<String> mOriginalValues;
    private ArrayFilter mFilter;

    public AutoCompleteAdapter(Context context, int resource, List<String> objects) {

        super(context, resource, objects);
        fullList = (ArrayList<String>) objects;
        mOriginalValues = new ArrayList<String>(fullList);

    }

    @Override
    public void add(String object) {
        super.add(object);
        fullList.add(object);
        mOriginalValues.add(object);
    }

    @Override
    public void clear() {
        super.clear();
        fullList.clear();
        mOriginalValues.clear();
    }

    @Override
    public int getCount() {
        return fullList.size();
    }

    @Override
    public String getItem(int position) {
        return fullList.get(position);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        TextView view = (TextView) super.getView(position, convertView, parent);
        view.setEllipsize(TextUtils.TruncateAt.START);
        return view;
    }

    @Override
    public Filter getFilter() {
        if (mFilter == null) {
            mFilter = new ArrayFilter();
        }
        return mFilter;
    }


    private class ArrayFilter extends Filter {

        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();

            if (mOriginalValues == null) {
                mOriginalValues = new ArrayList<String>(fullList);
            }
            ArrayList<String> list = new ArrayList<String>(mOriginalValues);
            results.values = list;
            results.count = list.size();


            return results;
        }

        @SuppressWarnings("unchecked")
        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {

            if (results.values != null) {
                fullList = (ArrayList<String>) results.values;
            } else {
                fullList = new ArrayList<String>();
            }
            if (results.count > 0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }
        }
    }
}

检查一下。这个适配器很好用。祝你好运。

【讨论】:

  • 是的。对不起。主要思想是保留原始数据列表。你试过了吗,对你有帮助吗?
  • 最后的建议。尝试使用 BaseAdapter 而不是 ArrayAdapter。在这种情况下,您必须自己实现一些逻辑,但这会更好,因为您将只有一个包含数据的列表。在 ArrayAdapter 实现中,您有像 fullList 这样的 ArrayList,但您看不到))。祝你好运!
  • 查看源代码。 ArrayAdapter 具有看起来像您需要的 ArrayFilter 实现。可能它会为您提供很好的解决方案。 github.com/android/platform_frameworks_base/blob/master/core/…
【解决方案2】:

最终代码,为我的案例添加过滤和getView

    private class PostAdapter extends ArrayAdapter<PostItem> implements Filterable {

    private ArrayList<PostItem> fullList;
    private ArrayList<PostItem> mOriginalValues;        
    private ArrayFilter mFilter;

    public PostAdapter(Context context, int resource, List<PostItem> objects) {
        super(context, resource, objects);
        fullList = (ArrayList<PostItem>) objects;
        mOriginalValues = new ArrayList<PostItem>(fullList);

    }

    @Override
    public void add(PostItem object) {
        super.add(object);
        fullList.add(object);
        mOriginalValues.add(object);
    }

    @Override
    public void clear() {
        super.clear();
        fullList.clear();
        mOriginalValues.clear();
    }

    @Override
    public int getCount() {
        return fullList.size();
    }

    @Override
    public PostItem getItem(int position) {
        return fullList.get(position);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {

        View v = convertView;

        if (v == null) {
            LayoutInflater vi;
            vi = LayoutInflater.from(getContext());
            v = vi.inflate(R.layout.post_item, parent, false);
        }

        PostItem post = getItem(position);

        if (post != null) {
            TextView title = (TextView) v.findViewById(R.id.postTitle);
            TextView date = (TextView) v.findViewById(R.id.postDate);

            v.setTag(post.id);
            title.setText(Html.fromHtml(post.title));
            date.setText(post.name + " "
                    + getResources().getString(R.string.post_on) + " "
                    + post.date);
        }

        return v;
    }

    @Override
    public Filter getFilter() {
        if (mFilter == null) {
            mFilter = new ArrayFilter();
        }
        return mFilter;
    }


    private class ArrayFilter extends Filter {

        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();
            ArrayList<PostItem> list =  new ArrayList<PostItem>();

            if (mOriginalValues == null) {
                mOriginalValues = new ArrayList<PostItem>(fullList);
            }

            for (PostItem item : mOriginalValues) {
                if (item.title.toLowerCase().contains(prefix.toString().toLowerCase())) {
                    list.add(item);
                }
            }

            results.values = list;
            results.count = list.size();
            return results;
        }

        @SuppressWarnings("unchecked")
        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {

            if (results.values != null) {
                fullList = (ArrayList<PostItem>) results.values;
            } else {
                fullList = new ArrayList<PostItem>();
            }
            if (results.count > 0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }
        }
    }
}

感谢您的帮助

【讨论】:

    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 2011-12-31
    • 1970-01-01
    • 2012-02-21
    • 2014-05-02
    • 1970-01-01
    相关资源
    最近更新 更多