/* * Copyright (C) 2013 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.support.v7.media; import android.content.BroadcastReceiver; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.IntentFilter; import android.content.res.Resources; import android.media.AudioManager; import android.os.Build; import android.support.annotation.RequiresApi; import android.support.v7.mediarouter.R; import android.view.Display; import java.util.ArrayList; import java.util.List; import java.util.Locale; /** * Provides routes for built-in system destinations such as the local display * and speaker. On Jellybean and newer platform releases, queries the framework * MediaRouter for framework-provided routes and registers non-framework-provided * routes as user routes. */ abstract class SystemMediaRouteProvider extends MediaRouteProvider { private static final String TAG = "SystemMediaRouteProvider"; public static final String PACKAGE_NAME = "android"; public static final String DEFAULT_ROUTE_ID = "DEFAULT_ROUTE"; protected SystemMediaRouteProvider(Context context) { super(context, new ProviderMetadata(new ComponentName(PACKAGE_NAME, SystemMediaRouteProvider.class.getName()))); } public static SystemMediaRouteProvider obtain(Context context, SyncCallback syncCallback) { if (Build.VERSION.SDK_INT >= 24) { return new Api24Impl(context, syncCallback); } if (Build.VERSION.SDK_INT >= 18) { return new JellybeanMr2Impl(context, syncCallback); } if (Build.VERSION.SDK_INT >= 17) { return new JellybeanMr1Impl(context, syncCallback); } if (Build.VERSION.SDK_INT >= 16) { return new JellybeanImpl(context, syncCallback); } return new LegacyImpl(context); } /** * Called by the media router when a route is added to synchronize state with * the framework media router. */ public void onSyncRouteAdded(MediaRouter.RouteInfo route) { } /** * Called by the media router when a route is removed to synchronize state with * the framework media router. */ public void onSyncRouteRemoved(MediaRouter.RouteInfo route) { } /** * Called by the media router when a route is changed to synchronize state with * the framework media router. */ public void onSyncRouteChanged(MediaRouter.RouteInfo route) { } /** * Called by the media router when a route is selected to synchronize state with * the framework media router. */ public void onSyncRouteSelected(MediaRouter.RouteInfo route) { } /** * Callbacks into the media router to synchronize state with the framework media router. */ public interface SyncCallback { MediaRouter.RouteInfo getSystemRouteByDescriptorId(String id); } protected Object getDefaultRoute() { return null; } protected Object getSystemRoute(MediaRouter.RouteInfo route) { return null; } /** * Legacy implementation for platform versions prior to Jellybean. */ static class LegacyImpl extends SystemMediaRouteProvider { static final int PLAYBACK_STREAM = AudioManager.STREAM_MUSIC; private static final ArrayList CONTROL_FILTERS; static { IntentFilter f = new IntentFilter(); f.addCategory(MediaControlIntent.CATEGORY_LIVE_AUDIO); f.addCategory(MediaControlIntent.CATEGORY_LIVE_VIDEO); CONTROL_FILTERS = new ArrayList(); CONTROL_FILTERS.add(f); } final AudioManager mAudioManager; private final VolumeChangeReceiver mVolumeChangeReceiver; int mLastReportedVolume = -1; public LegacyImpl(Context context) { super(context); mAudioManager = (AudioManager)context.getSystemService(Context.AUDIO_SERVICE); mVolumeChangeReceiver = new VolumeChangeReceiver(); context.registerReceiver(mVolumeChangeReceiver, new IntentFilter(VolumeChangeReceiver.VOLUME_CHANGED_ACTION)); publishRoutes(); } void publishRoutes() { Resources r = getContext().getResources(); int maxVolume = mAudioManager.getStreamMaxVolume(PLAYBACK_STREAM); mLastReportedVolume = mAudioManager.getStreamVolume(PLAYBACK_STREAM); MediaRouteDescriptor defaultRoute = new MediaRouteDescriptor.Builder( DEFAULT_ROUTE_ID, r.getString(R.string.mr_system_route_name)) .addControlFilters(CONTROL_FILTERS) .setPlaybackStream(PLAYBACK_STREAM) .setPlaybackType(MediaRouter.RouteInfo.PLAYBACK_TYPE_LOCAL) .setVolumeHandling(MediaRouter.RouteInfo.PLAYBACK_VOLUME_VARIABLE) .setVolumeMax(maxVolume) .setVolume(mLastReportedVolume) .build(); MediaRouteProviderDescriptor providerDescriptor = new MediaRouteProviderDescriptor.Builder() .addRoute(defaultRoute) .build(); setDescriptor(providerDescriptor); } @Override public RouteController onCreateRouteController(String routeId) { if (routeId.equals(DEFAULT_ROUTE_ID)) { return new DefaultRouteController(); } return null; } final class DefaultRouteController extends RouteController { @Override public void onSetVolume(int volume) { mAudioManager.setStreamVolume(PLAYBACK_STREAM, volume, 0); publishRoutes(); } @Override public void onUpdateVolume(int delta) { int volume = mAudioManager.getStreamVolume(PLAYBACK_STREAM); int maxVolume = mAudioManager.getStreamMaxVolume(PLAYBACK_STREAM); int newVolume = Math.min(maxVolume, Math.max(0, volume + delta)); if (newVolume != volume) { mAudioManager.setStreamVolume(PLAYBACK_STREAM, volume, 0); } publishRoutes(); } } final class VolumeChangeReceiver extends BroadcastReceiver { // These constants come from AudioManager. public static final String VOLUME_CHANGED_ACTION = "android.media.VOLUME_CHANGED_ACTION"; public static final String EXTRA_VOLUME_STREAM_TYPE = "android.media.EXTRA_VOLUME_STREAM_TYPE"; public static final String EXTRA_VOLUME_STREAM_VALUE = "android.media.EXTRA_VOLUME_STREAM_VALUE"; @Override public void onReceive(Context context, Intent intent) { if (intent.getAction().equals(VOLUME_CHANGED_ACTION)) { final int streamType = intent.getIntExtra(EXTRA_VOLUME_STREAM_TYPE, -1); if (streamType == PLAYBACK_STREAM) { final int volume = intent.getIntExtra(EXTRA_VOLUME_STREAM_VALUE, -1); if (volume >= 0 && volume != mLastReportedVolume) { publishRoutes(); } } } } } } /** * Jellybean implementation. */ @RequiresApi(16) static class JellybeanImpl extends SystemMediaRouteProvider implements MediaRouterJellybean.Callback, MediaRouterJellybean.VolumeCallback { private static final ArrayList LIVE_AUDIO_CONTROL_FILTERS; static { IntentFilter f = new IntentFilter(); f.addCategory(MediaControlIntent.CATEGORY_LIVE_AUDIO); LIVE_AUDIO_CONTROL_FILTERS = new ArrayList(); LIVE_AUDIO_CONTROL_FILTERS.add(f); } private static final ArrayList LIVE_VIDEO_CONTROL_FILTERS; static { IntentFilter f = new IntentFilter(); f.addCategory(MediaControlIntent.CATEGORY_LIVE_VIDEO); LIVE_VIDEO_CONTROL_FILTERS = new ArrayList(); LIVE_VIDEO_CONTROL_FILTERS.add(f); } private final SyncCallback mSyncCallback; protected final Object mRouterObj; protected final Object mCallbackObj; protected final Object mVolumeCallbackObj; protected final Object mUserRouteCategoryObj; protected int mRouteTypes; protected boolean mActiveScan; protected boolean mCallbackRegistered; // Maintains an association from framework routes to support library routes. // Note that we cannot use the tag field for this because an application may // have published its own user routes to the framework media router and already // used the tag for its own purposes. protected final ArrayList mSystemRouteRecords = new ArrayList(); // Maintains an association from support library routes to framework routes. protected final ArrayList mUserRouteRecords = new ArrayList(); private MediaRouterJellybean.SelectRouteWorkaround mSelectRouteWorkaround; private MediaRouterJellybean.GetDefaultRouteWorkaround mGetDefaultRouteWorkaround; public JellybeanImpl(Context context, SyncCallback syncCallback) { super(context); mSyncCallback = syncCallback; mRouterObj = MediaRouterJellybean.getMediaRouter(context); mCallbackObj = createCallbackObj(); mVolumeCallbackObj = createVolumeCallbackObj(); Resources r = context.getResources(); mUserRouteCategoryObj = MediaRouterJellybean.createRouteCategory( mRouterObj, r.getString(R.string.mr_user_route_category_name), false); updateSystemRoutes(); } @Override public RouteController onCreateRouteController(String routeId) { int index = findSystemRouteRecordByDescriptorId(routeId); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); return new SystemRouteController(record.mRouteObj); } return null; } @Override public void onDiscoveryRequestChanged(MediaRouteDiscoveryRequest request) { int newRouteTypes = 0; boolean newActiveScan = false; if (request != null) { final MediaRouteSelector selector = request.getSelector(); final List categories = selector.getControlCategories(); final int count = categories.size(); for (int i = 0; i < count; i++) { String category = categories.get(i); if (category.equals(MediaControlIntent.CATEGORY_LIVE_AUDIO)) { newRouteTypes |= MediaRouterJellybean.ROUTE_TYPE_LIVE_AUDIO; } else if (category.equals(MediaControlIntent.CATEGORY_LIVE_VIDEO)) { newRouteTypes |= MediaRouterJellybean.ROUTE_TYPE_LIVE_VIDEO; } else { newRouteTypes |= MediaRouterJellybean.ROUTE_TYPE_USER; } } newActiveScan = request.isActiveScan(); } if (mRouteTypes != newRouteTypes || mActiveScan != newActiveScan) { mRouteTypes = newRouteTypes; mActiveScan = newActiveScan; updateSystemRoutes(); } } @Override public void onRouteAdded(Object routeObj) { if (addSystemRouteNoPublish(routeObj)) { publishRoutes(); } } private void updateSystemRoutes() { updateCallback(); boolean changed = false; for (Object routeObj : MediaRouterJellybean.getRoutes(mRouterObj)) { changed |= addSystemRouteNoPublish(routeObj); } if (changed) { publishRoutes(); } } private boolean addSystemRouteNoPublish(Object routeObj) { if (getUserRouteRecord(routeObj) == null && findSystemRouteRecord(routeObj) < 0) { String id = assignRouteId(routeObj); SystemRouteRecord record = new SystemRouteRecord(routeObj, id); updateSystemRouteDescriptor(record); mSystemRouteRecords.add(record); return true; } return false; } private String assignRouteId(Object routeObj) { // TODO: The framework media router should supply a unique route id that // we can use here. For now we use a hash of the route name and take care // to dedupe it. boolean isDefault = (getDefaultRoute() == routeObj); String id = isDefault ? DEFAULT_ROUTE_ID : String.format(Locale.US, "ROUTE_%08x", getRouteName(routeObj).hashCode()); if (findSystemRouteRecordByDescriptorId(id) < 0) { return id; } for (int i = 2; ; i++) { String newId = String.format(Locale.US, "%s_%d", id, i); if (findSystemRouteRecordByDescriptorId(newId) < 0) { return newId; } } } @Override public void onRouteRemoved(Object routeObj) { if (getUserRouteRecord(routeObj) == null) { int index = findSystemRouteRecord(routeObj); if (index >= 0) { mSystemRouteRecords.remove(index); publishRoutes(); } } } @Override public void onRouteChanged(Object routeObj) { if (getUserRouteRecord(routeObj) == null) { int index = findSystemRouteRecord(routeObj); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); updateSystemRouteDescriptor(record); publishRoutes(); } } } @Override public void onRouteVolumeChanged(Object routeObj) { if (getUserRouteRecord(routeObj) == null) { int index = findSystemRouteRecord(routeObj); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); int newVolume = MediaRouterJellybean.RouteInfo.getVolume(routeObj); if (newVolume != record.mRouteDescriptor.getVolume()) { record.mRouteDescriptor = new MediaRouteDescriptor.Builder(record.mRouteDescriptor) .setVolume(newVolume) .build(); publishRoutes(); } } } } @Override public void onRouteSelected(int type, Object routeObj) { if (routeObj != MediaRouterJellybean.getSelectedRoute(mRouterObj, MediaRouterJellybean.ALL_ROUTE_TYPES)) { // The currently selected route has already changed so this callback // is stale. Drop it to prevent getting into sync loops. return; } UserRouteRecord userRouteRecord = getUserRouteRecord(routeObj); if (userRouteRecord != null) { userRouteRecord.mRoute.select(); } else { // Select the route if it already exists in the compat media router. // If not, we will select it instead when the route is added. int index = findSystemRouteRecord(routeObj); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); MediaRouter.RouteInfo route = mSyncCallback.getSystemRouteByDescriptorId( record.mRouteDescriptorId); if (route != null) { route.select(); } } } } @Override public void onRouteUnselected(int type, Object routeObj) { // Nothing to do when a route is unselected. // We only need to handle when a route is selected. } @Override public void onRouteGrouped(Object routeObj, Object groupObj, int index) { // Route grouping is deprecated and no longer supported. } @Override public void onRouteUngrouped(Object routeObj, Object groupObj) { // Route grouping is deprecated and no longer supported. } @Override public void onVolumeSetRequest(Object routeObj, int volume) { UserRouteRecord record = getUserRouteRecord(routeObj); if (record != null) { record.mRoute.requestSetVolume(volume); } } @Override public void onVolumeUpdateRequest(Object routeObj, int direction) { UserRouteRecord record = getUserRouteRecord(routeObj); if (record != null) { record.mRoute.requestUpdateVolume(direction); } } @Override public void onSyncRouteAdded(MediaRouter.RouteInfo route) { if (route.getProviderInstance() != this) { Object routeObj = MediaRouterJellybean.createUserRoute( mRouterObj, mUserRouteCategoryObj); UserRouteRecord record = new UserRouteRecord(route, routeObj); MediaRouterJellybean.RouteInfo.setTag(routeObj, record); MediaRouterJellybean.UserRouteInfo.setVolumeCallback(routeObj, mVolumeCallbackObj); updateUserRouteProperties(record); mUserRouteRecords.add(record); MediaRouterJellybean.addUserRoute(mRouterObj, routeObj); } else { // If the newly added route is the counterpart of the currently selected // route in the framework media router then ensure it is selected in // the compat media router. Object routeObj = MediaRouterJellybean.getSelectedRoute( mRouterObj, MediaRouterJellybean.ALL_ROUTE_TYPES); int index = findSystemRouteRecord(routeObj); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); if (record.mRouteDescriptorId.equals(route.getDescriptorId())) { route.select(); } } } } @Override public void onSyncRouteRemoved(MediaRouter.RouteInfo route) { if (route.getProviderInstance() != this) { int index = findUserRouteRecord(route); if (index >= 0) { UserRouteRecord record = mUserRouteRecords.remove(index); MediaRouterJellybean.RouteInfo.setTag(record.mRouteObj, null); MediaRouterJellybean.UserRouteInfo.setVolumeCallback(record.mRouteObj, null); MediaRouterJellybean.removeUserRoute(mRouterObj, record.mRouteObj); } } } @Override public void onSyncRouteChanged(MediaRouter.RouteInfo route) { if (route.getProviderInstance() != this) { int index = findUserRouteRecord(route); if (index >= 0) { UserRouteRecord record = mUserRouteRecords.get(index); updateUserRouteProperties(record); } } } @Override public void onSyncRouteSelected(MediaRouter.RouteInfo route) { if (!route.isSelected()) { // The currently selected route has already changed so this callback // is stale. Drop it to prevent getting into sync loops. return; } if (route.getProviderInstance() != this) { int index = findUserRouteRecord(route); if (index >= 0) { UserRouteRecord record = mUserRouteRecords.get(index); selectRoute(record.mRouteObj); } } else { int index = findSystemRouteRecordByDescriptorId(route.getDescriptorId()); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); selectRoute(record.mRouteObj); } } } protected void publishRoutes() { MediaRouteProviderDescriptor.Builder builder = new MediaRouteProviderDescriptor.Builder(); int count = mSystemRouteRecords.size(); for (int i = 0; i < count; i++) { builder.addRoute(mSystemRouteRecords.get(i).mRouteDescriptor); } setDescriptor(builder.build()); } protected int findSystemRouteRecord(Object routeObj) { final int count = mSystemRouteRecords.size(); for (int i = 0; i < count; i++) { if (mSystemRouteRecords.get(i).mRouteObj == routeObj) { return i; } } return -1; } protected int findSystemRouteRecordByDescriptorId(String id) { final int count = mSystemRouteRecords.size(); for (int i = 0; i < count; i++) { if (mSystemRouteRecords.get(i).mRouteDescriptorId.equals(id)) { return i; } } return -1; } protected int findUserRouteRecord(MediaRouter.RouteInfo route) { final int count = mUserRouteRecords.size(); for (int i = 0; i < count; i++) { if (mUserRouteRecords.get(i).mRoute == route) { return i; } } return -1; } protected UserRouteRecord getUserRouteRecord(Object routeObj) { Object tag = MediaRouterJellybean.RouteInfo.getTag(routeObj); return tag instanceof UserRouteRecord ? (UserRouteRecord)tag : null; } protected void updateSystemRouteDescriptor(SystemRouteRecord record) { // We must always recreate the route descriptor when making any changes // because they are intended to be immutable once published. MediaRouteDescriptor.Builder builder = new MediaRouteDescriptor.Builder( record.mRouteDescriptorId, getRouteName(record.mRouteObj)); onBuildSystemRouteDescriptor(record, builder); record.mRouteDescriptor = builder.build(); } protected String getRouteName(Object routeObj) { // Routes should not have null names but it may happen for badly configured // user routes. We tolerate this by using an empty name string here but // such unnamed routes will be discarded by the media router upstream // (with a log message so we can track down the problem). CharSequence name = MediaRouterJellybean.RouteInfo.getName(routeObj, getContext()); return name != null ? name.toString() : ""; } protected void onBuildSystemRouteDescriptor(SystemRouteRecord record, MediaRouteDescriptor.Builder builder) { int supportedTypes = MediaRouterJellybean.RouteInfo.getSupportedTypes( record.mRouteObj); if ((supportedTypes & MediaRouterJellybean.ROUTE_TYPE_LIVE_AUDIO) != 0) { builder.addControlFilters(LIVE_AUDIO_CONTROL_FILTERS); } if ((supportedTypes & MediaRouterJellybean.ROUTE_TYPE_LIVE_VIDEO) != 0) { builder.addControlFilters(LIVE_VIDEO_CONTROL_FILTERS); } builder.setPlaybackType( MediaRouterJellybean.RouteInfo.getPlaybackType(record.mRouteObj)); builder.setPlaybackStream( MediaRouterJellybean.RouteInfo.getPlaybackStream(record.mRouteObj)); builder.setVolume( MediaRouterJellybean.RouteInfo.getVolume(record.mRouteObj)); builder.setVolumeMax( MediaRouterJellybean.RouteInfo.getVolumeMax(record.mRouteObj)); builder.setVolumeHandling( MediaRouterJellybean.RouteInfo.getVolumeHandling(record.mRouteObj)); } protected void updateUserRouteProperties(UserRouteRecord record) { MediaRouterJellybean.UserRouteInfo.setName( record.mRouteObj, record.mRoute.getName()); MediaRouterJellybean.UserRouteInfo.setPlaybackType( record.mRouteObj, record.mRoute.getPlaybackType()); MediaRouterJellybean.UserRouteInfo.setPlaybackStream( record.mRouteObj, record.mRoute.getPlaybackStream()); MediaRouterJellybean.UserRouteInfo.setVolume( record.mRouteObj, record.mRoute.getVolume()); MediaRouterJellybean.UserRouteInfo.setVolumeMax( record.mRouteObj, record.mRoute.getVolumeMax()); MediaRouterJellybean.UserRouteInfo.setVolumeHandling( record.mRouteObj, record.mRoute.getVolumeHandling()); } protected void updateCallback() { if (mCallbackRegistered) { mCallbackRegistered = false; MediaRouterJellybean.removeCallback(mRouterObj, mCallbackObj); } if (mRouteTypes != 0) { mCallbackRegistered = true; MediaRouterJellybean.addCallback(mRouterObj, mRouteTypes, mCallbackObj); } } protected Object createCallbackObj() { return MediaRouterJellybean.createCallback(this); } protected Object createVolumeCallbackObj() { return MediaRouterJellybean.createVolumeCallback(this); } protected void selectRoute(Object routeObj) { if (mSelectRouteWorkaround == null) { mSelectRouteWorkaround = new MediaRouterJellybean.SelectRouteWorkaround(); } mSelectRouteWorkaround.selectRoute(mRouterObj, MediaRouterJellybean.ALL_ROUTE_TYPES, routeObj); } @Override protected Object getDefaultRoute() { if (mGetDefaultRouteWorkaround == null) { mGetDefaultRouteWorkaround = new MediaRouterJellybean.GetDefaultRouteWorkaround(); } return mGetDefaultRouteWorkaround.getDefaultRoute(mRouterObj); } @Override protected Object getSystemRoute(MediaRouter.RouteInfo route) { if (route == null) { return null; } int index = findSystemRouteRecordByDescriptorId(route.getDescriptorId()); if (index >= 0) { return mSystemRouteRecords.get(index).mRouteObj; } return null; } /** * Represents a route that is provided by the framework media router * and published by this route provider to the support library media router. */ protected static final class SystemRouteRecord { public final Object mRouteObj; public final String mRouteDescriptorId; public MediaRouteDescriptor mRouteDescriptor; // assigned immediately after creation public SystemRouteRecord(Object routeObj, String id) { mRouteObj = routeObj; mRouteDescriptorId = id; } } /** * Represents a route that is provided by the support library media router * and published by this route provider to the framework media router. */ protected static final class UserRouteRecord { public final MediaRouter.RouteInfo mRoute; public final Object mRouteObj; public UserRouteRecord(MediaRouter.RouteInfo route, Object routeObj) { mRoute = route; mRouteObj = routeObj; } } protected static final class SystemRouteController extends RouteController { private final Object mRouteObj; public SystemRouteController(Object routeObj) { mRouteObj = routeObj; } @Override public void onSetVolume(int volume) { MediaRouterJellybean.RouteInfo.requestSetVolume(mRouteObj, volume); } @Override public void onUpdateVolume(int delta) { MediaRouterJellybean.RouteInfo.requestUpdateVolume(mRouteObj, delta); } } } /** * Jellybean MR1 implementation. */ @RequiresApi(17) private static class JellybeanMr1Impl extends JellybeanImpl implements MediaRouterJellybeanMr1.Callback { private MediaRouterJellybeanMr1.ActiveScanWorkaround mActiveScanWorkaround; private MediaRouterJellybeanMr1.IsConnectingWorkaround mIsConnectingWorkaround; public JellybeanMr1Impl(Context context, SyncCallback syncCallback) { super(context, syncCallback); } @Override public void onRoutePresentationDisplayChanged(Object routeObj) { int index = findSystemRouteRecord(routeObj); if (index >= 0) { SystemRouteRecord record = mSystemRouteRecords.get(index); Display newPresentationDisplay = MediaRouterJellybeanMr1.RouteInfo.getPresentationDisplay(routeObj); int newPresentationDisplayId = (newPresentationDisplay != null ? newPresentationDisplay.getDisplayId() : -1); if (newPresentationDisplayId != record.mRouteDescriptor.getPresentationDisplayId()) { record.mRouteDescriptor = new MediaRouteDescriptor.Builder(record.mRouteDescriptor) .setPresentationDisplayId(newPresentationDisplayId) .build(); publishRoutes(); } } } @Override protected void onBuildSystemRouteDescriptor(SystemRouteRecord record, MediaRouteDescriptor.Builder builder) { super.onBuildSystemRouteDescriptor(record, builder); if (!MediaRouterJellybeanMr1.RouteInfo.isEnabled(record.mRouteObj)) { builder.setEnabled(false); } if (isConnecting(record)) { builder.setConnecting(true); } Display presentationDisplay = MediaRouterJellybeanMr1.RouteInfo.getPresentationDisplay(record.mRouteObj); if (presentationDisplay != null) { builder.setPresentationDisplayId(presentationDisplay.getDisplayId()); } } @Override protected void updateCallback() { super.updateCallback(); if (mActiveScanWorkaround == null) { mActiveScanWorkaround = new MediaRouterJellybeanMr1.ActiveScanWorkaround( getContext(), getHandler()); } mActiveScanWorkaround.setActiveScanRouteTypes(mActiveScan ? mRouteTypes : 0); } @Override protected Object createCallbackObj() { return MediaRouterJellybeanMr1.createCallback(this); } protected boolean isConnecting(SystemRouteRecord record) { if (mIsConnectingWorkaround == null) { mIsConnectingWorkaround = new MediaRouterJellybeanMr1.IsConnectingWorkaround(); } return mIsConnectingWorkaround.isConnecting(record.mRouteObj); } } /** * Jellybean MR2 implementation. */ @RequiresApi(18) private static class JellybeanMr2Impl extends JellybeanMr1Impl { public JellybeanMr2Impl(Context context, SyncCallback syncCallback) { super(context, syncCallback); } @Override protected void onBuildSystemRouteDescriptor(SystemRouteRecord record, MediaRouteDescriptor.Builder builder) { super.onBuildSystemRouteDescriptor(record, builder); CharSequence description = MediaRouterJellybeanMr2.RouteInfo.getDescription(record.mRouteObj); if (description != null) { builder.setDescription(description.toString()); } } @Override protected void selectRoute(Object routeObj) { MediaRouterJellybean.selectRoute(mRouterObj, MediaRouterJellybean.ALL_ROUTE_TYPES, routeObj); } @Override protected Object getDefaultRoute() { return MediaRouterJellybeanMr2.getDefaultRoute(mRouterObj); } @Override protected void updateUserRouteProperties(UserRouteRecord record) { super.updateUserRouteProperties(record); MediaRouterJellybeanMr2.UserRouteInfo.setDescription( record.mRouteObj, record.mRoute.getDescription()); } @Override protected void updateCallback() { if (mCallbackRegistered) { MediaRouterJellybean.removeCallback(mRouterObj, mCallbackObj); } mCallbackRegistered = true; MediaRouterJellybeanMr2.addCallback(mRouterObj, mRouteTypes, mCallbackObj, MediaRouter.CALLBACK_FLAG_UNFILTERED_EVENTS | (mActiveScan ? MediaRouter.CALLBACK_FLAG_PERFORM_ACTIVE_SCAN : 0)); } @Override protected boolean isConnecting(SystemRouteRecord record) { return MediaRouterJellybeanMr2.RouteInfo.isConnecting(record.mRouteObj); } } /** * Api24 implementation. */ @RequiresApi(24) private static class Api24Impl extends JellybeanMr2Impl { public Api24Impl(Context context, SyncCallback syncCallback) { super(context, syncCallback); } @Override protected void onBuildSystemRouteDescriptor(SystemRouteRecord record, MediaRouteDescriptor.Builder builder) { super.onBuildSystemRouteDescriptor(record, builder); builder.setDeviceType(MediaRouterApi24.RouteInfo.getDeviceType(record.mRouteObj)); } } }