/* * 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.app; import static com.android.settingslib.RestrictedLockUtils.EnforcedAdmin; import android.animation.AnimatorSet; import android.animation.ObjectAnimator; import android.animation.ValueAnimator; import android.app.Notification; import android.app.NotificationChannel; import android.app.NotificationChannelGroup; import android.app.NotificationManager; import android.app.role.RoleManager; 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.content.pm.ShortcutInfo; import android.graphics.drawable.Drawable; import android.os.Bundle; import android.os.UserHandle; import android.provider.Settings; import android.text.TextUtils; import android.util.Log; import android.view.View; import android.view.ViewGroup; import android.view.ViewTreeObserver; import android.view.animation.DecelerateInterpolator; import android.widget.Toast; import androidx.annotation.NonNull; import androidx.preference.PreferenceScreen; import com.android.settings.R; import com.android.settings.SettingsActivity; import com.android.settings.applications.AppInfoBase; import com.android.settings.dashboard.DashboardFragment; import com.android.settings.notification.NotificationBackend; import com.android.settingslib.RestrictedLockUtilsInternal; import com.android.settingslib.notification.ConversationIconFactory; import java.util.ArrayList; import java.util.List; abstract public class NotificationSettings extends DashboardFragment { private static final String TAG = "NotifiSettingsBase"; private static final boolean DEBUG = Log.isLoggable(TAG, Log.DEBUG); protected PackageManager mPm; protected NotificationBackend mBackend = new NotificationBackend(); protected NotificationManager mNm; protected RoleManager mRm; 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 Drawable mConversationDrawable; protected ShortcutInfo mConversationInfo; protected List mPreferenceFilter; protected boolean mShowLegacyChannelConfig = false; protected boolean mListeningToPackageRemove; protected List mControllers = new ArrayList<>(); protected DependentFieldListener mDependentFieldListener = new DependentFieldListener(); protected Intent mIntent; protected Bundle mArgs; private ViewGroup mLayoutView; private static final int DURATION_ANIMATE_PANEL_EXPAND_MS = 250; private final ViewTreeObserver.OnGlobalLayoutListener mOnGlobalLayoutListener = new ViewTreeObserver.OnGlobalLayoutListener() { @Override public void onGlobalLayout() { animateIn(); if (mLayoutView != null) { mLayoutView.getViewTreeObserver().removeOnGlobalLayoutListener(this); } } }; @Override public void onAttach(Context context) { super.onAttach(context); mContext = getActivity(); mIntent = getActivity().getIntent(); mArgs = getArguments(); mPm = getPackageManager(); mNm = NotificationManager.from(mContext); mRm = mContext.getSystemService(RoleManager.class); 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); if (mPkgInfo != null) { mUserId = UserHandle.getUserId(mUid); mSuspendedAppsAdmin = RestrictedLockUtilsInternal.checkIfApplicationIsSuspended( mContext, mPkg, mUserId); loadChannel(); loadAppRow(); loadChannelGroup(); loadPreferencesFilter(); collectConfigActivities(); if (use(HeaderPreferenceController.class) != null) { getSettingsLifecycle().addObserver(use(HeaderPreferenceController.class)); } if (use(ConversationHeaderPreferenceController.class) != null) { getSettingsLifecycle().addObserver( use(ConversationHeaderPreferenceController.class)); } for (NotificationPreferenceController controller : mControllers) { controller.onResume(mAppRow, mChannel, mChannelGroup, null, null, mSuspendedAppsAdmin, mPreferenceFilter); } } } @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(); loadConversation(); loadChannelGroup(); loadPreferencesFilter(); collectConfigActivities(); } protected void animatePanel() { if (mPreferenceFilter != null) { mLayoutView = getActivity().findViewById(R.id.main_content); mLayoutView.getViewTreeObserver().addOnGlobalLayoutListener(mOnGlobalLayoutListener); } } /** * Animate a Panel onto the screen. *

* Takes the entire panel and animates in from behind the navigation bar. *

* Relies on the Panel being having a fixed height to begin the animation. */ private void animateIn() { final AnimatorSet animatorSet = buildAnimatorSet(mLayoutView, mLayoutView.getHeight() /* startY */, 0.0f /* endY */, 0.0f /* startAlpha */, 1.0f /* endAlpha */, DURATION_ANIMATE_PANEL_EXPAND_MS); final ValueAnimator animator = new ValueAnimator(); animator.setFloatValues(0.0f, 1.0f); animatorSet.play(animator); animatorSet.start(); } /** * Build an {@link AnimatorSet} to animate the Panel, {@param parentView} in or out of the * screen, based on the positional parameters {@param startY}, {@param endY}, the parameters * for alpha changes {@param startAlpha}, {@param endAlpha}, and the {@param duration} in * milliseconds. */ @NonNull private static AnimatorSet buildAnimatorSet(@NonNull View targetView, float startY, float endY, float startAlpha, float endAlpha, int duration) { final AnimatorSet animatorSet = new AnimatorSet(); animatorSet.setDuration(duration); animatorSet.setInterpolator(new DecelerateInterpolator()); animatorSet.playTogether( ObjectAnimator.ofFloat(targetView, View.TRANSLATION_Y, startY, endY), ObjectAnimator.ofFloat(targetView, View.ALPHA, startAlpha, endAlpha)); return animatorSet; } private void loadPreferencesFilter() { Intent intent = getActivity().getIntent(); mPreferenceFilter = intent != null ? intent.getStringArrayListExtra(Settings.EXTRA_CHANNEL_FILTER_LIST) : null; } 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; } String conversationId = intent != null ? intent.getStringExtra(Settings.EXTRA_CONVERSATION_ID) : null; mChannel = mBackend.getChannel(mPkg, mUid, channelId, conversationId); } private void loadConversation() { if (mChannel == null || TextUtils.isEmpty(mChannel.getConversationId()) || mChannel.isDemoted()) { return; } mConversationInfo = mBackend.getConversationInfo( mContext, mPkg, mUid, mChannel.getConversationId()); if (mConversationInfo != null) { mConversationDrawable = mBackend.getConversationDrawable( mContext, mConversationInfo, mAppRow.pkg, mAppRow.uid, mChannel.isImportantConversation()); } } private void loadAppRow() { mAppRow = mBackend.loadAppRow(mContext, mPm, mRm, 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, null); } 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 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 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 class DependentFieldListener { protected void onFieldValueChanged() { // Reload the conversation drawable, which shows some channel/conversation state if (mConversationDrawable != null && mConversationDrawable instanceof ConversationIconFactory.ConversationIconDrawable) { ((ConversationIconFactory.ConversationIconDrawable) mConversationDrawable) .setImportant(mChannel.isImportantConversation()); } final PreferenceScreen screen = getPreferenceScreen(); for (NotificationPreferenceController controller : mControllers) { controller.displayPreference(screen); } updatePreferenceStates(); } } }