1 /*
2  * Copyright (C) 2011 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 package android.media;
18 
19 import android.annotation.UnsupportedAppUsage;
20 import android.app.PendingIntent;
21 import android.content.ComponentName;
22 import android.content.Intent;
23 import android.graphics.Bitmap;
24 import android.media.session.MediaSession;
25 import android.media.session.MediaSessionLegacyHelper;
26 import android.media.session.PlaybackState;
27 import android.os.Bundle;
28 import android.os.Looper;
29 import android.os.SystemClock;
30 import android.util.Log;
31 
32 /**
33  * RemoteControlClient enables exposing information meant to be consumed by remote controls
34  * capable of displaying metadata, artwork and media transport control buttons.
35  *
36  * <p>A remote control client object is associated with a media button event receiver. This
37  * event receiver must have been previously registered with
38  * {@link AudioManager#registerMediaButtonEventReceiver(ComponentName)} before the
39  * RemoteControlClient can be registered through
40  * {@link AudioManager#registerRemoteControlClient(RemoteControlClient)}.
41  *
42  * <p>Here is an example of creating a RemoteControlClient instance after registering a media
43  * button event receiver:
44  * <pre>ComponentName myEventReceiver = new ComponentName(getPackageName(), MyRemoteControlEventReceiver.class.getName());
45  * AudioManager myAudioManager = (AudioManager) getSystemService(Context.AUDIO_SERVICE);
46  * myAudioManager.registerMediaButtonEventReceiver(myEventReceiver);
47  * // build the PendingIntent for the remote control client
48  * Intent mediaButtonIntent = new Intent(Intent.ACTION_MEDIA_BUTTON);
49  * mediaButtonIntent.setComponent(myEventReceiver);
50  * PendingIntent mediaPendingIntent = PendingIntent.getBroadcast(getApplicationContext(), 0, mediaButtonIntent, 0);
51  * // create and register the remote control client
52  * RemoteControlClient myRemoteControlClient = new RemoteControlClient(mediaPendingIntent);
53  * myAudioManager.registerRemoteControlClient(myRemoteControlClient);</pre>
54  *
55  * @deprecated Use {@link MediaSession} instead.
56  */
57 @Deprecated public class RemoteControlClient
58 {
59     private final static String TAG = "RemoteControlClient";
60     private final static boolean DEBUG = false;
61 
62     /**
63      * Playback state of a RemoteControlClient which is stopped.
64      *
65      * @see #setPlaybackState(int)
66      */
67     public final static int PLAYSTATE_STOPPED            = 1;
68     /**
69      * Playback state of a RemoteControlClient which is paused.
70      *
71      * @see #setPlaybackState(int)
72      */
73     public final static int PLAYSTATE_PAUSED             = 2;
74     /**
75      * Playback state of a RemoteControlClient which is playing media.
76      *
77      * @see #setPlaybackState(int)
78      */
79     public final static int PLAYSTATE_PLAYING            = 3;
80     /**
81      * Playback state of a RemoteControlClient which is fast forwarding in the media
82      *    it is currently playing.
83      *
84      * @see #setPlaybackState(int)
85      */
86     public final static int PLAYSTATE_FAST_FORWARDING    = 4;
87     /**
88      * Playback state of a RemoteControlClient which is fast rewinding in the media
89      *    it is currently playing.
90      *
91      * @see #setPlaybackState(int)
92      */
93     public final static int PLAYSTATE_REWINDING          = 5;
94     /**
95      * Playback state of a RemoteControlClient which is skipping to the next
96      *    logical chapter (such as a song in a playlist) in the media it is currently playing.
97      *
98      * @see #setPlaybackState(int)
99      */
100     public final static int PLAYSTATE_SKIPPING_FORWARDS  = 6;
101     /**
102      * Playback state of a RemoteControlClient which is skipping back to the previous
103      *    logical chapter (such as a song in a playlist) in the media it is currently playing.
104      *
105      * @see #setPlaybackState(int)
106      */
107     public final static int PLAYSTATE_SKIPPING_BACKWARDS = 7;
108     /**
109      * Playback state of a RemoteControlClient which is buffering data to play before it can
110      *    start or resume playback.
111      *
112      * @see #setPlaybackState(int)
113      */
114     public final static int PLAYSTATE_BUFFERING          = 8;
115     /**
116      * Playback state of a RemoteControlClient which cannot perform any playback related
117      *    operation because of an internal error. Examples of such situations are no network
118      *    connectivity when attempting to stream data from a server, or expired user credentials
119      *    when trying to play subscription-based content.
120      *
121      * @see #setPlaybackState(int)
122      */
123     public final static int PLAYSTATE_ERROR              = 9;
124     /**
125      * @hide
126      * The value of a playback state when none has been declared.
127      * Intentionally hidden as an application shouldn't set such a playback state value.
128      */
129     public final static int PLAYSTATE_NONE               = 0;
130 
131     /**
132      * @hide
133      * The default playback type, "local", indicating the presentation of the media is happening on
134      * the same device (e.g. a phone, a tablet) as where it is controlled from.
135      */
136     public final static int PLAYBACK_TYPE_LOCAL = 0;
137     /**
138      * @hide
139      * A playback type indicating the presentation of the media is happening on
140      * a different device (i.e. the remote device) than where it is controlled from.
141      */
142     public final static int PLAYBACK_TYPE_REMOTE = 1;
143     private final static int PLAYBACK_TYPE_MIN = PLAYBACK_TYPE_LOCAL;
144     private final static int PLAYBACK_TYPE_MAX = PLAYBACK_TYPE_REMOTE;
145     /**
146      * @hide
147      * Playback information indicating the playback volume is fixed, i.e. it cannot be controlled
148      * from this object. An example of fixed playback volume is a remote player, playing over HDMI
149      * where the user prefer to control the volume on the HDMI sink, rather than attenuate at the
150      * source.
151      * @see #PLAYBACKINFO_VOLUME_HANDLING.
152      */
153     public final static int PLAYBACK_VOLUME_FIXED = 0;
154     /**
155      * @hide
156      * Playback information indicating the playback volume is variable and can be controlled from
157      * this object.
158      * @see #PLAYBACKINFO_VOLUME_HANDLING.
159      */
160     public final static int PLAYBACK_VOLUME_VARIABLE = 1;
161     /**
162      * @hide (to be un-hidden)
163      * The playback information value indicating the value of a given information type is invalid.
164      * @see #PLAYBACKINFO_VOLUME_HANDLING.
165      */
166     public final static int PLAYBACKINFO_INVALID_VALUE = Integer.MIN_VALUE;
167 
168     /**
169      * @hide
170      * An unknown or invalid playback position value.
171      */
172     public final static long PLAYBACK_POSITION_INVALID = -1;
173     /**
174      * @hide
175      * An invalid playback position value associated with the use of {@link #setPlaybackState(int)}
176      * used to indicate that playback position will remain unknown.
177      */
178     public final static long PLAYBACK_POSITION_ALWAYS_UNKNOWN = 0x8019771980198300L;
179     /**
180      * @hide
181      * The default playback speed, 1x.
182      */
183     public final static float PLAYBACK_SPEED_1X = 1.0f;
184 
185     //==========================================
186     // Public keys for playback information
187     /**
188      * @hide
189      * Playback information that defines the type of playback associated with this
190      * RemoteControlClient. See {@link #PLAYBACK_TYPE_LOCAL} and {@link #PLAYBACK_TYPE_REMOTE}.
191      */
192     public final static int PLAYBACKINFO_PLAYBACK_TYPE = 1;
193     /**
194      * @hide
195      * Playback information that defines at what volume the playback associated with this
196      * RemoteControlClient is performed. This information is only used when the playback type is not
197      * local (see {@link #PLAYBACKINFO_PLAYBACK_TYPE}).
198      */
199     public final static int PLAYBACKINFO_VOLUME = 2;
200     /**
201      * @hide
202      * Playback information that defines the maximum volume volume value that is supported
203      * by the playback associated with this RemoteControlClient. This information is only used
204      * when the playback type is not local (see {@link #PLAYBACKINFO_PLAYBACK_TYPE}).
205      */
206     public final static int PLAYBACKINFO_VOLUME_MAX = 3;
207     /**
208      * @hide
209      * Playback information that defines how volume is handled for the presentation of the media.
210      * @see #PLAYBACK_VOLUME_FIXED
211      * @see #PLAYBACK_VOLUME_VARIABLE
212      */
213     public final static int PLAYBACKINFO_VOLUME_HANDLING = 4;
214     /**
215      * @hide
216      * Playback information that defines over what stream type the media is presented.
217      */
218     public final static int PLAYBACKINFO_USES_STREAM = 5;
219 
220     //==========================================
221     // Public flags for the supported transport control capabilities
222     /**
223      * Flag indicating a RemoteControlClient makes use of the "previous" media key.
224      *
225      * @see #setTransportControlFlags(int)
226      * @see android.view.KeyEvent#KEYCODE_MEDIA_PREVIOUS
227      */
228     public final static int FLAG_KEY_MEDIA_PREVIOUS = 1 << 0;
229     /**
230      * Flag indicating a RemoteControlClient makes use of the "rewind" media key.
231      *
232      * @see #setTransportControlFlags(int)
233      * @see android.view.KeyEvent#KEYCODE_MEDIA_REWIND
234      */
235     public final static int FLAG_KEY_MEDIA_REWIND = 1 << 1;
236     /**
237      * Flag indicating a RemoteControlClient makes use of the "play" media key.
238      *
239      * @see #setTransportControlFlags(int)
240      * @see android.view.KeyEvent#KEYCODE_MEDIA_PLAY
241      */
242     public final static int FLAG_KEY_MEDIA_PLAY = 1 << 2;
243     /**
244      * Flag indicating a RemoteControlClient makes use of the "play/pause" media key.
245      *
246      * @see #setTransportControlFlags(int)
247      * @see android.view.KeyEvent#KEYCODE_MEDIA_PLAY_PAUSE
248      */
249     public final static int FLAG_KEY_MEDIA_PLAY_PAUSE = 1 << 3;
250     /**
251      * Flag indicating a RemoteControlClient makes use of the "pause" media key.
252      *
253      * @see #setTransportControlFlags(int)
254      * @see android.view.KeyEvent#KEYCODE_MEDIA_PAUSE
255      */
256     public final static int FLAG_KEY_MEDIA_PAUSE = 1 << 4;
257     /**
258      * Flag indicating a RemoteControlClient makes use of the "stop" media key.
259      *
260      * @see #setTransportControlFlags(int)
261      * @see android.view.KeyEvent#KEYCODE_MEDIA_STOP
262      */
263     public final static int FLAG_KEY_MEDIA_STOP = 1 << 5;
264     /**
265      * Flag indicating a RemoteControlClient makes use of the "fast forward" media key.
266      *
267      * @see #setTransportControlFlags(int)
268      * @see android.view.KeyEvent#KEYCODE_MEDIA_FAST_FORWARD
269      */
270     public final static int FLAG_KEY_MEDIA_FAST_FORWARD = 1 << 6;
271     /**
272      * Flag indicating a RemoteControlClient makes use of the "next" media key.
273      *
274      * @see #setTransportControlFlags(int)
275      * @see android.view.KeyEvent#KEYCODE_MEDIA_NEXT
276      */
277     public final static int FLAG_KEY_MEDIA_NEXT = 1 << 7;
278     /**
279      * Flag indicating a RemoteControlClient can receive changes in the media playback position
280      * through the {@link OnPlaybackPositionUpdateListener} interface. This flag must be set
281      * in order for components that display the RemoteControlClient information, to display and
282      * let the user control media playback position.
283      * @see #setTransportControlFlags(int)
284      * @see #setOnGetPlaybackPositionListener(OnGetPlaybackPositionListener)
285      * @see #setPlaybackPositionUpdateListener(OnPlaybackPositionUpdateListener)
286      */
287     public final static int FLAG_KEY_MEDIA_POSITION_UPDATE = 1 << 8;
288     /**
289      * Flag indicating a RemoteControlClient supports ratings.
290      * This flag must be set in order for components that display the RemoteControlClient
291      * information, to display ratings information, and, if ratings are declared editable
292      * (by calling {@link MediaMetadataEditor#addEditableKey(int)} with the
293      * {@link MediaMetadataEditor#RATING_KEY_BY_USER} key), it will enable the user to rate
294      * the media, with values being received through the interface set with
295      * {@link #setMetadataUpdateListener(OnMetadataUpdateListener)}.
296      * @see #setTransportControlFlags(int)
297      */
298     public final static int FLAG_KEY_MEDIA_RATING = 1 << 9;
299 
300     /**
301      * @hide
302      * The flags for when no media keys are declared supported.
303      * Intentionally hidden as an application shouldn't set the transport control flags
304      *     to this value.
305      */
306     public final static int FLAGS_KEY_MEDIA_NONE = 0;
307 
308     /**
309      * @hide
310      * Flag used to signal some type of metadata exposed by the RemoteControlClient is requested.
311      */
312     public final static int FLAG_INFORMATION_REQUEST_METADATA = 1 << 0;
313     /**
314      * @hide
315      * Flag used to signal that the transport control buttons supported by the
316      *     RemoteControlClient are requested.
317      * This can for instance happen when playback is at the end of a playlist, and the "next"
318      * operation is not supported anymore.
319      */
320     public final static int FLAG_INFORMATION_REQUEST_KEY_MEDIA = 1 << 1;
321     /**
322      * @hide
323      * Flag used to signal that the playback state of the RemoteControlClient is requested.
324      */
325     public final static int FLAG_INFORMATION_REQUEST_PLAYSTATE = 1 << 2;
326     /**
327      * @hide
328      * Flag used to signal that the album art for the RemoteControlClient is requested.
329      */
330     public final static int FLAG_INFORMATION_REQUEST_ALBUM_ART = 1 << 3;
331 
332     private MediaSession mSession;
333 
334     /**
335      * Class constructor.
336      * @param mediaButtonIntent The intent that will be sent for the media button events sent
337      *     by remote controls.
338      *     This intent needs to have been constructed with the {@link Intent#ACTION_MEDIA_BUTTON}
339      *     action, and have a component that will handle the intent (set with
340      *     {@link Intent#setComponent(ComponentName)}) registered with
341      *     {@link AudioManager#registerMediaButtonEventReceiver(ComponentName)}
342      *     before this new RemoteControlClient can itself be registered with
343      *     {@link AudioManager#registerRemoteControlClient(RemoteControlClient)}.
344      * @see AudioManager#registerMediaButtonEventReceiver(ComponentName)
345      * @see AudioManager#registerRemoteControlClient(RemoteControlClient)
346      */
RemoteControlClient(PendingIntent mediaButtonIntent)347     public RemoteControlClient(PendingIntent mediaButtonIntent) {
348         mRcMediaIntent = mediaButtonIntent;
349     }
350 
351     /**
352      * Class constructor for a remote control client whose internal event handling
353      * happens on a user-provided Looper.
354      * @param mediaButtonIntent The intent that will be sent for the media button events sent
355      *     by remote controls.
356      *     This intent needs to have been constructed with the {@link Intent#ACTION_MEDIA_BUTTON}
357      *     action, and have a component that will handle the intent (set with
358      *     {@link Intent#setComponent(ComponentName)}) registered with
359      *     {@link AudioManager#registerMediaButtonEventReceiver(ComponentName)}
360      *     before this new RemoteControlClient can itself be registered with
361      *     {@link AudioManager#registerRemoteControlClient(RemoteControlClient)}.
362      * @param looper The Looper running the event loop.
363      * @see AudioManager#registerMediaButtonEventReceiver(ComponentName)
364      * @see AudioManager#registerRemoteControlClient(RemoteControlClient)
365      */
RemoteControlClient(PendingIntent mediaButtonIntent, Looper looper)366     public RemoteControlClient(PendingIntent mediaButtonIntent, Looper looper) {
367         mRcMediaIntent = mediaButtonIntent;
368     }
369 
370     /**
371      * @hide
372      */
registerWithSession(MediaSessionLegacyHelper helper)373     public void registerWithSession(MediaSessionLegacyHelper helper) {
374         helper.addRccListener(mRcMediaIntent, mTransportListener);
375         mSession = helper.getSession(mRcMediaIntent);
376         setTransportControlFlags(mTransportControlFlags);
377     }
378 
379     /**
380      * @hide
381      */
unregisterWithSession(MediaSessionLegacyHelper helper)382     public void unregisterWithSession(MediaSessionLegacyHelper helper) {
383         helper.removeRccListener(mRcMediaIntent);
384         mSession = null;
385     }
386 
387     /**
388      * Get a {@link MediaSession} associated with this RCC. It will only have a
389      * session while it is registered with
390      * {@link AudioManager#registerRemoteControlClient}. The session returned
391      * should not be modified directly by the application but may be used with
392      * other APIs that require a session.
393      *
394      * @return A media session object or null.
395      */
getMediaSession()396     public MediaSession getMediaSession() {
397         return mSession;
398     }
399 
400     /**
401      * Class used to modify metadata in a {@link RemoteControlClient} object.
402      * Use {@link RemoteControlClient#editMetadata(boolean)} to create an instance of an editor,
403      * on which you set the metadata for the RemoteControlClient instance. Once all the information
404      * has been set, use {@link #apply()} to make it the new metadata that should be displayed
405      * for the associated client. Once the metadata has been "applied", you cannot reuse this
406      * instance of the MetadataEditor.
407      *
408      * @deprecated Use {@link MediaMetadata} and {@link MediaSession} instead.
409      */
410     @Deprecated public class MetadataEditor extends MediaMetadataEditor {
411 
412         // only use RemoteControlClient.editMetadata() to get a MetadataEditor instance
MetadataEditor()413         private MetadataEditor() { }
414         /**
415          * @hide
416          */
clone()417         public Object clone() throws CloneNotSupportedException {
418             throw new CloneNotSupportedException();
419         }
420 
421         /**
422          * The metadata key for the content artwork / album art.
423          */
424         public final static int BITMAP_KEY_ARTWORK = 100;
425 
426         /**
427          * @hide
428          * TODO(jmtrivi) have lockscreen move to the new key name and remove
429          */
430         public final static int METADATA_KEY_ARTWORK = BITMAP_KEY_ARTWORK;
431 
432         /**
433          * Adds textual information to be displayed.
434          * Note that none of the information added after {@link #apply()} has been called,
435          * will be displayed.
436          * @param key The identifier of a the metadata field to set. Valid values are
437          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_ALBUM},
438          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_ALBUMARTIST},
439          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_TITLE},
440          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_ARTIST},
441          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_AUTHOR},
442          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_COMPILATION},
443          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_COMPOSER},
444          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_DATE},
445          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_GENRE},
446          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_TITLE},
447          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_WRITER}.
448          * @param value The text for the given key, or {@code null} to signify there is no valid
449          *      information for the field.
450          * @return Returns a reference to the same MetadataEditor object, so you can chain put
451          *      calls together.
452          */
putString(int key, String value)453         public synchronized MetadataEditor putString(int key, String value)
454                 throws IllegalArgumentException {
455             super.putString(key, value);
456             if (mMetadataBuilder != null) {
457                 // MediaMetadata supports all the same fields as MetadataEditor
458                 String metadataKey = MediaMetadata.getKeyFromMetadataEditorKey(key);
459                 // But just in case, don't add things we don't understand
460                 if (metadataKey != null) {
461                     mMetadataBuilder.putText(metadataKey, value);
462                 }
463             }
464 
465             return this;
466         }
467 
468         /**
469          * Adds numerical information to be displayed.
470          * Note that none of the information added after {@link #apply()} has been called,
471          * will be displayed.
472          * @param key the identifier of a the metadata field to set. Valid values are
473          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_CD_TRACK_NUMBER},
474          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_DISC_NUMBER},
475          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_DURATION} (with a value
476          *      expressed in milliseconds),
477          *      {@link android.media.MediaMetadataRetriever#METADATA_KEY_YEAR}.
478          * @param value The long value for the given key
479          * @return Returns a reference to the same MetadataEditor object, so you can chain put
480          *      calls together.
481          * @throws IllegalArgumentException
482          */
putLong(int key, long value)483         public synchronized MetadataEditor putLong(int key, long value)
484                 throws IllegalArgumentException {
485             super.putLong(key, value);
486             if (mMetadataBuilder != null) {
487                 // MediaMetadata supports all the same fields as MetadataEditor
488                 String metadataKey = MediaMetadata.getKeyFromMetadataEditorKey(key);
489                 // But just in case, don't add things we don't understand
490                 if (metadataKey != null) {
491                     mMetadataBuilder.putLong(metadataKey, value);
492                 }
493             }
494             return this;
495         }
496 
497         /**
498          * Sets the album / artwork picture to be displayed on the remote control.
499          * @param key the identifier of the bitmap to set. The only valid value is
500          *      {@link #BITMAP_KEY_ARTWORK}
501          * @param bitmap The bitmap for the artwork, or null if there isn't any.
502          * @return Returns a reference to the same MetadataEditor object, so you can chain put
503          *      calls together.
504          * @throws IllegalArgumentException
505          * @see android.graphics.Bitmap
506          */
507         @Override
putBitmap(int key, Bitmap bitmap)508         public synchronized MetadataEditor putBitmap(int key, Bitmap bitmap)
509                 throws IllegalArgumentException {
510             super.putBitmap(key, bitmap);
511             if (mMetadataBuilder != null) {
512                 // MediaMetadata supports all the same fields as MetadataEditor
513                 String metadataKey = MediaMetadata.getKeyFromMetadataEditorKey(key);
514                 // But just in case, don't add things we don't understand
515                 if (metadataKey != null) {
516                     mMetadataBuilder.putBitmap(metadataKey, bitmap);
517                 }
518             }
519             return this;
520         }
521 
522         @Override
putObject(int key, Object object)523         public synchronized MetadataEditor putObject(int key, Object object)
524                 throws IllegalArgumentException {
525             super.putObject(key, object);
526             if (mMetadataBuilder != null &&
527                     (key == MediaMetadataEditor.RATING_KEY_BY_USER ||
528                     key == MediaMetadataEditor.RATING_KEY_BY_OTHERS)) {
529                 String metadataKey = MediaMetadata.getKeyFromMetadataEditorKey(key);
530                 if (metadataKey != null) {
531                     mMetadataBuilder.putRating(metadataKey, (Rating) object);
532                 }
533             }
534             return this;
535         }
536 
537         /**
538          * Clears all the metadata that has been set since the MetadataEditor instance was created
539          * (with {@link RemoteControlClient#editMetadata(boolean)}).
540          * Note that clearing the metadata doesn't reset the editable keys
541          * (use {@link MediaMetadataEditor#removeEditableKeys()} instead).
542          */
543         @Override
clear()544         public synchronized void clear() {
545             super.clear();
546         }
547 
548         /**
549          * Associates all the metadata that has been set since the MetadataEditor instance was
550          *     created with {@link RemoteControlClient#editMetadata(boolean)}, or since
551          *     {@link #clear()} was called, with the RemoteControlClient. Once "applied",
552          *     this MetadataEditor cannot be reused to edit the RemoteControlClient's metadata.
553          */
apply()554         public synchronized void apply() {
555             if (mApplied) {
556                 Log.e(TAG, "Can't apply a previously applied MetadataEditor");
557                 return;
558             }
559             synchronized (mCacheLock) {
560                 // Still build the old metadata so when creating a new editor
561                 // you get the expected values.
562                 // assign the edited data
563                 mMetadata = new Bundle(mEditorMetadata);
564                 // add the information about editable keys
565                 mMetadata.putLong(String.valueOf(KEY_EDITABLE_MASK), mEditableKeys);
566                 if ((mOriginalArtwork != null) && (!mOriginalArtwork.equals(mEditorArtwork))) {
567                     mOriginalArtwork.recycle();
568                 }
569                 mOriginalArtwork = mEditorArtwork;
570                 mEditorArtwork = null;
571 
572                 // USE_SESSIONS
573                 if (mSession != null && mMetadataBuilder != null) {
574                     mMediaMetadata = mMetadataBuilder.build();
575                     mSession.setMetadata(mMediaMetadata);
576                 }
577                 mApplied = true;
578             }
579         }
580     }
581 
582     /**
583      * Creates a {@link MetadataEditor}.
584      * @param startEmpty Set to false if you want the MetadataEditor to contain the metadata that
585      *     was previously applied to the RemoteControlClient, or true if it is to be created empty.
586      * @return a new MetadataEditor instance.
587      */
editMetadata(boolean startEmpty)588     public MetadataEditor editMetadata(boolean startEmpty) {
589         MetadataEditor editor = new MetadataEditor();
590         if (startEmpty) {
591             editor.mEditorMetadata = new Bundle();
592             editor.mEditorArtwork = null;
593             editor.mMetadataChanged = true;
594             editor.mArtworkChanged = true;
595             editor.mEditableKeys = 0;
596         } else {
597             editor.mEditorMetadata = new Bundle(mMetadata);
598             editor.mEditorArtwork = mOriginalArtwork;
599             editor.mMetadataChanged = false;
600             editor.mArtworkChanged = false;
601         }
602         // USE_SESSIONS
603         if (startEmpty || mMediaMetadata == null) {
604             editor.mMetadataBuilder = new MediaMetadata.Builder();
605         } else {
606             editor.mMetadataBuilder = new MediaMetadata.Builder(mMediaMetadata);
607         }
608         return editor;
609     }
610 
611     /**
612      * Sets the current playback state.
613      * @param state The current playback state, one of the following values:
614      *       {@link #PLAYSTATE_STOPPED},
615      *       {@link #PLAYSTATE_PAUSED},
616      *       {@link #PLAYSTATE_PLAYING},
617      *       {@link #PLAYSTATE_FAST_FORWARDING},
618      *       {@link #PLAYSTATE_REWINDING},
619      *       {@link #PLAYSTATE_SKIPPING_FORWARDS},
620      *       {@link #PLAYSTATE_SKIPPING_BACKWARDS},
621      *       {@link #PLAYSTATE_BUFFERING},
622      *       {@link #PLAYSTATE_ERROR}.
623      */
setPlaybackState(int state)624     public void setPlaybackState(int state) {
625         setPlaybackStateInt(state, PLAYBACK_POSITION_ALWAYS_UNKNOWN, PLAYBACK_SPEED_1X,
626                 false /* legacy API, converting to method with position and speed */);
627     }
628 
629     /**
630      * Sets the current playback state and the matching media position for the current playback
631      *   speed.
632      * @param state The current playback state, one of the following values:
633      *       {@link #PLAYSTATE_STOPPED},
634      *       {@link #PLAYSTATE_PAUSED},
635      *       {@link #PLAYSTATE_PLAYING},
636      *       {@link #PLAYSTATE_FAST_FORWARDING},
637      *       {@link #PLAYSTATE_REWINDING},
638      *       {@link #PLAYSTATE_SKIPPING_FORWARDS},
639      *       {@link #PLAYSTATE_SKIPPING_BACKWARDS},
640      *       {@link #PLAYSTATE_BUFFERING},
641      *       {@link #PLAYSTATE_ERROR}.
642      * @param timeInMs a 0 or positive value for the current media position expressed in ms
643      *    (same unit as for when sending the media duration, if applicable, with
644      *    {@link android.media.MediaMetadataRetriever#METADATA_KEY_DURATION} in the
645      *    {@link RemoteControlClient.MetadataEditor}). Negative values imply that position is not
646      *    known (e.g. listening to a live stream of a radio) or not applicable (e.g. when state
647      *    is {@link #PLAYSTATE_BUFFERING} and nothing had played yet).
648      * @param playbackSpeed a value expressed as a ratio of 1x playback: 1.0f is normal playback,
649      *    2.0f is 2x, 0.5f is half-speed, -2.0f is rewind at 2x speed. 0.0f means nothing is
650      *    playing (e.g. when state is {@link #PLAYSTATE_ERROR}).
651      */
setPlaybackState(int state, long timeInMs, float playbackSpeed)652     public void setPlaybackState(int state, long timeInMs, float playbackSpeed) {
653         setPlaybackStateInt(state, timeInMs, playbackSpeed, true);
654     }
655 
setPlaybackStateInt(int state, long timeInMs, float playbackSpeed, boolean hasPosition)656     private void setPlaybackStateInt(int state, long timeInMs, float playbackSpeed,
657             boolean hasPosition) {
658         synchronized(mCacheLock) {
659             if ((mPlaybackState != state) || (mPlaybackPositionMs != timeInMs)
660                     || (mPlaybackSpeed != playbackSpeed)) {
661                 // store locally
662                 mPlaybackState = state;
663                 // distinguish between an application not knowing the current playback position
664                 // at the moment and an application using the API where only the playback state
665                 // is passed, not the playback position.
666                 if (hasPosition) {
667                     if (timeInMs < 0) {
668                         mPlaybackPositionMs = PLAYBACK_POSITION_INVALID;
669                     } else {
670                         mPlaybackPositionMs = timeInMs;
671                     }
672                 } else {
673                     mPlaybackPositionMs = PLAYBACK_POSITION_ALWAYS_UNKNOWN;
674                 }
675                 mPlaybackSpeed = playbackSpeed;
676                 // keep track of when the state change occurred
677                 mPlaybackStateChangeTimeMs = SystemClock.elapsedRealtime();
678 
679                 // USE_SESSIONS
680                 if (mSession != null) {
681                     int pbState = getStateFromRccState(state);
682                     long position = hasPosition ? mPlaybackPositionMs
683                             : PlaybackState.PLAYBACK_POSITION_UNKNOWN;
684 
685                     PlaybackState.Builder bob = new PlaybackState.Builder(mSessionPlaybackState);
686                     bob.setState(pbState, position, playbackSpeed, SystemClock.elapsedRealtime());
687                     bob.setErrorMessage(null);
688                     mSessionPlaybackState = bob.build();
689                     mSession.setPlaybackState(mSessionPlaybackState);
690                 }
691             }
692         }
693     }
694 
695     /**
696      * Sets the flags for the media transport control buttons that this client supports.
697      * @param transportControlFlags A combination of the following flags:
698      *      {@link #FLAG_KEY_MEDIA_PREVIOUS},
699      *      {@link #FLAG_KEY_MEDIA_REWIND},
700      *      {@link #FLAG_KEY_MEDIA_PLAY},
701      *      {@link #FLAG_KEY_MEDIA_PLAY_PAUSE},
702      *      {@link #FLAG_KEY_MEDIA_PAUSE},
703      *      {@link #FLAG_KEY_MEDIA_STOP},
704      *      {@link #FLAG_KEY_MEDIA_FAST_FORWARD},
705      *      {@link #FLAG_KEY_MEDIA_NEXT},
706      *      {@link #FLAG_KEY_MEDIA_POSITION_UPDATE},
707      *      {@link #FLAG_KEY_MEDIA_RATING}.
708      */
setTransportControlFlags(int transportControlFlags)709     public void setTransportControlFlags(int transportControlFlags) {
710         synchronized(mCacheLock) {
711             // store locally
712             mTransportControlFlags = transportControlFlags;
713 
714             // USE_SESSIONS
715             if (mSession != null) {
716                 PlaybackState.Builder bob = new PlaybackState.Builder(mSessionPlaybackState);
717                 bob.setActions(getActionsFromRccControlFlags(transportControlFlags));
718                 mSessionPlaybackState = bob.build();
719                 mSession.setPlaybackState(mSessionPlaybackState);
720             }
721         }
722     }
723 
724     /**
725      * Interface definition for a callback to be invoked when one of the metadata values has
726      * been updated.
727      * Implement this interface to receive metadata updates after registering your listener
728      * through {@link RemoteControlClient#setMetadataUpdateListener(OnMetadataUpdateListener)}.
729      */
730     public interface OnMetadataUpdateListener {
731         /**
732          * Called on the implementer to notify that the metadata field for the given key has
733          * been updated to the new value.
734          * @param key the identifier of the updated metadata field.
735          * @param newValue the Object storing the new value for the key.
736          */
onMetadataUpdate(int key, Object newValue)737         public abstract void onMetadataUpdate(int key, Object newValue);
738     }
739 
740     /**
741      * Sets the listener to be called whenever the metadata is updated.
742      * New metadata values will be received in the same thread as the one in which
743      * RemoteControlClient was created.
744      * @param l the metadata update listener
745      */
setMetadataUpdateListener(OnMetadataUpdateListener l)746     public void setMetadataUpdateListener(OnMetadataUpdateListener l) {
747         synchronized(mCacheLock) {
748             mMetadataUpdateListener = l;
749         }
750     }
751 
752 
753     /**
754      * Interface definition for a callback to be invoked when the media playback position is
755      * requested to be updated.
756      * @see RemoteControlClient#FLAG_KEY_MEDIA_POSITION_UPDATE
757      */
758     public interface OnPlaybackPositionUpdateListener {
759         /**
760          * Called on the implementer to notify it that the playback head should be set at the given
761          * position. If the position can be changed from its current value, the implementor of
762          * the interface must also update the playback position using
763          * {@link #setPlaybackState(int, long, float)} to reflect the actual new
764          * position being used, regardless of whether it differs from the requested position.
765          * Failure to do so would cause the system to not know the new actual playback position,
766          * and user interface components would fail to show the user where playback resumed after
767          * the position was updated.
768          * @param newPositionMs the new requested position in the current media, expressed in ms.
769          */
onPlaybackPositionUpdate(long newPositionMs)770         void onPlaybackPositionUpdate(long newPositionMs);
771     }
772 
773     /**
774      * Interface definition for a callback to be invoked when the media playback position is
775      * queried.
776      * @see RemoteControlClient#FLAG_KEY_MEDIA_POSITION_UPDATE
777      */
778     public interface OnGetPlaybackPositionListener {
779         /**
780          * Called on the implementer of the interface to query the current playback position.
781          * @return a negative value if the current playback position (or the last valid playback
782          *     position) is not known, or a zero or positive value expressed in ms indicating the
783          *     current position, or the last valid known position.
784          */
onGetPlaybackPosition()785         long onGetPlaybackPosition();
786     }
787 
788     /**
789      * Sets the listener to be called whenever the media playback position is requested
790      * to be updated.
791      * Notifications will be received in the same thread as the one in which RemoteControlClient
792      * was created.
793      * @param l the position update listener to be called
794      */
setPlaybackPositionUpdateListener(OnPlaybackPositionUpdateListener l)795     public void setPlaybackPositionUpdateListener(OnPlaybackPositionUpdateListener l) {
796         synchronized(mCacheLock) {
797             mPositionUpdateListener = l;
798         }
799     }
800 
801     /**
802      * Sets the listener to be called whenever the media current playback position is needed.
803      * Queries will be received in the same thread as the one in which RemoteControlClient
804      * was created.
805      * @param l the listener to be called to retrieve the playback position
806      */
setOnGetPlaybackPositionListener(OnGetPlaybackPositionListener l)807     public void setOnGetPlaybackPositionListener(OnGetPlaybackPositionListener l) {
808         synchronized(mCacheLock) {
809             mPositionProvider = l;
810         }
811     }
812 
813     /**
814      * @hide
815      * Flag to reflect that the application controlling this RemoteControlClient sends playback
816      * position updates. The playback position being "readable" is considered from the application's
817      * point of view.
818      */
819     @UnsupportedAppUsage
820     public static int MEDIA_POSITION_READABLE = 1 << 0;
821     /**
822      * @hide
823      * Flag to reflect that the application controlling this RemoteControlClient can receive
824      * playback position updates. The playback position being "writable"
825      * is considered from the application's point of view.
826      */
827     @UnsupportedAppUsage
828     public static int MEDIA_POSITION_WRITABLE = 1 << 1;
829 
830     /** @hide */
831     public final static int DEFAULT_PLAYBACK_VOLUME_HANDLING = PLAYBACK_VOLUME_VARIABLE;
832     /** @hide */
833     // hard-coded to the same number of steps as AudioService.MAX_STREAM_VOLUME[STREAM_MUSIC]
834     public final static int DEFAULT_PLAYBACK_VOLUME = 15;
835 
836     /**
837      * Lock for all cached data
838      */
839     private final Object mCacheLock = new Object();
840     /**
841      * Cache for the playback state.
842      * Access synchronized on mCacheLock
843      */
844     private int mPlaybackState = PLAYSTATE_NONE;
845     /**
846      * Time of last play state change
847      * Access synchronized on mCacheLock
848      */
849     private long mPlaybackStateChangeTimeMs = 0;
850     /**
851      * Last playback position in ms reported by the user
852      */
853     private long mPlaybackPositionMs = PLAYBACK_POSITION_INVALID;
854     /**
855      * Last playback speed reported by the user
856      */
857     private float mPlaybackSpeed = PLAYBACK_SPEED_1X;
858     /**
859      * Cache for the artwork bitmap.
860      * Access synchronized on mCacheLock
861      * Artwork and metadata are not kept in one Bundle because the bitmap sometimes needs to be
862      * accessed to be resized, in which case a copy will be made. This would add overhead in
863      * Bundle operations.
864      */
865     private Bitmap mOriginalArtwork;
866     /**
867      * Cache for the transport control mask.
868      * Access synchronized on mCacheLock
869      */
870     private int mTransportControlFlags = FLAGS_KEY_MEDIA_NONE;
871     /**
872      * Cache for the metadata strings.
873      * Access synchronized on mCacheLock
874      * This is re-initialized in apply() and so cannot be final.
875      */
876     private Bundle mMetadata = new Bundle();
877     /**
878      * Listener registered by user of RemoteControlClient to receive requests for playback position
879      * update requests.
880      */
881     private OnPlaybackPositionUpdateListener mPositionUpdateListener;
882     /**
883      * Provider registered by user of RemoteControlClient to provide the current playback position.
884      */
885     private OnGetPlaybackPositionListener mPositionProvider;
886     /**
887      * Listener registered by user of RemoteControlClient to receive edit changes to metadata
888      * it exposes.
889      */
890     private OnMetadataUpdateListener mMetadataUpdateListener;
891     /**
892      * The current remote control client generation ID across the system, as known by this object
893      */
894     private int mCurrentClientGenId = -1;
895 
896     /**
897      * The media button intent description associated with this remote control client
898      * (can / should include target component for intent handling, used when persisting media
899      *    button event receiver across reboots).
900      */
901     private final PendingIntent mRcMediaIntent;
902 
903     /**
904      * Reflects whether any "plugged in" IRemoteControlDisplay has mWantsPositonSync set to true.
905      */
906     // TODO consider using a ref count for IRemoteControlDisplay requiring sync instead
907     private boolean mNeedsPositionSync = false;
908 
909     /**
910      * Cache for the current playback state using Session APIs.
911      */
912     private PlaybackState mSessionPlaybackState = null;
913 
914     /**
915      * Cache for metadata using Session APIs. This is re-initialized in apply().
916      */
917     private MediaMetadata mMediaMetadata;
918 
919     /**
920      * @hide
921      * Accessor to media button intent description (includes target component)
922      */
getRcMediaIntent()923     public PendingIntent getRcMediaIntent() {
924         return mRcMediaIntent;
925     }
926 
927     /**
928      * @hide
929      * Default value for the unique identifier
930      */
931     public final static int RCSE_ID_UNREGISTERED = -1;
932 
933     // USE_SESSIONS
934     private MediaSession.Callback mTransportListener = new MediaSession.Callback() {
935 
936         @Override
937         public void onSeekTo(long pos) {
938             RemoteControlClient.this.onSeekTo(mCurrentClientGenId, pos);
939         }
940 
941         @Override
942         public void onSetRating(Rating rating) {
943             if ((mTransportControlFlags & FLAG_KEY_MEDIA_RATING) != 0) {
944                 onUpdateMetadata(mCurrentClientGenId, MetadataEditor.RATING_KEY_BY_USER, rating);
945             }
946         }
947     };
948 
949     //===========================================================
950     // Message handlers
951 
onSeekTo(int generationId, long timeMs)952     private void onSeekTo(int generationId, long timeMs) {
953         synchronized (mCacheLock) {
954             if ((mCurrentClientGenId == generationId) && (mPositionUpdateListener != null)) {
955                 mPositionUpdateListener.onPlaybackPositionUpdate(timeMs);
956             }
957         }
958     }
959 
onUpdateMetadata(int generationId, int key, Object value)960     private void onUpdateMetadata(int generationId, int key, Object value) {
961         synchronized (mCacheLock) {
962             if ((mCurrentClientGenId == generationId) && (mMetadataUpdateListener != null)) {
963                 mMetadataUpdateListener.onMetadataUpdate(key, value);
964             }
965         }
966     }
967 
968     //===========================================================
969     // Internal utilities
970 
971     /**
972      * Returns whether, for the given playback state, the playback position is expected to
973      * be changing.
974      * @param playstate the playback state to evaluate
975      * @return true during any form of playback, false if it's not playing anything while in this
976      *     playback state
977      */
playbackPositionShouldMove(int playstate)978     static boolean playbackPositionShouldMove(int playstate) {
979         switch(playstate) {
980             case PLAYSTATE_STOPPED:
981             case PLAYSTATE_PAUSED:
982             case PLAYSTATE_BUFFERING:
983             case PLAYSTATE_ERROR:
984             case PLAYSTATE_SKIPPING_FORWARDS:
985             case PLAYSTATE_SKIPPING_BACKWARDS:
986                 return false;
987             case PLAYSTATE_PLAYING:
988             case PLAYSTATE_FAST_FORWARDING:
989             case PLAYSTATE_REWINDING:
990             default:
991                 return true;
992         }
993     }
994 
995     /**
996      * Period for playback position drift checks, 15s when playing at 1x or slower.
997      */
998     private final static long POSITION_REFRESH_PERIOD_PLAYING_MS = 15000;
999 
1000     /**
1001      * Minimum period for playback position drift checks, never more often when every 2s, when
1002      * fast forwarding or rewinding.
1003      */
1004     private final static long POSITION_REFRESH_PERIOD_MIN_MS = 2000;
1005 
1006     /**
1007      * The value above which the difference between client-reported playback position and
1008      * estimated position is considered a drift.
1009      */
1010     private final static long POSITION_DRIFT_MAX_MS = 500;
1011 
1012     /**
1013      * Compute the period at which the estimated playback position should be compared against the
1014      * actual playback position. Is a funciton of playback speed.
1015      * @param speed 1.0f is normal playback speed
1016      * @return the period in ms
1017      */
getCheckPeriodFromSpeed(float speed)1018     private static long getCheckPeriodFromSpeed(float speed) {
1019         if (Math.abs(speed) <= 1.0f) {
1020             return POSITION_REFRESH_PERIOD_PLAYING_MS;
1021         } else {
1022             return Math.max((long)(POSITION_REFRESH_PERIOD_PLAYING_MS / Math.abs(speed)),
1023                     POSITION_REFRESH_PERIOD_MIN_MS);
1024         }
1025     }
1026 
1027     /**
1028      * Get the {@link PlaybackState} state for the given
1029      * {@link RemoteControlClient} state.
1030      *
1031      * @param rccState The state used by {@link RemoteControlClient}.
1032      * @return The equivalent state used by {@link PlaybackState}.
1033      */
getStateFromRccState(int rccState)1034     private static int getStateFromRccState(int rccState) {
1035         switch (rccState) {
1036             case PLAYSTATE_BUFFERING:
1037                 return PlaybackState.STATE_BUFFERING;
1038             case PLAYSTATE_ERROR:
1039                 return PlaybackState.STATE_ERROR;
1040             case PLAYSTATE_FAST_FORWARDING:
1041                 return PlaybackState.STATE_FAST_FORWARDING;
1042             case PLAYSTATE_NONE:
1043                 return PlaybackState.STATE_NONE;
1044             case PLAYSTATE_PAUSED:
1045                 return PlaybackState.STATE_PAUSED;
1046             case PLAYSTATE_PLAYING:
1047                 return PlaybackState.STATE_PLAYING;
1048             case PLAYSTATE_REWINDING:
1049                 return PlaybackState.STATE_REWINDING;
1050             case PLAYSTATE_SKIPPING_BACKWARDS:
1051                 return PlaybackState.STATE_SKIPPING_TO_PREVIOUS;
1052             case PLAYSTATE_SKIPPING_FORWARDS:
1053                 return PlaybackState.STATE_SKIPPING_TO_NEXT;
1054             case PLAYSTATE_STOPPED:
1055                 return PlaybackState.STATE_STOPPED;
1056             default:
1057                 return -1;
1058         }
1059     }
1060 
1061     /**
1062      * Get the {@link RemoteControlClient} state for the given
1063      * {@link PlaybackState} state.
1064      *
1065      * @param state The state used by {@link PlaybackState}.
1066      * @return The equivalent state used by {@link RemoteControlClient}.
1067      */
getRccStateFromState(int state)1068     static int getRccStateFromState(int state) {
1069         switch (state) {
1070             case PlaybackState.STATE_BUFFERING:
1071                 return PLAYSTATE_BUFFERING;
1072             case PlaybackState.STATE_ERROR:
1073                 return PLAYSTATE_ERROR;
1074             case PlaybackState.STATE_FAST_FORWARDING:
1075                 return PLAYSTATE_FAST_FORWARDING;
1076             case PlaybackState.STATE_NONE:
1077                 return PLAYSTATE_NONE;
1078             case PlaybackState.STATE_PAUSED:
1079                 return PLAYSTATE_PAUSED;
1080             case PlaybackState.STATE_PLAYING:
1081                 return PLAYSTATE_PLAYING;
1082             case PlaybackState.STATE_REWINDING:
1083                 return PLAYSTATE_REWINDING;
1084             case PlaybackState.STATE_SKIPPING_TO_PREVIOUS:
1085                 return PLAYSTATE_SKIPPING_BACKWARDS;
1086             case PlaybackState.STATE_SKIPPING_TO_NEXT:
1087                 return PLAYSTATE_SKIPPING_FORWARDS;
1088             case PlaybackState.STATE_STOPPED:
1089                 return PLAYSTATE_STOPPED;
1090             default:
1091                 return -1;
1092         }
1093     }
1094 
getActionsFromRccControlFlags(int rccFlags)1095     private static long getActionsFromRccControlFlags(int rccFlags) {
1096         long actions = 0;
1097         long flag = 1;
1098         while (flag <= rccFlags) {
1099             if ((flag & rccFlags) != 0) {
1100                 actions |= getActionForRccFlag((int) flag);
1101             }
1102             flag = flag << 1;
1103         }
1104         return actions;
1105     }
1106 
getRccControlFlagsFromActions(long actions)1107     static int getRccControlFlagsFromActions(long actions) {
1108         int rccFlags = 0;
1109         long action = 1;
1110         while (action <= actions && action < Integer.MAX_VALUE) {
1111             if ((action & actions) != 0) {
1112                 rccFlags |= getRccFlagForAction(action);
1113             }
1114             action = action << 1;
1115         }
1116         return rccFlags;
1117     }
1118 
getActionForRccFlag(int flag)1119     private static long getActionForRccFlag(int flag) {
1120         switch (flag) {
1121             case FLAG_KEY_MEDIA_PREVIOUS:
1122                 return PlaybackState.ACTION_SKIP_TO_PREVIOUS;
1123             case FLAG_KEY_MEDIA_REWIND:
1124                 return PlaybackState.ACTION_REWIND;
1125             case FLAG_KEY_MEDIA_PLAY:
1126                 return PlaybackState.ACTION_PLAY;
1127             case FLAG_KEY_MEDIA_PLAY_PAUSE:
1128                 return PlaybackState.ACTION_PLAY_PAUSE;
1129             case FLAG_KEY_MEDIA_PAUSE:
1130                 return PlaybackState.ACTION_PAUSE;
1131             case FLAG_KEY_MEDIA_STOP:
1132                 return PlaybackState.ACTION_STOP;
1133             case FLAG_KEY_MEDIA_FAST_FORWARD:
1134                 return PlaybackState.ACTION_FAST_FORWARD;
1135             case FLAG_KEY_MEDIA_NEXT:
1136                 return PlaybackState.ACTION_SKIP_TO_NEXT;
1137             case FLAG_KEY_MEDIA_POSITION_UPDATE:
1138                 return PlaybackState.ACTION_SEEK_TO;
1139             case FLAG_KEY_MEDIA_RATING:
1140                 return PlaybackState.ACTION_SET_RATING;
1141         }
1142         return 0;
1143     }
1144 
getRccFlagForAction(long action)1145     private static int getRccFlagForAction(long action) {
1146         // We only care about the lower set of actions that can map to rcc
1147         // flags.
1148         int testAction = action < Integer.MAX_VALUE ? (int) action : 0;
1149         switch (testAction) {
1150             case (int) PlaybackState.ACTION_SKIP_TO_PREVIOUS:
1151                 return FLAG_KEY_MEDIA_PREVIOUS;
1152             case (int) PlaybackState.ACTION_REWIND:
1153                 return FLAG_KEY_MEDIA_REWIND;
1154             case (int) PlaybackState.ACTION_PLAY:
1155                 return FLAG_KEY_MEDIA_PLAY;
1156             case (int) PlaybackState.ACTION_PLAY_PAUSE:
1157                 return FLAG_KEY_MEDIA_PLAY_PAUSE;
1158             case (int) PlaybackState.ACTION_PAUSE:
1159                 return FLAG_KEY_MEDIA_PAUSE;
1160             case (int) PlaybackState.ACTION_STOP:
1161                 return FLAG_KEY_MEDIA_STOP;
1162             case (int) PlaybackState.ACTION_FAST_FORWARD:
1163                 return FLAG_KEY_MEDIA_FAST_FORWARD;
1164             case (int) PlaybackState.ACTION_SKIP_TO_NEXT:
1165                 return FLAG_KEY_MEDIA_NEXT;
1166             case (int) PlaybackState.ACTION_SEEK_TO:
1167                 return FLAG_KEY_MEDIA_POSITION_UPDATE;
1168             case (int) PlaybackState.ACTION_SET_RATING:
1169                 return FLAG_KEY_MEDIA_RATING;
1170         }
1171         return 0;
1172     }
1173 }
1174