7

当发送带有 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); 

}
4

2 回答 2

6

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

   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();
    }
}
于 2013-01-26T23:18:48.200 回答
2

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

于 2012-10-05T19:18:04.637 回答