gpt4 book ai didi

android - 服务停止时 Widget 的 PendingIntent 丢失

转载 作者:行者123 更新时间:2023-11-30 00:50:42 25 4
gpt4 key购买 nike

我有一个音乐播放器应用程序,带有 MusicPlayingService(扩展 MediaBrowserServiceCompat)和一个控制服务的小部件。 MusicPlayingService 还引用了一个 MusicStateManager(在 MusicServices onCreate 中创建),它实现了“MediaSessionCompat.Callback”并处理来自当前 MediaSession 的所有回调:mSession .setCallback(MusicStateManager);

public class MusicPlayingService extends MediaBrowserServiceCompat implements
AudioManager.OnAudioFocusChangeListener {

@RequiresPermission(Manifest.permission.READ_PHONE_STATE)
@Override
public void onCreate() {
super.onCreate();
Log.d(TAG, "MusicPlayingService started");

.....

mPlayerStateManager = new MusicStateManager(this);

.....
}

@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.d(TAG, "onStartCommand");
MediaButtonReceiver.handleIntent(mPlayerStateManager.getSession(), intent);
return START_STICKY;
}

我的 Widget 然后有一堆 Pending Intent,它们被发送到 MusicPlayingService 并由 MusicStateManager 处理。一切都很好。

public class MusicStateManager extends MediaSessionCompat.Callback {

....

public MusicStateManager(@NonNull MusicStateManager argService) {
Log.d(TAG, "setting service");
mPlayerService = argService;

ComponentName mediaButtonReceiver = new ComponentName(mPlayerService, HeadsetReceiver.class);
mSession = new MediaSessionCompat(mPlayerService, SESSION_TAG, mediaButtonReceiver, null);
mSession.setFlags(MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS |
MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS);

Intent mediaButtonIntent = new Intent(Intent.ACTION_MEDIA_BUTTON);
PendingIntent pendingMediaButtonIntent = PendingIntent.getBroadcast(mPlayerService, 0, mediaButtonIntent, 0);
mSession.setMediaButtonReceiver(pendingMediaButtonIntent);

Intent toggleIntent = new Intent(NotificationPlayer.toggleAction);
PendingIntent pendingToggleIntent = PendingIntent.getBroadcast(mPlayerService, 0, toggleIntent, 0);
mSession.setMediaButtonReceiver(pendingToggleIntent);

mSession.setCallback(this);
mSession.setActive(true);

}

/**
* Callback method called from MusicStateManager whenever the music is about to play.
*/
public void onPlay() {
Log.d(TAG, "onPlay");

.......
}

@Override
public void onCustomAction(String action, Bundle extras) {
Log.d(TAG, "received action: " + action); // NoI18N

if (ACTION_TOGGLE.equals(action)) {
mPlayerService.toggle();
}
}

事情是这样的。如果我强制停止应用程序,音乐服务将关闭(很明显)。如果我然后按下小部件上的按钮,它会再次启动我的服务(我可以从 LogCat 中的各种 Log.d(...) 中看到它。但是,小部件中的 PendingIntent 丢失了,并且从未处理过。

这对用户有奇怪的影响。大多数情况下,小部件上的按钮会起作用(当服务正在运行时)。但是,有时(当服务未运行时)按钮在第一次按下时不会执行任何操作,但如果您再次按下它们,它们就会起作用(因为第一次按下启动了服务,但未能处理 Intent 。

知道如何调试这个问题并查看我的 PendingIntent 会发生什么吗?

最佳答案

为媒体播放创建 PendingIntent 的最佳方法是使用 MediaButtonReceiver.buildMediaButtonPendingIntent() .这允许您传入将触发正确回调的 PlaybackStateCompat.ACTION_ 常量。

PendingIntent playPendingIntent = MediaButtonReceiver
.buildMediaButtonPendingIntent(context, PlaybackStateCompat.ACTION_PLAY);

您似乎已经在 onStartCommand() 中调用了正确的 MediaButtonReceiver.handleIntent(),但您还应该确保在创建 MediaSessionCompat 通过调用 setPlaybackState() 启用正确的操作:

// Ideally, you should keep a reference to this Builder to update it,
// rather than create it from scratch each time
PlaybackStateCompat.Builder playbackStateBuilder = new PlaybackStateCompat.Builder()
.setActions(PlaybackStateCompat.ACTION_PLAY |
PlaybackStateCompat.ACTION_PLAY_PAUSE);
mSession.setPlaybackState(playbackStateBuilder.build());

因为 MediaSessionCompat 默认情况下不允许任何媒体按钮。通过确保直接从实例化支持 ACTION_PLAY,您可以确保第一个 MediaButtonReceiver.handleIntent() 正确调用您的 MediaSessionCompat.Callback

关于android - 服务停止时 Widget 的 PendingIntent 丢失,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41080396/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com