7

私は Android 開発に不慣れで、解決が難しい問題に遭遇しました。AutoCompleteTextViewウィジェットを適切に使用する方法を理解しようとしています。AutoCompleteTextViewWeb サービスの XML データを使用して を作成したいと考えています。私はそれを機能させることができましたが、私は間違いなく出力に満足していません.

HashMapwith id => name のペアを入れてAutoCompleteTextView、クリックされたアイテムの ID を取得したいと思います。オートコンプリートのフィルタリングされたセットの出力をクリックすると、オートコンプリート ボックスの下にリストを入力したいと思いますが、これもうまくいきました。

これまでに行われたこと:

  • オートコンプリートは単純な ArrayList でうまく機能し、すべてのデータが正しくフィルタリングされます
  • クリック後に onItemClick イベントが適切に発生する
  • parent.getItemAtPosition(position) は、クリックされたアイテムの正しい文字列表現を返します

イベント onItemClick(AdapterView parent, View v, int position, long id) が思い通りに動作しません。クリックされた項目のフィルター処理されていない配列の位置を特定するにはどうすればよいですか? フィルタリングされたものの位置は、私が興味を持っていないものです。

その他の質問:

  • AutoCompleteTextView で HashMaps または Collections を処理する方法
  • onItemClick イベントで正しい itemId を取得する方法

この問題について非常に広範な調査を行いましたが、私の質問に答える価値のある情報は見つかりませんでした。

4

1 に答える 1

-1

AutoCompleteTextView で HashMaps または Collections を処理する方法

独自のカスタム アダプターを設定できます。アダプターでは、データを特定の位置に配置する場所はあなた次第です。

onItemClick イベントで正しい itemId を取得する方法

カスタム アダプタでフィルタを定義すると、そのフィルタによって提案されたアイテムが設定されます。2 つの異なるリストがあり、1 つは元の値を含み、もう 1 つはフィルター処理されたアイテムを含みます。私はこのようなことを意味します。

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 リスナーでは、返された値 (フィルタリングされたリストから) をマップのキーとして使用し、関連付けられた値を取得します。のインデックスを使用するリストを考えることができますHashMap。その後、 を使用Mapしてアイテムを描画したり、独自のデータを取得したりできます。

于 2011-06-11T09:35:52.370 に答える