Google Play音乐是否会占用所有ACTION_MEDIA_BUTTON意图?

当发送带有ACTION_MEDIA_BUTTON意图的sendOrderedBroadcast时(我模仿用户正在点击蓝牙耳机上的播放按钮),Google Play音乐会打开并播放最后播放的专辑而不是前景音乐播放应用。

如果我将其更改为sendBroadcast,Google Play音乐和当前音乐播放应用(在我的情况下为Pandora)都会制作播放按钮。

这仅发生在Android 4.0及更高版本中 。 Play Music是否会占用这个意图(一个bug)? 您是否怀疑Pandora未按照以下建议注册为当前媒体按钮处理程序: http : //android-developers.blogspot.com/2010/06/allowing-applications-to-play-nicer.html

有没有办法可以将此意图指向当前的音乐播放应用程序?

这是我的代码:

public void broadcastMediaIntent(MediaIntent intentType){ long eventtime = SystemClock.uptimeMillis(); Intent downIntent = new Intent(Intent.ACTION_MEDIA_BUTTON, null); Intent upIntent = new Intent(Intent.ACTION_MEDIA_BUTTON, null); KeyEvent downEvent = null; KeyEvent upEvent = null; switch(intentType){ case NEXT: downEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MEDIA_NEXT, 0); upEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_UP, KeyEvent.KEYCODE_MEDIA_NEXT, 0); break; case PLAY_PAUSE: downEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE, 0); upEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_UP, KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE, 0); break; case PREVIOUS: downEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MEDIA_PREVIOUS, 0); upEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_UP, KeyEvent.KEYCODE_MEDIA_PREVIOUS, 0); break; case FAST_FORWARD: downEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MEDIA_FAST_FORWARD, 0); upEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_UP, KeyEvent.KEYCODE_MEDIA_FAST_FORWARD, 0); break; case REWIND: downEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_DOWN, KeyEvent.KEYCODE_MEDIA_REWIND, 0); upEvent = new KeyEvent(eventtime, eventtime, KeyEvent.ACTION_UP, KeyEvent.KEYCODE_MEDIA_REWIND, 0); break; default: break; } downIntent.putExtra(Intent.EXTRA_KEY_EVENT, downEvent); sendOrderedBroadcast(downIntent, null); upIntent.putExtra(Intent.EXTRA_KEY_EVENT, upEvent); sendOrderedBroadcast(upIntent, null); } 

有一个API必须用作这些意图的首选接收者,但这当然可以由系统密钥的发送者处理。 看看这篇文章和文档 。 但在这种情况下,这取决于潘多拉和谷歌音乐,所以这可能不适合你。 您当然也可以将您的广播发送到特定的包(通过在意图中指定组件名称),然后您决定哪个应用程序获取它。 我在AudioManager中快速搜索了一个隐藏的API,但看起来并不乐观。

你试过配件吗? 如果这样可行,那么我将研究如何发送该意图。 如果不是,我会看看安装了哪些应用程序,并进行“智能”猜测或询问用户将意图发送到哪个应用程序。 也许最后一个是最好的方式,因为它使用公共API,并不会通过猜测错误来惹恼用户:)

编辑:这可能有效,但它也可能受到权限和证书的保护。 在锁屏中,这是媒体键的处理方式:

 void handleMediaKeyEvent(KeyEvent keyEvent) { IAudioService audioService = IAudioService.Stub.asInterface( ServiceManager.checkService(Context.AUDIO_SERVICE)); if (audioService != null) { try { audioService.dispatchMediaKeyEvent(keyEvent); } catch (RemoteException e) { Log.e("KeyguardViewBase", "dispatchMediaKeyEvent threw exception " + e); } } else { Slog.w("KeyguardViewBase", "Unable to find IAudioService for media key event"); } } 

但是,API是隐藏的,因此您必须解决这个问题。 这可能是我能帮助你的最好的。 另一种方法是注入事件。 一种方法是成为一种输入法,但这不太可能是前进的方法。 有几种方法可以将事件注入到您自己的活动中,但我不知道是否有任何注入系统的方法。 也许你可以看看仪器测试是如何做到的。

以下应该可以解决问题。 顺便说一下,你在哪里find了锁屏的源代码?

  public void handleMediaKeyEvent(KeyEvent keyEvent) { /* * Attempt to execute the following with reflection. * * [Code] * IAudioService audioService = IAudioService.Stub.asInterface(b); * audioService.dispatchMediaKeyEvent(keyEvent); */ try { // Get binder from ServiceManager.checkService(String) IBinder iBinder = (IBinder) Class.forName("android.os.ServiceManager") .getDeclaredMethod("checkService",String.class) .invoke(null, Context.AUDIO_SERVICE); // get audioService from IAudioService.Stub.asInterface(IBinder) Object audioService = Class.forName("android.media.IAudioService$Stub") .getDeclaredMethod("asInterface",IBinder.class) .invoke(null,iBinder); // Dispatch keyEvent using IAudioService.dispatchMediaKeyEvent(KeyEvent) Class.forName("android.media.IAudioService") .getDeclaredMethod("dispatchMediaKeyEvent",KeyEvent.class) .invoke(audioService, keyEvent); } catch (Exception e1) { e1.printStackTrace(); } }