android AutoCompleteTextView 自定义BaseAdapter

最近项目中需要做搜索功能,实现类似 Google、Baidu 搜索的 下拉提示效果。Android为我们提供了 AutoCompleteTextView 控件来完成此功能。

网上好多例子都是简单使用 ArrayAdapter 来实现的,界面比较简单,实际项目中用处不大;自己研究了一番,自定义Adapter 继承BaseAdapter 并实现Filterable 接口 实现了上述功能。

 

运行效果截图

package com.example.actv;  

import java.util.ArrayList;
import java.util.List;  

import com.example.actv.entity.PhoneContact;  

import android.os.Bundle;
import android.app.Activity;
import android.view.Menu;
import android.view.View;
import android.widget.AdapterView;
import android.widget.AdapterView.OnItemClickListener;
import android.widget.AutoCompleteTextView;  

public class MainActivity extends Activity implements OnItemClickListener {
    List<PhoneContact> mList;
    private AutoCompleteTextView mACTV;  

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);  

        buildAppData();
        findView();
    }  

    private void buildAppData() {
        String[] names = { "abc", "allen", "bird", "bike", "book", "cray",
                "david", "demon", "eclipse", "felling", "frank", "google",
                "green", "hill", "hook","jin zhiwen", "jack", "jay", "king","kevin","kobe",
                "lily", "lucy", "mike", "nike", "nail", "open","open cv",
                "panda", "pp", "queue", "ray allen", "risk", "tim cook","T-MAC","tony allen",
                "x man", "x phone", "yy", "world", "w3c", "zoom","zhu ziqing"};  

        mList = new ArrayList<PhoneContact>();  

        for (int i = 0; i < names.length; i++) {
            PhoneContact pc = new PhoneContact(100 + i, names[i], "1861234567"
                    + i, names[i].concat("@gmail.com"));
            mList.add(pc);
        }  

    }  

    private void findView() {
        mACTV = (AutoCompleteTextView) findViewById(R.id.mACTV);
        PhoneAdapter mAdapter = new PhoneAdapter(mList, getApplicationContext());
        mACTV.setAdapter(mAdapter);
        mACTV.setThreshold(1);  //设置输入一个字符 提示,默认为2  

        mACTV.setOnItemClickListener(this);
    }  

    @Override
    public boolean onCreateOptionsMenu(Menu menu) {
        // Inflate the menu; this adds items to the action bar if it is present.
        getMenuInflater().inflate(R.menu.main, menu);
        return true;
    }  

    @Override
    public void onItemClick(AdapterView<?> parent, View view, int position,
            long id) {  

        PhoneContact pc = mList.get(position);
        mACTV.setText(pc.getName()+" "+pc.getPhone());
    }  

}

  自定义Adapter

package com.example.actv;  

import java.util.ArrayList;
import java.util.List;  

import android.content.Context;
import android.view.View;
import android.view.ViewGroup;
import android.widget.BaseAdapter;
import android.widget.Filter;
import android.widget.Filterable;
import android.widget.TextView;  

import com.example.actv.entity.PhoneContact;  

public class PhoneAdapter extends BaseAdapter implements Filterable {
    private ArrayFilter mFilter;
    private List<PhoneContact> mList;
    private Context context;
    private ArrayList<PhoneContact> mUnfilteredData;  

    public PhoneAdapter(List<PhoneContact> mList, Context context) {
        this.mList = mList;
        this.context = context;
    }  

    @Override
    public int getCount() {  

        return mList==null ? 0:mList.size();
    }  

    @Override
    public Object getItem(int position) {
        // TODO Auto-generated method stub
        return mList.get(position);
    }  

    @Override
    public long getItemId(int position) {  

        return position;
    }  

    @Override
    public View getView(int position, View convertView, ViewGroup parent) {
        View view;
        ViewHolder holder;
        if(convertView==null){
            view = View.inflate(context, R.layout.phone_item, null);  

            holder = new ViewHolder();
            holder.tv_name = (TextView) view.findViewById(R.id.tv_name);
            holder.tv_phone = (TextView) view.findViewById(R.id.tv_phone);
            holder.tv_email = (TextView) view.findViewById(R.id.tv_email);  

            view.setTag(holder);
        }else{
            view = convertView;
            holder = (ViewHolder) view.getTag();
        }  

        PhoneContact pc = mList.get(position);  

        holder.tv_name.setText("姓名:"+pc.getName());
        holder.tv_phone.setText("电话:"+pc.getPhone());
        holder.tv_email.setText("Email:"+pc.getEmail());  

        return view;
    }  

    static class ViewHolder{
        public TextView tv_name;
        public TextView tv_phone;
        public TextView tv_email;
    }  

    @Override
    public Filter getFilter() {
        if (mFilter == null) {
            mFilter = new ArrayFilter();
        }
        return mFilter;
    }  

    private class ArrayFilter extends Filter {  

        @Override
        protected FilterResults performFiltering(CharSequence prefix) {
            FilterResults results = new FilterResults();  

            if (mUnfilteredData == null) {
                mUnfilteredData = new ArrayList<PhoneContact>(mList);
            }  

            if (prefix == null || prefix.length() == 0) {
                ArrayList<PhoneContact> list = mUnfilteredData;
                results.values = list;
                results.count = list.size();
            } else {
                String prefixString = prefix.toString().toLowerCase();  

                ArrayList<PhoneContact> unfilteredValues = mUnfilteredData;
                int count = unfilteredValues.size();  

                ArrayList<PhoneContact> newValues = new ArrayList<PhoneContact>(count);  

                for (int i = 0; i < count; i++) {
                    PhoneContact pc = unfilteredValues.get(i);
                    if (pc != null) {  

                        if(pc.getName()!=null && pc.getName().startsWith(prefixString)){  

                            newValues.add(pc);
                        }else if(pc.getEmail()!=null && pc.getEmail().startsWith(prefixString)){  

                            newValues.add(pc);
                        }
                    }
                }  

                results.values = newValues;
                results.count = newValues.size();
            }  

            return results;
        }  

        @Override
        protected void publishResults(CharSequence constraint,
                FilterResults results) {
             //noinspection unchecked
            mList = (List<PhoneContact>) results.values;
            if (results.count > 0) {
                notifyDataSetChanged();
            } else {
                notifyDataSetInvalidated();
            }
        }  

    }
}

  

注意:一定要实现 Filterable 接口,否则无效

 

MainActivity 布局文件

<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"
    tools:context=".MainActivity" >  

    <AutoCompleteTextView
        android:id="@+id/mACTV"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:hint="" >  

        <requestFocus />
    </AutoCompleteTextView>  

</RelativeLayout>

  

 

phone_item.xml

<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="wrap_content" >  

    <ImageView
        android:id="@+id/ivIcon"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:contentDescription="@string/app_name"
        android:src="@drawable/ic_launcher" />  

    <LinearLayout
        android:id="@+id/appInfo"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_marginLeft="5dip"
        android:layout_toRightOf="@id/ivIcon"
        android:orientation="vertical" >  

        <TextView
            android:id="@+id/tv_name"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/person_name"
            android:textColor="#000000"
            android:textSize="16sp" />  

        <TextView
            android:id="@+id/tv_phone"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/phone"
            android:textColor="#666666"
            android:textSize="13sp" />  

        <TextView
            android:id="@+id/tv_email"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:text="@string/email"
            android:textColor="#666666"
            android:textSize="13sp" />
    </LinearLayout>  

    <Button
        android:id="@+id/btnClick"
        android:layout_width="80dip"
        android:layout_height="wrap_content"
        android:layout_alignParentRight="true"
        android:layout_centerVertical="true"
        android:focusable="false"
        android:text="@string/call"
        android:textColor="#000000"
        android:textSize="16sp" />  

</RelativeLayout>

  

时间: 2024-09-24 03:32:09

android AutoCompleteTextView 自定义BaseAdapter的相关文章

Android零基础入门第42节:自定义BaseAdapter

原文:Android零基础入门第42节:自定义BaseAdapter     在ListView的使用中,有时候还需要在里面加入按钮等控件,实现单独的操作.也就是说,这个ListView不再只是展示数据,也不仅仅是这一行要来处理用户的操作,而是里面的控件要获得用户的焦点.读者可以试试用SimpleAdapter添加一个按钮到ListView的条目中,会发现可以添加,但是却无法获得焦点,点击操作被ListView的Item所覆盖.这时候最方便的方法就是使用灵活的适配器BaseAdapter了.  

Android 自定义 BaseAdapter 最佳实践

虽然现在很多新的项目都在使用RecyclerView,但是很多开发者在一些场景中还是倾向使用ListView或者GridView,然后就是需要写许多的Adapter.一次项目组在新启动一个新项目的时候,有个同事拿来了一个网上说的万能Adapter,在使用的时候发现即使在单个视图类型一旦逻辑判断比较复杂情况下非常不方便,更不用说在适配器Adapter中使用多视图类型了,这里仅是个人观点,也许没有掌握到精华. 当然了随着RecyclerView的使用,网上也有很多有关对RecyclerView多视图

Android自定义BaseAdapter最佳实践

虽然现在很多新的项目都在使用RecyclerView,但是很多开发者在一些场景中还是倾向使用ListView或者GridView,然后就是需要写许多的Adapter.一次项目组在新启动一个新项目的时候,有个同事拿来了一个网上说的万能Adapter,在使用的时候发现即使在单个视图类型一旦逻辑判断比较复杂情况下非常不方便,更不用说在适配器Adapter中使用多视图类型了,这里仅是个人观点,也许没有掌握到精华,这是有关万能适配器Adapter的一片博文 Android 快速开发系列 打造万能的List

android-求大神解答,自定义baseAdapter报错

问题描述 求大神解答,自定义baseAdapter报错 这是我的源码 package com.example.administrator.robot; import android.app.Activity;import android.content.Context;import android.support.v7.app.ActionBarActivity;import android.os.Bundle;import android.util.Log;import android.vie

Android AutoCompleteTextView连接数据库自动提示的方法(附demo源码下载)_Android

本文实例讲述了Android AutoCompleteTextView连接数据库自动提示的方法.分享给大家供大家参考,具体如下: 这个简单例子也体现MVC的思想.AutoCompleteTextView 就是View,而SimpleCursorAdapter就是Controller,SQLiteOpenHelper就相当于Model. 1.首先定义MVC中的Model,自定义DBHelper类继承SQLiteOpenHelper用于访问数据库 import android.content.Con

Android中,BaseAdapter类中怎么获取屏幕宽度?

问题描述 Android中,BaseAdapter类中怎么获取屏幕宽度? 本人小白,自定义了个MyAdapter继承BaseAdapter,现在想在MyAdapter中获取屏幕宽度,但是该类不是Activity也不是service,WindowManeger获取不到.getSystemService(Context.WINDOW_SERVICE),Resources的getResources();也没有. 解决方案 如果 控件宽度是适应屏幕的话 getView(int position View

android如何自定义TextView详解

Android控件中的TextView控件只有一个输入框,但是为了用于的操作方便我们应该实现一些功能: 1. 可以直接将内容删除的功能按钮 2. 可以记录用户以前输入的数据,同时能够将数据通过下拉显示,点击的时候实现输入 先上图: 下拉的图片没有做,所以和删除的图片使用同一个了,同志们可以直接在xml文件中更换就行了 分析: 肯定要使用自定义view来实现的,我们知道自定义view大概可以分为三类:自绘控件,组合控件,继承控件,我们这里是要进行增强的textView的功能,所以我这里使用的 是组

Android AutoCompleteTextView连接数据库自动提示的方法(附demo源码下载)

本文实例讲述了Android AutoCompleteTextView连接数据库自动提示的方法.分享给大家供大家参考,具体如下: 这个简单例子也体现MVC的思想.AutoCompleteTextView 就是View,而SimpleCursorAdapter就是Controller,SQLiteOpenHelper就相当于Model. 1.首先定义MVC中的Model,自定义DBHelper类继承SQLiteOpenHelper用于访问数据库 import android.content.Con

Android 使用自定义RecyclerView控件实现Gallery效果

上篇文章给大家介绍了Android 自定义 HorizontalScrollView 打造多图片OOM 的横向滑动效果.其实制作横向滚动的不得不说另一个控件,就是Google官方最近新增加的RecyclerView,据说是ListView的升级版本,本篇文章,首先介绍RecyclerView的用法,然后经行一定的分析:最后自定义一下RecyclerView实现我们需要的相册效果. 1.RecyclerView的基本用法 首先主Activity的布局文件: <RelativeLayout xmln