يمكنك العثور على عناصر التحكّم في الوسائط في Android بالقرب من "الإعدادات السريعة". يتم ترتيب الجلسات من تطبيقات متعدّدة في لوحة عرض دوّارة يمكن التمرير فيها سريعًا. تعرض لوحة العرض الدوّارة الجلسات بالترتيب التالي:
- أحداث البث المباشر التي يتم تشغيلها على الهاتف
- عمليات البث عن بُعد، مثل تلك التي يتم رصدها على الأجهزة الخارجية أو جلسات البث
- الجلسات السابقة التي يمكن استئنافها، بترتيب آخر مرة تم تشغيلها فيها
بدءًا من Android 13 (المستوى 33 لواجهة برمجة التطبيقات)، لضمان تمكّن المستخدمين من الوصول إلى مجموعة متنوعة
من عناصر التحكّم في الوسائط للتطبيقات التي تشغّل الوسائط، يتم اشتقاق أزرار الإجراءات في عناصر التحكّم في الوسائط
من الحالة Player
.
بهذه الطريقة، يمكنك تقديم مجموعة متسقة من عناصر التحكّم في الوسائط وتجربة استخدام محسّنة لعناصر التحكّم في الوسائط على جميع الأجهزة.
يعرض الشكل 1 مثالاً على كيفية ظهور ذلك على هاتف وجهاز لوحي، على التوالي.
يعرض النظام ما يصل إلى خمسة أزرار إجراءات استنادًا إلى حالة Player
كما هو описан في الجدول التالي. في الوضع المكثّف، لا يتم عرض سوى أول ثلاث خانات
لإجراءات. يتوافق ذلك مع طريقة عرض عناصر التحكّم في الوسائط في منصّات Android الأخرى، مثل Auto وAssistant وWear OS.
الشريحة | المعايير | الإجراء |
---|---|---|
1 |
playWhenReady
غير صحيح أو حالة التشغيل
الحالية هي STATE_ENDED .
|
تشغيل |
playWhenReady صحيح وحالة التشغيل الحالية هي STATE_BUFFERING .
|
مؤشر سريان العمل | |
playWhenReady صحيح وحالة التشغيل الحالية هي STATE_READY . |
إيقاف مؤقت | |
2 | يتوفّر طلب تشغيل COMMAND_SEEK_TO_PREVIOUS أو COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM . |
الصفحة السابقة |
لا يتوفّر طلب اللاعب COMMAND_SEEK_TO_PREVIOUS أو COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM ، ويتوفّر طلب مخصّص من التنسيق المخصّص الذي لم يتم وضعه بعد لملء الخانة. |
قرض مخصص | |
تتضمّن معلومات الجلسة الإضافية قيمة منطقية true للمفتاح EXTRAS_KEY_SLOT_RESERVATION_SEEK_TO_PREV . |
فارغ | |
3 | يتوفّر طلب تشغيل COMMAND_SEEK_TO_NEXT أو COMMAND_SEEK_TO_NEXT_MEDIA_ITEM . |
التالي |
لا يتوفّر طلب اللاعب COMMAND_SEEK_TO_NEXT أو COMMAND_SEEK_TO_NEXT_MEDIA_ITEM ، ويتوفّر طلب مخصّص من التنسيق المخصّص الذي لم يتم وضعه بعد لملء الخانة. |
قرض مخصص | |
تتضمّن معلومات الجلسة الإضافية قيمة منطقية true للمفتاح EXTRAS_KEY_SLOT_RESERVATION_SEEK_TO_NEXT . |
فارغ | |
4 | يتوفّر أمر مخصّص من التنسيق المخصّص لم يتم وضعه بعد لملء المساحة. | قرض مخصص |
5 | يتوفّر أمر مخصّص من التنسيق المخصّص لم يتم وضعه بعد لملء المساحة. | قرض مخصص |
يتم وضع الأوامر المخصّصة بالترتيب الذي تمت إضافتها به إلى التنسيق المخصّص.
تخصيص أزرار الأوامر
لتخصيص عناصر التحكّم في الوسائط في النظام باستخدام Jetpack Media3،
يمكنك ضبط التنسيق المخصّص للجلسة والأوامر المتاحة لعناصر التحكّم
وفقًا لذلك، عند تنفيذ MediaSessionService
:
في
onCreate()
، أنشئMediaSession
وحدِّد التنسيق المخصّص لأزرار الأوامر.في
MediaSession.Callback.onConnect()
، امنح الإذن لأجهزة التحكّم من خلال تحديد الأوامر المتاحة لها، بما في ذلك الأوامر المخصّصة، فيConnectionResult
.في
MediaSession.Callback.onCustomCommand()
، يجب الردّ على الأمر المخصّص الذي يختاره المستخدم.
Kotlin
class PlaybackService : MediaSessionService() { private val customCommandFavorites = SessionCommand(ACTION_FAVORITES, Bundle.EMPTY) private var mediaSession: MediaSession? = null override fun onCreate() { super.onCreate() val favoriteButton = CommandButton.Builder() .setDisplayName("Save to favorites") .setIconResId(R.drawable.favorite_icon) .setSessionCommand(customCommandFavorites) .build() val player = ExoPlayer.Builder(this).build() // Build the session with a custom layout. mediaSession = MediaSession.Builder(this, player) .setCallback(MyCallback()) .setCustomLayout(ImmutableList.of(favoriteButton)) .build() } private inner class MyCallback : MediaSession.Callback { override fun onConnect( session: MediaSession, controller: MediaSession.ControllerInfo ): ConnectionResult { // Set available player and session commands. return AcceptedResultBuilder(session) .setAvailablePlayerCommands( ConnectionResult.DEFAULT_PLAYER_COMMANDS.buildUpon() .remove(COMMAND_SEEK_TO_NEXT) .remove(COMMAND_SEEK_TO_NEXT_MEDIA_ITEM) .remove(COMMAND_SEEK_TO_PREVIOUS) .remove(COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM) .build() ) .setAvailableSessionCommands( ConnectionResult.DEFAULT_SESSION_COMMANDS.buildUpon() .add(customCommandFavorites) .build() ) .build() } override fun onCustomCommand( session: MediaSession, controller: MediaSession.ControllerInfo, customCommand: SessionCommand, args: Bundle ): ListenableFuture{ if (customCommand.customAction == ACTION_FAVORITES) { // Do custom logic here saveToFavorites(session.player.currentMediaItem) return Futures.immediateFuture(SessionResult(SessionResult.RESULT_SUCCESS)) } return super.onCustomCommand(session, controller, customCommand, args) } } }
Java
public class PlaybackService extends MediaSessionService { private static final SessionCommand CUSTOM_COMMAND_FAVORITES = new SessionCommand("ACTION_FAVORITES", Bundle.EMPTY); @Nullable private MediaSession mediaSession; public void onCreate() { super.onCreate(); CommandButton favoriteButton = new CommandButton.Builder() .setDisplayName("Save to favorites") .setIconResId(R.drawable.favorite_icon) .setSessionCommand(CUSTOM_COMMAND_FAVORITES) .build(); Player player = new ExoPlayer.Builder(this).build(); // Build the session with a custom layout. mediaSession = new MediaSession.Builder(this, player) .setCallback(new MyCallback()) .setCustomLayout(ImmutableList.of(favoriteButton)) .build(); } private static class MyCallback implements MediaSession.Callback { @Override public ConnectionResult onConnect( MediaSession session, MediaSession.ControllerInfo controller) { // Set available player and session commands. return new AcceptedResultBuilder(session) .setAvailablePlayerCommands( ConnectionResult.DEFAULT_PLAYER_COMMANDS.buildUpon() .remove(COMMAND_SEEK_TO_NEXT) .remove(COMMAND_SEEK_TO_NEXT_MEDIA_ITEM) .remove(COMMAND_SEEK_TO_PREVIOUS) .remove(COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM) .build()) .setAvailableSessionCommands( ConnectionResult.DEFAULT_SESSION_COMMANDS.buildUpon() .add(CUSTOM_COMMAND_FAVORITES) .build()) .build(); } public ListenableFutureonCustomCommand( MediaSession session, MediaSession.ControllerInfo controller, SessionCommand customCommand, Bundle args) { if (customCommand.customAction.equals(CUSTOM_COMMAND_FAVORITES.customAction)) { // Do custom logic here saveToFavorites(session.getPlayer().getCurrentMediaItem()); return Futures.immediateFuture(new SessionResult(SessionResult.RESULT_SUCCESS)); } return MediaSession.Callback.super.onCustomCommand( session, controller, customCommand, args); } } }
للاطّلاع على مزيد من المعلومات عن ضبط MediaSession
ليتمكّن العملاء، مثل
النظام، من الاتصال بتطبيق الوسائط، يُرجى الاطّلاع على مقالة
منح التحكّم للعملاء الآخرين.
باستخدام Jetpack Media3، عند تنفيذ MediaSession
، يتم تلقائيًا تعديل PlaybackState
وفقًا لأحدث إصدار من مشغّل الوسائط. وبالمثل، عند
تنفيذ MediaSessionService
، تنشر المكتبة تلقائيًا
MediaStyle
إشعارًا
نيابةً عنك وتحافظ على تحديثه.
الردّ على أزرار الإجراءات
عندما ينقر المستخدم على زر إجراء في عناصر التحكّم في الوسائط في النظام، يُرسِل MediaController
في النظام أمر تشغيل إلى MediaSession
. بعد ذلك، يفوّض
MediaSession
هذه الأوامر إلى المشغّل. تعالج جلسة الوسائط تلقائيًا الأوامر
المحدّدة في واجهة Player
Media3.
راجِع مقالة إضافة طلبات مخصّصة للحصول على إرشادات حول كيفية الردّ على طلب مخصّص.
السلوك في الإصدارات الأقدم من Android 13
للحفاظ على التوافق مع الإصدارات القديمة، يواصل واجهة المستخدم في النظام توفير تنسيق بديل
يستخدم إجراءات الإشعارات للتطبيقات التي لا يتم تحديثها لاستهداف Android 13،
أو التي لا تتضمّن معلومات PlaybackState
. يتم تحديد أزرار الإجراءات
من قائمة Notification.Action
المرفقة بالإشعار MediaStyle
. يعرض النظام ما يصل إلى خمسة إجراءات بالترتيب الذي تمت فيه
إضافتها. في الوضع المكثّف، يتم عرض ما يصل إلى ثلاثة أزرار، ويتم تحديدها من خلال
القيم التي تم تمريرها إلى setShowActionsInCompactView()
.
يتم وضع الإجراءات المخصّصة بالترتيب الذي تمت إضافتها به إلى
PlaybackState
.
يوضّح مثال الرمز البرمجي التالي كيفية إضافة إجراءات إلى MediaStyle notification :
Kotlin
import androidx.core.app.NotificationCompat import androidx.media3.session.MediaStyleNotificationHelper var notification = NotificationCompat.Builder(context, CHANNEL_ID) // Show controls on lock screen even when user hides sensitive content. .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) .setSmallIcon(R.drawable.ic_stat_player) // Add media control buttons that invoke intents in your media service .addAction(R.drawable.ic_prev, "Previous", prevPendingIntent) // #0 .addAction(R.drawable.ic_pause, "Pause", pausePendingIntent) // #1 .addAction(R.drawable.ic_next, "Next", nextPendingIntent) // #2 // Apply the media style template .setStyle(MediaStyleNotificationHelper.MediaStyle(mediaSession) .setShowActionsInCompactView(1 /* #1: pause button */)) .setContentTitle("Wonderful music") .setContentText("My Awesome Band") .setLargeIcon(albumArtBitmap) .build()
Java
import androidx.core.app.NotificationCompat; import androidx.media3.session.MediaStyleNotificationHelper; NotificationCompat.Builder notification = new NotificationCompat.Builder(context, CHANNEL_ID) .setVisibility(NotificationCompat.VISIBILITY_PUBLIC) .setSmallIcon(R.drawable.ic_stat_player) .addAction(R.drawable.ic_prev, "Previous", prevPendingIntent) .addAction(R.drawable.ic_pause, "Pause", pausePendingIntent) .addAction(R.drawable.ic_next, "Next", nextPendingIntent) .setStyle(new MediaStyleNotificationHelper.MediaStyle(mediaSession) .setShowActionsInCompactView(1 /* #1: pause button */)) .setContentTitle("Wonderful music") .setContentText("My Awesome Band") .setLargeIcon(albumArtBitmap) .build();
إتاحة استئناف تشغيل الوسائط
تتيح ميزة "استئناف الوسائط" للمستخدمين إعادة تشغيل الجلسات السابقة من لوحة العرض الدوّارة بدون الحاجة إلى تشغيل التطبيق. وعند بدء التشغيل، يتفاعل المستخدم مع عناصر التحكّم في الوسائط بالطريقة المعتادة.
يمكن تفعيل ميزة استئناف التشغيل وإيقافها باستخدام تطبيق "الإعدادات"، ضمن خيارات الصوت > الوسائط. يمكن للمستخدم أيضًا الوصول إلى "الإعدادات" من خلال النقر على رمز الترس الذي يظهر بعد التمرير سريعًا في لوحة العرض الدوّارة الموسّعة.
تقدّم Media3 واجهات برمجة تطبيقات لتسهيل إتاحة استئناف تشغيل الوسائط. اطّلِع على مستندات استئناف التشغيل باستخدام Media3 للحصول على إرشادات حول تنفيذ هذه الميزة.
استخدام واجهات برمجة التطبيقات القديمة للوسائط
يوضّح هذا القسم كيفية الدمج مع عناصر التحكّم في الوسائط في النظام باستخدام واجهات برمجة التطبيقات MediaCompat القديمة.
يسترجع النظام المعلومات التالية من MediaMetadata
MediaSession
، ويعرضها عند توفّرها:
METADATA_KEY_ALBUM_ART_URI
METADATA_KEY_TITLE
METADATA_KEY_DISPLAY_TITLE
METADATA_KEY_ARTIST
METADATA_KEY_DURATION
(إذا لم يتم ضبط المدة، لن يُظهر شريط التمرير مدى التقدّم)
لضمان تلقّي إشعار صالح ودقيق لعناصر التحكّم في الوسائط،
اضبط قيمة البيانات الوصفية METADATA_KEY_TITLE
أو METADATA_KEY_DISPLAY_TITLE
على عنوان الوسائط التي يتم تشغيلها حاليًا.
يعرض مشغّل الوسائط الوقت المنقضي للوسائط التي يتم تشغيلها حاليًا، بالإضافة إلى شريط تقدّم يتم ربطه بالرمز MediaSession
PlaybackState
.
يعرض مشغّل الوسائط مستوى تقدّم الوسائط التي يتم تشغيلها حاليًا، بالإضافة إلى
شريط تقدّم يتم ربطه برمز MediaSession
PlaybackState
. يسمح شريط التمرير
للمستخدمين بتغيير الموضع وعرض الوقت المنقضي لعنصر
الوسائط. لتفعيل شريط التمرير، يجب تنفيذ
PlaybackState.Builder#setActions
وتضمين ACTION_SEEK_TO
.
الشريحة | الإجراء | المعايير |
---|---|---|
1 | تشغيل |
الحالة الحالية لـ PlaybackState هي إحدى الحالات التالية:
|
مؤشر سريان العمل |
تكون الحالة الحالية للPlaybackState إحدى الحالات التالية:
|
|
إيقاف مؤقت | لا تنطبق أيّ من الحالات السابقة على الحالة الحالية لـ PlaybackState . |
|
2 | الصفحة السابقة | تشمل PlaybackState الإجراءات ACTION_SKIP_TO_PREVIOUS . |
قرض مخصص | PlaybackState الإجراءات لا تتضمّن ACTION_SKIP_TO_PREVIOUS وPlaybackState الإجراءات المخصّصة تتضمّن إجراءً مخصّصًا لم يتم وضعه بعد. |
|
فارغ | PlaybackState تشمل الإضافات قيمة منطقية true للمفتاح SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_PREV . |
|
3 | التالي | تشمل PlaybackState الإجراءات ACTION_SKIP_TO_NEXT . |
قرض مخصص | PlaybackState الإجراءات لا تتضمّن ACTION_SKIP_TO_NEXT وPlaybackState الإجراءات المخصّصة تتضمّن إجراءً مخصّصًا لم يتم وضعه بعد. |
|
فارغ | PlaybackState تشمل الإضافات قيمة منطقية true للمفتاح SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_NEXT . |
|
4 | قرض مخصص | PlaybackState تتضمّن الإجراءات المخصّصة إجراءً مخصّصًا لم يتم وضعه بعد. |
5 | قرض مخصص | PlaybackState تتضمّن الإجراءات المخصّصة إجراءً مخصّصًا لم يتم وضعه بعد. |
إضافة إجراءات عادية
توضّح أمثلة الرموز البرمجية التالية كيفية إضافة PlaybackState
إجراءات عادية
ومخصّصة.
بالنسبة إلى الإجراءات "تشغيل" و"إيقاف مؤقت" و"السابق" و"التالي"، اضبط هذه الإجراءات في
PlaybackState
لجلسة الوسائط.
Kotlin
val session = MediaSessionCompat(context, TAG) val playbackStateBuilder = PlaybackStateCompat.Builder() val style = NotificationCompat.MediaStyle() // For this example, the media is currently paused: val state = PlaybackStateCompat.STATE_PAUSED val position = 0L val playbackSpeed = 1f playbackStateBuilder.setState(state, position, playbackSpeed) // And the user can play, skip to next or previous, and seek val stateActions = PlaybackStateCompat.ACTION_PLAY or PlaybackStateCompat.ACTION_PLAY_PAUSE or PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS or PlaybackStateCompat.ACTION_SKIP_TO_NEXT or PlaybackStateCompat.ACTION_SEEK_TO // adding the seek action enables seeking with the seekbar playbackStateBuilder.setActions(stateActions) // ... do more setup here ... session.setPlaybackState(playbackStateBuilder.build()) style.setMediaSession(session.sessionToken) notificationBuilder.setStyle(style)
Java
MediaSessionCompat session = new MediaSessionCompat(context, TAG); PlaybackStateCompat.Builder playbackStateBuilder = new PlaybackStateCompat.Builder(); NotificationCompat.MediaStyle style = new NotificationCompat.MediaStyle(); // For this example, the media is currently paused: int state = PlaybackStateCompat.STATE_PAUSED; long position = 0L; float playbackSpeed = 1f; playbackStateBuilder.setState(state, position, playbackSpeed); // And the user can play, skip to next or previous, and seek long stateActions = PlaybackStateCompat.ACTION_PLAY | PlaybackStateCompat.ACTION_PLAY_PAUSE | PlaybackStateCompat.ACTION_SKIP_TO_PREVIOUS | PlaybackStateCompat.ACTION_SKIP_TO_NEXT | PlaybackStateCompat.ACTION_SEEK_TO; // adding this enables the seekbar thumb playbackStateBuilder.setActions(stateActions); // ... do more setup here ... session.setPlaybackState(playbackStateBuilder.build()); style.setMediaSession(session.getSessionToken()); notificationBuilder.setStyle(style);
إذا كنت لا تريد أي أزرار في خانتَي "السابق" أو "التالي"، لا تُضِف ACTION_SKIP_TO_PREVIOUS
أو ACTION_SKIP_TO_NEXT
، بل أضِف عناصر إضافية إلى الجلسة:
Kotlin
session.setExtras(Bundle().apply { putBoolean(SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_PREV, true) putBoolean(SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_NEXT, true) })
Java
Bundle extras = new Bundle(); extras.putBoolean(SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_PREV, true); extras.putBoolean(SESSION_EXTRAS_KEY_SLOT_RESERVATION_SKIP_TO_NEXT, true); session.setExtras(extras);
إضافة إجراءات مخصّصة
بالنسبة إلى الإجراءات الأخرى التي تريد عرضها في عناصر التحكّم بالوسائط، يمكنك إنشاء رمز
PlaybackStateCompat.CustomAction
وإضافته إلى PlaybackState
بدلاً من ذلك. تظهر هذه الإجراءات في
الترتيب الذي تمت إضافتها به.
Kotlin
val customAction = PlaybackStateCompat.CustomAction.Builder( "com.example.MY_CUSTOM_ACTION", // action ID "Custom Action", // title - used as content description for the button R.drawable.ic_custom_action ).build() playbackStateBuilder.addCustomAction(customAction)
Java
PlaybackStateCompat.CustomAction customAction = new PlaybackStateCompat.CustomAction.Builder( "com.example.MY_CUSTOM_ACTION", // action ID "Custom Action", // title - used as content description for the button R.drawable.ic_custom_action ).build(); playbackStateBuilder.addCustomAction(customAction);
الاستجابة لإجراءات PlaybackState
عندما ينقر المستخدم على زر، يستخدم SystemUI
MediaController.TransportControls
لإرسال أمر إلى MediaSession
. عليك تسجيل طلب معاودة اتصال
يمكنه الاستجابة بشكل صحيح لهذه الأحداث.
Kotlin
val callback = object: MediaSession.Callback() { override fun onPlay() { // start playback } override fun onPause() { // pause playback } override fun onSkipToPrevious() { // skip to previous } override fun onSkipToNext() { // skip to next } override fun onSeekTo(pos: Long) { // jump to position in track } override fun onCustomAction(action: String, extras: Bundle?) { when (action) { CUSTOM_ACTION_1 -> doCustomAction1(extras) CUSTOM_ACTION_2 -> doCustomAction2(extras) else -> { Log.w(TAG, "Unknown custom action $action") } } } } session.setCallback(callback)
Java
MediaSession.Callback callback = new MediaSession.Callback() { @Override public void onPlay() { // start playback } @Override public void onPause() { // pause playback } @Override public void onSkipToPrevious() { // skip to previous } @Override public void onSkipToNext() { // skip to next } @Override public void onSeekTo(long pos) { // jump to position in track } @Override public void onCustomAction(String action, Bundle extras) { if (action.equals(CUSTOM_ACTION_1)) { doCustomAction1(extras); } else if (action.equals(CUSTOM_ACTION_2)) { doCustomAction2(extras); } else { Log.w(TAG, "Unknown custom action " + action); } } };
استئناف تشغيل الوسائط
لظهور تطبيق المشغّل في منطقة الإعدادات السريعة،
يجب إنشاء إشعار MediaStyle
يتضمّن رمز MediaSession
مفتاح تنشيط صالحًا.
لعرض عنوان إشعار MediaStyle، استخدِم
NotificationBuilder.setContentTitle()
.
لعرض رمز العلامة التجارية لمشغّل الوسائط، استخدِم
NotificationBuilder.setSmallIcon()
.
لتفعيل ميزة استئناف التشغيل، يجب أن توفّر التطبيقات MediaBrowserService
وMediaSession
. يجب أن ينفِّذ MediaSession
طلب إعادة الاتصال onPlay()
.
تنفيذ MediaBrowserService
بعد تشغيل الجهاز، يبحث النظام عن التطبيقات الخمسة التي تم استخدامها مؤخرًا لتشغيل الوسائط، ويقدّم عناصر تحكّم يمكن استخدامها لإعادة تشغيل المحتوى من كل تطبيق.
يحاول النظام التواصل مع MediaBrowserService
من خلال اتصال من SystemUI. يجب أن يسمح تطبيقك بعمليات الربط هذه، وإلا لن يتمكّن من إعادة تشغيل المحتوى.
يمكن تحديد الاتصالات الواردة من SystemUI والتحقّق منها باستخدام اسم الحزمة
com.android.systemui
والتوقيع. تم توقيع SystemUI باستخدام توقيع
النظام الأساسي. يمكنك الاطّلاع على مثال على كيفية التحقّق من توقيع المنصة في تطبيق UAMP.
لكي تتيح إمكانية استئناف التشغيل، يجب أن ينفذ MediaBrowserService
السلوكَين التاليَين:
يجب أن يعرض
onGetRoot()
جذرًا غير صفري بسرعة. يجب التعامل مع المنطق المعقد الآخر فيonLoadChildren()
عند استدعاء
onLoadChildren()
على معرّف الوسائط الجذر، يجب أن تحتوي النتيجة على عنصر FLAG_PLAYABLE تابع.من المفترض أن يعرض
MediaBrowserService
عنصر الوسائط الذي تم تشغيله مؤخرًا عندتلقّي طلب البحث EXTRA_RECENT. يجب أن تكون القيمة المعروضة عنصر وسائط فعليًا بدلاً من دالة عامة.يجب أن يوفّر
MediaBrowserService
عنصر MediaDescription مناسبًا يتضمّن عنوانًا وعنوانًا فرعيًا غير فارغَين. يجب أيضًا ضبط معرّف موارد منتظم للرمز أو صورة نقطية للرمز.
توضّح أمثلة الرموز البرمجية التالية كيفية تنفيذ onGetRoot()
.
Kotlin
override fun onGetRoot( clientPackageName: String, clientUid: Int, rootHints: Bundle? ): BrowserRoot? { ... // Verify that the specified package is SystemUI. You'll need to write your // own logic to do this. if (isSystem(clientPackageName, clientUid)) { rootHints?.let { if (it.getBoolean(BrowserRoot.EXTRA_RECENT)) { // Return a tree with a single playable media item for resumption. val extras = Bundle().apply { putBoolean(BrowserRoot.EXTRA_RECENT, true) } return BrowserRoot(MY_RECENTS_ROOT_ID, extras) } } // You can return your normal tree if the EXTRA_RECENT flag is not present. return BrowserRoot(MY_MEDIA_ROOT_ID, null) } // Return an empty tree to disallow browsing. return BrowserRoot(MY_EMPTY_ROOT_ID, null)
Java
@Override public BrowserRoot onGetRoot(String clientPackageName, int clientUid, Bundle rootHints) { ... // Verify that the specified package is SystemUI. You'll need to write your // own logic to do this. if (isSystem(clientPackageName, clientUid)) { if (rootHints != null) { if (rootHints.getBoolean(BrowserRoot.EXTRA_RECENT)) { // Return a tree with a single playable media item for resumption. Bundle extras = new Bundle(); extras.putBoolean(BrowserRoot.EXTRA_RECENT, true); return new BrowserRoot(MY_RECENTS_ROOT_ID, extras); } } // You can return your normal tree if the EXTRA_RECENT flag is not present. return new BrowserRoot(MY_MEDIA_ROOT_ID, null); } // Return an empty tree to disallow browsing. return new BrowserRoot(MY_EMPTY_ROOT_ID, null); }