summaryrefslogtreecommitdiff
path: root/java/com/android/dialer/enrichedcall/FuzzyPhoneNumberMatcher.java
blob: 6f4d9752188d6c6b7d6bcff7d1c2fdab11123b57 (plain)
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
/*
 * Copyright (C) 2017 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.enrichedcall;

import android.support.annotation.NonNull;

/** Utility for comparing phone numbers. */
public class FuzzyPhoneNumberMatcher {

  private static final int REQUIRED_MATCHED_DIGITS = 7;

  /**
   * Returns {@code true} if the given numbers can be interpreted to be the same.
   *
   * <p>This method is called numerous times when rendering the call log. Using string methods is
   * too slow, so character by character matching is used instead.
   */
  public static boolean matches(@NonNull String lhs, @NonNull String rhs) {
    return lastSevenDigitsCharacterByCharacterMatches(lhs, rhs);
  }

  /**
   * This strategy examines the numbers character by character starting from the end. If the last
   * {@link #REQUIRED_MATCHED_DIGITS} match, it returns {@code true}.
   */
  private static boolean lastSevenDigitsCharacterByCharacterMatches(
      @NonNull String lhs, @NonNull String rhs) {
    int lhsIndex = lhs.length() - 1;
    int rhsIndex = rhs.length() - 1;

    int matchedDigits = 0;

    while (lhsIndex >= 0 && rhsIndex >= 0) {
      if (!Character.isDigit(lhs.charAt(lhsIndex))) {
        --lhsIndex;
        continue;
      }
      if (!Character.isDigit(rhs.charAt(rhsIndex))) {
        --rhsIndex;
        continue;
      }
      if (lhs.charAt(lhsIndex) != rhs.charAt(rhsIndex)) {
        break;
      }
      --lhsIndex;
      --rhsIndex;
      ++matchedDigits;
    }

    return matchedDigits >= REQUIRED_MATCHED_DIGITS;
  }
}