Android仿手机通讯录地址选择功能

感觉比较好的一个地址选择设计,而且发现有的App中也用到了。还是先上效果图

思路:

1.效果是仿照网上大神实现的类似通讯录样式做的;
2.右边a-z是自定义的一个bar,设置了点击监听事件,以及对话框弹出
3.关键是adapter,判断了字母显示和隐藏
4.用到汉字转拼音、按首字母排序等工具类
5.3个activity的跳转是用回调来实现,每个activity都实现了回调,这样就有了从区activity直接跳转到首页的效果
6.数据是调用的我本地的接口实现的,如果大家没有数据我可以想办法给你们提供测试的省市区数据接口。加载数据是用volley框架实现的

代码的一个结构

1.右侧自定义bar的部分代码

首先重写onDraw方法

/**
   * 重写
   * @param canvas
   */
  @Override
  protected void onDraw(Canvas canvas) {
    super.onDraw(canvas);

    int height=getHeight();//获取对应的高度
    int width=getWidth();//获取对应的宽度
    int singleHeight=height/b.length;//获取每一个字母的高度

    for(int i=0;i<b.length;i++){
      paint.setColor(Color.rgb(33,65,98));
      paint.setTypeface(Typeface.DEFAULT_BOLD);
      paint.setAntiAlias(true);
      paint.setTextSize(20);
      //选中
      if(i==choose)
      {
        paint.setColor(Color.parseColor("#3399ff"));//设置选中状态颜色
        paint.setFakeBoldText(true);
      }
      //x坐标等于中间-字符串宽度的一办(????????)
      float xPos=width/2-paint.measureText(b[i])/2;
      float yPos=singleHeight*i+singleHeight;
      canvas.drawText(b[i],xPos,yPos,paint);
      paint.reset();//重置画笔
    }
  }

重写dispatchTouchEvent方法

 /**
   * 重写
   * @param event
   * @return
   */
  @TargetApi(Build.VERSION_CODES.JELLY_BEAN)
  @Override
  public boolean dispatchTouchEvent(MotionEvent event) {

    int action=event.getAction();
    float y=event.getY();//点击Y坐标
    int oldChoose=choose;
    OnTouchingLetterChangedListener listener=onTouchingLetterChangedListener;
    int c=(int)(y/getHeight()*b.length);//点击y坐标所占总高度的比例*b数组的长度就等于点击b中的个数

    switch (action){
      case MotionEvent.ACTION_UP:
        setBackground(new ColorDrawable(0*00000000));
        choose=-1;//
        invalidate();
        if(mTextDialog!=null)
        {
          mTextDialog.setVisibility(View.INVISIBLE);
        }
        break;
      default:
        setBackgroundResource(R.drawable.sidebar_background);
        if(oldChoose!=c)
        {
          if(c>=0 && c<b.length)
          {
            if(listener!=null)
            {
              listener.onTouchingLetterChanged(b[c]);
            }
            if(mTextDialog!=null)
            {
              mTextDialog.setText(b[c]);
              mTextDialog.setVisibility(View.VISIBLE);
            }

            choose=c;
            invalidate();
          }
        }

        break;
    }
    return true;
  }

向外开发接口

 /**
   * 向外公开的方法
   * @param onTouchingLetterChangedListener
   */
  public void setOnTouchingLetterChangedListener(OnTouchingLetterChangedListener onTouchingLetterChangedListener){
    this.onTouchingLetterChangedListener=onTouchingLetterChangedListener;
  }

2.adapter关键代码,以province的adapter为例,继承自SectionIndexer

/**
   * 根据ListView的当前位置获取匪类的首字母的Char ascii值
   * @param position
   * @return
   */
  public int getSectionForPosition(int position){
    return list.get(position).getSortLetters().charAt(0);
  }

  /**
   * 根据分类的首字母的Char ascii值获取其第一次出现该首字母的位置
   * @param section
   * @return
   */
  public int getPositionForSection(int section){
    for(int i=0;i<getCount();i++){
      String sortStr=list.get(i).getSortLetters();
      char firstChar=sortStr.toUpperCase().charAt(0);
      if(firstChar==section)
      {
        return i;
      }
    }
    return -1;
  }

然后getView里面判断显示效果,是否显示字母,在哪里显示字母

 @Override
  public View getView(final int i, View view, ViewGroup viewGroup) {
    ViewHolder holder=null;
    final Province province=list.get(i);
    if(view==null)
    {
      holder=new ViewHolder();
      view=LayoutInflater.from(mContext).inflate(R.layout.item,null);
      holder.tvLetter= (TextView) view.findViewById(R.id.catalog);
      holder.tvTitle= (TextView) view.findViewById(R.id.title);
      view.setTag(holder);
    }
    else
    {
      holder= (ViewHolder) view.getTag();
    }
    //根据position获取分类的首字母的char ascii值
    int section=getSectionForPosition(i);

    //如果当前位置等于该分类首字母的Char的位置,则认为是第一次出现
    if(i==getPositionForSection(section))
    {
      holder.tvLetter.setVisibility(View.VISIBLE);
      holder.tvLetter.setText(province.getSortLetters());
    }
    else
    {
      holder.tvLetter.setVisibility(View.GONE);
    }

    holder.tvTitle.setText(this.list.get(i).getProvinceName());

    return view;
  }

3.再贴一个provinceActivity的类

public class ProvinceActivity extends Activity {

  private Context mContext;

  private ListView sortListView;
  private SideBar sideBar;
  private TextView dialog;
  private ProvinceAdapter adapter;

  /**
   * 汉字转换成拼音的类
   */
  private CharacterParser characterParser;
  private List<Province> sourceDateList;
  /**
   * 根据拼音来排列ListView里面的数据类
   */
  private PinyinComparator pinyinComparator;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.a_province);
    mContext=this;

    initView();
  }

  private void initView() {
    //实例化汉字转拼音类
    characterParser=CharacterParser.getInstance();

    pinyinComparator=new PinyinComparator();

    sideBar= (SideBar) findViewById(R.id.sidrbar);
    dialog= (TextView) findViewById(R.id.dialog);
    sideBar.setTextView(dialog);

    //设置右侧触摸监听
    sideBar.setOnTouchingLetterChangedListener(new SideBar.OnTouchingLetterChangedListener() {
      @Override
      public void onTouchingLetterChanged(String s) {
        //该字母首次出现的位置
        int position=adapter.getPositionForSection(s.charAt(0));
        if(position!=-1)
        {
          sortListView.setSelection(position);
        }
      }
    });

    sortListView= (ListView) findViewById(R.id.lv_pro);
    sortListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
      @Override
      public void onItemClick(AdapterView<?> adapterView, View view, int i, long l) {
        Intent intent=new Intent();
        intent.putExtra("provinceId",((Province)adapter.getItem(i)).getId());
        intent.putExtra("provinceName",((Province)adapter.getItem(i)).getProvinceName());
        intent.setClass(mContext,CityActivity.class);
        startActivityForResult(intent,0);
      }
    });

    //获取数据
    volley_get();

  }

  @Override
  protected void onActivityResult(int requestCode, int resultCode, Intent data) {
    if(requestCode==0)
    {
      if(resultCode==1)
      {
        setResult(1,data);
        finish();
      }
    }
    super.onActivityResult(requestCode, resultCode, data);
  }

  /**
   * Volley加载数据
   */
  private void volley_get(){
    RequestQueue mQueue=Volley.newRequestQueue(mContext);
    JsonObjectRequest jsonObjectRequest=new JsonObjectRequest("http://10.0.0.103:8080/StoAppPro/GetProvince",null,new Response.Listener<JSONObject>() {
      @Override
      public void onResponse(JSONObject jsonObject) {
        //Gson解析,直接将jsonObject的data值转换成list
        Gson gson=new Gson();
        Type listType=new TypeToken<List<Province>>(){}.getType();
        try {
          List<Province> list=gson.fromJson(jsonObject.get("data").toString(),listType);
          sourceDateList=filledData(list);

          Log.e("wj", sourceDateList.get(0).getId() + "");

          //根据a-z进行排序源数据
          Collections.sort(sourceDateList,pinyinComparator);

          //初始化适配器
          adapter=new ProvinceAdapter(mContext,sourceDateList);
          //绑定适配器
          sortListView.setAdapter(adapter);
        } catch (JSONException e) {
          e.printStackTrace();
        }
      }
    },new Response.ErrorListener() {
      @Override
      public void onErrorResponse(VolleyError volleyError) {

      }
    });
    mQueue.add(jsonObjectRequest);
  }

  /**
   * 为ListView填充数据
   * @param
   * @return
   */
  private List<Province> filledData(List<Province> list){
    List<Province> mSortList = new ArrayList<Province>();

    for(int i=0; i<list.size(); i++){
      Province province = new Province();
      province.setProvinceName(list.get(i).getProvinceName());
      province.setId(list.get(i).getId());
      //汉字转换成拼音
      String pinyin = characterParser.getSelling(list.get(i).getProvinceName());
      String sortString = pinyin.substring(0, 1).toUpperCase();//获取拼音首字母
      // 正则表达式,判断首字母是否是英文字母
      if(sortString.matches("[A-Z]")){
        province.setSortLetters(sortString.toUpperCase());
      }else{
        province.setSortLetters("#");
      }

      mSortList.add(province);
    }
    return mSortList;

  }
}

ok,粘贴了部分代码,而且很多关键地方我也在代码中加了注释。还是那句话,自己动手实现一把才能在今后用到的时候方便使用。

最后放上源码:Android仿手机通讯录地址选择功能

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • Android手机联系人快速索引(手机通讯录)

    最近需要实现一个手机通讯录的快速索引功能.根据姓名首字母快速索引功能.下面是一个手机联系人快速索引的效果,总体来说代码不算难,拼音转换的地方略有复杂.下面上源码:源码中有注释. 下面是效果图: MainActivity: import java.util.ArrayList; import java.util.Collections; import java.util.List; import android.app.Activity; import android.os.Bundle; imp

  • Android实现通讯录效果——获取手机号码和姓名

    首先给大家展示下运行效果图: 由于通讯录在手机里是以数据库贮存的 所以我们可以通过一个方法 context.getContentResolver().query(Phone.CONTENT_URI, null, null, null, null); 来获得通讯录 ,这个方法返回一个游标的数据类型,通过moveToNext()方法来获取所有的手机号码信息 当然读取手机通讯录需要权限 在adnroidManifest文件中声明即可 由于我也实现了打电话的功能 所以也要声明权限 <uses-permi

  • Android获取手机通讯录、sim卡联系人及调用拨号界面方法

    android获取手机通讯录联系人信息 复制代码 代码如下: private void getPhoneContacts() {        ContentResolver resolver = this.getContentResolver();                // 获取手机联系人       Cursor phoneCursor = resolver.query(Phone.CONTENT_URI,                  new String[] { Phone

  • Android编程操作手机通讯录的方法示例

    本文实例讲述了Android编程操作手机通讯录的方法.分享给大家供大家参考,具体如下: 手机通讯录的操作是经常被用到的,例如添加联系人,删除联系人或者取得联系人信息.类似的操作还有收藏夹的操作,下面就针对通讯录的操作来做个小例子.同样的这次也会使用到内容提供者的知识. 1. 要操作通信录就要得到授权,也就是读或者写通讯录的权力.这里也需要使用Junit <?xml version="1.0" encoding="utf-8"?> <manifest

  • android手机获取gps和基站的经纬度地址实现代码

    复制代码 代码如下: <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="fill_parent" android:layout_height="fill_parent" an

  • Android使用ContentResolver搜索手机通讯录的方法

    本文实例讲述了Android使用ContentResolver搜索手机通讯录的方法.分享给大家供大家参考,具体如下: 在这个程序中使用ContentResolver来访问通讯录里联系人的关键字,并将所有找到的联系人存入CursorAdapter里.输入搜索人员名字a ,会将所有以a开头的名字都显示出来,输入*,则所有通讯录中的人名显示于AutoCompleteView的AdapterView里,若发生了User选择事件后,会将勾选的联系人电话号码显示于TextView里. 此程序中用到了Cont

  • Android跳转到通讯录获取用户名称和手机号码的实现思路

    效果图如下所示: 先给大家说下实现android 跳转到通讯录的实现思路: 1.点击跳转到通讯录界面 2.获取通讯录姓名和手机号码 3.回调显示姓名和手机号码 1首先是跳转到通讯录界面 Uri uri = Uri.parse("content://contacts/people"); Intent intent = new Intent(Intent.ACTION_PICK, uri); startActivityForResult(intent, 0); 通过设置通讯录url跳转,可

  • Android仿手机通讯录地址选择功能

    感觉比较好的一个地址选择设计,而且发现有的App中也用到了.还是先上效果图 思路: 1.效果是仿照网上大神实现的类似通讯录样式做的: 2.右边a-z是自定义的一个bar,设置了点击监听事件,以及对话框弹出 3.关键是adapter,判断了字母显示和隐藏 4.用到汉字转拼音.按首字母排序等工具类 5.3个activity的跳转是用回调来实现,每个activity都实现了回调,这样就有了从区activity直接跳转到首页的效果 6.数据是调用的我本地的接口实现的,如果大家没有数据我可以想办法给你们提

  • Android仿微信通讯录滑动快速定位功能

    先给大家展示下效果图: 实现代码如下: 下面简单说下实现原理. public class IndexBar extends LinearLayout implements View.OnTouchListener { private static final String[] INDEXES = new String[]{"#", "A", "B", "C", "D", "E", &qu

  • Android仿手机QQ图案解锁功能

    本文实例为大家分享了Android仿手机QQ图案解锁的具体代码,供大家参考,具体内容如下 ps:请不要再问我,为什么导入之后会乱码了. 其实,代码基本上都是从原生系统中提取的:LockPatternView.加密工具类,以及解锁逻辑等,我只是稍作修改,大家都知道,原生系统界面比较丑陋,因此,我特意把QQ的apk解压了,从中拿了几张图案解锁的图片,一个简单的例子就这样诞生了. 好了,废话不多说,我们来看看效果(最后两张是最新4.4系统,炫一下,呵呵): 1.最关健的就是那个自定义九宫格View,代

  • Android仿微信通讯录列表侧边栏效果

    先看Android仿微信通讯录列表侧边栏效果图 这是比较常见的效果了吧 列表根据首字符的拼音字母来排序,且可以通过侧边栏的字母索引来进行定位. 实现这样一个效果并不难,只要自定义一个索引View,然后引入一个可以对汉字进行拼音解析的jar包--pinyin4j-2.5.0即可 首先,先来定义侧边栏控件View,只要直接画出来即可. 字母选中项会变为红色,且滑动时背景会变色,此时SideBar并不包含居中的提示文本 public class SideBar extends View { priva

  • Android读取手机通讯录联系人到自己项目

    本文实例为大家分享了Android读取手机通讯录联系人到项目的具体代码,供大家参考,具体内容如下 一.主界面代码如下: <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent" android:orientat

  • Android个人手机通讯录开发详解

    一.Android 个人手机通讯录开发 数据存储:SQLite 数据库 开发工具:Android Studio 二.Phone Module 简介 1. 界面展示 2. 文件结构简单分析 三.个人手机通讯录代码实现 1. 清单文件 (AndroidManifest.xml) <?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.andr

  • 轻松实现Android仿淘宝地区选择功能

    最近用淘宝客户端的时候,编辑地址的时候有个地区选择的功能.看上面的效果觉得挺酷,滚动的时候,是最后一个从下面飞上来挨着前一个.就自己鼓捣一个出来玩玩. 说了效果可能不太直观,下面上两张图看看效果 淘宝地区选择效果 再来一张自己的效果 gif的效果可能不太好,大家自己用Android手机打开淘宝看看 实现分析 展示很简单,ListView就可以了.对于动画效果,只需要在getView的时候获取到要展示的View,通过属性动画修改translationY就ok啦.由于地区选择是一个界面,所以这里还用

  • android仿微信通讯录搜索示例(匹配拼音,字母,索引位置)

    前言: 仿微信通讯录搜索功能,通过汉字或拼音首字母找到匹配的联系人并显示匹配的位置 一:先看效果图 字母索引 搜索匹配 二:功能分析 1:汉字转拼音 通讯录汉字转拼音(首个字符当考虑姓氏多音字), 现在转换拼音常见的有pinyin4j和tinypinyin, pinyin4j的功能强大,包含声调多音字,tinypinyin执行快占用内存少, 如果只是简单匹配通讯录,建议使用tinypinyin,用法也很简单这里不详细介绍 拼音类 public class CNPinyin <T extends

  • Android仿微信通讯录打造带悬停头部的分组列表(上)

    一 概述 本文是Android导航分组列表系列上,因时间和篇幅原因分上下,最终上下合璧,完整版效果如下: 上部残卷效果如下:两个ItemDecoration,一个实现悬停头部分组列表功能,一个实现分割线(官方demo) 网上关于实现带悬停分组头部的列表的方法有很多,像我看过有主席的自定义ExpandListView实现的,也看过有人用一个额外的父布局里面套 RecyclerView/ListView+一个头部View(位置固定在父布局上方)实现的. 对于以上解决方案,有以下几点个人觉得不好的地方

随机推荐