- Keep launching notification settings in the Settings task. This is not the expected behavior, but only a workaround until b/72420153 is fixed. Bug: 72420153 Test: Enter PIP, go to app > notifications > additional settings, ensure that it doesn't start in the PIP task Change-Id: I73e704a283285462d4884db21923818cfb6deead
430 lines
16 KiB
Java
430 lines
16 KiB
Java
/*
|
|
* Copyright (C) 2016 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 static android.app.NotificationManager.IMPORTANCE_LOW;
|
|
import static android.app.NotificationManager.IMPORTANCE_NONE;
|
|
import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
|
|
|
|
import android.app.Activity;
|
|
import android.app.Notification;
|
|
import android.app.NotificationChannel;
|
|
import android.app.NotificationChannelGroup;
|
|
import android.app.NotificationManager;
|
|
import android.content.BroadcastReceiver;
|
|
import android.content.Context;
|
|
import android.content.Intent;
|
|
import android.content.IntentFilter;
|
|
import android.content.pm.ActivityInfo;
|
|
import android.content.pm.PackageInfo;
|
|
import android.content.pm.PackageManager;
|
|
import android.content.pm.PackageManager.NameNotFoundException;
|
|
import android.content.pm.ResolveInfo;
|
|
import android.os.Bundle;
|
|
import android.os.UserHandle;
|
|
import android.provider.Settings;
|
|
import android.support.v7.preference.Preference;
|
|
import android.support.v7.preference.PreferenceGroup;
|
|
import android.support.v7.preference.PreferenceScreen;
|
|
import android.text.TextUtils;
|
|
import android.util.Log;
|
|
import android.widget.Toast;
|
|
|
|
import com.android.settings.R;
|
|
import com.android.settings.SettingsActivity;
|
|
import com.android.settings.applications.AppInfoBase;
|
|
import com.android.settings.core.SubSettingLauncher;
|
|
import com.android.settings.dashboard.DashboardFragment;
|
|
import com.android.settings.widget.MasterCheckBoxPreference;
|
|
import com.android.settingslib.RestrictedLockUtils;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.Comparator;
|
|
import java.util.List;
|
|
|
|
abstract public class NotificationSettingsBase extends DashboardFragment {
|
|
private static final String TAG = "NotifiSettingsBase";
|
|
private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
|
|
protected static final String ARG_FROM_SETTINGS = "fromSettings";
|
|
|
|
protected PackageManager mPm;
|
|
protected NotificationBackend mBackend = new NotificationBackend();
|
|
protected NotificationManager mNm;
|
|
protected Context mContext;
|
|
|
|
protected int mUid;
|
|
protected int mUserId;
|
|
protected String mPkg;
|
|
protected PackageInfo mPkgInfo;
|
|
protected EnforcedAdmin mSuspendedAppsAdmin;
|
|
protected NotificationChannelGroup mChannelGroup;
|
|
protected NotificationChannel mChannel;
|
|
protected NotificationBackend.AppRow mAppRow;
|
|
|
|
protected boolean mShowLegacyChannelConfig = false;
|
|
protected boolean mListeningToPackageRemove;
|
|
|
|
protected List<NotificationPreferenceController> mControllers = new ArrayList<>();
|
|
protected List<Preference> mDynamicPreferences = new ArrayList<>();
|
|
protected ImportanceListener mImportanceListener = new ImportanceListener();
|
|
|
|
protected Intent mIntent;
|
|
protected Bundle mArgs;
|
|
|
|
@Override
|
|
public void onAttach(Context context) {
|
|
super.onAttach(context);
|
|
mContext = getActivity();
|
|
mIntent = getActivity().getIntent();
|
|
mArgs = getArguments();
|
|
|
|
mPm = getPackageManager();
|
|
mNm = NotificationManager.from(mContext);
|
|
|
|
mPkg = mArgs != null && mArgs.containsKey(AppInfoBase.ARG_PACKAGE_NAME)
|
|
? mArgs.getString(AppInfoBase.ARG_PACKAGE_NAME)
|
|
: mIntent.getStringExtra(Settings.EXTRA_APP_PACKAGE);
|
|
mUid = mArgs != null && mArgs.containsKey(AppInfoBase.ARG_PACKAGE_UID)
|
|
? mArgs.getInt(AppInfoBase.ARG_PACKAGE_UID)
|
|
: mIntent.getIntExtra(Settings.EXTRA_APP_UID, -1);
|
|
|
|
if (mUid < 0) {
|
|
try {
|
|
mUid = mPm.getPackageUid(mPkg, 0);
|
|
} catch (NameNotFoundException e) {
|
|
}
|
|
}
|
|
|
|
mPkgInfo = findPackageInfo(mPkg, mUid);
|
|
|
|
mUserId = UserHandle.getUserId(mUid);
|
|
mSuspendedAppsAdmin = RestrictedLockUtils.checkIfApplicationIsSuspended(
|
|
mContext, mPkg, mUserId);
|
|
|
|
loadChannel();
|
|
loadAppRow();
|
|
loadChannelGroup();
|
|
collectConfigActivities();
|
|
|
|
getLifecycle().addObserver(use(HeaderPreferenceController.class));
|
|
|
|
for (NotificationPreferenceController controller : mControllers) {
|
|
controller.onResume(mAppRow, mChannel, mChannelGroup, mSuspendedAppsAdmin);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void onCreate(Bundle savedInstanceState) {
|
|
super.onCreate(savedInstanceState);
|
|
|
|
if (mIntent == null && mArgs == null) {
|
|
Log.w(TAG, "No intent");
|
|
toastAndFinish();
|
|
return;
|
|
}
|
|
|
|
if (mUid < 0 || TextUtils.isEmpty(mPkg) || mPkgInfo == null) {
|
|
Log.w(TAG, "Missing package or uid or packageinfo");
|
|
toastAndFinish();
|
|
return;
|
|
}
|
|
|
|
startListeningToPackageRemove();
|
|
}
|
|
|
|
@Override
|
|
public void onDestroy() {
|
|
stopListeningToPackageRemove();
|
|
super.onDestroy();
|
|
}
|
|
|
|
@Override
|
|
public void onResume() {
|
|
super.onResume();
|
|
if (mUid < 0 || TextUtils.isEmpty(mPkg) || mPkgInfo == null || mAppRow == null) {
|
|
Log.w(TAG, "Missing package or uid or packageinfo");
|
|
finish();
|
|
return;
|
|
}
|
|
// Reload app, channel, etc onResume in case they've changed. A little wasteful if we've
|
|
// just done onAttach but better than making every preference controller reload all
|
|
// the data
|
|
loadAppRow();
|
|
if (mAppRow == null) {
|
|
Log.w(TAG, "Can't load package");
|
|
finish();
|
|
return;
|
|
}
|
|
loadChannel();
|
|
loadChannelGroup();
|
|
collectConfigActivities();
|
|
}
|
|
|
|
private void loadChannel() {
|
|
Intent intent = getActivity().getIntent();
|
|
String channelId = intent != null ? intent.getStringExtra(Settings.EXTRA_CHANNEL_ID) : null;
|
|
if (channelId == null && intent != null) {
|
|
Bundle args = intent.getBundleExtra(SettingsActivity.EXTRA_SHOW_FRAGMENT_ARGUMENTS);
|
|
channelId = args != null ? args.getString(Settings.EXTRA_CHANNEL_ID) : null;
|
|
}
|
|
mChannel = mBackend.getChannel(mPkg, mUid, channelId);
|
|
}
|
|
|
|
private void loadAppRow() {
|
|
mAppRow = mBackend.loadAppRow(mContext, mPm, mPkgInfo);
|
|
}
|
|
|
|
private void loadChannelGroup() {
|
|
mShowLegacyChannelConfig = mBackend.onlyHasDefaultChannel(mAppRow.pkg, mAppRow.uid)
|
|
|| (mChannel != null
|
|
&& NotificationChannel.DEFAULT_CHANNEL_ID.equals(mChannel.getId()));
|
|
|
|
if (mShowLegacyChannelConfig) {
|
|
mChannel = mBackend.getChannel(
|
|
mAppRow.pkg, mAppRow.uid, NotificationChannel.DEFAULT_CHANNEL_ID);
|
|
}
|
|
if (mChannel != null && !TextUtils.isEmpty(mChannel.getGroup())) {
|
|
NotificationChannelGroup group = mBackend.getGroup(mPkg, mUid, mChannel.getGroup());
|
|
if (group != null) {
|
|
mChannelGroup = group;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected void toastAndFinish() {
|
|
Toast.makeText(mContext, R.string.app_not_found_dlg_text, Toast.LENGTH_SHORT).show();
|
|
getActivity().finish();
|
|
}
|
|
|
|
protected void collectConfigActivities() {
|
|
Intent intent = new Intent(Intent.ACTION_MAIN)
|
|
.addCategory(Notification.INTENT_CATEGORY_NOTIFICATION_PREFERENCES)
|
|
.setPackage(mAppRow.pkg);
|
|
final List<ResolveInfo> resolveInfos = mPm.queryIntentActivities(
|
|
intent,
|
|
0 //PackageManager.MATCH_DEFAULT_ONLY
|
|
);
|
|
if (DEBUG) {
|
|
Log.d(TAG, "Found " + resolveInfos.size() + " preference activities"
|
|
+ (resolveInfos.size() == 0 ? " ;_;" : ""));
|
|
}
|
|
for (ResolveInfo ri : resolveInfos) {
|
|
final ActivityInfo activityInfo = ri.activityInfo;
|
|
if (mAppRow.settingsIntent != null) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "Ignoring duplicate notification preference activity ("
|
|
+ activityInfo.name + ") for package "
|
|
+ activityInfo.packageName);
|
|
}
|
|
continue;
|
|
}
|
|
// TODO(78660939): This should actually start a new task
|
|
mAppRow.settingsIntent = intent
|
|
.setPackage(null)
|
|
.setClassName(activityInfo.packageName, activityInfo.name);
|
|
if (mChannel != null) {
|
|
mAppRow.settingsIntent.putExtra(Notification.EXTRA_CHANNEL_ID, mChannel.getId());
|
|
}
|
|
if (mChannelGroup != null) {
|
|
mAppRow.settingsIntent.putExtra(
|
|
Notification.EXTRA_CHANNEL_GROUP_ID, mChannelGroup.getId());
|
|
}
|
|
}
|
|
}
|
|
|
|
private PackageInfo findPackageInfo(String pkg, int uid) {
|
|
if (pkg == null || uid < 0) {
|
|
return null;
|
|
}
|
|
final String[] packages = mPm.getPackagesForUid(uid);
|
|
if (packages != null && pkg != null) {
|
|
final int N = packages.length;
|
|
for (int i = 0; i < N; i++) {
|
|
final String p = packages[i];
|
|
if (pkg.equals(p)) {
|
|
try {
|
|
return mPm.getPackageInfo(pkg, PackageManager.GET_SIGNATURES);
|
|
} catch (NameNotFoundException e) {
|
|
Log.w(TAG, "Failed to load package " + pkg, e);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
protected Preference populateSingleChannelPrefs(PreferenceGroup parent,
|
|
final NotificationChannel channel, final boolean groupBlocked) {
|
|
MasterCheckBoxPreference channelPref = new MasterCheckBoxPreference(
|
|
getPrefContext());
|
|
channelPref.setCheckBoxEnabled(mSuspendedAppsAdmin == null
|
|
&& isChannelBlockable(channel)
|
|
&& isChannelConfigurable(channel)
|
|
&& !groupBlocked);
|
|
channelPref.setKey(channel.getId());
|
|
channelPref.setTitle(channel.getName());
|
|
channelPref.setChecked(channel.getImportance() != IMPORTANCE_NONE);
|
|
Bundle channelArgs = new Bundle();
|
|
channelArgs.putInt(AppInfoBase.ARG_PACKAGE_UID, mUid);
|
|
channelArgs.putString(AppInfoBase.ARG_PACKAGE_NAME, mPkg);
|
|
channelArgs.putString(Settings.EXTRA_CHANNEL_ID, channel.getId());
|
|
channelArgs.putBoolean(ARG_FROM_SETTINGS, true);
|
|
channelPref.setIntent(new SubSettingLauncher(getActivity())
|
|
.setDestination(ChannelNotificationSettings.class.getName())
|
|
.setArguments(channelArgs)
|
|
.setTitle(R.string.notification_channel_title)
|
|
.setSourceMetricsCategory(getMetricsCategory())
|
|
.toIntent());
|
|
|
|
channelPref.setOnPreferenceChangeListener(
|
|
new Preference.OnPreferenceChangeListener() {
|
|
@Override
|
|
public boolean onPreferenceChange(Preference preference,
|
|
Object o) {
|
|
boolean value = (Boolean) o;
|
|
int importance = value ? IMPORTANCE_LOW : IMPORTANCE_NONE;
|
|
channel.setImportance(importance);
|
|
channel.lockFields(
|
|
NotificationChannel.USER_LOCKED_IMPORTANCE);
|
|
mBackend.updateChannel(mPkg, mUid, channel);
|
|
|
|
return true;
|
|
}
|
|
});
|
|
parent.addPreference(channelPref);
|
|
return channelPref;
|
|
}
|
|
|
|
protected boolean isChannelConfigurable(NotificationChannel channel) {
|
|
if (channel != null && mAppRow != null) {
|
|
return !channel.getId().equals(mAppRow.lockedChannelId);
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected boolean isChannelBlockable(NotificationChannel channel) {
|
|
if (channel != null && mAppRow != null) {
|
|
if (!mAppRow.systemApp) {
|
|
return true;
|
|
}
|
|
|
|
return channel.isBlockableSystem()
|
|
|| channel.getImportance() == NotificationManager.IMPORTANCE_NONE;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected boolean isChannelGroupBlockable(NotificationChannelGroup group) {
|
|
if (group != null && mAppRow != null) {
|
|
if (!mAppRow.systemApp) {
|
|
return true;
|
|
}
|
|
|
|
return group.isBlocked();
|
|
}
|
|
return false;
|
|
}
|
|
|
|
protected void setVisible(Preference p, boolean visible) {
|
|
setVisible(getPreferenceScreen(), p, visible);
|
|
}
|
|
|
|
protected void setVisible(PreferenceGroup parent, Preference p, boolean visible) {
|
|
final boolean isVisible = parent.findPreference(p.getKey()) != null;
|
|
if (isVisible == visible) return;
|
|
if (visible) {
|
|
parent.addPreference(p);
|
|
} else {
|
|
parent.removePreference(p);
|
|
}
|
|
}
|
|
|
|
protected void startListeningToPackageRemove() {
|
|
if (mListeningToPackageRemove) {
|
|
return;
|
|
}
|
|
mListeningToPackageRemove = true;
|
|
final IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_REMOVED);
|
|
filter.addDataScheme("package");
|
|
getContext().registerReceiver(mPackageRemovedReceiver, filter);
|
|
}
|
|
|
|
protected void stopListeningToPackageRemove() {
|
|
if (!mListeningToPackageRemove) {
|
|
return;
|
|
}
|
|
mListeningToPackageRemove = false;
|
|
getContext().unregisterReceiver(mPackageRemovedReceiver);
|
|
}
|
|
|
|
protected void onPackageRemoved() {
|
|
getActivity().finishAndRemoveTask();
|
|
}
|
|
|
|
protected final BroadcastReceiver mPackageRemovedReceiver = new BroadcastReceiver() {
|
|
@Override
|
|
public void onReceive(Context context, Intent intent) {
|
|
String packageName = intent.getData().getSchemeSpecificPart();
|
|
if (mPkgInfo == null || TextUtils.equals(mPkgInfo.packageName, packageName)) {
|
|
if (DEBUG) {
|
|
Log.d(TAG, "Package (" + packageName + ") removed. Removing"
|
|
+ "NotificationSettingsBase.");
|
|
}
|
|
onPackageRemoved();
|
|
}
|
|
}
|
|
};
|
|
|
|
protected Comparator<NotificationChannel> mChannelComparator =
|
|
(left, right) -> {
|
|
if (left.isDeleted() != right.isDeleted()) {
|
|
return Boolean.compare(left.isDeleted(), right.isDeleted());
|
|
} else if (left.getId().equals(NotificationChannel.DEFAULT_CHANNEL_ID)) {
|
|
// Uncategorized/miscellaneous legacy channel goes last
|
|
return 1;
|
|
} else if (right.getId().equals(NotificationChannel.DEFAULT_CHANNEL_ID)) {
|
|
return -1;
|
|
}
|
|
|
|
return left.getId().compareTo(right.getId());
|
|
};
|
|
|
|
protected class ImportanceListener {
|
|
protected void onImportanceChanged() {
|
|
final PreferenceScreen screen = getPreferenceScreen();
|
|
for (NotificationPreferenceController controller : mControllers) {
|
|
controller.displayPreference(screen);
|
|
}
|
|
updatePreferenceStates();
|
|
|
|
boolean hideDynamicFields = false;
|
|
if (mAppRow == null || mAppRow.banned) {
|
|
hideDynamicFields = true;
|
|
} else {
|
|
if (mChannel != null) {
|
|
hideDynamicFields = mChannel.getImportance() == IMPORTANCE_NONE;
|
|
} else if (mChannelGroup != null) {
|
|
hideDynamicFields = mChannelGroup.isBlocked();
|
|
}
|
|
}
|
|
for (Preference preference : mDynamicPreferences) {
|
|
setVisible(getPreferenceScreen(), preference, !hideDynamicFields);
|
|
}
|
|
}
|
|
}
|
|
}
|