/* * Copyright (C) 2017 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.systemui.statusbar.notification.row; import static android.app.AppOpsManager.OP_CAMERA; import static android.app.AppOpsManager.OP_RECORD_AUDIO; import static android.app.AppOpsManager.OP_SYSTEM_ALERT_WINDOW; import android.app.INotificationManager; import android.app.NotificationChannel; import android.content.Context; import android.content.Intent; import android.content.pm.LauncherApps; import android.content.pm.PackageManager; import android.content.pm.ShortcutManager; import android.net.Uri; import android.os.Bundle; import android.os.Handler; import android.os.UserHandle; import android.os.UserManager; import android.provider.Settings; import android.service.notification.StatusBarNotification; import android.util.ArraySet; import android.util.IconDrawableFactory; import android.util.Log; import android.view.HapticFeedbackConstants; import android.view.View; import android.view.accessibility.AccessibilityManager; import com.android.internal.annotations.VisibleForTesting; import com.android.internal.logging.MetricsLogger; import com.android.internal.logging.UiEventLogger; import com.android.internal.logging.nano.MetricsProto; import com.android.settingslib.notification.ConversationIconFactory; import com.android.systemui.R; import com.android.systemui.dagger.SysUISingleton; import com.android.systemui.dagger.qualifiers.Background; import com.android.systemui.dagger.qualifiers.Main; import com.android.systemui.people.widget.PeopleSpaceWidgetManager; import com.android.systemui.plugins.statusbar.NotificationMenuRowPlugin; import com.android.systemui.plugins.statusbar.StatusBarStateController; import com.android.systemui.settings.UserContextProvider; import com.android.systemui.shade.ShadeController; import com.android.systemui.statusbar.NotificationLockscreenUserManager; import com.android.systemui.statusbar.NotificationPresenter; import com.android.systemui.statusbar.StatusBarState; import com.android.systemui.statusbar.StatusBarStateControllerImpl; import com.android.systemui.statusbar.notification.AssistantFeedbackController; import com.android.systemui.statusbar.notification.NotificationActivityStarter; import com.android.systemui.statusbar.notification.collection.NotificationEntry; import com.android.systemui.statusbar.notification.collection.provider.HighPriorityProvider; import com.android.systemui.statusbar.notification.collection.render.NotifGutsViewListener; import com.android.systemui.statusbar.notification.collection.render.NotifGutsViewManager; import com.android.systemui.statusbar.notification.stack.NotificationListContainer; import com.android.systemui.statusbar.phone.CentralSurfaces; import com.android.systemui.statusbar.policy.DeviceProvisionedController; import com.android.systemui.wmshell.BubblesManager; import java.util.Optional; import javax.inject.Inject; import dagger.Lazy; /** * Handles various NotificationGuts related tasks, such as binding guts to a row, opening and * closing guts, and keeping track of the currently exposed notification guts. */ @SysUISingleton public class NotificationGutsManager implements NotifGutsViewManager { private static final String TAG = "NotificationGutsManager"; // Must match constant in Settings. Used to highlight preferences when linking to Settings. private static final String EXTRA_FRAGMENT_ARG_KEY = ":settings:fragment_args_key"; private final MetricsLogger mMetricsLogger; private final Context mContext; private final AccessibilityManager mAccessibilityManager; private final HighPriorityProvider mHighPriorityProvider; private final ChannelEditorDialogController mChannelEditorDialogController; private final OnUserInteractionCallback mOnUserInteractionCallback; // Dependencies: private final NotificationLockscreenUserManager mLockscreenUserManager; private final StatusBarStateController mStatusBarStateController; private final DeviceProvisionedController mDeviceProvisionedController; private final AssistantFeedbackController mAssistantFeedbackController; // which notification is currently being longpress-examined by the user private NotificationGuts mNotificationGutsExposed; private NotificationMenuRowPlugin.MenuItem mGutsMenuItem; private NotificationPresenter mPresenter; private NotificationActivityStarter mNotificationActivityStarter; private NotificationListContainer mListContainer; private OnSettingsClickListener mOnSettingsClickListener; private final Lazy> mCentralSurfacesOptionalLazy; private final Handler mMainHandler; private final Handler mBgHandler; private final Optional mBubblesManagerOptional; private Runnable mOpenRunnable; private final INotificationManager mNotificationManager; private final PeopleSpaceWidgetManager mPeopleSpaceWidgetManager; private final UserManager mUserManager; private final LauncherApps mLauncherApps; private final ShortcutManager mShortcutManager; private final UserContextProvider mContextTracker; private final UiEventLogger mUiEventLogger; private final ShadeController mShadeController; private NotifGutsViewListener mGutsListener; @Inject public NotificationGutsManager(Context context, Lazy> centralSurfacesOptionalLazy, @Main Handler mainHandler, @Background Handler bgHandler, AccessibilityManager accessibilityManager, HighPriorityProvider highPriorityProvider, INotificationManager notificationManager, UserManager userManager, PeopleSpaceWidgetManager peopleSpaceWidgetManager, LauncherApps launcherApps, ShortcutManager shortcutManager, ChannelEditorDialogController channelEditorDialogController, UserContextProvider contextTracker, AssistantFeedbackController assistantFeedbackController, Optional bubblesManagerOptional, UiEventLogger uiEventLogger, OnUserInteractionCallback onUserInteractionCallback, ShadeController shadeController, NotificationLockscreenUserManager notificationLockscreenUserManager, StatusBarStateController statusBarStateController, DeviceProvisionedController deviceProvisionedController, MetricsLogger metricsLogger) { mContext = context; mCentralSurfacesOptionalLazy = centralSurfacesOptionalLazy; mMainHandler = mainHandler; mBgHandler = bgHandler; mAccessibilityManager = accessibilityManager; mHighPriorityProvider = highPriorityProvider; mNotificationManager = notificationManager; mUserManager = userManager; mPeopleSpaceWidgetManager = peopleSpaceWidgetManager; mLauncherApps = launcherApps; mShortcutManager = shortcutManager; mContextTracker = contextTracker; mChannelEditorDialogController = channelEditorDialogController; mAssistantFeedbackController = assistantFeedbackController; mBubblesManagerOptional = bubblesManagerOptional; mUiEventLogger = uiEventLogger; mOnUserInteractionCallback = onUserInteractionCallback; mShadeController = shadeController; mLockscreenUserManager = notificationLockscreenUserManager; mStatusBarStateController = statusBarStateController; mDeviceProvisionedController = deviceProvisionedController; mMetricsLogger = metricsLogger; } public void setUpWithPresenter(NotificationPresenter presenter, NotificationListContainer listContainer, OnSettingsClickListener onSettingsClick) { mPresenter = presenter; mListContainer = listContainer; mOnSettingsClickListener = onSettingsClick; } public void setNotificationActivityStarter( NotificationActivityStarter notificationActivityStarter) { mNotificationActivityStarter = notificationActivityStarter; } public void onDensityOrFontScaleChanged(NotificationEntry entry) { setExposedGuts(entry.getGuts()); bindGuts(entry.getRow()); } /** * Sends an intent to open the notification settings for a particular package and optional * channel. */ public static final String EXTRA_SHOW_FRAGMENT_ARGUMENTS = ":settings:show_fragment_args"; private void startAppNotificationSettingsActivity(String packageName, final int appUid, final NotificationChannel channel, ExpandableNotificationRow row) { final Intent intent = new Intent(Settings.ACTION_APP_NOTIFICATION_SETTINGS); intent.putExtra(Settings.EXTRA_APP_PACKAGE, packageName); intent.putExtra(Settings.EXTRA_APP_UID, appUid); if (channel != null) { final Bundle args = new Bundle(); intent.putExtra(EXTRA_FRAGMENT_ARG_KEY, channel.getId()); args.putString(EXTRA_FRAGMENT_ARG_KEY, channel.getId()); intent.putExtra(EXTRA_SHOW_FRAGMENT_ARGUMENTS, args); } mNotificationActivityStarter.startNotificationGutsIntent(intent, appUid, row); } private void startAppDetailsSettingsActivity(String packageName, final int appUid, ExpandableNotificationRow row) { final Intent intent = new Intent(Settings.ACTION_APPLICATION_DETAILS_SETTINGS); intent.setData(Uri.fromParts("package", packageName, null)); intent.putExtra(Settings.EXTRA_APP_PACKAGE, packageName); intent.putExtra(Settings.EXTRA_APP_UID, appUid); mNotificationActivityStarter.startNotificationGutsIntent(intent, appUid, row); } protected void startAppOpsSettingsActivity(String pkg, int uid, ArraySet ops, ExpandableNotificationRow row) { if (ops.contains(OP_SYSTEM_ALERT_WINDOW)) { if (ops.contains(OP_CAMERA) || ops.contains(OP_RECORD_AUDIO)) { startAppDetailsSettingsActivity(pkg, uid, row); } else { Intent intent = new Intent(Settings.ACTION_MANAGE_APP_OVERLAY_PERMISSION); intent.setData(Uri.fromParts("package", pkg, null)); mNotificationActivityStarter.startNotificationGutsIntent(intent, uid, row); } } else if (ops.contains(OP_CAMERA) || ops.contains(OP_RECORD_AUDIO)) { Intent intent = new Intent(Intent.ACTION_MANAGE_APP_PERMISSIONS); intent.putExtra(Intent.EXTRA_PACKAGE_NAME, pkg); mNotificationActivityStarter.startNotificationGutsIntent(intent, uid, row); } } private void startConversationSettingsActivity(int uid, ExpandableNotificationRow row) { final Intent intent = new Intent(Settings.ACTION_CONVERSATION_SETTINGS); mNotificationActivityStarter.startNotificationGutsIntent(intent, uid, row); } private boolean bindGuts(final ExpandableNotificationRow row) { row.ensureGutsInflated(); return bindGuts(row, mGutsMenuItem); } @VisibleForTesting protected boolean bindGuts(final ExpandableNotificationRow row, NotificationMenuRowPlugin.MenuItem item) { NotificationEntry entry = row.getEntry(); row.setGutsView(item); row.setTag(entry.getSbn().getPackageName()); row.getGuts().setClosedListener((NotificationGuts g) -> { row.onGutsClosed(); if (!g.willBeRemoved() && !row.isRemoved()) { mListContainer.onHeightChanged( row, !mPresenter.isPresenterFullyCollapsed() /* needsAnimation */); } if (mNotificationGutsExposed == g) { mNotificationGutsExposed = null; mGutsMenuItem = null; } if (mGutsListener != null) { mGutsListener.onGutsClose(entry); } String key = entry.getKey(); }); View gutsView = item.getGutsView(); try { if (gutsView instanceof NotificationSnooze) { initializeSnoozeView(row, (NotificationSnooze) gutsView); } else if (gutsView instanceof NotificationInfo) { initializeNotificationInfo(row, (NotificationInfo) gutsView); } else if (gutsView instanceof NotificationConversationInfo) { initializeConversationNotificationInfo( row, (NotificationConversationInfo) gutsView); } else if (gutsView instanceof PartialConversationInfo) { initializePartialConversationNotificationInfo(row, (PartialConversationInfo) gutsView); } else if (gutsView instanceof FeedbackInfo) { initializeFeedbackInfo(row, (FeedbackInfo) gutsView); } return true; } catch (Exception e) { Log.e(TAG, "error binding guts", e); return false; } } /** * Sets up the {@link NotificationSnooze} inside the notification row's guts. * * @param row view to set up the guts for * @param notificationSnoozeView view to set up/bind within {@code row} */ private void initializeSnoozeView( final ExpandableNotificationRow row, NotificationSnooze notificationSnoozeView) { NotificationGuts guts = row.getGuts(); StatusBarNotification sbn = row.getEntry().getSbn(); notificationSnoozeView.setSnoozeListener(mListContainer.getSwipeActionHelper()); notificationSnoozeView.setStatusBarNotification(sbn); notificationSnoozeView.setSnoozeOptions(row.getEntry().getSnoozeCriteria()); guts.setHeightChangedListener((NotificationGuts g) -> { mListContainer.onHeightChanged(row, row.isShown() /* needsAnimation */); }); } /** * Sets up the {@link FeedbackInfo} inside the notification row's guts. * * @param row view to set up the guts for * @param feedbackInfo view to set up/bind within {@code row} */ private void initializeFeedbackInfo( final ExpandableNotificationRow row, FeedbackInfo feedbackInfo) { if (mAssistantFeedbackController.getFeedbackIcon(row.getEntry()) == null) { return; } StatusBarNotification sbn = row.getEntry().getSbn(); UserHandle userHandle = sbn.getUser(); PackageManager pmUser = CentralSurfaces.getPackageManagerForUser(mContext, userHandle.getIdentifier()); feedbackInfo.bindGuts(pmUser, sbn, row.getEntry(), row, mAssistantFeedbackController); } /** * Sets up the {@link NotificationInfo} inside the notification row's guts. * @param row view to set up the guts for * @param notificationInfoView view to set up/bind within {@code row} */ @VisibleForTesting void initializeNotificationInfo( final ExpandableNotificationRow row, NotificationInfo notificationInfoView) throws Exception { NotificationGuts guts = row.getGuts(); StatusBarNotification sbn = row.getEntry().getSbn(); String packageName = sbn.getPackageName(); // Settings link is only valid for notifications that specify a non-system user NotificationInfo.OnSettingsClickListener onSettingsClick = null; UserHandle userHandle = sbn.getUser(); PackageManager pmUser = CentralSurfaces.getPackageManagerForUser( mContext, userHandle.getIdentifier()); final NotificationInfo.OnAppSettingsClickListener onAppSettingsClick = (View v, Intent intent) -> { mMetricsLogger.action(MetricsProto.MetricsEvent.ACTION_APP_NOTE_SETTINGS); guts.resetFalsingCheck(); mNotificationActivityStarter.startNotificationGutsIntent(intent, sbn.getUid(), row); }; if (!userHandle.equals(UserHandle.ALL) || mLockscreenUserManager.getCurrentUserId() == UserHandle.USER_SYSTEM) { onSettingsClick = (View v, NotificationChannel channel, int appUid) -> { mMetricsLogger.action(MetricsProto.MetricsEvent.ACTION_NOTE_INFO); guts.resetFalsingCheck(); mOnSettingsClickListener.onSettingsClick(sbn.getKey()); startAppNotificationSettingsActivity(packageName, appUid, channel, row); }; } notificationInfoView.bindNotification( pmUser, mNotificationManager, mOnUserInteractionCallback, mChannelEditorDialogController, packageName, row.getEntry().getChannel(), row.getUniqueChannels(), row.getEntry(), onSettingsClick, onAppSettingsClick, mUiEventLogger, mDeviceProvisionedController.isDeviceProvisioned(), row.getIsNonblockable(), mHighPriorityProvider.isHighPriority(row.getEntry()), mAssistantFeedbackController, mMetricsLogger); } /** * Sets up the {@link PartialConversationInfo} inside the notification row's guts. * @param row view to set up the guts for * @param notificationInfoView view to set up/bind within {@code row} */ @VisibleForTesting void initializePartialConversationNotificationInfo( final ExpandableNotificationRow row, PartialConversationInfo notificationInfoView) throws Exception { NotificationGuts guts = row.getGuts(); StatusBarNotification sbn = row.getEntry().getSbn(); String packageName = sbn.getPackageName(); // Settings link is only valid for notifications that specify a non-system user NotificationInfo.OnSettingsClickListener onSettingsClick = null; UserHandle userHandle = sbn.getUser(); PackageManager pmUser = CentralSurfaces.getPackageManagerForUser( mContext, userHandle.getIdentifier()); if (!userHandle.equals(UserHandle.ALL) || mLockscreenUserManager.getCurrentUserId() == UserHandle.USER_SYSTEM) { onSettingsClick = (View v, NotificationChannel channel, int appUid) -> { mMetricsLogger.action(MetricsProto.MetricsEvent.ACTION_NOTE_INFO); guts.resetFalsingCheck(); mOnSettingsClickListener.onSettingsClick(sbn.getKey()); startAppNotificationSettingsActivity(packageName, appUid, channel, row); }; } notificationInfoView.bindNotification( pmUser, mNotificationManager, mChannelEditorDialogController, packageName, row.getEntry().getChannel(), row.getUniqueChannels(), row.getEntry(), onSettingsClick, mDeviceProvisionedController.isDeviceProvisioned(), row.getIsNonblockable()); } /** * Sets up the {@link ConversationInfo} inside the notification row's guts. * @param row view to set up the guts for * @param notificationInfoView view to set up/bind within {@code row} */ @VisibleForTesting void initializeConversationNotificationInfo( final ExpandableNotificationRow row, NotificationConversationInfo notificationInfoView) throws Exception { NotificationGuts guts = row.getGuts(); NotificationEntry entry = row.getEntry(); StatusBarNotification sbn = entry.getSbn(); String packageName = sbn.getPackageName(); // Settings link is only valid for notifications that specify a non-system user NotificationConversationInfo.OnSettingsClickListener onSettingsClick = null; UserHandle userHandle = sbn.getUser(); PackageManager pmUser = CentralSurfaces.getPackageManagerForUser( mContext, userHandle.getIdentifier()); final NotificationConversationInfo.OnAppSettingsClickListener onAppSettingsClick = (View v, Intent intent) -> { mMetricsLogger.action(MetricsProto.MetricsEvent.ACTION_APP_NOTE_SETTINGS); guts.resetFalsingCheck(); mNotificationActivityStarter.startNotificationGutsIntent(intent, sbn.getUid(), row); }; final NotificationConversationInfo.OnConversationSettingsClickListener onConversationSettingsListener = () -> { startConversationSettingsActivity(sbn.getUid(), row); }; if (!userHandle.equals(UserHandle.ALL) || mLockscreenUserManager.getCurrentUserId() == UserHandle.USER_SYSTEM) { onSettingsClick = (View v, NotificationChannel channel, int appUid) -> { mMetricsLogger.action(MetricsProto.MetricsEvent.ACTION_NOTE_INFO); guts.resetFalsingCheck(); mOnSettingsClickListener.onSettingsClick(sbn.getKey()); startAppNotificationSettingsActivity(packageName, appUid, channel, row); }; } ConversationIconFactory iconFactoryLoader = new ConversationIconFactory(mContext, mLauncherApps, pmUser, IconDrawableFactory.newInstance(mContext, false), mContext.getResources().getDimensionPixelSize( R.dimen.notification_guts_conversation_icon_size)); notificationInfoView.bindNotification( mShortcutManager, pmUser, mUserManager, mPeopleSpaceWidgetManager, mNotificationManager, mOnUserInteractionCallback, packageName, entry.getChannel(), entry, entry.getBubbleMetadata(), onSettingsClick, iconFactoryLoader, mContextTracker.getUserContext(), mDeviceProvisionedController.isDeviceProvisioned(), mMainHandler, mBgHandler, onConversationSettingsListener, mBubblesManagerOptional, mShadeController); } /** * Closes guts or notification menus that might be visible and saves any changes. * * @param removeLeavebehinds true if leavebehinds (e.g. snooze) should be closed. * @param force true if guts should be closed regardless of state (used for snooze only). * @param removeControls true if controls (e.g. info) should be closed. * @param x if closed based on touch location, this is the x touch location. * @param y if closed based on touch location, this is the y touch location. * @param resetMenu if any notification menus that might be revealed should be closed. */ public void closeAndSaveGuts(boolean removeLeavebehinds, boolean force, boolean removeControls, int x, int y, boolean resetMenu) { if (mNotificationGutsExposed != null) { mNotificationGutsExposed.removeCallbacks(mOpenRunnable); mNotificationGutsExposed.closeControls(removeLeavebehinds, removeControls, x, y, force); } if (resetMenu) { mListContainer.resetExposedMenuView(false /* animate */, true /* force */); } } /** * Returns the exposed NotificationGuts or null if none are exposed. */ public NotificationGuts getExposedGuts() { return mNotificationGutsExposed; } public void setExposedGuts(NotificationGuts guts) { mNotificationGutsExposed = guts; } public ExpandableNotificationRow.LongPressListener getNotificationLongClicker() { return this::openGuts; } /** * Opens guts on the given ExpandableNotificationRow {@code view}. This handles opening guts for * the normal half-swipe and long-press use cases via a circular reveal. When the blocking * helper needs to be shown on the row, this will skip the circular reveal. * * @param view ExpandableNotificationRow to open guts on * @param x x coordinate of origin of circular reveal * @param y y coordinate of origin of circular reveal * @param menuItem MenuItem the guts should display * @return true if guts was opened */ public boolean openGuts( View view, int x, int y, NotificationMenuRowPlugin.MenuItem menuItem) { if (menuItem.getGutsView() instanceof NotificationGuts.GutsContent) { NotificationGuts.GutsContent gutsView = (NotificationGuts.GutsContent) menuItem.getGutsView(); if (gutsView.needsFalsingProtection()) { if (mStatusBarStateController instanceof StatusBarStateControllerImpl) { ((StatusBarStateControllerImpl) mStatusBarStateController) .setLeaveOpenOnKeyguardHide(true); } Optional centralSurfacesOptional = mCentralSurfacesOptionalLazy.get(); if (centralSurfacesOptional.isPresent()) { Runnable r = () -> mMainHandler.post( () -> openGutsInternal(view, x, y, menuItem)); centralSurfacesOptional.get().executeRunnableDismissingKeyguard( r, null /* cancelAction */, false /* dismissShade */, true /* afterKeyguardGone */, true /* deferred */); return true; } /** * When {@link CentralSurfaces} doesn't exist, falling through to call * {@link #openGutsInternal(View,int,int,NotificationMenuRowPlugin.MenuItem)}. */ } } return openGutsInternal(view, x, y, menuItem); } @VisibleForTesting boolean openGutsInternal( View view, int x, int y, NotificationMenuRowPlugin.MenuItem menuItem) { if (!(view instanceof ExpandableNotificationRow)) { return false; } if (view.getWindowToken() == null) { Log.e(TAG, "Trying to show notification guts, but not attached to window"); return false; } final ExpandableNotificationRow row = (ExpandableNotificationRow) view; if (row.isNotificationRowLongClickable()) { view.performHapticFeedback(HapticFeedbackConstants.LONG_PRESS); } if (row.areGutsExposed()) { closeAndSaveGuts(false /* removeLeavebehind */, false /* force */, true /* removeControls */, -1 /* x */, -1 /* y */, true /* resetMenu */); return false; } row.ensureGutsInflated(); NotificationGuts guts = row.getGuts(); mNotificationGutsExposed = guts; if (!bindGuts(row, menuItem)) { // exception occurred trying to fill in all the data, bail. return false; } // Assume we are a status_bar_notification_row if (guts == null) { // This view has no guts. Examples are the more card or the dismiss all view return false; } // ensure that it's laid but not visible until actually laid out guts.setVisibility(View.INVISIBLE); // Post to ensure the the guts are properly laid out. mOpenRunnable = new Runnable() { @Override public void run() { if (row.getWindowToken() == null) { Log.e(TAG, "Trying to show notification guts in post(), but not attached to " + "window"); return; } guts.setVisibility(View.VISIBLE); final boolean needsFalsingProtection = (mStatusBarStateController.getState() == StatusBarState.KEYGUARD && !mAccessibilityManager.isTouchExplorationEnabled()); guts.openControls( x, y, needsFalsingProtection, row::onGutsOpened); if (mGutsListener != null) { mGutsListener.onGutsOpen(row.getEntry(), guts); } row.closeRemoteInput(); mListContainer.onHeightChanged(row, true /* needsAnimation */); mGutsMenuItem = menuItem; } }; guts.post(mOpenRunnable); return true; } /** * @param gutsListener the listener for open and close guts events */ public void setGutsListener(NotifGutsViewListener gutsListener) { mGutsListener = gutsListener; } public interface OnSettingsClickListener { public void onSettingsClick(String key); } }