本文为 android-UniversalMusicPlayer源码解析 分析的版本:39fa286,Demo 地址:android-UniversalMusicPlayer
分析者:kevinshine

1. 功能介绍

1.1 android-UniversalMusicPlayer

该项目为googlesamples中提供的一个完整音频播放APP示例。其中不但介绍了标准的播放器前后台实现,还包括了如何横跨Android Auto, Android Wear, Android TV等平台的提供一致性用户体验。以下简称uamp。

1.2 MediaSession架构

Android5.0 提出了全新的MediaSession概念用于播放器与控制器之间进行交互,它的出现为多控制端单服务端的结构提供了有力的支持(一套方案支持Auto,Wear)。uamp底层播放架构都是基于

v4包中的MediaSessionCompat,所以uamp同时也是非常好的MediaSession框架demo。

参考网上的一张结构图可以更好的理解:

opus android_opus android

2. 详细设计

2.1 软件架构介绍

uamp的代码按照标准的MVC模式划分目录,model,ui,playback分别代表了各个层。

opus android_app_02

2.2 类关系图

opus android_ide_03

3. 流程图

简单的播放流程如下:

opus android_app_04

4. 总体设计

参考uamp的分层罗辑,从MVC各个层的主要类开始分析

4.1 MusicPlayerActivity

整个应用的入口,在其onCreate中处理Intent并根据需要展示目录或进入播放界面。
其父类BaseActivity的onCreate中,初始化了MediaBrowser

public void onCreate(Bundle savedInstanceState) {
        ........
        // Connect a media browser just to get the media session token. There are other ways
        // this can be done, for example by sharing the session token directly.
        mMediaBrowser = new MediaBrowserCompat(this,
            new ComponentName(this, MusicService.class), mConnectionCallback, null);
    }

并在onStart和onStop中维护生命周期

protected void onStart() {
        .......
        mMediaBrowser.connect();
    }

    protected void onStop() {
        ......

        mMediaBrowser.disconnect();
    }

MediaBrowser用来访问指定的MediaBrowserService中的内容,即mMediaBrowser用来访问MusicService。
在MediaBrowser的connect方法中会bind到MusicService并将Binder进行包装,封装交互参数。
当建立连接成功后会回调到mConnectionCallback的onConnected()方法,通过connectToSession将获取到的MediaBrowser中的token作为参数创建MediaControllerCompat,再通过setSupportMediaController(mediaController)方法赋值到MusicPlayerActivity,则MusicPlayerActivity与MusicService建立起了连接。

4.2 FullScreenPlayerActivity

顾名思义这个界面是个全屏的播放详情及控制界面,里面包含了全部的播放控制按钮(play,pause,seekbar等),以及专辑、曲目信息显示。与BaseActivity相同,通过MediaBrowser的建立与MusicService的联系。在播放控制View的OnClickListener实现中,获取当前MediaController的TransportControls,并发送对应指令。

MediaControllerCompat.TransportControls controls =
                            getSupportMediaController().getTransportControls();
                    switch (state.getState()) {
                        case PlaybackStateCompat.STATE_PLAYING: // fall through
                        case PlaybackStateCompat.STATE_BUFFERING:
                            controls.pause();
                            stopSeekbarUpdate();
                            break;
                        case PlaybackStateCompat.STATE_PAUSED:
                        case PlaybackStateCompat.STATE_STOPPED:
                            controls.play();
                            scheduleSeekbarUpdate();
                            break;
                        default:
                            LogHelper.d(TAG, "onClick with state ", state.getState());
                    }

Seekbar的更新使用了ScheduledExecutorService

private final ScheduledExecutorService mExecutorService =
        Executors.newSingleThreadScheduledExecutor();


    private void scheduleSeekbarUpdate() {
        .....
            mScheduleFuture = mExecutorService.scheduleAtFixedRate(
                    new Runnable() {
                        @Override
                        public void run() {
                            mHandler.post(mUpdateProgressTask);
                        }
                    }, PROGRESS_UPDATE_INITIAL_INTERVAL,
                    PROGRESS_UPDATE_INTERNAL, TimeUnit.MILLISECONDS);
        ....
    }

    //更新seekbar
    private void updateProgress() {
        ........
        long currentPosition = mLastPlaybackState.getPosition();
        if (mLastPlaybackState.getState() != PlaybackStateCompat.STATE_PAUSED) {
            long timeDelta = SystemClock.elapsedRealtime() -
                    mLastPlaybackState.getLastPositionUpdateTime();
            currentPosition += (int) timeDelta * mLastPlaybackState.getPlaybackSpeed();
        }
        mSeekbar.setProgress((int) currentPosition);
    }
4.3 MusicService

MusicService的继承自MediaBrowserServiceCompat,所以首先它是一个标准的Service。作为后台播放的核心类,其中包含对于MediaNotificationManager、PlaybackManager,MusicProvider等主要功能模块的使用。

在MusicService的onCreate中,对各个类进行初始化。从命名上也可大致了解各个类的作用。
* MediaNotificationManager:封装了对于NotificationManagerCompat的操作,包括点击事件的响应,显示状态同步,Service前后台状态切换
* PlaybackManager:播放状态控制
* QueueManager:管理播放队列,对其操作都定义在PlaybackManager中
* MusicProvider:音乐资源的Provider,定义了所有的资源操作接口

初始化

public void onCreate() {
        ......

        //初始化MusicProvider,并调用retrieveMedia()异步加载数据
        mMusicProvider = new MusicProvider();
        mMusicProvider.retrieveMediaAsync(null /* Callback */);
        ......
        //初始化QueueManager,后续当调用QueueManager中的方法如setCurrentQueue()时,
        //会回调MetadataUpdateListener中的方法向外通知
        QueueManager queueManager = new QueueManager(mMusicProvider, getResources(),
                new QueueManager.MetadataUpdateListener() {
                    @Override
                    public void onMetadataChanged(MediaMetadataCompat metadata) {
                        mSession.setMetadata(metadata);
                    }

                    @Override
                    public void onMetadataRetrieveError() {
                        mPlaybackManager.updatePlaybackState(
                                getString(R.string.error_no_metadata));
                    }

                    @Override
                    public void onCurrentQueueIndexUpdated(int queueIndex) {
                        mPlaybackManager.handlePlayRequest();
                    }

                    @Override
                    public void onQueueUpdated(String title,
                                               List<MediaSessionCompat.QueueItem> newQueue) {
                        mSession.setQueue(newQueue);
                        mSession.setQueueTitle(title);
                    }
                });

        //初始化PlaybackManager
        LocalPlayback playback = new LocalPlayback(this, mMusicProvider);
        mPlaybackManager = new PlaybackManager(this, getResources(), mMusicProvider, queueManager,
                playback);

        //最核心的对象,定义了该服务的Session,并通过setSessionToken赋值,client端通过此Token与Service关联
        mSession = new MediaSessionCompat(this, "MusicService");
        setSessionToken(mSession.getSessionToken());
        mSession.setCallback(mPlaybackManager.getMediaSessionCallback());
        mSession.setFlags(MediaSessionCompat.FLAG_HANDLES_MEDIA_BUTTONS |
                MediaSessionCompat.FLAG_HANDLES_TRANSPORT_CONTROLS);

        Context context = getApplicationContext();
        Intent intent = new Intent(context, NowPlayingActivity.class);
        PendingIntent pi = PendingIntent.getActivity(context, 99 /*request code*/,
                intent, PendingIntent.FLAG_UPDATE_CURRENT);
        mSession.setSessionActivity(pi);

        mSessionExtras = new Bundle();
        ......

        //初始化MediaNotificationManager
        try {
            mMediaNotificationManager = new MediaNotificationManager(this);
        } catch (RemoteException e) {
            throw new IllegalStateException("Could not create a MediaNotificationManager", e);
        }
        .......
    }

由于MusicService本质上就是一个Service,传统的通过startService传递Intent并在onStartCommand中接收的方式也是可行的,但鉴于MediaSession Framework的存在,除特殊需求(如BroadcastReceiver中)不建议此方式。

内容浏览
onGetRoot,onLoadChildren两个方法分别用来获取BrowserRoot和某节点的子节点。
其中onGetRoot方法中还负责验证包名,若包名不在白名单中则返回null,拒绝访问。

DelayedStopHandler
当播放器进入Pause状态后,会通过DelayedStopHandler延迟关闭Service,防止频繁切换状态时Service频繁创建

4.4 MediaNotificationManager

MediaNotificationManager继承自BroadcastReceiver通过接收ACTION_PAUSE,ACTION_PLAY等广播调用相应的处理方法。
其中保存了MusicService的引用,并通过SessionTonken初始化MediaControllerCompat控制MusicService的播放行为。

public MediaNotificationManager(MusicService service) throws RemoteException {
        mService = service;
        updateSessionToken();

        mNotificationColor = ResourceHelper.getThemeColor(mService, R.attr.colorPrimary,
                Color.DKGRAY);

        mNotificationManager = NotificationManagerCompat.from(service);

        ........

        mNotificationManager.cancelAll();
    }

当播放器进入play/pause状态时,会回调相应处理方法,并通过startNotification显示Notification,
并将MusicService切换为前台Service。
当MusicService被销毁或进入Stop状态时也会触发stopNotification(),cancel当前的Notification并切换Service状态。

public void startNotification() {
        if (!mStarted) {
            mMetadata = mController.getMetadata();
            mPlaybackState = mController.getPlaybackState();

            // The notification must be updated after setting started to true
            Notification notification = createNotification();
            if (notification != null) {
                mController.registerCallback(mCb);
                IntentFilter filter = new IntentFilter();
                filter.addAction(ACTION_NEXT);
                filter.addAction(ACTION_PAUSE);
                filter.addAction(ACTION_PLAY);
                filter.addAction(ACTION_PREV);
                filter.addAction(ACTION_STOP_CASTING);
                mService.registerReceiver(this, filter);

                mService.startForeground(NOTIFICATION_ID, notification);
                mStarted = true;
            }
        }
    }

public void stopNotification() {
        if (mStarted) {
            mStarted = false;
            mController.unregisterCallback(mCb);
            try {
                mNotificationManager.cancel(NOTIFICATION_ID);
                mService.unregisterReceiver(this);
            } catch (IllegalArgumentException ex) {
                // ignore if the receiver is not registered.
            }
            mService.stopForeground(true);
        }
    }
4.5 PlaybackManager

PlaybackManager为播放服务中的核心控制器,串联起MusicService、QueueManager、Playback等。其主要功能有
1.处理交互过程中产生的请求(play、pause…)

public void handlePlayRequest() {
        LogHelper.d(TAG, "handlePlayRequest: mState=" + mPlayback.getState());
        MediaSessionCompat.QueueItem currentMusic = mQueueManager.getCurrentMusic();
        if (currentMusic != null) {
            mServiceCallback.onPlaybackStart();
            mPlayback.play(currentMusic);
        }
    }

    public void handlePauseRequest() {
        LogHelper.d(TAG, "handlePauseRequest: mState=" + mPlayback.getState());
        if (mPlayback.isPlaying()) {
            mPlayback.pause();
            mServiceCallback.onPlaybackStop();
        }
    }

    public void handleStopRequest(String withError) {
        LogHelper.d(TAG, "handleStopRequest: mState=" + mPlayback.getState() + " error=", withError);
        mPlayback.stop(true);
        mServiceCallback.onPlaybackStop();
        updatePlaybackState(withError);
    }

2.定义并创建MediaSessionCallback,在MusicService初始化过程中会将其赋值给MediaSessionCompat的Callback,当Session中的状态发生变化触发相应事件时,会回调到对应的方法中,并更新PlaybackManager中属性的状态。其中onCustomAction用来扩展自定义的需求,action定义请求类型,bundle用来保存参数。

private class MediaSessionCallback extends MediaSessionCompat.Callback {
        @Override
        public void onPlay() {
            LogHelper.d(TAG, "play");
            if (mQueueManager.getCurrentMusic() == null) {
                mQueueManager.setRandomQueue();
            }
            handlePlayRequest();
        }

        @Override
        public void onSkipToQueueItem(long queueId) {
            LogHelper.d(TAG, "OnSkipToQueueItem:" + queueId);
            mQueueManager.setCurrentQueueItem(queueId);
            mQueueManager.updateMetadata();
        }

        @Override
        public void onSeekTo(long position) {
            LogHelper.d(TAG, "onSeekTo:", position);
            mPlayback.seekTo((int) position);
        }

        .......


        @Override
        public void onCustomAction(@NonNull String action, Bundle extras) {
            if (CUSTOM_ACTION_THUMBS_UP.equals(action)) {
                LogHelper.i(TAG, "onCustomAction: favorite for current track");
                MediaSessionCompat.QueueItem currentMusic = mQueueManager.getCurrentMusic();
                if (currentMusic != null) {
                    String mediaId = currentMusic.getDescription().getMediaId();
                    if (mediaId != null) {
                        String musicId = MediaIDHelper.extractMusicIDFromMediaID(mediaId);
                        mMusicProvider.setFavorite(musicId, !mMusicProvider.isFavorite(musicId));
                    }
                }
                // playback state needs to be updated because the "Favorite" icon on the
                // custom action will change to reflect the new favorite state.
                updatePlaybackState(null);
            } else {
                LogHelper.e(TAG, "Unsupported action: ", action);
            }
        }
}
4.6 QueueManager

顾名思义在QueueManager中包含了对Queue的各项操作

//播放队列
    private List<MediaSessionCompat.QueueItem> mPlayingQueue;
    //当前播放的元素下标
    private int mCurrentIndex;

    public QueueManager(@NonNull MusicProvider musicProvider,
                        @NonNull Resources resources,
                        @NonNull MetadataUpdateListener listener) {
        this.mMusicProvider = musicProvider;
        this.mListener = listener;
        this.mResources = resources;

        //使用Collections.synchronizedList创建线程同步的容器
        mPlayingQueue = Collections.synchronizedList(new ArrayList<MediaSessionCompat.QueueItem>());
        mCurrentIndex = 0;
    }

QueueManager的初始化是在MusicService中完成的,QueueManager.MetadataUpdateListener会在Queue状态发生变化时通知session或调用对应的PlaybackManager中的方法。

QueueManager queueManager = new QueueManager(mMusicProvider, getResources(),
        new QueueManager.MetadataUpdateListener() {
            @Override
            public void onMetadataChanged(MediaMetadataCompat metadata) {
                mSession.setMetadata(metadata);
            }

            @Override
            public void onMetadataRetrieveError() {
                mPlaybackManager.updatePlaybackState(
                        getString(R.string.error_no_metadata));
            }

            @Override
            public void onCurrentQueueIndexUpdated(int queueIndex) {
                mPlaybackManager.handlePlayRequest();
            }

            @Override
            public void onQueueUpdated(String title,
                                       List<MediaSessionCompat.QueueItem> newQueue) {
                mSession.setQueue(newQueue);
                mSession.setQueueTitle(title);
            }
        });

5.各种Callback和Listener看着确实有些晕,整理一下

用户点击播放控制按钮 ->
调用MediaController的对应方法 ->
通知MediaSession ->
回调MediaSessionCallback(PlaybackManager中定义并创建,在MusicService创建时赋值给mSession)->
调用PlaybackManager中对应的方法处理,其中分两种情况,第一种:类似handlePlayRequest的方法,会回调到PlaybackServiceCallback中的方法,其实就是回调到了PlaybackServiceCallback的实现类MusicService中的对应方法,再调用Playback向播放器发送指令。第二种:类似onSkipToQueueItem的方法,会调用QueueManager中的相关方法,再通过QueueManager中定义的MetadataUpdateListener调用MusicService中的方法,如mSession.setMetadata(metadata);当Session中的状态或属性变化时,注册在其上的Callback的对应方法会触发,其中定义的更新界面等操作得到执行 -> 流程结束。