Add NLS specific screens for notification listener approval
Fixes: 141689199 Fixes: 143639217 Test: atest Change-Id: I4ead087e0015ad33d6be4f9357de50a4298b3347
This commit is contained in:
@@ -0,0 +1,75 @@
|
||||
/*
|
||||
* Copyright (C) 2019 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.specialaccess.notificationaccess;
|
||||
|
||||
import android.app.Dialog;
|
||||
import android.app.settings.SettingsEnums;
|
||||
import android.content.ComponentName;
|
||||
import android.content.DialogInterface;
|
||||
import android.os.Bundle;
|
||||
|
||||
import androidx.appcompat.app.AlertDialog;
|
||||
import androidx.fragment.app.Fragment;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.core.instrumentation.InstrumentedDialogFragment;
|
||||
|
||||
public class FriendlyWarningDialogFragment extends InstrumentedDialogFragment {
|
||||
static final String KEY_COMPONENT = "c";
|
||||
static final String KEY_LABEL = "l";
|
||||
|
||||
public FriendlyWarningDialogFragment setServiceInfo(ComponentName cn, CharSequence label,
|
||||
Fragment target) {
|
||||
Bundle args = new Bundle();
|
||||
args.putString(KEY_COMPONENT, cn.flattenToString());
|
||||
args.putCharSequence(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 CharSequence label = args.getCharSequence(KEY_LABEL);
|
||||
final ComponentName cn = ComponentName.unflattenFromString(args
|
||||
.getString(KEY_COMPONENT));
|
||||
NotificationAccessDetails parent = (NotificationAccessDetails) 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,
|
||||
new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
parent.disable(cn);
|
||||
}
|
||||
})
|
||||
.setNegativeButton(R.string.notification_listener_disable_warning_cancel,
|
||||
(dialog, id) -> {
|
||||
// pass
|
||||
})
|
||||
.create();
|
||||
}
|
||||
}
|
@@ -17,6 +17,8 @@
|
||||
package com.android.settings.applications.specialaccess.notificationaccess;
|
||||
|
||||
import android.app.ActivityManager;
|
||||
import android.app.NotificationManager;
|
||||
import android.content.ComponentName;
|
||||
import android.content.Context;
|
||||
|
||||
import com.android.settings.core.BasePreferenceController;
|
||||
@@ -33,4 +35,9 @@ public class NotificationAccessController extends BasePreferenceController {
|
||||
? AVAILABLE_UNSEARCHABLE
|
||||
: UNSUPPORTED_ON_DEVICE;
|
||||
}
|
||||
|
||||
public static boolean hasAccess(Context context, ComponentName cn) {
|
||||
return context.getSystemService(NotificationManager.class)
|
||||
.isNotificationListenerAccessGranted(cn);
|
||||
}
|
||||
}
|
||||
|
@@ -0,0 +1,226 @@
|
||||
/*
|
||||
* Copyright (C) 2019 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.specialaccess.notificationaccess;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.ActivityManager;
|
||||
import android.app.NotificationManager;
|
||||
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.content.pm.ServiceInfo;
|
||||
import android.os.AsyncTask;
|
||||
import android.os.Bundle;
|
||||
import android.os.UserManager;
|
||||
import android.provider.Settings;
|
||||
import android.service.notification.NotificationListenerService;
|
||||
import android.util.IconDrawableFactory;
|
||||
import android.util.Log;
|
||||
import android.util.Slog;
|
||||
|
||||
import androidx.annotation.VisibleForTesting;
|
||||
import androidx.appcompat.app.AlertDialog;
|
||||
import androidx.preference.Preference;
|
||||
import androidx.preference.SwitchPreference;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.applications.AppInfoBase;
|
||||
import com.android.settings.overlay.FeatureFactory;
|
||||
import com.android.settings.widget.EntityHeaderController;
|
||||
import com.android.settingslib.applications.AppUtils;
|
||||
|
||||
import java.util.List;
|
||||
import java.util.Objects;
|
||||
|
||||
public class NotificationAccessDetails extends AppInfoBase {
|
||||
private static final String TAG = "NotifAccessDetails";
|
||||
private static final String SWITCH_PREF_KEY = "notification_access_switch";
|
||||
|
||||
private boolean mCreated;
|
||||
private ComponentName mComponentName;
|
||||
private CharSequence mServiceName;
|
||||
private boolean mIsNls;
|
||||
|
||||
private NotificationManager mNm;
|
||||
private PackageManager mPm;
|
||||
|
||||
@Override
|
||||
public void onCreate(Bundle savedInstanceState) {
|
||||
final Intent intent = getIntent();
|
||||
if (mComponentName == null && intent != null) {
|
||||
String cn = intent.getStringExtra(Settings.EXTRA_NOTIFICATION_LISTENER_COMPONENT_NAME);
|
||||
if (cn != null) {
|
||||
mComponentName = ComponentName.unflattenFromString(cn);
|
||||
if (mComponentName != null) {
|
||||
final Bundle args = getArguments();
|
||||
args.putString(ARG_PACKAGE_NAME, mComponentName.getPackageName());
|
||||
}
|
||||
}
|
||||
}
|
||||
super.onCreate(savedInstanceState);
|
||||
mNm = getContext().getSystemService(NotificationManager.class);
|
||||
mPm = getPackageManager();
|
||||
addPreferencesFromResource(R.xml.notification_access_permission_details);
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onActivityCreated(Bundle savedInstanceState) {
|
||||
super.onActivityCreated(savedInstanceState);
|
||||
if (mCreated) {
|
||||
Log.w(TAG, "onActivityCreated: ignoring duplicate call");
|
||||
return;
|
||||
}
|
||||
mCreated = true;
|
||||
if (mPackageInfo == null) return;
|
||||
loadNotificationListenerService();
|
||||
final Activity activity = getActivity();
|
||||
final Preference pref = EntityHeaderController
|
||||
.newInstance(activity, this, null /* header */)
|
||||
.setRecyclerView(getListView(), getSettingsLifecycle())
|
||||
.setIcon(IconDrawableFactory.newInstance(getContext())
|
||||
.getBadgedIcon(mPackageInfo.applicationInfo))
|
||||
.setLabel(mPackageInfo.applicationInfo.loadLabel(mPm))
|
||||
.setSummary(mServiceName)
|
||||
.setIsInstantApp(AppUtils.isInstant(mPackageInfo.applicationInfo))
|
||||
.setPackageName(mPackageName)
|
||||
.setUid(mPackageInfo.applicationInfo.uid)
|
||||
.setHasAppInfoLink(true)
|
||||
.setButtonActions(EntityHeaderController.ActionType.ACTION_NONE,
|
||||
EntityHeaderController.ActionType.ACTION_NONE)
|
||||
.done(activity, getPrefContext());
|
||||
getPreferenceScreen().addPreference(pref);
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getMetricsCategory() {
|
||||
return SettingsEnums.NOTIFICATION_ACCESS_DETAIL;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected boolean refreshUi() {
|
||||
final Context context = getContext();
|
||||
if (context.getSystemService(ActivityManager.class).isLowRamDeviceStatic()) {
|
||||
Slog.d(TAG, "not available on low ram devices");
|
||||
return false;
|
||||
}
|
||||
if (mComponentName == null) {
|
||||
// No service given
|
||||
Slog.d(TAG, "No component name provided");
|
||||
return false;
|
||||
}
|
||||
if (!mIsNls) {
|
||||
// This component doesn't have the right androidmanifest definition to be an NLS
|
||||
Slog.d(TAG, "Provided component name is not an NLS");
|
||||
return false;
|
||||
}
|
||||
if (UserManager.get(getContext()).isManagedProfile()) {
|
||||
// Apps in the work profile do not support notification listeners.
|
||||
Slog.d(TAG, "NLSes aren't allowed in work profiles");
|
||||
return false;
|
||||
}
|
||||
updatePreference(findPreference(SWITCH_PREF_KEY));
|
||||
return true;
|
||||
}
|
||||
|
||||
@Override
|
||||
protected AlertDialog createDialog(int id, int errorCode) {
|
||||
return null;
|
||||
}
|
||||
|
||||
public void updatePreference(SwitchPreference preference) {
|
||||
final CharSequence label = mPackageInfo.applicationInfo.loadLabel(mPm);
|
||||
preference.setChecked(isServiceEnabled(mComponentName));
|
||||
preference.setOnPreferenceChangeListener((p, newValue) -> {
|
||||
final boolean access = (Boolean) newValue;
|
||||
if (!access) {
|
||||
if (!isServiceEnabled(mComponentName)) {
|
||||
return true; // already disabled
|
||||
}
|
||||
// show a friendly dialog
|
||||
new FriendlyWarningDialogFragment()
|
||||
.setServiceInfo(mComponentName, label, this)
|
||||
.show(getFragmentManager(), "friendlydialog");
|
||||
return false;
|
||||
} else {
|
||||
if (isServiceEnabled(mComponentName)) {
|
||||
return true; // already enabled
|
||||
}
|
||||
// show a scary dialog
|
||||
new ScaryWarningDialogFragment()
|
||||
.setServiceInfo(mComponentName, label, this)
|
||||
.show(getFragmentManager(), "dialog");
|
||||
return false;
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@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);
|
||||
}
|
||||
|
||||
public void disable(final ComponentName cn) {
|
||||
logSpecialPermissionChange(true, cn.getPackageName());
|
||||
mNm.setNotificationListenerAccessGranted(cn, false);
|
||||
AsyncTask.execute(() -> {
|
||||
if (!mNm.isNotificationPolicyAccessGrantedForPackage(
|
||||
cn.getPackageName())) {
|
||||
mNm.removeAutomaticZenRules(cn.getPackageName());
|
||||
}
|
||||
});
|
||||
refreshUi();
|
||||
}
|
||||
|
||||
protected void enable(ComponentName cn) {
|
||||
logSpecialPermissionChange(true, cn.getPackageName());
|
||||
mNm.setNotificationListenerAccessGranted(cn, true);
|
||||
refreshUi();
|
||||
}
|
||||
|
||||
protected boolean isServiceEnabled(ComponentName cn) {
|
||||
return mNm.isNotificationListenerAccessGranted(cn);
|
||||
}
|
||||
|
||||
protected void loadNotificationListenerService() {
|
||||
mIsNls = false;
|
||||
|
||||
if (mComponentName == null) {
|
||||
return;
|
||||
}
|
||||
Intent intent = new Intent(NotificationListenerService.SERVICE_INTERFACE)
|
||||
.setComponent(mComponentName);
|
||||
List<ResolveInfo> installedServices = mPm.queryIntentServicesAsUser(
|
||||
intent, PackageManager.GET_SERVICES | PackageManager.GET_META_DATA, mUserId);
|
||||
for (ResolveInfo resolveInfo : installedServices) {
|
||||
ServiceInfo info = resolveInfo.serviceInfo;
|
||||
if (android.Manifest.permission.BIND_NOTIFICATION_LISTENER_SERVICE.equals(
|
||||
info.permission)) {
|
||||
if (Objects.equals(mComponentName, info.getComponentName())) {
|
||||
mIsNls = true;
|
||||
mServiceName = info.loadLabel(mPm);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
@@ -0,0 +1,79 @@
|
||||
/*
|
||||
* Copyright (C) 2019 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.specialaccess.notificationaccess;
|
||||
|
||||
import android.app.Dialog;
|
||||
import android.app.settings.SettingsEnums;
|
||||
import android.content.ComponentName;
|
||||
import android.content.DialogInterface;
|
||||
import android.os.Bundle;
|
||||
|
||||
import androidx.appcompat.app.AlertDialog;
|
||||
import androidx.fragment.app.Fragment;
|
||||
|
||||
import com.android.settings.R;
|
||||
import com.android.settings.core.instrumentation.InstrumentedDialogFragment;
|
||||
|
||||
|
||||
public class ScaryWarningDialogFragment extends InstrumentedDialogFragment {
|
||||
private static final String KEY_COMPONENT = "c";
|
||||
private static final String KEY_LABEL = "l";
|
||||
|
||||
@Override
|
||||
public int getMetricsCategory() {
|
||||
return SettingsEnums.DIALOG_SERVICE_ACCESS_WARNING;
|
||||
}
|
||||
|
||||
public ScaryWarningDialogFragment setServiceInfo(ComponentName cn, CharSequence label,
|
||||
Fragment target) {
|
||||
Bundle args = new Bundle();
|
||||
args.putString(KEY_COMPONENT, cn.flattenToString());
|
||||
args.putCharSequence(KEY_LABEL, label);
|
||||
setArguments(args);
|
||||
setTargetFragment(target, 0);
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public Dialog onCreateDialog(Bundle savedInstanceState) {
|
||||
final Bundle args = getArguments();
|
||||
final CharSequence label = args.getCharSequence(KEY_LABEL);
|
||||
final ComponentName cn = ComponentName.unflattenFromString(args
|
||||
.getString(KEY_COMPONENT));
|
||||
NotificationAccessDetails parent = (NotificationAccessDetails) getTargetFragment();
|
||||
|
||||
final String title = getResources().getString(
|
||||
R.string.notification_listener_security_warning_title, label);
|
||||
final String summary = getResources().getString(
|
||||
R.string.notification_listener_security_warning_summary, label);
|
||||
return new AlertDialog.Builder(getContext())
|
||||
.setMessage(summary)
|
||||
.setTitle(title)
|
||||
.setCancelable(true)
|
||||
.setPositiveButton(R.string.allow,
|
||||
new DialogInterface.OnClickListener() {
|
||||
@Override
|
||||
public void onClick(DialogInterface dialog, int which) {
|
||||
parent.enable(cn);
|
||||
}
|
||||
})
|
||||
.setNegativeButton(R.string.deny,
|
||||
(dialog, id) -> {
|
||||
// pass
|
||||
})
|
||||
.create();
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user