إضافة الميزات الأساسية إلى جهاز استقبال Android TV

تضم هذه الصفحة مقتطفات رموز وأوصافًا للميزات المتاحة لتخصيص تطبيق مستقبِل Android TV.

ضبط المكتبات

لإتاحة واجهات برمجة تطبيقات Cast Connect لتطبيق Android TV:

نظام التشغيل Android
  1. افتح الملف build.gradle داخل دليل وحدة التطبيق.
  2. تأكّد من تضمين google() في repositories المدرَجة.
      repositories {
        google()
      }
  3. حسب نوع الجهاز المستهدَف لتطبيقك، أضِف أحدث إصدارات المكتبات إلى تبعياتك:
    • بالنسبة إلى تطبيق مستقبِل Android:
        dependencies {
          implementation 'com.google.android.gms:play-services-cast-tv:21.0.0'
          implementation 'com.google.android.gms:play-services-cast:21.3.0'
        }
    • بالنسبة إلى تطبيق "مرسِل Android":
        dependencies {
          implementation 'com.google.android.gms:play-services-cast:21.0.0'
          implementation 'com.google.android.gms:play-services-cast-framework:21.3.0'
        }
    احرص على تعديل رقم الإصدار هذا في كل مرة يتم فيها تعديل الخدمات.
  4. احفظ التغييرات وانقر على Sync Project with Gradle Files في شريط الأدوات.
iOS
  1. تأكَّد من أنّ Podfile تستهدف google-cast-sdk 4.8.0 أو إصدار أحدث.
  2. الإصدار 13 أو الإصدارات الأحدث من iOS يمكنك الاطّلاع على ملاحظات الإصدار للحصول على مزيد من التفاصيل.
      platform: ios, '13'
    
      def target_pods
         pod 'google-cast-sdk', '~>4.8.0'
      end
الويب
  1. يتطلب الإصدار M87 من متصفّح Chromium أو إصدارًا أحدث.
  2. إضافة مكتبة Web sender API إلى مشروعك
      <script src="//www.gstatic.com/cv/js/sender/v1/cast_sender.js?loadCastFramework=1"></script>

متطلبات AndroidX

تتطلب الإصدارات الجديدة من خدمات Google Play تحديث التطبيق لاستخدام مساحة الاسم androidx. اتّبِع تعليمات نقل البيانات إلى AndroidX.

تطبيق Android TV: المتطلبات الأساسية

لإتاحة بث المحتوى في تطبيق Android TV في تطبيق Android TV، يجب إنشاء أحداث ودعمها من جلسة وسائط. تقدّم البيانات التي تقدّمها جلسة الوسائط الخاصة بك معلومات أساسية عن حالة وسائطك، مثل الموضع وحالة التشغيل وما إلى ذلك. تستخدم مكتبة Cast Connect جلسة الوسائط أيضًا للإشارة إلى تلقّي رسائل معينة من أحد المُرسِلين، مثل الإيقاف المؤقت.

لمزيد من المعلومات عن جلسة الوسائط وطريقة إعداد جلسة وسائط، يُرجى الاطّلاع على دليل استخدام جلسة وسائط.

دورة حياة جلسة الوسائط

من المفترض أن ينشئ تطبيقك جلسة وسائط عند بدء التشغيل وتطلقه عندما لا يمكن التحكّم فيه بعد الآن. على سبيل المثال، إذا كان تطبيقك عبارة عن تطبيق فيديو، عليك إصدار الجلسة عندما يخرج المستخدم من نشاط التشغيل، إما عن طريق اختيار "رجوع" لتصفّح محتوى آخر أو بإنشاء خلفية للتطبيق. إذا كان تطبيقك موسيقيًا، يجب إصداره عندما يتوقّف تطبيقك عن تشغيل أي وسائط.

جارٍ تعديل حالة الجلسة.

ويجب إبقاء البيانات في جلسة الوسائط محدّثة مع حالة المشغّل. على سبيل المثال، عند إيقاف التشغيل مؤقتًا، يجب تعديل حالة التشغيل والإجراءات المتاحة. وتُدرِج الجداول التالية الحالات التي تقع على عاتقك مسؤولية إبقاءها محدّثة.

MediaDataCompat

حقل البيانات الوصفية الوصف
MetaDATA_KEY_TITLE (مطلوبة) تمثّل هذه السمة عنوان الوسائط.
MetaDATA_KEY_DISPLAY_SUBTITLE العنوان الفرعي
MetaDATA_KEY_DISPLAY_ICON_URI عنوان URL للرمز.
MetaDATA_KEY_DURATION (مطلوبة) مدة عرض الوسائط
MetaDATA_KEY_MEDIA_URI رقم تعريف المحتوى
MetaDATA_KEY_ARTIST الفنان
MetaDATA_KEY_ألبوم الألبوم

تشغيل StateCompat

الطريقة المطلوبة الوصف
setAction() لضبط أوامر الوسائط المتوافقة.
setState() اضبط حالة التشغيل والموضع الحالي.

MediaSessionCompat

الطريقة المطلوبة الوصف
setالتكرارMode() لضبط وضع التكرار.
setShuffleMode() لضبط وضع الترتيب العشوائي.
setmetadata() لضبط البيانات الوصفية للوسائط
setتشغيلState() لضبط حالة التشغيل.
كوتلين
private fun updateMediaSession() {
    val metadata = MediaMetadataCompat.Builder()
         .putString(MediaMetadataCompat.METADATA_KEY_TITLE, "title")
         .putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_SUBTITLE, "subtitle")
         .putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_ICON_URI, mMovie.getCardImageUrl())
         .build()

    val playbackState = PlaybackStateCompat.Builder()
         .setState(
             PlaybackStateCompat.STATE_PLAYING,
             player.getPosition(),
             player.getPlaybackSpeed(),
             System.currentTimeMillis()
        )
         .build()

    mediaSession.setMetadata(metadata)
    mediaSession.setPlaybackState(playbackState)
}
JavaScript
private void updateMediaSession() {
  MediaMetadataCompat metadata =
      new MediaMetadataCompat.Builder()
          .putString(MediaMetadataCompat.METADATA_KEY_TITLE, "title")
          .putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_SUBTITLE, "subtitle")
          .putString(MediaMetadataCompat.METADATA_KEY_DISPLAY_ICON_URI,mMovie.getCardImageUrl())
          .build();

  PlaybackStateCompat playbackState =
      new PlaybackStateCompat.Builder()
          .setState(
               PlaybackStateCompat.STATE_PLAYING,
               player.getPosition(),
               player.getPlaybackSpeed(),
               System.currentTimeMillis())
          .build();

  mediaSession.setMetadata(metadata);
  mediaSession.setPlaybackState(playbackState);
}

التعامل مع عناصر التحكّم في النقل

يجب أن ينفِّذ تطبيقك استدعاء التحكّم في نقل جلسات الوسائط. يعرض الجدول التالي إجراءات التحكّم في النقل التي يحتاج إلى التعامل معها:

MediaSessionCompat.Callback

المهام الوصف
onPlay() استئناف
onPause() إيقاف مؤقت
onSeekTo() التقديم إلى موضع
onStop() إيقاف الوسائط الحالية
كوتلين
class MyMediaSessionCallback : MediaSessionCompat.Callback() {
  override fun onPause() {
    // Pause the player and update the play state.
    ...
  }

  override fun onPlay() {
    // Resume the player and update the play state.
    ...
  }

  override fun onSeekTo (long pos) {
    // Seek and update the play state.
    ...
  }
  ...
}

mediaSession.setCallback( MyMediaSessionCallback() );
JavaScript
public MyMediaSessionCallback extends MediaSessionCompat.Callback {
  public void onPause() {
    // Pause the player and update the play state.
    ...
  }

  public void onPlay() {
    // Resume the player and update the play state.
    ...
  }

  public void onSeekTo (long pos) {
    // Seek and update the play state.
    ...
  }
  ...
}

mediaSession.setCallback(new MyMediaSessionCallback());

ضبط إعدادات دعم البث

عندما يتم إرسال طلب إطلاق من خلال تطبيق مرسِل، يتم إنشاء هدف باستخدام مساحة اسم تطبيق. يكون تطبيقك مسؤولاً عن التعامل معه وإنشاء مثال على عنصر CastReceiverContext عند تشغيل تطبيق التلفزيون. يلزم توفُّر عنصر CastReceiverContext للتفاعل مع البث مع تشغيل تطبيق التلفزيون. يتيح هذا الكائن لتطبيق التلفزيون إمكانية قبول رسائل وسائط البث الواردة من أي مُرسِلين متصلين.

إعداد Android TV

إضافة فلتر intent للإطلاق

يمكنك إضافة فلتر الأهداف الجديد إلى النشاط الذي تريد التعامل مع نية التشغيل من تطبيق المُرسِل:

<activity android:name="com.example.activity">
  <intent-filter>
      <action android:name="com.google.android.gms.cast.tv.action.LAUNCH" />
      <category android:name="android.intent.category.DEFAULT" />
  </intent-filter>
</activity>

تحديد موفّر خيارات المستلِم

عليك تنفيذ السمة ReceiverOptionsProvider لتوفير CastReceiverOptions:

كوتلين
class MyReceiverOptionsProvider : ReceiverOptionsProvider {
  override fun getOptions(context: Context?): CastReceiverOptions {
    return CastReceiverOptions.Builder(context)
          .setStatusText("My App")
          .build()
    }
}
JavaScript
public class MyReceiverOptionsProvider implements ReceiverOptionsProvider {
  @Override
  public CastReceiverOptions getOptions(Context context) {
    return new CastReceiverOptions.Builder(context)
        .setStatusText("My App")
        .build();
  }
}

حدِّد بعد ذلك موفِّر الخيارات في AndroidManifest:

 <meta-data
    android:name="com.google.android.gms.cast.tv.RECEIVER_OPTIONS_PROVIDER_CLASS_NAME"
    android:value="com.example.mysimpleatvapplication.MyReceiverOptionsProvider" />

يتم استخدام ReceiverOptionsProvider لتوفير CastReceiverOptions عند إعداد CastReceiverContext.

سياق جهاز استقبال البث

إعداد CastReceiverContext عند إنشاء تطبيقك:

كوتلين
override fun onCreate() {
  CastReceiverContext.initInstance(this)

  ...
}
JavaScript
@Override
public void onCreate() {
  CastReceiverContext.initInstance(this);

  ...
}

يجب تفعيل CastReceiverContext عند انتقال التطبيق إلى المقدّمة:

كوتلين
CastReceiverContext.getInstance().start()
JavaScript
CastReceiverContext.getInstance().start();

الاتصال stop() على CastReceiverContext بعد ظهور التطبيق في الخلفية لتطبيقات أو تطبيقات الفيديو التي لا تتيح تشغيل الخلفية:

كوتلين
// Player has stopped.
CastReceiverContext.getInstance().stop()
JavaScript
// Player has stopped.
CastReceiverContext.getInstance().stop();

بالإضافة إلى ذلك، إذا كان تطبيقك يتيح التشغيل في الخلفية، يمكنك الاتصال بالرقم stop() على CastReceiverContext عندما يتوقّف التشغيل أثناء التشغيل في الخلفية.

ننصحك بشدة باستخدام LifecycleObserver من المكتبة androidx.lifecycle لإدارة المكالمات CastReceiverContext.start() و CastReceiverContext.stop()، خاصةً إذا كان التطبيق المحلي يتضمن أنشطة متعددة. يؤدي ذلك إلى تجنُّب شروط السباقات عندما تتصل بـ "start()" و"stop()" من أنشطة مختلفة.

كوتلين
// Create a LifecycleObserver class.
class MyLifecycleObserver : DefaultLifecycleObserver {
  override fun onStart(owner: LifecycleOwner) {
    // App prepares to enter foreground.
    CastReceiverContext.getInstance().start()
  }

  override fun onStop(owner: LifecycleOwner) {
    // App has moved to the background or has terminated.
    CastReceiverContext.getInstance().stop()
  }
}

// Add the observer when your application is being created.
class MyApplication : Application() {
  fun onCreate() {
    super.onCreate()

    // Initialize CastReceiverContext.
    CastReceiverContext.initInstance(this /* android.content.Context */)

    // Register LifecycleObserver
    ProcessLifecycleOwner.get().lifecycle.addObserver(
        MyLifecycleObserver())
  }
}
JavaScript
// Create a LifecycleObserver class.
public class MyLifecycleObserver implements DefaultLifecycleObserver {
  @Override
  public void onStart(LifecycleOwner owner) {
    // App prepares to enter foreground.
    CastReceiverContext.getInstance().start();
  }

  @Override
  public void onStop(LifecycleOwner owner) {
    // App has moved to the background or has terminated.
    CastReceiverContext.getInstance().stop();
  }
}

// Add the observer when your application is being created.
public class MyApplication extends Application {
  @Override
  public void onCreate() {
    super.onCreate();

    // Initialize CastReceiverContext.
    CastReceiverContext.initInstance(this /* android.content.Context */);

    // Register LifecycleObserver
    ProcessLifecycleOwner.get().getLifecycle().addObserver(
        new MyLifecycleObserver());
  }
}
// In AndroidManifest.xml set MyApplication as the application class
<application
    ...
    android:name=".MyApplication">

ربط MediaSession بحساب MediaManager

عند إنشاء MediaSession، عليك أيضًا تقديم الرمز المميّز MediaSession الحالي إلى CastReceiverContext حتى يتمكّن من معرفة مكان إرسال الأوامر واسترجاع حالة تشغيل الوسائط:

كوتلين
val mediaManager: MediaManager = receiverContext.getMediaManager()
mediaManager.setSessionCompatToken(currentMediaSession.getSessionToken())
JavaScript
MediaManager mediaManager = receiverContext.getMediaManager();
mediaManager.setSessionCompatToken(currentMediaSession.getSessionToken());

عندما يتم إصدار MediaSession بسبب تشغيل غير نشط، يجب ضبط رمز مميّز فارغ على MediaManager:

كوتلين
myPlayer.stop()
mediaSession.release()
mediaManager.setSessionCompatToken(null)
JavaScript
myPlayer.stop();
mediaSession.release();
mediaManager.setSessionCompatToken(null);

إذا كان تطبيقك يتيح تشغيل الوسائط في الخلفية، بدلاً من الاتصال CastReceiverContext.stop() عندما يتم إرسال تطبيقك إلى الخلفية، يجب عدم الاتصال به إلا عندما يكون تطبيقك في الخلفية وعدم تشغيل الوسائط بعد ذلك. على سبيل المثال:

كوتلين
class MyLifecycleObserver : DefaultLifecycleObserver {
  ...
  // App has moved to the background.
  override fun onPause(owner: LifecycleOwner) {
    mIsBackground = true
    myStopCastReceiverContextIfNeeded()
  }
}

// Stop playback on the player.
private fun myStopPlayback() {
  myPlayer.stop()

  myStopCastReceiverContextIfNeeded()
}

// Stop the CastReceiverContext when both the player has
// stopped and the app has moved to the background.
private fun myStopCastReceiverContextIfNeeded() {
  if (mIsBackground && myPlayer.isStopped()) {
    CastReceiverContext.getInstance().stop()
  }
}
JavaScript
public class MyLifecycleObserver implements DefaultLifecycleObserver {
  ...
  // App has moved to the background.
  @Override
  public void onPause(LifecycleOwner owner) {
    mIsBackground = true;

    myStopCastReceiverContextIfNeeded();
  }
}

// Stop playback on the player.
private void myStopPlayback() {
  myPlayer.stop();

  myStopCastReceiverContextIfNeeded();
}

// Stop the CastReceiverContext when both the player has
// stopped and the app has moved to the background.
private void myStopCastReceiverContextIfNeeded() {
  if (mIsBackground && myPlayer.isStopped()) {
    CastReceiverContext.getInstance().stop();
  }
}

استخدام مشغّل Exo مع Cast Connect

إذا كنت تستخدم Exoplayer، يمكنك استخدام MediaSessionConnector لحفظ الجلسة تلقائيًا وجميع المعلومات ذات الصلة، بما في ذلك حالة التشغيل بدلاً من تتبُّع التغييرات يدويًا.

يمكن استخدام السمة MediaSessionConnector.MediaButtonEventHandler للتعامل مع أحداث MediaButton من خلال الاتصال setMediaButtonEventHandler(MediaButtonEventHandler) بحيث يتم التعامل معها تلقائيًا من خلال MediaSessionCompat.Callback.

للدمج MediaSessionConnector في تطبيقك، أضِف ما يلي إلى فئة نشاط اللاعبين أو إلى أي مكان تدير فيه جلسة الوسائط:

كوتلين
class PlayerActivity : Activity() {
  private var mMediaSession: MediaSessionCompat? = null
  private var mMediaSessionConnector: MediaSessionConnector? = null
  private var mMediaManager: MediaManager? = null

  override fun onCreate(savedInstanceState: Bundle?) {
    ...
    mMediaSession = MediaSessionCompat(this, LOG_TAG)
    mMediaSessionConnector = MediaSessionConnector(mMediaSession!!)
    ...
  }

  override fun onStart() {
    ...
    mMediaManager = receiverContext.getMediaManager()
    mMediaManager!!.setSessionCompatToken(currentMediaSession.getSessionToken())
    mMediaSessionConnector!!.setPlayer(mExoPlayer)
    mMediaSessionConnector!!.setMediaMetadataProvider(mMediaMetadataProvider)
    mMediaSession!!.isActive = true
    ...
  }

  override fun onStop() {
    ...
    mMediaSessionConnector!!.setPlayer(null)
    mMediaSession!!.release()
    mMediaManager!!.setSessionCompatToken(null)
    ...
  }
}
JavaScript
public class PlayerActivity extends Activity {
  private MediaSessionCompat mMediaSession;
  private MediaSessionConnector mMediaSessionConnector;
  private MediaManager mMediaManager;

  @Override
  protected void onCreate(Bundle savedInstanceState) {
    ...
    mMediaSession = new MediaSessionCompat(this, LOG_TAG);
    mMediaSessionConnector = new MediaSessionConnector(mMediaSession);
    ...
  }

  @Override
  protected void onStart() {
    ...
    mMediaManager = receiverContext.getMediaManager();
    mMediaManager.setSessionCompatToken(currentMediaSession.getSessionToken());

    mMediaSessionConnector.setPlayer(mExoPlayer);
    mMediaSessionConnector.setMediaMetadataProvider(mMediaMetadataProvider);
    mMediaSession.setActive(true);
    ...
  }

  @Override
  protected void onStop() {
    ...
    mMediaSessionConnector.setPlayer(null);
    mMediaSession.release();
    mMediaManager.setSessionCompatToken(null);
    ...
  }
}

إعداد تطبيق المُرسِل

تفعيل دعم Cast Connect

بعد تحديث تطبيق المُرسِل باستخدام ميزة Cast Connect، يمكنك الإعلان عن جاهزيته عن طريق ضبط علامة androidReceiverCompatible على LaunchOptions.

نظام التشغيل Android

ويجب أن يتوفّر الإصدار play-services-cast-framework من نظام التشغيل 19.0.0 أو إصدار أحدث.

العلامة androidReceiverCompatible محدَّدة في LaunchOptions (التي تُعد جزءًا من CastOptions):

كوتلين
class CastOptionsProvider : OptionsProvider {
  override fun getCastOptions(context: Context?): CastOptions {
    val launchOptions: LaunchOptions = Builder()
          .setAndroidReceiverCompatible(true)
          .build()
    return CastOptions.Builder()
          .setLaunchOptions(launchOptions)
          ...
          .build()
    }
}
JavaScript
public class CastOptionsProvider implements OptionsProvider {
  @Override
  public CastOptions getCastOptions(Context context) {
    LaunchOptions launchOptions = new LaunchOptions.Builder()
              .setAndroidReceiverCompatible(true)
              .build();
    return new CastOptions.Builder()
        .setLaunchOptions(launchOptions)
        ...
        .build();
  }
}
iOS

ويجب استخدام الإصدار v4.4.8 من google-cast-sdk أو إصدار أحدث.

العلامة androidReceiverCompatible محدَّدة في GCKLaunchOptions (التي تُعد جزءًا من GCKCastOptions):

let options = GCKCastOptions(discoveryCriteria: GCKDiscoveryCriteria(applicationID: kReceiverAppID))
...
let launchOptions = GCKLaunchOptions()
launchOptions.androidReceiverCompatible = true
options.launchOptions = launchOptions
GCKCastContext.setSharedInstanceWith(options)
الويب

يتطلب إصدار متصفِّح Chromium M87 أو إصدارًا أحدث.

const context = cast.framework.CastContext.getInstance();
const castOptions = new cast.framework.CastOptions();
castOptions.receiverApplicationId = kReceiverAppID;
castOptions.androidReceiverCompatible = true;
context.setOptions(castOptions);

إعداد Play Console

إعداد تطبيق Android TV

أضِف اسم الحزمة لتطبيق Android TV في Play Developer Console لربطه بمعرّف تطبيق البث.

تسجيل أجهزة مطوّري البرامج

سجِّل الرقم التسلسلي لجهاز Android TV الذي ستستخدمه للتطوير في Play Developer Console.

بدون التسجيل، لن يعمل تطبيق Cast Connect إلا مع التطبيقات المثبتة من متجر Google Play لأسباب أمنية.

للاطّلاع على المزيد من المعلومات حول تسجيل جهاز Cast أو Android TV لعملية تطوير Cast، يُرجى الانتقال إلى صفحة التسجيل.

جارٍ تحميل الوسائط

إذا سبق لك تنفيذ دعم الروابط لصفحات في التطبيق في تطبيق Android TV، يجب أن يكون لديك تعريف مشابه تم إعداده في بيان Android TV:

<activity android:name="com.example.activity">
  <intent-filter>
     <action android:name="android.intent.action.VIEW" />
     <category android:name="android.intent.category.DEFAULT" />
     <data android:scheme="https"/>
     <data android:host="www.example.com"/>
     <data android:pathPattern=".*"/>
  </intent-filter>
</activity>

التحميل حسب الكيان على المُرسِل

في المُرسِلين، يمكنك تمرير الرابط لصفحة معيّنة في التطبيق من خلال ضبط entity في معلومات الوسائط لطلب التحميل:

كوتلين
val mediaToLoad = MediaInfo.Builder("some-id")
    .setEntity("https://example.com/watch/some-id")
    ...
    .build()
val loadRequest = MediaLoadRequestData.Builder()
    .setMediaInfo(mediaToLoad)
    .setCredentials("user-credentials")
    ...
    .build()
remoteMediaClient.load(loadRequest)
نظام التشغيل Android
Java
MediaInfo mediaToLoad =
    new MediaInfo.Builder("some-id")
        .setEntity("https://example.com/watch/some-id")
        ...
        .build();
MediaLoadRequestData loadRequest =
    new MediaLoadRequestData.Builder()
        .setMediaInfo(mediaToLoad)
        .setCredentials("user-credentials")
        ...
        .build();
remoteMediaClient.load(loadRequest);
iOS
let mediaInfoBuilder = GCKMediaInformationBuilder(entity: "https://example.com/watch/some-id")
...
mediaInformation = mediaInfoBuilder.build()

let mediaLoadRequestDataBuilder = GCKMediaLoadRequestDataBuilder()
mediaLoadRequestDataBuilder.mediaInformation = mediaInformation
mediaLoadRequestDataBuilder.credentials = "user-credentials"
...
let mediaLoadRequestData = mediaLoadRequestDataBuilder.build()

remoteMediaClient?.loadMedia(with: mediaLoadRequestData)
الويب

يتطلب إصدار متصفِّح Chromium M87 أو إصدارًا أحدث.

let mediaInfo = new chrome.cast.media.MediaInfo('some-id"', 'video/mp4');
mediaInfo.entity = 'https://example.com/watch/some-id';
...

let request = new chrome.cast.media.LoadRequest(mediaInfo);
request.credentials = 'user-credentials';
...

cast.framework.CastContext.getInstance().getCurrentSession().loadMedia(request);

يتم إرسال أمر التحميل من خلال هدف باستخدام رابط لصفحة في التطبيق واسم الحزمة الذي حدّدته في Play Console.

إعداد بيانات اعتماد ATV على المُرسِل

من المحتمل أن يكون تطبيق مستقبِل الويب وتطبيق Android TV متوافقَين مع روابط لصفحات في التطبيق وcredentials (على سبيل المثال، إذا كنت تتعامل مع المصادقة بشكل مختلف على النظامَين الأساسيَين). لحلّ هذه المشكلة، يمكنك توفير السمتَين البديلتين entity وcredentials لجهاز Android TV:

نظام التشغيل Android
كوتلين
val mediaToLoad = MediaInfo.Builder("some-id")
        .setEntity("https://example.com/watch/some-id")
        .setAtvEntity("myscheme://example.com/atv/some-id")
        ...
        .build()
val loadRequest = MediaLoadRequestData.Builder()
        .setMediaInfo(mediaToLoad)
        .setCredentials("user-credentials")
        .setAtvCredentials("atv-user-credentials")
        ...
        .build()
remoteMediaClient.load(loadRequest)
JavaScript
MediaInfo mediaToLoad =
    new MediaInfo.Builder("some-id")
        .setEntity("https://example.com/watch/some-id")
        .setAtvEntity("myscheme://example.com/atv/some-id")
        ...
        .build();
MediaLoadRequestData loadRequest =
    new MediaLoadRequestData.Builder()
        .setMediaInfo(mediaToLoad)
        .setCredentials("user-credentials")
        .setAtvCredentials("atv-user-credentials")
        ...
        .build();
remoteMediaClient.load(loadRequest);
iOS
let mediaInfoBuilder = GCKMediaInformationBuilder(entity: "https://example.com/watch/some-id")
mediaInfoBuilder.atvEntity = "myscheme://example.com/atv/some-id"
...
mediaInformation = mediaInfoBuilder.build()

let mediaLoadRequestDataBuilder = GCKMediaLoadRequestDataBuilder()
mediaLoadRequestDataBuilder.mediaInformation = mediaInformation
mediaLoadRequestDataBuilder.credentials = "user-credentials"
mediaLoadRequestDataBuilder.atvCredentials = "atv-user-credentials"
...
let mediaLoadRequestData = mediaLoadRequestDataBuilder.build()

remoteMediaClient?.loadMedia(with: mediaLoadRequestData)
الويب

يتطلب إصدار متصفِّح Chromium M87 أو إصدارًا أحدث.

let mediaInfo = new chrome.cast.media.MediaInfo('some-id"', 'video/mp4');
mediaInfo.entity = 'https://example.com/watch/some-id';
mediaInfo.atvEntity = 'myscheme://example.com/atv/some-id';
...

let request = new chrome.cast.media.LoadRequest(mediaInfo);
request.credentials = 'user-credentials';
request.atvCredentials = 'atv-user-credentials';
...

cast.framework.CastContext.getInstance().getCurrentSession().loadMedia(request);

في حال إطلاق تطبيق Web Receiver، يتم استخدام entity وcredentials في طلب التحميل. ومع ذلك، إذا تم إطلاق تطبيق Android TV، تلغي حزمة تطوير البرامج (SDK) تطبيقَي entity وcredentials باستخدام atvEntity وatvCredentials (إذا تم تحديدهما).

التحميل حسب Content ID أو MediaQueueData

إذا لم تكن تستخدم entity أو atvEntity، وكنت تستخدم Content ID أو عنوان URL للمحتوى في معلومات الوسائط أو تستخدم بيانات طلب تحميل الوسائط الأكثر تفصيلاً، يجب إضافة فلتر الأهداف المحدد مسبقًا التالي في تطبيق Android TV:

<activity android:name="com.example.activity">
  <intent-filter>
     <action android:name="com.google.android.gms.cast.tv.action.LOAD"/>
     <category android:name="android.intent.category.DEFAULT" />
  </intent-filter>
</activity>

بالنسبة إلى جهة الإرسال، على غرار التحميل حسب الكيان، يمكنك إنشاء طلب تحميل يتضمن معلومات المحتوى والاتصال بـ load().

نظام التشغيل Android
كوتلين
val mediaToLoad = MediaInfo.Builder("some-id").build()
val loadRequest = MediaLoadRequestData.Builder()
    .setMediaInfo(mediaToLoad)
    .setCredentials("user-credentials")
    ...
    .build()
remoteMediaClient.load(loadRequest)
JavaScript
MediaInfo mediaToLoad =
    new MediaInfo.Builder("some-id").build();
MediaLoadRequestData loadRequest =
    new MediaLoadRequestData.Builder()
        .setMediaInfo(mediaToLoad)
        .setCredentials("user-credentials")
        ...
        .build();
remoteMediaClient.load(loadRequest);
iOS
let mediaInfoBuilder = GCKMediaInformationBuilder(contentId: "some-id")
...
mediaInformation = mediaInfoBuilder.build()

let mediaLoadRequestDataBuilder = GCKMediaLoadRequestDataBuilder()
mediaLoadRequestDataBuilder.mediaInformation = mediaInformation
mediaLoadRequestDataBuilder.credentials = "user-credentials"
...
let mediaLoadRequestData = mediaLoadRequestDataBuilder.build()

remoteMediaClient?.loadMedia(with: mediaLoadRequestData)
الويب

يتطلب إصدار متصفِّح Chromium M87 أو إصدارًا أحدث.

let mediaInfo = new chrome.cast.media.MediaInfo('some-id"', 'video/mp4');
...

let request = new chrome.cast.media.LoadRequest(mediaInfo);
...

cast.framework.CastContext.getInstance().getCurrentSession().loadMedia(request);

التعامل مع طلبات التحميل

في نشاطك، إذا أردت معالجة طلبات التحميل هذه، عليك معالجة عناصر intent في عمليات استدعاء مراحل النشاط:

كوتلين
class MyActivity : Activity() {
  override fun onStart() {
    super.onStart()
    val mediaManager = CastReceiverContext.getInstance().getMediaManager()
    // Pass the intent to the SDK. You can also do this in onCreate().
    if (mediaManager.onNewIntent(intent)) {
        // If the SDK recognizes the intent, you should early return.
        return
    }
    // If the SDK doesn't recognize the intent, you can handle the intent with
    // your own logic.
    ...
  }

  // For some cases, a new load intent triggers onNewIntent() instead of
  // onStart().
  override fun onNewIntent(intent: Intent) {
    val mediaManager = CastReceiverContext.getInstance().getMediaManager()
    // Pass the intent to the SDK. You can also do this in onCreate().
    if (mediaManager.onNewIntent(intent)) {
        // If the SDK recognizes the intent, you should early return.
        return
    }
    // If the SDK doesn't recognize the intent, you can handle the intent with
    // your own logic.
    ...
  }
}
JavaScript
public class MyActivity extends Activity {
  @Override
  protected void onStart() {
    super.onStart();
    MediaManager mediaManager =
        CastReceiverContext.getInstance().getMediaManager();
    // Pass the intent to the SDK. You can also do this in onCreate().
    if (mediaManager.onNewIntent(getIntent())) {
      // If the SDK recognizes the intent, you should early return.
      return;
    }
    // If the SDK doesn't recognize the intent, you can handle the intent with
    // your own logic.
    ...
  }

  // For some cases, a new load intent triggers onNewIntent() instead of
  // onStart().
  @Override
  protected void onNewIntent(Intent intent) {
    MediaManager mediaManager =
        CastReceiverContext.getInstance().getMediaManager();
    // Pass the intent to the SDK. You can also do this in onCreate().
    if (mediaManager.onNewIntent(intent)) {
      // If the SDK recognizes the intent, you should early return.
      return;
    }
    // If the SDK doesn't recognize the intent, you can handle the intent with
    // your own logic.
    ...
  }
}

إذا رصدت ميزة MediaManager أنّ النية بالشراء، فإنّها تستخرج عنصر MediaLoadRequestData من النية بالشراء واستدعاء MediaLoadCommandCallback.onLoad(). يجب إلغاء هذه الطريقة لمعالجة طلب التحميل. يجب أن يتم تسجيل معاودة الاتصال قبل أن يتم طلب MediaManager.onNewIntent() (من الأفضل استخدام طريقة onCreate() أو نشاط).

كوتلين
class MyActivity : Activity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val mediaManager = CastReceiverContext.getInstance().getMediaManager()
        mediaManager.setMediaLoadCommandCallback(MyMediaLoadCommandCallback())
    }
}

class MyMediaLoadCommandCallback : MediaLoadCommandCallback() {
  override fun onLoad(
        senderId: String?,
        loadRequestData: MediaLoadRequestData
  ): Task {
      return Tasks.call {
        // Resolve the entity into your data structure and load media.
        val mediaInfo = loadRequestData.getMediaInfo()
        if (!checkMediaInfoSupported(mediaInfo)) {
            // Throw MediaException to indicate load failure.
            throw MediaException(
                MediaError.Builder()
                    .setDetailedErrorCode(DetailedErrorCode.LOAD_FAILED)
                    .setReason(MediaError.ERROR_REASON_INVALID_REQUEST)
                    .build()
            )
        }
        myFillMediaInfo(MediaInfoWriter(mediaInfo))
        myPlayerLoad(mediaInfo.getContentUrl())

        // Update media metadata and state (this clears all previous status
        // overrides).
        castReceiverContext.getMediaManager()
            .setDataFromLoad(loadRequestData)
        ...
        castReceiverContext.getMediaManager().broadcastMediaStatus()

        // Return the resolved MediaLoadRequestData to indicate load success.
        return loadRequestData
     }
  }

  private fun myPlayerLoad(contentURL: String) {
    myPlayer.load(contentURL)

    // Update the MediaSession state.
    val playbackState: PlaybackStateCompat = Builder()
        .setState(
            player.getState(), player.getPosition(), System.currentTimeMillis()
        )
        ...
        .build()
    mediaSession.setPlaybackState(playbackState)
  }
JavaScript
public class MyActivity extends Activity {
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);

    MediaManager mediaManager =
        CastReceiverContext.getInstance().getMediaManager();
    mediaManager.setMediaLoadCommandCallback(new MyMediaLoadCommandCallback());
  }
}

public class MyMediaLoadCommandCallback extends MediaLoadCommandCallback {
  @Override
  public Task onLoad(String senderId, MediaLoadRequestData loadRequestData) {
    return Tasks.call(() -> {
        // Resolve the entity into your data structure and load media.
        MediaInfo mediaInfo = loadRequestData.getMediaInfo();
        if (!checkMediaInfoSupported(mediaInfo)) {
          // Throw MediaException to indicate load failure.
          throw new MediaException(
              new MediaError.Builder()
                  .setDetailedErrorCode(DetailedErrorCode.LOAD_FAILED)
                  .setReason(MediaError.ERROR_REASON_INVALID_REQUEST)
                  .build());
        }
        myFillMediaInfo(new MediaInfoWriter(mediaInfo));
        myPlayerLoad(mediaInfo.getContentUrl());

        // Update media metadata and state (this clears all previous status
        // overrides).
        castReceiverContext.getMediaManager()
            .setDataFromLoad(loadRequestData);
        ...
        castReceiverContext.getMediaManager().broadcastMediaStatus();

        // Return the resolved MediaLoadRequestData to indicate load success.
        return loadRequestData;
    });
}

private void myPlayerLoad(String contentURL) {
  myPlayer.load(contentURL);

  // Update the MediaSession state.
  PlaybackStateCompat playbackState =
      new PlaybackStateCompat.Builder()
          .setState(
              player.getState(), player.getPosition(), System.currentTimeMillis())
          ...
          .build();
  mediaSession.setPlaybackState(playbackState);
}

لمعالجة هدف التحميل، يمكنك تحليل الغرض من بنية البيانات التي حدّدناها (MediaLoadRequestData لطلبات التحميل).

طلبات الوسائط المتوافقة

الدعم الأساسي للتحكّم في التشغيل

تتضمن أوامر الدمج الأساسية الأوامر التي تتوافق مع جلسة الوسائط. ويتم إعلام هذه الأوامر من خلال استدعاءات جلسات الوسائط. يجب تسجيل معاودة الاتصال بجلسة الوسائط حتى تتمكّن من إجراء ذلك (قد يكون سبق لك إجراء ذلك).

كوتلين
private class MyMediaSessionCallback : MediaSessionCompat.Callback() {
  override fun onPause() {
    // Pause the player and update the play state.
    myPlayer.pause()
  }

  override fun onPlay() {
    // Resume the player and update the play state.
    myPlayer.play()
  }

  override fun onSeekTo(pos: Long) {
    // Seek and update the play state.
    myPlayer.seekTo(pos)
  }
    ...
 }

mediaSession.setCallback(MyMediaSessionCallback())
JavaScript
private class MyMediaSessionCallback extends MediaSessionCompat.Callback {
  @Override
  public void onPause() {
    // Pause the player and update the play state.
    myPlayer.pause();
  }
  @Override
  public void onPlay() {
    // Resume the player and update the play state.
    myPlayer.play();
  }
  @Override
  public void onSeekTo(long pos) {
    // Seek and update the play state.
    myPlayer.seekTo(pos);
  }

  ...
}

mediaSession.setCallback(new MyMediaSessionCallback());

إتاحة أوامر التحكّم في البث

لا تتوفّر بعض أوامر البث في MediaSession، مثل skipAd() أو setActiveMediaTracks(). ويجب أيضًا تنفيذ بعض أوامر قائمة المحتوى التالي لأن قائمة انتظار البث غير متوافقة بالكامل مع قائمة انتظار MediaSession.

كوتلين
class MyMediaCommandCallback : MediaCommandCallback() {
    override fun onSkipAd(requestData: RequestData?): Task {
        // Skip your ad
        ...
        return Tasks.forResult(null)
    }
}

val mediaManager = CastReceiverContext.getInstance().getMediaManager()
mediaManager.setMediaCommandCallback(MyMediaCommandCallback())
JavaScript
public class MyMediaCommandCallback extends MediaCommandCallback {
  @Override
  public Task onSkipAd(RequestData requestData) {
    // Skip your ad
    ...
    return Tasks.forResult(null);
  }
}

MediaManager mediaManager =
    CastReceiverContext.getInstance().getMediaManager();
mediaManager.setMediaCommandCallback(new MyMediaCommandCallback());

تحديد أوامر الوسائط المتوافقة

كما هو الحال مع جهاز استقبال البث، يجب أن يحدِّد تطبيق Android TV الأوامر المتوافقة، حتى يتمكّن المُرسِلون من تفعيل عناصر تحكّم معيّنة في واجهة المستخدم أو إيقافها. بالنسبة إلى الطلبات التي تشكّل جزءًا من MediaSession، حدِّد الأوامر في PlaybackStateCompat. يجب تحديد الأوامر الإضافية في MediaStatusModifier.

كوتلين
// Set media session supported commands
val playbackState: PlaybackStateCompat = PlaybackStateCompat.Builder()
    .setActions(PlaybackStateCompat.ACTION_PLAY or PlaybackStateCompat.ACTION_PAUSE)
    .setState(PlaybackStateCompat.STATE_PLAYING)
    .build()

mediaSession.setPlaybackState(playbackState)

// Set additional commands in MediaStatusModifier
val mediaManager = CastReceiverContext.getInstance().getMediaManager()
mediaManager.getMediaStatusModifier()
    .setMediaCommandSupported(MediaStatus.COMMAND_QUEUE_NEXT)
JavaScript
// Set media session supported commands
PlaybackStateCompat playbackState =
    new PlaybackStateCompat.Builder()
        .setActions(PlaybackStateCompat.ACTION_PLAY | PlaybackStateCompat.ACTION_PAUSE)
        .setState(PlaybackStateCompat.STATE_PLAYING)
        .build();

mediaSession.setPlaybackState(playbackState);

// Set additional commands in MediaStatusModifier
MediaManager mediaManager = CastReceiverContext.getInstance().getMediaManager();
mediaManager.getMediaStatusModifier()
            .setMediaCommandSupported(MediaStatus.COMMAND_QUEUE_NEXT);

إخفاء الأزرار غير المتوافقة

إذا كان تطبيق Android TV لا يتيح سوى التحكّم في الوسائط الأساسية ولكن تطبيق استقبال الويب يتيح استخدام تحكّم أكثر تقدّمًا، عليك التأكّد من أنّ تطبيق المُرسِل يتصرف بشكل صحيح عند البث إلى تطبيق Android TV. على سبيل المثال، إذا كان تطبيق Android TV لا يتيح تغيير معدّل التشغيل أثناء تفعيل تطبيق مستقبِل الويب، عليك ضبط الإجراءات المتوافقة بشكل صحيح على كل منصّة والتأكّد من عرض تطبيق المُرسِل بشكل صحيح.

تعديل MediaStatus

لإتاحة الميزات المتقدّمة، مثل المقاطع الصوتية والإعلانات وأحداث البث المباشر وقائمة المحتوى التالي، يجب أن يقدّم تطبيق Android TV معلومات إضافية لا يمكن التأكّد منها من خلال MediaSession.

ونقدّم لك الفئة MediaStatusModifier لتحقيق ذلك. ستعمل MediaStatusModifier دائمًا على MediaSession التي ضبطتها في CastReceiverContext.

لإنشاء بث مباشر MediaStatus:

كوتلين
val mediaManager: MediaManager = castReceiverContext.getMediaManager()
val statusModifier: MediaStatusModifier = mediaManager.getMediaStatusModifier()

statusModifier
    .setLiveSeekableRange(seekableRange)
    .setAdBreakStatus(adBreakStatus)
    .setCustomData(customData)

mediaManager.broadcastMediaStatus()
JavaScript
MediaManager mediaManager = castReceiverContext.getMediaManager();
MediaStatusModifier statusModifier = mediaManager.getMediaStatusModifier();

statusModifier
    .setLiveSeekableRange(seekableRange)
    .setAdBreakStatus(adBreakStatus)
    .setCustomData(customData);

mediaManager.broadcastMediaStatus();

ستحصل مكتبة برامجنا على القاعدة MediaStatus الأساسية من MediaSession، ويمكن لتطبيق Android TV تحديد الحالة الإضافية وإلغاء الحالة من خلال معدِّل MediaStatus.

يمكن ضبط بعض الحالات والبيانات الوصفية في MediaSession وMediaStatusModifier. ننصحك بشدّة بضبطها فقط في MediaSession. لا يزال بإمكانك استخدام المُعدِّل لإلغاء الولايات في MediaSession، وننصح بعدم تطبيق الحالة في المُعدِّل دائمًا على أولوية أعلى من القيم التي يقدمها MediaSession.

الاعتراض على MediaStatus قبل إرسال الرسائل

وكما هو الحال مع حزمة تطوير برامج مستقبِل الويب، إذا كنت تريد إجراء بعض اللمسات الأخيرة قبل الإرسال، يمكنك تحديد MediaStatusInterceptor لمعالجة MediaStatus الإرسال المطلوب. نمرّر الرمز MediaStatusWriter للتلاعب بـ MediaStatus قبل إرساله.

كوتلين
mediaManager.setMediaStatusInterceptor(object : MediaStatusInterceptor {
    override fun intercept(mediaStatusWriter: MediaStatusWriter) {
      // Perform customization.
        mediaStatusWriter.setCustomData(JSONObject("{data: \"my Hello\"}"))
    }
})
JavaScript
mediaManager.setMediaStatusInterceptor(new MediaStatusInterceptor() {
    @Override
    public void intercept(MediaStatusWriter mediaStatusWriter) {
        // Perform customization.
        mediaStatusWriter.setCustomData(new JSONObject("{data: \"my Hello\"}"));
    }
});

التعامل مع بيانات اعتماد المستخدم

قد يسمح تطبيق Android TV لمستخدمين محدَّدين فقط ببدء جلسة التطبيق أو الانضمام إليها. على سبيل المثال، لا تسمح لمُرسِل الإطلاق أو الانضمام إلا في الحالات التالية:

  • يتم تسجيل دخول تطبيق المُرسِل إلى الحساب والملف الشخصي على تطبيق ATV نفسه.
  • تم تسجيل دخول تطبيق المُرسِل إلى الحساب نفسه، ولكن تم تسجيل ملف شخصي مختلف كتطبيق ATV.

إذا كان تطبيقك يمكنه التعامل مع مستخدمين متعددين أو مجهولين، يمكنك السماح لأي مستخدم إضافي بالانضمام إلى جلسة ATV. إذا قدّم المستخدم بيانات الاعتماد، يجب أن يتعامل تطبيق ATV مع بيانات الاعتماد لكي يتم تتبُّع مستوى تقدُّمه وبيانات المستخدم الأخرى بشكل صحيح.

عند تشغيل تطبيق المُرسِل أو الانضمام إلى تطبيق Android TV، يجب أن يوفّر تطبيق المُرسِل بيانات الاعتماد التي تمثّل مَن ينضم إلى الجلسة.

قبل تشغيل المُرسِل لتطبيق Android TV والانضمام إليه، يمكنك تحديد أداة فحص الإطلاق لمعرفة ما إذا كان بيانات اعتماد المُرسِل مسموحًا بها أم لا. وإذا لم يكن الأمر كذلك، ستعود حزمة SDK Cast Connect إلى إطلاق جهاز استقبال الويب.

بيانات اعتماد إطلاق تطبيق المُرسِل

من جانب المُرسِل، يمكنك تحديد CredentialsData لتمثيل المستخدم الذي ينضم إلى الجلسة.

credentials هو سلسلة يمكن تحديدها بواسطة المستخدم، شرط أن يتمكن تطبيق ATV من فهمها. تحدّد السمة credentialsType المنصة التي تأتي منها CredentialsData أو يمكن أن تكون قيمة مخصّصة. يتم ضبطها تلقائيًا على النظام الأساسي الذي يتم إرساله منه.

يتم تمرير CredentialsData إلى تطبيق Android TV فقط خلال وقت الإطلاق أو الانضمام. إذا ضبطته مرة أخرى أثناء الربط، لن يتم تمريره إلى تطبيق Android TV. إذا غيّر المُرسِل الملف الشخصي أثناء الاتصال، يمكنك مواصلة البقاء في الجلسة أو الاتصال SessionManager.endCurrentCastSession(boolean stopCasting) إذا كنت تعتقد أن الملف الشخصي الجديد غير متوافق مع الجلسة.

يمكن استرداد CredentialsData لكل مُرسِل باستخدام getSenders على CastReceiverContext للحصول على SenderInfo، getCastLaunchRequest() للحصول على CastLaunchRequest، ثم getCredentialsData().

نظام التشغيل Android

ويجب أن يتوفّر الإصدار play-services-cast-framework من نظام التشغيل 19.0.0 أو إصدار أحدث.

كوتلين
CastContext.getSharedInstance().setLaunchCredentialsData(
    CredentialsData.Builder()
        .setCredentials("{\"userId\": \"abc\"}")
        .build()
)
JavaScript
CastContext.getSharedInstance().setLaunchCredentialsData(
    new CredentialsData.Builder()
        .setCredentials("{\"userId\": \"abc\"}")
        .build());
iOS

ويجب استخدام الإصدار v4.8.0 من google-cast-sdk أو إصدار أحدث.

يمكن الاتصال في أي وقت بعد ضبط الخيارات: GCKCastContext.setSharedInstanceWith(options).

GCKCastContext.sharedInstance().setLaunch(
    GCKCredentialsData(credentials: "{\"userId\": \"abc\"}")
الويب

يتطلب إصدار متصفِّح Chromium M87 أو إصدارًا أحدث.

يمكن الاتصال في أي وقت بعد ضبط الخيارات: cast.framework.CastContext.getInstance().setOptions(options);.

let credentialsData =
    new chrome.cast.CredentialsData("{\"userId\": \"abc\"}");
cast.framework.CastContext.getInstance().setLaunchCredentialsData(credentialsData);

تنفيذ أداة التحقّق من طلبات إطلاق منتجات ATV

يتم تمرير CredentialsData إلى تطبيق Android TV عندما يحاول أحد المُرسِلين تشغيل التطبيق أو الانضمام إليه. يمكنك تنفيذ LaunchRequestChecker. للسماح بهذا الطلب أو رفضه.

في حال رفض طلب، يتم تحميل مستقبِل الويب بدلاً من تشغيله في الأصل على تطبيق ATV. وعليك رفض طلب إذا لم تتمكّن خدمة ATV من التعامل مع المستخدم الذي يطلب إطلاق التطبيق أو الانضمام إليه. ومن الأمثلة على ذلك أنّ مستخدمًا قد سجّل الدخول إلى تطبيق ATV مختلفًا عن الإصدار الذي يطلبه ويتعذّر على تطبيقك التعامل مع بيانات الاعتماد والتبديل التي يستخدمها المستخدم، أو أنّه لم يتم تسجيل دخول مستخدم حاليًا إلى تطبيق ATV.

في حال السماح بطلب، يتم تشغيل تطبيق ATV. يمكنك تخصيص هذا السلوك استنادًا إلى ما إذا كان تطبيقك يتيح إرسال طلبات التحميل عندما لا يكون المستخدم مسجّلاً الدخول إلى تطبيق ATV أو إذا كان هناك مستخدم غير مطابق. ويمكن البحث عن هذا السلوك بشكل كامل في LaunchRequestChecker.

أنشئ صفًا يؤدي إلى تنفيذ واجهة CastReceiverOptions.LaunchRequestChecker:

كوتلين
class MyLaunchRequestChecker : LaunchRequestChecker {
  override fun checkLaunchRequestSupported(launchRequest: CastLaunchRequest): Task {
    return Tasks.call {
      myCheckLaunchRequest(
           launchRequest
      )
    }
  }
}

private fun myCheckLaunchRequest(launchRequest: CastLaunchRequest): Boolean {
  val credentialsData = launchRequest.getCredentialsData()
     ?: return false // or true if you allow anonymous users to join.

  // The request comes from a mobile device, e.g. checking user match.
  return if (credentialsData.credentialsType == CredentialsData.CREDENTIALS_TYPE_ANDROID) {
     myCheckMobileCredentialsAllowed(credentialsData.getCredentials())
  } else false // Unrecognized credentials type.
}
JavaScript
public class MyLaunchRequestChecker
    implements CastReceiverOptions.LaunchRequestChecker {
  @Override
  public Task checkLaunchRequestSupported(CastLaunchRequest launchRequest) {
    return Tasks.call(() -> myCheckLaunchRequest(launchRequest));
  }
}

private boolean myCheckLaunchRequest(CastLaunchRequest launchRequest) {
  CredentialsData credentialsData = launchRequest.getCredentialsData();
  if (credentialsData == null) {
    return false;  // or true if you allow anonymous users to join.
  }

  // The request comes from a mobile device, e.g. checking user match.
  if (credentialsData.getCredentialsType().equals(CredentialsData.CREDENTIALS_TYPE_ANDROID)) {
    return myCheckMobileCredentialsAllowed(credentialsData.getCredentials());
  }

  // Unrecognized credentials type.
  return false;
}

بعد ذلك، اضبطها في ReceiverOptionsProvider:

كوتلين
class MyReceiverOptionsProvider : ReceiverOptionsProvider {
  override fun getOptions(context: Context?): CastReceiverOptions {
    return CastReceiverOptions.Builder(context)
        ...
        .setLaunchRequestChecker(MyLaunchRequestChecker())
        .build()
  }
}
JavaScript
public class MyReceiverOptionsProvider implements ReceiverOptionsProvider {
  @Override
  public CastReceiverOptions getOptions(Context context) {
    return new CastReceiverOptions.Builder(context)
        ...
        .setLaunchRequestChecker(new MyLaunchRequestChecker())
        .build();
  }
}

يؤدي حلّ true في LaunchRequestChecker إلى إطلاق تطبيق ATV وإطلاق false تطبيق مستقبِل الويب.

إرسال رسائل مخصصة واستلامها

يسمح لك بروتوكول البث بإرسال رسائل سلاسل مخصّصة بين المُرسِلين وتطبيق المستلِم. يجب تسجيل مساحة اسم (قناة) لإرسال رسائل قبل إعداد CastReceiverContext.

Android TV: تحديد مساحة اسم مخصّصة

يجب تحديد مساحات الأسماء المتوافقة في CastReceiverOptions أثناء الإعداد:

كوتلين
class MyReceiverOptionsProvider : ReceiverOptionsProvider {
  override fun getOptions(context: Context?): CastReceiverOptions {
    return CastReceiverOptions.Builder(context)
        .setCustomNamespaces(
            Arrays.asList("urn:x-cast:com.example.cast.mynamespace")
        )
        .build()
  }
}
JavaScript
public class MyReceiverOptionsProvider implements ReceiverOptionsProvider {
  @Override
  public CastReceiverOptions getOptions(Context context) {
    return new CastReceiverOptions.Builder(context)
        .setCustomNamespaces(
              Arrays.asList("urn:x-cast:com.example.cast.mynamespace"))
        .build();
  }
}

Android TV: إرسال الرسائل

كوتلين
// If senderId is null, then the message is broadcasted to all senders.
CastReceiverContext.getInstance().sendMessage(
    "urn:x-cast:com.example.cast.mynamespace", senderId, customString)
JavaScript
// If senderId is null, then the message is broadcasted to all senders.
CastReceiverContext.getInstance().sendMessage(
    "urn:x-cast:com.example.cast.mynamespace", senderId, customString);

Android TV: استلام رسائل مساحة الاسم المخصّصة

كوتلين
class MyCustomMessageListener : MessageReceivedListener {
    override fun onMessageReceived(
        namespace: String, senderId: String?, message: String ) {
        ...
    }
}

CastReceiverContext.getInstance().setMessageReceivedListener(
    "urn:x-cast:com.example.cast.mynamespace", new MyCustomMessageListener());
JavaScript
class MyCustomMessageListener implements CastReceiverContext.MessageReceivedListener {
  @Override
  public void onMessageReceived(
      String namespace, String senderId, String message) {
    ...
  }
}

CastReceiverContext.getInstance().setMessageReceivedListener(
    "urn:x-cast:com.example.cast.mynamespace", new MyCustomMessageListener());