Android在ListFragment中实现搜索功能

3
我正在尝试在我的ListFragment中添加搜索功能。 我已经在标准Activity中完成了此操作,但现在我想将其移植到片段中。 目前我正在做的是:

  1. 我有一个Main.java,它托管片段。

  2. 我有一个ListFrag.java,它用R.id.frag_list托管列表数组List.TERM

我的Main.java像这样实现addTextChangedListener:

et.addTextChangedListener(new TextWatcher()
    {
    public void afterTextChanged(Editable s)
    {
                                                                    // Abstract Method of TextWatcher Interface.
    }
    public void beforeTextChanged(CharSequence s,
    int start, int count, int after)
    {
    // Abstract Method of TextWatcher Interface.
    }
    public void onTextChanged(CharSequence s,
    int start, int before, int count)
    {
    textlength = et.getText().length();
    array_sort.clear();
    for (int i = 0; i < List.TERM.length; i++)
    {
    if (textlength <= List.TERM[i].length())
    {
    if(et.getText().toString().equalsIgnoreCase(
    (String)
    List.TERM[i].subSequence(0,
    textlength)))
    {array_sort.add(List.TERM[i]);
    }
    }
    }
    lv.setAdapter(new ArrayAdapter<String>
    (Main.this,
            R.layout.main,R.id.frag_list, array_sort));
    }
    });

我的main.xml文件如下所示:
<LinearLayout
    android:layout_width="fill_parent"
    android:background="@drawable/searchbar"
    android:layout_height="60dp"
    android:layout_marginLeft="15dp"
    android:layout_marginRight="15dp"
    >
    <EditText android:id="@+id/EditText01"
    android:layout_height="wrap_content"
    android:layout_width="fill_parent"
    android:hint="@string/prompt">
    </EditText>
    </LinearLayout>

<fragment
    android:id="@+id/frag_list"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
android:layout_marginLeft="50dp"
android:layout_marginTop="90dp"
android:layout_marginBottom="40dp"
    class="com.bernard.ListFrag" />

当我运行应用程序时,我点击editText。在我尝试输入一个字母的瞬间,应用程序崩溃,并显示错误日志:
08-11 12:59:57.090: E/AndroidRuntime(20220): FATAL EXCEPTION: main
08-11 12:59:57.090: E/AndroidRuntime(20220): java.lang.NullPointerException
08-11 12:59:57.090: E/AndroidRuntime(20220):    at com.bernard.Main$1.onTextChanged(Main.java:54)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.widget.TextView.sendOnTextChanged(TextView.java:8105)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.widget.TextView.handleTextChanged(TextView.java:8152)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.widget.TextView$ChangeWatcher.onTextChanged(TextView.java:8532)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.sendTextChange(SpannableStringBuilder.java:892)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.change(SpannableStringBuilder.java:352)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.change(SpannableStringBuilder.java:266)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:443)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:420)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.text.SpannableStringBuilder.replace(SpannableStringBuilder.java:29)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.view.inputmethod.BaseInputConnection.replaceText(BaseInputConnection.java:669)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.view.inputmethod.BaseInputConnection.setComposingText(BaseInputConnection.java:431)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at com.android.internal.view.IInputConnectionWrapper.executeMessage(IInputConnectionWrapper.java:369)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at com.android.internal.view.IInputConnectionWrapper$MyHandler.handleMessage(IInputConnectionWrapper.java:85)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.os.Handler.dispatchMessage(Handler.java:99)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.os.Looper.loop(Looper.java:156)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at android.app.ActivityThread.main(ActivityThread.java:5025)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at java.lang.reflect.Method.invokeNative(Native Method)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at java.lang.reflect.Method.invoke(Method.java:511)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:784)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:551)
08-11 12:59:57.090: E/AndroidRuntime(20220):    at dalvik.system.NativeStart.main(Native Method)

我猜测这种错误可能是:

  1. 我应该在片段下实现editText和text监听器,而不是在Main.java下实现。
  2. 我没有从Main.java调用新列表,因此会出现NullPointerException。

有没有人有修复这个问题的建议?谢谢 :)


你能指出Main.java文件中的第54行是哪一行吗?浏览onTextChanged方法可以发现有几个地方可能会抛出NullPointerException异常,因此如果没有这些信息,很难确定实际上是哪一个从你的堆栈跟踪中抛出了异常。 - Nick
它指的是:lv.setAdapter(new ArrayAdapter<String>(...); - bunbun
2个回答

0

尝试替换:

for (int i = 0; i < List.TERM.length; i++)
    {
    if (textlength <= List.TERM[i].length())
    {
    if(et.getText().toString().equalsIgnoreCase(
    (String)
    List.TERM[i].subSequence(0,
    textlength)))
    {array_sort.add(List.TERM[i]);
    }
    }
    }
    lv.setAdapter(new ArrayAdapter<String>
    (Main.this,
            R.layout.main,R.id.frag_list, array_sort));
    }
    });

使用:

           // reuse the existing adapter instead of creating a new one:
           ArrayAdapter<String> ad = (ArrayAdapter<String>) getListView().getAdapter();
           ad.clear();
           for (int i = 0; i < List.TERM.length; i++) {
              if (textlength <= List.TERM[i].length()) {
                 if (et.getText().toString().equalsIgnoreCase(
                       (String)
                             List.TERM[i].subSequence(0,
                                   textlength))) {
                    ad.add(List.TERM[i]);
                 }
              }
           }
           ad.notifyDataSetChanged(); // tell the ListView to update

哦,还要确保在实例化ArrayAdapter的任何地方,都将一个具体的实例(例如new ArrayList<String>())作为列表参数传递进去。 - Nick
以上评论的澄清:如果您只是像许多ArrayAdapter教程所示那样传递一个数组,您将得到一个模型被冻结且不能像上面所示那样被重用的ArrayAdapter。 - Nick
非常感谢您的帮助!抱歉,我不太理解。我的意思是我应该在Main.java还是ListFrag.java中进行操作?另外,在您提出的解决方案的第二行中,我收到了“无法为Main类型定义getListView()方法”的消息。 - bunbun
我曾经以为根据类名ListFrag.java,它是继承于ListFragment,因此可以访问getListView()。但现在我看到这段代码是在Main.java中,这有点令我困惑。您可以尝试用lv变量来替换getListView(),但由于我没有看到它的初始化方式,我对lv是否有效表示怀疑。 - Nick
哈哈,这是我的主要关注点。Main.java 给了我访问 R.id.EditText01 的权限,但 ListFrag.java 没有。所以我想根据 EditText01 中的输入获取一个新列表,并将其传输到 ListFrag。我考虑在 fragment 中实现 EditText… 但只是想知道它是否仍然可以在 Main.java 中工作。 - bunbun
你应该在ListFrag类中创建一些setModel(...)方法。从Main.java中基于EditText01等生成新的列表,然后在那里调用ListFrag实例的setModel(...)方法,将新列表传递进去。你的setModel(...)方法的主体本质上就是我上面发布的片段。 - Nick

0

我自己找到了解决方法!也感谢Nick的帮助:D

虽然有点粗糙,但它能够工作。如果你认为你有更好的解决方案,请分享!

所以我做的是创建一个列表片段视图,根据http://developer.android.com/guide/components/fragments.html#CommunicatingWithActivity中的示例创建一个片段视图。

诀窍在于,在ListFrag.java中,我只需在onCreateView下实现原始活动中的所有内容即可。 重要部分如下:

@Override
public View onCreateView(LayoutInflater inflater, ViewGroup container,
        Bundle savedInstanceState) {
    // Inflate the layout for this fragment
    final View V = inflater.inflate(R.layout.list_fragment, container, false);

    lv = (ListView) V.findViewById(R.id.ListView01);
    final ArrayList <String>ar = new ArrayList<String>();
    et = (EditText) V.findViewById(R.id.EditText01);
    final String[] words = list.TERM;
    // Populate list with our static array of titles.
    lv.setAdapter(new ArrayAdapter<String>(getActivity(),
            android.R.layout.simple_list_item_activated_1, list.TERM));
    lv.setTextFilterEnabled(true); 

    et.addTextChangedListener(new TextWatcher()
    {
    public void afterTextChanged(Editable s)
    {
                                                                    // Abstract Method of TextWatcher Interface.
    }
    public void beforeTextChanged(CharSequence s,
    int start, int count, int after)
    {
    // Abstract Method of TextWatcher Interface.
    }
    public void onTextChanged(CharSequence s,
    int start, int before, int count)
    {
    textlength = et.getText().length();
    array_sort.clear();
    for (int i = 0; i < words.length; i++)
    {
    if (textlength <= words[i].length())
    {
    if(et.getText().toString().equalsIgnoreCase(
    (String)
    words[i].subSequence(0,
    textlength)))
    {array_sort.add(words[i]);
    }
    }
    }
    lv.setAdapter(new ArrayAdapter<String>
    (getActivity(),
            android.R.layout.simple_list_item_activated_1, array_sort));
    }
    });

    //Intent after selection is made
    lv.setOnItemClickListener(new OnItemClickListener() {
        @Override
        public void onItemClick(AdapterView<?> parent, View view,
                int position, long id) {
            String name = lv.getItemAtPosition(position).toString();
            for (int index = 0; index < words.length; index++)
            {
                if (name.equals(words[index]))
                {
                    position = index;
                    break;
                }
            }
            String d1  = words[position];
            ar.add(d1.toString()); 

            showDetails(position);

        }
    });



    return V;
}

这就是了,一个片段内的listview的全面搜索。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接