在使用适配器的教程的帮助下,我已经成功地从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
}
});
}
最佳答案
SimpleAdapter已经启用了过滤功能。您只需要调用以下内容:
getListAdapter().getFilter().filter("Text to search for");
而已。 Android将处理您的数据过滤并显示与给定字符串匹配的数据。
更新-自定义过滤器
自定义过滤器搜索的方式涉及更多。简短的答案是,您需要编写自己的SimpleAdapter版本。 AFAIK,没有一个Android提供的适配器为如何自定义过滤逻辑提供了简单的解决方案。这就是为什么我要构建Advanced-Adapters的原因,不幸的是,我还没有为SimpleAdapter编写解决方案。现在是ArrayAdapter和SparseArray的所有替代产品。
您可能会发现很多人和例子,他们说要扩展SimpleAdapter类并重写
getFilter()
方法,以使其返回由您实现的新Filter类。然后,您可以使用mCommentList
数组列表来确定如何过滤数据。尽管这在某些情况下会起作用,但这是错误的方法,并且在各种情况下都会失败。正确的解决方案是创建自己的适配器类,其行为类似于SimpleAdapter,但扩展BaseAdapter类……这基本上意味着从头开始创建自己的适配器。对于Android来说,它是一个新的艰巨的任务,关于适配器的工作原理,还有很多要学习和理解的地方。
最简单的方法是将SimpleAdapter source code直接复制到您的项目中,然后手动将过滤器代码的逻辑更改为所需的方式。具体调整的逻辑发生在this for loop中。请记住,这在后台线程的
performFiltering()
方法中!因此,所有同步块。