/* * 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 android.service.notification; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.StringDef; import android.annotation.SystemApi; import android.app.Notification; import android.os.Build; import android.os.Bundle; import android.os.Parcel; import android.os.Parcelable; import android.os.UserHandle; import java.lang.annotation.Retention; import java.lang.annotation.RetentionPolicy; /** * Ranking updates from the Assistant. * * The updates are provides as a {@link Bundle} of signals, using the keys provided in this * class. * Each {@code KEY} specifies what type of data it supports and what kind of Adjustment it * realizes on the notification rankings. * * Notifications affected by the Adjustment will be re-ranked if necessary. * * @hide */ @SystemApi public final class Adjustment implements Parcelable { private final String mPackage; private final String mKey; private final CharSequence mExplanation; private final Bundle mSignals; private final int mUser; @Nullable private String mIssuer; /** @hide */ @StringDef (prefix = { "KEY_" }, value = { KEY_PEOPLE, KEY_SNOOZE_CRITERIA, KEY_GROUP_KEY, KEY_USER_SENTIMENT, KEY_CONTEXTUAL_ACTIONS, KEY_TEXT_REPLIES, KEY_IMPORTANCE, KEY_IMPORTANCE_PROPOSAL, KEY_SENSITIVE_CONTENT, KEY_RANKING_SCORE, KEY_NOT_CONVERSATION }) @Retention(RetentionPolicy.SOURCE) public @interface Keys {} /** * Data type: ArrayList of {@code String}, where each is a representation of a * {@link android.provider.ContactsContract.Contacts#CONTENT_LOOKUP_URI}. * See {@link android.app.Notification.Builder#addPerson(String)}. * @hide */ @SystemApi public static final String KEY_PEOPLE = "key_people"; /** * Parcelable {@code ArrayList} of {@link SnoozeCriterion}. These criteria may be visible to * users. If a user chooses to snooze a notification until one of these criterion, the * assistant will be notified via * {@link NotificationAssistantService#onNotificationSnoozedUntilContext}. */ public static final String KEY_SNOOZE_CRITERIA = "key_snooze_criteria"; /** * Data type: String. Used to change what {@link Notification#getGroup() group} a notification * belongs to. * @hide */ public static final String KEY_GROUP_KEY = "key_group_key"; /** * Data type: int, one of {@link NotificationListenerService.Ranking#USER_SENTIMENT_POSITIVE}, * {@link NotificationListenerService.Ranking#USER_SENTIMENT_NEUTRAL}, * {@link NotificationListenerService.Ranking#USER_SENTIMENT_NEGATIVE}. Used to express how * a user feels about notifications in the same {@link android.app.NotificationChannel} as * the notification represented by {@link #getKey()}. */ public static final String KEY_USER_SENTIMENT = "key_user_sentiment"; /** * Data type: ArrayList of {@link android.app.Notification.Action}. * Used to suggest contextual actions for a notification. * * @see Notification.Action.Builder#setContextual(boolean) */ public static final String KEY_CONTEXTUAL_ACTIONS = "key_contextual_actions"; /** * Data type: ArrayList of {@link CharSequence}. * Used to suggest smart replies for a notification. */ public static final String KEY_TEXT_REPLIES = "key_text_replies"; /** * Data type: int, one of importance values e.g. * {@link android.app.NotificationManager#IMPORTANCE_MIN}. * *
If used from * {@link NotificationAssistantService#onNotificationEnqueued(StatusBarNotification)}, and * received before the notification is posted, it can block a notification from appearing or * silence it. Importance adjustments received too late from * {@link NotificationAssistantService#onNotificationEnqueued(StatusBarNotification)} will be * ignored. *
*If used from * {@link NotificationAssistantService#adjustNotification(Adjustment)}, it can * visually demote or cancel a notification, but use this with care if they notification was * recently posted because the notification may already have made noise. *
*/ public static final String KEY_IMPORTANCE = "key_importance"; /** * Weaker than {@link #KEY_IMPORTANCE}, this adjustment suggests an importance rather than * mandates an importance change. * * A notification listener can interpet this suggestion to show the user a prompt to change * notification importance for the notification (or type, or app) moving forward. * * Data type: int, one of importance values e.g. * {@link android.app.NotificationManager#IMPORTANCE_MIN}. */ public static final String KEY_IMPORTANCE_PROPOSAL = "key_importance_proposal"; /** * Data type: boolean, when true it suggests that the content text of this notification is * sensitive. The system uses this information to improve privacy around the notification * content. In {@link Build.VERSION_CODES#VANILLA_ICE_CREAM}, sensitive notification content is * redacted from updates to most {@link NotificationListenerService * NotificationListenerServices}. Also if an app posts a sensitive notification while * {@link android.media.projection.MediaProjection screen-sharing} is active, that app's windows * are blocked from screen-sharing and a {@link android.widget.Toast Toast} is shown to inform * the user about this. */ public static final String KEY_SENSITIVE_CONTENT = "key_sensitive_content"; /** * Data type: float, a ranking score from 0 (lowest) to 1 (highest). * Used to rank notifications inside that fall under the same classification (i.e. alerting, * silenced). */ public static final String KEY_RANKING_SCORE = "key_ranking_score"; /** * Data type: boolean, when true it suggests this is NOT a conversation notification. * @hide */ @SystemApi public static final String KEY_NOT_CONVERSATION = "key_not_conversation"; /** * Create a notification adjustment. * * @param pkg The package of the notification. * @param key The notification key. * @param signals A bundle of signals that should inform notification display, ordering, and * interruptiveness. * @param explanation A human-readable justification for the adjustment. * @hide */ @SystemApi public Adjustment(String pkg, String key, Bundle signals, CharSequence explanation, int user) { mPackage = pkg; mKey = key; mSignals = signals; mExplanation = explanation; mUser = user; } /** * Create a notification adjustment. * * @param pkg The package of the notification. * @param key The notification key. * @param signals A bundle of signals that should inform notification display, ordering, and * interruptiveness. * @param explanation A human-readable justification for the adjustment. * @param userHandle User handle for for whose the adjustments will be applied. */ public Adjustment(@NonNull String pkg, @NonNull String key, @NonNull Bundle signals, @NonNull CharSequence explanation, @NonNull UserHandle userHandle) { mPackage = pkg; mKey = key; mSignals = signals; mExplanation = explanation; mUser = userHandle.getIdentifier(); } /** * @hide */ @SystemApi protected Adjustment(Parcel in) { if (in.readInt() == 1) { mPackage = in.readString(); } else { mPackage = null; } if (in.readInt() == 1) { mKey = in.readString(); } else { mKey = null; } if (in.readInt() == 1) { mExplanation = in.readCharSequence(); } else { mExplanation = null; } mSignals = in.readBundle(); mUser = in.readInt(); mIssuer = in.readString(); } public static final @android.annotation.NonNull Creator