/* * 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.app; import android.annotation.FlaggedApi; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.SystemApi; import android.annotation.TestApi; import android.app.NotificationManager.Importance; import android.compat.annotation.UnsupportedAppUsage; import android.content.ContentResolver; import android.content.Context; import android.content.Intent; import android.content.pm.ShortcutInfo; import android.media.AudioAttributes; import android.media.RingtoneManager; import android.net.Uri; import android.os.Parcel; import android.os.Parcelable; import android.os.VibrationEffect; import android.os.vibrator.persistence.VibrationXmlParser; import android.os.vibrator.persistence.VibrationXmlSerializer; import android.provider.Settings; import android.service.notification.NotificationListenerService; import android.text.TextUtils; import android.util.Log; import android.util.proto.ProtoOutputStream; import com.android.internal.util.Preconditions; import com.android.internal.util.XmlUtils; import com.android.modules.utils.TypedXmlPullParser; import com.android.modules.utils.TypedXmlSerializer; import org.json.JSONException; import org.json.JSONObject; import org.xmlpull.v1.XmlPullParser; import org.xmlpull.v1.XmlSerializer; import java.io.FileNotFoundException; import java.io.IOException; import java.io.PrintWriter; import java.io.StringReader; import java.io.StringWriter; import java.util.Arrays; import java.util.Objects; /** * A representation of settings that apply to a collection of similarly themed notifications. */ public final class NotificationChannel implements Parcelable { private static final String TAG = "NotificationChannel"; /** * The id of the default channel for an app. This id is reserved by the system. All * notifications posted from apps targeting {@link android.os.Build.VERSION_CODES#N_MR1} or * earlier without a notification channel specified are posted to this channel. */ public static final String DEFAULT_CHANNEL_ID = "miscellaneous"; /** * The formatter used by the system to create an id for notification * channels when it automatically creates conversation channels on behalf of an app. The format * string takes two arguments, in this order: the * {@link #getId()} of the original notification channel, and the * {@link ShortcutInfo#getId() id} of the conversation. * @hide */ public static final String CONVERSATION_CHANNEL_ID_FORMAT = "%1$s : %2$s"; /** * TODO: STOPSHIP remove * Conversation id to use for apps that aren't providing them yet. * @hide */ public static final String PLACEHOLDER_CONVERSATION_ID = ":placeholder_id"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing sound, like a tone picker * ({@link #setSound(Uri, AudioAttributes)}). */ public static final String EDIT_SOUND = "sound"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing vibration ({@link #enableVibration(boolean)}, * {@link #setVibrationPattern(long[])}). */ public static final String EDIT_VIBRATION = "vibration"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing importance ({@link #setImportance(int)}) and/or conversation * priority. */ public static final String EDIT_IMPORTANCE = "importance"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing behavior on devices that are locked or have a turned off * display ({@link #setLockscreenVisibility(int)}, {@link #enableLights(boolean)}, * {@link #setLightColor(int)}). */ public static final String EDIT_LOCKED_DEVICE = "locked"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing do not disturb bypass {(@link #setBypassDnd(boolean)}) . */ public static final String EDIT_ZEN = "zen"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing conversation settings (demoting or restoring a channel to * be a Conversation, changing bubble behavior, or setting the priority of a conversation). */ public static final String EDIT_CONVERSATION = "conversation"; /** * Extra value for {@link Settings#EXTRA_CHANNEL_FILTER_LIST}. Include to show fields * that have to do with editing launcher behavior (showing badges)}. */ public static final String EDIT_LAUNCHER = "launcher"; /** * The maximum length for text fields in a NotificationChannel. Fields will be truncated at this * limit. * @hide */ public static final int MAX_TEXT_LENGTH = 1000; /** * @hide */ public static final int MAX_VIBRATION_LENGTH = 1000; private static final String TAG_CHANNEL = "channel"; private static final String ATT_NAME = "name"; private static final String ATT_DESC = "desc"; private static final String ATT_ID = "id"; private static final String ATT_DELETED = "deleted"; private static final String ATT_PRIORITY = "priority"; private static final String ATT_VISIBILITY = "visibility"; private static final String ATT_IMPORTANCE = "importance"; private static final String ATT_LIGHTS = "lights"; private static final String ATT_LIGHT_COLOR = "light_color"; private static final String ATT_VIBRATION = "vibration"; private static final String ATT_VIBRATION_EFFECT = "vibration_effect"; private static final String ATT_VIBRATION_ENABLED = "vibration_enabled"; private static final String ATT_SOUND = "sound"; private static final String ATT_USAGE = "usage"; private static final String ATT_FLAGS = "flags"; private static final String ATT_CONTENT_TYPE = "content_type"; private static final String ATT_SHOW_BADGE = "show_badge"; private static final String ATT_USER_LOCKED = "locked"; /** * This attribute represents both foreground services and user initiated jobs in U+. * It was not renamed in U on purpose, in order to avoid creating an unnecessary migration path. */ private static final String ATT_FG_SERVICE_SHOWN = "fgservice"; private static final String ATT_GROUP = "group"; private static final String ATT_BLOCKABLE_SYSTEM = "blockable_system"; private static final String ATT_ALLOW_BUBBLE = "allow_bubbles"; private static final String ATT_ORIG_IMP = "orig_imp"; private static final String ATT_PARENT_CHANNEL = "parent"; private static final String ATT_CONVERSATION_ID = "conv_id"; private static final String ATT_IMP_CONVERSATION = "imp_conv"; private static final String ATT_DEMOTE = "dem"; private static final String ATT_DELETED_TIME_MS = "del_time"; private static final String DELIMITER = ","; /** * @hide */ public static final int USER_LOCKED_PRIORITY = 0x00000001; /** * @hide */ public static final int USER_LOCKED_VISIBILITY = 0x00000002; /** * @hide */ public static final int USER_LOCKED_IMPORTANCE = 0x00000004; /** * @hide */ public static final int USER_LOCKED_LIGHTS = 0x00000008; /** * @hide */ public static final int USER_LOCKED_VIBRATION = 0x00000010; /** * @hide */ @SystemApi public static final int USER_LOCKED_SOUND = 0x00000020; /** * @hide */ public static final int USER_LOCKED_SHOW_BADGE = 0x00000080; /** * @hide */ public static final int USER_LOCKED_ALLOW_BUBBLE = 0x00000100; /** * @hide */ public static final int[] LOCKABLE_FIELDS = new int[] { USER_LOCKED_PRIORITY, USER_LOCKED_VISIBILITY, USER_LOCKED_IMPORTANCE, USER_LOCKED_LIGHTS, USER_LOCKED_VIBRATION, USER_LOCKED_SOUND, USER_LOCKED_SHOW_BADGE, USER_LOCKED_ALLOW_BUBBLE }; /** * @hide */ public static final int DEFAULT_ALLOW_BUBBLE = -1; /** * @hide */ public static final int ALLOW_BUBBLE_ON = 1; /** * @hide */ public static final int ALLOW_BUBBLE_OFF = 0; private static final int DEFAULT_LIGHT_COLOR = 0; private static final int DEFAULT_VISIBILITY = NotificationManager.VISIBILITY_NO_OVERRIDE; private static final int DEFAULT_IMPORTANCE = NotificationManager.IMPORTANCE_UNSPECIFIED; private static final boolean DEFAULT_DELETED = false; private static final boolean DEFAULT_SHOW_BADGE = true; private static final long DEFAULT_DELETION_TIME_MS = -1; @UnsupportedAppUsage private String mId; private String mName; private String mDesc; private int mImportance = DEFAULT_IMPORTANCE; private int mOriginalImportance = DEFAULT_IMPORTANCE; private boolean mBypassDnd; private int mLockscreenVisibility = DEFAULT_VISIBILITY; private Uri mSound = Settings.System.DEFAULT_NOTIFICATION_URI; private boolean mSoundRestored = false; private boolean mLights; private int mLightColor = DEFAULT_LIGHT_COLOR; private long[] mVibrationPattern; private VibrationEffect mVibrationEffect; // Bitwise representation of fields that have been changed by the user, preventing the app from // making changes to these fields. private int mUserLockedFields; private boolean mUserVisibleTaskShown; private boolean mVibrationEnabled; private boolean mShowBadge = DEFAULT_SHOW_BADGE; private boolean mDeleted = DEFAULT_DELETED; private String mGroup; private AudioAttributes mAudioAttributes = Notification.AUDIO_ATTRIBUTES_DEFAULT; // If this is a blockable system notification channel. private boolean mBlockableSystem = false; private int mAllowBubbles = DEFAULT_ALLOW_BUBBLE; private boolean mImportanceLockedDefaultApp; private String mParentId = null; private String mConversationId = null; private boolean mDemoted = false; private boolean mImportantConvo = false; private long mDeletedTime = DEFAULT_DELETION_TIME_MS; /** Do not (de)serialize this value: it only affects logic in system_server and that logic * is reset on each boot {@link NotificationAttentionHelper#buzzBeepBlinkLocked}. */ private long mLastNotificationUpdateTimeMs = 0; /** * Creates a notification channel. * * @param id The id of the channel. Must be unique per package. The value may be truncated if * it is too long. * @param name The user visible name of the channel. You can rename this channel when the system * locale changes by listening for the {@link Intent#ACTION_LOCALE_CHANGED} * broadcast. The recommended maximum length is 40 characters; the value may be * truncated if it is too long. * @param importance The importance of the channel. This controls how interruptive notifications * posted to this channel are. */ public NotificationChannel(String id, CharSequence name, @Importance int importance) { this.mId = getTrimmedString(id); this.mName = name != null ? getTrimmedString(name.toString()) : null; this.mImportance = importance; } /** * @hide */ protected NotificationChannel(Parcel in) { if (in.readByte() != 0) { mId = getTrimmedString(in.readString()); } else { mId = null; } if (in.readByte() != 0) { mName = getTrimmedString(in.readString()); } else { mName = null; } if (in.readByte() != 0) { mDesc = getTrimmedString(in.readString()); } else { mDesc = null; } mImportance = in.readInt(); mBypassDnd = in.readByte() != 0; mLockscreenVisibility = in.readInt(); if (in.readByte() != 0) { mSound = Uri.CREATOR.createFromParcel(in); mSound = Uri.parse(getTrimmedString(mSound.toString())); } else { mSound = null; } mLights = in.readByte() != 0; mVibrationPattern = in.createLongArray(); if (mVibrationPattern != null && mVibrationPattern.length > MAX_VIBRATION_LENGTH) { mVibrationPattern = Arrays.copyOf(mVibrationPattern, MAX_VIBRATION_LENGTH); } if (Flags.notificationChannelVibrationEffectApi()) { mVibrationEffect = in.readInt() != 0 ? VibrationEffect.CREATOR.createFromParcel(in) : null; } mUserLockedFields = in.readInt(); mUserVisibleTaskShown = in.readByte() != 0; mVibrationEnabled = in.readByte() != 0; mShowBadge = in.readByte() != 0; mDeleted = in.readByte() != 0; if (in.readByte() != 0) { mGroup = getTrimmedString(in.readString()); } else { mGroup = null; } mAudioAttributes = in.readInt() > 0 ? AudioAttributes.CREATOR.createFromParcel(in) : null; mLightColor = in.readInt(); mBlockableSystem = in.readBoolean(); mAllowBubbles = in.readInt(); mOriginalImportance = in.readInt(); mParentId = getTrimmedString(in.readString()); mConversationId = getTrimmedString(in.readString()); mDemoted = in.readBoolean(); mImportantConvo = in.readBoolean(); mDeletedTime = in.readLong(); mImportanceLockedDefaultApp = in.readBoolean(); } @Override public void writeToParcel(Parcel dest, int flags) { if (mId != null) { dest.writeByte((byte) 1); dest.writeString(mId); } else { dest.writeByte((byte) 0); } if (mName != null) { dest.writeByte((byte) 1); dest.writeString(mName); } else { dest.writeByte((byte) 0); } if (mDesc != null) { dest.writeByte((byte) 1); dest.writeString(mDesc); } else { dest.writeByte((byte) 0); } dest.writeInt(mImportance); dest.writeByte(mBypassDnd ? (byte) 1 : (byte) 0); dest.writeInt(mLockscreenVisibility); if (mSound != null) { dest.writeByte((byte) 1); mSound.writeToParcel(dest, 0); } else { dest.writeByte((byte) 0); } dest.writeByte(mLights ? (byte) 1 : (byte) 0); dest.writeLongArray(mVibrationPattern); if (Flags.notificationChannelVibrationEffectApi()) { if (mVibrationEffect != null) { dest.writeInt(1); mVibrationEffect.writeToParcel(dest, /* flags= */ 0); } else { dest.writeInt(0); } } dest.writeInt(mUserLockedFields); dest.writeByte(mUserVisibleTaskShown ? (byte) 1 : (byte) 0); dest.writeByte(mVibrationEnabled ? (byte) 1 : (byte) 0); dest.writeByte(mShowBadge ? (byte) 1 : (byte) 0); dest.writeByte(mDeleted ? (byte) 1 : (byte) 0); if (mGroup != null) { dest.writeByte((byte) 1); dest.writeString(mGroup); } else { dest.writeByte((byte) 0); } if (mAudioAttributes != null) { dest.writeInt(1); mAudioAttributes.writeToParcel(dest, 0); } else { dest.writeInt(0); } dest.writeInt(mLightColor); dest.writeBoolean(mBlockableSystem); dest.writeInt(mAllowBubbles); dest.writeInt(mOriginalImportance); dest.writeString(mParentId); dest.writeString(mConversationId); dest.writeBoolean(mDemoted); dest.writeBoolean(mImportantConvo); dest.writeLong(mDeletedTime); dest.writeBoolean(mImportanceLockedDefaultApp); } /** * @hide */ public NotificationChannel copy() { NotificationChannel copy = new NotificationChannel(mId, mName, mImportance); copy.setDescription(mDesc); copy.setBypassDnd(mBypassDnd); copy.setLockscreenVisibility(mLockscreenVisibility); copy.setSound(mSound, mAudioAttributes); copy.setLightColor(mLightColor); copy.enableLights(mLights); copy.setVibrationPattern(mVibrationPattern); if (Flags.notificationChannelVibrationEffectApi()) { copy.setVibrationEffect(mVibrationEffect); } copy.lockFields(mUserLockedFields); copy.setUserVisibleTaskShown(mUserVisibleTaskShown); copy.enableVibration(mVibrationEnabled); copy.setShowBadge(mShowBadge); copy.setDeleted(mDeleted); copy.setGroup(mGroup); copy.setBlockable(mBlockableSystem); copy.setAllowBubbles(mAllowBubbles); copy.setOriginalImportance(mOriginalImportance); copy.setConversationId(mParentId, mConversationId); copy.setDemoted(mDemoted); copy.setImportantConversation(mImportantConvo); copy.setDeletedTimeMs(mDeletedTime); copy.setImportanceLockedByCriticalDeviceFunction(mImportanceLockedDefaultApp); copy.setLastNotificationUpdateTimeMs(mLastNotificationUpdateTimeMs); return copy; } /** * @hide */ @TestApi public void lockFields(int field) { mUserLockedFields |= field; } /** * @hide */ public void unlockFields(int field) { mUserLockedFields &= ~field; } /** * @hide */ @TestApi public void setUserVisibleTaskShown(boolean shown) { mUserVisibleTaskShown = shown; } /** * @hide */ @TestApi public void setDeleted(boolean deleted) { mDeleted = deleted; } /** * @hide */ @TestApi public void setDeletedTimeMs(long time) { mDeletedTime = time; } /** * @hide */ @TestApi public void setImportantConversation(boolean importantConvo) { mImportantConvo = importantConvo; } /** * Allows users to block notifications sent through this channel, if this channel belongs to * a package that otherwise would have notifications "fixed" as enabled. * * If the channel does not belong to a package that has a fixed notification permission, this * method does nothing, since such channels are blockable by default and cannot be set to be * unblockable. * @param blockable if {@code true}, allows users to block notifications on this channel. */ public void setBlockable(boolean blockable) { mBlockableSystem = blockable; } // Modifiable by apps post channel creation /** * Sets the user visible name of this channel. * *

The recommended maximum length is 40 characters; the value may be truncated if it is too * long. */ public void setName(CharSequence name) { mName = name != null ? getTrimmedString(name.toString()) : null; } /** * Sets the user visible description of this channel. * *

The recommended maximum length is 300 characters; the value may be truncated if it is too * long. */ public void setDescription(String description) { mDesc = getTrimmedString(description); } private String getTrimmedString(String input) { if (input != null && input.length() > MAX_TEXT_LENGTH) { return input.substring(0, MAX_TEXT_LENGTH); } return input; } /** * @hide */ public void setId(String id) { mId = id; } // Modifiable by apps on channel creation. /** * Sets what group this channel belongs to. * * Group information is only used for presentation, not for behavior. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}, unless the * channel is not currently part of a group. * * @param groupId the id of a group created by * {@link NotificationManager#createNotificationChannelGroup(NotificationChannelGroup)}. */ public void setGroup(String groupId) { this.mGroup = groupId; } /** * Sets whether notifications posted to this channel can appear as application icon badges * in a Launcher. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. * * @param showBadge true if badges should be allowed to be shown. */ public void setShowBadge(boolean showBadge) { this.mShowBadge = showBadge; } /** * Sets the sound that should be played for notifications posted to this channel and its * audio attributes. Notification channels with an {@link #getImportance() importance} of at * least {@link NotificationManager#IMPORTANCE_DEFAULT} should have a sound. * * Note: An app-specific sound can be provided in the Uri parameter, but because channels are * persistent for the duration of the app install, and are backed up and restored, the Uri * should be stable. For this reason it is not recommended to use a * {@link ContentResolver#SCHEME_ANDROID_RESOURCE} uri, as resource ids can change on app * upgrade. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. */ public void setSound(Uri sound, AudioAttributes audioAttributes) { this.mSound = sound; this.mAudioAttributes = audioAttributes; } /** * Sets whether notifications posted to this channel should display notification lights, * on devices that support that feature. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. */ public void enableLights(boolean lights) { this.mLights = lights; } /** * Sets the notification light color for notifications posted to this channel, if lights are * {@link #enableLights(boolean) enabled} on this channel and the device supports that feature. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. */ public void setLightColor(int argb) { this.mLightColor = argb; } /** * Sets whether notification posted to this channel should vibrate. The vibration pattern can * be set with {@link #setVibrationPattern(long[])}. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. */ public void enableVibration(boolean vibration) { this.mVibrationEnabled = vibration; } /** * Sets the vibration pattern for notifications posted to this channel. If the provided * pattern is valid (non-null, non-empty with at least 1 non-zero value), will enable vibration * on this channel (equivalent to calling {@link #enableVibration(boolean)} with {@code true}). * Otherwise, vibration will be disabled unless {@link #enableVibration(boolean)} is * used with {@code true}, in which case the default vibration will be used. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. */ public void setVibrationPattern(long[] vibrationPattern) { this.mVibrationEnabled = vibrationPattern != null && vibrationPattern.length > 0; this.mVibrationPattern = vibrationPattern; if (Flags.notificationChannelVibrationEffectApi()) { try { this.mVibrationEffect = VibrationEffect.createWaveform(vibrationPattern, /* repeat= */ -1); } catch (IllegalArgumentException | NullPointerException e) { this.mVibrationEffect = null; } } } /** * Sets a {@link VibrationEffect} for notifications posted to this channel. If the * provided effect is non-null, will enable vibration on this channel (equivalent * to calling {@link #enableVibration(boolean)} with {@code true}). Otherwise * vibration will be disabled unless {@link #enableVibration(boolean)} is used with * {@code true}, in which case the default vibration will be used. * *

The effect passed here will be returned from {@link #getVibrationEffect()}. * If the provided {@link VibrationEffect} is an equivalent to a wave-form * vibration pattern, the equivalent wave-form pattern will be returned from * {@link #getVibrationPattern()}. * *

Note that some {@link VibrationEffect}s may not be playable on some devices. * In cases where such an effect is passed here, vibration will still be enabled * for the channel, but the default vibration will be used. Nonetheless, the * provided effect will be stored and be returned from {@link #getVibrationEffect} * calls, and could be used by the same channel on a different device, for example, * in cases the user backs up and restores to a device that does have the ability * to play the effect, where that effect will be used instead of the default. To * avoid such issues that could make the vibration behavior of your notification * channel differ among different devices, it's recommended that you avoid * vibration effect primitives, as the support for them differs widely among * devices (read {@link VibrationEffect.Composition} for more on vibration * primitives). * *

Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. * * @see #getVibrationEffect() * @see android.os.Vibrator#areEffectsSupported(int...) * @see android.os.Vibrator#arePrimitivesSupported(int...) */ @FlaggedApi(Flags.FLAG_NOTIFICATION_CHANNEL_VIBRATION_EFFECT_API) public void setVibrationEffect(@Nullable VibrationEffect effect) { this.mVibrationEnabled = effect != null; this.mVibrationEffect = effect; this.mVibrationPattern = effect == null ? null : effect.computeCreateWaveformOffOnTimingsOrNull(); } /** * Sets the level of interruption of this notification channel. * * Only modifiable before the channel is submitted to * {@link NotificationManager#createNotificationChannel(NotificationChannel)}. * * @param importance the amount the user should be interrupted by * notifications from this channel. */ public void setImportance(@Importance int importance) { this.mImportance = importance; } // Modifiable by a notification ranker. /** * Sets whether or not notifications posted to this channel can interrupt the user in * {@link android.app.NotificationManager.Policy#INTERRUPTION_FILTER_PRIORITY} mode. * * Only modifiable by the system and notification ranker. */ public void setBypassDnd(boolean bypassDnd) { this.mBypassDnd = bypassDnd; } /** * Sets whether notifications posted to this channel appear on the lockscreen or not, and if so, * whether they appear in a redacted form. See e.g. {@link Notification#VISIBILITY_SECRET}. * * Only modifiable by the system and notification ranker. */ public void setLockscreenVisibility(int lockscreenVisibility) { this.mLockscreenVisibility = lockscreenVisibility; } /** * As of Android 11 this value is no longer respected. * @see #canBubble() * @see Notification#getBubbleMetadata() */ public void setAllowBubbles(boolean allowBubbles) { mAllowBubbles = allowBubbles ? ALLOW_BUBBLE_ON : ALLOW_BUBBLE_OFF; } /** * @hide */ public void setAllowBubbles(int allowed) { mAllowBubbles = allowed; } /** * Sets this channel as being converastion-centric. Different settings and functionality may be * exposed for conversation-centric channels. * * @param parentChannelId The {@link #getId()} id} of the generic channel that notifications of * this type would be posted to in absence of a specific conversation id. * For example, if this channel represents 'Messages from Person A', the * parent channel would be 'Messages.' * @param conversationId The {@link ShortcutInfo#getId()} of the shortcut representing this * channel's conversation. */ public void setConversationId(@NonNull String parentChannelId, @NonNull String conversationId) { mParentId = parentChannelId; mConversationId = conversationId; } /** * Returns the id of this channel. */ public String getId() { return mId; } /** * Returns the user visible name of this channel. */ public CharSequence getName() { return mName; } /** * Returns the user visible description of this channel. */ public String getDescription() { return mDesc; } /** * Returns the user specified importance e.g. {@link NotificationManager#IMPORTANCE_LOW} for * notifications posted to this channel. Note: This value might be > * {@link NotificationManager#IMPORTANCE_NONE}, but notifications posted to this channel will * not be shown to the user if the parent {@link NotificationChannelGroup} or app is blocked. * See {@link NotificationChannelGroup#isBlocked()} and * {@link NotificationManager#areNotificationsEnabled()}. */ public int getImportance() { return mImportance; } /** * Whether or not notifications posted to this channel can bypass the Do Not Disturb * {@link NotificationManager#INTERRUPTION_FILTER_PRIORITY} mode when the active policy allows * priority channels to bypass notification filtering. */ public boolean canBypassDnd() { return mBypassDnd; } /** * Whether or not this channel represents a conversation. */ public boolean isConversation() { return !TextUtils.isEmpty(getConversationId()); } /** * Whether or not notifications in this conversation are considered important. * *

Important conversations may get special visual treatment, and might be able to bypass DND. * *

This is only valid for channels that represent conversations, that is, * where {@link #isConversation()} is true. */ public boolean isImportantConversation() { return mImportantConvo; } /** * Returns the notification sound for this channel. */ public Uri getSound() { return mSound; } /** * Returns the audio attributes for sound played by notifications posted to this channel. */ public AudioAttributes getAudioAttributes() { return mAudioAttributes; } /** * Returns whether notifications posted to this channel trigger notification lights. */ public boolean shouldShowLights() { return mLights; } /** * Returns the notification light color for notifications posted to this channel. Irrelevant * unless {@link #shouldShowLights()}. */ public int getLightColor() { return mLightColor; } /** * Returns whether notifications posted to this channel always vibrate. */ public boolean shouldVibrate() { return mVibrationEnabled; } /** * Returns the vibration pattern for notifications posted to this channel. Will be ignored if * vibration is not enabled ({@link #shouldVibrate()}). */ public long[] getVibrationPattern() { return mVibrationPattern; } /** * Returns the {@link VibrationEffect} for notifications posted to this channel. * The returned effect is derived from either the effect provided in the * {@link #setVibrationEffect(VibrationEffect)} method, or the equivalent vibration effect * of the pattern set via the {@link #setVibrationPattern(long[])} method, based on setter * method that was called last. * * The returned effect will be ignored in one of the following cases: *

* * @return the {@link VibrationEffect} set via {@link * #setVibrationEffect(VibrationEffect)}, or the equivalent of the * vibration set via {@link #setVibrationPattern(long[])}. * * @see VibrationEffect#createWaveform(long[], int) */ @FlaggedApi(Flags.FLAG_NOTIFICATION_CHANNEL_VIBRATION_EFFECT_API) @Nullable public VibrationEffect getVibrationEffect() { return mVibrationEffect; } /** * Returns whether or not notifications posted to this channel are shown on the lockscreen in * full or redacted form. */ public int getLockscreenVisibility() { return mLockscreenVisibility; } /** * Returns whether notifications posted to this channel can appear as badges in a Launcher * application. * * Note that badging may be disabled for other reasons. */ public boolean canShowBadge() { return mShowBadge; } /** * Returns what group this channel belongs to. * * This is used only for visually grouping channels in the UI. */ public String getGroup() { return mGroup; } /** * Returns whether notifications posted to this channel are allowed to display outside of the * notification shade, in a floating window on top of other apps. * * @see Notification#getBubbleMetadata() */ public boolean canBubble() { return mAllowBubbles == ALLOW_BUBBLE_ON; } /** * @hide */ public int getAllowBubbles() { return mAllowBubbles; } /** * Returns the {@link #getId() id} of the parent notification channel to this channel, if it's * a conversation related channel. See {@link #setConversationId(String, String)}. */ public @Nullable String getParentChannelId() { return mParentId; } /** * Returns the {@link ShortcutInfo#getId() id} of the conversation backing this channel, if it's * associated with a conversation. See {@link #setConversationId(String, String)}. */ public @Nullable String getConversationId() { return mConversationId; } /** * @hide */ @SystemApi public boolean isDeleted() { return mDeleted; } /** * @hide */ public long getDeletedTimeMs() { return mDeletedTime; } /** * @hide */ @SystemApi public int getUserLockedFields() { return mUserLockedFields; } /** * @hide */ public boolean isUserVisibleTaskShown() { return mUserVisibleTaskShown; } /** * Returns whether this channel is always blockable, even if the app is 'fixed' as * non-blockable. */ public boolean isBlockable() { return mBlockableSystem; } /** * @hide */ @TestApi public void setImportanceLockedByCriticalDeviceFunction(boolean locked) { mImportanceLockedDefaultApp = locked; } /** * @hide */ @TestApi public boolean isImportanceLockedByCriticalDeviceFunction() { return mImportanceLockedDefaultApp; } /** * @hide */ @TestApi public int getOriginalImportance() { return mOriginalImportance; } /** * @hide */ @TestApi public void setOriginalImportance(int importance) { mOriginalImportance = importance; } /** * @hide */ @TestApi public void setDemoted(boolean demoted) { mDemoted = demoted; } /** * Returns whether the user has decided that this channel does not represent a conversation. The * value will always be false for channels that never claimed to be conversations - that is, * for channels where {@link #getConversationId()} and {@link #getParentChannelId()} are empty. */ public boolean isDemoted() { return mDemoted; } /** * Returns whether the user has chosen the importance of this channel, either to affirm the * initial selection from the app, or changed it to be higher or lower. * @see #getImportance() */ public boolean hasUserSetImportance() { return (mUserLockedFields & USER_LOCKED_IMPORTANCE) != 0; } /** * Returns whether the user has chosen the sound of this channel. * @see #getSound() */ public boolean hasUserSetSound() { return (mUserLockedFields & USER_LOCKED_SOUND) != 0; } /** * Returns the time of the notification post or last update for this channel. * @return time of post / last update * @hide */ public long getLastNotificationUpdateTimeMs() { return mLastNotificationUpdateTimeMs; } /** * Sets the time of the notification post or last update for this channel. * @hide */ public void setLastNotificationUpdateTimeMs(long updateTimeMs) { mLastNotificationUpdateTimeMs = updateTimeMs; } /** * @hide */ public void populateFromXmlForRestore(XmlPullParser parser, boolean pkgInstalled, Context context) { populateFromXml(XmlUtils.makeTyped(parser), true, pkgInstalled, context); } /** * @hide */ @SystemApi public void populateFromXml(XmlPullParser parser) { populateFromXml(XmlUtils.makeTyped(parser), false, true, null); } /** * If {@param forRestore} is true, {@param Context} MUST be non-null. */ private void populateFromXml(TypedXmlPullParser parser, boolean forRestore, boolean pkgInstalled, @Nullable Context context) { Preconditions.checkArgument(!forRestore || context != null, "forRestore is true but got null context"); // Name, id, and importance are set in the constructor. setDescription(parser.getAttributeValue(null, ATT_DESC)); setBypassDnd(Notification.PRIORITY_DEFAULT != safeInt(parser, ATT_PRIORITY, Notification.PRIORITY_DEFAULT)); setLockscreenVisibility(safeInt(parser, ATT_VISIBILITY, DEFAULT_VISIBILITY)); Uri sound = safeUri(parser, ATT_SOUND); final AudioAttributes audioAttributes = safeAudioAttributes(parser); final int usage = audioAttributes.getUsage(); setSound(forRestore ? restoreSoundUri(context, sound, pkgInstalled, usage) : sound, audioAttributes); enableLights(safeBool(parser, ATT_LIGHTS, false)); setLightColor(safeInt(parser, ATT_LIGHT_COLOR, DEFAULT_LIGHT_COLOR)); // Set the pattern before the effect, so that we can properly handle cases where the pattern // is null, but the effect is not null (i.e. for non-waveform VibrationEffects - the ones // which cannot be represented as a vibration pattern). setVibrationPattern(safeLongArray(parser, ATT_VIBRATION, null)); if (Flags.notificationChannelVibrationEffectApi()) { VibrationEffect vibrationEffect = safeVibrationEffect(parser, ATT_VIBRATION_EFFECT); if (vibrationEffect != null) { // Restore the effect only if it is not null. This allows to avoid undoing a // `setVibrationPattern` call above, if that was done with a non-null pattern // (e.g. back up from a version that did not support `setVibrationEffect`). setVibrationEffect(vibrationEffect); } } enableVibration(safeBool(parser, ATT_VIBRATION_ENABLED, false)); setShowBadge(safeBool(parser, ATT_SHOW_BADGE, false)); setDeleted(safeBool(parser, ATT_DELETED, false)); setDeletedTimeMs(XmlUtils.readLongAttribute( parser, ATT_DELETED_TIME_MS, DEFAULT_DELETION_TIME_MS)); setGroup(parser.getAttributeValue(null, ATT_GROUP)); lockFields(safeInt(parser, ATT_USER_LOCKED, 0)); setUserVisibleTaskShown(safeBool(parser, ATT_FG_SERVICE_SHOWN, false)); setBlockable(safeBool(parser, ATT_BLOCKABLE_SYSTEM, false)); setAllowBubbles(safeInt(parser, ATT_ALLOW_BUBBLE, DEFAULT_ALLOW_BUBBLE)); setOriginalImportance(safeInt(parser, ATT_ORIG_IMP, DEFAULT_IMPORTANCE)); setConversationId(parser.getAttributeValue(null, ATT_PARENT_CHANNEL), parser.getAttributeValue(null, ATT_CONVERSATION_ID)); setDemoted(safeBool(parser, ATT_DEMOTE, false)); setImportantConversation(safeBool(parser, ATT_IMP_CONVERSATION, false)); } /** * Returns whether the sound for this channel was successfully restored * from backup. * @return false if the sound was not restored successfully. true otherwise (default value) * @hide */ public boolean isSoundRestored() { return mSoundRestored; } @Nullable private Uri getCanonicalizedSoundUri(ContentResolver contentResolver, @NonNull Uri uri) { if (Settings.System.DEFAULT_NOTIFICATION_URI.equals(uri)) { return uri; } if (ContentResolver.SCHEME_ANDROID_RESOURCE.equals(uri.getScheme())) { try { contentResolver.getResourceId(uri); return uri; } catch (FileNotFoundException e) { return null; } } if (ContentResolver.SCHEME_FILE.equals(uri.getScheme())) { return uri; } return contentResolver.canonicalize(uri); } @Nullable private Uri getUncanonicalizedSoundUri( ContentResolver contentResolver, @NonNull Uri uri, int usage) { if (Settings.System.DEFAULT_NOTIFICATION_URI.equals(uri) || ContentResolver.SCHEME_ANDROID_RESOURCE.equals(uri.getScheme()) || ContentResolver.SCHEME_FILE.equals(uri.getScheme())) { return uri; } int ringtoneType = 0; // Consistent with UI(SoundPreferenceController.handlePreferenceTreeClick). if (AudioAttributes.USAGE_ALARM == usage) { ringtoneType = RingtoneManager.TYPE_ALARM; } else if (AudioAttributes.USAGE_NOTIFICATION_RINGTONE == usage) { ringtoneType = RingtoneManager.TYPE_RINGTONE; } else { ringtoneType = RingtoneManager.TYPE_NOTIFICATION; } try { return RingtoneManager.getRingtoneUriForRestore( contentResolver, uri.toString(), ringtoneType); } catch (Exception e) { Log.e(TAG, "Failed to uncanonicalized sound uri for " + uri + " " + e); return Settings.System.DEFAULT_NOTIFICATION_URI; } } /** * Restore/validate sound Uri from backup * @param context The Context * @param uri The sound Uri to restore * @param pkgInstalled If the parent package is installed * @return restored and validated Uri * @hide */ @Nullable public Uri restoreSoundUri( Context context, @Nullable Uri uri, boolean pkgInstalled, int usage) { if (uri == null || Uri.EMPTY.equals(uri)) { return null; } ContentResolver contentResolver = context.getContentResolver(); // There are backups out there with uncanonical uris (because we fixed this after // shipping). If uncanonical uris are given to MediaProvider.uncanonicalize it won't // verify the uri against device storage and we'll possibly end up with a broken uri. // We then canonicalize the uri to uncanonicalize it back, which means we properly check // the uri and in the case of not having the resource we end up with the default - better // than broken. As a side effect we'll canonicalize already canonicalized uris, this is fine // according to the docs because canonicalize method has to handle canonical uris as well. Uri canonicalizedUri = getCanonicalizedSoundUri(contentResolver, uri); if (canonicalizedUri == null) { // Uri failed to restore with package installed if (!mSoundRestored && pkgInstalled) { mSoundRestored = true; // We got a null because the uri in the backup does not exist here, so we return // default return Settings.System.DEFAULT_NOTIFICATION_URI; } else { // Flag as unrestored and try again later (on package install) mSoundRestored = false; return uri; } } mSoundRestored = true; return getUncanonicalizedSoundUri(contentResolver, canonicalizedUri, usage); } /** * @hide */ @SystemApi public void writeXml(XmlSerializer out) throws IOException { writeXml(XmlUtils.makeTyped(out), false, null); } /** * @hide */ public void writeXmlForBackup(XmlSerializer out, Context context) throws IOException { writeXml(XmlUtils.makeTyped(out), true, context); } private Uri getSoundForBackup(Context context) { Uri sound = getSound(); if (sound == null || Uri.EMPTY.equals(sound)) { return null; } Uri canonicalSound = getCanonicalizedSoundUri(context.getContentResolver(), sound); if (canonicalSound == null) { // The content provider does not support canonical uris so we backup the default return Settings.System.DEFAULT_NOTIFICATION_URI; } return canonicalSound; } /** * If {@param forBackup} is true, {@param Context} MUST be non-null. */ private void writeXml(TypedXmlSerializer out, boolean forBackup, @Nullable Context context) throws IOException { Preconditions.checkArgument(!forBackup || context != null, "forBackup is true but got null context"); out.startTag(null, TAG_CHANNEL); out.attribute(null, ATT_ID, getId()); if (getName() != null) { out.attribute(null, ATT_NAME, getName().toString()); } if (getDescription() != null) { out.attribute(null, ATT_DESC, getDescription()); } if (getImportance() != DEFAULT_IMPORTANCE) { out.attributeInt(null, ATT_IMPORTANCE, getImportance()); } if (canBypassDnd()) { out.attributeInt(null, ATT_PRIORITY, Notification.PRIORITY_MAX); } if (getLockscreenVisibility() != DEFAULT_VISIBILITY) { out.attributeInt(null, ATT_VISIBILITY, getLockscreenVisibility()); } Uri sound = forBackup ? getSoundForBackup(context) : getSound(); if (sound != null) { out.attribute(null, ATT_SOUND, sound.toString()); } if (getAudioAttributes() != null) { out.attributeInt(null, ATT_USAGE, getAudioAttributes().getUsage()); out.attributeInt(null, ATT_CONTENT_TYPE, getAudioAttributes().getContentType()); out.attributeInt(null, ATT_FLAGS, getAudioAttributes().getFlags()); } if (shouldShowLights()) { out.attributeBoolean(null, ATT_LIGHTS, shouldShowLights()); } if (getLightColor() != DEFAULT_LIGHT_COLOR) { out.attributeInt(null, ATT_LIGHT_COLOR, getLightColor()); } if (shouldVibrate()) { out.attributeBoolean(null, ATT_VIBRATION_ENABLED, shouldVibrate()); } if (getVibrationPattern() != null) { out.attribute(null, ATT_VIBRATION, longArrayToString(getVibrationPattern())); } if (getVibrationEffect() != null) { out.attribute(null, ATT_VIBRATION_EFFECT, vibrationToString(getVibrationEffect())); } if (getUserLockedFields() != 0) { out.attributeInt(null, ATT_USER_LOCKED, getUserLockedFields()); } if (isUserVisibleTaskShown()) { out.attributeBoolean(null, ATT_FG_SERVICE_SHOWN, isUserVisibleTaskShown()); } if (canShowBadge()) { out.attributeBoolean(null, ATT_SHOW_BADGE, canShowBadge()); } if (isDeleted()) { out.attributeBoolean(null, ATT_DELETED, isDeleted()); } if (getDeletedTimeMs() >= 0) { out.attributeLong(null, ATT_DELETED_TIME_MS, getDeletedTimeMs()); } if (getGroup() != null) { out.attribute(null, ATT_GROUP, getGroup()); } if (isBlockable()) { out.attributeBoolean(null, ATT_BLOCKABLE_SYSTEM, isBlockable()); } if (getAllowBubbles() != DEFAULT_ALLOW_BUBBLE) { out.attributeInt(null, ATT_ALLOW_BUBBLE, getAllowBubbles()); } if (getOriginalImportance() != DEFAULT_IMPORTANCE) { out.attributeInt(null, ATT_ORIG_IMP, getOriginalImportance()); } if (getParentChannelId() != null) { out.attribute(null, ATT_PARENT_CHANNEL, getParentChannelId()); } if (getConversationId() != null) { out.attribute(null, ATT_CONVERSATION_ID, getConversationId()); } if (isDemoted()) { out.attributeBoolean(null, ATT_DEMOTE, isDemoted()); } if (isImportantConversation()) { out.attributeBoolean(null, ATT_IMP_CONVERSATION, isImportantConversation()); } // mImportanceLockedDefaultApp has a different source of truth and so isn't written to // this xml file out.endTag(null, TAG_CHANNEL); } /** * @hide */ @SystemApi public JSONObject toJson() throws JSONException { JSONObject record = new JSONObject(); record.put(ATT_ID, getId()); record.put(ATT_NAME, getName()); record.put(ATT_DESC, getDescription()); if (getImportance() != DEFAULT_IMPORTANCE) { record.put(ATT_IMPORTANCE, NotificationListenerService.Ranking.importanceToString(getImportance())); } if (canBypassDnd()) { record.put(ATT_PRIORITY, Notification.PRIORITY_MAX); } if (getLockscreenVisibility() != DEFAULT_VISIBILITY) { record.put(ATT_VISIBILITY, Notification.visibilityToString(getLockscreenVisibility())); } if (getSound() != null) { record.put(ATT_SOUND, getSound().toString()); } if (getAudioAttributes() != null) { record.put(ATT_USAGE, Integer.toString(getAudioAttributes().getUsage())); record.put(ATT_CONTENT_TYPE, Integer.toString(getAudioAttributes().getContentType())); record.put(ATT_FLAGS, Integer.toString(getAudioAttributes().getFlags())); } record.put(ATT_LIGHTS, Boolean.toString(shouldShowLights())); record.put(ATT_LIGHT_COLOR, Integer.toString(getLightColor())); record.put(ATT_VIBRATION_ENABLED, Boolean.toString(shouldVibrate())); record.put(ATT_USER_LOCKED, Integer.toString(getUserLockedFields())); record.put(ATT_FG_SERVICE_SHOWN, Boolean.toString(isUserVisibleTaskShown())); record.put(ATT_VIBRATION, longArrayToString(getVibrationPattern())); if (getVibrationEffect() != null) { record.put(ATT_VIBRATION_EFFECT, vibrationToString(getVibrationEffect())); } record.put(ATT_SHOW_BADGE, Boolean.toString(canShowBadge())); record.put(ATT_DELETED, Boolean.toString(isDeleted())); record.put(ATT_DELETED_TIME_MS, Long.toString(getDeletedTimeMs())); record.put(ATT_GROUP, getGroup()); record.put(ATT_BLOCKABLE_SYSTEM, isBlockable()); record.put(ATT_ALLOW_BUBBLE, getAllowBubbles()); // TODO: original importance return record; } private static AudioAttributes safeAudioAttributes(TypedXmlPullParser parser) { int usage = safeInt(parser, ATT_USAGE, AudioAttributes.USAGE_NOTIFICATION); int contentType = safeInt(parser, ATT_CONTENT_TYPE, AudioAttributes.CONTENT_TYPE_SONIFICATION); int flags = safeInt(parser, ATT_FLAGS, 0); return new AudioAttributes.Builder() .setUsage(usage) .setContentType(contentType) .setFlags(flags) .build(); } private static Uri safeUri(TypedXmlPullParser parser, String att) { final String val = parser.getAttributeValue(null, att); return val == null ? null : Uri.parse(val); } private static String vibrationToString(VibrationEffect effect) { StringWriter writer = new StringWriter(); try { VibrationXmlSerializer.serialize( effect, writer, VibrationXmlSerializer.FLAG_ALLOW_HIDDEN_APIS); } catch (IOException e) { Log.e(TAG, "Unable to serialize vibration: " + effect, e); } return writer.toString(); } private static VibrationEffect safeVibrationEffect(TypedXmlPullParser parser, String att) { final String val = parser.getAttributeValue(null, att); if (val != null) { try { return VibrationXmlParser.parseVibrationEffect( new StringReader(val), VibrationXmlParser.FLAG_ALLOW_HIDDEN_APIS); } catch (IOException e) { Log.e(TAG, "Unable to read serialized vibration effect", e); } } return null; } private static int safeInt(TypedXmlPullParser parser, String att, int defValue) { return parser.getAttributeInt(null, att, defValue); } private static boolean safeBool(TypedXmlPullParser parser, String att, boolean defValue) { return parser.getAttributeBoolean(null, att, defValue); } private static long[] safeLongArray(TypedXmlPullParser parser, String att, long[] defValue) { final String attributeValue = parser.getAttributeValue(null, att); if (TextUtils.isEmpty(attributeValue)) return defValue; String[] values = attributeValue.split(DELIMITER); long[] longValues = new long[values.length]; for (int i = 0; i < values.length; i++) { try { longValues[i] = Long.parseLong(values[i]); } catch (NumberFormatException e) { longValues[i] = 0; } } return longValues; } private static String longArrayToString(long[] values) { StringBuilder sb = new StringBuilder(); if (values != null && values.length > 0) { for (int i = 0; i < values.length - 1; i++) { sb.append(values[i]).append(DELIMITER); } sb.append(values[values.length - 1]); } return sb.toString(); } public static final @android.annotation.NonNull Creator CREATOR = new Creator() { @Override public NotificationChannel createFromParcel(Parcel in) { return new NotificationChannel(in); } @Override public NotificationChannel[] newArray(int size) { return new NotificationChannel[size]; } }; @Override public int describeContents() { return 0; } @Override public boolean equals(@Nullable Object o) { if (this == o) return true; if (o == null || getClass() != o.getClass()) return false; NotificationChannel that = (NotificationChannel) o; return getImportance() == that.getImportance() && mBypassDnd == that.mBypassDnd && getLockscreenVisibility() == that.getLockscreenVisibility() && mLights == that.mLights && getLightColor() == that.getLightColor() && getUserLockedFields() == that.getUserLockedFields() && isUserVisibleTaskShown() == that.isUserVisibleTaskShown() && mVibrationEnabled == that.mVibrationEnabled && mShowBadge == that.mShowBadge && isDeleted() == that.isDeleted() && getDeletedTimeMs() == that.getDeletedTimeMs() && isBlockable() == that.isBlockable() && mAllowBubbles == that.mAllowBubbles && Objects.equals(getId(), that.getId()) && Objects.equals(getName(), that.getName()) && Objects.equals(mDesc, that.mDesc) && Objects.equals(getSound(), that.getSound()) && Arrays.equals(mVibrationPattern, that.mVibrationPattern) && Objects.equals(getVibrationEffect(), that.getVibrationEffect()) && Objects.equals(getGroup(), that.getGroup()) && Objects.equals(getAudioAttributes(), that.getAudioAttributes()) && mImportanceLockedDefaultApp == that.mImportanceLockedDefaultApp && mOriginalImportance == that.mOriginalImportance && Objects.equals(getParentChannelId(), that.getParentChannelId()) && Objects.equals(getConversationId(), that.getConversationId()) && isDemoted() == that.isDemoted() && isImportantConversation() == that.isImportantConversation(); } @Override public int hashCode() { int result = Objects.hash(getId(), getName(), mDesc, getImportance(), mBypassDnd, getLockscreenVisibility(), getSound(), mLights, getLightColor(), getUserLockedFields(), isUserVisibleTaskShown(), mVibrationEnabled, mShowBadge, isDeleted(), getDeletedTimeMs(), getGroup(), getAudioAttributes(), isBlockable(), mAllowBubbles, mImportanceLockedDefaultApp, mOriginalImportance, getVibrationEffect(), mParentId, mConversationId, mDemoted, mImportantConvo); result = 31 * result + Arrays.hashCode(mVibrationPattern); return result; } /** @hide */ public void dump(PrintWriter pw, String prefix, boolean redacted) { String redactedName = redacted ? TextUtils.trimToLengthWithEllipsis(mName, 3) : mName; String output = "NotificationChannel{" + "mId='" + mId + '\'' + ", mName=" + redactedName + getFieldsString() + '}'; pw.println(prefix + output); } @Override public String toString() { return "NotificationChannel{" + "mId='" + mId + '\'' + ", mName=" + mName + getFieldsString() + '}'; } private String getFieldsString() { return ", mDescription=" + (!TextUtils.isEmpty(mDesc) ? "hasDescription " : "") + ", mImportance=" + mImportance + ", mBypassDnd=" + mBypassDnd + ", mLockscreenVisibility=" + mLockscreenVisibility + ", mSound=" + mSound + ", mLights=" + mLights + ", mLightColor=" + mLightColor + ", mVibrationPattern=" + Arrays.toString(mVibrationPattern) + ", mVibrationEffect=" + (mVibrationEffect == null ? "null" : mVibrationEffect.toString()) + ", mUserLockedFields=" + Integer.toHexString(mUserLockedFields) + ", mUserVisibleTaskShown=" + mUserVisibleTaskShown + ", mVibrationEnabled=" + mVibrationEnabled + ", mShowBadge=" + mShowBadge + ", mDeleted=" + mDeleted + ", mDeletedTimeMs=" + mDeletedTime + ", mGroup='" + mGroup + '\'' + ", mAudioAttributes=" + mAudioAttributes + ", mBlockableSystem=" + mBlockableSystem + ", mAllowBubbles=" + mAllowBubbles + ", mImportanceLockedDefaultApp=" + mImportanceLockedDefaultApp + ", mOriginalImp=" + mOriginalImportance + ", mParent=" + mParentId + ", mConversationId=" + mConversationId + ", mDemoted=" + mDemoted + ", mImportantConvo=" + mImportantConvo + ", mLastNotificationUpdateTimeMs=" + mLastNotificationUpdateTimeMs; } /** @hide */ public void dumpDebug(ProtoOutputStream proto, long fieldId) { final long token = proto.start(fieldId); proto.write(NotificationChannelProto.ID, mId); proto.write(NotificationChannelProto.NAME, mName); proto.write(NotificationChannelProto.DESCRIPTION, mDesc); proto.write(NotificationChannelProto.IMPORTANCE, mImportance); proto.write(NotificationChannelProto.CAN_BYPASS_DND, mBypassDnd); proto.write(NotificationChannelProto.LOCKSCREEN_VISIBILITY, mLockscreenVisibility); if (mSound != null) { proto.write(NotificationChannelProto.SOUND, mSound.toString()); } proto.write(NotificationChannelProto.USE_LIGHTS, mLights); proto.write(NotificationChannelProto.LIGHT_COLOR, mLightColor); if (mVibrationPattern != null) { for (long v : mVibrationPattern) { proto.write(NotificationChannelProto.VIBRATION, v); } } proto.write(NotificationChannelProto.USER_LOCKED_FIELDS, mUserLockedFields); proto.write(NotificationChannelProto.USER_VISIBLE_TASK_SHOWN, mUserVisibleTaskShown); proto.write(NotificationChannelProto.IS_VIBRATION_ENABLED, mVibrationEnabled); proto.write(NotificationChannelProto.SHOW_BADGE, mShowBadge); proto.write(NotificationChannelProto.IS_DELETED, mDeleted); proto.write(NotificationChannelProto.GROUP, mGroup); if (mAudioAttributes != null) { mAudioAttributes.dumpDebug(proto, NotificationChannelProto.AUDIO_ATTRIBUTES); } proto.write(NotificationChannelProto.IS_BLOCKABLE_SYSTEM, mBlockableSystem); proto.write(NotificationChannelProto.ALLOW_APP_OVERLAY, mAllowBubbles); proto.end(token); } }