Using MediaController on Android 5

7k views Asked by At

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?

2

There are 2 answers

6
mangini On BEST ANSWER

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:

  1. If you just want to skip musics or play/pause, you can send media button events (via AudioManager) and/or request/release audio focus.

  2. 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

Original response:

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.

0
Sagar On

Check for following permission

private void notificationAccessPermission() {
    if (Settings.Secure.getString(this.getContentResolver(), "enabled_notification_listeners").contains(getApplicationContext().getPackageName())) {
        //service is enabled do something
    } else {
        Intent intent = new
                Intent("android.settings.ACTION_NOTIFICATION_LISTENER_SETTINGS");
        startActivity(intent);
    }
}