| /* | 
 |  * Copyright (C) 2015 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 android.telecom; | 
 |  | 
 | import android.app.ActivityManager; | 
 | import android.app.role.RoleManager; | 
 | import android.content.Context; | 
 | import android.content.Intent; | 
 | import android.content.pm.ActivityInfo; | 
 | import android.content.pm.PackageManager; | 
 | import android.content.pm.ResolveInfo; | 
 | import android.net.Uri; | 
 | import android.os.AsyncTask; | 
 | import android.os.Binder; | 
 | import android.os.Process; | 
 | import android.os.UserHandle; | 
 | import android.text.TextUtils; | 
 | import android.util.Slog; | 
 |  | 
 | import com.android.internal.util.CollectionUtils; | 
 |  | 
 | import java.util.ArrayList; | 
 | import java.util.List; | 
 | import java.util.concurrent.CompletableFuture; | 
 | import java.util.concurrent.ExecutionException; | 
 | import java.util.concurrent.TimeUnit; | 
 | import java.util.concurrent.TimeoutException; | 
 | import java.util.function.Consumer; | 
 |  | 
 | /** | 
 |  * Class for managing the default dialer application that will receive incoming calls, and be | 
 |  * allowed to make emergency outgoing calls. | 
 |  * | 
 |  * @hide | 
 |  */ | 
 | public class DefaultDialerManager { | 
 |     private static final String TAG = "DefaultDialerManager"; | 
 |  | 
 |     /** | 
 |      * Sets the specified package name as the default dialer application for the current user. | 
 |      * The caller of this method needs to have permission to write to secure settings and | 
 |      * manage users on the device. | 
 |      * | 
 |      * @return {@code true} if the default dialer application was successfully changed, | 
 |      *         {@code false} otherwise. | 
 |      * | 
 |      * @hide | 
 |      * */ | 
 |     public static boolean setDefaultDialerApplication(Context context, String packageName) { | 
 |         return setDefaultDialerApplication(context, packageName, ActivityManager.getCurrentUser()); | 
 |     } | 
 |  | 
 |     /** | 
 |      * Sets the specified package name as the default dialer application for the specified user. | 
 |      * The caller of this method needs to have permission to write to secure settings and | 
 |      * manage users on the device. | 
 |      * | 
 |      * @return {@code true} if the default dialer application was successfully changed, | 
 |      *         {@code false} otherwise. | 
 |      * | 
 |      * @hide | 
 |      * */ | 
 |     public static boolean setDefaultDialerApplication(Context context, String packageName, | 
 |             int user) { | 
 |         long identity = Binder.clearCallingIdentity(); | 
 |         try { | 
 |             CompletableFuture<Void> future = new CompletableFuture<>(); | 
 |             Consumer<Boolean> callback = successful -> { | 
 |                 if (successful) { | 
 |                     future.complete(null); | 
 |                 } else { | 
 |                     future.completeExceptionally(new RuntimeException()); | 
 |                 } | 
 |             }; | 
 |             context.getSystemService(RoleManager.class).addRoleHolderAsUser( | 
 |                     RoleManager.ROLE_DIALER, packageName, 0, UserHandle.of(user), | 
 |                     AsyncTask.THREAD_POOL_EXECUTOR, callback); | 
 |             future.get(5, TimeUnit.SECONDS); | 
 |             return true; | 
 |         } catch (InterruptedException | ExecutionException | TimeoutException e) { | 
 |             Slog.e(TAG, "Failed to set default dialer to " + packageName + " for user " + user, e); | 
 |             return false; | 
 |         } finally { | 
 |             Binder.restoreCallingIdentity(identity); | 
 |         } | 
 |     } | 
 |  | 
 |     /** | 
 |      * Returns the installed dialer application for the current user that will be used to receive | 
 |      * incoming calls, and is allowed to make emergency calls. | 
 |      * | 
 |      * The application will be returned in order of preference: | 
 |      * 1) User selected phone application (if still installed) | 
 |      * 2) Pre-installed system dialer (if not disabled) | 
 |      * 3) Null | 
 |      * | 
 |      * The caller of this method needs to have permission to manage users on the device. | 
 |      * | 
 |      * @hide | 
 |      * */ | 
 |     public static String getDefaultDialerApplication(Context context) { | 
 |         return getDefaultDialerApplication(context, context.getUserId()); | 
 |     } | 
 |  | 
 |     /** | 
 |      * Returns the installed dialer application for the specified user that will be used to receive | 
 |      * incoming calls, and is allowed to make emergency calls. | 
 |      * | 
 |      * The application will be returned in order of preference: | 
 |      * 1) User selected phone application (if still installed) | 
 |      * 2) Pre-installed system dialer (if not disabled) | 
 |      * 3) Null | 
 |      * | 
 |      * The caller of this method needs to have permission to manage users on the device. | 
 |      * | 
 |      * @hide | 
 |      * */ | 
 |     public static String getDefaultDialerApplication(Context context, int user) { | 
 |         long identity = Binder.clearCallingIdentity(); | 
 |         try { | 
 |             return CollectionUtils.firstOrNull(context.getSystemService(RoleManager.class) | 
 |                     .getRoleHoldersAsUser(RoleManager.ROLE_DIALER, UserHandle.of(user))); | 
 |         } finally { | 
 |             Binder.restoreCallingIdentity(identity); | 
 |         } | 
 |     } | 
 |  | 
 |     /** | 
 |      * Returns a list of installed and available dialer applications. | 
 |      * | 
 |      * In order to appear in the list, a dialer application must implement an intent-filter with | 
 |      * the DIAL intent for the following schemes: | 
 |      * | 
 |      * 1) Empty scheme | 
 |      * 2) tel Uri scheme | 
 |      * | 
 |      * @hide | 
 |      **/ | 
 |     public static List<String> getInstalledDialerApplications(Context context, int userId) { | 
 |         PackageManager packageManager = context.getPackageManager(); | 
 |  | 
 |         // Get the list of apps registered for the DIAL intent with empty scheme | 
 |         Intent intent = new Intent(Intent.ACTION_DIAL); | 
 |         List<ResolveInfo> resolveInfoList = | 
 |                 packageManager.queryIntentActivitiesAsUser(intent, 0, userId); | 
 |  | 
 |         List<String> packageNames = new ArrayList<>(); | 
 |  | 
 |         for (ResolveInfo resolveInfo : resolveInfoList) { | 
 |             final ActivityInfo activityInfo = resolveInfo.activityInfo; | 
 |             if (activityInfo != null | 
 |                     && !packageNames.contains(activityInfo.packageName) | 
 |                     // ignore cross profile intent handler | 
 |                     && resolveInfo.targetUserId == UserHandle.USER_CURRENT) { | 
 |                 packageNames.add(activityInfo.packageName); | 
 |             } | 
 |         } | 
 |  | 
 |         final Intent dialIntentWithTelScheme = new Intent(Intent.ACTION_DIAL); | 
 |         dialIntentWithTelScheme.setData(Uri.fromParts(PhoneAccount.SCHEME_TEL, "", null)); | 
 |         return filterByIntent(context, packageNames, dialIntentWithTelScheme, userId); | 
 |     } | 
 |  | 
 |     public static List<String> getInstalledDialerApplications(Context context) { | 
 |         return getInstalledDialerApplications(context, Process.myUserHandle().getIdentifier()); | 
 |     } | 
 |  | 
 |     /** | 
 |      * Determines if the package name belongs to the user-selected default dialer or the preloaded | 
 |      * system dialer, and thus should be allowed to perform certain privileged operations. | 
 |      * | 
 |      * @param context A valid context. | 
 |      * @param packageName of the package to check for. | 
 |      * | 
 |      * @return {@code true} if the provided package name corresponds to the user-selected default | 
 |      *         dialer or the preloaded system dialer, {@code false} otherwise. | 
 |      * | 
 |      * @hide | 
 |      */ | 
 |     public static boolean isDefaultOrSystemDialer(Context context, String packageName) { | 
 |         if (TextUtils.isEmpty(packageName)) { | 
 |             return false; | 
 |         } | 
 |         final TelecomManager tm = getTelecomManager(context); | 
 |         return packageName.equals(tm.getDefaultDialerPackage()) | 
 |                 || packageName.equals(tm.getSystemDialerPackage()); | 
 |     } | 
 |  | 
 |     /** | 
 |      * Filter a given list of package names for those packages that contain an activity that has | 
 |      * an intent filter for a given intent. | 
 |      * | 
 |      * @param context A valid context | 
 |      * @param packageNames List of package names to filter. | 
 |      * @param userId The UserId | 
 |      * @return The filtered list. | 
 |      */ | 
 |     private static List<String> filterByIntent(Context context, List<String> packageNames, | 
 |             Intent intent, int userId) { | 
 |         if (packageNames == null || packageNames.isEmpty()) { | 
 |             return new ArrayList<>(); | 
 |         } | 
 |  | 
 |         final List<String> result = new ArrayList<>(); | 
 |         final List<ResolveInfo> resolveInfoList = context.getPackageManager() | 
 |                 .queryIntentActivitiesAsUser(intent, 0, userId); | 
 |         final int length = resolveInfoList.size(); | 
 |         for (int i = 0; i < length; i++) { | 
 |             final ActivityInfo info = resolveInfoList.get(i).activityInfo; | 
 |             if (info != null && packageNames.contains(info.packageName) | 
 |                     && !result.contains(info.packageName)) { | 
 |                 result.add(info.packageName); | 
 |             } | 
 |         } | 
 |  | 
 |         return result; | 
 |     } | 
 |  | 
 |  | 
 |     private static TelecomManager getTelecomManager(Context context) { | 
 |         return (TelecomManager) context.getSystemService(Context.TELECOM_SERVICE); | 
 |     } | 
 | } |