/* * Copyright (C) 2012 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.dialer.dialpad; import android.text.TextUtils; import com.android.dialer.dialpad.SmartDialPrefix.PhoneNumberTokens; import com.google.common.annotations.VisibleForTesting; import com.google.common.collect.Lists; import java.util.ArrayList; /** * {@link #SmartDialNameMatcher} contains utility functions to remove accents from accented * characters and normalize a phone number. It also contains the matching logic that determines if * a contact's display name matches a numeric query. The boolean variable * {@link #ALLOW_INITIAL_MATCH} controls the behavior of the matching logic and determines * whether we allow matches like 57 - (J)ohn (S)mith. */ public class SmartDialNameMatcher { private String mQuery; // Whether or not we allow matches like 57 - (J)ohn (S)mith private static final boolean ALLOW_INITIAL_MATCH = true; // The maximum length of the initial we will match - typically set to 1 to minimize false // positives private static final int INITIAL_LENGTH_LIMIT = 1; private final ArrayList mMatchPositions = Lists.newArrayList(); public static final SmartDialMap LATIN_SMART_DIAL_MAP = new LatinSmartDialMap(); private final SmartDialMap mMap; private String mNameMatchMask = ""; private String mPhoneNumberMatchMask = ""; @VisibleForTesting public SmartDialNameMatcher(String query) { this(query, LATIN_SMART_DIAL_MAP); } public SmartDialNameMatcher(String query, SmartDialMap map) { mQuery = query; mMap = map; } /** * Constructs empty highlight mask. Bit 0 at a position means there is no match, Bit 1 means * there is a match and should be highlighted in the TextView. * @param builder StringBuilder object * @param length Length of the desired mask. */ private void constructEmptyMask(StringBuilder builder, int length) { for (int i = 0; i < length; ++i) { builder.append("0"); } } /** * Replaces the 0-bit at a position with 1-bit, indicating that there is a match. * @param builder StringBuilder object. * @param matchPos Match Positions to mask as 1. */ private void replaceBitInMask(StringBuilder builder, SmartDialMatchPosition matchPos) { for (int i = matchPos.start; i < matchPos.end; ++i) { builder.replace(i, i + 1, "1"); } } /** * Strips a phone number of unnecessary characters (spaces, dashes, etc.) * * @param number Phone number we want to normalize * @return Phone number consisting of digits from 0-9 */ public static String normalizeNumber(String number, SmartDialMap map) { return normalizeNumber(number, 0, map); } /** * Strips a phone number of unnecessary characters (spaces, dashes, etc.) * * @param number Phone number we want to normalize * @param offset Offset to start from * @return Phone number consisting of digits from 0-9 */ public static String normalizeNumber(String number, int offset, SmartDialMap map) { final StringBuilder s = new StringBuilder(); for (int i = offset; i < number.length(); i++) { char ch = number.charAt(i); if (map.isValidDialpadNumericChar(ch)) { s.append(ch); } } return s.toString(); } /** * Matches a phone number against a query. Let the test application overwrite the NANP setting. * * @param phoneNumber - Raw phone number * @param query - Normalized query (only contains numbers from 0-9) * @param useNanp - Overwriting nanp setting boolean, used for testing. * @return {@literal null} if the number and the query don't match, a valid * SmartDialMatchPosition with the matching positions otherwise */ @VisibleForTesting public SmartDialMatchPosition matchesNumber(String phoneNumber, String query, boolean useNanp) { StringBuilder builder = new StringBuilder(); constructEmptyMask(builder, phoneNumber.length()); mPhoneNumberMatchMask = builder.toString(); // Try matching the number as is SmartDialMatchPosition matchPos = matchesNumberWithOffset(phoneNumber, query, 0); if (matchPos == null) { final PhoneNumberTokens phoneNumberTokens = SmartDialPrefix.parsePhoneNumber(phoneNumber); if (phoneNumberTokens == null) { return matchPos; } if (phoneNumberTokens.countryCodeOffset != 0) { matchPos = matchesNumberWithOffset(phoneNumber, query, phoneNumberTokens.countryCodeOffset); } if (matchPos == null && phoneNumberTokens.nanpCodeOffset != 0 && useNanp) { matchPos = matchesNumberWithOffset(phoneNumber, query, phoneNumberTokens.nanpCodeOffset); } } if (matchPos != null) { replaceBitInMask(builder, matchPos); mPhoneNumberMatchMask = builder.toString(); } return matchPos; } /** * Matches a phone number against the saved query, taking care of formatting characters and also * taking into account country code prefixes and special NANP number treatment. * * @param phoneNumber - Raw phone number * @return {@literal null} if the number and the query don't match, a valid * SmartDialMatchPosition with the matching positions otherwise */ public SmartDialMatchPosition matchesNumber(String phoneNumber) { return matchesNumber(phoneNumber, mQuery, true); } /** * Matches a phone number against a query, taking care of formatting characters and also * taking into account country code prefixes and special NANP number treatment. * * @param phoneNumber - Raw phone number * @param query - Normalized query (only contains numbers from 0-9) * @return {@literal null} if the number and the query don't match, a valid * SmartDialMatchPosition with the matching positions otherwise */ public SmartDialMatchPosition matchesNumber(String phoneNumber, String query) { return matchesNumber(phoneNumber, query, true); } /** * Matches a phone number against a query, taking care of formatting characters * * @param phoneNumber - Raw phone number * @param query - Normalized query (only contains numbers from 0-9) * @param offset - The position in the number to start the match against (used to ignore * leading prefixes/country codes) * @return {@literal null} if the number and the query don't match, a valid * SmartDialMatchPosition with the matching positions otherwise */ private SmartDialMatchPosition matchesNumberWithOffset(String phoneNumber, String query, int offset) { if (TextUtils.isEmpty(phoneNumber) || TextUtils.isEmpty(query)) { return null; } int queryAt = 0; int numberAt = offset; for (int i = offset; i < phoneNumber.length(); i++) { if (queryAt == query.length()) { break; } char ch = phoneNumber.charAt(i); if (mMap.isValidDialpadNumericChar(ch)) { if (ch != query.charAt(queryAt)) { return null; } queryAt++; } else { if (queryAt == 0) { // Found a separator before any part of the query was matched, so advance the // offset to avoid prematurely highlighting separators before the rest of the // query. // E.g. don't highlight the first '-' if we're matching 1-510-111-1111 with // '510'. // However, if the current offset is 0, just include the beginning separators // anyway, otherwise the highlighting ends up looking weird. // E.g. if we're matching (510)-111-1111 with '510', we should include the // first '('. if (offset != 0) { offset++; } } } numberAt++; } return new SmartDialMatchPosition(0 + offset, numberAt); } /** * This function iterates through each token in the display name, trying to match the query * to the numeric equivalent of the token. * * A token is defined as a range in the display name delimited by characters that have no * latin alphabet equivalents (e.g. spaces - ' ', periods - ',', underscores - '_' or chinese * characters - '王'). Transliteration from non-latin characters to latin character will be * done on a best effort basis - e.g. 'Ü' - 'u'. * * For example, * the display name "Phillips Thomas Jr" contains three tokens: "phillips", "thomas", and "jr". * * A match must begin at the start of a token. * For example, typing 846(Tho) would match "Phillips Thomas", but 466(hom) would not. * * Also, a match can extend across tokens. * For example, typing 37337(FredS) would match (Fred S)mith. * * @param displayName The normalized(no accented characters) display name we intend to match * against. * @param query The string of digits that we want to match the display name to. * @param matchList An array list of {@link SmartDialMatchPosition}s that we add matched * positions to. * @return Returns true if a combination of the tokens in displayName match the query * string contained in query. If the function returns true, matchList will contain an * ArrayList of match positions (multiple matches correspond to initial matches). */ @VisibleForTesting boolean matchesCombination(String displayName, String query, ArrayList matchList) { StringBuilder builder = new StringBuilder(); constructEmptyMask(builder, displayName.length()); mNameMatchMask = builder.toString(); final int nameLength = displayName.length(); final int queryLength = query.length(); if (nameLength < queryLength) { return false; } if (queryLength == 0) { return false; } // The current character index in displayName // E.g. 3 corresponds to 'd' in "Fred Smith" int nameStart = 0; // The current character in the query we are trying to match the displayName against int queryStart = 0; // The start position of the current token we are inspecting int tokenStart = 0; // The number of non-alphabetic characters we've encountered so far in the current match. // E.g. if we've currently matched 3733764849 to (Fred Smith W)illiam, then the // seperatorCount should be 2. This allows us to correctly calculate offsets for the match // positions int seperatorCount = 0; ArrayList partial = new ArrayList(); // Keep going until we reach the end of displayName while (nameStart < nameLength && queryStart < queryLength) { char ch = displayName.charAt(nameStart); // Strip diacritics from accented characters if any ch = mMap.normalizeCharacter(ch); if (mMap.isValidDialpadCharacter(ch)) { if (mMap.isValidDialpadAlphabeticChar(ch)) { ch = mMap.getDialpadNumericCharacter(ch); } if (ch != query.charAt(queryStart)) { // Failed to match the current character in the query. // Case 1: Failed to match the first character in the query. Skip to the next // token since there is no chance of this token matching the query. // Case 2: Previous characters in the query matched, but the current character // failed to match. This happened in the middle of a token. Skip to the next // token since there is no chance of this token matching the query. // Case 3: Previous characters in the query matched, but the current character // failed to match. This happened right at the start of the current token. In // this case, we should restart the query and try again with the current token. // Otherwise, we would fail to match a query like "964"(yog) against a name // Yo-Yoghurt because the query match would fail on the 3rd character, and // then skip to the end of the "Yoghurt" token. if (queryStart == 0 || mMap.isValidDialpadCharacter(mMap.normalizeCharacter( displayName.charAt(nameStart - 1)))) { // skip to the next token, in the case of 1 or 2. while (nameStart < nameLength && mMap.isValidDialpadCharacter(mMap.normalizeCharacter( displayName.charAt(nameStart)))) { nameStart++; } nameStart++; } // Restart the query and set the correct token position queryStart = 0; seperatorCount = 0; tokenStart = nameStart; } else { if (queryStart == queryLength - 1) { // As much as possible, we prioritize a full token match over a sub token // one so if we find a full token match, we can return right away matchList.add(new SmartDialMatchPosition( tokenStart, queryLength + tokenStart + seperatorCount)); for (SmartDialMatchPosition match : matchList) { replaceBitInMask(builder, match); } mNameMatchMask = builder.toString(); return true; } else if (ALLOW_INITIAL_MATCH && queryStart < INITIAL_LENGTH_LIMIT) { // we matched the first character. // branch off and see if we can find another match with the remaining // characters in the query string and the remaining tokens // find the next separator in the query string int j; for (j = nameStart; j < nameLength; j++) { if (!mMap.isValidDialpadCharacter(mMap.normalizeCharacter( displayName.charAt(j)))) { break; } } // this means there is at least one character left after the separator if (j < nameLength - 1) { final String remainder = displayName.substring(j + 1); final ArrayList partialTemp = Lists.newArrayList(); if (matchesCombination( remainder, query.substring(queryStart + 1), partialTemp)) { // store the list of possible match positions SmartDialMatchPosition.advanceMatchPositions(partialTemp, j + 1); partialTemp.add(0, new SmartDialMatchPosition(nameStart, nameStart + 1)); // we found a partial token match, store the data in a // temp buffer and return it if we end up not finding a full // token match partial = partialTemp; } } } nameStart++; queryStart++; // we matched the current character in the name against one in the query, // continue and see if the rest of the characters match } } else { // found a separator, we skip this character and continue to the next one nameStart++; if (queryStart == 0) { // This means we found a separator before the start of a token, // so we should increment the token's start position to reflect its true // start position tokenStart = nameStart; } else { // Otherwise this separator was found in the middle of a token being matched, // so increase the separator count seperatorCount++; } } } // if we have no complete match at this point, then we attempt to fall back to the partial // token match(if any). If we don't allow initial matching (ALLOW_INITIAL_MATCH = false) // then partial will always be empty. if (!partial.isEmpty()) { matchList.addAll(partial); for (SmartDialMatchPosition match : matchList) { replaceBitInMask(builder, match); } mNameMatchMask = builder.toString(); return true; } return false; } public boolean matches(String displayName) { mMatchPositions.clear(); return matchesCombination(displayName, mQuery, mMatchPositions); } public ArrayList getMatchPositions() { // Return a clone of mMatchPositions so that the caller can use it without // worrying about it changing return new ArrayList(mMatchPositions); } public void setQuery(String query) { mQuery = query; } public String getNameMatchPositionsInString() { return mNameMatchMask; } public String getNumberMatchPositionsInString() { return mPhoneNumberMatchMask; } public String getQuery() { return mQuery; } }