I want to use the new MediaController
in Android 5 instead of the old RemoteController
for getting the currently playing track and changing the track.
I couldn't find any examples so I tried it by myself.
To get the current MediaController
, I have implemented a Class which extends MediaController.Callback
and implements MediaSessionManager.OnActiveSessionsChangedListener
.
With this method I try to get the current mediaController
:
@Override
public void onActiveSessionsChanged(List<MediaController> mediaControllers) {
Log.i(Util.LOG_TAG, "sessions changed");
if (mediaControllers != null && !mediaControllers.isEmpty()) {
this.mediaController = mediaControllers.get(0);
mediaController.registerCallback(this);
}
}
But I think I have to register my class. So I have done
MediaSessionManager mgr = (MediaSessionManager)context.getSystemService(Context.MEDIA_SESSION_SERVICE);
mgr.addOnActiveSessionsChangedListener(this, null);
But when I do this I get an Error that I lack the permission to control media. When I tried to add this permission I noticed that this permission can't be used by third party apps.
What am I doing wrong?
UPDATE: A comment clarified that the question was actually about viewing/controlling other apps' MediaSessions, not your own.
Although there is no way to do it directly, due to privacy reasons, you have two options with different levels of available controls/information and different requirements for user interaction:
If you just want to skip musics or play/pause, you can send media button events (via AudioManager) and/or request/release audio focus.
If you need to also know the metadata of what is currently playing, there is a more intrusive way that requires explicit user interaction:
First, create and register a NotificationListenerService:
public class NotificationListener extends NotificationListenerService {
public NotificationListener() {
}
}
In AndroidManifest.xml:
<service android:name=".NotificationListener"
android:permission="android.permission.BIND_NOTIFICATION_LISTENER_SERVICE"
android:enabled="true" android:exported="true">
<intent-filter>
<action android:name="android.service.notification.NotificationListenerService" />
</intent-filter>
</service>
Then you will be able to fetch MediaSessions by specifying your NotificationListenerService componentName in the getActiveSessions call:
MediaSessionManager mm = (MediaSessionManager) this.getSystemService(
Context.MEDIA_SESSION_SERVICE);
List<MediaController> controllers = mm.getActiveSessions(
new ComponentName(this, NotificationListener.class));
Log.i(TAG, "found " + controllers.size() + " controllers");
One caveat is that the user will need to explicitly give your app Notification access permission, by going to Settings -> Sound & Notification -> Notification access
The object that describes your MediaSession and that can be passed along to allow other components/apps to control an existing MediaSesion is the MediaSession.Token. With a token, you can create a MediaController directly, without resorting to the MediaSessionManager. The code for that would be something like:
MediaController mediaController = new MediaController(getActivity(),
sessionToken);
This doesn't require any special permission. If you are also using the MediaBrowser or the MediaBrowserService, you should get the token associated with the MediaBrowser, by using it's getSessionToken() method.
We just released an example that uses a MediaBrowserService to handle music browsing, playback and media style notification and provides a simple Activity to control playback.