/*
* Copyright (C) 2014 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.support.v4.media;
import static android.support.annotation.RestrictTo.Scope.LIBRARY_GROUP;
import android.graphics.Bitmap;
import android.net.Uri;
import android.os.Build;
import android.os.Bundle;
import android.os.Parcel;
import android.os.Parcelable;
import android.support.annotation.Nullable;
import android.support.annotation.RestrictTo;
import android.text.TextUtils;
/**
* A simple set of metadata for a media item suitable for display. This can be
* created using the Builder or retrieved from existing metadata using
* {@link MediaMetadataCompat#getDescription()}.
*/
public final class MediaDescriptionCompat implements Parcelable {
/**
* Used as a long extra field to indicate the bluetooth folder type of the media item as
* specified in the section 6.10.2.2 of the Bluetooth AVRCP 1.5. This is valid only for
* {@link MediaBrowserCompat.MediaItem} with
* {@link MediaBrowserCompat.MediaItem#FLAG_BROWSABLE}. The value should be one of the
* following:
*
* - {@link #BT_FOLDER_TYPE_MIXED}
* - {@link #BT_FOLDER_TYPE_TITLES}
* - {@link #BT_FOLDER_TYPE_ALBUMS}
* - {@link #BT_FOLDER_TYPE_ARTISTS}
* - {@link #BT_FOLDER_TYPE_GENRES}
* - {@link #BT_FOLDER_TYPE_PLAYLISTS}
* - {@link #BT_FOLDER_TYPE_YEARS}
*
*
* @see #getExtras()
*/
public static final String EXTRA_BT_FOLDER_TYPE = "android.media.extra.BT_FOLDER_TYPE";
/**
* The type of folder that is unknown or contains media elements of mixed types as specified in
* the section 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_MIXED = 0;
/**
* The type of folder that contains media elements only as specified in the section 6.10.2.2 of
* the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_TITLES = 1;
/**
* The type of folder that contains folders categorized by album as specified in the section
* 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_ALBUMS = 2;
/**
* The type of folder that contains folders categorized by artist as specified in the section
* 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_ARTISTS = 3;
/**
* The type of folder that contains folders categorized by genre as specified in the section
* 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_GENRES = 4;
/**
* The type of folder that contains folders categorized by playlist as specified in the section
* 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_PLAYLISTS = 5;
/**
* The type of folder that contains folders categorized by year as specified in the section
* 6.10.2.2 of the Bluetooth AVRCP 1.5.
*/
public static final long BT_FOLDER_TYPE_YEARS = 6;
/**
* Used as a long extra field to indicate the download status of the media item. The value
* should be one of the following:
*
* - {@link #STATUS_NOT_DOWNLOADED}
* - {@link #STATUS_DOWNLOADING}
* - {@link #STATUS_DOWNLOADED}
*
*
* @see #getExtras()
*/
public static final String EXTRA_DOWNLOAD_STATUS = "android.media.extra.DOWNLOAD_STATUS";
/**
* The status value to indicate the media item is not downloaded.
*
* @see #EXTRA_DOWNLOAD_STATUS
*/
public static final long STATUS_NOT_DOWNLOADED = 0;
/**
* The status value to indicate the media item is being downloaded.
*
* @see #EXTRA_DOWNLOAD_STATUS
*/
public static final long STATUS_DOWNLOADING = 1;
/**
* The status value to indicate the media item is downloaded for later offline playback.
*
* @see #EXTRA_DOWNLOAD_STATUS
*/
public static final long STATUS_DOWNLOADED = 2;
/**
* Custom key to store a media URI on API 21-22 devices (before it became part of the
* framework class) when parceling/converting to and from framework objects.
*
* @hide
*/
@RestrictTo(LIBRARY_GROUP)
public static final String DESCRIPTION_KEY_MEDIA_URI =
"android.support.v4.media.description.MEDIA_URI";
/**
* Custom key to store whether the original Bundle provided by the developer was null
*
* @hide
*/
@RestrictTo(LIBRARY_GROUP)
public static final String DESCRIPTION_KEY_NULL_BUNDLE_FLAG =
"android.support.v4.media.description.NULL_BUNDLE_FLAG";
/**
* A unique persistent id for the content or null.
*/
private final String mMediaId;
/**
* A primary title suitable for display or null.
*/
private final CharSequence mTitle;
/**
* A subtitle suitable for display or null.
*/
private final CharSequence mSubtitle;
/**
* A description suitable for display or null.
*/
private final CharSequence mDescription;
/**
* A bitmap icon suitable for display or null.
*/
private final Bitmap mIcon;
/**
* A Uri for an icon suitable for display or null.
*/
private final Uri mIconUri;
/**
* Extras for opaque use by apps/system.
*/
private final Bundle mExtras;
/**
* A Uri to identify this content.
*/
private final Uri mMediaUri;
/**
* A cached copy of the equivalent framework object.
*/
private Object mDescriptionObj;
MediaDescriptionCompat(String mediaId, CharSequence title, CharSequence subtitle,
CharSequence description, Bitmap icon, Uri iconUri, Bundle extras, Uri mediaUri) {
mMediaId = mediaId;
mTitle = title;
mSubtitle = subtitle;
mDescription = description;
mIcon = icon;
mIconUri = iconUri;
mExtras = extras;
mMediaUri = mediaUri;
}
MediaDescriptionCompat(Parcel in) {
mMediaId = in.readString();
mTitle = TextUtils.CHAR_SEQUENCE_CREATOR.createFromParcel(in);
mSubtitle = TextUtils.CHAR_SEQUENCE_CREATOR.createFromParcel(in);
mDescription = TextUtils.CHAR_SEQUENCE_CREATOR.createFromParcel(in);
mIcon = in.readParcelable(null);
mIconUri = in.readParcelable(null);
mExtras = in.readBundle();
mMediaUri = in.readParcelable(null);
}
/**
* Returns the media id or null. See
* {@link MediaMetadataCompat#METADATA_KEY_MEDIA_ID}.
*/
@Nullable
public String getMediaId() {
return mMediaId;
}
/**
* Returns a title suitable for display or null.
*
* @return A title or null.
*/
@Nullable
public CharSequence getTitle() {
return mTitle;
}
/**
* Returns a subtitle suitable for display or null.
*
* @return A subtitle or null.
*/
@Nullable
public CharSequence getSubtitle() {
return mSubtitle;
}
/**
* Returns a description suitable for display or null.
*
* @return A description or null.
*/
@Nullable
public CharSequence getDescription() {
return mDescription;
}
/**
* Returns a bitmap icon suitable for display or null.
*
* @return An icon or null.
*/
@Nullable
public Bitmap getIconBitmap() {
return mIcon;
}
/**
* Returns a Uri for an icon suitable for display or null.
*
* @return An icon uri or null.
*/
@Nullable
public Uri getIconUri() {
return mIconUri;
}
/**
* Returns any extras that were added to the description.
*
* @return A bundle of extras or null.
*/
@Nullable
public Bundle getExtras() {
return mExtras;
}
/**
* Returns a Uri representing this content or null.
*
* @return A media Uri or null.
*/
@Nullable
public Uri getMediaUri() {
return mMediaUri;
}
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
if (Build.VERSION.SDK_INT < 21) {
dest.writeString(mMediaId);
TextUtils.writeToParcel(mTitle, dest, flags);
TextUtils.writeToParcel(mSubtitle, dest, flags);
TextUtils.writeToParcel(mDescription, dest, flags);
dest.writeParcelable(mIcon, flags);
dest.writeParcelable(mIconUri, flags);
dest.writeBundle(mExtras);
dest.writeParcelable(mMediaUri, flags);
} else {
MediaDescriptionCompatApi21.writeToParcel(getMediaDescription(), dest, flags);
}
}
@Override
public String toString() {
return mTitle + ", " + mSubtitle + ", " + mDescription;
}
/**
* Gets the underlying framework {@link android.media.MediaDescription}
* object.
*
* This method is only supported on
* {@link android.os.Build.VERSION_CODES#LOLLIPOP} and later.
*
*
* @return An equivalent {@link android.media.MediaDescription} object, or
* null if none.
*/
public Object getMediaDescription() {
if (mDescriptionObj != null || Build.VERSION.SDK_INT < 21) {
return mDescriptionObj;
}
Object bob = MediaDescriptionCompatApi21.Builder.newInstance();
MediaDescriptionCompatApi21.Builder.setMediaId(bob, mMediaId);
MediaDescriptionCompatApi21.Builder.setTitle(bob, mTitle);
MediaDescriptionCompatApi21.Builder.setSubtitle(bob, mSubtitle);
MediaDescriptionCompatApi21.Builder.setDescription(bob, mDescription);
MediaDescriptionCompatApi21.Builder.setIconBitmap(bob, mIcon);
MediaDescriptionCompatApi21.Builder.setIconUri(bob, mIconUri);
// Media URI was not added until API 23, so add it to the Bundle of extras to
// ensure the data is not lost - this ensures that
// fromMediaDescription(getMediaDescription(mediaDescriptionCompat)) returns
// an equivalent MediaDescriptionCompat on all API levels
Bundle extras = mExtras;
if (Build.VERSION.SDK_INT < 23 && mMediaUri != null) {
if (extras == null) {
extras = new Bundle();
extras.putBoolean(DESCRIPTION_KEY_NULL_BUNDLE_FLAG, true);
}
extras.putParcelable(DESCRIPTION_KEY_MEDIA_URI, mMediaUri);
}
MediaDescriptionCompatApi21.Builder.setExtras(bob, extras);
if (Build.VERSION.SDK_INT >= 23) {
MediaDescriptionCompatApi23.Builder.setMediaUri(bob, mMediaUri);
}
mDescriptionObj = MediaDescriptionCompatApi21.Builder.build(bob);
return mDescriptionObj;
}
/**
* Creates an instance from a framework
* {@link android.media.MediaDescription} object.
*
* This method is only supported on API 21+.
*
*
* @param descriptionObj A {@link android.media.MediaDescription} object, or
* null if none.
* @return An equivalent {@link MediaMetadataCompat} object, or null if
* none.
*/
public static MediaDescriptionCompat fromMediaDescription(Object descriptionObj) {
if (descriptionObj != null && Build.VERSION.SDK_INT >= 21) {
Builder bob = new Builder();
bob.setMediaId(MediaDescriptionCompatApi21.getMediaId(descriptionObj));
bob.setTitle(MediaDescriptionCompatApi21.getTitle(descriptionObj));
bob.setSubtitle(MediaDescriptionCompatApi21.getSubtitle(descriptionObj));
bob.setDescription(MediaDescriptionCompatApi21.getDescription(descriptionObj));
bob.setIconBitmap(MediaDescriptionCompatApi21.getIconBitmap(descriptionObj));
bob.setIconUri(MediaDescriptionCompatApi21.getIconUri(descriptionObj));
Bundle extras = MediaDescriptionCompatApi21.getExtras(descriptionObj);
Uri mediaUri = extras == null ? null :
(Uri) extras.getParcelable(DESCRIPTION_KEY_MEDIA_URI);
if (mediaUri != null) {
if (extras.containsKey(DESCRIPTION_KEY_NULL_BUNDLE_FLAG) && extras.size() == 2) {
// The extras were only created for the media URI, so we set it back to null to
// ensure mediaDescriptionCompat.getExtras() equals
// fromMediaDescription(getMediaDescription(mediaDescriptionCompat)).getExtras()
extras = null;
} else {
// Remove media URI keys to ensure mediaDescriptionCompat.getExtras().keySet()
// equals fromMediaDescription(getMediaDescription(mediaDescriptionCompat))
// .getExtras().keySet()
extras.remove(DESCRIPTION_KEY_MEDIA_URI);
extras.remove(DESCRIPTION_KEY_NULL_BUNDLE_FLAG);
}
}
bob.setExtras(extras);
if (mediaUri != null) {
bob.setMediaUri(mediaUri);
} else if (Build.VERSION.SDK_INT >= 23) {
bob.setMediaUri(MediaDescriptionCompatApi23.getMediaUri(descriptionObj));
}
MediaDescriptionCompat descriptionCompat = bob.build();
descriptionCompat.mDescriptionObj = descriptionObj;
return descriptionCompat;
} else {
return null;
}
}
public static final Parcelable.Creator CREATOR =
new Parcelable.Creator() {
@Override
public MediaDescriptionCompat createFromParcel(Parcel in) {
if (Build.VERSION.SDK_INT < 21) {
return new MediaDescriptionCompat(in);
} else {
return fromMediaDescription(MediaDescriptionCompatApi21.fromParcel(in));
}
}
@Override
public MediaDescriptionCompat[] newArray(int size) {
return new MediaDescriptionCompat[size];
}
};
/**
* Builder for {@link MediaDescriptionCompat} objects.
*/
public static final class Builder {
private String mMediaId;
private CharSequence mTitle;
private CharSequence mSubtitle;
private CharSequence mDescription;
private Bitmap mIcon;
private Uri mIconUri;
private Bundle mExtras;
private Uri mMediaUri;
/**
* Creates an initially empty builder.
*/
public Builder() {
}
/**
* Sets the media id.
*
* @param mediaId The unique id for the item or null.
* @return this
*/
public Builder setMediaId(@Nullable String mediaId) {
mMediaId = mediaId;
return this;
}
/**
* Sets the title.
*
* @param title A title suitable for display to the user or null.
* @return this
*/
public Builder setTitle(@Nullable CharSequence title) {
mTitle = title;
return this;
}
/**
* Sets the subtitle.
*
* @param subtitle A subtitle suitable for display to the user or null.
* @return this
*/
public Builder setSubtitle(@Nullable CharSequence subtitle) {
mSubtitle = subtitle;
return this;
}
/**
* Sets the description.
*
* @param description A description suitable for display to the user or
* null.
* @return this
*/
public Builder setDescription(@Nullable CharSequence description) {
mDescription = description;
return this;
}
/**
* Sets the icon.
*
* @param icon A {@link Bitmap} icon suitable for display to the user or
* null.
* @return this
*/
public Builder setIconBitmap(@Nullable Bitmap icon) {
mIcon = icon;
return this;
}
/**
* Sets the icon uri.
*
* @param iconUri A {@link Uri} for an icon suitable for display to the
* user or null.
* @return this
*/
public Builder setIconUri(@Nullable Uri iconUri) {
mIconUri = iconUri;
return this;
}
/**
* Sets a bundle of extras.
*
* @param extras The extras to include with this description or null.
* @return this
*/
public Builder setExtras(@Nullable Bundle extras) {
mExtras = extras;
return this;
}
/**
* Sets the media uri.
*
* @param mediaUri The content's {@link Uri} for the item or null.
* @return this
*/
public Builder setMediaUri(@Nullable Uri mediaUri) {
mMediaUri = mediaUri;
return this;
}
/**
* Creates a {@link MediaDescriptionCompat} instance with the specified
* fields.
*
* @return A MediaDescriptionCompat instance.
*/
public MediaDescriptionCompat build() {
return new MediaDescriptionCompat(mMediaId, mTitle, mSubtitle, mDescription, mIcon,
mIconUri, mExtras, mMediaUri);
}
}
}