过滤自定义ArrayAdapter或在活动中实现搜索

时间:2014-08-05 20:08:51

标签: android mysql listview search adapter

我已经在使用Adapter的教程的帮助下从MYSQL数据库成功实现了listview。现在我需要在显示的列表视图或某种搜索对话框中添加一个过滤器。我是android的新手。我想搜索数据库并在下一个活动中显示结果。将非常感谢详细的解释。我知道如何调用搜索对话框,但我很想继续前进。这是由listview中的数据库组成的活动:

DBLib.java

    package com.example.test;

import java.util.ArrayList;
import java.util.HashMap;

import org.json.JSONArray;
import org.json.JSONException;
import org.json.JSONObject;

import android.app.ListActivity;
import android.app.ProgressDialog;
import android.app.SearchManager;
import android.content.Intent;
import android.os.AsyncTask;
import android.os.Bundle;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.Button;
import android.widget.EditText;
import android.widget.ListAdapter;
import android.widget.ListView;
import android.widget.SimpleAdapter;

public class DBLib extends ListActivity implements OnClickListener{

    private ProgressDialog pDialog;

    private Button b_search;
    private EditText et;

    private static final String READ_DB_URL ="http://crshaggy.byethost7.com/webservice/warehouse.php";

    private static final String TAG_TITLE = "title";
    private static final String TAG_POSTS = "posts";
    private static final String TAG_ID = "id";

    private JSONArray mComments = null;

    private ArrayList<HashMap<String, String>> mCommentList;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.dblib_list);
        b_search=(Button)findViewById(R.id.listsearch);
        b_search.setOnClickListener(this);






    }

    @Override
    public void onClick(View v) {
        onSearchRequested();

    }


        @Override
    protected void onResume() {
        // TODO Auto-generated method stub
        super.onResume();
        // loading the comments via AsyncTask
        new LoadComments().execute();
    }

    public void updateJSONdata() {

    mCommentList = new ArrayList<HashMap<String, String>>();
    JSONParser jParser = new JSONParser();
    JSONObject json = jParser.getJSONFromUrl(READ_DB_URL);

    try{
        mComments = json.getJSONArray(TAG_POSTS);

        for (int i = 0; i < mComments.length(); i++) {
            JSONObject c = mComments.getJSONObject(i);

            String title = c.getString(TAG_TITLE);

        HashMap<String, String> map = new HashMap<String, String>();

        map.put(TAG_TITLE, title);

        mCommentList.add(map);
    }
} catch (JSONException e) {
            e.printStackTrace();
        }
    }


    private void updateList() {

        ListAdapter adapter = new SimpleAdapter(this, mCommentList, 
        R.layout.post, new String[] { TAG_TITLE, TAG_ID}, 
        new int[] { R.id.title});

        setListAdapter(adapter);

        ListView lv = getListView();    
        lv.setOnItemClickListener(new OnItemClickListener() {

            @Override
            public void onItemClick(AdapterView<?> parent, View view,
                    int position, long id) {
                    }
                });
            }

    public class LoadComments extends AsyncTask<Void, Void, Boolean>{

        @Override
        protected void onPreExecute() {
            super.onPreExecute();
            pDialog = new ProgressDialog(DBLib.this);
            pDialog.setMessage("Loading Warehouse...");
            pDialog.setIndeterminate(false);
            pDialog.setCancelable(true);
            pDialog.show();
        }

        @Override
        protected Boolean doInBackground(Void... arg0) {
            updateJSONdata();
            return null;

        }

        @Override
        protected void onPostExecute(Boolean result) {
            super.onPostExecute(result);
            pDialog.dismiss();
            updateList();
        }
    }




    }

列表布局

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/customgrey"
    android:paddingBottom="@dimen/activity_vertical_margin"
    android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    tools:context="com.example.test.DBLib" >

    <Button
        android:id="@+id/listsearch"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_alignParentTop="true"
        android:text="@string/searchbutton" />

    <ListView
        android:id="@android:id/list"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_alignParentLeft="true"
        android:layout_below="@+id/listsearch" >
    </ListView>

</RelativeLayout>

修改1 我已经更新了我的代码,我添加了一个onTextChanged()方法,但它给出了一个空指针异常。我刚刚更改了我在这里发布的OnPostCreate()方法:

@Override
        protected void onPostExecute(Boolean result) {
            super.onPostExecute(result);
            pDialog.dismiss();
            updateList();

            inputSearch = (EditText) findViewById(R.id.et_search);

            inputSearch.addTextChangedListener(new TextWatcher() {

                @Override
                public void onTextChanged(CharSequence cs, int arg1, int arg2, int arg3) {
                    // When user changed the Text
                    DBLib.this.adapter.getFilter().filter(cs);   
                }

                @Override
                public void beforeTextChanged(CharSequence arg0, int arg1, int arg2,
                        int arg3) {
                    // TODO Auto-generated method stub

                }

                @Override
                public void afterTextChanged(Editable arg0) {
                    // TODO Auto-generated method stub                          
                }
            });


}

1 个答案:

答案 0 :(得分:0)

SimpleAdapter已经启用了过滤功能。你只需要调用类似的东西:

getListAdapter().getFilter().filter("Text to search for");

就是这样。 Android将处理您的数据过滤并显示与给定字符串匹配的数据。

更新 - 自定义过滤器

自定义过滤器搜索的方式更多。简而言之,您需要编写自己的SimpleAdapter版本。 AFAIK,Android提供的适配器都不能为如何自定义过滤逻辑提供简单的解决方案。这就是我构建Advanced-Adapters的原因,遗憾的是我还没有为SimpleAdapter编写解决方案。它现在只是ArrayAdapter和SparseArray的替代品。

你可能会发现很多人和例子都说扩展SimpleAdapter类并覆盖getFilter()方法让它返回你实现的新Filter类。然后,您可以使用mCommentList arraylist来了解如何过滤数据。虽然这在某些情况下会起作用,但这是不正确的方法,并且会因各种情况而失败。

正确的解决方案是创建自己的适配器类,其行为类似于SimpleAdapter,但扩展了BaseAdapter类......这基本上意味着从头开始创建自己的适配器。作为Android新手,这可能是一项非常艰巨的任务,需要学习和理解适配器的工作方式。

最简单的方法是将SimpleAdapter source code复制到项目中,然后手动将过滤器代码的逻辑更改为您想要的方式。具体调整的逻辑发生在this for loop内。请记住,这是在后台线程上的performFiltering()方法内!因此所有同步的块。