1 /* 2 * Copyright (C) 2012 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 package com.android.dialer.dialpad; 18 19 import android.text.TextUtils; 20 21 import com.android.dialer.dialpad.SmartDialPrefix.PhoneNumberTokens; 22 23 import com.google.common.annotations.VisibleForTesting; 24 import com.google.common.collect.Lists; 25 26 import java.util.ArrayList; 27 28 /** 29 * {@link #SmartDialNameMatcher} contains utility functions to remove accents from accented 30 * characters and normalize a phone number. It also contains the matching logic that determines if 31 * a contact's display name matches a numeric query. The boolean variable 32 * {@link #ALLOW_INITIAL_MATCH} controls the behavior of the matching logic and determines 33 * whether we allow matches like 57 - (J)ohn (S)mith. 34 */ 35 public class SmartDialNameMatcher { 36 37 private String mQuery; 38 39 // Whether or not we allow matches like 57 - (J)ohn (S)mith 40 private static final boolean ALLOW_INITIAL_MATCH = true; 41 42 // The maximum length of the initial we will match - typically set to 1 to minimize false 43 // positives 44 private static final int INITIAL_LENGTH_LIMIT = 1; 45 46 private final ArrayList<SmartDialMatchPosition> mMatchPositions = Lists.newArrayList(); 47 48 public static final SmartDialMap LATIN_SMART_DIAL_MAP = new LatinSmartDialMap(); 49 50 private final SmartDialMap mMap; 51 52 private String mNameMatchMask = ""; 53 private String mPhoneNumberMatchMask = ""; 54 55 @VisibleForTesting SmartDialNameMatcher(String query)56 public SmartDialNameMatcher(String query) { 57 this(query, LATIN_SMART_DIAL_MAP); 58 } 59 SmartDialNameMatcher(String query, SmartDialMap map)60 public SmartDialNameMatcher(String query, SmartDialMap map) { 61 mQuery = query; 62 mMap = map; 63 } 64 65 /** 66 * Constructs empty highlight mask. Bit 0 at a position means there is no match, Bit 1 means 67 * there is a match and should be highlighted in the TextView. 68 * @param builder StringBuilder object 69 * @param length Length of the desired mask. 70 */ constructEmptyMask(StringBuilder builder, int length)71 private void constructEmptyMask(StringBuilder builder, int length) { 72 for (int i = 0; i < length; ++i) { 73 builder.append("0"); 74 } 75 } 76 77 /** 78 * Replaces the 0-bit at a position with 1-bit, indicating that there is a match. 79 * @param builder StringBuilder object. 80 * @param matchPos Match Positions to mask as 1. 81 */ replaceBitInMask(StringBuilder builder, SmartDialMatchPosition matchPos)82 private void replaceBitInMask(StringBuilder builder, SmartDialMatchPosition matchPos) { 83 for (int i = matchPos.start; i < matchPos.end; ++i) { 84 builder.replace(i, i + 1, "1"); 85 } 86 } 87 88 /** 89 * Strips a phone number of unnecessary characters (spaces, dashes, etc.) 90 * 91 * @param number Phone number we want to normalize 92 * @return Phone number consisting of digits from 0-9 93 */ normalizeNumber(String number, SmartDialMap map)94 public static String normalizeNumber(String number, SmartDialMap map) { 95 return normalizeNumber(number, 0, map); 96 } 97 98 /** 99 * Strips a phone number of unnecessary characters (spaces, dashes, etc.) 100 * 101 * @param number Phone number we want to normalize 102 * @param offset Offset to start from 103 * @return Phone number consisting of digits from 0-9 104 */ normalizeNumber(String number, int offset, SmartDialMap map)105 public static String normalizeNumber(String number, int offset, SmartDialMap map) { 106 final StringBuilder s = new StringBuilder(); 107 for (int i = offset; i < number.length(); i++) { 108 char ch = number.charAt(i); 109 if (map.isValidDialpadNumericChar(ch)) { 110 s.append(ch); 111 } 112 } 113 return s.toString(); 114 } 115 116 /** 117 * Matches a phone number against a query. Let the test application overwrite the NANP setting. 118 * 119 * @param phoneNumber - Raw phone number 120 * @param query - Normalized query (only contains numbers from 0-9) 121 * @param useNanp - Overwriting nanp setting boolean, used for testing. 122 * @return {@literal null} if the number and the query don't match, a valid 123 * SmartDialMatchPosition with the matching positions otherwise 124 */ 125 @VisibleForTesting matchesNumber(String phoneNumber, String query, boolean useNanp)126 public SmartDialMatchPosition matchesNumber(String phoneNumber, String query, boolean useNanp) { 127 StringBuilder builder = new StringBuilder(); 128 constructEmptyMask(builder, phoneNumber.length()); 129 mPhoneNumberMatchMask = builder.toString(); 130 131 // Try matching the number as is 132 SmartDialMatchPosition matchPos = matchesNumberWithOffset(phoneNumber, query, 0); 133 if (matchPos == null) { 134 final PhoneNumberTokens phoneNumberTokens = 135 SmartDialPrefix.parsePhoneNumber(phoneNumber); 136 137 if (phoneNumberTokens == null) { 138 return matchPos; 139 } 140 if (phoneNumberTokens.countryCodeOffset != 0) { 141 matchPos = matchesNumberWithOffset(phoneNumber, query, 142 phoneNumberTokens.countryCodeOffset); 143 } 144 if (matchPos == null && phoneNumberTokens.nanpCodeOffset != 0 && useNanp) { 145 matchPos = matchesNumberWithOffset(phoneNumber, query, 146 phoneNumberTokens.nanpCodeOffset); 147 } 148 } 149 if (matchPos != null) { 150 replaceBitInMask(builder, matchPos); 151 mPhoneNumberMatchMask = builder.toString(); 152 } 153 return matchPos; 154 } 155 156 /** 157 * Matches a phone number against the saved query, taking care of formatting characters and also 158 * taking into account country code prefixes and special NANP number treatment. 159 * 160 * @param phoneNumber - Raw phone number 161 * @return {@literal null} if the number and the query don't match, a valid 162 * SmartDialMatchPosition with the matching positions otherwise 163 */ matchesNumber(String phoneNumber)164 public SmartDialMatchPosition matchesNumber(String phoneNumber) { 165 return matchesNumber(phoneNumber, mQuery, true); 166 } 167 168 /** 169 * Matches a phone number against a query, taking care of formatting characters and also 170 * taking into account country code prefixes and special NANP number treatment. 171 * 172 * @param phoneNumber - Raw phone number 173 * @param query - Normalized query (only contains numbers from 0-9) 174 * @return {@literal null} if the number and the query don't match, a valid 175 * SmartDialMatchPosition with the matching positions otherwise 176 */ matchesNumber(String phoneNumber, String query)177 public SmartDialMatchPosition matchesNumber(String phoneNumber, String query) { 178 return matchesNumber(phoneNumber, query, true); 179 } 180 181 /** 182 * Matches a phone number against a query, taking care of formatting characters 183 * 184 * @param phoneNumber - Raw phone number 185 * @param query - Normalized query (only contains numbers from 0-9) 186 * @param offset - The position in the number to start the match against (used to ignore 187 * leading prefixes/country codes) 188 * @return {@literal null} if the number and the query don't match, a valid 189 * SmartDialMatchPosition with the matching positions otherwise 190 */ matchesNumberWithOffset(String phoneNumber, String query, int offset)191 private SmartDialMatchPosition matchesNumberWithOffset(String phoneNumber, String query, 192 int offset) { 193 if (TextUtils.isEmpty(phoneNumber) || TextUtils.isEmpty(query)) { 194 return null; 195 } 196 int queryAt = 0; 197 int numberAt = offset; 198 for (int i = offset; i < phoneNumber.length(); i++) { 199 if (queryAt == query.length()) { 200 break; 201 } 202 char ch = phoneNumber.charAt(i); 203 if (mMap.isValidDialpadNumericChar(ch)) { 204 if (ch != query.charAt(queryAt)) { 205 return null; 206 } 207 queryAt++; 208 } else { 209 if (queryAt == 0) { 210 // Found a separator before any part of the query was matched, so advance the 211 // offset to avoid prematurely highlighting separators before the rest of the 212 // query. 213 // E.g. don't highlight the first '-' if we're matching 1-510-111-1111 with 214 // '510'. 215 // However, if the current offset is 0, just include the beginning separators 216 // anyway, otherwise the highlighting ends up looking weird. 217 // E.g. if we're matching (510)-111-1111 with '510', we should include the 218 // first '('. 219 if (offset != 0) { 220 offset++; 221 } 222 } 223 } 224 numberAt++; 225 } 226 return new SmartDialMatchPosition(0 + offset, numberAt); 227 } 228 229 /** 230 * This function iterates through each token in the display name, trying to match the query 231 * to the numeric equivalent of the token. 232 * 233 * A token is defined as a range in the display name delimited by characters that have no 234 * latin alphabet equivalents (e.g. spaces - ' ', periods - ',', underscores - '_' or chinese 235 * characters - '王'). Transliteration from non-latin characters to latin character will be 236 * done on a best effort basis - e.g. 'Ü' - 'u'. 237 * 238 * For example, 239 * the display name "Phillips Thomas Jr" contains three tokens: "phillips", "thomas", and "jr". 240 * 241 * A match must begin at the start of a token. 242 * For example, typing 846(Tho) would match "Phillips Thomas", but 466(hom) would not. 243 * 244 * Also, a match can extend across tokens. 245 * For example, typing 37337(FredS) would match (Fred S)mith. 246 * 247 * @param displayName The normalized(no accented characters) display name we intend to match 248 * against. 249 * @param query The string of digits that we want to match the display name to. 250 * @param matchList An array list of {@link SmartDialMatchPosition}s that we add matched 251 * positions to. 252 * @return Returns true if a combination of the tokens in displayName match the query 253 * string contained in query. If the function returns true, matchList will contain an 254 * ArrayList of match positions (multiple matches correspond to initial matches). 255 */ 256 @VisibleForTesting matchesCombination(String displayName, String query, ArrayList<SmartDialMatchPosition> matchList)257 boolean matchesCombination(String displayName, String query, 258 ArrayList<SmartDialMatchPosition> matchList) { 259 StringBuilder builder = new StringBuilder(); 260 constructEmptyMask(builder, displayName.length()); 261 mNameMatchMask = builder.toString(); 262 final int nameLength = displayName.length(); 263 final int queryLength = query.length(); 264 265 if (nameLength < queryLength) { 266 return false; 267 } 268 269 if (queryLength == 0) { 270 return false; 271 } 272 273 // The current character index in displayName 274 // E.g. 3 corresponds to 'd' in "Fred Smith" 275 int nameStart = 0; 276 277 // The current character in the query we are trying to match the displayName against 278 int queryStart = 0; 279 280 // The start position of the current token we are inspecting 281 int tokenStart = 0; 282 283 // The number of non-alphabetic characters we've encountered so far in the current match. 284 // E.g. if we've currently matched 3733764849 to (Fred Smith W)illiam, then the 285 // seperatorCount should be 2. This allows us to correctly calculate offsets for the match 286 // positions 287 int seperatorCount = 0; 288 289 ArrayList<SmartDialMatchPosition> partial = new ArrayList<SmartDialMatchPosition>(); 290 // Keep going until we reach the end of displayName 291 while (nameStart < nameLength && queryStart < queryLength) { 292 char ch = displayName.charAt(nameStart); 293 // Strip diacritics from accented characters if any 294 ch = mMap.normalizeCharacter(ch); 295 if (mMap.isValidDialpadCharacter(ch)) { 296 if (mMap.isValidDialpadAlphabeticChar(ch)) { 297 ch = mMap.getDialpadNumericCharacter(ch); 298 } 299 if (ch != query.charAt(queryStart)) { 300 // Failed to match the current character in the query. 301 302 // Case 1: Failed to match the first character in the query. Skip to the next 303 // token since there is no chance of this token matching the query. 304 305 // Case 2: Previous characters in the query matched, but the current character 306 // failed to match. This happened in the middle of a token. Skip to the next 307 // token since there is no chance of this token matching the query. 308 309 // Case 3: Previous characters in the query matched, but the current character 310 // failed to match. This happened right at the start of the current token. In 311 // this case, we should restart the query and try again with the current token. 312 // Otherwise, we would fail to match a query like "964"(yog) against a name 313 // Yo-Yoghurt because the query match would fail on the 3rd character, and 314 // then skip to the end of the "Yoghurt" token. 315 316 if (queryStart == 0 || mMap.isValidDialpadCharacter(mMap.normalizeCharacter( 317 displayName.charAt(nameStart - 1)))) { 318 // skip to the next token, in the case of 1 or 2. 319 while (nameStart < nameLength && 320 mMap.isValidDialpadCharacter(mMap.normalizeCharacter( 321 displayName.charAt(nameStart)))) { 322 nameStart++; 323 } 324 nameStart++; 325 } 326 327 // Restart the query and set the correct token position 328 queryStart = 0; 329 seperatorCount = 0; 330 tokenStart = nameStart; 331 } else { 332 if (queryStart == queryLength - 1) { 333 334 // As much as possible, we prioritize a full token match over a sub token 335 // one so if we find a full token match, we can return right away 336 matchList.add(new SmartDialMatchPosition( 337 tokenStart, queryLength + tokenStart + seperatorCount)); 338 for (SmartDialMatchPosition match : matchList) { 339 replaceBitInMask(builder, match); 340 } 341 mNameMatchMask = builder.toString(); 342 return true; 343 } else if (ALLOW_INITIAL_MATCH && queryStart < INITIAL_LENGTH_LIMIT) { 344 // we matched the first character. 345 // branch off and see if we can find another match with the remaining 346 // characters in the query string and the remaining tokens 347 // find the next separator in the query string 348 int j; 349 for (j = nameStart; j < nameLength; j++) { 350 if (!mMap.isValidDialpadCharacter(mMap.normalizeCharacter( 351 displayName.charAt(j)))) { 352 break; 353 } 354 } 355 // this means there is at least one character left after the separator 356 if (j < nameLength - 1) { 357 final String remainder = displayName.substring(j + 1); 358 final ArrayList<SmartDialMatchPosition> partialTemp = 359 Lists.newArrayList(); 360 if (matchesCombination( 361 remainder, query.substring(queryStart + 1), partialTemp)) { 362 363 // store the list of possible match positions 364 SmartDialMatchPosition.advanceMatchPositions(partialTemp, j + 1); 365 partialTemp.add(0, 366 new SmartDialMatchPosition(nameStart, nameStart + 1)); 367 // we found a partial token match, store the data in a 368 // temp buffer and return it if we end up not finding a full 369 // token match 370 partial = partialTemp; 371 } 372 } 373 } 374 nameStart++; 375 queryStart++; 376 // we matched the current character in the name against one in the query, 377 // continue and see if the rest of the characters match 378 } 379 } else { 380 // found a separator, we skip this character and continue to the next one 381 nameStart++; 382 if (queryStart == 0) { 383 // This means we found a separator before the start of a token, 384 // so we should increment the token's start position to reflect its true 385 // start position 386 tokenStart = nameStart; 387 } else { 388 // Otherwise this separator was found in the middle of a token being matched, 389 // so increase the separator count 390 seperatorCount++; 391 } 392 } 393 } 394 // if we have no complete match at this point, then we attempt to fall back to the partial 395 // token match(if any). If we don't allow initial matching (ALLOW_INITIAL_MATCH = false) 396 // then partial will always be empty. 397 if (!partial.isEmpty()) { 398 matchList.addAll(partial); 399 for (SmartDialMatchPosition match : matchList) { 400 replaceBitInMask(builder, match); 401 } 402 mNameMatchMask = builder.toString(); 403 return true; 404 } 405 return false; 406 } 407 matches(String displayName)408 public boolean matches(String displayName) { 409 mMatchPositions.clear(); 410 return matchesCombination(displayName, mQuery, mMatchPositions); 411 } 412 getMatchPositions()413 public ArrayList<SmartDialMatchPosition> getMatchPositions() { 414 // Return a clone of mMatchPositions so that the caller can use it without 415 // worrying about it changing 416 return new ArrayList<SmartDialMatchPosition>(mMatchPositions); 417 } 418 setQuery(String query)419 public void setQuery(String query) { 420 mQuery = query; 421 } 422 getNameMatchPositionsInString()423 public String getNameMatchPositionsInString() { 424 return mNameMatchMask; 425 } 426 getNumberMatchPositionsInString()427 public String getNumberMatchPositionsInString() { 428 return mPhoneNumberMatchMask; 429 } 430 getQuery()431 public String getQuery() { 432 return mQuery; 433 } 434 } 435