1 /*
2  * Copyright (C) 2013 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.phonenumberutil;
18 
19 import android.content.Context;
20 import android.database.Cursor;
21 import android.net.Uri;
22 import android.os.Trace;
23 import android.provider.CallLog;
24 import android.support.annotation.NonNull;
25 import android.support.annotation.Nullable;
26 import android.telecom.PhoneAccountHandle;
27 import android.telephony.PhoneNumberUtils;
28 import android.telephony.SubscriptionInfo;
29 import android.telephony.TelephonyManager;
30 import android.text.BidiFormatter;
31 import android.text.TextDirectionHeuristics;
32 import android.text.TextUtils;
33 import com.android.dialer.common.Assert;
34 import com.android.dialer.common.LogUtil;
35 import com.android.dialer.compat.telephony.TelephonyManagerCompat;
36 import com.android.dialer.i18n.LocaleUtils;
37 import com.android.dialer.oem.MotorolaUtils;
38 import com.android.dialer.oem.PhoneNumberUtilsAccessor;
39 import com.android.dialer.phonenumbergeoutil.PhoneNumberGeoUtilComponent;
40 import com.android.dialer.telecom.TelecomUtil;
41 import com.google.common.base.Optional;
42 import java.util.Arrays;
43 import java.util.HashSet;
44 import java.util.List;
45 import java.util.Set;
46 
47 public class PhoneNumberHelper {
48 
49   private static final Set<String> LEGACY_UNKNOWN_NUMBERS =
50       new HashSet<>(Arrays.asList("-1", "-2", "-3"));
51 
52   /** Returns true if it is possible to place a call to the given number. */
canPlaceCallsTo(CharSequence number, int presentation)53   public static boolean canPlaceCallsTo(CharSequence number, int presentation) {
54     return presentation == CallLog.Calls.PRESENTATION_ALLOWED
55         && !TextUtils.isEmpty(number)
56         && !isLegacyUnknownNumbers(number);
57   }
58 
59   /**
60    * Move the given cursor to a position where the number it points to matches the number in a
61    * contact lookup URI.
62    *
63    * <p>We assume the cursor is one returned by the Contacts Provider when the URI asks for a
64    * specific number. This method's behavior is undefined when the cursor doesn't meet the
65    * assumption.
66    *
67    * <p>When determining whether two phone numbers are identical enough for caller ID purposes, the
68    * Contacts Provider ignores special characters such as '#'. This makes it possible for the cursor
69    * returned by the Contacts Provider to have multiple rows even when the URI asks for a specific
70    * number.
71    *
72    * <p>For example, suppose the user has two contacts whose numbers are "#123" and "123",
73    * respectively. When the URI asks for number "123", both numbers will be returned. Therefore, the
74    * following strategy is employed to find a match.
75    *
76    * <p>In the following description, we use E to denote a number the cursor points to (an existing
77    * contact number), and L to denote the number in the contact lookup URI.
78    *
79    * <p>If neither E nor L contains special characters, return true to indicate a match is found.
80    *
81    * <p>If either E or L contains special characters, return true when the raw numbers of E and L
82    * are the same. Otherwise, move the cursor to its next position and start over.
83    *
84    * <p>Return false in all other circumstances to indicate that no match can be found.
85    *
86    * <p>When no match can be found, the cursor is after the last result when the method returns.
87    *
88    * @param cursor A cursor returned by the Contacts Provider.
89    * @param columnIndexForNumber The index of the column where phone numbers are stored. It is the
90    *     caller's responsibility to pass the correct column index.
91    * @param contactLookupUri A URI used to retrieve a contact via the Contacts Provider. It is the
92    *     caller's responsibility to ensure the URI is one that asks for a specific phone number.
93    * @return true if a match can be found.
94    */
updateCursorToMatchContactLookupUri( @ullable Cursor cursor, int columnIndexForNumber, @Nullable Uri contactLookupUri)95   public static boolean updateCursorToMatchContactLookupUri(
96       @Nullable Cursor cursor, int columnIndexForNumber, @Nullable Uri contactLookupUri) {
97     if (cursor == null || contactLookupUri == null) {
98       return false;
99     }
100 
101     if (!cursor.moveToFirst()) {
102       return false;
103     }
104 
105     Assert.checkArgument(
106         0 <= columnIndexForNumber && columnIndexForNumber < cursor.getColumnCount());
107 
108     String lookupNumber = contactLookupUri.getLastPathSegment();
109     if (TextUtils.isEmpty(lookupNumber)) {
110       return false;
111     }
112 
113     boolean lookupNumberHasSpecialChars = numberHasSpecialChars(lookupNumber);
114 
115     do {
116       String existingContactNumber = cursor.getString(columnIndexForNumber);
117       boolean existingContactNumberHasSpecialChars = numberHasSpecialChars(existingContactNumber);
118 
119       if ((!lookupNumberHasSpecialChars && !existingContactNumberHasSpecialChars)
120           || sameRawNumbers(existingContactNumber, lookupNumber)) {
121         return true;
122       }
123 
124     } while (cursor.moveToNext());
125 
126     return false;
127   }
128 
129   /** Returns true if the input phone number contains special characters. */
numberHasSpecialChars(String number)130   public static boolean numberHasSpecialChars(String number) {
131     return !TextUtils.isEmpty(number) && number.contains("#");
132   }
133 
134   /** Returns true if the raw numbers of the two input phone numbers are the same. */
sameRawNumbers(String number1, String number2)135   public static boolean sameRawNumbers(String number1, String number2) {
136     String rawNumber1 =
137         PhoneNumberUtils.stripSeparators(PhoneNumberUtils.convertKeypadLettersToDigits(number1));
138     String rawNumber2 =
139         PhoneNumberUtils.stripSeparators(PhoneNumberUtils.convertKeypadLettersToDigits(number2));
140 
141     return rawNumber1.equals(rawNumber2);
142   }
143 
144   /**
145    * An enhanced version of {@link PhoneNumberUtils#isLocalEmergencyNumber(Context, String)}.
146    *
147    * <p>This methods supports checking the number for all SIMs.
148    *
149    * @param context the context which the number should be checked against
150    * @param number the number to tbe checked
151    * @return true if the specified number is an emergency number for any SIM in the device.
152    */
153   @SuppressWarnings("Guava")
isLocalEmergencyNumber(Context context, String number)154   public static boolean isLocalEmergencyNumber(Context context, String number) {
155     List<PhoneAccountHandle> phoneAccountHandles =
156         TelecomUtil.getSubscriptionPhoneAccounts(context);
157 
158     // If the number of phone accounts with a subscription is no greater than 1, only one SIM is
159     // installed in the device. We hand over the job to PhoneNumberUtils#isLocalEmergencyNumber.
160     if (phoneAccountHandles.size() <= 1) {
161       return PhoneNumberUtils.isLocalEmergencyNumber(context, number);
162     }
163 
164     for (PhoneAccountHandle phoneAccountHandle : phoneAccountHandles) {
165       Optional<SubscriptionInfo> subscriptionInfo =
166           TelecomUtil.getSubscriptionInfo(context, phoneAccountHandle);
167       if (subscriptionInfo.isPresent()
168           && PhoneNumberUtilsAccessor.isLocalEmergencyNumber(
169               context, subscriptionInfo.get().getSubscriptionId(), number)) {
170         return true;
171       }
172     }
173 
174     return false;
175   }
176 
177   /**
178    * Returns true if the given number is the number of the configured voicemail. To be able to
179    * mock-out this, it is not a static method.
180    */
isVoicemailNumber( Context context, PhoneAccountHandle accountHandle, CharSequence number)181   public static boolean isVoicemailNumber(
182       Context context, PhoneAccountHandle accountHandle, CharSequence number) {
183     if (TextUtils.isEmpty(number)) {
184       return false;
185     }
186     return TelecomUtil.isVoicemailNumber(context, accountHandle, number.toString());
187   }
188 
189   /**
190    * Returns true if the given number is a SIP address. To be able to mock-out this, it is not a
191    * static method.
192    */
isSipNumber(CharSequence number)193   public static boolean isSipNumber(CharSequence number) {
194     return number != null && isUriNumber(number.toString());
195   }
196 
isUnknownNumberThatCanBeLookedUp( Context context, PhoneAccountHandle accountHandle, CharSequence number, int presentation)197   public static boolean isUnknownNumberThatCanBeLookedUp(
198       Context context, PhoneAccountHandle accountHandle, CharSequence number, int presentation) {
199     if (presentation == CallLog.Calls.PRESENTATION_UNKNOWN) {
200       return false;
201     }
202     if (presentation == CallLog.Calls.PRESENTATION_RESTRICTED) {
203       return false;
204     }
205     if (presentation == CallLog.Calls.PRESENTATION_UNAVAILABLE) {
206       return false;
207     }
208     if (presentation == CallLog.Calls.PRESENTATION_PAYPHONE) {
209       return false;
210     }
211     if (TextUtils.isEmpty(number)) {
212       return false;
213     }
214     if (isVoicemailNumber(context, accountHandle, number)) {
215       return false;
216     }
217     if (isLegacyUnknownNumbers(number)) {
218       return false;
219     }
220     return true;
221   }
222 
isLegacyUnknownNumbers(CharSequence number)223   public static boolean isLegacyUnknownNumbers(CharSequence number) {
224     return number != null && LEGACY_UNKNOWN_NUMBERS.contains(number.toString());
225   }
226 
227   /**
228    * @param countryIso Country ISO used if there is no country code in the number, may be null
229    *     otherwise.
230    * @return a geographical description string for the specified number.
231    */
getGeoDescription( Context context, String number, @Nullable String countryIso)232   public static String getGeoDescription(
233       Context context, String number, @Nullable String countryIso) {
234     return PhoneNumberGeoUtilComponent.get(context)
235         .getPhoneNumberGeoUtil()
236         .getGeoDescription(context, number, countryIso);
237   }
238 
239   /**
240    * @param phoneAccountHandle {@code PhonAccountHandle} used to get current network country ISO.
241    *     May be null if no account is in use or selected, in which case default account will be
242    *     used.
243    * @return The ISO 3166-1 two letters country code of the country the user is in based on the
244    *     network location. If the network location does not exist, fall back to the locale setting.
245    */
getCurrentCountryIso( Context context, @Nullable PhoneAccountHandle phoneAccountHandle)246   public static String getCurrentCountryIso(
247       Context context, @Nullable PhoneAccountHandle phoneAccountHandle) {
248     Trace.beginSection("PhoneNumberHelper.getCurrentCountryIso");
249     // Without framework function calls, this seems to be the most accurate location service
250     // we can rely on.
251     String countryIso =
252         TelephonyManagerCompat.getNetworkCountryIsoForPhoneAccountHandle(
253             context, phoneAccountHandle);
254     if (TextUtils.isEmpty(countryIso)) {
255       countryIso = LocaleUtils.getLocale(context).getCountry();
256       LogUtil.i(
257           "PhoneNumberHelper.getCurrentCountryIso",
258           "No CountryDetector; falling back to countryIso based on locale: " + countryIso);
259     }
260     countryIso = countryIso.toUpperCase();
261     Trace.endSection();
262 
263     return countryIso;
264   }
265 
266   /**
267    * An enhanced version of {@link PhoneNumberUtils#formatNumber(String, String, String)}.
268    *
269    * <p>The {@link Context} parameter allows us to tweak formatting according to device properties.
270    *
271    * <p>Returns the formatted phone number (e.g, 1-123-456-7890) or the original number if
272    * formatting fails or is intentionally ignored.
273    */
formatNumber( Context context, @Nullable String number, @Nullable String numberE164, String countryIso)274   public static String formatNumber(
275       Context context, @Nullable String number, @Nullable String numberE164, String countryIso) {
276     // The number can be null e.g. schema is voicemail and uri content is empty.
277     if (number == null) {
278       return null;
279     }
280 
281     if (MotorolaUtils.shouldDisablePhoneNumberFormatting(context)) {
282       return number;
283     }
284 
285     String formattedNumber = PhoneNumberUtils.formatNumber(number, numberE164, countryIso);
286     return formattedNumber != null ? formattedNumber : number;
287   }
288 
289   /** @see #formatNumber(Context, String, String, String). */
formatNumber(Context context, @Nullable String number, String countryIso)290   public static String formatNumber(Context context, @Nullable String number, String countryIso) {
291     return formatNumber(context, number, /* numberE164 = */ null, countryIso);
292   }
293 
294   @Nullable
formatNumberForDisplay( Context context, @Nullable String number, @NonNull String countryIso)295   public static CharSequence formatNumberForDisplay(
296       Context context, @Nullable String number, @NonNull String countryIso) {
297     if (number == null) {
298       return null;
299     }
300 
301     return PhoneNumberUtils.createTtsSpannable(
302         BidiFormatter.getInstance()
303             .unicodeWrap(formatNumber(context, number, countryIso), TextDirectionHeuristics.LTR));
304   }
305 
306   /**
307    * Determines if the specified number is actually a URI (i.e. a SIP address) rather than a regular
308    * PSTN phone number, based on whether or not the number contains an "@" character.
309    *
310    * @param number Phone number
311    * @return true if number contains @
312    *     <p>TODO: Remove if PhoneNumberUtils.isUriNumber(String number) is made public.
313    */
isUriNumber(String number)314   public static boolean isUriNumber(String number) {
315     // Note we allow either "@" or "%40" to indicate a URI, in case
316     // the passed-in string is URI-escaped.  (Neither "@" nor "%40"
317     // will ever be found in a legal PSTN number.)
318     return number != null && (number.contains("@") || number.contains("%40"));
319   }
320 
321   /**
322    * @param number SIP address of the form "username@domainname" (or the URI-escaped equivalent
323    *     "username%40domainname")
324    *     <p>TODO: Remove if PhoneNumberUtils.getUsernameFromUriNumber(String number) is made public.
325    * @return the "username" part of the specified SIP address, i.e. the part before the "@"
326    *     character (or "%40").
327    */
getUsernameFromUriNumber(String number)328   public static String getUsernameFromUriNumber(String number) {
329     // The delimiter between username and domain name can be
330     // either "@" or "%40" (the URI-escaped equivalent.)
331     int delimiterIndex = number.indexOf('@');
332     if (delimiterIndex < 0) {
333       delimiterIndex = number.indexOf("%40");
334     }
335     if (delimiterIndex < 0) {
336       LogUtil.i(
337           "PhoneNumberHelper.getUsernameFromUriNumber",
338           "getUsernameFromUriNumber: no delimiter found in SIP address: "
339               + LogUtil.sanitizePii(number));
340       return number;
341     }
342     return number.substring(0, delimiterIndex);
343   }
344 
isVerizon(Context context)345   private static boolean isVerizon(Context context) {
346     // Verizon MCC/MNC codes copied from com/android/voicemailomtp/res/xml/vvm_config.xml.
347     // TODO(sail): Need a better way to do per carrier and per OEM configurations.
348     switch (context.getSystemService(TelephonyManager.class).getSimOperator()) {
349       case "310004":
350       case "310010":
351       case "310012":
352       case "310013":
353       case "310590":
354       case "310890":
355       case "310910":
356       case "311110":
357       case "311270":
358       case "311271":
359       case "311272":
360       case "311273":
361       case "311274":
362       case "311275":
363       case "311276":
364       case "311277":
365       case "311278":
366       case "311279":
367       case "311280":
368       case "311281":
369       case "311282":
370       case "311283":
371       case "311284":
372       case "311285":
373       case "311286":
374       case "311287":
375       case "311288":
376       case "311289":
377       case "311390":
378       case "311480":
379       case "311481":
380       case "311482":
381       case "311483":
382       case "311484":
383       case "311485":
384       case "311486":
385       case "311487":
386       case "311488":
387       case "311489":
388         return true;
389       default:
390         return false;
391     }
392   }
393 
394   /**
395    * Gets the label to display for a phone call where the presentation is set as
396    * PRESENTATION_RESTRICTED. For Verizon we want this to be displayed as "Restricted". For all
397    * other carriers we want this to be be displayed as "Private number".
398    */
getDisplayNameForRestrictedNumber(Context context)399   public static String getDisplayNameForRestrictedNumber(Context context) {
400     if (isVerizon(context)) {
401       return context.getString(R.string.private_num_verizon);
402     } else {
403       return context.getString(R.string.private_num_non_verizon);
404     }
405   }
406 }
407