1 /* 2 * Copyright (C) 2015 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except 5 * in compliance with the License. You may obtain a copy of the License at 6 * 7 * http://www.apache.org/licenses/LICENSE-2.0 8 * 9 * Unless required by applicable law or agreed to in writing, software distributed under the License 10 * is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express 11 * or implied. See the License for the specific language governing permissions and limitations under 12 * the License. 13 */ 14 15 package android.telecom; 16 17 import android.app.ActivityManager; 18 import android.content.Context; 19 import android.content.Intent; 20 import android.content.pm.ActivityInfo; 21 import android.content.pm.PackageManager; 22 import android.content.pm.ResolveInfo; 23 import android.net.Uri; 24 import android.os.Process; 25 import android.os.UserHandle; 26 import android.provider.Settings; 27 import android.text.TextUtils; 28 29 import java.util.ArrayList; 30 import java.util.List; 31 32 /** 33 * Class for managing the default dialer application that will receive incoming calls, and be 34 * allowed to make emergency outgoing calls. 35 * 36 * @hide 37 */ 38 public class DefaultDialerManager { 39 private static final String TAG = "DefaultDialerManager"; 40 41 /** 42 * Sets the specified package name as the default dialer application for the current user. 43 * The caller of this method needs to have permission to write to secure settings and 44 * manage users on the device. 45 * 46 * @return {@code true} if the default dialer application was successfully changed, 47 * {@code false} otherwise. 48 * 49 * @hide 50 * */ setDefaultDialerApplication(Context context, String packageName)51 public static boolean setDefaultDialerApplication(Context context, String packageName) { 52 return setDefaultDialerApplication(context, packageName, ActivityManager.getCurrentUser()); 53 } 54 55 /** 56 * Sets the specified package name as the default dialer application for the specified user. 57 * The caller of this method needs to have permission to write to secure settings and 58 * manage users on the device. 59 * 60 * @return {@code true} if the default dialer application was successfully changed, 61 * {@code false} otherwise. 62 * 63 * @hide 64 * */ setDefaultDialerApplication(Context context, String packageName, int user)65 public static boolean setDefaultDialerApplication(Context context, String packageName, 66 int user) { 67 // Get old package name 68 String oldPackageName = Settings.Secure.getStringForUser(context.getContentResolver(), 69 Settings.Secure.DIALER_DEFAULT_APPLICATION, user); 70 71 if (packageName != null && oldPackageName != null && packageName.equals(oldPackageName)) { 72 // No change 73 return false; 74 } 75 76 // Only make the change if the new package belongs to a valid phone application 77 List<String> packageNames = getInstalledDialerApplications(context); 78 79 if (packageNames.contains(packageName)) { 80 // Update the secure setting. 81 Settings.Secure.putStringForUser(context.getContentResolver(), 82 Settings.Secure.DIALER_DEFAULT_APPLICATION, packageName, user); 83 return true; 84 } 85 return false; 86 } 87 88 /** 89 * Returns the installed dialer application for the current user that will be used to receive 90 * incoming calls, and is allowed to make emergency calls. 91 * 92 * The application will be returned in order of preference: 93 * 1) User selected phone application (if still installed) 94 * 2) Pre-installed system dialer (if not disabled) 95 * 3) Null 96 * 97 * The caller of this method needs to have permission to manage users on the device. 98 * 99 * @hide 100 * */ getDefaultDialerApplication(Context context)101 public static String getDefaultDialerApplication(Context context) { 102 return getDefaultDialerApplication(context, context.getUserId()); 103 } 104 105 /** 106 * Returns the installed dialer application for the specified user that will be used to receive 107 * incoming calls, and is allowed to make emergency calls. 108 * 109 * The application will be returned in order of preference: 110 * 1) User selected phone application (if still installed) 111 * 2) Pre-installed system dialer (if not disabled) 112 * 3) Null 113 * 114 * The caller of this method needs to have permission to manage users on the device. 115 * 116 * @hide 117 * */ getDefaultDialerApplication(Context context, int user)118 public static String getDefaultDialerApplication(Context context, int user) { 119 String defaultPackageName = Settings.Secure.getStringForUser(context.getContentResolver(), 120 Settings.Secure.DIALER_DEFAULT_APPLICATION, user); 121 122 final List<String> packageNames = getInstalledDialerApplications(context, user); 123 124 // Verify that the default dialer has not been disabled or uninstalled. 125 if (packageNames.contains(defaultPackageName)) { 126 return defaultPackageName; 127 } 128 129 // No user-set dialer found, fallback to system dialer 130 String systemDialerPackageName = getTelecomManager(context).getSystemDialerPackage(); 131 132 if (TextUtils.isEmpty(systemDialerPackageName)) { 133 // No system dialer configured at build time 134 return null; 135 } 136 137 if (packageNames.contains(systemDialerPackageName)) { 138 return systemDialerPackageName; 139 } else { 140 return null; 141 } 142 } 143 144 /** 145 * Returns a list of installed and available dialer applications. 146 * 147 * In order to appear in the list, a dialer application must implement an intent-filter with 148 * the DIAL intent for the following schemes: 149 * 150 * 1) Empty scheme 151 * 2) tel Uri scheme 152 * 153 * @hide 154 **/ getInstalledDialerApplications(Context context, int userId)155 public static List<String> getInstalledDialerApplications(Context context, int userId) { 156 PackageManager packageManager = context.getPackageManager(); 157 158 // Get the list of apps registered for the DIAL intent with empty scheme 159 Intent intent = new Intent(Intent.ACTION_DIAL); 160 List<ResolveInfo> resolveInfoList = 161 packageManager.queryIntentActivitiesAsUser(intent, 0, userId); 162 163 List<String> packageNames = new ArrayList<>(); 164 165 for (ResolveInfo resolveInfo : resolveInfoList) { 166 final ActivityInfo activityInfo = resolveInfo.activityInfo; 167 if (activityInfo != null 168 && !packageNames.contains(activityInfo.packageName) 169 // ignore cross profile intent handler 170 && resolveInfo.targetUserId == UserHandle.USER_CURRENT) { 171 packageNames.add(activityInfo.packageName); 172 } 173 } 174 175 final Intent dialIntentWithTelScheme = new Intent(Intent.ACTION_DIAL); 176 dialIntentWithTelScheme.setData(Uri.fromParts(PhoneAccount.SCHEME_TEL, "", null)); 177 return filterByIntent(context, packageNames, dialIntentWithTelScheme, userId); 178 } 179 getInstalledDialerApplications(Context context)180 public static List<String> getInstalledDialerApplications(Context context) { 181 return getInstalledDialerApplications(context, Process.myUserHandle().getIdentifier()); 182 } 183 184 /** 185 * Determines if the package name belongs to the user-selected default dialer or the preloaded 186 * system dialer, and thus should be allowed to perform certain privileged operations. 187 * 188 * @param context A valid context. 189 * @param packageName of the package to check for. 190 * 191 * @return {@code true} if the provided package name corresponds to the user-selected default 192 * dialer or the preloaded system dialer, {@code false} otherwise. 193 * 194 * @hide 195 */ isDefaultOrSystemDialer(Context context, String packageName)196 public static boolean isDefaultOrSystemDialer(Context context, String packageName) { 197 if (TextUtils.isEmpty(packageName)) { 198 return false; 199 } 200 final TelecomManager tm = getTelecomManager(context); 201 return packageName.equals(tm.getDefaultDialerPackage()) 202 || packageName.equals(tm.getSystemDialerPackage()); 203 } 204 205 /** 206 * Filter a given list of package names for those packages that contain an activity that has 207 * an intent filter for a given intent. 208 * 209 * @param context A valid context 210 * @param packageNames List of package names to filter. 211 * @param userId The UserId 212 * @return The filtered list. 213 */ filterByIntent(Context context, List<String> packageNames, Intent intent, int userId)214 private static List<String> filterByIntent(Context context, List<String> packageNames, 215 Intent intent, int userId) { 216 if (packageNames == null || packageNames.isEmpty()) { 217 return new ArrayList<>(); 218 } 219 220 final List<String> result = new ArrayList<>(); 221 final List<ResolveInfo> resolveInfoList = context.getPackageManager() 222 .queryIntentActivitiesAsUser(intent, 0, userId); 223 final int length = resolveInfoList.size(); 224 for (int i = 0; i < length; i++) { 225 final ActivityInfo info = resolveInfoList.get(i).activityInfo; 226 if (info != null && packageNames.contains(info.packageName) 227 && !result.contains(info.packageName)) { 228 result.add(info.packageName); 229 } 230 } 231 232 return result; 233 } 234 235 getTelecomManager(Context context)236 private static TelecomManager getTelecomManager(Context context) { 237 return (TelecomManager) context.getSystemService(Context.TELECOM_SERVICE); 238 } 239 } 240