首页
学习
活动
专区
工具
TVP
发布
社区首页 >问答首页 >在安卓系统中扩展ArrayAdapter

在安卓系统中扩展ArrayAdapter
EN

Stack Overflow用户
提问于 2012-04-30 14:13:02
回答 2查看 17.9K关注 0票数 19

我需要覆盖来自类ArrayAdaptergetFilter()方法,并且我找到了来自here in the github的源代码

代码语言:javascript
复制
//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类来创建MyAdapter,而不是仅仅从ArrayAdapter中复制和粘贴代码

如何重写父类的内部私有类?

编辑:我得到的异常..

Edit2:现在我在问题中添加了完整的代码。如果我复制和编辑数组适配器,它就能完美地工作。问题是只有当我扩展..!!现在,代码和搜索都运行得很完美。我和Log.i确认过了..但UI中自动完成建议的下拉列表不起作用。我只得到我输入的第一个字符,下一个字符过滤会发生,但UI更新不会发生。

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2012-05-08 12:30:52

在stackoverflow社区的一些帮助下,我删除了异常,后来我发现返回的建议并没有真正改变,因为mObjects是从超类或基类返回的。问题是有两个公共方法getCount()getItem(int position),它们获取计数并从基类的mObjects中获取列表。所以我只需要在我的类中添加这两个方法。

代码语言:javascript
复制
public int getCount() {
    return mObjects.size();
}

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

现在将返回派生类的mObjects。它们是在UI的下拉列表中更新的。

我不是java专家,但这解决了我的问题。如果您有任何建议可以使代码更好,请将其添加到评论中或随时编辑答案。

票数 11
EN

Stack Overflow用户

发布于 2012-05-04 14:13:06

更新:

您应该覆盖T类中的toString()方法,并返回筛选条件字符串。

例如,

代码语言:javascript
复制
class T {

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

缺省情况下,Object类的toString()实现返回getClass().getName() + '@' + Integer.toHexString(hashCode())

原件:

应将数组列表引用传递给ArrayAdapter构造函数。我认为你没有这样做,因此它不能获得一个列表进行过滤。

在此示例代码中,objects数组引用被传递给超级构造函数。

代码语言:javascript
复制
public CustomAdapter(Context context, int tvResId, ArrayList<String> objects) {
    super(context, textViewResourceId, objects);
    this.objects = objects;
}

如果您的过滤是基于列表项中的简单字符串,那么您可以按照@Iiorry的建议实现TextWatcher。但是,如果过滤比这复杂一点,那么您还需要实现Filterable

票数 7
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/10379261

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档