/* * Copyright (C) 2014 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.NotificationChannel; import android.app.NotificationChannelGroup; import android.app.settings.SettingsEnums; import android.content.Context; import android.os.AsyncTask; import android.os.Bundle; import android.text.TextUtils; import android.util.Log; import androidx.preference.Preference; import androidx.preference.PreferenceCategory; import androidx.preference.PreferenceGroup; import androidx.preference.PreferenceScreen; import androidx.preference.SwitchPreference; import com.android.internal.widget.LockPatternUtils; import com.android.settings.R; import com.android.settings.widget.MasterSwitchPreference; import com.android.settingslib.RestrictedSwitchPreference; import com.android.settingslib.core.AbstractPreferenceController; import java.util.ArrayList; import java.util.Collections; import java.util.Comparator; import java.util.List; /** These settings are per app, so should not be returned in global search results. */ public class AppNotificationSettings extends NotificationSettingsBase { private static final String TAG = "AppNotificationSettings"; private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); private static String KEY_GENERAL_CATEGORY = "categories"; private static String KEY_ADVANCED_CATEGORY = "app_advanced"; private static String KEY_BADGE = "badge"; private static String KEY_APP_LINK = "app_link"; private static String KEY_BUBBLE = "bubble_link_pref"; private static String[] LEGACY_NON_ADVANCED_KEYS = {KEY_BADGE, KEY_APP_LINK, KEY_BUBBLE}; private List mChannelGroupList; @Override public int getMetricsCategory() { return SettingsEnums.NOTIFICATION_APP_NOTIFICATION; } @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); final PreferenceScreen screen = getPreferenceScreen(); if (mShowLegacyChannelConfig && screen != null) { // if showing legacy settings, pull advanced settings out of the advanced category PreferenceGroup advanced = (PreferenceGroup) findPreference(KEY_ADVANCED_CATEGORY); removePreference(KEY_ADVANCED_CATEGORY); if (advanced != null) { for (String key : LEGACY_NON_ADVANCED_KEYS) { Preference pref = advanced.findPreference(key); advanced.removePreference(pref); if (pref != null) { screen.addPreference(pref); } } } } } @Override public void onResume() { super.onResume(); if (mUid < 0 || TextUtils.isEmpty(mPkg) || mPkgInfo == null) { Log.w(TAG, "Missing package or uid or packageinfo"); finish(); return; } if (!mShowLegacyChannelConfig) { // Load channel settings new AsyncTask() { @Override protected Void doInBackground(Void... unused) { mChannelGroupList = mBackend.getGroups(mPkg, mUid).getList(); Collections.sort(mChannelGroupList, mChannelGroupComparator); return null; } @Override protected void onPostExecute(Void unused) { if (getHost() == null) { return; } populateList(); } }.execute(); } for (NotificationPreferenceController controller : mControllers) { controller.onResume(mAppRow, mChannel, mChannelGroup, mSuspendedAppsAdmin); controller.displayPreference(getPreferenceScreen()); } updatePreferenceStates(); } @Override protected String getLogTag() { return TAG; } @Override protected int getPreferenceScreenResId() { return R.xml.app_notification_settings; } @Override protected List createPreferenceControllers(Context context) { mControllers = new ArrayList<>(); mControllers.add(new HeaderPreferenceController(context, this)); mControllers.add(new BlockPreferenceController(context, mImportanceListener, mBackend)); mControllers.add(new BadgePreferenceController(context, mBackend)); mControllers.add(new AllowSoundPreferenceController( context, mImportanceListener, mBackend)); mControllers.add(new ImportancePreferenceController( context, mImportanceListener, mBackend)); mControllers.add(new MinImportancePreferenceController( context, mImportanceListener, mBackend)); mControllers.add(new HighImportancePreferenceController( context, mImportanceListener, mBackend)); mControllers.add(new SoundPreferenceController(context, this, mImportanceListener, mBackend)); mControllers.add(new LightsPreferenceController(context, mBackend)); mControllers.add(new VibrationPreferenceController(context, mBackend)); mControllers.add(new VisibilityPreferenceController(context, new LockPatternUtils(context), mBackend)); mControllers.add(new DndPreferenceController(context, mBackend)); mControllers.add(new AppLinkPreferenceController(context)); mControllers.add(new DescriptionPreferenceController(context)); mControllers.add(new NotificationsOffPreferenceController(context)); mControllers.add(new DeletedChannelsPreferenceController(context, mBackend)); mControllers.add(new BubbleSummaryPreferenceController(context, mBackend)); return new ArrayList<>(mControllers); } private void populateList() { if (!mDynamicPreferences.isEmpty()) { for (Preference p : mDynamicPreferences) { getPreferenceScreen().removePreference(p); } mDynamicPreferences.clear(); } if (mChannelGroupList.isEmpty()) { PreferenceCategory groupCategory = new PreferenceCategory(getPrefContext()); groupCategory.setTitle(R.string.notification_channels); groupCategory.setKey(KEY_GENERAL_CATEGORY); getPreferenceScreen().addPreference(groupCategory); mDynamicPreferences.add(groupCategory); Preference empty = new Preference(getPrefContext()); empty.setTitle(R.string.no_channels); empty.setEnabled(false); groupCategory.addPreference(empty); } else { populateGroupList(); mImportanceListener.onImportanceChanged(); } } private void populateGroupList() { for (NotificationChannelGroup group : mChannelGroupList) { PreferenceCategory groupCategory = new PreferenceCategory(getPrefContext()); groupCategory.setOrderingAsAdded(true); getPreferenceScreen().addPreference(groupCategory); mDynamicPreferences.add(groupCategory); if (group.getId() == null) { if (mChannelGroupList.size() > 1) { groupCategory.setTitle(R.string.notification_channels_other); } groupCategory.setKey(KEY_GENERAL_CATEGORY); } else { groupCategory.setTitle(group.getName()); groupCategory.setKey(group.getId()); populateGroupToggle(groupCategory, group); } if (!group.isBlocked()) { final List channels = group.getChannels(); Collections.sort(channels, mChannelComparator); int N = channels.size(); for (int i = 0; i < N; i++) { final NotificationChannel channel = channels.get(i); populateSingleChannelPrefs(groupCategory, channel, group.isBlocked()); } } } } protected void populateGroupToggle(final PreferenceGroup parent, NotificationChannelGroup group) { RestrictedSwitchPreference preference = new RestrictedSwitchPreference(getPrefContext()); preference.setTitle(R.string.notification_switch_label); preference.setEnabled(mSuspendedAppsAdmin == null && isChannelGroupBlockable(group)); preference.setChecked(!group.isBlocked()); preference.setOnPreferenceClickListener(preference1 -> { final boolean allowGroup = ((SwitchPreference) preference1).isChecked(); group.setBlocked(!allowGroup); mBackend.updateChannelGroup(mAppRow.pkg, mAppRow.uid, group); onGroupBlockStateChanged(group); return true; }); parent.addPreference(preference); } private Comparator mChannelGroupComparator = new Comparator() { @Override public int compare(NotificationChannelGroup left, NotificationChannelGroup right) { // Non-grouped channels (in placeholder group with a null id) come last if (left.getId() == null && right.getId() != null) { return 1; } else if (right.getId() == null && left.getId() != null) { return -1; } return left.getId().compareTo(right.getId()); } }; protected void onGroupBlockStateChanged(NotificationChannelGroup group) { if (group == null) { return; } PreferenceGroup groupGroup = ( PreferenceGroup) getPreferenceScreen().findPreference(group.getId()); if (groupGroup != null) { if (group.isBlocked()) { List toRemove = new ArrayList<>(); int childCount = groupGroup.getPreferenceCount(); for (int i = 0; i < childCount; i++) { Preference pref = groupGroup.getPreference(i); if (pref instanceof MasterSwitchPreference) { toRemove.add(pref); } } for (Preference pref : toRemove) { groupGroup.removePreference(pref); } } else { final List channels = group.getChannels(); Collections.sort(channels, mChannelComparator); int N = channels.size(); for (int i = 0; i < N; i++) { final NotificationChannel channel = channels.get(i); populateSingleChannelPrefs(groupGroup, channel, group.isBlocked()); } } } } }