MediasessionService के साथ बैकग्राउंड में वीडियो चलाने की सुविधा

अक्सर, किसी ऐप्लिकेशन के फ़ोरग्राउंड में न होने पर भी मीडिया चलाना होता है. उदाहरण के लिए, जब कोई उपयोगकर्ता अपने डिवाइस को लॉक करता है या किसी दूसरे ऐप्लिकेशन का इस्तेमाल करता है, तब भी संगीत प्लेयर आम तौर पर संगीत चलाता रहता है. Media3 लाइब्रेरी, इंटरफ़ेस की एक सीरीज़ उपलब्ध कराती है. इसकी मदद से, बैकग्राउंड में वीडियो चलाने की सुविधा दी जा सकती है.

MediaSessionService का इस्तेमाल करना

बैकग्राउंड में वीडियो चलाने की सुविधा चालू करने के लिए, आपको Player और MediaSession को अलग सेवा में शामिल करना होगा. इससे, डिवाइस तब भी मीडिया दिखाना जारी रखता है, जब आपका ऐप्लिकेशन स्क्रीन पर न हो.

MediaSessionService की मदद से, मीडिया सेशन को ऐप्लिकेशन की गतिविधि से अलग चलाया जा सकता है
पहली इमेज: MediaSessionService की मदद से, मीडिया सेशन को ऐप्लिकेशन की गतिविधि से अलग चलाया जा सकता है

किसी सेवा में प्लेयर को होस्ट करते समय, आपको MediaSessionService का इस्तेमाल करना चाहिए. ऐसा करने के लिए, MediaSessionService को एक्सटेंड करने वाली क्लास बनाएं और उसमें अपना मीडिया सेशन बनाएं.

MediaSessionService का इस्तेमाल करने से, Google Assistant, सिस्टम मीडिया कंट्रोल या Wear OS जैसे साथी डिवाइसों जैसे बाहरी क्लाइंट, आपकी सेवा को खोज सकते हैं, उससे कनेक्ट कर सकते हैं, और वीडियो चलाने की सुविधा को कंट्रोल कर सकते हैं. इसके लिए, उन्हें आपके ऐप्लिकेशन के यूज़र इंटरफ़ेस (यूआई) की गतिविधि को ऐक्सेस करने की ज़रूरत नहीं होती. असल में, एक ही समय में एक ही MediaSessionService से कनेक्ट किए गए कई क्लाइंट ऐप्लिकेशन हो सकते हैं. साथ ही, हर ऐप्लिकेशन का अपना MediaController हो सकता है.

सेवा की लाइफ़साइकल लागू करना

आपको अपनी सेवा के लाइफ़साइकल के तीन तरीके लागू करने होंगे:

  • onCreate() को तब कॉल किया जाता है, जब पहला कंट्रोलर कनेक्ट होने वाला होता है और सेवा को इंस्टैंशिएट किया जाता है और शुरू किया जाता है. Player और MediaSession बनाने के लिए, यह सबसे अच्छी जगह है.
  • onTaskRemoved(Intent) को तब कॉल किया जाता है, जब उपयोगकर्ता हाल ही के टास्क से ऐप्लिकेशन को हटा देता है. अगर वीडियो चल रहा है, तो ऐप्लिकेशन के पास सेवा को फ़ोरग्राउंड में चलाए रखने का विकल्प होता है. अगर प्लेयर को रोका गया है, तो इसका मतलब है कि सेवा फ़ोरग्राउंड में नहीं है और उसे रोकना होगा.
  • onDestroy() को तब कॉल किया जाता है, जब सेवा बंद की जा रही हो. प्लेयर और सेशन के साथ-साथ सभी संसाधनों को रिलीज़ करना ज़रूरी है.

Kotlin

class PlaybackService : MediaSessionService() {
  private var mediaSession: MediaSession? = null

  // Create your player and media session in the onCreate lifecycle event
  override fun onCreate() {
    super.onCreate()
    val player = ExoPlayer.Builder(this).build()
    mediaSession = MediaSession.Builder(this, player).build()
  }

  // The user dismissed the app from the recent tasks
  override fun onTaskRemoved(rootIntent: Intent?) {
    val player = mediaSession?.player!!
    if (!player.playWhenReady
        || player.mediaItemCount == 0
        || player.playbackState == Player.STATE_ENDED) {
      // Stop the service if not playing, continue playing in the background
      // otherwise.
      stopSelf()
    }
  }

  // Remember to release the player and media session in onDestroy
  override fun onDestroy() {
    mediaSession?.run {
      player.release()
      release()
      mediaSession = null
    }
    super.onDestroy()
  }
}

Java

public class PlaybackService extends MediaSessionService {
  private MediaSession mediaSession = null;

  // Create your Player and MediaSession in the onCreate lifecycle event
  @Override
  public void onCreate() {
    super.onCreate();
    ExoPlayer player = new ExoPlayer.Builder(this).build();
    mediaSession = new MediaSession.Builder(this, player).build();
  }

  // The user dismissed the app from the recent tasks
  @Override
  public void onTaskRemoved(@Nullable Intent rootIntent) {
    Player player = mediaSession.getPlayer();
    if (!player.getPlayWhenReady()
        || player.getMediaItemCount() == 0
        || player.getPlaybackState() == Player.STATE_ENDED) {
      // Stop the service if not playing, continue playing in the background
      // otherwise.
      stopSelf();
    }
  }

  // Remember to release the player and media session in onDestroy
  @Override
  public void onDestroy() {
    mediaSession.getPlayer().release();
    mediaSession.release();
    mediaSession = null;
    super.onDestroy();
  }
}

बैकग्राउंड में वीडियो चलाने के विकल्प के तौर पर, ऐप्लिकेशन किसी भी स्थिति में सेवा को बंद कर सकता है. ऐसा तब होता है, जब उपयोगकर्ता ऐप्लिकेशन को बंद कर देता है:

Kotlin

override fun onTaskRemoved(rootIntent: Intent?) {
  val player = mediaSession.player
  if (player.playWhenReady) {
    // Make sure the service is not in foreground.
    player.pause()
  }
  stopSelf()
}

Java

@Override
public void onTaskRemoved(@Nullable Intent rootIntent) {
  Player player = mediaSession.getPlayer();
  if (player.getPlayWhenReady()) {
    // Make sure the service is not in foreground.
    player.pause();
  }
  stopSelf();
}

मीडिया सेशन का ऐक्सेस देना

onGetSession() तरीके को बदलकर, दूसरे क्लाइंट को उस मीडिया सेशन का ऐक्सेस दें जो सेवा बनाने के दौरान बनाया गया था.

Kotlin

class PlaybackService : MediaSessionService() {
  private var mediaSession: MediaSession? = null
  // [...] lifecycle methods omitted

  override fun onGetSession(controllerInfo: MediaSession.ControllerInfo): MediaSession? =
    mediaSession
}

Java

public class PlaybackService extends MediaSessionService {
  private MediaSession mediaSession = null;
  // [...] lifecycle methods omitted

  @Override
  public MediaSession onGetSession(MediaSession.ControllerInfo controllerInfo) {
    return mediaSession;
  }
}

मेनिफ़ेस्ट में सेवा की जानकारी देना

किसी ऐप्लिकेशन को फ़ोरग्राउंड सेवा चलाने के लिए अनुमति की ज़रूरत होती है. मेनिफ़ेस्ट में FOREGROUND_SERVICE अनुमति जोड़ें. अगर आपने एपीआई लेवल 34 और उसके बाद के वर्शन को टारगेट किया है, तो FOREGROUND_SERVICE_MEDIA_PLAYBACK भी जोड़ें:

<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PLAYBACK" />

आपको मेनिफ़ेस्ट में Service क्लास की जानकारी भी देनी होगी. इसके लिए, MediaSessionService के इंटेंट फ़िल्टर का इस्तेमाल करें.

<service
    android:name=".PlaybackService"
    android:foregroundServiceType="mediaPlayback"
    android:exported="true">
    <intent-filter>
        <action android:name="androidx.media3.session.MediaSessionService"/>
    </intent-filter>
</service>

अगर आपका ऐप्लिकेशन Android 10 (एपीआई लेवल 29) और उसके बाद के वर्शन वाले डिवाइस पर चल रहा है, तो आपको ऐसा foregroundServiceType तय करना होगा जिसमें mediaPlayback शामिल हो.

MediaController का इस्तेमाल करके प्लेबैक कंट्रोल करना

अपने प्लेयर के यूज़र इंटरफ़ेस (यूआई) वाली गतिविधि या फ़्रैगमेंट में, MediaController का इस्तेमाल करके यूज़र इंटरफ़ेस और मीडिया सेशन के बीच लिंक बनाया जा सकता है. आपका यूज़र इंटरफ़ेस (यूआई), मीडिया कंट्रोलर का इस्तेमाल करके, सेशन के दौरान यूआई से प्लेयर को निर्देश भेजता है. MediaController बनाने और उसका इस्तेमाल करने के बारे में जानने के लिए, MediaController बनाएं गाइड देखें.

यूज़र इंटरफ़ेस (यूआई) के निर्देशों को मैनेज करना

MediaSession को अपने MediaSession.Callback के ज़रिए, कंट्रोलर से निर्देश मिलते हैं. MediaSession को शुरू करने पर, MediaSession.Callback डिफ़ॉल्ट रूप से लागू हो जाता है. यह उन सभी निर्देशों को अपने-आप मैनेज करता है जिन्हें MediaController आपके प्लेयर को भेजता है.

सूचना

MediaSessionService आपके लिए अपने-आप एक MediaNotification बना देता है, जो ज़्यादातर मामलों में काम करता है. डिफ़ॉल्ट रूप से, पब्लिश की गई सूचना एक MediaStyle सूचना होती है. यह आपके मीडिया सेशन की नई जानकारी के साथ अपडेट रहती है और प्लेबैक कंट्रोल दिखाती है. MediaNotification को आपके सेशन की जानकारी होती है. साथ ही, इसका इस्तेमाल उसी सेशन से कनेक्ट किए गए किसी भी दूसरे ऐप्लिकेशन के लिए, वीडियो चलाने या रोकने के लिए किया जा सकता है.

उदाहरण के लिए, MediaSessionService का इस्तेमाल करने वाला संगीत स्ट्रीमिंग ऐप्लिकेशन, एक ऐसा MediaNotification बनाता है जिसमें आपके MediaSession कॉन्फ़िगरेशन के आधार पर, प्लेबैक कंट्रोल के साथ-साथ, चल रहे मौजूदा मीडिया आइटम का टाइटल, कलाकार, और एल्बम आर्ट दिखाया जाता है.

ज़रूरी मेटाडेटा, मीडिया में दिया जा सकता है या मीडिया आइटम के हिस्से के तौर पर एलान किया जा सकता है. जैसे, यहां दिए गए स्निपेट में:

Kotlin

val mediaItem =
    MediaItem.Builder()
      .setMediaId("media-1")
      .setUri(mediaUri)
      .setMediaMetadata(
        MediaMetadata.Builder()
          .setArtist("David Bowie")
          .setTitle("Heroes")
          .setArtworkUri(artworkUri)
          .build()
      )
      .build()

mediaController.setMediaItem(mediaItem)
mediaController.prepare()
mediaController.play()

Java

MediaItem mediaItem =
    new MediaItem.Builder()
        .setMediaId("media-1")
        .setUri(mediaUri)
        .setMediaMetadata(
            new MediaMetadata.Builder()
                .setArtist("David Bowie")
                .setTitle("Heroes")
                .setArtworkUri(artworkUri)
                .build())
        .build();

mediaController.setMediaItem(mediaItem);
mediaController.prepare();
mediaController.play();

ऐप्लिकेशन, Android मीडिया कंट्रोल के कमांड बटन को पसंद के मुताबिक बना सकते हैं. Android के मीडिया कंट्रोल को पसंद के मुताबिक बनाने के बारे में ज़्यादा पढ़ें.

सूचनाएं पसंद के मुताबिक बनाना

सूचना को पसंद के मुताबिक बनाने के लिए, DefaultMediaNotificationProvider.Builder के साथ MediaNotification.Provider बनाएं या सेवा देने वाली कंपनी के इंटरफ़ेस को पसंद के मुताबिक बनाएं. setMediaNotificationProvider का इस्तेमाल करके, सेवा देने वाली कंपनी को अपने MediaSessionService में जोड़ें.

रोके गए वीडियो को फिर से चलाना

मीडिया बटन, Android डिवाइसों और दूसरे डिवाइसों पर मौजूद हार्डवेयर बटन होते हैं. जैसे, ब्लूटूथ हेडसेट पर मौजूद चलाएं या रोकें बटन. Media3, सेवा के चालू होने पर मीडिया बटन के इनपुट को मैनेज करता है.

Media3 मीडिया बटन रिसीवर के बारे में जानकारी देना

Media3 में एक एपीआई शामिल है, जिसकी मदद से उपयोगकर्ता किसी ऐप्लिकेशन के बंद होने के बाद भी वीडियो चलाना जारी रख सकते हैं. यह सुविधा, डिवाइस के फिर से चालू होने के बाद भी काम करती है. डिफ़ॉल्ट रूप से, गाने का प्लेबैक फिर से शुरू करने की सुविधा बंद रहती है. इसका मतलब है कि जब आपकी सेवा चालू नहीं होती, तो उपयोगकर्ता गाने का प्लेबैक फिर से शुरू नहीं कर सकता. ऑप्ट-इन करने के लिए, अपने मेनिफ़ेस्ट में MediaButtonReceiver का एलान करें:

<receiver android:name="androidx.media3.session.MediaButtonReceiver"
  android:exported="true">
  <intent-filter>
    <action android:name="android.intent.action.MEDIA_BUTTON" />
  </intent-filter>
</receiver>

गाने के प्लेबैक को फिर से शुरू करने का कॉलबैक लागू करना

जब ब्लूटूथ डिवाइस या Android सिस्टम यूज़र इंटरफ़ेस (यूआई) की वीडियो फिर से चलाने की सुविधा से, वीडियो फिर से चलाने का अनुरोध किया जाता है, तो onPlaybackResumption() कॉलबैक का तरीका इस्तेमाल किया जाता है.

Kotlin

override fun onPlaybackResumption(
    mediaSession: MediaSession,
    controller: ControllerInfo
): ListenableFuture<MediaItemsWithStartPosition> {
  val settable = SettableFuture.create<MediaItemsWithStartPosition>()
  scope.launch {
    // Your app is responsible for storing the playlist and the start position
    // to use here
    val resumptionPlaylist = restorePlaylist()
    settable.set(resumptionPlaylist)
  }
  return settable
}

Java

@Override
public ListenableFuture<MediaItemsWithStartPosition> onPlaybackResumption(
    MediaSession mediaSession,
    ControllerInfo controller
) {
  SettableFuture<MediaItemsWithStartPosition> settableFuture = SettableFuture.create();
  settableFuture.addListener(() -> {
    // Your app is responsible for storing the playlist and the start position
    // to use here
    MediaItemsWithStartPosition resumptionPlaylist = restorePlaylist();
    settableFuture.set(resumptionPlaylist);
  }, MoreExecutors.directExecutor());
  return settableFuture;
}

अगर आपने वीडियो चलाने की रफ़्तार, दोहराने का मोड या गाने को शफ़ल करने का मोड जैसे अन्य पैरामीटर सेव किए हैं, तो Media3 के प्लेयर को तैयार करने और कॉलबैक पूरा होने पर वीडियो चलाने से पहले, onPlaybackResumption() में इन पैरामीटर के साथ प्लेयर को कॉन्फ़िगर किया जा सकता है.

बेहतर कंट्रोलर कॉन्फ़िगरेशन और पुराने सिस्टम के साथ काम करने की सुविधा

आम तौर पर, ऐप्लिकेशन के यूज़र इंटरफ़ेस (यूआई) में MediaController का इस्तेमाल, वीडियो चलाने और प्लेलिस्ट दिखाने के लिए किया जाता है. साथ ही, यह सेशन बाहरी क्लाइंट को भी दिखता है. जैसे, मोबाइल या टीवी पर Android मीडिया कंट्रोल और Assistant, स्मार्टवॉच के लिए Wear OS, और कार में Android Auto. Media3 सेशन डेमो ऐप्लिकेशन, ऐसे ऐप्लिकेशन का उदाहरण है जो इस तरह की स्थिति को लागू करता है.

ये बाहरी क्लाइंट, लेगसी AndroidX लाइब्रेरी के MediaControllerCompat या Android फ़्रेमवर्क के android.media.session.MediaController जैसे एपीआई का इस्तेमाल कर सकते हैं. Media3, लेगसी लाइब्रेरी के साथ पूरी तरह से काम करता है. साथ ही, यह Android फ़्रेमवर्क एपीआई के साथ इंटरऑपरेबिलिटी की सुविधा देता है.

मीडिया कंट्रोल करने से जुड़ी सूचना का इस्तेमाल करना

यह समझना ज़रूरी है कि ये लेगसी या फ़्रेमवर्क कंट्रोलर, फ़्रेमवर्क PlaybackState.getActions() और PlaybackState.getCustomActions() से एक ही वैल्यू पढ़ते हैं. फ़्रेमवर्क सेशन की कार्रवाइयों और कस्टम कार्रवाइयों का पता लगाने के लिए, कोई ऐप्लिकेशन मीडिया सूचना कंट्रोलर का इस्तेमाल कर सकता है. साथ ही, इसके उपलब्ध निर्देशों और कस्टम लेआउट को सेट कर सकता है. यह सेवा, मीडिया सूचना कंट्रोलर को आपके सेशन से कनेक्ट करती है. साथ ही, फ़्रेमवर्क सेशन की कार्रवाइयों और कस्टम कार्रवाइयों को कॉन्फ़िगर करने के लिए, आपके कॉलबैक के onConnect() से मिले ConnectionResult का इस्तेमाल करती है.

सिर्फ़ मोबाइल पर काम करने वाले ऐप्लिकेशन के लिए, MediaSession.Callback.onConnect() को लागू किया जा सकता है. इससे, फ़्रेमवर्क सेशन के लिए उपलब्ध निर्देश और पसंद के मुताबिक लेआउट सेट किए जा सकते हैं. इसके लिए, यह तरीका अपनाएं:

Kotlin

override fun onConnect(
  session: MediaSession,
  controller: MediaSession.ControllerInfo
): ConnectionResult {
  if (session.isMediaNotificationController(controller)) {
    val sessionCommands =
      ConnectionResult.DEFAULT_SESSION_COMMANDS.buildUpon()
        .add(customCommandSeekBackward)
        .add(customCommandSeekForward)
        .build()
    val playerCommands =
      ConnectionResult.DEFAULT_PLAYER_COMMANDS.buildUpon()
        .remove(COMMAND_SEEK_TO_PREVIOUS)
        .remove(COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM)
        .remove(COMMAND_SEEK_TO_NEXT)
        .remove(COMMAND_SEEK_TO_NEXT_MEDIA_ITEM)
        .build()
    // Custom layout and available commands to configure the legacy/framework session.
    return AcceptedResultBuilder(session)
      .setCustomLayout(
        ImmutableList.of(
          createSeekBackwardButton(customCommandSeekBackward),
          createSeekForwardButton(customCommandSeekForward))
      )
      .setAvailablePlayerCommands(playerCommands)
      .setAvailableSessionCommands(sessionCommands)
      .build()
  }
  // Default commands with default custom layout for all other controllers.
  return AcceptedResultBuilder(session).build()
}

Java

@Override
public ConnectionResult onConnect(
    MediaSession session, MediaSession.ControllerInfo controller) {
  if (session.isMediaNotificationController(controller)) {
    SessionCommands sessionCommands =
        ConnectionResult.DEFAULT_SESSION_COMMANDS
            .buildUpon()
            .add(customCommandSeekBackward)
            .add(customCommandSeekForward)
            .build();
    Player.Commands playerCommands =
        ConnectionResult.DEFAULT_PLAYER_COMMANDS
            .buildUpon()
            .remove(COMMAND_SEEK_TO_PREVIOUS)
            .remove(COMMAND_SEEK_TO_PREVIOUS_MEDIA_ITEM)
            .remove(COMMAND_SEEK_TO_NEXT)
            .remove(COMMAND_SEEK_TO_NEXT_MEDIA_ITEM)
            .build();
    // Custom layout and available commands to configure the legacy/framework session.
    return new AcceptedResultBuilder(session)
        .setCustomLayout(
            ImmutableList.of(
                createSeekBackwardButton(customCommandSeekBackward),
                createSeekForwardButton(customCommandSeekForward)))
        .setAvailablePlayerCommands(playerCommands)
        .setAvailableSessionCommands(sessionCommands)
        .build();
  }
  // Default commands without default custom layout for all other controllers.
  return new AcceptedResultBuilder(session).build();
}

Android Auto को कस्टम निर्देश भेजने की अनुमति देना

MediaLibraryService का इस्तेमाल करते समय और मोबाइल ऐप्लिकेशन के साथ Android Auto के साथ काम करने के लिए, Android Auto कंट्रोलर के लिए सही निर्देश उपलब्ध होने चाहिए. ऐसा न होने पर, Media3 उस कंट्रोलर से आने वाले कस्टम निर्देशों को स्वीकार नहीं करेगा:

Kotlin

override fun onConnect(
  session: MediaSession,
  controller: MediaSession.ControllerInfo
): ConnectionResult {
  val sessionCommands =
    ConnectionResult.DEFAULT_SESSION_AND_LIBRARY_COMMANDS.buildUpon()
      .add(customCommandSeekBackward)
      .add(customCommandSeekForward)
      .build()
  if (session.isMediaNotificationController(controller)) {
    // [...] See above.
  } else if (session.isAutoCompanionController(controller)) {
    // Available session commands to accept incoming custom commands from Auto.
    return AcceptedResultBuilder(session)
      .setAvailableSessionCommands(sessionCommands)
      .build()
  }
  // Default commands with default custom layout for all other controllers.
  return AcceptedResultBuilder(session).build()
}

Java

@Override
public ConnectionResult onConnect(
    MediaSession session, MediaSession.ControllerInfo controller) {
  SessionCommands sessionCommands =
      ConnectionResult.DEFAULT_SESSION_COMMANDS
          .buildUpon()
          .add(customCommandSeekBackward)
          .add(customCommandSeekForward)
          .build();
  if (session.isMediaNotificationController(controller)) {
    // [...] See above.
  } else if (session.isAutoCompanionController(controller)) {
    // Available commands to accept incoming custom commands from Auto.
    return new AcceptedResultBuilder(session)
        .setAvailableSessionCommands(sessionCommands)
        .build();
  }
  // Default commands without default custom layout for all other controllers.
  return new AcceptedResultBuilder(session).build();
}

सेशन के डेमो ऐप्लिकेशन में एक ऑटोमोटिव मॉड्यूल है. इससे, Automotive OS के साथ काम करने की सुविधा के बारे में पता चलता है. इसके लिए, एक अलग APK की ज़रूरत होती है.