Android手机联系人带字母索引的快速查找_Android

喜欢另辟蹊径的我,在这里废话不多说了,直接上代码和图片了。
效果图如下:

第一步:MainActivity的代码如下:

package net.loonggg.test; 

import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.TreeSet; 

import android.os.Bundle;
import android.app.Activity;
import android.graphics.Color;
import android.view.MotionEvent;
import android.view.View;
import android.view.View.OnTouchListener;
import android.view.Window;
import android.widget.LinearLayout;
import android.widget.ListView;
import android.widget.TextView;
import android.widget.LinearLayout.LayoutParams; 

public class MainActivity extends Activity {
  private HashMap<String, Integer> selector;// 存放含有索引字母的位置
  private LinearLayout layoutIndex;
  private ListView listView;
  private TextView tv_show;
  private ListViewAdapter adapter;
  private String[] indexStr = { "#", "A", "B", "C", "D", "E", "F", "G", "H",
      "I", "J", "K", "L", "M", "N", "O", "P", "Q", "R", "S", "T", "U",
      "V", "W", "X", "Y", "Z" };
  private List<Person> persons = null;
  private List<Person> newPersons = new ArrayList<Person>();
  private int height;// 字体高度
  private boolean flag = false; 

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    // 去标题栏
    requestWindowFeature(Window.FEATURE_NO_TITLE);
    setContentView(R.layout.activity_main);
    layoutIndex = (LinearLayout) this.findViewById(R.id.layout);
    layoutIndex.setBackgroundColor(Color.parseColor("#00ffffff"));
    listView = (ListView) findViewById(R.id.listView);
    tv_show = (TextView) findViewById(R.id.tv);
    tv_show.setVisibility(View.GONE);
    setData();
    String[] allNames = sortIndex(persons);
    sortList(allNames); 

    selector = new HashMap<String, Integer>();
    for (int j = 0; j < indexStr.length; j++) {// 循环字母表,找出newPersons中对应字母的位置
      for (int i = 0; i < newPersons.size(); i++) {
        if (newPersons.get(i).getName().equals(indexStr[j])) {
          selector.put(indexStr[j], i);
        }
      } 

    }
    adapter = new ListViewAdapter(this, newPersons);
    listView.setAdapter(adapter);
  } 

  /**
   * 重新排序获得一个新的List集合
   *
   * @param allNames
   */
  private void sortList(String[] allNames) {
    for (int i = 0; i < allNames.length; i++) {
      if (allNames[i].length() != 1) {
        for (int j = 0; j < persons.size(); j++) {
          if (allNames[i].equals(persons.get(j).getPinYinName())) {
            Person p = new Person(persons.get(j).getName(), persons
                .get(j).getPinYinName());
            newPersons.add(p);
          }
        }
      } else {
        newPersons.add(new Person(allNames[i]));
      }
    }
  } 

  @Override
  public void onWindowFocusChanged(boolean hasFocus) {
    // 在oncreate里面执行下面的代码没反应,因为oncreate里面得到的getHeight=0
    if (!flag) {// 这里为什么要设置个flag进行标记,我这里不先告诉你们,请读者研究,因为这对你们以后的开发有好处
      height = layoutIndex.getMeasuredHeight() / indexStr.length;
      getIndexView();
      flag = true;
    }
  } 

  /**
   * 获取排序后的新数据
   *
   * @param persons
   * @return
   */
  public String[] sortIndex(List<Person> persons) {
    TreeSet<String> set = new TreeSet<String>();
    // 获取初始化数据源中的首字母,添加到set中
    for (Person person : persons) {
      set.add(StringHelper.getPinYinHeadChar(person.getName()).substring(
          0, 1));
    }
    // 新数组的长度为原数据加上set的大小
    String[] names = new String[persons.size() + set.size()];
    int i = 0;
    for (String string : set) {
      names[i] = string;
      i++;
    }
    String[] pinYinNames = new String[persons.size()];
    for (int j = 0; j < persons.size(); j++) {
      persons.get(j).setPinYinName(
          StringHelper
              .getPingYin(persons.get(j).getName().toString()));
      pinYinNames[j] = StringHelper.getPingYin(persons.get(j).getName()
          .toString());
    }
    // 将原数据拷贝到新数据中
    System.arraycopy(pinYinNames, 0, names, set.size(), pinYinNames.length);
    // 自动按照首字母排序
    Arrays.sort(names, String.CASE_INSENSITIVE_ORDER);
    return names;
  } 

  /**
   * 绘制索引列表
   */
  public void getIndexView() {
    LinearLayout.LayoutParams params = new LayoutParams(
        LayoutParams.WRAP_CONTENT, height);
    for (int i = 0; i < indexStr.length; i++) {
      final TextView tv = new TextView(this);
      tv.setLayoutParams(params);
      tv.setText(indexStr[i]);
      tv.setPadding(10, 0, 10, 0);
      layoutIndex.addView(tv);
      layoutIndex.setOnTouchListener(new OnTouchListener() { 

        @Override
        public boolean onTouch(View v, MotionEvent event) 

        {
          float y = event.getY();
          int index = (int) (y / height);
          if (index > -1 && index < indexStr.length) {// 防止越界
            String key = indexStr[index];
            if (selector.containsKey(key)) {
              int pos = selector.get(key);
              if (listView.getHeaderViewsCount() > 0) {// 防止ListView有标题栏,本例中没有。
                listView.setSelectionFromTop(
                    pos + listView.getHeaderViewsCount(), 0);
              } else {
                listView.setSelectionFromTop(pos, 0);// 滑动到第一项
              }
              tv_show.setVisibility(View.VISIBLE);
              tv_show.setText(indexStr[index]);
            }
          }
          switch (event.getAction()) {
          case MotionEvent.ACTION_DOWN:
            layoutIndex.setBackgroundColor(Color
                .parseColor("#606060"));
            break; 

          case MotionEvent.ACTION_MOVE: 

            break;
          case MotionEvent.ACTION_UP:
            layoutIndex.setBackgroundColor(Color
                .parseColor("#00ffffff"));
            tv_show.setVisibility(View.GONE);
            break;
          }
          return true;
        }
      });
    }
  } 

  /**
   * 设置模拟数据
   */
  private void setData() {
    persons = new ArrayList<Person>();
    Person p1 = new Person("耿琦");
    persons.add(p1);
    Person p2 = new Person("王宝强");
    persons.add(p2);
    Person p3 = new Person("柳岩");
    persons.add(p3);
    Person p4 = new Person("文章");
    persons.add(p4);
    Person p5 = new Person("马伊琍");
    persons.add(p5);
    Person p6 = new Person("李晨");
    persons.add(p6);
    Person p7 = new Person("张馨予");
    persons.add(p7);
    Person p8 = new Person("韩红");
    persons.add(p8);
    Person p9 = new Person("韩寒");
    persons.add(p9);
    Person p10 = new Person("丹丹");
    persons.add(p10);
    Person p11 = new Person("丹凤眼");
    persons.add(p11);
    Person p12 = new Person("哈哈");
    persons.add(p12);
    Person p13 = new Person("萌萌");
    persons.add(p13);
    Person p14 = new Person("蒙混");
    persons.add(p14);
    Person p15 = new Person("烟花");
    persons.add(p15);
    Person p16 = new Person("眼黑");
    persons.add(p16);
    Person p17 = new Person("许三多");
    persons.add(p17);
    Person p18 = new Person("程咬金");
    persons.add(p18);
    Person p19 = new Person("程哈哈");
    persons.add(p19);
    Person p20 = new Person("爱死你");
    persons.add(p20);
    Person p21 = new Person("阿莱");
    persons.add(p21); 

  } 

}

 此Activity的布局文件如下:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
  android:layout_width="fill_parent"
  android:layout_height="match_parent"
  android:background="#ffffff"
  android:orientation="vertical" > 

  <TextView
    android:layout_width="fill_parent"
    android:layout_height="wrap_content"
    android:padding="10dp"
    android:text="列表显示"
    android:textColor="#000000"
    android:textSize="16sp" /> 

  <FrameLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#ffffff" > 

    <ListView
      android:id="@+id/listView"
      android:layout_width="match_parent"
      android:layout_height="wrap_content"
      android:cacheColorHint="#00000000"
      android:fadingEdge="none"
      android:scrollbars="none" >
    </ListView> 

    <TextView
      android:id="@+id/tv"
      android:layout_width="60dp"
      android:layout_height="60dp"
      android:layout_gravity="center"
      android:background="#f0606060"
      android:gravity="center"
      android:text="A"
      android:textColor="#ffffff"
      android:textSize="30sp" /> 

    <LinearLayout
      android:id="@+id/layout"
      android:layout_width="wrap_content"
      android:layout_height="fill_parent"
      android:layout_gravity="right"
      android:background="#d7d7d7"
      android:gravity="center"
      android:orientation="vertical" >
    </LinearLayout>
  </FrameLayout> 

</LinearLayout> 

第二步:自定义了一个Adapter,代码如下:

package net.loonggg.test; 

import java.util.List; 

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

public class ListViewAdapter extends BaseAdapter {
  private Context context;
  private List<Person> list;
  private ViewHolder viewHolder; 

  public ListViewAdapter(Context context, List<Person> list) {
    this.context = context;
    this.list = list;
  } 

  @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 boolean isEnabled(int position) {
    // TODO Auto-generated method stub
    if (list.get(position).getName().length() == 1)// 如果是字母索引
      return false;// 表示不能点击
    return super.isEnabled(position);
  } 

  @Override
  public View getView(int position, View convertView, ViewGroup parent) {
    String item = list.get(position).getName();
    viewHolder = new ViewHolder();
    if (item.length() == 1) {
      convertView = LayoutInflater.from(context).inflate(R.layout.index,
          null);
      viewHolder.indexTv = (TextView) convertView
          .findViewById(R.id.indexTv);
    } else {
      convertView = LayoutInflater.from(context).inflate(R.layout.item,
          null);
      viewHolder.itemTv = (TextView) convertView
          .findViewById(R.id.itemTv);
    }
    if (item.length() == 1) {
      viewHolder.indexTv.setText(list.get(position).getName());
    } else {
      viewHolder.itemTv.setText(list.get(position).getName());
    }
    return convertView;
  } 

  private class ViewHolder {
    private TextView indexTv;
    private TextView itemTv;
  } 

} 

第三步:用到的ListView中的子布局文件如下:
1、index.xml布局文件代码如下:

<?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="wrap_content"
  android:background="#9c9c9c"
  android:orientation="vertical"
  android:paddingBottom="5dp"
  android:paddingLeft="10dp"
  android:paddingTop="5dp" > 

  <TextView
    android:id="@+id/indexTv"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textColor="#0f0f0f" /> 

</LinearLayout>

2、item.xml布局文件代码如下:

<?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="wrap_content"
  android:background="#ffffff"
  android:orientation="vertical"
  android:paddingBottom="5dp"
  android:paddingLeft="20dp"
  android:paddingTop="5dp" > 

  <TextView
    android:id="@+id/itemTv"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:textColor="#000000"
    android:textSize="20sp" /> 

</LinearLayout> 

第四步:用到的实体类及工具类如下:
1、Person.java代码如下:

package net.loonggg.test; 

public class Person {
  private String name;
  private String pinYinName; 

  public Person(String name) {
    super();
    this.name = name;
  } 

  public Person(String name, String pinYinName) {
    super();
    this.name = name;
    this.pinYinName = pinYinName;
  } 

  public String getName() {
    return name;
  } 

  public void setName(String name) {
    this.name = name;
  } 

  public String getPinYinName() {
    return pinYinName;
  } 

  public void setPinYinName(String pinYinName) {
    this.pinYinName = pinYinName;
  } 

} 

2、工具类代码如下:

package net.loonggg.test; 

import net.sourceforge.pinyin4j.PinyinHelper;
import net.sourceforge.pinyin4j.format.HanyuPinyinCaseType;
import net.sourceforge.pinyin4j.format.HanyuPinyinOutputFormat;
import net.sourceforge.pinyin4j.format.HanyuPinyinToneType;
import net.sourceforge.pinyin4j.format.HanyuPinyinVCharType;
import net.sourceforge.pinyin4j.format.exception.BadHanyuPinyinOutputFormatCombination; 

public class StringHelper {
  /**
   * 得到 全拼
   *
   * @param src
   * @return
   */
  public static String getPingYin(String src) {
    char[] t1 = null;
    t1 = src.toCharArray();
    String[] t2 = new String[t1.length];
    HanyuPinyinOutputFormat t3 = new HanyuPinyinOutputFormat();
    t3.setCaseType(HanyuPinyinCaseType.LOWERCASE);
    t3.setToneType(HanyuPinyinToneType.WITHOUT_TONE);
    t3.setVCharType(HanyuPinyinVCharType.WITH_V);
    String t4 = "";
    int t0 = t1.length;
    try {
      for (int i = 0; i < t0; i++) {
        // 判断是否为汉字字符
        if (java.lang.Character.toString(t1[i]).matches(
            "[\\u4E00-\\u9FA5]+")) {
          t2 = PinyinHelper.toHanyuPinyinStringArray(t1[i], t3);
          t4 += t2[0];
        } else {
          t4 += java.lang.Character.toString(t1[i]);
        }
      }
      return t4;
    } catch (BadHanyuPinyinOutputFormatCombination e1) {
      e1.printStackTrace();
    }
    return t4;
  } 

  /**
   * 得到首字母
   *
   * @param str
   * @return
   */
  public static String getHeadChar(String str) { 

    String convert = "";
    char word = str.charAt(0);
    String[] pinyinArray = PinyinHelper.toHanyuPinyinStringArray(word);
    if (pinyinArray != null) {
      convert += pinyinArray[0].charAt(0);
    } else {
      convert += word;
    }
    return convert.toUpperCase();
  } 

  /**
   * 得到中文首字母缩写
   *
   * @param str
   * @return
   */
  public static String getPinYinHeadChar(String str) { 

    String convert = "";
    for (int j = 0; j < str.length(); j++) {
      char word = str.charAt(j);
      String[] pinyinArray = PinyinHelper.toHanyuPinyinStringArray(word);
      if (pinyinArray != null) {
        convert += pinyinArray[0].charAt(0);
      } else {
        convert += word;
      }
    }
    return convert.toUpperCase();
  }
}

到这里就完事,非常简单吧!

以上是小编为您精心准备的的内容,在的博客、问答、公众号、人物、课程等栏目也有的相关内容,欢迎继续使用右上角搜索按钮进行搜索Android联系人索引
联系人 右侧字母索引、iphone快速查找联系人、快速查找联系人、苹果7快速查找联系人、ios 快速查找联系人,以便于您获取更多的相关知识。

时间: 2024-08-03 05:33:02

Android手机联系人带字母索引的快速查找_Android的相关文章

Android手机联系人带字母索引的快速查找

喜欢另辟蹊径的我,在这里废话不多说了,直接上代码和图片了. 效果图如下: 第一步:MainActivity的代码如下: package net.loonggg.test; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.TreeSet; import android.os.Bundle; import and

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

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

bug-求助: Android 中sectionIndexer 实现字母索引,在滑动的时候字母的宽高不一致?

问题描述 求助: Android 中sectionIndexer 实现字母索引,在滑动的时候字母的宽高不一致? framework层刚入门,在改这个bug,横屏下fastscroll滚动式的字母索引,宽高不一致,比如W,M这种字母比HLI这类宽下,所以在同一个padding值下就有一个宽高不一,现在就是要求必须一样.我在系统的fastScroll的style中发现,定义的padding是16dp.我将其改成141210等等,就ok了,没有这个问题,我就是想知道,为啥padding都是一样的情况下

android手机自带的短信软件会拦截掉短信广播的吗?

问题描述 android手机自带的短信软件会拦截掉短信广播的吗? 在androidAVD上测试时可以正常收到短信广播的,但是在真机测试过程中受到短信却没反应,所以想询问一下android手机自带的短信接收到广播之后是不是就会拦截掉,不让其他的软件接收. 解决方案 你的广播频率如果人家不知道怎么拦截,是不是你在真机上测试时没有开通权限呀 解决方案二: 有权限的啊,装的时候太特别注意了这个,安装的时候有看到说接受短信的权限.我广播优先级没设定.

PC版与Android手机版带断点续传的多线程下载_Android

一.多线程下载         多线程下载就是抢占服务器资源         原理:服务器CPU 分配给每条线程的时间片相同,服务器带宽平均分配给每条线程,所以客户端开启的线程越多,就能抢占到更多的服务器资源.       1.设置开启线程数,发送http请求到下载地址,获取下载文件的总长度           然后创建一个长度一致的临时文件,避免下载到一半存储空间不够了,并计算每个线程下载多少数据              2.计算每个线程下载数据的开始和结束位置           再次发送

源码-现在Android手机都带有云服务,是怎么实现的

问题描述 现在Android手机都带有云服务,是怎么实现的 现在Android手机都带有云服务,是怎么实现的,,是对Android底层源码的改造吗 解决方案 http://www.imooc.com/learn/254 解决方案二: 不需要,只要对某些上层api进行改造,加上云端存储就可以了. 解决方案三: 题主,你可以百度一下Bmob或者七牛

PC版与Android手机版带断点续传的多线程下载

一.多线程下载 多线程下载就是抢占服务器资源 原理:服务器CPU 分配给每条线程的时间片相同,服务器带宽平均分配给每条线程,所以客户端开启的线程越多,就能抢占到更多的服务器资源. 1.设置开启线程数,发送http请求到下载地址,获取下载文件的总长度           然后创建一个长度一致的临时文件,避免下载到一半存储空间不够了,并计算每个线程下载多少数据              2.计算每个线程下载数据的开始和结束位置           再次发送请求,用 Range 头请求开始位置和结束位

浅谈Android手机联系人开发之增删查改功能

最近在做手机联系人的功能模块的时候,遇到了很多的坑,在网上搜索的有一些所谓的最全的手机联系人开发的介绍还存在一些bug,所以我把我最近的项目心得和方法写下来,既能帮助大家减少了解android开发手机联系人的门槛,好,废话少说,接下来直奔主题. 一.深入浅出手机联系人的前奏(小米手机的data表跟模拟器的data表不一样) 1.手机联系人主要是对contacts2.db数据库表的操纵,这个数据库中有三个表是比较重要的,分别是data,raw_contacts,mimetyps这三个表.在下面的增

Android自定义View实现字母导航栏的代码_Android

思路分析: 1.自定义View实现字母导航栏 2.ListView实现联系人列表 3.字母导航栏滑动事件处理 4.字母导航栏与中间字母的联动 5.字母导航栏与ListView的联动 效果图: 首先,我们先甩出主布局文件,方便后面代码的说明 <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:android="http://schemas.android.com/apk/re