Android基于MediaBroswerService的App实现概述

158 阅读3分钟
  • Media session

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

  • Media controller

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

使用MediaBrowserService可以让Android Wear, Auto非常容易找我们的App,连接它,浏览它的内容,控制其播放,而完全不需要接触我们的UI Activity。

服务端实现

  • 服务端基础配置

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<List> 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<MediaBrowserCompat.MediaItem> children) { //children 为来自Service的列表数据 } });

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

@Override public void onLoadChildren(@NonNull final String parentMediaId, @NonNull final Result<List> 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(); }

  • 服务端

总结

最后对于程序员来说,要学习的知识内容、技术有太多太多,要想不被环境淘汰就只有不断提升自己,从来都是我们去适应环境,而不是环境来适应我们!

这里附上上述的技术体系图相关的几十套腾讯、头条、阿里、美团等公司2021年的面试题,把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分。

相信它会给大家带来很多收获:

上述【高清技术脑图】以及【配套的架构技术PDF】可以关注我免费获取

Android学习PDF+架构视频+面试文档+源码笔记

当程序员容易,当一个优秀的程序员是需要不断学习的,从初级程序员到高级程序员,从初级架构师到资深架构师,或者走向管理,从技术经理到技术总监,每个阶段都需要掌握不同的能力。早早确定自己的职业方向,才能在工作和能力提升中甩开同龄人。