草庐IT

android - AutoCompleteTextView onItemClick 使用 HashMap 的项目位置或 id

coder 2023-11-28 原文

我是 Android 开发的新手,遇到了一个我觉得很难解决的问题。我想弄清楚如何正确使用 AutoCompleteTextView 小部件。我想创建一个 AutoCompleteTextView,使用来自 Web 服务的 XML 数据。我设法让它工作,但我对输出结果肯定不满意。

我想将一个带有 id => 名称对的 HashMap 放入 AutoCompleteTextView 中并获取被点击项目的 id。当我单击自动完成过滤集输出时,我想在自动完成框下方填充一个列表,我也设法开始工作。

到目前为止完成:

  • 自动完成适用于简单的 ArrayList,所有数据过滤正确
  • onItemClick 事件在点击后正确触发
  • parent.getItemAtPosition(position) 返回被点击项目的正确字符串表示

事件 onItemClick(AdapterView parent, View v, int position, long id) 的行为不符合我的意愿。我怎样才能找出被点击项目的未过滤数组位置?过滤后的位置是我不感兴趣的位置。

进一步的问题:

  • 如何在 AutoCompleteTextView 中处理 HashMap 或集合
  • 如何在onItemClick事件中获取正确的itemId

我对这个问题进行了广泛的研究,但没有找到任何有值(value)的信息来回答我的问题。

最佳答案

How to handle HashMaps or Collections in AutoCompleteTextView

您可以设置自己的自定义适配器。在您的适配器中,您可以将数据传送到给定位置取决于您。

How to get the right itemId in the onItemClick event

在您的自定义适配器中,您定义了一个过滤器,该过滤器设置了建议的项目。您有两个不同的列表,一个包含原始值,另一个包含过滤后的项目。我的意思是这样的。

private class AutoCompleteItemAdapter extends ArrayAdapter<YourItemClass> implements Filterable {

    private NameFilter      mFilter;
    List<YourItemClass> suggestions;
    List<YourItemClass> mOriginalValues;

    public AutoCompleteItemAdapter(Context context, int resource, List<YourItemClass> suggestions) {
        super(context, resource, suggestions);
        this.suggestions = suggestions;
        this.mOriginalValues = suggestions;
    }

    public void updateData(List<YourItemClass> suggestions) {
               mLock.lock();
               try{
                   this.suggestions = suggestions;
                       this.mOriginalValues = suggestions;
            finally{
                mLock.unlock();
            }
        }

    @Override
    public int getCount() {
        mLock.lock();
        try {
            return suggestions.size();
        } finally {
            mLock.unlock();
        }
    }

    @Override
    public YourItemClass getItem(int position) {
        return mOriginalValues.get(position);
    }

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        // draw your item here...
    }

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

    private class NameFilter extends Filter {
        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();

            if (mOriginalValues == null) {
                mLock.lock();
                try {
                    mOriginalValues = new ArrayList<YourItemClass>(suggestions);
                } finally {
                    mLock.unlock();
                }
            }

            if (prefix == null || prefix.length() == 0) {
                mLock.lock();
                try {
                    ArrayList<YourItemClass> list = new ArrayList<YourItemClass>(mOriginalValues);
                    results.values = list;
                    results.count = list.size();
                } finally {
                    mLock.unlock();
                }
            } else {
                String prefixString = prefix.toString().toLowerCase();

                final List<YourItemClass> values = mOriginalValues;
                final int count = values.size();

                final ArrayList<YourItemClass> newValues = new ArrayList<YourItemClass>(count);

                //              FILTERING
                //
                    // add your hits to the newValues collection
                //
                //
                results.values = newValues;
                results.count = newValues.size();
            }
            return results;
        }


        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            mLock.lock();
            try {
                if (results == null || results.values == null) return; 
                suggestions = new ArrayList<YourItemClass>();
                suggestions = (List<YourItemClass>) results.values;
                if (results.count > 0) {
                    notifyDataSetChanged();
                } else {
                    notifyDataSetInvalidated();
                }
            } finally {
                mLock.unlock();
            }
        }
    }
}

现在这可能会引发一些并发问题,正如我们所提到的,适配器可能会询问列表的大小,并且会输出更大的值,这可能会导致 getView 函数出现问题。 (例如:尝试使用底层数据绘制 5 个元素只有 4 个,因为我们进行了另一个过滤)这就是我们使用 AutoCompleteTextView 的方式,到目前为止效果很好,没问题。我刚才提到我仍然很担心,我有一种模糊的感觉,认为有更好的解决方案。

在您的 onClick 监听器中,您使用返回值(来自过滤列表)作为 map 中的键,并获取关联值。您可以认为您的列表使用了 HashMap 的索引。之后,您可以使用您的 Map 绘制您的项目,或获取您自己的数据。

关于android - AutoCompleteTextView onItemClick 使用 HashMap 的项目位置或 id,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2678819/

有关android - AutoCompleteTextView onItemClick 使用 HashMap 的项目位置或 id的更多相关文章

随机推荐