Android के लिए क्लाउड मीडिया प्रोवाइडर बनाएं

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

शुरू करने से पहले

क्लाउड मीडिया की सेवा देने वाली कंपनी बनाने से पहले, इन बातों का ध्यान रखें.

ज़रूरी शर्तें

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

Android, ज़रूरी शर्तें पूरी करने वाली सभी क्लाउड मीडिया की सेवा देने वाली कंपनियों की सूची, सर्वर पर सेव रखता है. हर ओईएम कॉन्फ़िगर किए जा सकने वाले ओवरले का इस्तेमाल करके, डिफ़ॉल्ट क्लाउड मीडिया की सेवा देने वाली कंपनी चुन सकता है. चुने गए ऐप्लिकेशन, सभी तकनीकी ज़रूरी शर्तें पूरी करते हों और क्वालिटी की सभी जांच में पास हुए हों. ओईएम की क्लाउड मीडिया की सेवा देने वाली कंपनी के पायलट प्रोग्राम की प्रोसेस और ज़रूरी शर्तों के बारे में ज़्यादा जानने के लिए, पूछताछ फ़ॉर्म भरें.

तय करें कि आपको क्लाउड मीडिया की सेवा देने वाली कंपनी बनानी है या नहीं

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

हर प्रोफ़ाइल के लिए, एक चालू क्लाउड मीडिया की सेवा देने वाली कंपनी

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

डिफ़ॉल्ट रूप से, Android का फ़ोटो पिकर, क्लाउड मीडिया की सेवा देने वाली कंपनी को अपने-आप चुनने की कोशिश करेगा.

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

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

क्लाउड मीडिया की सेवा देने वाली कंपनी बनाना

यहां दिए गए डायग्राम में, Android ऐप्लिकेशन, Android के फ़ोटो पिकर, स्थानीय डिवाइस के MediaProvider, और CloudMediaProvider के बीच, फ़ोटो चुनने के सेशन से पहले और उसके दौरान होने वाले इवेंट का क्रम दिखाया गया है.

फ़ोटो पिकर से क्लाउड मीडिया सेवा देने वाली कंपनी तक के फ़्लो को दिखाने वाला सीक्वेंस डायग्राम
पहली इमेज: फ़ोटो चुनने के सेशन के दौरान होने वाले इवेंट का क्रम दिखाने वाला डायग्राम.
  1. सिस्टम, उपयोगकर्ता की पसंदीदा क्लाउड मीडिया की सेवा देने वाली कंपनी को शुरू करता है और समय-समय पर मीडिया के मेटाडेटा को Android के फ़ोटो पिकर के बैकएंड में सिंक करता है.
  2. जब कोई Android ऐप्लिकेशन, फ़ोटो पिकर लॉन्च करता है, तो उपयोगकर्ता को स्थानीय या क्लाउड से ली गई आइटम का मर्ज किया गया ग्रिड दिखाने से पहले, फ़ोटो पिकर, क्लाउड मीडिया की सेवा देने वाली कंपनी के साथ, कम समय में होने वाला इंक्रीमेंटल सिंक करता है. इससे यह पक्का किया जाता है कि नतीजे, ज़्यादा से ज़्यादा अप-टू-डेट हों. जवाब मिलने के बाद या समयसीमा खत्म होने पर, फ़ोटो पिकर ग्रिड में अब ऐक्सेस की जा सकने वाली सभी फ़ोटो दिखती हैं. इसमें, आपके डिवाइस पर स्थानीय तौर पर सेव की गई फ़ोटो के साथ-साथ, क्लाउड से सिंक की गई फ़ोटो भी शामिल होती हैं.
  3. उपयोगकर्ता के स्क्रोल करने पर, फ़ोटो पिकर, यूज़र इंटरफ़ेस (यूआई) में दिखाने के लिए, क्लाउड मीडिया की सेवा देने वाली कंपनी से मीडिया के थंबनेल फ़ेच करता है.
  4. जब उपयोगकर्ता सेशन पूरा करता है और नतीजों में क्लाउड मीडिया का कोई आइटम शामिल होता है, तो फ़ोटो पिकर, कॉन्टेंट के लिए फ़ाइल डिस्क्रिप्टर का अनुरोध करता है, यूआरआई जनरेट करता है, और कॉल करने वाले ऐप्लिकेशन को फ़ाइल का ऐक्सेस देता है.
  5. अब ऐप्लिकेशन, यूआरआई खोल सकता है और उसके पास मीडिया के कॉन्टेंट का सिर्फ़ पढ़ने का ऐक्सेस होता है. डिफ़ॉल्ट रूप से, संवेदनशील मेटाडेटा को छिपा दिया जाता है. फ़ोटो पिकर, Android ऐप्लिकेशन और क्लाउड मीडिया की सेवा देने वाली कंपनी के बीच डेटा के आदान-प्रदान को कोऑर्डिनेट करने के लिए, FUSE फ़ाइल सिस्टम का इस्तेमाल करता है.

सामान्य समस्याएं

यहां कुछ अहम बातें दी गई हैं जिन्हें लागू करने से पहले ध्यान में रखना चाहिए:

डुप्लीकेट फ़ाइलें डालने से बचना

Android के फ़ोटो पिकर के पास, क्लाउड मीडिया की स्थिति की जांच करने का कोई तरीका नहीं है. इसलिए, CloudMediaProvider को कर्सर की उस लाइन में MEDIA_STORE_URI उपलब्ध कराना होगा जिसमें ऐसी कोई भी फ़ाइल मौजूद हो जो क्लाउड और स्थानीय डिवाइस, दोनों पर मौजूद हो. ऐसा न करने पर, उपयोगकर्ता को फ़ोटो पिकर में डुप्लीकेट फ़ाइलें दिखेंगी.

झलक दिखाने के लिए, इमेज के साइज़ को ऑप्टिमाइज़ करना

यह बहुत ज़रूरी है कि onOpenPreview से वापस लाई गई फ़ाइल, पूरी रिज़ॉल्यूशन वाली इमेज न हो. साथ ही, यह अनुरोध किए गए Size के मुताबिक हो. बहुत बड़ी इमेज को यूज़र इंटरफ़ेस (यूआई) में लोड होने में ज़्यादा समय लगेगा. वहीं, बहुत छोटी इमेज, डिवाइस की स्क्रीन के साइज़ के हिसाब से पिक्सेलेट या धुंधली हो सकती है.

सही ओरिएंटेशन को मैनेज करना

अगर onOpenPreview में वापस लाए गए थंबनेल में उनका EXIF डेटा शामिल नहीं है, तो उन्हें सही ओरिएंटेशन में वापस लाया जाना चाहिए. ऐसा करने से, झलक दिखाने वाले ग्रिड में थंबनेल गलत तरीके से रोटेट नहीं होंगे.

बिना अनुमति के ऐक्सेस करने से रोकने की सुविधा

ContentProvider से कॉल करने वाले को डेटा वापस भेजने से पहले, MANAGE_CLOUD_MEDIA_PROVIDERS_PERMISSION की जांच करें. इससे, बिना अनुमति वाले ऐप्लिकेशन, क्लाउड डेटा को ऐक्सेस नहीं कर पाएंगे.

CloudMediaProvider क्लास

android.content.ContentProvider से ली गई CloudMediaProvider क्लास में, यहां दिए गए उदाहरण में दिखाए गए तरीकों जैसे तरीके शामिल होते हैं:

Kotlin

abstract class CloudMediaProvider : ContentProvider() {

    @NonNull
    abstract override fun onGetMediaCollectionInfo(@NonNull bundle: Bundle): Bundle

    @NonNull
    override fun onQueryAlbums(@NonNull bundle: Bundle): Cursor = TODO("Implement onQueryAlbums")

    @NonNull
    abstract override fun onQueryDeletedMedia(@NonNull bundle: Bundle): Cursor

    @NonNull
    abstract override fun onQueryMedia(@NonNull bundle: Bundle): Cursor

    @NonNull
    abstract override fun onOpenMedia(
        @NonNull string: String,
        @Nullable bundle: Bundle?,
        @Nullable cancellationSignal: CancellationSignal?
    ): ParcelFileDescriptor

    @NonNull
    abstract override fun onOpenPreview(
        @NonNull string: String,
        @NonNull point: Point,
        @Nullable bundle: Bundle?,
        @Nullable cancellationSignal: CancellationSignal?
    ): AssetFileDescriptor

    @Nullable
    override fun onCreateCloudMediaSurfaceController(
        @NonNull bundle: Bundle,
        @NonNull callback: CloudMediaSurfaceStateChangedCallback
    ): CloudMediaSurfaceController? = null
}

Java

public abstract class CloudMediaProvider extends android.content.ContentProvider {

  @NonNull
  public abstract android.os.Bundle onGetMediaCollectionInfo(@NonNull android.os.Bundle);

  @NonNull
  public android.database.Cursor onQueryAlbums(@NonNull android.os.Bundle);

  @NonNull
  public abstract android.database.Cursor onQueryDeletedMedia(@NonNull android.os.Bundle);

  @NonNull
  public abstract android.database.Cursor onQueryMedia(@NonNull android.os.Bundle);

  @NonNull
  public abstract android.os.ParcelFileDescriptor onOpenMedia(@NonNull String, @Nullable android.os.Bundle, @Nullable android.os.CancellationSignal) throws java.io.FileNotFoundException;

  @NonNull
  public abstract android.content.res.AssetFileDescriptor onOpenPreview(@NonNull String, @NonNull android.graphics.Point, @Nullable android.os.Bundle, @Nullable android.os.CancellationSignal) throws java.io.FileNotFoundException;

  @Nullable
  public android.provider.CloudMediaProvider.CloudMediaSurfaceController onCreateCloudMediaSurfaceController(@NonNull android.os.Bundle, @NonNull android.provider.CloudMediaProvider.CloudMediaSurfaceStateChangedCallback);
}

CloudMediaProviderContract क्लास

Android के फ़ोटो पिकर में, प्राथमिक CloudMediaProvider को लागू करने वाली क्लास के अलावा, CloudMediaProviderContract क्लास भी शामिल होती है. इस क्लास में, फ़ोटो पिकर और क्लाउड मीडिया की सेवा देने वाली कंपनी के बीच इंटरऑपरेबिलिटी के बारे में बताया गया है. इसमें, सिंक करने की कार्रवाइयों के लिए MediaCollectionInfo, अनुमानित Cursor कॉलम, और Bundle एक्स्ट्रा जैसे पहलू शामिल हैं.

Kotlin

object CloudMediaProviderContract {

    const val EXTRA_ALBUM_ID = "android.provider.extra.ALBUM_ID"
    const val EXTRA_LOOPING_PLAYBACK_ENABLED = "android.provider.extra.LOOPING_PLAYBACK_ENABLED"
    const val EXTRA_MEDIA_COLLECTION_ID = "android.provider.extra.MEDIA_COLLECTION_ID"
    const val EXTRA_PAGE_SIZE = "android.provider.extra.PAGE_SIZE"
    const val EXTRA_PAGE_TOKEN = "android.provider.extra.PAGE_TOKEN"
    const val EXTRA_PREVIEW_THUMBNAIL = "android.provider.extra.PREVIEW_THUMBNAIL"
    const val EXTRA_SURFACE_CONTROLLER_AUDIO_MUTE_ENABLED = "android.provider.extra.SURFACE_CONTROLLER_AUDIO_MUTE_ENABLED"
    const val EXTRA_SYNC_GENERATION = "android.provider.extra.SYNC_GENERATION"
    const val MANAGE_CLOUD_MEDIA_PROVIDERS_PERMISSION = "com.android.providers.media.permission.MANAGE_CLOUD_MEDIA_PROVIDERS"
    const val PROVIDER_INTERFACE = "android.content.action.CLOUD_MEDIA_PROVIDER"

    object MediaColumns {
        const val DATE_TAKEN_MILLIS = "date_taken_millis"
        const val DURATION_MILLIS = "duration_millis"
        const val HEIGHT = "height"
        const val ID = "id"
        const val IS_FAVORITE = "is_favorite"
        const val MEDIA_STORE_URI = "media_store_uri"
        const val MIME_TYPE = "mime_type"
        const val ORIENTATION = "orientation"
        const val SIZE_BYTES = "size_bytes"
        const val STANDARD_MIME_TYPE_EXTENSION = "standard_mime_type_extension"
        const val STANDARD_MIME_TYPE_EXTENSION_ANIMATED_WEBP = 3 // 0x3
        const val STANDARD_MIME_TYPE_EXTENSION_GIF = 1 // 0x1
        const val STANDARD_MIME_TYPE_EXTENSION_MOTION_PHOTO = 2 // 0x2
        const val STANDARD_MIME_TYPE_EXTENSION_NONE = 0 // 0x0
        const val SYNC_GENERATION = "sync_generation"
        const val WIDTH = "width"
    }

    object AlbumColumns {
        const val DATE_TAKEN_MILLIS = "date_taken_millis"
        const val DISPLAY_NAME = "display_name"
        const val ID = "id"
        const val MEDIA_COUNT = "album_media_count"
        const val MEDIA_COVER_ID = "album_media_cover_id"
    }

    object MediaCollectionInfo {
        const val ACCOUNT_CONFIGURATION_INTENT = "account_configuration_intent"
        const val ACCOUNT_NAME = "account_name"
        const val LAST_MEDIA_SYNC_GENERATION = "last_media_sync_generation"
        const val MEDIA_COLLECTION_ID = "media_collection_id"
    }
}

Java

public final class CloudMediaProviderContract {

  public static final String EXTRA_ALBUM_ID = "android.provider.extra.ALBUM_ID";
  public static final String EXTRA_LOOPING_PLAYBACK_ENABLED = "android.provider.extra.LOOPING_PLAYBACK_ENABLED";
  public static final String EXTRA_MEDIA_COLLECTION_ID = "android.provider.extra.MEDIA_COLLECTION_ID";
  public static final String EXTRA_PAGE_SIZE = "android.provider.extra.PAGE_SIZE";
  public static final String EXTRA_PAGE_TOKEN = "android.provider.extra.PAGE_TOKEN";
  public static final String EXTRA_PREVIEW_THUMBNAIL = "android.provider.extra.PREVIEW_THUMBNAIL";
  public static final String EXTRA_SURFACE_CONTROLLER_AUDIO_MUTE_ENABLED = "android.provider.extra.SURFACE_CONTROLLER_AUDIO_MUTE_ENABLED";
  public static final String EXTRA_SYNC_GENERATION = "android.provider.extra.SYNC_GENERATION";
  public static final String MANAGE_CLOUD_MEDIA_PROVIDERS_PERMISSION = "com.android.providers.media.permission.MANAGE_CLOUD_MEDIA_PROVIDERS";
  public static final String PROVIDER_INTERFACE = "android.content.action.CLOUD_MEDIA_PROVIDER";
}

// Columns available for every media item
public static final class CloudMediaProviderContract.MediaColumns {

  public static final String DATE_TAKEN_MILLIS = "date_taken_millis";
  public static final String DURATION_MILLIS = "duration_millis";
  public static final String HEIGHT = "height";
  public static final String ID = "id";
  public static final String IS_FAVORITE = "is_favorite";
  public static final String MEDIA_STORE_URI = "media_store_uri";
  public static final String MIME_TYPE = "mime_type";
  public static final String ORIENTATION = "orientation";
  public static final String SIZE_BYTES = "size_bytes";
  public static final String STANDARD_MIME_TYPE_EXTENSION = "standard_mime_type_extension";
  public static final int STANDARD_MIME_TYPE_EXTENSION_ANIMATED_WEBP = 3; // 0x3
  public static final int STANDARD_MIME_TYPE_EXTENSION_GIF = 1; // 0x1 
  public static final int STANDARD_MIME_TYPE_EXTENSION_MOTION_PHOTO = 2; // 0x2 
  public static final int STANDARD_MIME_TYPE_EXTENSION_NONE = 0; // 0x0 
  public static final String SYNC_GENERATION = "sync_generation";
  public static final String WIDTH = "width";
}

// Columns available for every album item
public static final class CloudMediaProviderContract.AlbumColumns {

  public static final String DATE_TAKEN_MILLIS = "date_taken_millis";
  public static final String DISPLAY_NAME = "display_name";
  public static final String ID = "id";
  public static final String MEDIA_COUNT = "album_media_count";
  public static final String MEDIA_COVER_ID = "album_media_cover_id";
}

// Media Collection metadata that is cached by the OS to compare sync states.
public static final class CloudMediaProviderContract.MediaCollectionInfo {

  public static final String ACCOUNT_CONFIGURATION_INTENT = "account_configuration_intent";
  public static final String ACCOUNT_NAME = "account_name";
  public static final String LAST_MEDIA_SYNC_GENERATION = "last_media_sync_generation";
  public static final String MEDIA_COLLECTION_ID = "media_collection_id";
}

onGetMediaCollectionInfo

onGetMediaCollectionInfo() तरीके का इस्तेमाल, ऑपरेटिंग सिस्टम, क्लाउड मीडिया के कैश मेमोरी में सेव किए गए आइटम की वैधता का आकलन करने और क्लाउड मीडिया की सेवा देने वाली कंपनी के साथ ज़रूरी सिंक तय करने के लिए करता है. ऑपरेटिंग सिस्टम की ओर से बार-बार कॉल किए जाने की संभावना की वजह से, onGetMediaCollectionInfo() को परफ़ॉर्मेंस के लिहाज़ से अहम माना जाता है. इसलिए, लंबे समय तक चलने वाली कार्रवाइयों या ऐसे साइड इफ़ेक्ट से बचना ज़रूरी है जिनसे परफ़ॉर्मेंस पर बुरा असर पड़ सकता है. ऑपरेटिंग सिस्टम, इस तरीके से मिले पिछले जवाबों को कैश मेमोरी में सेव करता है और सही कार्रवाइयां तय करने के लिए, उनकी तुलना बाद के जवाबों से करता है.

Kotlin

abstract fun onGetMediaCollectionInfo(extras: Bundle): Bundle

Java

@NonNull
public abstract Bundle onGetMediaCollectionInfo(@NonNull Bundle extras);

वापस लाए गए MediaCollectionInfo बंडल में ये कॉन्स्टैंट शामिल होते हैं:

onQueryMedia

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

यह तरीका, मीडिया कलेक्शन में मौजूद सभी मीडिया आइटम को दिखाने वाला Cursor लौटाता है. इसे, दिए गए एक्स्ट्रा के हिसाब से फ़िल्टर किया जा सकता है. साथ ही, इसे MediaColumns#DATE_TAKEN_MILLIS के हिसाब से, उल्टे क्रम में क्रम से लगाया जा सकता है. इसमें, सबसे हाल के आइटम पहले दिखते हैं.

वापस लाए गए CloudMediaProviderContract बंडल में ये कॉन्स्टैंट शामिल होते हैं:

क्लाउड मीडिया की सेवा देने वाली कंपनी को, वापस लाए गए Bundle के हिस्से के तौर पर, CloudMediaProviderContract#EXTRA_MEDIA_COLLECTION_ID सेट करना होगा. इसे सेट न करने पर गड़बड़ी होती है और वापस लाया गया Cursor अमान्य हो जाता है. अगर क्लाउड मीडिया की सेवा देने वाली कंपनी ने, दिए गए एक्स्ट्रा में मौजूद किसी फ़िल्टर को मैनेज किया है, तो उसे वापस लाए गए Cursor#setExtras के हिस्से के तौर पर, ContentResolver#EXTRA_HONORED_ARGS में कुंजी जोड़नी होगी.

onQueryDeletedMedia

onQueryDeletedMedia() तरीके का इस्तेमाल, यह पक्का करने के लिए किया जाता है कि क्लाउड खाते में मिटाए गए आइटम, फ़ोटो पिकर के यूज़र इंटरफ़ेस (यूआई) से सही तरीके से हटाए गए हों. कम समय में होने वाले सिंक के लिए संवेदनशील होने की वजह से, इन कॉल को इनके हिस्से के तौर पर शुरू किया जा सकता है:

  • बैकग्राउंड में होने वाला, पहले से किया गया सिंक
  • फ़ोटो पिकर के सेशन (इसके लिए, यह ज़रूरी है कि सिंक की पूरी या इंक्रीमेंटल स्थिति की ज़रूरत हो)

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

यह तरीका, मौजूदा सेवा देने वाली कंपनी के वर्शन में, मीडिया कलेक्शन में मौजूद सभी मिटाए गए मीडिया आइटम को दिखाने वाला Cursor लौटाता है. इसे onGetMediaCollectionInfo() से वापस लाया जाता है. इन आइटम को, एक्स्ट्रा के हिसाब से फ़िल्टर किया जा सकता है. क्लाउड मीडिया की सेवा देने वाली कंपनी को, वापस लाए गए Cursor#setExtras के हिस्से के तौर पर, CloudMediaProviderContract#EXTRA_MEDIA_COLLECTION_ID सेट करना होगा. इसे सेट न करने पर गड़बड़ी होती है और Cursor अमान्य हो जाता है. अगर सेवा देने वाली कंपनी ने, दिए गए एक्स्ट्रा में मौजूद किसी फ़िल्टर को मैनेज किया है, तो उसे ContentResolver#EXTRA_HONORED_ARGS में कुंजी जोड़नी होगी.

onQueryAlbums

onQueryAlbums() तरीके का इस्तेमाल, क्लाउड मीडिया की सेवा देने वाली कंपनी में मौजूद क्लाउड एल्बम की सूची और उनसे जुड़े मेटाडेटा को फ़ेच करने के लिए किया जाता है. ज़्यादा जानकारी के लिए, CloudMediaProviderContract.AlbumColumns देखें.

यह तरीका, मीडिया कलेक्शन में मौजूद सभी एल्बम आइटम को दिखाने वाला Cursor लौटाता है. इसे, दिए गए एक्स्ट्रा के हिसाब से फ़िल्टर किया जा सकता है. साथ ही, इसे AlbumColumns#DATE_TAKEN_MILLIS के हिसाब से, उल्टे क्रम में क्रम से लगाया जा सकता है. इसमें, सबसे हाल के आइटम पहले दिखते हैं. क्लाउड मीडिया की सेवा देने वाली कंपनी को, वापस लाए गए Cursor के हिस्से के तौर पर, CloudMediaProviderContract#EXTRA_MEDIA_COLLECTION_ID सेट करना होगा. इसे सेट न करने पर गड़बड़ी होती है और वापस लाया गया Cursor अमान्य हो जाता है. अगर सेवा देने वाली कंपनी ने, दिए गए एक्स्ट्रा में मौजूद किसी फ़िल्टर को मैनेज किया है, तो उसे वापस लाए गए Cursor के हिस्से के तौर पर, ContentResolver#EXTRA_HONORED_ARGS में कुंजी जोड़नी होगी.

onOpenMedia

onOpenMedia() तरीके को, दिए गए mediaId से पहचाने गए पूरे साइज़ के मीडिया को वापस लाना चाहिए. अगर डिवाइस पर कॉन्टेंट डाउनलोड करते समय यह तरीका ब्लॉक हो जाता है, तो आपको समय-समय पर, दिए गए CancellationSignal की जांच करनी चाहिए, ताकि छोड़े गए अनुरोधों को रद्द किया जा सके.

onOpenPreview

onOpenPreview() तरीके को, दिए गए size के आइटम के लिए, दिए गए mediaId का थंबनेल वापस लाना चाहिए. थंबनेल, ओरिजनल CloudMediaProviderContract.MediaColumns#MIME_TYPE में होना चाहिए. साथ ही, उम्मीद है कि यह onOpenMedia से वापस लाए गए आइटम की तुलना में, बहुत कम रिज़ॉल्यूशन वाला होगा. अगर डिवाइस पर कॉन्टेंट डाउनलोड करते समय यह तरीका ब्लॉक हो जाता है, तो आपको समय-समय पर, दिए गए CancellationSignal की जांच करनी चाहिए, ताकि छोड़े गए अनुरोधों को रद्द किया जा सके.

onCreateCloudMediaSurfaceController

onCreateCloudMediaSurfaceController() तरीके को, CloudMediaSurfaceController वापस लाना चाहिए. इसका इस्तेमाल, मीडिया आइटम की झलक दिखाने के लिए किया जाता है. अगर झलक दिखाने की सुविधा उपलब्ध नहीं है, तो null वापस लाना चाहिए.

CloudMediaSurfaceController मीडिया आइटम की झलक दिखाने की सुविधा को मैनेज करता है Surface के दिए गए इंस्टेंस पर. इस क्लास के तरीके, एसिंक्रोनस होने चाहिए. साथ ही, इन्हें किसी भी बड़ी कार्रवाई को करके ब्लॉक नहीं करना चाहिए. CloudMediaSurfaceController का एक इंस्टेंस, कई मीडिया आइटम को दिखाने के लिए ज़िम्मेदार होता है. ये आइटम, कई Surface से जुड़े होते हैं.

CloudMediaSurfaceController में, लाइफ़साइकल के इन कॉलबैक की सूची के लिए सहायता उपलब्ध है: