• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2016 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package com.android.settings.notification;
18 
19 import static android.app.NotificationManager.IMPORTANCE_LOW;
20 import static android.app.NotificationManager.IMPORTANCE_NONE;
21 import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin;
22 
23 import android.app.Activity;
24 import android.app.Notification;
25 import android.app.NotificationChannel;
26 import android.app.NotificationChannelGroup;
27 import android.app.NotificationManager;
28 import android.content.BroadcastReceiver;
29 import android.content.Context;
30 import android.content.Intent;
31 import android.content.IntentFilter;
32 import android.content.pm.ActivityInfo;
33 import android.content.pm.PackageInfo;
34 import android.content.pm.PackageManager;
35 import android.content.pm.PackageManager.NameNotFoundException;
36 import android.content.pm.ResolveInfo;
37 import android.os.Bundle;
38 import android.os.UserHandle;
39 import android.provider.Settings;
40 import android.support.v7.preference.Preference;
41 import android.support.v7.preference.PreferenceGroup;
42 import android.support.v7.preference.PreferenceScreen;
43 import android.text.TextUtils;
44 import android.util.Log;
45 import android.widget.Toast;
46 
47 import com.android.settings.R;
48 import com.android.settings.SettingsActivity;
49 import com.android.settings.applications.AppInfoBase;
50 import com.android.settings.core.SubSettingLauncher;
51 import com.android.settings.dashboard.DashboardFragment;
52 import com.android.settings.widget.MasterCheckBoxPreference;
53 import com.android.settingslib.RestrictedLockUtils;
54 
55 import java.util.ArrayList;
56 import java.util.Comparator;
57 import java.util.List;
58 
59 abstract public class NotificationSettingsBase extends DashboardFragment {
60     private static final String TAG = "NotifiSettingsBase";
61     private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG);
62     protected static final String ARG_FROM_SETTINGS = "fromSettings";
63 
64     protected PackageManager mPm;
65     protected NotificationBackend mBackend = new NotificationBackend();
66     protected NotificationManager mNm;
67     protected Context mContext;
68 
69     protected int mUid;
70     protected int mUserId;
71     protected String mPkg;
72     protected PackageInfo mPkgInfo;
73     protected EnforcedAdmin mSuspendedAppsAdmin;
74     protected NotificationChannelGroup mChannelGroup;
75     protected NotificationChannel mChannel;
76     protected NotificationBackend.AppRow mAppRow;
77 
78     protected boolean mShowLegacyChannelConfig = false;
79     protected boolean mListeningToPackageRemove;
80 
81     protected List<NotificationPreferenceController> mControllers = new ArrayList<>();
82     protected List<Preference> mDynamicPreferences = new ArrayList<>();
83     protected ImportanceListener mImportanceListener = new ImportanceListener();
84 
85     protected Intent mIntent;
86     protected Bundle mArgs;
87 
88     @Override
onAttach(Context context)89     public void onAttach(Context context) {
90         super.onAttach(context);
91         mContext = getActivity();
92         mIntent = getActivity().getIntent();
93         mArgs = getArguments();
94 
95         mPm = getPackageManager();
96         mNm = NotificationManager.from(mContext);
97 
98         mPkg = mArgs != null && mArgs.containsKey(AppInfoBase.ARG_PACKAGE_NAME)
99                 ? mArgs.getString(AppInfoBase.ARG_PACKAGE_NAME)
100                 : mIntent.getStringExtra(Settings.EXTRA_APP_PACKAGE);
101         mUid = mArgs != null && mArgs.containsKey(AppInfoBase.ARG_PACKAGE_UID)
102                 ? mArgs.getInt(AppInfoBase.ARG_PACKAGE_UID)
103                 : mIntent.getIntExtra(Settings.EXTRA_APP_UID, -1);
104 
105         if (mUid < 0) {
106             try {
107                 mUid = mPm.getPackageUid(mPkg, 0);
108             } catch (NameNotFoundException e) {
109             }
110         }
111 
112         mPkgInfo = findPackageInfo(mPkg, mUid);
113 
114         mUserId = UserHandle.getUserId(mUid);
115         mSuspendedAppsAdmin = RestrictedLockUtils.checkIfApplicationIsSuspended(
116                 mContext, mPkg, mUserId);
117 
118         loadChannel();
119         loadAppRow();
120         loadChannelGroup();
121         collectConfigActivities();
122 
123         getLifecycle().addObserver(use(HeaderPreferenceController.class));
124 
125         for (NotificationPreferenceController controller : mControllers) {
126             controller.onResume(mAppRow, mChannel, mChannelGroup, mSuspendedAppsAdmin);
127         }
128     }
129 
130     @Override
onCreate(Bundle savedInstanceState)131     public void onCreate(Bundle savedInstanceState) {
132         super.onCreate(savedInstanceState);
133 
134         if (mIntent == null && mArgs == null) {
135             Log.w(TAG, "No intent");
136             toastAndFinish();
137             return;
138         }
139 
140         if (mUid < 0 || TextUtils.isEmpty(mPkg) || mPkgInfo == null) {
141             Log.w(TAG, "Missing package or uid or packageinfo");
142             toastAndFinish();
143             return;
144         }
145 
146         startListeningToPackageRemove();
147     }
148 
149     @Override
onDestroy()150     public void onDestroy() {
151         stopListeningToPackageRemove();
152         super.onDestroy();
153     }
154 
155     @Override
onResume()156     public void onResume() {
157         super.onResume();
158         if (mUid < 0 || TextUtils.isEmpty(mPkg) || mPkgInfo == null || mAppRow == null) {
159             Log.w(TAG, "Missing package or uid or packageinfo");
160             finish();
161             return;
162         }
163         // Reload app, channel, etc onResume in case they've changed. A little wasteful if we've
164         // just done onAttach but better than making every preference controller reload all
165         // the data
166         loadAppRow();
167         if (mAppRow == null) {
168             Log.w(TAG, "Can't load package");
169             finish();
170             return;
171         }
172         loadChannel();
173         loadChannelGroup();
174         collectConfigActivities();
175     }
176 
loadChannel()177     private void loadChannel() {
178         Intent intent = getActivity().getIntent();
179         String channelId = intent != null ? intent.getStringExtra(Settings.EXTRA_CHANNEL_ID) : null;
180         if (channelId == null && intent != null) {
181             Bundle args = intent.getBundleExtra(SettingsActivity.EXTRA_SHOW_FRAGMENT_ARGUMENTS);
182             channelId = args != null ? args.getString(Settings.EXTRA_CHANNEL_ID) : null;
183         }
184         mChannel = mBackend.getChannel(mPkg, mUid, channelId);
185     }
186 
loadAppRow()187     private void loadAppRow() {
188         mAppRow = mBackend.loadAppRow(mContext, mPm, mPkgInfo);
189     }
190 
loadChannelGroup()191     private void loadChannelGroup() {
192         mShowLegacyChannelConfig = mBackend.onlyHasDefaultChannel(mAppRow.pkg, mAppRow.uid)
193                 || (mChannel != null
194                 && NotificationChannel.DEFAULT_CHANNEL_ID.equals(mChannel.getId()));
195 
196         if (mShowLegacyChannelConfig) {
197             mChannel = mBackend.getChannel(
198                     mAppRow.pkg, mAppRow.uid, NotificationChannel.DEFAULT_CHANNEL_ID);
199         }
200         if (mChannel != null && !TextUtils.isEmpty(mChannel.getGroup())) {
201             NotificationChannelGroup group = mBackend.getGroup(mPkg, mUid, mChannel.getGroup());
202             if (group != null) {
203                 mChannelGroup = group;
204             }
205         }
206     }
207 
toastAndFinish()208     protected void toastAndFinish() {
209         Toast.makeText(mContext, R.string.app_not_found_dlg_text, Toast.LENGTH_SHORT).show();
210         getActivity().finish();
211     }
212 
collectConfigActivities()213     protected void collectConfigActivities() {
214         Intent intent = new Intent(Intent.ACTION_MAIN)
215                 .addCategory(Notification.INTENT_CATEGORY_NOTIFICATION_PREFERENCES)
216                 .setPackage(mAppRow.pkg);
217         final List<ResolveInfo> resolveInfos = mPm.queryIntentActivities(
218                 intent,
219                 0 //PackageManager.MATCH_DEFAULT_ONLY
220         );
221         if (DEBUG) {
222             Log.d(TAG, "Found " + resolveInfos.size() + " preference activities"
223                     + (resolveInfos.size() == 0 ? " ;_;" : ""));
224         }
225         for (ResolveInfo ri : resolveInfos) {
226             final ActivityInfo activityInfo = ri.activityInfo;
227             if (mAppRow.settingsIntent != null) {
228                 if (DEBUG) {
229                     Log.d(TAG, "Ignoring duplicate notification preference activity ("
230                             + activityInfo.name + ") for package "
231                             + activityInfo.packageName);
232                 }
233                 continue;
234             }
235             // TODO(78660939): This should actually start a new task
236             mAppRow.settingsIntent = intent
237                     .setPackage(null)
238                     .setClassName(activityInfo.packageName, activityInfo.name);
239             if (mChannel != null) {
240                 mAppRow.settingsIntent.putExtra(Notification.EXTRA_CHANNEL_ID, mChannel.getId());
241             }
242             if (mChannelGroup != null) {
243                 mAppRow.settingsIntent.putExtra(
244                         Notification.EXTRA_CHANNEL_GROUP_ID, mChannelGroup.getId());
245             }
246         }
247     }
248 
findPackageInfo(String pkg, int uid)249     private PackageInfo findPackageInfo(String pkg, int uid) {
250         if (pkg == null || uid < 0) {
251             return null;
252         }
253         final String[] packages = mPm.getPackagesForUid(uid);
254         if (packages != null && pkg != null) {
255             final int N = packages.length;
256             for (int i = 0; i < N; i++) {
257                 final String p = packages[i];
258                 if (pkg.equals(p)) {
259                     try {
260                         return mPm.getPackageInfo(pkg, PackageManager.GET_SIGNATURES);
261                     } catch (NameNotFoundException e) {
262                         Log.w(TAG, "Failed to load package " + pkg, e);
263                     }
264                 }
265             }
266         }
267         return null;
268     }
269 
populateSingleChannelPrefs(PreferenceGroup parent, final NotificationChannel channel, final boolean groupBlocked)270     protected Preference populateSingleChannelPrefs(PreferenceGroup parent,
271             final NotificationChannel channel, final boolean groupBlocked) {
272         MasterCheckBoxPreference channelPref = new MasterCheckBoxPreference(
273                 getPrefContext());
274         channelPref.setCheckBoxEnabled(mSuspendedAppsAdmin == null
275                 && isChannelBlockable(channel)
276                 && isChannelConfigurable(channel)
277                 && !groupBlocked);
278         channelPref.setKey(channel.getId());
279         channelPref.setTitle(channel.getName());
280         channelPref.setChecked(channel.getImportance() != IMPORTANCE_NONE);
281         Bundle channelArgs = new Bundle();
282         channelArgs.putInt(AppInfoBase.ARG_PACKAGE_UID, mUid);
283         channelArgs.putString(AppInfoBase.ARG_PACKAGE_NAME, mPkg);
284         channelArgs.putString(Settings.EXTRA_CHANNEL_ID, channel.getId());
285         channelArgs.putBoolean(ARG_FROM_SETTINGS, true);
286         channelPref.setIntent(new SubSettingLauncher(getActivity())
287                 .setDestination(ChannelNotificationSettings.class.getName())
288                 .setArguments(channelArgs)
289                 .setTitle(R.string.notification_channel_title)
290                 .setSourceMetricsCategory(getMetricsCategory())
291                 .toIntent());
292 
293         channelPref.setOnPreferenceChangeListener(
294                 new Preference.OnPreferenceChangeListener() {
295                     @Override
296                     public boolean onPreferenceChange(Preference preference,
297                             Object o) {
298                         boolean value = (Boolean) o;
299                         int importance = value ? IMPORTANCE_LOW : IMPORTANCE_NONE;
300                         channel.setImportance(importance);
301                         channel.lockFields(
302                                 NotificationChannel.USER_LOCKED_IMPORTANCE);
303                         mBackend.updateChannel(mPkg, mUid, channel);
304 
305                         return true;
306                     }
307                 });
308         parent.addPreference(channelPref);
309         return channelPref;
310     }
311 
isChannelConfigurable(NotificationChannel channel)312     protected boolean isChannelConfigurable(NotificationChannel channel) {
313         if (channel != null && mAppRow != null) {
314             return !channel.getId().equals(mAppRow.lockedChannelId);
315         }
316         return false;
317     }
318 
isChannelBlockable(NotificationChannel channel)319     protected boolean isChannelBlockable(NotificationChannel channel) {
320         if (channel != null && mAppRow != null) {
321             if (!mAppRow.systemApp) {
322                 return true;
323             }
324 
325             return channel.isBlockableSystem()
326                     || channel.getImportance() == NotificationManager.IMPORTANCE_NONE;
327         }
328         return false;
329     }
330 
isChannelGroupBlockable(NotificationChannelGroup group)331     protected boolean isChannelGroupBlockable(NotificationChannelGroup group) {
332         if (group != null && mAppRow != null) {
333             if (!mAppRow.systemApp) {
334                 return true;
335             }
336 
337             return group.isBlocked();
338         }
339         return false;
340     }
341 
setVisible(Preference p, boolean visible)342     protected void setVisible(Preference p, boolean visible) {
343         setVisible(getPreferenceScreen(), p, visible);
344     }
345 
setVisible(PreferenceGroup parent, Preference p, boolean visible)346     protected void setVisible(PreferenceGroup parent, Preference p, boolean visible) {
347         final boolean isVisible = parent.findPreference(p.getKey()) != null;
348         if (isVisible == visible) return;
349         if (visible) {
350             parent.addPreference(p);
351         } else {
352             parent.removePreference(p);
353         }
354     }
355 
startListeningToPackageRemove()356     protected void startListeningToPackageRemove() {
357         if (mListeningToPackageRemove) {
358             return;
359         }
360         mListeningToPackageRemove = true;
361         final IntentFilter filter = new IntentFilter(Intent.ACTION_PACKAGE_REMOVED);
362         filter.addDataScheme("package");
363         getContext().registerReceiver(mPackageRemovedReceiver, filter);
364     }
365 
stopListeningToPackageRemove()366     protected void stopListeningToPackageRemove() {
367         if (!mListeningToPackageRemove) {
368             return;
369         }
370         mListeningToPackageRemove = false;
371         getContext().unregisterReceiver(mPackageRemovedReceiver);
372     }
373 
onPackageRemoved()374     protected void onPackageRemoved() {
375         getActivity().finishAndRemoveTask();
376     }
377 
378     protected final BroadcastReceiver mPackageRemovedReceiver = new BroadcastReceiver() {
379         @Override
380         public void onReceive(Context context, Intent intent) {
381             String packageName = intent.getData().getSchemeSpecificPart();
382             if (mPkgInfo == null || TextUtils.equals(mPkgInfo.packageName, packageName)) {
383                 if (DEBUG) {
384                     Log.d(TAG, "Package (" + packageName + ") removed. Removing"
385                             + "NotificationSettingsBase.");
386                 }
387                 onPackageRemoved();
388             }
389         }
390     };
391 
392     protected Comparator<NotificationChannel> mChannelComparator =
393             (left, right) -> {
394                 if (left.isDeleted() != right.isDeleted()) {
395                     return Boolean.compare(left.isDeleted(), right.isDeleted());
396                 } else if (left.getId().equals(NotificationChannel.DEFAULT_CHANNEL_ID)) {
397                     // Uncategorized/miscellaneous legacy channel goes last
398                     return 1;
399                 } else if (right.getId().equals(NotificationChannel.DEFAULT_CHANNEL_ID)) {
400                     return -1;
401                 }
402 
403                 return left.getId().compareTo(right.getId());
404             };
405 
406     protected class ImportanceListener {
onImportanceChanged()407         protected void onImportanceChanged() {
408             final PreferenceScreen screen = getPreferenceScreen();
409             for (NotificationPreferenceController controller : mControllers) {
410                 controller.displayPreference(screen);
411             }
412             updatePreferenceStates();
413 
414             boolean hideDynamicFields = false;
415             if (mAppRow == null || mAppRow.banned) {
416                 hideDynamicFields = true;
417             } else {
418                 if (mChannel != null) {
419                     hideDynamicFields = mChannel.getImportance() == IMPORTANCE_NONE;
420                 } else if (mChannelGroup != null) {
421                     hideDynamicFields = mChannelGroup.isBlocked();
422                 }
423             }
424             for (Preference preference : mDynamicPreferences) {
425                 setVisible(getPreferenceScreen(), preference, !hideDynamicFields);
426             }
427         }
428     }
429 }
430