admin管理员组

文章数量:1646328

最近在做一个需求,要在桌面去控制qq音乐的播放(即,可以播放暂停,上一曲,下一曲显示歌名和歌手名)。接下来一一说下完成这个需求的心路历程。

  • 1.在网上查腾讯有没有针对qq音乐提供SDK接口,结果发现,没有,如果需要控制,得和qq音乐合作,实现qplay 协议。

  • 2.一位前辈和我说,让我试试Android 的MediaSession 是否可行,我也试了下,还是行不通。一般的话只要音乐播放器实现了谷歌提供的MediaSession 框架,然后就可以被第三方应用控制。但是人家qq音乐没实现这个。哈哈哈

  • 3.这下没辙了,另外网上看到了一篇文章,这里也分享下,这里说的就是百度的一个app可以控制很多的第三方音乐播发器,但是这些第三方app基本都是车载的。例如控制qq音乐,需要先在qq音乐里面打开车载功能。这一看,这就是百度和腾讯有合作的莫。具体可以参考这篇文章 我通过了这篇文章中的方法获取我手机中所有实现MediaSession的应用,最后获取到了今日头条,英语流利说等,没有QQ音乐。

  • 4.这下没办法了。然后插上耳机,放了一首音乐冷静冷静。突然,发现耳机线上的两个按钮,咦,这不就能控制qq音乐吗,于是就尝试了下在我的app里面点击模拟发MediaButton 按键,果然,可以正常的控制播放暂停上一首,下一首qq音乐,当然此时系统中必须只有一个音乐播放器。这里就先这样。关于如何模拟发送MediaButton 按键,可以看我这篇文章android 中用代码模拟发送按键

    一般MediaButton 有如下几个按键:而且一般的音乐播放器里面都会实现对MediaButton的接收,具体原理是接收广播,有兴趣的可以自行研究。

      KeyEvent.KEYCODE_MEDIA_NEXT
      KeyEvent.KEYCODE_MEDIA_PREVIOUS
      KeyEvent.KEYCODE_MEDIA_PLAY_PAUSE
      KeyEvent.KEYCODE_MEDIA_PAUSE
      KeyEvent.KEYCODE_MEDIA_PLAY
    

    可以控制了,但是音乐名和歌手名还拿不到。不过可以从音乐的通知中获取音乐名和歌手名,一般主流的音乐播放器,播放音乐时,都会发一个通知,即Notification,如下图,这个通知中就带有音乐的播放信息。音乐我这边有系统的源码,故想要在系统中去截取notification中的音乐的信息。但是后来发现,完全不用。谷歌提供了一个notificationListener,来专门监听通知。下面来说下notification 的具体用法

notificationListener 的具体用法

1. 创建一个服务并且继承 NotificationListenerService(这个核心的代码我下面再具体列出来)
2.在AndroidManifest中添加服务
            <service
            android:name=".service.NotificationListener"
            android:label="@string/app_name"
            android:permission="android.permission.BIND_NOTIFICATION_LISTENER_SERVICE">
            <intent-filter>
                <action android:name="android.service.notification.NotificationListenerService" />
            </intent-filter>
        </service>

3.在初次使用时需要申请通知使用的权限

if (!isNotificationServiceEnabled()) { 初次使用时 判断是否获取了通知使用权
            enableNotificationListenerAlertDialog = buildNotificationServiceAlertDialog();
            enableNotificationListenerAlertDialog.show();
        }
 private boolean isNotificationServiceEnabled() {
        String pkgName = mActivityView.getContext().getPackageName();
        final String flat = Settings.Secure.getString(mActivityView.getContext().getContentResolver(),
                ENABLED_NOTIFICATION_LISTENERS);
        if (!TextUtils.isEmpty(flat)) {
            final String[] names = flat.split(":");
            for (int i = 0; i < names.length; i++) {
                final ComponentName cn = ComponentName.unflattenFromString(names[i]);
                if (cn != null) {
                    if (TextUtils.equals(pkgName, cn.getPackageName())) {
                        return true;
                    }
                }
            }
        }
        return false;
    }
    
    private AlertDialog buildNotificationServiceAlertDialog() {
        AlertDialog.Builder alertDialogBuilder = new AlertDialog.Builder(mActivityView.getContext());
        alertDialogBuilder.setTitle(R.string.notification_listener_service);
        alertDialogBuilder.setMessage(R.string.notification_listener_service_explanation);
        alertDialogBuilder.setPositiveButton(R.string.yes,
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int id) {
                        mActivityView.getContext().startActivity(new Intent(ACTION_NOTIFICATION_LISTENER_SETTINGS));  // 如果没有获取,则跳转到setting 去获取
                    }
                });
        alertDialogBuilder.setNegativeButton(R.string.no,
                new DialogInterface.OnClickListener() {
                    public void onClick(DialogInterface dialog, int id) {
                        // If you choose to not enable the notification listener
                        // the app. will not work as expected
                    }
                });
        return (alertDialogBuilder.create());
    }
    

申请完权限,接下来就可以来监听接收通知了,

public class MyNotificationListenerService extends NotificationListenerService {

@Override
public void onListenerConnected() {
    //当连接成功时调用,一般在开启监听后会回调一次该方法
}

@Override
public void onNotificationPosted(StatusBarNotification sbn) {
      //当收到一条消息时回调,sbn里面带有这条消息的具体信息
}

@Override
public void onNotificationRemoved(StatusBarNotification sbn) {
     //当移除一条消息的时候回调,sbn是被移除的消息
}

所以一般就是在 onNotificationPosted 方法里面去监听拿到消息的具体信息,如下,一般有如下信息。但是我基本都打印了返回值为String 类型的,但是基本都是空。最后我发现,qq音乐发送上面图中的通知,其实是发了一个自定义的RemoteViews(关于RemoteViews 这里不再多说)

Bundle extras = sbn.getNotification().extras;
String title = extras.getString(Notification.EXTRA_TITLE);
String content = extras.getString(Notification.EXTRA_TEXT);
String packageName = sbn.getPackageName();

所以,我通过 sbn.getNotification().bigContentView 拿到了上图中的view,我有尝试通过windowManager 去将此view 显示出来,发现是可以正常显示的。那么接下来的问题就是解析这个view了。

一般的话android中去分析一个view的结构,可以使用AndroidStudio 中的 Layout Inspector工具。然后通过此工具我发现我需要的歌手名在此view的第二个子view 中的第一个和第二个子view。于是:

if (sbn.getNotification().bigContentView != null) {
     ViewGroup view = (ViewGroup) sbn.getNotification().bigContentView.apply(this, null);
     TextView txMusciName = (TextView) ((ViewGroup) view.getChildAt(1)).getChildAt(0);
     TextView txMusicSinger = (TextView) ((ViewGroup) view.getChildAt(1)).getChildAt(1);
     String  qqMusicName = (String) txMusciName.getText();
     String  qqMuiscSinger = (String) txMusicSinger.getText();
     
    // 此处因为这个服务比较特殊,拿到数据之后想要更新UI,则可以通过发广播的方式。
          Intent intent = new Intent(ControlPresent.MUSIC_ACTION_CHANGE_BROADCAST);
                intent.putExtra("qqMusicName", qqMusicName);
                intent.putExtra("qqMuiscSinger", qqMuiscSinger);
                sendBroadcast(intent);

以上,则可以正常的控制qq音乐了。

本文标签: 第三方音乐播放何去音乐android