| /* |
| * 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 com.android.settings.applications; |
| |
| import static android.app.AppOpsManager.OP_GET_USAGE_STATS; |
| import static android.app.AppOpsManager.OP_LOADER_USAGE_STATS; |
| import static android.app.admin.DevicePolicyResources.Strings.Settings.WORK_PROFILE_DISABLE_USAGE_ACCESS_WARNING; |
| |
| import android.Manifest; |
| import android.app.AppOpsManager; |
| import android.app.admin.DevicePolicyManager; |
| import android.app.settings.SettingsEnums; |
| import android.content.ComponentName; |
| import android.content.Context; |
| import android.content.Intent; |
| import android.content.pm.PackageManager; |
| import android.content.pm.ResolveInfo; |
| import android.os.Bundle; |
| import android.provider.Settings; |
| |
| import androidx.annotation.VisibleForTesting; |
| import androidx.appcompat.app.AlertDialog; |
| import androidx.preference.Preference; |
| import androidx.preference.Preference.OnPreferenceChangeListener; |
| import androidx.preference.Preference.OnPreferenceClickListener; |
| import androidx.preference.SwitchPreference; |
| |
| import com.android.settings.R; |
| import com.android.settings.applications.AppStateUsageBridge.UsageState; |
| import com.android.settings.overlay.FeatureFactory; |
| import com.android.settingslib.core.instrumentation.MetricsFeatureProvider; |
| |
| public class UsageAccessDetails extends AppInfoWithHeader implements OnPreferenceChangeListener, |
| OnPreferenceClickListener { |
| |
| private static final String KEY_APP_OPS_PREFERENCE_SCREEN = "app_ops_preference_screen"; |
| private static final String KEY_APP_OPS_SETTINGS_SWITCH = "app_ops_settings_switch"; |
| private static final String KEY_APP_OPS_SETTINGS_DESC = "app_ops_settings_description"; |
| |
| // Use a bridge to get the usage stats but don't initialize it to connect with all state. |
| // TODO: Break out this functionality into its own class. |
| private AppStateUsageBridge mUsageBridge; |
| private AppOpsManager mAppOpsManager; |
| private SwitchPreference mSwitchPref; |
| private Preference mUsageDesc; |
| private Intent mSettingsIntent; |
| private UsageState mUsageState; |
| private DevicePolicyManager mDpm; |
| |
| @Override |
| public void onCreate(Bundle savedInstanceState) { |
| super.onCreate(savedInstanceState); |
| |
| Context context = getActivity(); |
| mUsageBridge = new AppStateUsageBridge(context, mState, null); |
| mAppOpsManager = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE); |
| mDpm = context.getSystemService(DevicePolicyManager.class); |
| |
| addPreferencesFromResource(R.xml.app_ops_permissions_details); |
| mSwitchPref = (SwitchPreference) findPreference(KEY_APP_OPS_SETTINGS_SWITCH); |
| mUsageDesc = findPreference(KEY_APP_OPS_SETTINGS_DESC); |
| |
| getPreferenceScreen().setTitle(R.string.usage_access); |
| mSwitchPref.setTitle(R.string.permit_usage_access); |
| mUsageDesc.setTitle(R.string.usage_access_description); |
| |
| mSwitchPref.setOnPreferenceChangeListener(this); |
| |
| mSettingsIntent = new Intent(Intent.ACTION_MAIN) |
| .addCategory(Settings.INTENT_CATEGORY_USAGE_ACCESS_CONFIG) |
| .setPackage(mPackageName); |
| } |
| |
| @Override |
| public boolean onPreferenceClick(Preference preference) { |
| return false; |
| } |
| |
| @Override |
| public boolean onPreferenceChange(Preference preference, Object newValue) { |
| if (preference == mSwitchPref) { |
| if (mUsageState != null && (Boolean) newValue != mUsageState.isPermissible()) { |
| if (mUsageState.isPermissible() && mDpm.isProfileOwnerApp(mPackageName)) { |
| new AlertDialog.Builder(getContext()) |
| .setIcon(com.android.internal.R.drawable.ic_dialog_alert_material) |
| .setTitle(android.R.string.dialog_alert_title) |
| .setMessage(mDpm.getResources().getString( |
| WORK_PROFILE_DISABLE_USAGE_ACCESS_WARNING, |
| () -> getString(R.string.work_profile_usage_access_warning))) |
| .setPositiveButton(R.string.okay, null) |
| .show(); |
| } |
| setHasAccess(!mUsageState.isPermissible()); |
| refreshUi(); |
| } |
| return true; |
| } |
| return false; |
| } |
| |
| private static boolean doesAnyPermissionMatch(String permissionToMatch, String[] permissions) { |
| for (String permission : permissions) { |
| if (permissionToMatch.equals(permission)) { |
| return true; |
| } |
| } |
| return false; |
| } |
| |
| private void setHasAccess(boolean newState) { |
| logSpecialPermissionChange(newState, mPackageName); |
| |
| final int newAppOpMode = newState ? AppOpsManager.MODE_ALLOWED : AppOpsManager.MODE_IGNORED; |
| final int uid = mPackageInfo.applicationInfo.uid; |
| if (doesAnyPermissionMatch(Manifest.permission.PACKAGE_USAGE_STATS, |
| mUsageState.packageInfo.requestedPermissions)) { |
| mAppOpsManager.setMode(OP_GET_USAGE_STATS, uid, mPackageName, newAppOpMode); |
| } |
| if (doesAnyPermissionMatch(Manifest.permission.LOADER_USAGE_STATS, |
| mUsageState.packageInfo.requestedPermissions)) { |
| mAppOpsManager.setMode(OP_LOADER_USAGE_STATS, uid, mPackageName, newAppOpMode); |
| } |
| } |
| |
| @VisibleForTesting |
| void logSpecialPermissionChange(boolean newState, String packageName) { |
| int logCategory = newState ? SettingsEnums.APP_SPECIAL_PERMISSION_USAGE_VIEW_ALLOW |
| : SettingsEnums.APP_SPECIAL_PERMISSION_USAGE_VIEW_DENY; |
| final MetricsFeatureProvider metricsFeatureProvider = |
| FeatureFactory.getFactory(getContext()).getMetricsFeatureProvider(); |
| metricsFeatureProvider.action( |
| metricsFeatureProvider.getAttribution(getActivity()), |
| logCategory, |
| getMetricsCategory(), |
| packageName, |
| 0); |
| } |
| |
| @Override |
| protected boolean refreshUi() { |
| retrieveAppEntry(); |
| if (mAppEntry == null) { |
| return false; |
| } |
| if (mPackageInfo == null) { |
| return false; // onCreate must have failed, make sure to exit |
| } |
| mUsageState = mUsageBridge.getUsageInfo(mPackageName, |
| mPackageInfo.applicationInfo.uid); |
| |
| boolean hasAccess = mUsageState.isPermissible(); |
| mSwitchPref.setChecked(hasAccess); |
| mSwitchPref.setEnabled(mUsageState.permissionDeclared); |
| |
| ResolveInfo resolveInfo = mPm.resolveActivityAsUser(mSettingsIntent, |
| PackageManager.GET_META_DATA, mUserId); |
| if (resolveInfo != null) { |
| Bundle metaData = resolveInfo.activityInfo.metaData; |
| mSettingsIntent.setComponent(new ComponentName(resolveInfo.activityInfo.packageName, |
| resolveInfo.activityInfo.name)); |
| if (metaData != null |
| && metaData.containsKey(Settings.METADATA_USAGE_ACCESS_REASON)) { |
| mSwitchPref.setSummary( |
| metaData.getString(Settings.METADATA_USAGE_ACCESS_REASON)); |
| } |
| } |
| |
| return true; |
| } |
| |
| @Override |
| protected AlertDialog createDialog(int id, int errorCode) { |
| return null; |
| } |
| |
| @Override |
| public int getMetricsCategory() { |
| return SettingsEnums.APPLICATIONS_USAGE_ACCESS_DETAIL; |
| } |
| |
| } |