Android实现模拟搜索功能
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
Android实现模拟搜索功能本⽂实例为⼤家分享了Android实现模拟搜索功能的具体代码,供⼤家参考,具体内容如下先看效果图,合适了再接着往下看:
我们看到的这个页⾯,是由两部分组成,顶部的⾃定义的搜索框,和listView组成。
⾸先我们来实现布局页⾯,⾃定义搜索框,和设置listView
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="/apk/res/android"
xmlns:app="/apk/res-auto"
xmlns:tools="/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".SearchBoxActivity"
android:orientation="vertical"
>
<EditText
android:id="@+id/et_search"
android:layout_width="match_parent"
android:layout_height="40dp"
android:hint="搜索名称"
android:background="@drawable/btn_search"
android:layout_marginLeft="10dp"
android:layout_marginRight="10dp"
android:layout_marginTop="10dp"
android:maxLines="1"
android:maxLength="20"
android:inputType="text"
android:drawableLeft="@drawable/search"
/>
<ListView
android:id="@+id/listView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
/>
</LinearLayout>
其中EditeText控件中的 android:background="@drawable/btn_search"
这个btn_search.xml 是在drawable⽬录下定义的。
btn_search.xml
<?xml version="1.0" encoding="utf-8"?>
<shape xmlns:android="/apk/res/android"
android:shape="rectangle">
<padding
android:bottom="5dp"
android:left="5dp"
android:right="5dp"
android:top="5dp" />
<stroke
android:width="2dp"
android:color="@color/blue" />
<solid android:color="@color/white" />
<corners android:radius="20dp" />
</shape>
之后我们就来实现搜索搜索功能。
使⽤ListView控件就要给这个控件设置适配器,我们就先来创建⼀个适配器SearchAdapter,⾥⾯的list集合泛型是我⾃⼰创建的⼀个类,类⾥⾯只有⼀个String属性,实现了get和set⽅法,还有构造器。
在适配器中创建了⼀个内部类MyFilter,继承了Filter类,这个Filter类是Google官⽅提供的,实现数据过滤。
之后我们重写其中的两个⽅法performFiltering 和publishResults ⾃⼰制定过滤规则。
public class SearchAdapter extends BaseAdapter implements Filterable {
private Context context;
private ArrayList<Simulation> list = new ArrayList<>();
private MyFilter filter; //创建MyFilter对象
private FilterListener listener = null; //接⼝对象
public SearchAdapter(Context context, ArrayList<Simulation> list, FilterListener listener) {
this.context = context;
this.list = list;
this.listener = listener;
}
@Override
public int getCount() {
return list.size();
}
@Override
public Object getItem(int position) {
return list.get(position);
}
@Override
public long getItemId(int position) {
return position;
}
@Override
public View getView(int position, View convertView, ViewGroup parent) {
try {
final ViewHold hold;
if (convertView == null) {
hold = new ViewHold();
convertView = LayoutInflater.from(context).inflate(yout.item_search, null);
_simulation = convertView.findViewById(_simulation);
convertView.setTag(hold);
} else {
hold = (ViewHold) convertView.getTag();
}
Simulation simulation = list.get(position);
_simulation.setText(simulation.getText());
} catch (Exception e) {
e.printStackTrace();
}
return convertView;
}
public Filter getFilter() {
if (filter == null) {
filter = new MyFilter(list);
}
return filter;
}
/**
* 创建内部类MyFilter继承Filter类,并重写相关⽅法,实现数据的过滤
*/
class MyFilter extends Filter {
//创建集合保存原始数据
private ArrayList<Simulation> original = new ArrayList<>();
public MyFilter(ArrayList<Simulation> original) {
this.original = original;
}
//该⽅法返回搜索过滤后的数据
@Override
protected FilterResults performFiltering(CharSequence constraint) {
//创建FilterResults对象
FilterResults filterResults = new FilterResults();
/**
* 没有搜索内容的话就还是给filterResults赋值原始数据的值和⼤⼩
* 执⾏了搜索的话,根据搜索规则过滤即可,最后把过滤后的数据的值和⼤⼩赋值给filterResults
*/
if (TextUtils.isEmpty(constraint)) {
//取出当前的数据源的值和集合元素个数
//此时返回的filterResults就是原始的数据,不进⾏过滤
filterResults.values = original;
filterResults.count = original.size();
} else {
ArrayList<Simulation> mList = new ArrayList<>();
//创建集合保护过滤后的数据
for (Simulation s : original) {
//这⾥的toLowerCase():是将字符串中的字母全部变为⼩写,⽽⾮字母则不做改变
if (s.getText().trim().toLowerCase().contains(constraint.toString().trim().toLowerCase())) {
//规则匹配的话就往集合中添加该数据
mList.add(s);
}
}
filterResults.values = mList;
filterResults.count = mList.size();
}
return filterResults;
}
//该⽅法⽤来刷新⽤户界⾯,根据过滤后的数据重新展⽰列表
@Override
protected void publishResults(CharSequence constraint, FilterResults results) {
//获取过滤后的数据
list = (ArrayList<Simulation>) results.values;
//如果接⼝对象不为空,那么调⽤接⼝中的⽅法获取过滤后的数据,具体的实现在new这个接⼝的时候重写的⽅法⾥执⾏ if (listener != null) {
listener.getFilterData(list);
}
//刷新数据源显⽰
//通知数据观察者当前所关联的数据源已经发⽣改变,任何与该数据有关的视图都应该去刷新⾃⼰。
notifyDataSetChanged();
}
}
public interface FilterListener{
void getFilterData(List<Simulation> list);
}
public final class ViewHold {
private TextView tv_simulation;
}
}
之后我们在SearchBoxActivity中,对EditText控件的TextChanged进⾏实时监听,然后对输⼊的关键字与ListView中的数据源进⾏循环遍历、过滤,再把新数据源通过适配器刷新到ListView上。
这么⼀个过程。
public class SearchBoxActivity extends AppCompatActivity {
private static final String TAG = "SearchBoxActivity";
private EditText et_search;
private ListView listView;
private SearchAdapter searchAdapter;
private ArrayList<Simulation> list = new ArrayList<>();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(yout.activity_search_box);
et_search = findViewById(R.id.et_search);
listView = findViewById(R.id.listView);
String data[] = new String[]{"⼤数据", "Android开发", "Java开发", "web前端开发", "⽹页开发", "IOS开发"};
for (int i = 0; i < 6; i++) {
Simulation simulation = new Simulation(data[i]);
list.add(simulation);
}
searchAdapter = new SearchAdapter(this, list, new SearchAdapter.FilterListener() {
@Override
public void getFilterData(List<Simulation> list) {
//这⾥可以拿到过滤后的数据,所以在这⾥可以对搜索后的数据进⾏操作
Log.e(TAG, "接⼝回调成功");
Log.e(TAG, list.toString());
setItemClick(list);
}
});
//设置适配器
listView.setAdapter(searchAdapter);
//设置监听
setListeners();
}
private void setListeners() {
//没有进⾏搜索的时候,也要添加对listView的item单击监听
setItemClick(list);
/**
* 对编辑框添加⽂本改变监听,搜索的具体功能是在这⾥实现
* ⽂字改变的时候进⾏搜索,关键⽅法是重写onTextChanged()⽅法
*/
et_search.addTextChangedListener(new TextWatcher() {
//每次EditText⽂本改变之前的时候,会回调这个⽅法
@Override
public void beforeTextChanged(CharSequence s, int start, int count, int after) {
//s 输⼊框中改变前的字符串信息
//start 输⼊框中改变前的字符串的起始位置
//count 输⼊框中改变前后的字符串改变数量⼀般为0
//after 输⼊框中改变后的字符串与起始位置的偏移量
}
//每次EditText⽂本改变的时候,会回调这个⽅法
@Override
public void onTextChanged(CharSequence s, int start, int before, int count) {
//第⼀个参数s 的含义: 输⼊框中改变后的字符串信息
//start 输⼊框中改变后的字符串的起始位置
//before 输⼊框中改变前的字符串的位置默认为0
//count 输⼊框中改变后的⼀共输⼊字符串的数量
if (searchAdapter != null) {
searchAdapter.getFilter().filter(s);
}
}
//每次EditText⽂本改变之后的时候,会回调这个⽅法
@Override
public void afterTextChanged(Editable s) {
//edit 输⼊结束呈现在输⼊框中的信息
}
});
}
private void setItemClick(List<Simulation> filter_list) {
listView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
@Override
public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
Toast.makeText(SearchBoxActivity.this, filter_list.get(position).getText(), Toast.LENGTH_SHORT).show(); }
});
}
}
这样就实现了模拟搜索的功能,并且在代码中已经给出了详细的注释。
以上就是本⽂的全部内容,希望对⼤家的学习有所帮助,也希望⼤家多多⽀持。