শেয়ার্ড স্টোরেজ থেকে মিডিয়া ফাইল অ্যাক্সেস করুন

আরও সমৃদ্ধ ব্যবহারকারীর অভিজ্ঞতা প্রদান করতে, অনেক অ্যাপ ব্যবহারকারীদের অবদান রাখতে এবং মিডিয়া অ্যাক্সেস করতে দেয় যা বাহ্যিক স্টোরেজ ভলিউমে উপলব্ধ। ফ্রেমওয়ার্ক মিডিয়া সংগ্রহে একটি অপ্টিমাইজ করা সূচক সরবরাহ করে, যাকে মিডিয়া স্টোর বলা হয়, যা ব্যবহারকারীদের এই মিডিয়া ফাইলগুলিকে আরও সহজে পুনরুদ্ধার এবং আপডেট করতে দেয়। আপনার অ্যাপ আনইনস্টল হওয়ার পরেও, এই ফাইলগুলি ব্যবহারকারীর ডিভাইসে থেকে যায়।

ফটো পিকার

মিডিয়া স্টোর ব্যবহার করার বিকল্প হিসাবে, অ্যান্ড্রয়েড ফটো পিকার টুল ব্যবহারকারীদের তাদের সম্পূর্ণ মিডিয়া লাইব্রেরিতে আপনার অ্যাপ অ্যাক্সেস দেওয়ার প্রয়োজন ছাড়াই মিডিয়া ফাইল নির্বাচন করার জন্য একটি নিরাপদ, অন্তর্নির্মিত উপায় প্রদান করে। এটি শুধুমাত্র সমর্থিত ডিভাইসগুলিতে উপলব্ধ। আরও তথ্যের জন্য, ফটো পিকার গাইড দেখুন।

মিডিয়া স্টোর

মিডিয়া স্টোর বিমূর্ততার সাথে ইন্টারঅ্যাক্ট করতে, একটি ContentResolver অবজেক্ট ব্যবহার করুন যা আপনি আপনার অ্যাপের প্রসঙ্গ থেকে পুনরুদ্ধার করেন:

কোটলিন

val projection = arrayOf(media-database-columns-to-retrieve)
val selection = sql-where-clause-with-placeholder-variables
val selectionArgs = values-of-placeholder-variables
val sortOrder = sql-order-by-clause

applicationContext.contentResolver.query(
    MediaStore.media-type.Media.EXTERNAL_CONTENT_URI,
    projection,
    selection,
    selectionArgs,
    sortOrder
)?.use { cursor ->
    while (cursor.moveToNext()) {
        // Use an ID column from the projection to get
        // a URI representing the media item itself.
    }
}

জাভা

String[] projection = new String[] {
        media-database-columns-to-retrieve
};
String selection = sql-where-clause-with-placeholder-variables;
String[] selectionArgs = new String[] {
        values-of-placeholder-variables
};
String sortOrder = sql-order-by-clause;

Cursor cursor = getApplicationContext().getContentResolver().query(
    MediaStore.media-type.Media.EXTERNAL_CONTENT_URI,
    projection,
    selection,
    selectionArgs,
    sortOrder
);

while (cursor.moveToNext()) {
    // Use an ID column from the projection to get
    // a URI representing the media item itself.
}

সিস্টেমটি স্বয়ংক্রিয়ভাবে একটি বাহ্যিক স্টোরেজ ভলিউম স্ক্যান করে এবং মিডিয়া ফাইলগুলিকে নিম্নলিখিত সুসংজ্ঞায়িত সংগ্রহগুলিতে যোগ করে:

  • ছবি, ফটোগ্রাফ এবং স্ক্রিনশট সহ, যা DCIM/ এবং Pictures/ ডিরেক্টরিতে সংরক্ষণ করা হয়। সিস্টেম এই ফাইলগুলিকে MediaStore.Images টেবিলে যোগ করে।
  • ভিডিও, যা DCIM/ , Movies/ , এবং Pictures/ ডিরেক্টরিতে সংরক্ষিত থাকে। সিস্টেম এই ফাইলগুলিকে MediaStore.Video টেবিলে যুক্ত করে।
  • অডিও ফাইল, যা Alarms/ , Audiobooks/ , Music/ , Notifications/ , Podcasts/ , এবং Ringtones/ ডিরেক্টরিতে সংরক্ষিত থাকে। অতিরিক্তভাবে, সিস্টেম অডিও প্লেলিস্টগুলিকে স্বীকৃতি দেয় যেগুলি Music/ বা Movies/ ডিরেক্টরিগুলির পাশাপাশি ভয়েস রেকর্ডিংগুলি যেগুলি Recordings/ ডিরেক্টরিতে রয়েছে৷ সিস্টেম এই ফাইলগুলিকে MediaStore.Audio টেবিলে যোগ করে। Recordings/ ডিরেক্টরিটি Android 11 (API লেভেল 30) এবং তার নিচের সংস্করণে উপলভ্য নয়।
  • ডাউনলোড করা ফাইল, যা Download/ ডিরেক্টরিতে সংরক্ষিত থাকে। যে ডিভাইসগুলিতে Android 10 (API লেভেল 29) এবং উচ্চতর চলমান, এই ফাইলগুলি MediaStore.Downloads টেবিলে সংরক্ষণ করা হয়। এই টেবিলটি Android 9 (API লেভেল 28) এবং তার নিচের সংস্করণে উপলভ্য নয়।

মিডিয়া স্টোরে MediaStore.Files নামে একটি সংগ্রহও রয়েছে। এটির বিষয়বস্তু আপনার অ্যাপ স্কোপড স্টোরেজ ব্যবহার করে কিনা তার উপর নির্ভর করে, যে অ্যাপগুলি Android 10 বা তার বেশির দিকে লক্ষ্য করে।

  • স্কোপড স্টোরেজ সক্ষম করা থাকলে, সংগ্রহটি শুধুমাত্র আপনার অ্যাপ তৈরি করা ফটো, ভিডিও এবং অডিও ফাইল দেখায়। বেশিরভাগ ডেভেলপারদের অন্য অ্যাপ থেকে মিডিয়া ফাইল দেখার জন্য MediaStore.Files ব্যবহার করার প্রয়োজন নেই, তবে আপনার যদি এটি করার জন্য একটি নির্দিষ্ট প্রয়োজন থাকে, তাহলে আপনি READ_EXTERNAL_STORAGE অনুমতি ঘোষণা করতে পারেন। যাইহোক, আমরা সুপারিশ করি যে আপনি আপনার অ্যাপ তৈরি করেনি এমন ফাইলগুলি খুলতে MediaStore API ব্যবহার করুন।
  • স্কোপড স্টোরেজ অনুপলব্ধ হলে বা ব্যবহার করা না হলে, সংগ্রহটি সব ধরনের মিডিয়া ফাইল দেখায়।

প্রয়োজনীয় অনুমতি অনুরোধ করুন

মিডিয়া ফাইলগুলিতে অপারেশন করার আগে, নিশ্চিত করুন যে আপনার অ্যাপটি এই ফাইলগুলি অ্যাক্সেস করার জন্য প্রয়োজনীয় অনুমতিগুলি ঘোষণা করেছে৷ সতর্ক থাকুন, তবে, আপনার অ্যাপের প্রয়োজন নেই বা ব্যবহার করে এমন অনুমতিগুলি ঘোষণা করবেন না।

স্টোরেজ অনুমতি

স্টোরেজ অ্যাক্সেস করার জন্য আপনার অ্যাপের অনুমতি প্রয়োজন কিনা তা নির্ভর করে এটি শুধুমাত্র তার নিজস্ব মিডিয়া ফাইল বা অন্য অ্যাপের তৈরি করা ফাইল অ্যাক্সেস করে কিনা।

আপনার নিজস্ব মিডিয়া ফাইল অ্যাক্সেস করুন

যে ডিভাইসগুলিতে Android 10 বা উচ্চতর সংস্করণ চলে, সেগুলিতে MediaStore.Downloads সংগ্রহের ফাইলগুলি সহ আপনার অ্যাপের মালিকানাধীন মিডিয়া ফাইলগুলি অ্যাক্সেস এবং সংশোধন করার জন্য আপনার স্টোরেজ-সম্পর্কিত অনুমতির প্রয়োজন নেই৷ আপনি যদি একটি ক্যামেরা অ্যাপ তৈরি করেন, উদাহরণস্বরূপ, এটি নেওয়া ফটোগুলি অ্যাক্সেস করার জন্য আপনাকে স্টোরেজ-সম্পর্কিত অনুমতিগুলির অনুরোধ করার দরকার নেই, কারণ আপনি মিডিয়া স্টোরে যে ছবিগুলি লিখছেন সেগুলির মালিক আপনার অ্যাপ৷

অন্যান্য অ্যাপের মিডিয়া ফাইল অ্যাক্সেস করুন

অন্য অ্যাপগুলি তৈরি করা মিডিয়া ফাইলগুলি অ্যাক্সেস করতে, আপনাকে অবশ্যই উপযুক্ত স্টোরেজ-সম্পর্কিত অনুমতিগুলি ঘোষণা করতে হবে এবং ফাইলগুলিকে অবশ্যই নিম্নলিখিত মিডিয়া সংগ্রহগুলির মধ্যে একটিতে থাকতে হবে:

যতক্ষণ পর্যন্ত একটি ফাইল MediaStore.Images , MediaStore.Video , বা MediaStore.Audio কোয়েরি থেকে দেখা যায়, ততক্ষণ এটি MediaStore.Files ক্যোয়ারী ব্যবহার করেও দেখা যায়৷

নিম্নলিখিত কোড স্নিপেট দেখায় কিভাবে উপযুক্ত স্টোরেজ অনুমতি ঘোষণা করতে হয়:

<!-- Required only if your app needs to access images or photos
     that other apps created. -->
<uses-permission android:name="android.permission.READ_MEDIA_IMAGES" />

<!-- Required only if your app needs to access videos
     that other apps created. -->
<uses-permission android:name="android.permission.READ_MEDIA_VIDEO" />

<!-- Required only if your app needs to access audio files
     that other apps created. -->
<uses-permission android:name="android.permission.READ_MEDIA_AUDIO" />

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

<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"
                 android:maxSdkVersion="29" />

লিগ্যাসি ডিভাইসে চলমান অ্যাপের জন্য অতিরিক্ত অনুমতি প্রয়োজন

যদি আপনার অ্যাপটি এমন কোনো ডিভাইসে ব্যবহার করা হয় যা Android 9 বা তার চেয়ে কম সংস্করণ চালায়, অথবা যদি আপনার অ্যাপটি সাময়িকভাবে স্কোপড স্টোরেজ থেকে অপ্ট আউট করে থাকে, তাহলে আপনাকে অবশ্যই যেকোনো মিডিয়া ফাইল অ্যাক্সেস করার জন্য READ_EXTERNAL_STORAGE অনুমতির অনুরোধ করতে হবে। আপনি যদি মিডিয়া ফাইলগুলি পরিবর্তন করতে চান তবে আপনাকে অবশ্যই WRITE_EXTERNAL_STORAGE অনুমতির জন্য অনুরোধ করতে হবে।

অন্যান্য অ্যাপের ডাউনলোড অ্যাক্সেস করার জন্য স্টোরেজ অ্যাক্সেস ফ্রেমওয়ার্ক প্রয়োজন

আপনার অ্যাপ যদি MediaStore.Downloads সংগ্রহের মধ্যে একটি ফাইল অ্যাক্সেস করতে চায় যা আপনার অ্যাপ তৈরি করেনি, তাহলে আপনাকে অবশ্যই স্টোরেজ অ্যাক্সেস ফ্রেমওয়ার্ক ব্যবহার করতে হবে। এই ফ্রেমওয়ার্কটি কীভাবে ব্যবহার করবেন সে সম্পর্কে আরও জানতে, শেয়ার্ড স্টোরেজ থেকে নথি এবং অন্যান্য ফাইল অ্যাক্সেস করুন

মিডিয়া অবস্থানের অনুমতি

যদি আপনার অ্যাপটি Android 10 (API স্তর 29) বা উচ্চতরকে লক্ষ্য করে এবং ফটোগুলি থেকে অসংশোধিত EXIF ​​মেটাডেটা পুনরুদ্ধার করতে হয়, তাহলে আপনাকে আপনার অ্যাপের ম্যানিফেস্টে ACCESS_MEDIA_LOCATION অনুমতি ঘোষণা করতে হবে, তারপর রানটাইমে এই অনুমতির অনুরোধ করুন৷

মিডিয়া স্টোরের আপডেটের জন্য চেক করুন

মিডিয়া ফাইলগুলিকে আরও নির্ভরযোগ্যভাবে অ্যাক্সেস করতে, বিশেষ করে যদি আপনার অ্যাপ মিডিয়া স্টোর থেকে ইউআরআই বা ডেটা ক্যাশে করে, আপনি যখন আপনার মিডিয়া ডেটা শেষবার সিঙ্ক করেছিলেন সেই তুলনায় মিডিয়া স্টোর সংস্করণ পরিবর্তিত হয়েছে কিনা তা পরীক্ষা করুন৷ আপডেটের জন্য এই চেকটি সম্পাদন করতে, getVersion() কল করুন। প্রত্যাবর্তিত সংস্করণটি একটি অনন্য স্ট্রিং যা মিডিয়া স্টোরের উল্লেখযোগ্য পরিবর্তন হলেই পরিবর্তিত হয়। যদি প্রত্যাবর্তিত সংস্করণটি শেষ সিঙ্ক করা সংস্করণ থেকে আলাদা হয়, তাহলে আপনার অ্যাপের মিডিয়া ক্যাশে পুনরায় স্ক্যান করুন এবং পুনরায় সিঙ্ক করুন৷

অ্যাপ প্রক্রিয়া শুরুর সময় এই চেকটি সম্পূর্ণ করুন। আপনি যখন মিডিয়া স্টোরে প্রশ্ন করবেন তখন সংস্করণটি পরীক্ষা করার দরকার নেই।

সংস্করণ নম্বর সম্পর্কিত কোনো বাস্তবায়ন বিবরণ অনুমান করবেন না।

একটি মিডিয়া সংগ্রহের জন্য অনুসন্ধান করুন

5 মিনিট বা তার বেশি সময়কালের মতো শর্তগুলির একটি নির্দিষ্ট সেটকে সন্তুষ্ট করে এমন মিডিয়া খুঁজে পেতে, নিম্নলিখিত কোড স্নিপেটে দেখানো একটির মতো একটি SQL-এর মতো নির্বাচন বিবৃতি ব্যবহার করুন:

কোটলিন

// Need the READ_EXTERNAL_STORAGE permission if accessing video files that your
// app didn't create.

// Container for information about each video.
data class Video(val uri: Uri,
    val name: String,
    val duration: Int,
    val size: Int
)
val videoList = mutableListOf<Video>()

val collection =
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        MediaStore.Video.Media.getContentUri(
            MediaStore.VOLUME_EXTERNAL
        )
    } else {
        MediaStore.Video.Media.EXTERNAL_CONTENT_URI
    }

val projection = arrayOf(
    MediaStore.Video.Media._ID,
    MediaStore.Video.Media.DISPLAY_NAME,
    MediaStore.Video.Media.DURATION,
    MediaStore.Video.Media.SIZE
)

// Show only videos that are at least 5 minutes in duration.
val selection = "${MediaStore.Video.Media.DURATION} >= ?"
val selectionArgs = arrayOf(
    TimeUnit.MILLISECONDS.convert(5, TimeUnit.MINUTES).toString()
)

// Display videos in alphabetical order based on their display name.
val sortOrder = "${MediaStore.Video.Media.DISPLAY_NAME} ASC"

val query = ContentResolver.query(
    collection,
    projection,
    selection,
    selectionArgs,
    sortOrder
)
query?.use { cursor ->
    // Cache column indices.
    val idColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media._ID)
    val nameColumn =
            cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DISPLAY_NAME)
    val durationColumn =
            cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATION)
    val sizeColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media.SIZE)

    while (cursor.moveToNext()) {
        // Get values of columns for a given video.
        val id = cursor.getLong(idColumn)
        val name = cursor.getString(nameColumn)
        val duration = cursor.getInt(durationColumn)
        val size = cursor.getInt(sizeColumn)

        val contentUri: Uri = ContentUris.withAppendedId(
            MediaStore.Video.Media.EXTERNAL_CONTENT_URI,
            id
        )

        // Stores column values and the contentUri in a local object
        // that represents the media file.
        videoList += Video(contentUri, name, duration, size)
    }
}

জাভা

// Need the READ_EXTERNAL_STORAGE permission if accessing video files that your
// app didn't create.

// Container for information about each video.
class Video {
    private final Uri uri;
    private final String name;
    private final int duration;
    private final int size;

    public Video(Uri uri, String name, int duration, int size) {
        this.uri = uri;
        this.name = name;
        this.duration = duration;
        this.size = size;
    }
}
List<Video> videoList = new ArrayList<Video>();

Uri collection;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
    collection = MediaStore.Video.Media.getContentUri(MediaStore.VOLUME_EXTERNAL);
} else {
    collection = MediaStore.Video.Media.EXTERNAL_CONTENT_URI;
}

String[] projection = new String[] {
    MediaStore.Video.Media._ID,
    MediaStore.Video.Media.DISPLAY_NAME,
    MediaStore.Video.Media.DURATION,
    MediaStore.Video.Media.SIZE
};
String selection = MediaStore.Video.Media.DURATION +
        " >= ?";
String[] selectionArgs = new String[] {
    String.valueOf(TimeUnit.MILLISECONDS.convert(5, TimeUnit.MINUTES));
};
String sortOrder = MediaStore.Video.Media.DISPLAY_NAME + " ASC";

try (Cursor cursor = getApplicationContext().getContentResolver().query(
    collection,
    projection,
    selection,
    selectionArgs,
    sortOrder
)) {
    // Cache column indices.
    int idColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media._ID);
    int nameColumn =
            cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DISPLAY_NAME);
    int durationColumn =
            cursor.getColumnIndexOrThrow(MediaStore.Video.Media.DURATION);
    int sizeColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media.SIZE);

    while (cursor.moveToNext()) {
        // Get values of columns for a given video.
        long id = cursor.getLong(idColumn);
        String name = cursor.getString(nameColumn);
        int duration = cursor.getInt(durationColumn);
        int size = cursor.getInt(sizeColumn);

        Uri contentUri = ContentUris.withAppendedId(
                MediaStore.Video.Media.EXTERNAL_CONTENT_URI, id);

        // Stores column values and the contentUri in a local object
        // that represents the media file.
        videoList.add(new Video(contentUri, name, duration, size));
    }
}

আপনার অ্যাপে এই ধরনের প্রশ্ন করার সময়, নিম্নলিখিতগুলি মনে রাখবেন:

  • একটি কর্মী থ্রেডে query() পদ্ধতিতে কল করুন।
  • কলামের সূচকগুলি ক্যাশে করুন যাতে প্রতিবার আপনি প্রশ্নের ফলাফল থেকে একটি সারি প্রক্রিয়া করার সময় আপনাকে getColumnIndexOrThrow() কল করতে হবে না।
  • এই উদাহরণে দেখানো কন্টেন্ট URI-তে ID যোগ করুন।
  • যে ডিভাইসগুলি Android 10 এবং উচ্চতর চালায় সেগুলির কলামের নাম প্রয়োজন যা MediaStore API-এ সংজ্ঞায়িত করা আছে। আপনার অ্যাপের মধ্যে একটি নির্ভরশীল লাইব্রেরি যদি API-তে অনির্ধারিত একটি কলামের নাম আশা করে, যেমন "MimeType" , তাহলে আপনার অ্যাপের প্রক্রিয়ায় কলামের নামটি গতিশীলভাবে অনুবাদ করতে CursorWrapper ব্যবহার করুন।

ফাইল থাম্বনেল লোড

যদি আপনার অ্যাপ একাধিক মিডিয়া ফাইল দেখায় এবং ব্যবহারকারীকে এই ফাইলগুলির মধ্যে একটি বেছে নেওয়ার অনুরোধ করে, তাহলে ফাইলগুলির পরিবর্তে ফাইলগুলির পূর্বরূপ সংস্করণ বা থাম্বনেইলগুলি লোড করা আরও কার্যকর।

একটি প্রদত্ত মিডিয়া ফাইলের জন্য থাম্বনেইল লোড করতে, loadThumbnail() ব্যবহার করুন এবং আপনি যে থাম্বনেইলটি লোড করতে চান তার আকারে পাস করুন, যেমনটি নিম্নলিখিত কোড স্নিপেটে দেখানো হয়েছে:

কোটলিন

// Load thumbnail of a specific media item.
val thumbnail: Bitmap =
        applicationContext.contentResolver.loadThumbnail(
        content-uri, Size(640, 480), null)

জাভা

// Load thumbnail of a specific media item.
Bitmap thumbnail =
        getApplicationContext().getContentResolver().loadThumbnail(
        content-uri, new Size(640, 480), null);

একটি মিডিয়া ফাইল খুলুন

একটি মিডিয়া ফাইল খোলার জন্য আপনি যে নির্দিষ্ট যুক্তি ব্যবহার করেন তা নির্ভর করে মিডিয়া বিষয়বস্তু একটি ফাইল বর্ণনাকারী, একটি ফাইল স্ট্রীম, বা একটি সরাসরি ফাইল পাথ হিসাবে সেরাভাবে উপস্থাপন করা হয় কিনা তার উপর।

ফাইল বর্ণনাকারী

একটি ফাইল বর্ণনাকারী ব্যবহার করে একটি মিডিয়া ফাইল খুলতে, নিম্নলিখিত কোড স্নিপেটে দেখানো মত যুক্তি ব্যবহার করুন:

কোটলিন

// Open a specific media item using ParcelFileDescriptor.
val resolver = applicationContext.contentResolver

// "rw" for read-and-write.
// "rwt" for truncating or overwriting existing file contents.
val readOnlyMode = "r"
resolver.openFileDescriptor(content-uri, readOnlyMode).use { pfd ->
    // Perform operations on "pfd".
}

জাভা

// Open a specific media item using ParcelFileDescriptor.
ContentResolver resolver = getApplicationContext()
        .getContentResolver();

// "rw" for read-and-write.
// "rwt" for truncating or overwriting existing file contents.
String readOnlyMode = "r";
try (ParcelFileDescriptor pfd =
        resolver.openFileDescriptor(content-uri, readOnlyMode)) {
    // Perform operations on "pfd".
} catch (IOException e) {
    e.printStackTrace();
}

ফাইল স্ট্রীম

একটি ফাইল স্ট্রিম ব্যবহার করে একটি মিডিয়া ফাইল খুলতে, নিম্নলিখিত কোড স্নিপেটে দেখানো মত যুক্তি ব্যবহার করুন:

কোটলিন

// Open a specific media item using InputStream.
val resolver = applicationContext.contentResolver
resolver.openInputStream(content-uri).use { stream ->
    // Perform operations on "stream".
}

জাভা

// Open a specific media item using InputStream.
ContentResolver resolver = getApplicationContext()
        .getContentResolver();
try (InputStream stream = resolver.openInputStream(content-uri)) {
    // Perform operations on "stream".
}

সরাসরি ফাইল পাথ

তৃতীয় পক্ষের মিডিয়া লাইব্রেরিগুলির সাথে আপনার অ্যাপটিকে আরও মসৃণভাবে কাজ করতে সহায়তা করার জন্য, Android 11 (API স্তর 30) এবং উচ্চতর আপনাকে শেয়ার্ড স্টোরেজ থেকে মিডিয়া ফাইলগুলি অ্যাক্সেস করতে MediaStore API ছাড়া অন্য API ব্যবহার করতে দেয়৷ আপনি পরিবর্তে নিম্নলিখিত API গুলির মধ্যে যেকোনো একটি ব্যবহার করে সরাসরি মিডিয়া ফাইলগুলি অ্যাক্সেস করতে পারেন:

  • File এপিআই
  • নেটিভ লাইব্রেরি, যেমন fopen()

আপনার কাছে কোনো স্টোরেজ-সম্পর্কিত অনুমতি না থাকলে, আপনি File API ব্যবহার করে আপনার অ্যাপ-নির্দিষ্ট ডিরেক্টরির ফাইলগুলির পাশাপাশি মিডিয়া ফাইলগুলি অ্যাক্সেস করতে পারেন যা আপনার অ্যাপের জন্য দায়ী

যদি আপনার অ্যাপ File এপিআই ব্যবহার করে কোনো ফাইল অ্যাক্সেস করার চেষ্টা করে এবং এতে প্রয়োজনীয় অনুমতি না থাকে, তাহলে একটি FileNotFoundException ঘটে।

Android 10 (API লেভেল 29) চালিত একটি ডিভাইসে শেয়ার্ড স্টোরেজে অন্যান্য ফাইল অ্যাক্সেস করতে, আমরা আপনাকে আপনার অ্যাপের true ফাইলে requestLegacyExternalStorage সেট করে সাময়িকভাবে স্কোপড স্টোরেজ থেকে অপ্ট আউট করার পরামর্শ দিচ্ছি। Android 10-এ নেটিভ ফাইল পদ্ধতি ব্যবহার করে মিডিয়া ফাইল অ্যাক্সেস করতে, আপনাকে অবশ্যই READ_EXTERNAL_STORAGE অনুমতির অনুরোধ করতে হবে।

মিডিয়া বিষয়বস্তু অ্যাক্সেস করার সময় বিবেচনা

মিডিয়া বিষয়বস্তু অ্যাক্সেস করার সময়, নিম্নলিখিত বিভাগে আলোচনা করা বিবেচ্য বিষয়গুলি মনে রাখবেন৷

ক্যাশে ডেটা

যদি আপনার অ্যাপ মিডিয়া স্টোর থেকে ইউআরআই বা ডেটা ক্যাশে করে, তাহলে মিডিয়া স্টোরের আপডেটের জন্য পর্যায়ক্রমে চেক করুন । এই চেকটি আপনার অ্যাপ-সাইড, ক্যাশে করা ডেটা সিস্টেম-সাইড, প্রদানকারী ডেটার সাথে সিঙ্কে থাকতে দেয়।

কর্মক্ষমতা

আপনি যখন সরাসরি ফাইল পাথ ব্যবহার করে মিডিয়া ফাইলগুলির ক্রমিক পাঠ সম্পাদন করেন, তখন কার্যক্ষমতা MediaStore API-এর সাথে তুলনীয়।

আপনি যখন সরাসরি ফাইল পাথ ব্যবহার করে মিডিয়া ফাইলগুলির র্যান্ডম পঠন এবং লেখা সম্পাদন করেন, তবে, প্রক্রিয়াটি দ্বিগুণ ধীর হতে পারে। এই পরিস্থিতিতে, আমরা পরিবর্তে MediaStore API ব্যবহার করার পরামর্শ দিই।

ডেটা কলাম

আপনি যখন একটি বিদ্যমান মিডিয়া ফাইল অ্যাক্সেস করেন, আপনি আপনার যুক্তিতে DATA কলামের মান ব্যবহার করতে পারেন। কারণ এই মানটির একটি বৈধ ফাইল পাথ রয়েছে। যাইহোক, অনুমান করবেন না যে ফাইলটি সর্বদা উপলব্ধ। যে কোনো ফাইল-ভিত্তিক I/O ত্রুটিগুলি হ্যান্ডেল করার জন্য প্রস্তুত থাকুন।

একটি মিডিয়া ফাইল তৈরি বা আপডেট করতে, অন্যদিকে, DATA কলামের মান ব্যবহার করবেন না। পরিবর্তে, DISPLAY_NAME এবং RELATIVE_PATH কলামের মানগুলি ব্যবহার করুন৷

স্টোরেজ ভলিউম

যে অ্যাপগুলি Android 10 বা তার উচ্চতরকে টার্গেট করে সেগুলি অনন্য নাম অ্যাক্সেস করতে পারে যা সিস্টেম প্রতিটি বাহ্যিক স্টোরেজ ভলিউমের জন্য বরাদ্দ করে৷ এই নামকরণ সিস্টেম আপনাকে দক্ষতার সাথে বিষয়বস্তু সংগঠিত এবং সূচী করতে সহায়তা করে এবং এটি আপনাকে নতুন মিডিয়া ফাইলগুলি কোথায় সংরক্ষণ করা হয় তার উপর নিয়ন্ত্রণ দেয়।

নিম্নলিখিত ভলিউমগুলি মনে রাখার জন্য বিশেষভাবে দরকারী:

  • VOLUME_EXTERNAL ভলিউম ডিভাইসে শেয়ার করা সমস্ত স্টোরেজ ভলিউমের একটি দৃশ্য প্রদান করে৷ আপনি এই সিন্থেটিক ভলিউমের বিষয়বস্তু পড়তে পারেন, কিন্তু আপনি বিষয়বস্তু পরিবর্তন করতে পারবেন না।
  • VOLUME_EXTERNAL_PRIMARY ভলিউম ডিভাইসে প্রাথমিক ভাগ করা স্টোরেজ ভলিউম উপস্থাপন করে৷ আপনি এই ভলিউমের বিষয়বস্তু পড়তে এবং পরিবর্তন করতে পারেন।

আপনি MediaStore.getExternalVolumeNames() কল করে অন্যান্য ভলিউম আবিষ্কার করতে পারেন:

কোটলিন

val volumeNames: Set<String> = MediaStore.getExternalVolumeNames(context)
val firstVolumeName = volumeNames.iterator().next()

জাভা

Set<String> volumeNames = MediaStore.getExternalVolumeNames(context);
String firstVolumeName = volumeNames.iterator().next();

অবস্থান যেখানে মিডিয়া বন্দী করা হয়েছে

কিছু ফটোগ্রাফ এবং ভিডিও তাদের মেটাডেটাতে অবস্থানের তথ্য ধারণ করে, যা সেই স্থানটি দেখায় যেখানে একটি ফটো তোলা হয়েছিল বা যেখানে একটি ভিডিও রেকর্ড করা হয়েছিল৷

আপনি কীভাবে আপনার অ্যাপে এই অবস্থানের তথ্য অ্যাক্সেস করবেন তা নির্ভর করে আপনার ফটোগ্রাফ বা ভিডিওর জন্য অবস্থানের তথ্য অ্যাক্সেস করতে হবে।

ফটোগ্রাফ

আপনার অ্যাপ স্কোপড স্টোরেজ ব্যবহার করলে, সিস্টেমটি ডিফল্টরূপে অবস্থানের তথ্য লুকিয়ে রাখে। এই তথ্য অ্যাক্সেস করতে, নিম্নলিখিত পদক্ষেপগুলি সম্পূর্ণ করুন:

  1. আপনার অ্যাপের ম্যানিফেস্টে ACCESS_MEDIA_LOCATION অনুমতির জন্য অনুরোধ করুন।
  2. আপনার MediaStore অবজেক্ট থেকে, setRequireOriginal() কল করে ফটোগ্রাফের সঠিক বাইটগুলি পান এবং ফটোগ্রাফের URI পাস করুন, যেমনটি নিম্নলিখিত কোড স্নিপেটে দেখানো হয়েছে:

    কোটলিন

    val photoUri: Uri = Uri.withAppendedPath(
            MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
            cursor.getString(idColumnIndex)
    )
    
    // Get location data using the Exifinterface library.
    // Exception occurs if ACCESS_MEDIA_LOCATION permission isn't granted.
    photoUri = MediaStore.setRequireOriginal(photoUri)
    contentResolver.openInputStream(photoUri)?.use { stream ->
        ExifInterface(stream).run {
            // If lat/long is null, fall back to the coordinates (0, 0).
            val latLong = latLong ?: doubleArrayOf(0.0, 0.0)
        }
    }
    

    জাভা

    Uri photoUri = Uri.withAppendedPath(
            MediaStore.Images.Media.EXTERNAL_CONTENT_URI,
            cursor.getString(idColumnIndex));
    
    final double[] latLong;
    
    // Get location data using the Exifinterface library.
    // Exception occurs if ACCESS_MEDIA_LOCATION permission isn't granted.
    photoUri = MediaStore.setRequireOriginal(photoUri);
    InputStream stream = getContentResolver().openInputStream(photoUri);
    if (stream != null) {
        ExifInterface exifInterface = new ExifInterface(stream);
        double[] returnedLatLong = exifInterface.getLatLong();
    
        // If lat/long is null, fall back to the coordinates (0, 0).
        latLong = returnedLatLong != null ? returnedLatLong : new double[2];
    
        // Don't reuse the stream associated with
        // the instance of "ExifInterface".
        stream.close();
    } else {
        // Failed to load the stream, so return the coordinates (0, 0).
        latLong = new double[2];
    }
    

ভিডিও

একটি ভিডিওর মেটাডেটার মধ্যে অবস্থানের তথ্য অ্যাক্সেস করতে, নিম্নলিখিত কোড স্নিপেটে দেখানো হিসাবে MediaMetadataRetriever ক্লাস ব্যবহার করুন। এই ক্লাসটি ব্যবহার করার জন্য আপনার অ্যাপকে কোনো অতিরিক্ত অনুমতির অনুরোধ করতে হবে না।

কোটলিন

val retriever = MediaMetadataRetriever()
val context = applicationContext

// Find the videos that are stored on a device by querying the video collection.
val query = ContentResolver.query(
    collection,
    projection,
    selection,
    selectionArgs,
    sortOrder
)
query?.use { cursor ->
    val idColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media._ID)
    while (cursor.moveToNext()) {
        val id = cursor.getLong(idColumn)
        val videoUri: Uri = ContentUris.withAppendedId(
            MediaStore.Video.Media.EXTERNAL_CONTENT_URI,
            id
        )
        extractVideoLocationInfo(videoUri)
    }
}

private fun extractVideoLocationInfo(videoUri: Uri) {
    try {
        retriever.setDataSource(context, videoUri)
    } catch (e: RuntimeException) {
        Log.e(APP_TAG, "Cannot retrieve video file", e)
    }
    // Metadata uses a standardized format.
    val locationMetadata: String? =
            retriever.extractMetadata(MediaMetadataRetriever.METADATA_KEY_LOCATION)
}

জাভা

MediaMetadataRetriever retriever = new MediaMetadataRetriever();
Context context = getApplicationContext();

// Find the videos that are stored on a device by querying the video collection.
try (Cursor cursor = context.getContentResolver().query(
    collection,
    projection,
    selection,
    selectionArgs,
    sortOrder
)) {
    int idColumn = cursor.getColumnIndexOrThrow(MediaStore.Video.Media._ID);
    while (cursor.moveToNext()) {
        long id = cursor.getLong(idColumn);
        Uri videoUri = ContentUris.withAppendedId(
                MediaStore.Video.Media.EXTERNAL_CONTENT_URI, id);
        extractVideoLocationInfo(videoUri);
    }
}

private void extractVideoLocationInfo(Uri videoUri) {
    try {
        retriever.setDataSource(context, videoUri);
    } catch (RuntimeException e) {
        Log.e(APP_TAG, "Cannot retrieve video file", e);
    }
    // Metadata uses a standardized format.
    String locationMetadata = retriever.extractMetadata(
            MediaMetadataRetriever.METADATA_KEY_LOCATION);
}

শেয়ারিং

কিছু অ্যাপ ব্যবহারকারীদের একে অপরের সাথে মিডিয়া ফাইল শেয়ার করতে দেয়। উদাহরণস্বরূপ, সোশ্যাল মিডিয়া অ্যাপগুলি ব্যবহারকারীদের বন্ধুদের সাথে ফটো এবং ভিডিও শেয়ার করতে দেয়৷

মিডিয়া ফাইলগুলি ভাগ করতে, একটি সামগ্রী প্রদানকারী তৈরির নির্দেশিকাতে সুপারিশকৃত একটি content:// URI ব্যবহার করুন৷

মিডিয়া ফাইলের অ্যাপ অ্যাট্রিবিউশন

অ্যান্ড্রয়েড 10 বা উচ্চতরকে লক্ষ্য করে এমন কোনও অ্যাপের জন্য যখন স্কোপড স্টোরেজ সক্ষম করা হয়, তখন সিস্টেমটি প্রতিটি মিডিয়া ফাইলের জন্য একটি অ্যাপকে অ্যাট্রিবিউট করে , যা নির্ধারণ করে যে আপনার অ্যাপটি কোন স্টোরেজ অনুমতির অনুরোধ না করলে সেটি অ্যাক্সেস করতে পারবে। প্রতিটি ফাইল শুধুমাত্র একটি অ্যাপে দায়ী করা যেতে পারে। তাই, যদি আপনার অ্যাপ এমন একটি মিডিয়া ফাইল তৈরি করে যা ফটো, ভিডিও বা অডিও ফাইল মিডিয়া সংগ্রহে সংরক্ষিত থাকে, তাহলে আপনার অ্যাপের ফাইলটিতে অ্যাক্সেস থাকবে।

ব্যবহারকারী যদি আপনার অ্যাপ আনইনস্টল করে এবং পুনরায় ইনস্টল করে, তবে, আপনাকে অবশ্যই READ_EXTERNAL_STORAGE আপনার অ্যাপটি তৈরি করা ফাইলগুলি অ্যাক্সেস করার জন্য অনুরোধ করতে হবে। এই অনুমতি অনুরোধটি প্রয়োজন কারণ সিস্টেম ফাইলটিকে নতুন ইনস্টল করা সংস্করণের পরিবর্তে অ্যাপের পূর্বে ইনস্টল করা সংস্করণের জন্য দায়ী বলে মনে করে৷

একটি আইটেম যোগ করুন

একটি বিদ্যমান সংগ্রহে একটি মিডিয়া আইটেম যোগ করতে, নিম্নলিখিত অনুরূপ কোড ব্যবহার করুন. এই কোড স্নিপেটটি Android 10 বা উচ্চতর সংস্করণে চালিত ডিভাইসগুলিতে VOLUME_EXTERNAL_PRIMARY ভলিউম অ্যাক্সেস করে৷ এর কারণ হল, এই ডিভাইসগুলিতে, আপনি শুধুমাত্র একটি ভলিউমের বিষয়বস্তু পরিবর্তন করতে পারবেন যদি এটি প্রাথমিক ভলিউম হয়, যেমন স্টোরেজ ভলিউম বিভাগে বর্ণনা করা হয়েছে।

কোটলিন

// Add a specific media item.
val resolver = applicationContext.contentResolver

// Find all audio files on the primary external storage device.
val audioCollection =
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        MediaStore.Audio.Media.getContentUri(
            MediaStore.VOLUME_EXTERNAL_PRIMARY
        )
    } else {
        MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
    }

// Publish a new song.
val newSongDetails = ContentValues().apply {
    put(MediaStore.Audio.Media.DISPLAY_NAME, "My Song.mp3")
}

// Keep a handle to the new song's URI in case you need to modify it
// later.
val myFavoriteSongUri = resolver
        .insert(audioCollection, newSongDetails)

জাভা

// Add a specific media item.
ContentResolver resolver = getApplicationContext()
        .getContentResolver();

// Find all audio files on the primary external storage device.
Uri audioCollection;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
    audioCollection = MediaStore.Audio.Media
            .getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY);
} else {
    audioCollection = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
}

// Publish a new song.
ContentValues newSongDetails = new ContentValues();
newSongDetails.put(MediaStore.Audio.Media.DISPLAY_NAME,
        "My Song.mp3");

// Keep a handle to the new song's URI in case you need to modify it
// later.
Uri myFavoriteSongUri = resolver
        .insert(audioCollection, newSongDetails);

মিডিয়া ফাইলগুলির জন্য মুলতুবি স্থিতি টগল করুন

যদি আপনার অ্যাপটি সম্ভাব্য সময়-সাপেক্ষ ক্রিয়াকলাপগুলি সম্পাদন করে, যেমন মিডিয়া ফাইলগুলিতে লেখা, এটি প্রক্রিয়াকরণের সময় ফাইলটিতে একচেটিয়া অ্যাক্সেস থাকা দরকারী৷ যে ডিভাইসগুলিতে Android 10 বা তার বেশি চলে, আপনার অ্যাপটি IS_PENDING পতাকার মান 1-এ সেট করে এই একচেটিয়া অ্যাক্সেস পেতে পারে। আপনার অ্যাপটি IS_PENDING এর মান 0-তে পরিবর্তন না করা পর্যন্ত শুধুমাত্র আপনার অ্যাপটি ফাইলটি দেখতে পারে।

নিম্নলিখিত কোড স্নিপেট পূর্ববর্তী কোড স্নিপেটে তৈরি করে। এই স্নিপেটটি দেখায় কিভাবে IS_PENDING পতাকা ব্যবহার করতে হয় যখন MediaStore.Audio সংগ্রহের সাথে সম্পর্কিত ডিরেক্টরিতে একটি দীর্ঘ গান সংরক্ষণ করা হয়:

কোটলিন

// Add a media item that other apps don't see until the item is
// fully written to the media store.
val resolver = applicationContext.contentResolver

// Find all audio files on the primary external storage device.
val audioCollection =
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        MediaStore.Audio.Media.getContentUri(
            MediaStore.VOLUME_EXTERNAL_PRIMARY
        )
    } else {
        MediaStore.Audio.Media.EXTERNAL_CONTENT_URI
    }

val songDetails = ContentValues().apply {
    put(MediaStore.Audio.Media.DISPLAY_NAME, "My Workout Playlist.mp3")
    put(MediaStore.Audio.Media.IS_PENDING, 1)
}

val songContentUri = resolver.insert(audioCollection, songDetails)

// "w" for write.
resolver.openFileDescriptor(songContentUri, "w", null).use { pfd ->
    // Write data into the pending audio file.
}

// Now that you're finished, release the "pending" status and let other apps
// play the audio track.
songDetails.clear()
songDetails.put(MediaStore.Audio.Media.IS_PENDING, 0)
resolver.update(songContentUri, songDetails, null, null)

জাভা

// Add a media item that other apps don't see until the item is
// fully written to the media store.
ContentResolver resolver = getApplicationContext()
        .getContentResolver();

// Find all audio files on the primary external storage device.
Uri audioCollection;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
    audioCollection = MediaStore.Audio.Media
            .getContentUri(MediaStore.VOLUME_EXTERNAL_PRIMARY);
} else {
    audioCollection = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
}

ContentValues songDetails = new ContentValues();
songDetails.put(MediaStore.Audio.Media.DISPLAY_NAME,
        "My Workout Playlist.mp3");
songDetails.put(MediaStore.Audio.Media.IS_PENDING, 1);

Uri songContentUri = resolver
        .insert(audioCollection, songDetails);

// "w" for write.
try (ParcelFileDescriptor pfd =
        resolver.openFileDescriptor(songContentUri, "w", null)) {
    // Write data into the pending audio file.
}

// Now that you're finished, release the "pending" status and let other apps
// play the audio track.
songDetails.clear();
songDetails.put(MediaStore.Audio.Media.IS_PENDING, 0);
resolver.update(songContentUri, songDetails, null, null);

ফাইল অবস্থানের জন্য একটি ইঙ্গিত দিন

যখন আপনার অ্যাপ অ্যান্ড্রয়েড 10 চালিত একটি ডিভাইসে মিডিয়া সঞ্চয় করে, তখন ডিফল্টভাবে মিডিয়া তার প্রকারের উপর ভিত্তি করে সংগঠিত হয়। উদাহরণস্বরূপ, ডিফল্টভাবে নতুন ছবি ফাইলগুলি Environment.DIRECTORY_PICTURES ডিরেক্টরিতে স্থাপন করা হয়, যা MediaStore.Images সংগ্রহের সাথে মিলে যায়।

যদি আপনার অ্যাপটি একটি নির্দিষ্ট অবস্থান সম্পর্কে সচেতন থাকে যেখানে ফাইলগুলি সংরক্ষণ করা যেতে পারে, যেমন Pictures/MyVacationPictures নামক একটি ফটো অ্যালবাম, আপনি সিস্টেমটিকে নতুন লিখিত ফাইলগুলি কোথায় সংরক্ষণ করতে হবে তার একটি ইঙ্গিত প্রদান করতে MediaColumns.RELATIVE_PATH সেট করতে পারেন৷

একটি আইটেম আপডেট করুন

আপনার অ্যাপের মালিকানাধীন একটি মিডিয়া ফাইল আপডেট করতে, নিম্নলিখিতগুলির মতো কোড ব্যবহার করুন:

কোটলিন

// Updates an existing media item.
val mediaId = // MediaStore.Audio.Media._ID of item to update.
val resolver = applicationContext.contentResolver

// When performing a single item update, prefer using the ID.
val selection = "${MediaStore.Audio.Media._ID} = ?"

// By using selection + args you protect against improper escaping of // values.
val selectionArgs = arrayOf(mediaId.toString())

// Update an existing song.
val updatedSongDetails = ContentValues().apply {
    put(MediaStore.Audio.Media.DISPLAY_NAME, "My Favorite Song.mp3")
}

// Use the individual song's URI to represent the collection that's
// updated.
val numSongsUpdated = resolver.update(
        myFavoriteSongUri,
        updatedSongDetails,
        selection,
        selectionArgs)

জাভা

// Updates an existing media item.
long mediaId = // MediaStore.Audio.Media._ID of item to update.
ContentResolver resolver = getApplicationContext()
        .getContentResolver();

// When performing a single item update, prefer using the ID.
String selection = MediaStore.Audio.Media._ID + " = ?";

// By using selection + args you protect against improper escaping of
// values. Here, "song" is an in-memory object that caches the song's
// information.
String[] selectionArgs = new String[] { getId().toString() };

// Update an existing song.
ContentValues updatedSongDetails = new ContentValues();
updatedSongDetails.put(MediaStore.Audio.Media.DISPLAY_NAME,
        "My Favorite Song.mp3");

// Use the individual song's URI to represent the collection that's
// updated.
int numSongsUpdated = resolver.update(
        myFavoriteSongUri,
        updatedSongDetails,
        selection,
        selectionArgs);

স্কোপড স্টোরেজ অনুপলব্ধ হলে বা সক্ষম না হলে, পূর্ববর্তী কোড স্নিপেটে দেখানো প্রক্রিয়াটি আপনার অ্যাপের মালিকানাধীন নয় এমন ফাইলগুলির জন্যও কাজ করে।

নেটিভ কোডে আপডেট করুন

আপনি যদি নেটিভ লাইব্রেরি ব্যবহার করে মিডিয়া ফাইল লিখতে চান, তাহলে আপনার জাভা-ভিত্তিক বা কোটলিন-ভিত্তিক কোড থেকে আপনার নেটিভ কোডে ফাইলের সংশ্লিষ্ট ফাইল ডিসক্রিপ্টর পাস করুন।

নিম্নলিখিত কোড স্নিপেটটি দেখায় কিভাবে একটি মিডিয়া অবজেক্টের ফাইল বর্ণনাকারীকে আপনার অ্যাপের নেটিভ কোডে পাস করতে হয়:

কোটলিন

val contentUri: Uri = ContentUris.withAppendedId(
        MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
        cursor.getLong(BaseColumns._ID))
val fileOpenMode = "r"
val parcelFd = resolver.openFileDescriptor(contentUri, fileOpenMode)
val fd = parcelFd?.detachFd()
// Pass the integer value "fd" into your native code. Remember to call
// close(2) on the file descriptor when you're done using it.

জাভা

Uri contentUri = ContentUris.withAppendedId(
        MediaStore.Audio.Media.EXTERNAL_CONTENT_URI,
        cursor.getLong(Integer.parseInt(BaseColumns._ID)));
String fileOpenMode = "r";
ParcelFileDescriptor parcelFd =
        resolver.openFileDescriptor(contentUri, fileOpenMode);
if (parcelFd != null) {
    int fd = parcelFd.detachFd();
    // Pass the integer value "fd" into your native code. Remember to call
    // close(2) on the file descriptor when you're done using it.
}

অন্যান্য অ্যাপের মিডিয়া ফাইল আপডেট করুন

যদি আপনার অ্যাপ স্কোপড স্টোরেজ ব্যবহার করে, তবে এটি সাধারণত একটি মিডিয়া ফাইল আপডেট করতে পারে না যা মিডিয়া স্টোরে একটি ভিন্ন অ্যাপ অবদান রাখে।

আপনি ফাইলটি পরিবর্তন করার জন্য ব্যবহারকারীর সম্মতি পেতে পারেন, তবে, প্ল্যাটফর্মটি থ্রো করা RecoverableSecurityException ক্যাচ করে। তারপরে আপনি অনুরোধ করতে পারেন যে ব্যবহারকারী আপনার অ্যাপটিকে সেই নির্দিষ্ট আইটেমে লেখার অ্যাক্সেস মঞ্জুর করুন, যেমনটি নিম্নলিখিত কোড স্নিপেটে দেখানো হয়েছে:

কোটলিন

// Apply a grayscale filter to the image at the given content URI.
try {
    // "w" for write.
    contentResolver.openFileDescriptor(image-content-uri, "w")?.use {
        setGrayscaleFilter(it)
    }
} catch (securityException: SecurityException) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        val recoverableSecurityException = securityException as?
            RecoverableSecurityException ?:
            throw RuntimeException(securityException.message, securityException)

        val intentSender =
            recoverableSecurityException.userAction.actionIntent.intentSender
        intentSender?.let {
            startIntentSenderForResult(intentSender, image-request-code,
                    null, 0, 0, 0, null)
        }
    } else {
        throw RuntimeException(securityException.message, securityException)
    }
}

জাভা

try {
    // "w" for write.
    ParcelFileDescriptor imageFd = getContentResolver()
            .openFileDescriptor(image-content-uri, "w");
    setGrayscaleFilter(imageFd);
} catch (SecurityException securityException) {
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
        RecoverableSecurityException recoverableSecurityException;
        if (securityException instanceof RecoverableSecurityException) {
            recoverableSecurityException =
                    (RecoverableSecurityException)securityException;
        } else {
            throw new RuntimeException(
                    securityException.getMessage(), securityException);
        }
        IntentSender intentSender =recoverableSecurityException.getUserAction()
                .getActionIntent().getIntentSender();
        startIntentSenderForResult(intentSender, image-request-code,
                null, 0, 0, 0, null);
    } else {
        throw new RuntimeException(
                securityException.getMessage(), securityException);
    }
}

এই প্রক্রিয়াটি প্রতিবার সম্পূর্ণ করুন যখন আপনার অ্যাপের এমন একটি মিডিয়া ফাইল সংশোধন করতে হবে যা এটি তৈরি করেনি।

বিকল্পভাবে, যদি আপনার অ্যাপ অ্যান্ড্রয়েড 11 বা উচ্চতর সংস্করণে চলে, তাহলে আপনি ব্যবহারকারীদের আপনার অ্যাপকে মিডিয়া ফাইলের একটি গ্রুপে লেখার অ্যাক্সেস দিতে দিতে পারেন। createWriteRequest() পদ্ধতিটি ব্যবহার করুন, যেমন মিডিয়া ফাইলের গ্রুপগুলি কীভাবে পরিচালনা করবেন সে সম্পর্কে বিভাগে বর্ণিত হয়েছে।

যদি আপনার অ্যাপের অন্য একটি ব্যবহারের ক্ষেত্রে থাকে যা স্কোপড স্টোরেজ দ্বারা আচ্ছাদিত না হয়, তাহলে একটি বৈশিষ্ট্য অনুরোধ ফাইল করুন এবং সাময়িকভাবে স্কোপড স্টোরেজ থেকে অপ্ট আউট করুন

একটি আইটেম সরান

মিডিয়া স্টোরে আপনার অ্যাপের আর প্রয়োজন নেই এমন একটি আইটেম সরাতে, নিম্নলিখিত কোড স্নিপেটে যা দেখানো হয়েছে তার মতো যুক্তি ব্যবহার করুন:

কোটলিন

// Remove a specific media item.
val resolver = applicationContext.contentResolver

// URI of the image to remove.
val imageUri = "..."

// WHERE clause.
val selection = "..."
val selectionArgs = "..."

// Perform the actual removal.
val numImagesRemoved = resolver.delete(
        imageUri,
        selection,
        selectionArgs)

জাভা

// Remove a specific media item.
ContentResolver resolver = getApplicationContext()
        getContentResolver();

// URI of the image to remove.
Uri imageUri = "...";

// WHERE clause.
String selection = "...";
String[] selectionArgs = "...";

// Perform the actual removal.
int numImagesRemoved = resolver.delete(
        imageUri,
        selection,
        selectionArgs);

স্কোপড স্টোরেজ অনুপলব্ধ হলে বা সক্ষম না থাকলে, আপনি অন্য অ্যাপের মালিকানাধীন ফাইলগুলি সরাতে আগের কোড স্নিপেট ব্যবহার করতে পারেন। যদি স্কোপড স্টোরেজ সক্ষম করা থাকে, তবে, মিডিয়া আইটেমগুলি আপডেট করার বিষয়ে বিভাগে বর্ণিত হিসাবে আপনার অ্যাপ সরাতে চায় এমন প্রতিটি ফাইলের জন্য আপনাকে একটি RecoverableSecurityException ধরতে হবে।

যদি আপনার অ্যাপ অ্যান্ড্রয়েড 11 বা উচ্চতর সংস্করণে চলে, তাহলে আপনি ব্যবহারকারীদের সরানোর জন্য মিডিয়া ফাইলের একটি গ্রুপ বেছে নিতে দিতে পারেন। createTrashRequest() পদ্ধতি বা createDeleteRequest() পদ্ধতি ব্যবহার করুন, মিডিয়া ফাইলগুলির গ্রুপগুলি কীভাবে পরিচালনা করবেন সে সম্পর্কে বিভাগে বর্ণিত।

যদি আপনার অ্যাপের অন্য একটি ব্যবহারের ক্ষেত্রে থাকে যা স্কোপড স্টোরেজ দ্বারা আচ্ছাদিত না হয়, তাহলে একটি বৈশিষ্ট্য অনুরোধ ফাইল করুন এবং সাময়িকভাবে স্কোপড স্টোরেজ থেকে অপ্ট আউট করুন

মিডিয়া ফাইলের আপডেট সনাক্ত করুন

আপনার অ্যাপের স্টোরেজ ভলিউম শনাক্ত করতে হতে পারে মিডিয়া ফাইল যা অ্যাপ যোগ করা বা পরিবর্তন করা হয়েছে, আগের সময়ের তুলনায়। এই পরিবর্তনগুলি সবচেয়ে নির্ভরযোগ্যভাবে সনাক্ত করতে, আগ্রহের স্টোরেজ ভলিউম getGeneration() এ পাস করুন। যতক্ষণ পর্যন্ত মিডিয়া স্টোর সংস্করণ পরিবর্তন না হয়, এই পদ্ধতির রিটার্ন মান সময়ের সাথে একঘেয়েভাবে বৃদ্ধি পায়।

বিশেষ করে, getGeneration() মিডিয়া কলামের তারিখের চেয়ে বেশি শক্তিশালী, যেমন DATE_ADDED এবং DATE_MODIFIED । কারণ যখন কোনো অ্যাপ setLastModified() কল করে বা ব্যবহারকারী যখন সিস্টেম ঘড়ি পরিবর্তন করে তখন মিডিয়া কলামের মান পরিবর্তন হতে পারে।

মিডিয়া ফাইলের গ্রুপ পরিচালনা করুন

অ্যান্ড্রয়েড 11 এবং উচ্চতর সংস্করণে, আপনি ব্যবহারকারীকে মিডিয়া ফাইলগুলির একটি গ্রুপ নির্বাচন করতে বলতে পারেন, তারপর এই মিডিয়া ফাইলগুলিকে একক অপারেশনে আপডেট করতে পারেন। এই পদ্ধতিগুলি ডিভাইস জুড়ে আরও ভাল সামঞ্জস্য অফার করে এবং পদ্ধতিগুলি ব্যবহারকারীদের তাদের মিডিয়া সংগ্রহগুলি পরিচালনা করা সহজ করে তোলে।

এই "ব্যাচ আপডেট" কার্যকারিতা প্রদানকারী পদ্ধতিগুলির মধ্যে নিম্নলিখিতগুলি অন্তর্ভুক্ত রয়েছে:

createWriteRequest()
অনুরোধ করুন যে ব্যবহারকারী আপনার অ্যাপটিকে মিডিয়া ফাইলগুলির নির্দিষ্ট গ্রুপে লেখার অ্যাক্সেস মঞ্জুর করুন৷
createFavoriteRequest()
অনুরোধ করুন যে ব্যবহারকারী নির্দিষ্ট মিডিয়া ফাইলগুলিকে ডিভাইসে তাদের কিছু "প্রিয়" মিডিয়া হিসাবে চিহ্নিত করুন৷ এই ফাইলটিতে পড়ার অ্যাক্সেস আছে এমন যেকোনো অ্যাপ দেখতে পাবে যে ব্যবহারকারী ফাইলটিকে "প্রিয়" হিসেবে চিহ্নিত করেছেন।
createTrashRequest()

অনুরোধ করুন যে ব্যবহারকারী নির্দিষ্ট মিডিয়া ফাইলগুলি ডিভাইসের ট্র্যাশে রাখুন৷ ট্র্যাশে থাকা আইটেমগুলি একটি সিস্টেম-সংজ্ঞায়িত সময়ের পরে স্থায়ীভাবে মুছে ফেলা হয়।

createDeleteRequest()

অনুরোধ করুন যে ব্যবহারকারী নির্দিষ্ট মিডিয়া ফাইলগুলিকে আগে থেকে ট্র্যাশে না রেখে অবিলম্বে স্থায়ীভাবে মুছে ফেলুন৷

এই পদ্ধতিগুলির যেকোনো একটি কল করার পরে, সিস্টেমটি একটি PendingIntent অবজেক্ট তৈরি করে। আপনার অ্যাপ্লিকেশানটি এই অভিপ্রায়কে আহ্বান করার পরে, ব্যবহারকারীরা একটি ডায়ালগ দেখতে পান যা নির্দিষ্ট মিডিয়া ফাইলগুলিকে আপডেট বা মুছে ফেলার জন্য আপনার অ্যাপের জন্য তাদের সম্মতির অনুরোধ করে৷

উদাহরণস্বরূপ, createWriteRequest() এর জন্য একটি কল কীভাবে গঠন করা যায় তা এখানে রয়েছে:

কোটলিন

val urisToModify = /* A collection of content URIs to modify. */
val editPendingIntent = MediaStore.createWriteRequest(contentResolver,
        urisToModify)

// Launch a system prompt requesting user permission for the operation.
startIntentSenderForResult(editPendingIntent.intentSender, EDIT_REQUEST_CODE,
    null, 0, 0, 0)

জাভা

List<Uri> urisToModify = /* A collection of content URIs to modify. */
PendingIntent editPendingIntent = MediaStore.createWriteRequest(contentResolver,
                  urisToModify);

// Launch a system prompt requesting user permission for the operation.
startIntentSenderForResult(editPendingIntent.getIntentSender(),
    EDIT_REQUEST_CODE, null, 0, 0, 0);

ব্যবহারকারীর প্রতিক্রিয়া মূল্যায়ন. ব্যবহারকারী সম্মতি প্রদান করলে, মিডিয়া অপারেশনের সাথে এগিয়ে যান। অন্যথায়, কেন আপনার অ্যাপটির অনুমতি প্রয়োজন তা ব্যবহারকারীকে ব্যাখ্যা করুন:

কোটলিন

override fun onActivityResult(requestCode: Int, resultCode: Int,
                 data: Intent?) {
    ...
    when (requestCode) {
        EDIT_REQUEST_CODE ->
            if (resultCode == Activity.RESULT_OK) {
                /* Edit request granted; proceed. */
            } else {
                /* Edit request not granted; explain to the user. */
            }
    }
}

জাভা

@Override
protected void onActivityResult(int requestCode, int resultCode,
                   @Nullable Intent data) {
    ...
    if (requestCode == EDIT_REQUEST_CODE) {
        if (resultCode == Activity.RESULT_OK) {
            /* Edit request granted; proceed. */
        } else {
            /* Edit request not granted; explain to the user. */
        }
    }
}

আপনি createFavoriteRequest() , createTrashRequest() , এবং createDeleteRequest() এর সাথে এই একই সাধারণ প্যাটার্ন ব্যবহার করতে পারেন।

মিডিয়া পরিচালনার অনুমতি

ব্যবহারকারীরা মিডিয়া পরিচালনা করার জন্য একটি নির্দিষ্ট অ্যাপকে বিশ্বাস করতে পারে, যেমন মিডিয়া ফাইলগুলিতে ঘন ঘন সম্পাদনা করা। যদি আপনার অ্যাপটি Android 11 বা তার উচ্চতরকে লক্ষ্য করে এবং এটি ডিভাইসের ডিফল্ট গ্যালারি অ্যাপ না হয়, তাহলে প্রতিবার আপনার অ্যাপ যখন কোনো ফাইল পরিবর্তন বা মুছে ফেলার চেষ্টা করে তখন আপনাকে অবশ্যই ব্যবহারকারীকে একটি নিশ্চিতকরণ ডায়ালগ দেখাতে হবে।

যদি আপনার অ্যাপটি Android 12 (API লেভেল 31) বা উচ্চতরকে টার্গেট করে, তাহলে আপনি অনুরোধ করতে পারেন যে ব্যবহারকারীরা মিডিয়া ম্যানেজমেন্টের বিশেষ অনুমতিতে আপনার অ্যাপ অ্যাক্সেস মঞ্জুর করুন। এই অনুমতি আপনার অ্যাপটিকে প্রতিটি ফাইল অপারেশনের জন্য ব্যবহারকারীকে অনুরোধ না করেই নিম্নলিখিতগুলির প্রতিটি করতে দেয়:

  • createWriteRequest() ব্যবহার করে ফাইলগুলি পরিবর্তন করুন।
  • createTrashRequest() ব্যবহার করে ফাইলগুলিকে ট্র্যাশে এবং বাইরে সরান।
  • createDeleteRequest() ব্যবহার করে ফাইল মুছুন।

এটি করতে, নিম্নলিখিত পদক্ষেপগুলি সম্পূর্ণ করুন:

  1. আপনার অ্যাপের ম্যানিফেস্ট ফাইলে MANAGE_MEDIA অনুমতি এবং READ_EXTERNAL_STORAGE অনুমতি ঘোষণা করুন।

    একটি নিশ্চিতকরণ ডায়ালগ না দেখিয়ে createWriteRequest() এ কল করতে, ACCESS_MEDIA_LOCATION অনুমতিও ঘোষণা করুন৷

  2. আপনার অ্যাপে, ব্যবহারকারী কেন আপনার অ্যাপে মিডিয়া ম্যানেজমেন্ট অ্যাক্সেস দিতে চান তা ব্যাখ্যা করতে একটি UI দেখান।

  3. ACTION_REQUEST_MANAGE_MEDIA অভিপ্রায় ক্রিয়া আহ্বান করুন৷ এটি ব্যবহারকারীদের সিস্টেম সেটিংসে মিডিয়া ম্যানেজমেন্ট অ্যাপের স্ক্রিনে নিয়ে যায়। এখান থেকে, ব্যবহারকারীরা বিশেষ অ্যাপ অ্যাক্সেস মঞ্জুর করতে পারেন।

মিডিয়া স্টোরের বিকল্প প্রয়োজন এমন ক্ষেত্রে ব্যবহার করুন

যদি আপনার অ্যাপ প্রাথমিকভাবে নিম্নলিখিত ভূমিকাগুলির মধ্যে একটি সম্পাদন করে, তাহলে MediaStore API-এর বিকল্প বিবেচনা করুন।

অন্যান্য ধরনের ফাইলের সাথে কাজ করা

যদি আপনার অ্যাপ এমন নথি এবং ফাইলগুলির সাথে কাজ করে যেগুলিতে একচেটিয়াভাবে মিডিয়া সামগ্রী থাকে না, যেমন ফাইলগুলি যেগুলি EPUB বা PDF ফাইল এক্সটেনশন ব্যবহার করে, তাহলে ACTION_OPEN_DOCUMENT উদ্দেশ্য অ্যাকশনটি ব্যবহার করুন, যেমন নথি এবং অন্যান্য ফাইলগুলি সংরক্ষণ এবং অ্যাক্সেস করার নির্দেশিকাতে বর্ণিত হয়েছে৷

সহচর অ্যাপে ফাইল শেয়ারিং

যে ক্ষেত্রে আপনি একটি সঙ্গী অ্যাপের স্যুট প্রদান করেন, যেমন একটি মেসেজিং অ্যাপ এবং একটি প্রোফাইল অ্যাপ, content:// URI ব্যবহার করে ফাইল শেয়ারিং সেট আপ করুন । আমরা নিরাপত্তার সর্বোত্তম অনুশীলন হিসাবে এই কর্মপ্রবাহের সুপারিশ করি৷

অতিরিক্ত সম্পদ

কিভাবে মিডিয়া সঞ্চয় এবং অ্যাক্সেস করতে হয় সে সম্পর্কে আরও তথ্যের জন্য, নিম্নলিখিত সংস্থানগুলি দেখুন৷

নমুনা

ভিডিও