国庆回了趟家,昨天真不想走,离家近的感觉太好。唉,不扯这些,说说今天的正事吧。
上篇博客中介绍了自定义AutoCompleteTextView ,但是用到了一个很蹩脚的技术,就是我们事先把每个汉字的拼音当作一个字段输入进去了,在实际的开发中肯定不会这样做,我们要通过代码自动生成汉字的拼音,就像我们的手机通讯录,比如我们要查找“张三”这个人,我们只需要输入“zs”、“cs”或者“zhangsan”、“changsan”就能搜索到该人,那么我们该怎么来实现这样的功能呢?
本文所述案例是在上篇博客的基础上实现的,如果还没阅读上篇博客,请看android开发之自定义AutoCompleteTextView。
本文要实现的整体效果如下图所示:
在上篇博客中我们自定义了AutoCompleteTextView的Adapter,本文中,我们继续对这个Adapter进行深化改造。
主要改造两个地方,第一个地方是在构造方法中初始化拼音集合:
改造后的构造方法:
- public MyActAdapter(Context context, List<Book> books, int maxMatch) {
- this.books = books;
- this.context = context;
- this.maxMatch = maxMatch;
- initPinYinList();
- }
- 1
- 2
- 3
- 4
- 5
- 6
这个方法主要是初始化两个List集合,一个是pinYinList 另一个是pinYinAllList ,前者是所有书的作者姓名拼音的首字母集合,后者是所有书的作者姓名拼音全拼集合。
- private void initPinYinList() {
- pinYinList = new ArrayList<Set<String>>();
- pinYinAllList = new ArrayList<Set<String>>();
- PinYin4j pinyin = new PinYin4j();
- for (int i = 0; i < books.size(); i++) {
- pinYinList.add(pinyin.getPinyin(books.get(i).getAuthor().toString()));
- pinYinAllList.add(pinyin.getAllPinyin(books.get(i).getAuthor().toString()));
- }
- }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
这里还涉及到两个类,如下:
PinYin4j.java
- package com.example.myact;
-
- import java.util.Arrays;
- import java.util.HashSet;
- import java.util.Set;
-
- public class PinYin4j {
-
-
- public PinYin4j(){
- }
- /**
- * 字符串集合转换字符串(逗号分隔)
- *
- * @author wangsong
- * @param stringSet
- * @return
- */
- public String makeStringByStringSet(Set<String> stringSet) {
- StringBuilder str = new StringBuilder();
- int i = 0;
- for (String s : stringSet) {
- if (i == stringSet.size() - 1) {
- str.append(s);
- } else {
- str.append(s + ",");
- }
- i++;
- }
- return str.toString().toLowerCase();
- }
-
-
- /**
- * 获取汉字拼音全拼
- *
- * @author wangsong
- * @param src
- * @return Set<String>
- */
- public Set<String> getAllPinyin(String src) {
- char[] srcChar;
- srcChar = src.toCharArray();
- String[][] temp = new String[src.length()][];
- for (int i = 0; i < srcChar.length; i++) {
- char c = srcChar[i];
- if (String.valueOf(c).matches("[\\u4E00-\\u9FA5]+")) {
- String[] t = PinyinHelper.getUnformattedHanyuPinyinStringArray(c);
- temp[i] = new String[t.length];
- for(int j=0;j<t.length;j++){
- temp[i][j]=t[j].replaceAll("\\d", "");//获取全拼
- }
- } else if (((int) c >= 65 && (int) c <= 90)
- || ((int) c >= 97 && (int) c <= 122)||c>=48&&c<=57||c==42) {
- temp[i] = new String[] { String.valueOf(srcChar[i]) };
- } else {
- temp[i] = new String[] {"null!"};
- }
-
- }
- String[] pingyinArray = paiLie(temp);
- return array2Set(pingyinArray);
- }
- /**
- * 获取汉字拼音首字母集合
- *
- * @author wangsong
- * @param src
- * @return Set<String>
- */
- public Set<String> getPinyin(String src) {
- char[] srcChar;
- srcChar = src.toCharArray();
- String[][] temp = new String[src.length()][];
- for (int i = 0; i < srcChar.length; i++) {
- char c = srcChar[i];
- if (String.valueOf(c).matches("[\\u4E00-\\u9FA5]+")) {
- String[] t = PinyinHelper.getUnformattedHanyuPinyinStringArray(c);
- temp[i] = new String[t.length];
- for(int j=0;j<t.length;j++){
- temp[i][j]=t[j].substring(0,1);
- }
- } else if (((int) c >= 65 && (int) c <= 90)
- || ((int) c >= 97 && (int) c <= 122)||c>=48&&c<=57||c==42) {
- temp[i] = new String[] { String.valueOf(srcChar[i]) };
- } else {
- temp[i] = new String[] {"null!"};
- }
-
- }
- String[] pingyinArray = paiLie(temp);
- return array2Set(pingyinArray);
- }
-
- /*
- * 求2维数组所有排列组合情况
- * 比如:{{1,2},{3},{4},{5,6}}共有2中排列,为:1345,1346,2345,2346
- */
- private String[] paiLie(String[][] str){
- int max=1;
- for(int i=0;i<str.length;i++){
- max*=str[i].length;
- }
- String[] result=new String[max];
- for(int i = 0; i < max; i++){
- String s = "";
- int temp = 1; //注意这个temp的用法。
- for(int j = 0; j < str.length; j++){
- temp *= str[j].length;
- s += str[j][i / (max / temp) % str[j].length];
- }
- result[i]=s;
- }
-
- return result;
- }
- /**
- * 去掉重复项
- * @param tArray
- * @return
- */
- public static <T extends Object> Set<T> array2Set(T[] tArray) {
- Set<T> tSet = new HashSet<T>(Arrays.asList(tArray));
- // TODO 没有一步到位的方法,根据具体的作用,选择合适的Set的子类来转换。
- return tSet;
- }
- }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
- 80
- 81
- 82
- 83
- 84
- 85
- 86
- 87
- 88
- 89
- 90
- 91
- 92
- 93
- 94
- 95
- 96
- 97
- 98
- 99
- 100
- 101
- 102
- 103
- 104
- 105
- 106
- 107
- 108
- 109
- 110
- 111
- 112
- 113
- 114
- 115
- 116
- 117
- 118
- 119
- 120
- 121
- 122
- 123
- 124
- 125
- 126
- 127
PinyinHelper.java
- package com.example.myact;
-
- import java.io.BufferedInputStream;
- import java.io.FileNotFoundException;
- import java.io.IOException;
- import java.util.Properties;
-
- public class PinyinHelper {
- private static PinyinHelper instance;
- private Properties properties = null;
-
- public static String[] getUnformattedHanyuPinyinStringArray(char ch) {
- return getInstance().getHanyuPinyinStringArray(ch);
- }
-
- private PinyinHelper() {
- initResource();
- }
-
- public static PinyinHelper getInstance() {
- if (instance == null) {
- instance = new PinyinHelper();
- }
- return instance;
- }
-
- private void initResource() {
- try {
- final String resourceName = "/assets/unicode_to_hanyu_pinyin.txt";
- // final String resourceName = "/assets/unicode_py.ini";
-
- properties = new Properties();
- properties.load(getResourceInputStream(resourceName));
-
- } catch (FileNotFoundException ex) {
- ex.printStackTrace();
- } catch (IOException ex) {
- ex.printStackTrace();
- }
- }
-
- private BufferedInputStream getResourceInputStream(String resourceName) {
- return new BufferedInputStream(
- PinyinHelper.class.getResourceAsStream(resourceName));
- }
-
- private String[] getHanyuPinyinStringArray(char ch) {
- String pinyinRecord = getHanyuPinyinRecordFromChar(ch);
-
- if (null != pinyinRecord) {
- int indexOfLeftBracket = pinyinRecord.indexOf(Field.LEFT_BRACKET);
- int indexOfRightBracket = pinyinRecord
- .lastIndexOf(Field.RIGHT_BRACKET);
-
- String stripedString = pinyinRecord.substring(indexOfLeftBracket
- + Field.LEFT_BRACKET.length(), indexOfRightBracket);
-
- return stripedString.split(Field.COMMA);
-
- } else
- return null;
-
- }
-
- private String getHanyuPinyinRecordFromChar(char ch) {
- int codePointOfChar = ch;
- String codepointHexStr = Integer.toHexString(codePointOfChar)
- .toUpperCase();
- String foundRecord = properties.getProperty(codepointHexStr);
- return foundRecord;
- }
-
- class Field {
- static final String LEFT_BRACKET = "(";
- static final String RIGHT_BRACKET = ")";
- static final String COMMA = ",";
- }
-
- public static String[] toHanyuPinyinStringArray(char ch) {
- return getUnformattedHanyuPinyinStringArray(ch);
- }
- }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
- 74
- 75
- 76
- 77
- 78
- 79
- 80
- 81
- 82
这里是初始化拼音集合。
第二个改造的地方就是在过滤器中增加过滤的条件。
这是最新的过滤器,和上文相比,这里只是增加了一个else分支,在else分支中判断搜索条件是否符合要求。
- private class ArrayFilter extends Filter {
-
- @Override
- protected FilterResults performFiltering(CharSequence constraint) {
- FilterResults results = new FilterResults();
- if (mFilterBooks == null) {
- mFilterBooks = new ArrayList<Book>(books);
- }
- // 如果没有过滤条件则不过滤
- if (constraint == null || constraint.length() == 0) {
- results.values = mFilterBooks;
- results.count = mFilterBooks.size();
- } else {
- List<Book> retList = new ArrayList<Book>();
- // 过滤条件
- String str = constraint.toString().toLowerCase();
- Book book;
- // 循环变量数据源,如果有属性满足过滤条件,则添加到result中
- for (int i = 0; i < mFilterBooks.size(); i++) {
- book = mFilterBooks.get(i);
- if (book.getAuthor().contains(str)
- || book.getName().contains(str)
- || (book.getId() + "").contains(str)
- || (book.getPrice() + "").contains(str)
- || book.getPinyin().contains(str)) {
- retList.add(book);
- } else {
- //查看作者姓名拼音首字母是否符合过滤条件
- Set<String> pinyinSet = pinYinList.get(i);
- Iterator<String> pinyin = pinyinSet.iterator();
- while (pinyin.hasNext()) {
- if (pinyin.next().toString().contains(str)) {
- retList.add(book);
- break;
- }
- }
- //查看作者姓名拼音全拼是否符合过滤条件
- Set<String> pinyinAllSet = pinYinAllList.get(i);
- Iterator<String> pinyinAll = pinyinAllSet.iterator();
- while (pinyinAll.hasNext()) {
- if (pinyinAll.next().toString().contains(str)) {
- retList.add(book);
- break;
- }
- }
- }
- // if (maxMatch > 0) {
- // if (retList.size() > maxMatch - 1) {
- // break;
- // }
- // }
- }
- results.values = retList;
- results.count = retList.size();
- }
- return results;
- }
-
- // 在这里返回过滤结果
- @Override
- protected void publishResults(CharSequence constraint,
- FilterResults results) {
- // notifyDataSetInvalidated(),会重绘控件(还原到初始状态)
- // notifyDataSetChanged(),重绘当前可见区域
- books = (List<Book>) results.values;
- if (results.count > 0) {
- notifyDataSetChanged();
- } else {
- notifyDataSetInvalidated();
- }
- }
-
- }
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
- 72
- 73
其实还是很简单的,有问题欢迎留言讨论。
版权声明:本文为博主原创文章,未经博主允许不得转载。若有错误地方,还望批评指正,不胜感激。