19

getFilter()クラスのメソッドをオーバーライドする必要があり、github で hereArrayAdapter のソース コードを見つけました

//package name

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.Filter;
import android.widget.Filterable;

public class CustomAdapter<T> extends ArrayAdapter<T> implements Filterable{

    private ArrayList<T> mOriginalValues;
    private List<T> mObjects;
    private CustomFilter mFilter;
    private final Object mLock = new Object();
    public CustomAdapter(Context context, int textViewResourceId, T[] objects) {
        super(context, textViewResourceId, objects);

        mObjects = Arrays.asList(objects);
        // TODO Auto-generated constructor stub
    }

    @Override
    public Filter getFilter() {
        // TODO Auto-generated method stub
        if (mFilter == null) {
            mFilter = new CustomFilter();
        }
        return mFilter;
    }


    private class CustomFilter extends Filter {
        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();
            Log.d("bajji", "its ---> " + prefix);
            if (mOriginalValues == null) {
                synchronized (mLock) {
                    mOriginalValues = new ArrayList<T>(mObjects);
                }
            }

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

                ArrayList<T> values;
                synchronized (mLock) {
                    values = new ArrayList<T>(mOriginalValues);
                }

                final int count = values.size();
                final ArrayList<T> newValues = new ArrayList<T>();
                final ArrayList<T> approxValues = new ArrayList<T>();
                final ArrayList<T> secondApproxValues = new ArrayList<T>();


                for (int i = 0; i < count; i++) {
                    final T value = values.get(i);
                    final String valueText = value.toString().toLowerCase();
                    boolean flag = true;
                    // First match against the whole, non-splitted value
                    if (valueText.startsWith(prefixString)) {
                        newValues.add(value);
                        flag = false;
                    } else {
                        final String[] words = valueText.split(" ");
                        final int wordCount = words.length;

                        // Start at index 0, in case valueText starts with space(s)
                        for (int k = 0; k < wordCount; k++) {
                            if (words[k].startsWith(prefixString)) {
                                newValues.add(value);
                                flag = false;
                                break;
                            } 
                        }
                    }

                    if(flag) {
                        if(approxMatch(valueText, prefixString) <= 3) { //change the stuff and do a levi work
                            approxValues.add(value);
                        }
                        else {
                            final String[] words = valueText.split(" ");
                            final int wordCount = words.length;

                            // Start at index 0, in case valueText starts with space(s)
                            for (int k = 0; k < wordCount; k++) {
                                if(approxMatch(words[k], prefixString) <= 3) {
                                    //leve work
                                    secondApproxValues.add(value);
                                    break;
                                }
                            }
                        }
                    }
                }
                newValues.addAll(approxValues);
                newValues.addAll(secondApproxValues);
                results.values = newValues;
                results.count = newValues.size();
            }
            return results;
        }
        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            //noinspection unchecked
            mObjects = (List<T>) results.values;
            if (results.count > 0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }
        }
    }

    private int approxMatch (String s, String t) {
          // an approxmimate string matching algo
          return p;
    }
}

問題は、getFilter メソッドにメソッドを持つプライベート内部クラスのオブジェクトがあり、ArrayFilterそこpeformFilteringに別のコードを配置する必要があるため、クラスをオーバーライドする必要があることです。そして、メソッドで例外が発生します。

ArrayAdapter を拡張する派生クラスで、それに似たプライベート内部クラスを作成してArrayFilter呼び出したところMyFilter、メソッドで同じ例外が再び発生しましたperformFiltering

私の問題を解決する解決策を見つけました。クラス内のすべてのコードをコピーして、ArrayAdapterという新しいクラスを作成しMyAdapter、内部クラス内のいくつかのコードを変更ArrayFilterしたところ、アプリは思い通りに動作しました。しかし、それは最善の解決策ではないと感じています。

Android にはさまざまな api レベルがあるため、配列アダプターが別の api レベルで変更された場合、それらの変更をコードに追加する必要があります。したがって、コードを単にコピーして貼り付けるのではなく、クラスArrayAdapterを拡張して作成するのが最善の方法だと思います。MyAdapterArrayAdapter

親クラスの内部プライベート クラスをオーバーライドするにはどうすればよいですか?

編集:私が得る例外.. 例外

Edit2:今、質問に完全なコードを追加しました。アレイアダプターをコピーして編集すると完全に機能します..問題は拡張するときだけです..!! 現在、コードと検索は完全に機能しています。で確認しLog.iましたが、UI のオートコンプリート候補のドロップダウン リストが機能していません。入力した最初の文字のみを取得します。次の文字のフィルタリングが行われますが、UI の更新は行われません。

4

4 に答える 4

11

stackoverflow コミュニティの助けを借りて例外を削除しましたがmObjects、スーパークラスまたは基本クラスから返されたため、返された提案が実際には変化しないことが後でわかりました。問題は、2 つのパブリック メソッドがgetCount()ありgetItem(int position)、カウントを取得して、基本クラスの mObjects からリストをフェッチすることでした。したがって、クラスにこれら2つのメソッドを追加するだけです..

public int getCount() {
    return mObjects.size();
}

public T getItem(int position) {
    return mObjects.get(position);
}

mObjectsこれで派生クラスが返されます。UI のドロップダウン リストで更新されるもの。

私は Java の専門家ではありませんが、これで問題は解決します。コードを改善するための提案がある場合は、コメントに追加するか、回答を自由に編集してください。

于 2012-05-08T04:30:52.033 に答える
7

アップデート:

toString()クラスのメソッドをオーバーライドしてT、フィルター基準文字列を返す必要があります。

例えば、

class T {

    String firstName;
    String lastName;
    ...
    ...
    @override
    String toString() {
        return firstName + lastName;
    }
}

デフォルトでは、クラスのtoString()実装は次を返しますObjectgetClass().getName() + '@' + Integer.toHexString(hashCode())

オリジナル:

配列リスト参照は、ArrayAdapterコンストラクターに渡す必要があります。これを行っていないため、フィルタリング用のリストを取得できないと思います。

このサンプル コードobjectsでは、配列参照がスーパー コンストラクターに渡されます。

public CustomAdapter(Context context, int tvResId, ArrayList<String> objects) {
    super(context, textViewResourceId, objects);
    this.objects = objects;
}

フィルタリングがリスト項目の単純な文字列に基づいている場合はTextWatcher、@ Iiorry の提案に従って実装できます。ただし、フィルタリングがそれより少し複雑な場合は、Filterable同様に実装する必要があります。

于 2012-05-04T06:13:06.330 に答える
3

独自のフィルターを実装する必要はありません...@Waqasが提案したように、getFilterをオーバーライドするだけです。

これが私がしたことであり、それは機能します。必要に応じて自由に調整してください...

お役に立てば幸いです。

EditText(searchFilder):

searchFilter.addTextChangedListener(mOnSearchBoxTextChanged);
private TextWatcher mOnSearchBoxTextChanged = new TextWatcher() {

        public void afterTextChanged(Editable s) {
        }

        public void beforeTextChanged(CharSequence s, int start, int count,
                int after) {
        }

        public void onTextChanged(CharSequence s, int start, int before,
                int count) {
            mAdapter.getFilter().filter(s.toString());
        }

    };

アダプタgetFilter():

    class MyAdapter extends BaseAdapter implements Filterable {

            //the rest of the adapter....

            @Override
            public Filter getFilter() {
                if (newFilter == null) {
                    newFilter = new Filter() {
                        @Override
                        protected void publishResults(CharSequence prefix,
                                FilterResults results) {

                            CLog.logD( "Friends list filtered");

                            mAdapter.notifyDataSetChanged();
                        }

                        @Override
                protected FilterResults performFiltering(CharSequence constraint) {
                    constraint = constraint.toString().toLowerCase();

                    filteredFriendList = new ArrayList<FriendsListItemView>();

                    if (constraint!= null && constraint.toString().length() > 0) {
                        for (int i = 0; i < friendsList.size(); i++) {
                            FriendsListItemView newFriend = friendsList.get(i);
                            String name = newFriend.userName.getText().toString();

                            if (name.toLowerCase().contains(constraint)) {
                                filteredFriendList.add(newFriend);
                            } 
                        }

                    } else {
                        if (filteredFriendList.size() == 0) {
                            filteredFriendList = friendsList;
                        }
                    }

                    FilterResults newFilterResults = new FilterResults();
                    newFilterResults.count = filteredFriendList.size();
                    newFilterResults.values = filteredFriendList;
                    return newFilterResults;
                }
            };
                }
                return newFilter;
            }
        }
于 2012-05-03T13:35:03.187 に答える
1

おそらくこれはあなたを助けるかもしれません:

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import android.content.Context;
import android.util.Log;
import android.widget.ArrayAdapter;
import android.widget.Filter;

public class CustomAdapter<T> extends ArrayAdapter<T> {

    private final ArrayList<T> mOriginalValues;
    private List<T> mObjects;
    private CustomFilter mFilter;
    private final Object mLock = new Object();
    public CustomAdapter(Context context, int textViewResourceId, T[] objects) {
        super(context, textViewResourceId, objects);

        mObjects = Arrays.asList(objects);
        mOriginalValues = (ArrayList<T>) Arrays.asList(objects);
        // TODO Auto-generated constructor stub
    }

    @Override
    public Filter getFilter() {
        // TODO Auto-generated method stub
        if (mFilter == null) {
            mFilter = new CustomFilter();
        }
        return mFilter;
    }


    private class CustomFilter extends Filter {
        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();
            Log.d("bajji", "its ---> " + prefix);

            if (prefix == null || prefix.toString().trim().length() == 0) {
                ArrayList<T> list;
                synchronized (mLock) {
                    list = new ArrayList<T>(mOriginalValues);
                }
                results.values = list;
                results.count = list.size();
            } else {
                String prefixString = prefix.toString().toLowerCase();

                ArrayList<T> values;
                synchronized (mLock) {
                    values = new ArrayList<T>(mOriginalValues);
                }

                final int count = values.size();
                final ArrayList<T> newValues = new ArrayList<T>();
                final ArrayList<T> approxValues = new ArrayList<T>();
                final ArrayList<T> secondApproxValues = new ArrayList<T>();


                for (int i = 0; i < count; i++) {
                    final T value = values.get(i);
                    final String valueText = value.toString().toLowerCase();
                    boolean flag = true;
                    // First match against the whole, non-splitted value
                    if (valueText.startsWith(prefixString)) {
                        newValues.add(value);
                        flag = false;
                    } else {
                        final String[] words = valueText.split(" ");
                        final int wordCount = words.length;

                        // Start at index 0, in case valueText starts with space(s)
                        for (int k = 0; k < wordCount; k++) {
                            if (words[k].startsWith(prefixString)) {
                                newValues.add(value);
                                flag = false;
                                break;
                            } 
                        }
                    }

                    if(flag) {
                        if(approxMatch(valueText, prefixString) <= 3) { //change the stuff and do a levi work
                            approxValues.add(value);
                        }
                        else {
                            final String[] words = valueText.split(" ");
                            final int wordCount = words.length;

                            // Start at index 0, in case valueText starts with space(s)
                            for (int k = 0; k < wordCount; k++) {
                                if(approxMatch(words[k], prefixString) <= 3) {
                                    //leve work
                                    secondApproxValues.add(value);
                                    break;
                                }
                            }
                        }
                    }
                }
                newValues.addAll(approxValues);
                newValues.addAll(secondApproxValues);
                results.values = newValues;
                results.count = newValues.size();
            }
            return results;
        }
        @Override
        protected void publishResults(CharSequence constraint, FilterResults results) {
            //noinspection unchecked
            mObjects = (List<T>) results.values;
            notifyDataSetChanged();
            clear();
            for(T tmp : mObjects){
                add(tmp);
            }

            notifyDataSetChanged();
        }
    }

    private int approxMatch (String s, String t) {
          // an approxmimate string matching algo
          return p;
    }
}

getViewアダプターのメソッドで、mObjectsオブジェクトを参照して、ListView で使用可能な最近の値を取得する必要があります。

于 2012-05-07T08:57:41.720 に答える