我正在尝试在Android中实现通知播放器。我遵循this指南来实现这一目标。但是,根据我的要求,我需要在后台通知播放器中实现“跳到下一个”和“跳到上一个”。
到目前为止,已触发MediaSessionCompact.Callback的onPlay()和onPause()。但是,在调试时,我注意到在通知播放器中按下下一个和上一个按钮时,不会触发onSkipToNext()和onSkipToPrevious()回调方法。
感谢您的输入。我在下面添加了相关代码。
BackgroundAudioService
private void showPlayingNotification() {
NotificationCompat.Builder builder = MediaStyleHelper.from(BackgroundAudioService.this, mMediaSessionCompat);
if (builder == null) {
return;
}
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_previous, "Previous", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS)));
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_pause, "Pause", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_PLAY_PAUSE)));
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_next, "Next", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_SKIP_TO_NEXT)));
builder.setStyle(new NotificationCompat.MediaStyle().setShowActionsInCompactView(0).setMediaSession(mMediaSessionCompat.getSessionToken()));
builder.setSmallIcon(R.mipmap.ic_launcher);
NotificationManagerCompat.from(BackgroundAudioService.this).notify(1, builder.build());
}
private void showPausedNotification() {
NotificationCompat.Builder builder = MediaStyleHelper.from(this, mMediaSessionCompat);
if (builder == null) {
return;
}
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_previous, "Previous", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS)));
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_play, "Play", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_PLAY_PAUSE)));
builder.addAction(new NotificationCompat.Action(android.R.drawable.ic_media_next, "Next", MediaButtonReceiver.buildMediaButtonPendingIntent(this, PlaybackStateCompat.ACTION_SKIP_TO_NEXT)));
builder.setStyle(new NotificationCompat.MediaStyle().setShowActionsInCompactView(0).setMediaSession(mMediaSessionCompat.getSessionToken()));
builder.setSmallIcon(R.mipmap.ic_launcher);
NotificationManagerCompat.from(this).notify(1, builder.build());
}
private MediaSessionCompat.Callback mMediaSessionCallback = new MediaSessionCompat.Callback() {
@Override
public void onPlay() {
super.onPlay();
if (!successfullyRetrievedAudioFocus()) {
return;
}
mMediaSessionCompat.setActive(true);
setMediaPlaybackState(PlaybackStateCompat.STATE_PLAYING);
Log.d("MyLog", "STATE_PLAYING");
showPlayingNotification();
mMediaPlayer.start();
}
@Override
public void onPause() {
super.onPause();
if (mMediaPlayer.isPlaying()) {
mMediaPlayer.pause();
setMediaPlaybackState(PlaybackStateCompat.STATE_PAUSED);
Log.d("MyLog", "STATE_PAUSED");
showPausedNotification();
}
}
@Override
public void onSkipToNext() {
super.onSkipToNext();
setMediaPlaybackState(PlaybackStateCompat.STATE_SKIPPING_TO_NEXT);
Log.d("MyLog", "STATE_SKIPPING_TO_NEXT");
}
@Override
public void onSkipToPrevious() {
super.onSkipToPrevious();
setMediaPlaybackState(PlaybackStateCompat.STATE_SKIPPING_TO_PREVIOUS);
Log.d("MyLog", "STATE_SKIPPING_TO_PREVIOUS");
}
}
MediaStyleHelper
public static NotificationCompat.Builder from(
Context context, MediaSessionCompat mediaSession) {
MediaControllerCompat controller = mediaSession.getController();
MediaMetadataCompat mediaMetadata = controller.getMetadata();
MediaDescriptionCompat description = mediaMetadata.getDescription();
NotificationCompat.Builder builder = new NotificationCompat.Builder(context);
builder
.setContentTitle(description.getTitle())
.setContentText(description.getSubtitle())
.setSubText(description.getDescription())
.setLargeIcon(description.getIconBitmap())
.setContentIntent(controller.getSessionActivity())
.setDeleteIntent(
MediaButtonReceiver.buildMediaButtonPendingIntent(context, PlaybackStateCompat.ACTION_STOP))
.setVisibility(NotificationCompat.VISIBILITY_PUBLIC);
return builder;
}
MainActivity 的onCreate
mMediaBrowserCompat = new MediaBrowserCompat(this, new ComponentName(this, BackgroundAudioService.class),
mMediaBrowserCompatConnectionCallback, getIntent().getExtras());
MainActivity
private MediaBrowserCompat.ConnectionCallback mMediaBrowserCompatConnectionCallback = new MediaBrowserCompat.ConnectionCallback() {
@Override
public void onConnected() {
super.onConnected();
try {
mMediaControllerCompat = new MediaControllerCompat(MainActivity.this, mMediaBrowserCompat.getSessionToken());
mMediaControllerCompat.registerCallback(mMediaControllerCompatCallback);
setSupportMediaController(mMediaControllerCompat);
getSupportMediaController().getTransportControls().playFromMediaId(String.valueOf(R.raw.warner_tautz_off_broadway), null);
} catch( RemoteException e ) {
}
}
};
private MediaControllerCompat.Callback mMediaControllerCompatCallback = new MediaControllerCompat.Callback() {
@Override
public void onPlaybackStateChanged(PlaybackStateCompat state) {
super.onPlaybackStateChanged(state);
if( state == null ) {
return;
}
switch( state.getState() ) {
case PlaybackStateCompat.STATE_PLAYING: {
mCurrentState = STATE_PLAYING;
break;
}
case PlaybackStateCompat.STATE_PAUSED: {
mCurrentState = STATE_PAUSED;
break;
}
case PlaybackStateCompat.STATE_SKIPPING_TO_NEXT: {
mCurrentState = STATE_SKIPPING_TO_NEXT;
break;
}
case PlaybackStateCompat.STATE_SKIPPING_TO_PREVIOUS: {
mCurrentState = STATE_SKIPPING_TO_PREVIOUS;
}
}
}
};
更新
似乎onMediaButtonEvent()在通知播放器中的下一个和上一个触发,两个按钮的 Action 名称为“android.intent.action.MEDIA_BUTTON”。
@Override
public boolean onMediaButtonEvent(Intent mediaButtonEvent) {
Log.d("MyLog", "executing onMediaButtonEvent: " + mediaButtonEvent.getAction());
return super.onMediaButtonEvent(mediaButtonEvent);
}
最佳答案
遵循相同的指南后就遇到了这个问题。
在BackgroundAudioService类中寻找一种称为setMediaPlaybackState(int state)
的方法。在这里,您可以定义一个PlaybackStateCompat.Builder并使用setActions定义 session 中的当前可用功能。
除非将ACTION_SKIP_TO_NEXT和ACTION_SKIP_TO_PREVIOUS作为支持的 Action 传入,否则不会调用OnSkipToNext和onSkipToPrevious。将setActions更新为以下内容后,我设法使其正常运行。
private void setMediaPlaybackState(int state) {
PlaybackStateCompat.Builder playbackstateBuilder = new PlaybackStateCompat.Builder();
switch (state) {
case PlaybackStateCompat.STATE_PLAYING: {
playbackstateBuilder.setActions(PlaybackStateCompat.ACTION_PLAY_PAUSE | PlaybackStateCompat.ACTION_PAUSE
| PlaybackStateCompat.ACTION_SKIP_TO_NEXT | PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS);
break;
}
default: {
playbackstateBuilder.setActions(PlaybackStateCompat.ACTION_PLAY_PAUSE | PlaybackStateCompat.ACTION_PLAY
| PlaybackStateCompat.ACTION_SKIP_TO_NEXT | PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS);
break;
}
}
playbackstateBuilder.setState(state, PlaybackStateCompat.PLAYBACK_POSITION_UNKNOWN, 0);
mMediaSessionCompat.setPlaybackState(playbackstateBuilder.build());
}
如果有一种更简洁的方法可以很好地做到这一点,但似乎所有操作都必须设置在一个匹配中,并且不能在多个调用之间进行拆分。