当发送带有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);
}
答案 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 post和the 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是隐藏的,因此您必须解决这个问题。这可能是我能帮到你的最好的。另一种方法是注入事件。一种方法是成为一种输入法,但这不太可能是前进的方法。有几种方法可以将事件注入到您自己的活动中,但我不知道是否有任何注入系统的方法。也许你可以看看仪器测试是如何做到的。