7

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

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

到目前为止完成:

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

onItemClick(AdapterView parent, View v, int position, long id) 事件的行为不像我想要的那样。如何找出单击项目的未过滤数组位置?过滤后的位置是我不感兴趣的。

进一步的问题:

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

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

4

1 回答 1

-1

如何在 AutoCompleteTextView 中处理 HashMap 或集合

您可以设置自己的自定义适配器。在您的适配器中,您可以将数据放到给定位置。

如何在 onItemClick 事件中获取正确的 itemId

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

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 函数出现问题。(fi.:尝试使用基础数据绘制 5 个元素只有 4 个,因为我们进行了另一次过滤)这是我们使用 的方式AutoCompleteTextView,到目前为止效果很好,没问题。我刚才提到我仍然很担心,我有一种模糊的感觉,认为有更好的解决方案。

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

于 2011-06-11T09:35:52.370 回答