For an Android app I’m working on I need to find out what song, if any, is playing in the background on the device. Turns out there’s no documented way to do this on Android - but it also turns out that using an open source platform is great. After a few hours of poking around I had a working proof of concept, and I’m sure someone more familiar with Android would’ve figured it out much faster.
IMPORTANT: I’m using an undocumented interface. While this works on Android 1.5R3 and T-Mobile G1 (HTC Magic), and likely on other versions/devices, there is no guarantee it’ll keep working in future releases - use at your own risk.
The default way to play music on the G1 is the Music app. Since it can play music in the background, it seemed likely there is some way to interact with the service it uses. This of course doesn’t help if the user is using some other music player, but it should be good enough.
First step was to figure out exactly what service that is - I used something like the following code to find out what services are running while I’m playing music:
ActivityManager am = (ActivityManager)this.getSystemService(ACTIVITY_SERVICE);
List<ActivityManager.RunningServiceInfo> rs = am.getRunningServices(50);
for (int i=0; i<rs.size(); i++) {
ActivityManager.RunningServiceInfo rsi = rs.get(i);
Log.i("Service", "Process " + rsi.process + " with component " + rsi.service.getClassName());
}
This prints up to 50 running services. The interesting line in the output is
process com.android.music with component com.android.music.MediaPlaybackService
OK, now to figure out how to interact with MediaPlaybackService. It turns out to be surprisingly simple.
First, get a copy of IMediaPlaybackService.aidl from the source code of the Music app, and include it in your Android app. The contents of this file are the methods we have access to in MediaPlaybackService. Of course, if the Music app on the device changes, and our copy of the .aidl file is inaccurate, we have a problem - hence the warning above.
Next, implement a ServiceConnection we can use to connect to MediaPlaybackService.
private class MediaPlayerServiceConnection implements ServiceConnection {
public IMediaPlaybackService mService;
public void onServiceConnected(ComponentName name, IBinder service) {
Log.i("MediaPlayerServiceConnection", "Connected! Name: " + name.getClassName());
// This is the important line
mService = IMediaPlaybackService.Stub.asInterface(service);
// If all went well, now we can use the interface
try {
Log.i("MediaPlayerServiceConnection", "Playing track: " + mService.getTrackName());
Log.i("MediaPlayerServiceConnection", "By artist: " + mService.getArtistName());
if (mService.isPlaying()) {
Log.i("MediaPlayerServiceConnection", "Music player is playing.");
} else {
Log.i("MediaPlayerServiceConnection", "Music player is not playing.");
}
} catch (Exception e) {
e.printStackTrace();
throw new RuntimeException(e);
}
}
public void onServiceDisconnected(ComponentName name) {
Log.i("MediaPlayerServiceConnection", "Disconnected!");
}
}
Finally, bind the service:
Intent i = new Intent();
i.setClassName("com.android.music", "com.android.music.MediaPlaybackService");
ServiceConnection conn = new MediaPlayerServiceConnection();
this.bindService(i, conn, 0);
That’s it! From here on it should be possible to use the MediaPlaybackService like any other service.
This comment has been removed by the author.
ReplyDelete