diff options
author | cretin45 <cretin45@gmail.com> | 2014-07-23 16:32:45 -0700 |
---|---|---|
committer | Ed Mancebo <emancebo@cyngn.com> | 2015-01-06 21:40:35 +0000 |
commit | 7d23a38654b9c0a2abcfadf02bd0cb2ee3d95aec (patch) | |
tree | dc373ea5fcdd2c6dd4ed0acddfed02cb76efafe6 | |
parent | b559010d742a7d6961e5f07ef80345e972f85442 (diff) | |
download | android_packages_apps_Dialer-7d23a38654b9c0a2abcfadf02bd0cb2ee3d95aec.tar.gz android_packages_apps_Dialer-7d23a38654b9c0a2abcfadf02bd0cb2ee3d95aec.tar.bz2 android_packages_apps_Dialer-7d23a38654b9c0a2abcfadf02bd0cb2ee3d95aec.zip |
Dialer: Refactor SmartDial for additional languages
Adds Chinese support and improved Korean support.
p4: Refactor for transliterating the displayName.
Change-Id: Id3175c6f4eda4962be9d25bd43c24a834c3c5ff1
-rw-r--r-- | Android.mk | 3 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/ChineseSmartDialMap.java | 468 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/GreekSmartDialMap.java | 12 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/HebrewSmartDialMap.java | 12 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/KoreanSmartDialMap.java | 96 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/LatinSmartDialMap.java | 12 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/RussianSmartDialMap.java | 12 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/SmartDialMap.java | 13 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/SmartDialNameMatcher.java | 2 | ||||
-rw-r--r-- | src/com/android/dialer/dialpad/SmartDialPrefix.java | 4 |
10 files changed, 632 insertions, 2 deletions
diff --git a/Android.mk b/Android.mk index c7323411f..70102cf9f 100644 --- a/Android.mk +++ b/Android.mk @@ -18,6 +18,9 @@ res_dirs := res \ $(phone_common_dir)/res LOCAL_SRC_FILES := $(call all-java-files-under, $(src_dirs)) +LOCAL_SRC_FILES += ../../providers/ContactsProvider/src/com/android/providers/contacts/NameSplitter.java \ + ../../providers/ContactsProvider/src/com/android/providers/contacts/HanziToPinyin.java \ + ../../providers/ContactsProvider/src/com/android/providers/contacts/util/NeededForTesting.java LOCAL_RESOURCE_DIR := $(addprefix $(LOCAL_PATH)/, $(res_dirs)) LOCAL_ASSET_DIR := $(LOCAL_PATH)/assets diff --git a/src/com/android/dialer/dialpad/ChineseSmartDialMap.java b/src/com/android/dialer/dialpad/ChineseSmartDialMap.java new file mode 100644 index 000000000..3e2c13cca --- /dev/null +++ b/src/com/android/dialer/dialpad/ChineseSmartDialMap.java @@ -0,0 +1,468 @@ +package com.android.dialer.dialpad; + +import com.android.providers.contacts.HanziToPinyin; + +import android.text.TextUtils; + +import java.util.ArrayList; + +public class ChineseSmartDialMap implements SmartDialMap { + + private static final char[] LATIN_LETTERS_TO_DIGITS = { + '2', '2', '2', // A,B,C -> 2 + '3', '3', '3', // D,E,F -> 3 + '4', '4', '4', // G,H,I -> 4 + '5', '5', '5', // J,K,L -> 5 + '6', '6', '6', // M,N,O -> 6 + '7', '7', '7', '7', // P,Q,R,S -> 7 + '8', '8', '8', // T,U,V -> 8 + '9', '9', '9', '9' // W,X,Y,Z -> 9 + }; + + @Override + public boolean isValidDialpadAlphabeticChar(char ch) { + return (ch >= 'a' && ch <= 'z'); + } + + @Override + public boolean isValidDialpadNumericChar(char ch) { + return (ch >= '0' && ch <= '9'); + } + + @Override + public boolean isValidDialpadCharacter(char ch) { + return (isValidDialpadAlphabeticChar(ch) || isValidDialpadNumericChar(ch)); + } + + /* + * The switch statement in this function was generated using the python code: + * from unidecode import unidecode + * for i in range(192, 564): + * char = unichr(i) + * decoded = unidecode(char) + * # Unicode characters that decompose into multiple characters i.e. + * # into ss are not supported for now + * if (len(decoded) == 1 and decoded.isalpha()): + * print "case '" + char + "': return '" + unidecode(char) + "';" + * + * This gives us a way to map characters containing accents/diacritics to their + * alphabetic equivalents. The unidecode library can be found at: + * http://pypi.python.org/pypi/Unidecode/0.04.1 + * + * Also remaps all upper case latin characters to their lower case equivalents. + */ + @Override + public char normalizeCharacter(char ch) { + switch (ch) { + case 'À': return 'a'; + case 'Á': return 'a'; + case 'Â': return 'a'; + case 'Ã': return 'a'; + case 'Ä': return 'a'; + case 'Å': return 'a'; + case 'Ç': return 'c'; + case 'È': return 'e'; + case 'É': return 'e'; + case 'Ê': return 'e'; + case 'Ë': return 'e'; + case 'Ì': return 'i'; + case 'Í': return 'i'; + case 'Î': return 'i'; + case 'Ï': return 'i'; + case 'Ð': return 'd'; + case 'Ñ': return 'n'; + case 'Ò': return 'o'; + case 'Ó': return 'o'; + case 'Ô': return 'o'; + case 'Õ': return 'o'; + case 'Ö': return 'o'; + case '×': return 'x'; + case 'Ø': return 'o'; + case 'Ù': return 'u'; + case 'Ú': return 'u'; + case 'Û': return 'u'; + case 'Ü': return 'u'; + case 'Ý': return 'u'; + case 'à': return 'a'; + case 'á': return 'a'; + case 'â': return 'a'; + case 'ã': return 'a'; + case 'ä': return 'a'; + case 'å': return 'a'; + case 'ç': return 'c'; + case 'è': return 'e'; + case 'é': return 'e'; + case 'ê': return 'e'; + case 'ë': return 'e'; + case 'ì': return 'i'; + case 'í': return 'i'; + case 'î': return 'i'; + case 'ï': return 'i'; + case 'ð': return 'd'; + case 'ñ': return 'n'; + case 'ò': return 'o'; + case 'ó': return 'o'; + case 'ô': return 'o'; + case 'õ': return 'o'; + case 'ö': return 'o'; + case 'ø': return 'o'; + case 'ù': return 'u'; + case 'ú': return 'u'; + case 'û': return 'u'; + case 'ü': return 'u'; + case 'ý': return 'y'; + case 'ÿ': return 'y'; + case 'Ā': return 'a'; + case 'ā': return 'a'; + case 'Ă': return 'a'; + case 'ă': return 'a'; + case 'Ą': return 'a'; + case 'ą': return 'a'; + case 'Ć': return 'c'; + case 'ć': return 'c'; + case 'Ĉ': return 'c'; + case 'ĉ': return 'c'; + case 'Ċ': return 'c'; + case 'ċ': return 'c'; + case 'Č': return 'c'; + case 'č': return 'c'; + case 'Ď': return 'd'; + case 'ď': return 'd'; + case 'Đ': return 'd'; + case 'đ': return 'd'; + case 'Ē': return 'e'; + case 'ē': return 'e'; + case 'Ĕ': return 'e'; + case 'ĕ': return 'e'; + case 'Ė': return 'e'; + case 'ė': return 'e'; + case 'Ę': return 'e'; + case 'ę': return 'e'; + case 'Ě': return 'e'; + case 'ě': return 'e'; + case 'Ĝ': return 'g'; + case 'ĝ': return 'g'; + case 'Ğ': return 'g'; + case 'ğ': return 'g'; + case 'Ġ': return 'g'; + case 'ġ': return 'g'; + case 'Ģ': return 'g'; + case 'ģ': return 'g'; + case 'Ĥ': return 'h'; + case 'ĥ': return 'h'; + case 'Ħ': return 'h'; + case 'ħ': return 'h'; + case 'Ĩ': return 'i'; + case 'ĩ': return 'i'; + case 'Ī': return 'i'; + case 'ī': return 'i'; + case 'Ĭ': return 'i'; + case 'ĭ': return 'i'; + case 'Į': return 'i'; + case 'į': return 'i'; + case 'İ': return 'i'; + case 'ı': return 'i'; + case 'Ĵ': return 'j'; + case 'ĵ': return 'j'; + case 'Ķ': return 'k'; + case 'ķ': return 'k'; + case 'ĸ': return 'k'; + case 'Ĺ': return 'l'; + case 'ĺ': return 'l'; + case 'Ļ': return 'l'; + case 'ļ': return 'l'; + case 'Ľ': return 'l'; + case 'ľ': return 'l'; + case 'Ŀ': return 'l'; + case 'ŀ': return 'l'; + case 'Ł': return 'l'; + case 'ł': return 'l'; + case 'Ń': return 'n'; + case 'ń': return 'n'; + case 'Ņ': return 'n'; + case 'ņ': return 'n'; + case 'Ň': return 'n'; + case 'ň': return 'n'; + case 'Ō': return 'o'; + case 'ō': return 'o'; + case 'Ŏ': return 'o'; + case 'ŏ': return 'o'; + case 'Ő': return 'o'; + case 'ő': return 'o'; + case 'Ŕ': return 'r'; + case 'ŕ': return 'r'; + case 'Ŗ': return 'r'; + case 'ŗ': return 'r'; + case 'Ř': return 'r'; + case 'ř': return 'r'; + case 'Ś': return 's'; + case 'ś': return 's'; + case 'Ŝ': return 's'; + case 'ŝ': return 's'; + case 'Ş': return 's'; + case 'ş': return 's'; + case 'Š': return 's'; + case 'š': return 's'; + case 'Ţ': return 't'; + case 'ţ': return 't'; + case 'Ť': return 't'; + case 'ť': return 't'; + case 'Ŧ': return 't'; + case 'ŧ': return 't'; + case 'Ũ': return 'u'; + case 'ũ': return 'u'; + case 'Ū': return 'u'; + case 'ū': return 'u'; + case 'Ŭ': return 'u'; + case 'ŭ': return 'u'; + case 'Ů': return 'u'; + case 'ů': return 'u'; + case 'Ű': return 'u'; + case 'ű': return 'u'; + case 'Ų': return 'u'; + case 'ų': return 'u'; + case 'Ŵ': return 'w'; + case 'ŵ': return 'w'; + case 'Ŷ': return 'y'; + case 'ŷ': return 'y'; + case 'Ÿ': return 'y'; + case 'Ź': return 'z'; + case 'ź': return 'z'; + case 'Ż': return 'z'; + case 'ż': return 'z'; + case 'Ž': return 'z'; + case 'ž': return 'z'; + case 'ſ': return 's'; + case 'ƀ': return 'b'; + case 'Ɓ': return 'b'; + case 'Ƃ': return 'b'; + case 'ƃ': return 'b'; + case 'Ɔ': return 'o'; + case 'Ƈ': return 'c'; + case 'ƈ': return 'c'; + case 'Ɖ': return 'd'; + case 'Ɗ': return 'd'; + case 'Ƌ': return 'd'; + case 'ƌ': return 'd'; + case 'ƍ': return 'd'; + case 'Ɛ': return 'e'; + case 'Ƒ': return 'f'; + case 'ƒ': return 'f'; + case 'Ɠ': return 'g'; + case 'Ɣ': return 'g'; + case 'Ɩ': return 'i'; + case 'Ɨ': return 'i'; + case 'Ƙ': return 'k'; + case 'ƙ': return 'k'; + case 'ƚ': return 'l'; + case 'ƛ': return 'l'; + case 'Ɯ': return 'w'; + case 'Ɲ': return 'n'; + case 'ƞ': return 'n'; + case 'Ɵ': return 'o'; + case 'Ơ': return 'o'; + case 'ơ': return 'o'; + case 'Ƥ': return 'p'; + case 'ƥ': return 'p'; + case 'ƫ': return 't'; + case 'Ƭ': return 't'; + case 'ƭ': return 't'; + case 'Ʈ': return 't'; + case 'Ư': return 'u'; + case 'ư': return 'u'; + case 'Ʊ': return 'y'; + case 'Ʋ': return 'v'; + case 'Ƴ': return 'y'; + case 'ƴ': return 'y'; + case 'Ƶ': return 'z'; + case 'ƶ': return 'z'; + case 'ƿ': return 'w'; + case 'Ǎ': return 'a'; + case 'ǎ': return 'a'; + case 'Ǐ': return 'i'; + case 'ǐ': return 'i'; + case 'Ǒ': return 'o'; + case 'ǒ': return 'o'; + case 'Ǔ': return 'u'; + case 'ǔ': return 'u'; + case 'Ǖ': return 'u'; + case 'ǖ': return 'u'; + case 'Ǘ': return 'u'; + case 'ǘ': return 'u'; + case 'Ǚ': return 'u'; + case 'ǚ': return 'u'; + case 'Ǜ': return 'u'; + case 'ǜ': return 'u'; + case 'Ǟ': return 'a'; + case 'ǟ': return 'a'; + case 'Ǡ': return 'a'; + case 'ǡ': return 'a'; + case 'Ǥ': return 'g'; + case 'ǥ': return 'g'; + case 'Ǧ': return 'g'; + case 'ǧ': return 'g'; + case 'Ǩ': return 'k'; + case 'ǩ': return 'k'; + case 'Ǫ': return 'o'; + case 'ǫ': return 'o'; + case 'Ǭ': return 'o'; + case 'ǭ': return 'o'; + case 'ǰ': return 'j'; + case 'Dz': return 'd'; + case 'Ǵ': return 'g'; + case 'ǵ': return 'g'; + case 'Ƿ': return 'w'; + case 'Ǹ': return 'n'; + case 'ǹ': return 'n'; + case 'Ǻ': return 'a'; + case 'ǻ': return 'a'; + case 'Ǿ': return 'o'; + case 'ǿ': return 'o'; + case 'Ȁ': return 'a'; + case 'ȁ': return 'a'; + case 'Ȃ': return 'a'; + case 'ȃ': return 'a'; + case 'Ȅ': return 'e'; + case 'ȅ': return 'e'; + case 'Ȇ': return 'e'; + case 'ȇ': return 'e'; + case 'Ȉ': return 'i'; + case 'ȉ': return 'i'; + case 'Ȋ': return 'i'; + case 'ȋ': return 'i'; + case 'Ȍ': return 'o'; + case 'ȍ': return 'o'; + case 'Ȏ': return 'o'; + case 'ȏ': return 'o'; + case 'Ȑ': return 'r'; + case 'ȑ': return 'r'; + case 'Ȓ': return 'r'; + case 'ȓ': return 'r'; + case 'Ȕ': return 'u'; + case 'ȕ': return 'u'; + case 'Ȗ': return 'u'; + case 'ȗ': return 'u'; + case 'Ș': return 's'; + case 'ș': return 's'; + case 'Ț': return 't'; + case 'ț': return 't'; + case 'Ȝ': return 'y'; + case 'ȝ': return 'y'; + case 'Ȟ': return 'h'; + case 'ȟ': return 'h'; + case 'Ȥ': return 'z'; + case 'ȥ': return 'z'; + case 'Ȧ': return 'a'; + case 'ȧ': return 'a'; + case 'Ȩ': return 'e'; + case 'ȩ': return 'e'; + case 'Ȫ': return 'o'; + case 'ȫ': return 'o'; + case 'Ȭ': return 'o'; + case 'ȭ': return 'o'; + case 'Ȯ': return 'o'; + case 'ȯ': return 'o'; + case 'Ȱ': return 'o'; + case 'ȱ': return 'o'; + case 'Ȳ': return 'y'; + case 'ȳ': return 'y'; + case 'A': return 'a'; + case 'B': return 'b'; + case 'C': return 'c'; + case 'D': return 'd'; + case 'E': return 'e'; + case 'F': return 'f'; + case 'G': return 'g'; + case 'H': return 'h'; + case 'I': return 'i'; + case 'J': return 'j'; + case 'K': return 'k'; + case 'L': return 'l'; + case 'M': return 'm'; + case 'N': return 'n'; + case 'O': return 'o'; + case 'P': return 'p'; + case 'Q': return 'q'; + case 'R': return 'r'; + case 'S': return 's'; + case 'T': return 't'; + case 'U': return 'u'; + case 'V': return 'v'; + case 'W': return 'w'; + case 'X': return 'x'; + case 'Y': return 'y'; + case 'Z': return 'z'; + default: + return ch; + } + } + + @Override + public byte getDialpadIndex(char ch) { + if (ch >= '0' && ch <= '9') { + return (byte) (ch - '0'); + } else if (ch >= 'a' && ch <= 'z') { + return (byte) (LATIN_LETTERS_TO_DIGITS[ch - 'a'] - '0'); + } else { + return -1; + } + } + + @Override + public char getDialpadNumericCharacter(char ch) { + if (ch >= 'a' && ch <= 'z') { + return LATIN_LETTERS_TO_DIGITS[ch - 'a']; + } + return ch; + } + + /* + * Generates a space delimited string of pinyins + */ + private String tokenizeToPinyins(String displayName) { + HanziToPinyin hanziToPinyin = HanziToPinyin.getInstance(); + ArrayList<HanziToPinyin.Token> tokens = hanziToPinyin.getTokens(displayName); + ArrayList<String> pinyins = new ArrayList<String>(); + for (HanziToPinyin.Token token : tokens) { + if (token.type != HanziToPinyin.Token.PINYIN) { + return displayName; + } else { + pinyins.add(token.target); + } + } + return TextUtils.join(" ", pinyins); + } + + @Override + public String transliterateName(String index) { + return tokenizeToPinyins(index); + } + + /* + * Since the dialpad is matching off the generated pinyin, the + * Chinese characters can't be highlighted. Just return true if + * there is a match without highlighting a character position. + */ + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + String pinyinName = tokenizeToPinyins(displayName); + if (displayName.equals(pinyinName)) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } + final int nameLength = pinyinName.length(); + final int queryLength = query.length(); + + if (nameLength < queryLength) { + return false; + } + + if (queryLength == 0) { + return false; + } + + return true; + } + + +} diff --git a/src/com/android/dialer/dialpad/GreekSmartDialMap.java b/src/com/android/dialer/dialpad/GreekSmartDialMap.java index 77929b8f9..beb25f38e 100644 --- a/src/com/android/dialer/dialpad/GreekSmartDialMap.java +++ b/src/com/android/dialer/dialpad/GreekSmartDialMap.java @@ -16,6 +16,8 @@ package com.android.dialer.dialpad; +import java.util.ArrayList; + public class GreekSmartDialMap implements SmartDialMap { private static final char[] LATIN_LETTERS_TO_DIGITS = { @@ -480,4 +482,14 @@ public class GreekSmartDialMap implements SmartDialMap { return ch; } + @Override + public String transliterateName(String index) { + return index; + } + + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } } diff --git a/src/com/android/dialer/dialpad/HebrewSmartDialMap.java b/src/com/android/dialer/dialpad/HebrewSmartDialMap.java index 73d10ed92..43011f6d6 100644 --- a/src/com/android/dialer/dialpad/HebrewSmartDialMap.java +++ b/src/com/android/dialer/dialpad/HebrewSmartDialMap.java @@ -1,5 +1,7 @@ package com.android.dialer.dialpad; +import java.util.ArrayList; + public class HebrewSmartDialMap implements SmartDialMap { private static final char[] LATIN_LETTERS_TO_DIGITS = { @@ -415,4 +417,14 @@ public class HebrewSmartDialMap implements SmartDialMap { return ch; } + @Override + public String transliterateName(String index) { + return index; + } + + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } }
\ No newline at end of file diff --git a/src/com/android/dialer/dialpad/KoreanSmartDialMap.java b/src/com/android/dialer/dialpad/KoreanSmartDialMap.java index 527c0d386..61c50a155 100644 --- a/src/com/android/dialer/dialpad/KoreanSmartDialMap.java +++ b/src/com/android/dialer/dialpad/KoreanSmartDialMap.java @@ -16,6 +16,15 @@ package com.android.dialer.dialpad; +import com.android.providers.contacts.NameSplitter; + +import android.app.AppGlobals; +import android.content.Context; +import android.provider.ContactsContract; + +import java.util.ArrayList; +import java.util.Locale; + /** * Ported the logic from 10.1: * https://github.com/CyanogenMod/android_packages_apps_Contacts/blob/cm-10.1/src/com/android/contacts/dialpad/util/NameToNumberKorean.java @@ -500,4 +509,91 @@ public class KoreanSmartDialMap implements SmartDialMap { } } + /** + * Splits the displayName into lastname/first names + */ + private NameSplitter.Name splitName(String displayName) { + Context context = AppGlobals.getInitialApplication(); + NameSplitter nameSplitter = new NameSplitter( + context.getString(com.android.internal.R.string.common_name_prefixes), + context.getString(com.android.internal.R.string.common_last_name_prefixes), + context.getString(com.android.internal.R.string.common_name_suffixes), + context.getString(com.android.internal.R.string.common_name_conjunctions), + Locale.KOREA); + NameSplitter.Name name = new NameSplitter.Name(); + nameSplitter.split(name, displayName, ContactsContract.FullNameStyle.KOREAN); + return name; + } + + /** + * Korean names don't contain spaces between lastname/first names. + * Split the name and add a space so better prefixes can be generated. + * @return lastname + " " + firstname + */ + private String separateFirstNameLastName(String displayName) { + for (int i=0; i<displayName.length(); i++) { + char ch = (char)displayName.codePointAt(i); + if (ch <= UNICODE_HANGUL_START || ch >= UNICODE_HANGUL_END) { + return displayName; + } + } + NameSplitter.Name name = splitName(displayName); + if (name.familyName != null && name.givenNames != null) { + return name.familyName + " " + name.givenNames; + } else { + return displayName; + } + } + + @Override + public String transliterateName(String index) { + return separateFirstNameLastName(index); + } + + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + matchList.clear(); + final int nameLength = displayName.length(); + final int queryLength = query.length(); + + if (nameLength < queryLength) { + return false; + } + + if (queryLength == 0) { + return false; + } + for (int i=0; i<nameLength; i++) { + char ch = (char)displayName.codePointAt(i); + if (ch <= UNICODE_HANGUL_START || ch >= UNICODE_HANGUL_END) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } + } + /* + * For the matcher to work, we need to separate first/last names. + * Adjust the positions of the matches to account for the added space. + */ + NameSplitter.Name name = splitName(displayName); + if (name.familyName != null && name.givenNames != null) { + int separatorIndex = name.familyName.length(); + boolean matches = smartDialNameMatcher.matchesCombination(name.familyName + " " + name.givenNames, + query, matchList); + if (matches) { + for (SmartDialMatchPosition smartDialMatchPosition : matchList) { + if (smartDialMatchPosition.start > separatorIndex) { + smartDialMatchPosition.start--; + } + if (smartDialMatchPosition.end > separatorIndex) { + smartDialMatchPosition.end--; + } + } + return true; + } else { + return false; + } + } else { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } + } } diff --git a/src/com/android/dialer/dialpad/LatinSmartDialMap.java b/src/com/android/dialer/dialpad/LatinSmartDialMap.java index ef1ec0adc..84763e684 100644 --- a/src/com/android/dialer/dialpad/LatinSmartDialMap.java +++ b/src/com/android/dialer/dialpad/LatinSmartDialMap.java @@ -1,5 +1,7 @@ package com.android.dialer.dialpad; +import java.util.ArrayList; + public class LatinSmartDialMap implements SmartDialMap { private static final char[] LATIN_LETTERS_TO_DIGITS = { @@ -410,4 +412,14 @@ public class LatinSmartDialMap implements SmartDialMap { return ch; } + @Override + public String transliterateName(String index) { + return index; + } + + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } } diff --git a/src/com/android/dialer/dialpad/RussianSmartDialMap.java b/src/com/android/dialer/dialpad/RussianSmartDialMap.java index 639049ebe..7a9ace23f 100644 --- a/src/com/android/dialer/dialpad/RussianSmartDialMap.java +++ b/src/com/android/dialer/dialpad/RussianSmartDialMap.java @@ -16,6 +16,8 @@ package com.android.dialer.dialpad; +import java.util.ArrayList; + public class RussianSmartDialMap implements SmartDialMap { private static final char[] LATIN_LETTERS_TO_DIGITS = { @@ -476,4 +478,14 @@ public class RussianSmartDialMap implements SmartDialMap { return ch; } + @Override + public String transliterateName(String index) { + return index; + } + + @Override + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, + String displayName, String query, ArrayList<SmartDialMatchPosition> matchList) { + return smartDialNameMatcher.matchesCombination(displayName, query, matchList); + } } diff --git a/src/com/android/dialer/dialpad/SmartDialMap.java b/src/com/android/dialer/dialpad/SmartDialMap.java index b51891a8c..8773a1532 100644 --- a/src/com/android/dialer/dialpad/SmartDialMap.java +++ b/src/com/android/dialer/dialpad/SmartDialMap.java @@ -1,5 +1,7 @@ package com.android.dialer.dialpad; +import java.util.ArrayList; + /** * Note: These methods currently take characters as arguments. For future planned language support, * they will need to be changed to use codepoints instead of characters. @@ -40,4 +42,15 @@ public interface SmartDialMap { * from accented characters. */ public char normalizeCharacter(char ch); + + /* + * Allow the SmartDialMaps to convert the characters if needed. + */ + public String transliterateName(String index); + + /* + * Allow the SmartDialMaps to provide their own character to dialpad matching if needed. + */ + public boolean matchesCombination(SmartDialNameMatcher smartDialNameMatcher, String displayName, String query, + ArrayList<SmartDialMatchPosition> matchList); } diff --git a/src/com/android/dialer/dialpad/SmartDialNameMatcher.java b/src/com/android/dialer/dialpad/SmartDialNameMatcher.java index 77f417144..4daf0c837 100644 --- a/src/com/android/dialer/dialpad/SmartDialNameMatcher.java +++ b/src/com/android/dialer/dialpad/SmartDialNameMatcher.java @@ -415,7 +415,7 @@ public class SmartDialNameMatcher { if (mMultiMatchObject != null && mMultiMatchMethod != null) { return matchesMultiLanguage(displayName, mQuery, mMatchPositions); } else { - return matchesCombination(displayName, mQuery, mMatchPositions); + return mMap.matchesCombination(this, displayName, mQuery, mMatchPositions); } } diff --git a/src/com/android/dialer/dialpad/SmartDialPrefix.java b/src/com/android/dialer/dialpad/SmartDialPrefix.java index 6f993e4c1..6d2893b71 100644 --- a/src/com/android/dialer/dialpad/SmartDialPrefix.java +++ b/src/com/android/dialer/dialpad/SmartDialPrefix.java @@ -100,6 +100,8 @@ public class SmartDialPrefix { mMap = new HebrewSmartDialMap(); } else if (locale.equals("KR")) { mMap = new KoreanSmartDialMap(); + } else if (locale.equals("CN")) { + mMap = new ChineseSmartDialMap(); } else { mMap = new LatinSmartDialMap(); } @@ -180,7 +182,7 @@ public class SmartDialPrefix { */ public static ArrayList<String> generateNamePrefixes(String index) { final ArrayList<String> result = Lists.newArrayList(); - + index = mMap.transliterateName(index); /** Parses the name into a list of tokens.*/ final ArrayList<String> indexTokens = parseToIndexTokens(index); |