Android实现模拟搜索功能
本文实例为大家分享了Android实现模拟搜索功能的具体代码,供大家参考,具体内容如下
先看效果图,合适了再接着往下看:
我们看到的这个页面,是由两部分组成,顶部的自定义的搜索框,和listView组成。
首先我们来实现布局页面,自定义搜索框,和设置listView
<?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/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="http://schemas.android.com/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(R.layout.item_search, null); hold.tv_simulation = convertView.findViewById(R.id.tv_simulation); convertView.setTag(hold); } else { hold = (ViewHold) convertView.getTag(); } Simulation simulation = list.get(position); hold.tv_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(R.layout.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(); } }); } }
这样就实现了模拟搜索的功能,并且在代码中已经给出了详细的注释。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持猪先飞。
相关文章
- 下面我们来看一篇关于Android子控件超出父控件的范围显示出来方法,希望这篇文章能够帮助到各位朋友,有碰到此问题的朋友可以进来看看哦。 <RelativeLayout xmlns:an...2016-10-02
Android开发中findViewById()函数用法与简化
findViewById方法在android开发中是获取页面控件的值了,有没有发现我们一个页面控件多了会反复研究写findViewById呢,下面我们一起来看它的简化方法。 Android中Fin...2016-09-20- 如果我们的项目需要做来电及短信的功能,那么我们就得在Android模拟器开发这些功能,本来就来告诉我们如何在Android模拟器上模拟来电及来短信的功能。 在Android模拟...2016-09-20
- 夜神android模拟器如何设置代理呢?对于这个问题其实操作起来是非常的简单,下面小编来为各位详细介绍夜神android模拟器设置代理的方法,希望例子能够帮助到各位。 app...2016-09-20
- 为了增强android应用的用户体验,我们可以在一些Button按钮上自定义动态的设置一些样式,比如交互时改变字体、颜色、背景图等。 今天来看一个通过重写Button来动态实...2016-09-20
- 如果我们要在Android应用APP中加载html5页面,我们可以使用WebView,本文我们分享两个WebView加载html5页面实例应用。 实例一:WebView加载html5实现炫酷引导页面大多...2016-09-20
- 深入理解Android中View和ViewGroup从组成架构上看,似乎ViewGroup在View之上,View需要继承ViewGroup,但实际上不是这样的。View是基类,ViewGroup是它的子类。本教程我们深...2016-09-20
- 这篇文章主要介绍了vue+高德地图实现地图搜索及点击定位操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...2020-09-09
- 下面我们来看一篇关于Android自定义WebView网络视频播放控件开发例子,这个文章写得非常的不错下面给各位共享一下吧。 因为业务需要,以下代码均以Youtube网站在线视...2016-10-02
- java开发的Android应用,性能一直是一个大问题,,或许是Java语言本身比较消耗内存。本文我们来谈谈Android 性能优化之MemoryFile文件读写。 Android匿名共享内存对外A...2016-09-20
- TextView默认是横着显示了,今天我们一起来看看Android设置TextView竖着显示如何来实现吧,今天我们就一起来看看操作细节,具体的如下所示。 在开发Android程序的时候,...2016-10-02
android.os.BinderProxy cannot be cast to com解决办法
本文章来给大家介绍关于android.os.BinderProxy cannot be cast to com解决办法,希望此文章对各位有帮助呀。 Android在绑定服务的时候出现java.lang.ClassCastExc...2016-09-20- 这篇文章主要介绍了Android 实现钉钉自动打卡功能的步骤,帮助大家更好的理解和学习使用Android,感兴趣的朋友可以了解下...2021-03-15
- 下面我们来看一篇关于Android 开发之布局细节对比:RTL模式 ,希望这篇文章对各位同学会带来帮助,具体的细节如下介绍。 前言 讲真,好久没写博客了,2016都过了一半了,赶紧...2016-10-02
- 首先如果要在程序中使用sdcard进行存储,我们必须要在AndroidManifset.xml文件进行下面的权限设置: 在AndroidManifest.xml中加入访问SDCard的权限如下: <!--...2016-09-20
- 下面来给各位简单的介绍一下关于Android开发之PhoneGap打包及错误解决办法,希望碰到此类问题的同学可进入参考一下哦。 在我安装、配置好PhoneGap项目的所有依赖...2016-09-20
用Intel HAXM给Android模拟器Emulator加速
Android 模拟器 Emulator 速度真心不给力,, 现在我们来介绍使用 Intel HAXM 技术为 Android 模拟器加速,使模拟器运行度与真机比肩。 周末试玩了一下在Eclipse中使...2016-09-20- 在安卓开发时我碰到一个问题就是需要实现全屏,但又需要我们来判断出用户是使用了全屏或非全屏了,下面我分别找了两段代码,大家可参考。 先来看一个android屏幕全屏实...2016-09-20
Android开发中布局中的onClick简单完成多控件时的监听的利与弊
本文章来为各位介绍一篇关于Android开发中布局中的onClick简单完成多控件时的监听的利与弊的例子,希望这个例子能够帮助到各位朋友. 首先在一个控件加上这么一句:and...2016-09-20Ubuntu 系统下安装Android开发环境 Android Studio 1.0 步骤
Android Studio 是一个Android开发环境,基于IntelliJ IDEA. 类似 Eclipse ADT,Android Studio 提供了集成的 Android 开发工具用于开发和调试,可以在Linux,Mac OS X,Window...2016-09-20