android媒体协商,Android基于MediaBroswerService的App实现概述

前言

如何实现一个音乐播放App,而后让其能够被第三方的Android app打开,并获取其中的歌单,曲目列表,同时控制其播放呢?现有应用市场上,已经有相应的实现。好比百度CarLife对QQ音乐,喜马拉雅等的调用。android

c05b32b60526356473c966e2baca30ae.png

f05857c22dc7442161a8e090793cb8a1.png

在百度的Carlife App中,咱们能够看到,只要咱们本地的装了QQ音乐App,其就能够唤起,而后获取其中的歌曲数据,而后进行播放,这个是如何实现的呢?bash

需求

能够获取音乐播放器的歌曲列表

能够控制音乐播放器的播放

能够将音乐播放器的状态同步到第三方App

可以和第三方App间进行相互通讯

相似于CarLife 对音乐App的唤起,首先第三方App开启后,便可拉起音乐App,而后获取其中的歌单,打开歌单以后,获取歌单内的歌曲列表,点击进行播放,能够进行播放,暂停,下一首,上一首的控制。session

技术实现

谷歌官方提供了MediaBroswerService,经过其能够帮助咱们实现上述的需求。架构

MediaBroswerService

Android多媒体架构

Android多媒体播放采用client,server架构,一个server能够对应多个client,client在使用的时候须要先链接到server,双方经过设置的一些callback来进行状态的同步。app

ef458b9e4065d56ab0e0837c649627fb.png

使用MediaBrowserService播放ide

842e7eef7b028deeb2b570e6b1142f8a.png

客户端须要建立MediaBrowser,服务端须要实现MediaBrowserService,在创建链接后,两端之间的交互主要经过MediaController和MediaSession。两个类之间经过预先定义的callback进行交互,MediaSession控制着播放器的播放,MediaController来控制着UI的变化。ui

dad0f94a3688d4784506f49e8bf4fa2b.png

Media session

一个session持有了播放器的状态和关于正在播放的一些信息,一个seesion能够接收来自一个或多个媒体播放器的callback。这使得经过其它设备来控制成为可能。this

Media controller

咱们的UI只是和Media controller交互,而不是Player 自己,Media controller会将一些控制信息传递给Media Session,它也会在seesion发生变化的时候,获得来自session的回调,一个media controller一次只能够链接一个session。当使用一个media contoller和Session的时候,咱们能够在运行期部署多个播放器,在其执行的时候根据设备去修改app的外观。spa

使用MediaBrowserService可让Android Wear, Auto很是容易找咱们的App,链接它,浏览它的内容,控制其播放,而彻底不须要接触咱们的UI Activity。code

服务端实现

服务端基础配置

mainfeat 配置

复制代码

MediaPlaybackService的初始化

public class MediaPlaybackService extends MediaBrowserServiceCompat {

@Override

public void onCreate() {

super.onCreate();

// 1. 初始化 MediaSession

mSession = new MediaSessionCompat(this, "MusicService");

// 2. 设置 MedisSessionCallback

mSession.setCallback(mSessionCallback);

// 3. 开启 MediaButton 和 TransportControls 的支持

mSession.setFlags(MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS |

MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS);

// 4. 初始化 PlaybackState

mStateBuilder = new PlaybackStateCompat.Builder()

.setActions(

PlaybackStateCompat.ACTION_PLAY | PlaybackStateCompat.ACTION_PLAY_PAUSE);

mSession.setPlaybackState(mStateBuilder.build());

// 5. 关联 SessionToken

setSessionToken(mSession.getSessionToken());

}

}

复制代码

根据包名作权限判断以后,返回根路径

@Override

public BrowserRoot onGetRoot(String clientPackageName, int clientUid, Bundle rootHints) {

// 根据包名对每一个访问端作一些访问权限判断等,返回为空,链接将会断开

}

复制代码

用来根据mediaID来返回第三放App所须要得到媒体数据

@Override

public void onLoadChildren(final String parentMediaId,

final Result> result) {

// 根据parentMediaId返回播放列表相关信息

}

复制代码

客户端链接

private void initMediaBrowser() {

//1.待链接的服务

ComponentName componentName = new ComponentName("com.example.android.uamp","com.example.android.uamp.MusicService");

//2.建立MediaBrowser

mMediaBrowser = new MediaBrowserCompat(this, componentName, mConnectionCallbacks, null);

//3.创建链接

mMediaBrowser.connect();

}

复制代码

设置相应的callback,链接Callback,数据变化Callback

链接状态同步

数据变化Callback设置

private final MediaBrowserCompat.ConnectionCallback mConnectionCallbacks =

new MediaBrowserCompat.ConnectionCallback() {

@Override

public void onConnected() {

//链接成功回调

}

@Override

public void onConnectionSuspended() {

//链接中断回调

}

@Override

public void onConnectionFailed() {

//链接失败回调

}

};

复制代码

MediaControllerCompat.Callback controllerCallback =

new MediaControllerCompat.Callback() {

public void onSessionDestroyed() {

//Session销毁

}

@Override

public void onRepeatModeChanged(int repeatMode) {

//循环模式发生变化

}

@Override

public void onShuffleModeChanged(int shuffleMode) {

//随机模式发生变化

}

@Override

public void onMetadataChanged(MediaMetadataCompat metadata) {

//数据变化

}

@Override

public void onPlaybackStateChanged(PlaybackStateCompat state) {

//播放状态变化

}

};

复制代码

客户端与服务端数据交互

MediaBrowser经过调用subscribe,会回调到MediaService的onLoadChildren,在这里作一个判断而后构造相应的列表将列表数据返回。返回数据以后。

根据MediaID获取数据

客户端经过调用subscribe方法,传递MediaID,在SubscriptionCallback的方法中进行处理。

mMediaBrowser.subscribe("ID", new MediaBrowserCompat.SubscriptionCallback() {

@Override

public void onChildrenLoaded(@NonNull String parentId, @NonNull List children) {

//children 为来自Service的列表数据

}

});

复制代码

服务端和客户端之间传递的数据为MediaItem列表。MediaItem中具有的字段:MediaId,Title,SubTitle,Description,Icon,IconUri,MediaUri等字段。经过其能够帮助咱们携带一些数据来进行歌曲的展现和播放。

@Override

public void onLoadChildren(@NonNull final String parentMediaId,

@NonNull final Result> result) {

List items = new ArrayList<>();

//根据MediaID作数据填充

switch (parentMediaId) {

case:

default: break;

}

result.sendResult(items);

}

复制代码

发送自定义数据获取内容

客户端经过调用sendCustomAction,根据与服务端的协商,制定相应的action类型,进行数据的传递交互。

mMediaBrowser.sendCustomAction(action, extras, new MediaBrowserCompat.CustomActionCallback() {

@Override

public void onProgressUpdate(String action, Bundle extras, Bundle data) {

super.onProgressUpdate(action, extras, data);

}

@Override

public void onResult(String action, Bundle extras, Bundle resultData) {

super.onResult(action, extras, resultData);

}

@Override

public void onError(String action, Bundle extras, Bundle data) {

super.onError(action, extras, data);

}

});

复制代码

服务端实现onCustomAction,根据action类型返回相应的数据

@Override

public void onCustomAction(@NonNull String action, Bundle extras, @NonNull Result result) {

//分支判断

if (GET_LIST.equals(action)) {

Bundle bundle = new Bundle();

ArrayList list = new ArrayList<>();

//填充数据

bundle.putStringArrayList(LIST_NAMES, list);

result.sendResult(bundle);

}

}

复制代码

播放控制

客户端

客户端经过getMediaController getTransportControls()来进行播放,暂停,上一首,下一首的控制。

//获取播放状态

int pbState = MediaControllerCompat.getMediaController(MainActivity.this).getPlaybackState().getState();

//根据播放状态进行播放控制

if (pbState == PlaybackStateCompat.STATE_PLAYING) {

MediaControllerCompat.getMediaController(MainActivity.this).getTransportControls().pause();

} else {

MediaControllerCompat.getMediaController(MainActivity.this).getTransportControls().play();

}

复制代码

服务端

在服务端为MediaSession设置SessionCallback,来实现相应的播放功能。

mSession.setCallback(mSessionCallback);

复制代码

c961b5747d2c146ffe6a9ca0a7a4ae2b.png

客户端经过MediaController能够进行播放,暂停,根据MediaID播放下一个音乐,音乐播放快进等。全部的操做会回调到服务端的MediaSessionCallback的play,seekTo等方法,须要咱们本身实现,在其中控制播放队列,而后根据列表播放的状况来动态的变动队列。

播放状态同步

对于播放状态的同步,好比当前播放到哪个歌曲,当前是暂停仍是播放中。客户端经过Controller回调就能够获得相应的变化,可是,变化状态,服务端如何发送呢?

setMetadata(android.media.MediaMetadata));

setPlaybackState(android.media.session.PlaybackState));

复制代码

设置当前的歌曲信息,设置当前的播放状态。设置以后,客户端将会获得更新。

获取手机内的媒体服务

private void discoverBrowseableMediaApps(Context context) {

PackageManager packageManager = context.getPackageManager();

Intent intent = new Intent(MediaBrowserService.SERVICE_INTERFACE);

List services = packageManager.queryIntentServices(intent, 0);

for (ResolveInfo resolveInfo : services) {

if (resolveInfo.serviceInfo != null && resolveInfo.serviceInfo.applicationInfo != null) {

ApplicationInfo applicationInfo = resolveInfo.serviceInfo.applicationInfo;

String label = (String) packageManager.getApplicationLabel(applicationInfo);

Drawable icon = packageManager.getApplicationIcon(applicationInfo);

String className = resolveInfo.serviceInfo.name;

String packageName = resolveInfo.serviceInfo.packageName;

MusicService service = new MusicService();

service.icon = icon;

service.lable = label;

service.className = className;

service.packageName = packageName;

musicServiceList.add(service);

}

}

}

复制代码

部分机型出现获取不到,须要App开启,才能够获取的到。

结语

经过本篇文章,对MediaBroswerService作了一个简单的介绍,但对于播放器的具体实现,特别是在服务端仍是比较复杂的,须要维护歌曲队列,进行播放,同时负责状态的更新。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值