Toolbar 结合 SearchView实现搜索功能(具备历史记录)

效果(不动请戳大图)

toolbar.gif

ToolBar添加 menu

toolbar.inflateMenu(R.menu.toolbar_menu);

menu 文件

<menu>
<item
        android:id="@+id/action_search_kl"
        android:title="@string/action_search_knowledge"
        android:orderInCategory="80"
        app:actionViewClass="android.support.v7.widget.SearchView" 
        android:icon="@drawable/ic_search"
        app:showAsAction="always" />
    
</menu>

根据 itemId 找到该 item


MenuItem item=  toolbar.findViewById(R.id.action_search_kl);

通过 menuitem 获取里面的 view,进行各种样式更改

private void initSearchView(final MenuItem item){
  
//通过 item 获取 actionview
    final SearchView searchView = (SearchView) item.getActionView();
        searchView.setQueryHint("搜索知识库");    
    
    //改变默认的搜索图标
    ((ImageView)searchView.findViewById(R.id.search_button)).setImageResource(R.drawable.ic_search);
    
    //搜索监听
        searchView.setOnQueryTextListener(new SearchView.OnQueryTextListener() {
            @Override
                public boolean onQueryTextSubmit(String query) {
                    //在输入法按下搜索或者回车时,会调用次方法,在这里可以作保存历史记录的操作,我这里用了 sharepreference保存
            
             SPUtils spUtils = new SPUtils("knowledgeHistory");
                        spUtils.put(query, query);
                        presenter.searchKnowledge(query);
            
            
        
                        return false;
                    }        @Override
                public boolean onQueryTextChange(String newText) {
//输入字符则回调此方法


//当输入字符为空时,重新设置 item

if(newText==null||newText.length()==0){
 
 //由于实现了历史数据的功能,在此重新设置此 item才能实时生效
     initSearchView(item); }

            return false;
                    }
            });    
     //根据id-search_src_text获取TextView    
 searchViewOfKnowledge = (SearchView.SearchAutoComplete) searchView.findViewById(R.id.search_src_text);    
//改变输入文字的颜色
    searchViewOfKnowledge.setTextColor(ContextCompat.getColor(HomeTabActivity.this, R.color.colorAccent));
    try {
    
//取出历史数据,你可以利用其他方式
        final List<String> arr = new ArrayList<>();  
        
        SPUtils spUtils = new SPUtils("knowledgeHistory");
                Map<String, ?> map = spUtils.getAll();
        
                for (String key : map.keySet()) {
                    arr.add(map.get(key).toString());
                    }
        
        //显示历史数据列表
                searchViewOfKnowledge.setThreshold(0);

//历史数据列表的 adapter,必须继承 ArrayAdater 或实现 filterable接口
        HistoryAdapter adapter = new HistoryAdapter(HomeTabActivity.this, R.layout.item_history, arr,searchView);
        
       //设置 adapter
                searchViewOfKnowledge.setAdapter(adapter);
        
        //如果重写了 Adapter 的 getView 方法,可以不用实现 item 监听(实现了也没用),否则必须实现监听,不然会报错 
                searchViewOfKnowledge.setOnItemClickListener(new AdapterView.OnItemClickListener() {
        
                    @Override
                        public void onItemClick(AdapterView<?> adapterView, View view, int position, long l) {
                            searchView.setQuery(arr.get(position), true);
                            }
                    });
            } catch (NullPointerException e) {
            e.printStackTrace();
            }
    
//searchview 的关闭监听
    searchView.setOnCloseListener(new SearchView.OnCloseListener() {        @Override        public boolean onClose() {
                    return false;
                    }
            });}


历史列表的 adapter,带删除功能

item 布局

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout 
xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
        android:gravity="center_vertical"
        android:layout_width="match_parent"
        android:background="@drawable/touch_bg"
        android:padding="10dp"
        android:clickable="true"
        android:layout_height="wrap_content">
    <TextView
            android:id="@+id/titleTv"
                android:text="hahaha"
                android:layout_centerVertical="true"
                android:layout_width="match_parent"
                android:layout_height="wrap_content" />
    <ImageButton
            android:background="@drawable/ic_close"
                android:layout_width="wrap_content"
                android:layout_height="wrap_content"
                android:layout_centerVertical="true"
                android:layout_alignParentRight="true"
                android:layout_gravity="right"
                android:id="@+id/imageButton" />
        </RelativeLayout>


adapter

构造方法必须带有布局ID,传入SearchView 是为了点击 item 之后可以在SearchView 直接显示点击的文字

public HistoryAdapter(Context context, int resourceId, List<String> titles,SearchView searchView) {
    super(context, resourceId, objects);
        this.context=context;
        this.titles=titles;
        this.resourceId=resourceId;
        this.searchView=searchView;
    }

省略 getItem,getCount,getItemId方法,主要看 getView 中删除按钮的点击方法


holder.imageButton.setOnClickListener(new View.OnClickListener() {
    @Override
        public void onClick(View view) {
            SPUtils spUtils = new SPUtils("knowledgeHistory");
        
            //从数据中移除该项
            spUtils.remove(titles.get(i));
        
            //从数据中移除该项
                titles.remove(i);
        
        //进行刷新
                notifyDataSetChanged();    }});


必须重写 ArrayAdapter的getFilter()方法,如果不重写此方法,只会一直显示历史列表,则无法实现输入文字进行筛选的效果


//用于保存原始数据

private List<String> mOriginalValues;

@NonNull@Override
public Filter getFilter() {
    Filter filter = new Filter() {
        @SuppressWarnings("unchecked")
                @Override
                protected void publishResults(CharSequence constraint,FilterResults results) {
                    titles = (List<String>) results.values; // 得到筛选后的列表结果
                        notifyDataSetChanged();  // 刷新数据        }


@Override
        protected FilterResults performFiltering(CharSequence constraint) {
                    FilterResults results = new FilterResults();  
                            List<String> filteredArrList = new ArrayList<String>();
            if (mOriginalValues == null) {
            
            //保存一份未筛选前的完整数据
                            mOriginalValues = new ArrayList<String>(titles);
                           }             if (constraint == null || constraint.length() == 0) {                //如果接收到的文字为空,则不作比较,直接返回未筛选前的完整数据
                                results.count = mOriginalValues.size();
                                 results.values = mOriginalValues;
                            } else {
                            //遍历原始数据,与接收到的文字作比较,得到筛选结果
                                constraint = constraint.toString().toLowerCase();
                                for (int i = 0; i < mOriginalValues.size(); i++) {
                                    String data = mOriginalValues.get(i);                    if(data.toLowerCase().startsWith(constraint.toString())) {
                                            filteredArrList.add(data);
                                            }
                                    }
                                //返回得到的筛选列表
                                results.count = filteredArrList.size();
                                results.values = filteredArrList;            }            return results;        }    };    return filter;} 

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,189评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,577评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,857评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,703评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,705评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,620评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,995评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,656评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,898评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,639评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,720评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,395评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,982评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,953评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,195评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,907评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,472评论 2 342

推荐阅读更多精彩内容