Android 5 中具有焦点的 WebView 控件块媒体按钮事件

WebView control with focus blocks media button events in Android 5

编辑 2016 年 5 月 8 日

我找到了在我的应用程序中接收媒体按钮事件时遇到问题的原因。请参阅下面的答案。我编辑了这个问题的标题,以便更容易找到问题。原标题是"What can possibly block media buttons on Android Lollipop".

原始问题,2015 年 4 月:

挠我的头,盯着所有代码看了 2 天都无济于事...我的 Android 应用程序应该对媒体按钮做出反应(例如,从耳机,用蓝牙耳机测试) , 喜欢 play/pause, 接下来, 倒带。在 KitKat 及以下版本上运行良好。我发誓直到几天前,它甚至还适用于 Lollipop。现在什么也没有,没有任何迹象表明它听到媒体按钮被按下的声音。 任何人都可以快速建议在哪里寻找问题吗? 用两部 Lollipop 手机测试,同样的蓝牙耳机,同样的耳机适用于较低版本的 Android。同样的耳机工作正常,在其他应用程序中听到按下媒体按钮的声音。我可能会破坏什么???

我现在测试了听媒体按钮的新旧方式。在 AndroidManifest.xml:

<receiver android:name=".MediaButtonIntentReceiver" android:enabled="false">
  <intent-filter>
      <action android:name="android.intent.action.MEDIA_BUTTON" />
  </intent-filter>
</receiver>

事实上,它说 enabled="false" 没问题 - 我根据需要启用和禁用接收器,并且 MediaButtonIntentReceiver.java 使 KitKat 上的事件正常,而在 Lollipop 上则更低,完全静音。

我接下来切换到最新的 appcompat (v22.1) 并尝试使用 MediaSessionCompat object 和相关代码如下。这在我编写的一个 activity 小测试应用程序中非常有效 - 收到我的 Logcat 消息确认它听到棒棒糖上按下的媒体键。但是当插入我的应用程序时,在 Lollipop 上再次不起作用。什么鬼???

private MediaSessionCompat _mediaSession;
final String MEDIA_SESSION_TAG = "some_tag";

void initMediaSessions(Context context) {
    // Start a new MediaSession
    if (context == null)
        return;
    Lt.d("initMediaSession()...");
    ComponentName eventReceiver = new ComponentName(context.getPackageName(), MediaButtonIntentReceiver.class.getName());
    PendingIntent buttonReceiverIntent = PendingIntent.getBroadcast(
            context,
            0,
            new Intent(Intent.ACTION_MEDIA_BUTTON),
            PendingIntent.FLAG_UPDATE_CURRENT
    );

    // Parameters for new MediaSessionCompat():
    // context  The context.
    // tag  A short name for debugging purposes.
    // mediaButtonEventReceiver     The component name for your receiver. This must be non-null to support platform
    // versions earlier than LOLLIPOP. May not be null below Lollipop.
    // mbrIntent    The PendingIntent for your receiver component that handles media button events. This is optional
    // and will be used on JELLY_BEAN_MR2 and later instead of the component name.
    _mediaSession = new MediaSessionCompat(context, MEDIA_SESSION_TAG, eventReceiver, buttonReceiverIntent);
    _mediaSession.setCallback(new MediaSessionCallback());
    _mediaSession.setFlags(MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS |
                            MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS);
    _mediaSession.setActive(true);
    PlaybackStateCompat state = new PlaybackStateCompat.Builder()
            .setActions(
                    PlaybackStateCompat.ACTION_PLAY | PlaybackStateCompat.ACTION_PLAY_PAUSE |
                            PlaybackStateCompat.ACTION_PAUSE |
                            PlaybackStateCompat.ACTION_SKIP_TO_NEXT | PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS)
            .setState(PlaybackStateCompat.STATE_STOPPED, 0, 1, SystemClock.elapsedRealtime())
            .build();
    _mediaSession.setPlaybackState(state);
}


final class MediaSessionCallback extends MediaSessionCompat.Callback {
    @Override
    public void onPlay() {
        Lt.d("play");
    }

    @Override
    public void onPause() {
        Lt.d("pause");
    }

    @Override
    public void onStop() {
        Lt.d("stop.");
    }

    @Override
    public void onSkipToNext() {
        Lt.d("skipToNext");
    }

    @Override
    public void onSkipToPrevious() {
        Lt.d("skipToPrevious");
    }

    @Override
    public boolean onMediaButtonEvent(final Intent mediaButtonIntent) {
        Lt.d("GOT MediaButton EVENT");
        KeyEvent keyEvent = (KeyEvent) mediaButtonIntent.getExtras().get(Intent.EXTRA_KEY_EVENT);
        // ...do something with keyEvent, super... does nothing.
        return super.onMediaButtonEvent(mediaButtonIntent);
    }
}

2016 年 5 月 8 日:确定已解决 当主屏幕打开时,我终于找到了媒体按钮无法与我的应用程序一起使用的原因。我后来注意到,如果任何其他应用程序处于活动状态,或者屏幕关闭等,它们就会工作 - 但当我的主 activity 启动时。事实证明,至少在 Android 5 中,我在主屏幕上使用的 WebView 控件会阻止媒体按钮,如果它具有焦点。呼叫:

webView.setFocusable(false);

webView.setFocusableInTouchMode(false);

或者在布局中设置相同的属性,问题就解决了。对于 Android 6,WebView 的可聚焦设置对于接收媒体按钮事件无关紧要。

4 月2015 年 2 月 24 日:解决了吗? - 不太... 从评论到这个 post: Handling media buttons in Android 5.0 Lollipop by @mangini:

To receive media button events while your activity is in foreground, call setMediaController() on your activity with a MediaController instance connected to your active MediaSession. – mangini Jan 23 at 18:35

我一直在测试屏幕打开并且我的 activity 在它上面打开,因为它在旧版本的 Android 下也总是这样工作......只要我按下主页按钮,或关闭屏幕,我的旧 MediaButtonIntentReceiver 开始正常工作。

更新 媒体按钮事件仅在我的锁屏管理器启用时才会出现在我的应用程序中,使用已贬值的 RemoteControlClient,当然我的 activity 不在前台.如果我禁用锁屏管理器(这是我的应用程序中的一个选项)并删除 RemoteControlClient,同样,我无法获得任何媒体事件。

Lollipop 下的这个问题一团糟。或者至少现在我的脑海里存在着关于它的一团糟。希望有人 post 编写一个清晰的代码示例来处理媒体按钮,它可以在 Android 和 Lollipop 的旧版本下工作...

格雷格

请覆盖函数 return false。

@Override
    public boolean dispatchKeyEvent(KeyEvent event) {
        return false;
    }