我已经在使用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
}
});
}
答案 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()
方法内!因此所有同步的块。