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

时间:2012-09-24 21:51:46

标签: android android-intent

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

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

这仅发生在Android 4.0及更高版本中。 Play Music是否会占用这个意图(一个错误)?您是否怀疑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); 

}

2 个答案:

答案 0 :(得分:5)

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

   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();
    }
}

答案 1 :(得分:2)

您必须使用一个API作为这些意图的首选接收者,但这当然可以由系统密钥的发送者处理。查看this postthe docs。但在这种情况下,这取决于潘多拉和谷歌音乐,所以这可能不适合你。您当然也可以将您的广播发送到特定的包(通过在意图中指定组件名称),然后您决定哪个应用程序获取它。我在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是隐藏的,因此您必须解决这个问题。这可能是我能帮到你的最好的。另一种方法是注入事件。一种方法是成为一种输入法,但这不太可能是前进的方法。有几种方法可以将事件注入到您自己的活动中,但我不知道是否有任何注入系统的方法。也许你可以看看仪器测试是如何做到的。