/* * Copyright (C) 2010 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.notification; import android.app.Dialog; import android.app.NotificationManager; import android.app.settings.SettingsEnums; import android.content.ComponentName; import android.content.Context; import android.os.AsyncTask; import android.os.Bundle; import android.os.UserManager; import android.provider.SearchIndexableResource; import android.provider.Settings; import android.service.notification.NotificationListenerService; import android.widget.Toast; import androidx.annotation.VisibleForTesting; import androidx.appcompat.app.AlertDialog; import androidx.fragment.app.Fragment; import com.android.settings.R; import com.android.settings.core.instrumentation.InstrumentedDialogFragment; import com.android.settings.overlay.FeatureFactory; import com.android.settings.search.BaseSearchIndexProvider; import com.android.settings.search.Indexable; import com.android.settings.utils.ManagedServiceSettings; import com.android.settingslib.search.SearchIndexable; import java.util.ArrayList; import java.util.List; /** * Settings screen for managing notification listener permissions */ @SearchIndexable public class NotificationAccessSettings extends ManagedServiceSettings { private static final String TAG = "NotificationAccessSettings"; private static final Config CONFIG = new Config.Builder() .setTag(TAG) .setSetting(Settings.Secure.ENABLED_NOTIFICATION_LISTENERS) .setIntentAction(NotificationListenerService.SERVICE_INTERFACE) .setPermission(android.Manifest.permission.BIND_NOTIFICATION_LISTENER_SERVICE) .setNoun("notification listener") .setWarningDialogTitle(R.string.notification_listener_security_warning_title) .setWarningDialogSummary(R.string.notification_listener_security_warning_summary) .setEmptyText(R.string.no_notification_listeners) .build(); private NotificationManager mNm; @Override public void onCreate(Bundle icicle) { super.onCreate(icicle); final Context ctx = getContext(); if (UserManager.get(ctx).isManagedProfile()) { // Apps in the work profile do not support notification listeners. Toast.makeText(ctx, R.string.notification_settings_work_profile, Toast.LENGTH_SHORT) .show(); finish(); } } @Override public int getMetricsCategory() { return SettingsEnums.NOTIFICATION_ACCESS; } @Override public void onAttach(Context context) { super.onAttach(context); mNm = context.getSystemService(NotificationManager.class); } @Override protected Config getConfig() { return CONFIG; } @Override protected boolean setEnabled(ComponentName service, String title, boolean enable) { logSpecialPermissionChange(enable, service.getPackageName()); if (!enable) { if (!isServiceEnabled(service)) { return true; // already disabled } // show a friendly dialog new FriendlyWarningDialogFragment() .setServiceInfo(service, title, this) .show(getFragmentManager(), "friendlydialog"); return false; } else { if (isServiceEnabled(service)) { return true; // already enabled } // show a scary dialog new ScaryWarningDialogFragment() .setServiceInfo(service, title, this) .show(getFragmentManager(), "dialog"); return false; } } @Override protected boolean isServiceEnabled(ComponentName cn) { return mNm.isNotificationListenerAccessGranted(cn); } @Override protected void enable(ComponentName service) { mNm.setNotificationListenerAccessGranted(service, true); } @Override protected int getPreferenceScreenResId() { return R.xml.notification_access_settings; } @VisibleForTesting void logSpecialPermissionChange(boolean enable, String packageName) { int logCategory = enable ? SettingsEnums.APP_SPECIAL_PERMISSION_NOTIVIEW_ALLOW : SettingsEnums.APP_SPECIAL_PERMISSION_NOTIVIEW_DENY; FeatureFactory.getFactory(getContext()).getMetricsFeatureProvider().action(getContext(), logCategory, packageName); } private static void disable(final NotificationAccessSettings parent, final ComponentName cn) { parent.mNm.setNotificationListenerAccessGranted(cn, false); AsyncTask.execute(() -> { if (!parent.mNm.isNotificationPolicyAccessGrantedForPackage( cn.getPackageName())) { parent.mNm.removeAutomaticZenRules(cn.getPackageName()); } }); } public static class FriendlyWarningDialogFragment extends InstrumentedDialogFragment { static final String KEY_COMPONENT = "c"; static final String KEY_LABEL = "l"; public FriendlyWarningDialogFragment setServiceInfo(ComponentName cn, String label, Fragment target) { Bundle args = new Bundle(); args.putString(KEY_COMPONENT, cn.flattenToString()); args.putString(KEY_LABEL, label); setArguments(args); setTargetFragment(target, 0); return this; } @Override public int getMetricsCategory() { return SettingsEnums.DIALOG_DISABLE_NOTIFICATION_ACCESS; } @Override public Dialog onCreateDialog(Bundle savedInstanceState) { final Bundle args = getArguments(); final String label = args.getString(KEY_LABEL); final ComponentName cn = ComponentName.unflattenFromString(args .getString(KEY_COMPONENT)); NotificationAccessSettings parent = (NotificationAccessSettings) getTargetFragment(); final String summary = getResources().getString( R.string.notification_listener_disable_warning_summary, label); return new AlertDialog.Builder(getContext()) .setMessage(summary) .setCancelable(true) .setPositiveButton(R.string.notification_listener_disable_warning_confirm, (dialog, id) -> disable(parent, cn)) .setNegativeButton(R.string.notification_listener_disable_warning_cancel, (dialog, id) -> { // pass }) .create(); } } public static final Indexable.SearchIndexProvider SEARCH_INDEX_DATA_PROVIDER = new BaseSearchIndexProvider() { @Override public List getXmlResourcesToIndex(Context context, boolean enabled) { final List result = new ArrayList<>(); final SearchIndexableResource sir = new SearchIndexableResource(context); sir.xmlResId = R.xml.notification_access_settings; result.add(sir); return result; } }; }