ExoPlayer নিম্নলিখিত চিত্র বিন্যাস সমর্থন করে। বাহ্যিক লাইব্রেরিগুলির সাথে কীভাবে একীভূত করা যায় তার জন্য চিত্র লোডিং লাইব্রেরিগুলি দেখুন যা ফর্ম্যাটের একটি ভিন্ন সেটের জন্য সমর্থন প্রদান করতে পারে।
চিত্র বিন্যাস | সমর্থিত | নোট |
---|---|---|
বিএমপি | হ্যাঁ | |
জিআইএফ | না | কোনও এক্সট্র্যাক্টর সমর্থন নেই |
জেপিইজি | হ্যাঁ | |
জেপিইজি মোশন ফটো | হ্যাঁ | স্থির চিত্র এবং ভিডিও সমর্থিত |
জেপিইজি আল্ট্রা এইচডিআর | হ্যাঁ | Android 14 এর আগে বা নন-HDR ডিসপ্লেতে SDR-এ ফিরে যায় |
পিএনজি | হ্যাঁ | |
ওয়েবপি | হ্যাঁ | |
HEIF/HEIC | হ্যাঁ | |
HIIC গতি ফটো | আংশিকভাবে | শুধুমাত্র স্থির চিত্র সমর্থিত* |
আভিফ (বেসলাইন) | হ্যাঁ | শুধুমাত্র Android 14+ এ ডিকোড করা হয়েছে |
* HEIC মোশন ফটোগুলির ভিডিও অংশ মেটাডেটা রিট্রিভারের সাথে প্রাপ্ত করা যেতে পারে এবং একটি স্বতন্ত্র ফাইল হিসাবে চালানো যেতে পারে।
মিডিয়া আইটেম ব্যবহার করে
একটি প্লেলিস্টের অংশ হিসাবে একটি ইমেজ প্লে করতে, ইমেজ ইউআরআই সহ একটি MediaItem
তৈরি করুন এবং এটি প্লেয়ারে পাঠান৷ MediaItem
একটি imageDurationMs
থাকতে হবে যাতে ইমেজটি কতক্ষণ প্রদর্শিত হবে তা উল্লেখ করতে হবে।
কোটলিন
// Create a player instance. val player = ExoPlayer.Builder(context).build() // Set the media item to be played with the desired duration. player.setMediaItem( MediaItem.Builder().setUri(imageUri).setImageDurationMs(2000).build()) // Prepare the player. player.prepare()
জাভা
// Create a player instance. ExoPlayer player = new ExoPlayer.Builder(context).build(); // Set the media item to be played with the desired duration. player.setMediaItem( new MediaItem.Builder().setUri(imageUri).setImageDurationMs(2000).build()); // Prepare the player. player.prepare();
গতি ফটো
মোশন ফটো হল একটি ছোট ভিডিওর সাথে একটি স্থির চিত্রকে একত্রিত করা ফাইল।
- যদি ছবির সময়কাল
setImageDuration
দিয়ে সংজ্ঞায়িত করা হয়, মোশন ফটোটি ঘোষিত সময়কালের জন্য একটি স্থির চিত্র হিসাবে প্রদর্শিত হয়। - ছবির সময়কাল যদি অনির্ধারিত থাকে, তাহলে মোশন ফটো ভিডিও হিসেবে চালানো হয়।
প্রগ্রেসিভ মিডিয়া সোর্স ব্যবহার করে
আরও কাস্টমাইজেশন বিকল্পের জন্য, আপনি একটি ProgressiveMediaSource
তৈরি করতে পারেন এবং একটি MediaItem
এর পরিবর্তে সরাসরি প্লেয়ারে পাঠাতে পারেন।
কোটলিন
// Create a data source factory. val dataSourceFactory = DefaultHttpDataSource.Factory() // Create a media item with the image URI and the desired duration. val mediaItem = MediaItem.Builder().setUri(imageUri).setImageDurationMs(2000).build() // Create a progressive media source for this media item. val mediaSource = ProgressiveMediaSource.Factory(dataSourceFactory) .createMediaSource(mediaItem) // Create a player instance. val player = ExoPlayer.Builder(context).build() // Set the media source to be played. player.setMediaSource(mediaSource) // Prepare the player. player.prepare()
জাভা
// Create a data source factory. DataSource.Factory dataSourceFactory = new DefaultHttpDataSource.Factory(); // Create a media item with the image URI and the desired duration. MediaItem mediaItem = new MediaItem.Builder().setUri(imageUri).setImageDurationMs(2000).build(); // Create a progressive media source for this media item. MediaSource mediaSource = new ProgressiveMediaSource.Factory(dataSourceFactory) .createMediaSource(mediaItem); // Create a player instance. ExoPlayer player = new ExoPlayer.Builder(context).build(); // Set the media source to be played. player.setMediaSource(mediaSource); // Prepare the player. player.prepare();
প্লেব্যাক কাস্টমাইজিং
ExoPlayer আপনার অ্যাপের প্রয়োজন অনুযায়ী প্লেব্যাকের অভিজ্ঞতার জন্য একাধিক উপায় প্রদান করে। উদাহরণের জন্য কাস্টমাইজেশন পৃষ্ঠা দেখুন।
চিত্র লোডিং লাইব্রেরি
চিত্রগুলি প্রায়শই বহিরাগত চিত্র লোডিং লাইব্রেরি দ্বারা পরিচালিত হয়, উদাহরণস্বরূপ গ্লাইড বা কয়েল ।
এই লাইব্রেরিগুলিকে প্লেব্যাক পাইপলাইনে একত্রিত করার জন্য 3টি ধাপ প্রয়োজন:
-
APPLICATION_EXTERNALLY_LOADED_IMAGE
MIME প্রকার সহ একটিMediaItem
সংজ্ঞায়িত করুন৷ - ইমেজ লোডিং লাইব্রেরি থেকে একটি
Bitmap
পুনরুদ্ধার করতে একটি ইমেজ ডিকোডার প্রদান করুন। - ক্যাশিং এবং প্রিলোডিং ট্রিগার করতে একটি বহিরাগত লোডার প্রদান করুন।
বাহ্যিকভাবে লোড করা ছবি MIME প্রকার সহ MediaItem৷
ইমেজ লোডিং লাইব্রেরি কোড পাথগুলি ব্যবহার করার জন্য Player
যোগ করা MediaItem
অবশ্যই APPLICATION_EXTERNALLY_LOADED_IMAGE
MIME প্রকার স্পষ্টভাবে সংজ্ঞায়িত করতে হবে:
কোটলিন
val mediaItem = MediaItem.Builder() .setUri(imageUri) .setMimeType(MimeTypes.APPLICATION_EXTERNALLY_LOADED_IMAGE) .build()
জাভা
MediaItem mediaItem = new MediaItem.Builder() .setUri(imageUri) .setMimeType(MimeTypes.APPLICATION_EXTERNALLY_LOADED_IMAGE) .build();
একটি ইমেজ লোডিং লাইব্রেরি ব্যবহার করে ইমেজ ডিকোডার
ImageRenderer
একটি ইমেজ ইউআরআই-এর জন্য Bitmap
পুনরুদ্ধার করতে ImageDecoder
উদাহরণ ব্যবহার করে। এই ডিকোডারটি বাহ্যিক চিত্র লোডিং লাইব্রেরি ব্যবহার করার জন্য লেখা যেতে পারে, যেমনটি গ্লাইডের সাথে নিম্নলিখিত উদাহরণে দেখানো হয়েছে:
কোটলিন
val glideImageDecoder: ImageDecoder = object : ImageDecoder { private val inputBuffer = DecoderInputBuffer(DecoderInputBuffer.BUFFER_REPLACEMENT_MODE_NORMAL) private val outputBuffer: ImageOutputBuffer = object : ImageOutputBuffer() { override fun release() { clear() bitmap = null } } private var pendingDecode: AtomicBoolean? = null private var decodeError: ImageDecoderException? = null override fun dequeueInputBuffer(): DecoderInputBuffer? { return if (pendingDecode == null) inputBuffer else null } override fun queueInputBuffer(inputBuffer: DecoderInputBuffer) { if (inputBuffer.isEndOfStream) { outputBuffer.addFlag(C.BUFFER_FLAG_END_OF_STREAM) inputBuffer.clear() return } val currentDecode = AtomicBoolean(true) pendingDecode = currentDecode val imageUri = Uri.parse( String( inputBuffer.data!!.array(), inputBuffer.data!!.position(), inputBuffer.data!!.limit() - inputBuffer.data!!.position(), Charsets.UTF_8, ) ) val imageTimeUs = inputBuffer.timeUs Glide.with(context) .asBitmap() .load(imageUri) .into( object : CustomTarget<Bitmap?>() { override fun onResourceReady( resource: Bitmap, transition: Transition<in Bitmap?>?, ) { if (currentDecode.get()) { outputBuffer.timeUs = imageTimeUs outputBuffer.bitmap = resource pendingDecode = null } } override fun onLoadFailed(errorDrawable: Drawable?) { if (currentDecode.get()) { decodeError = ImageDecoderException("Glide load failed") } } override fun onLoadCleared(placeholder: Drawable?) {} } ) inputBuffer.clear() } @Throws(ImageDecoderException::class) override fun dequeueOutputBuffer(): ImageOutputBuffer? { if (decodeError != null) { throw decodeError as ImageDecoderException } val hasOutput = (pendingDecode == null && (outputBuffer.bitmap != null || outputBuffer.isEndOfStream)) return if (hasOutput) outputBuffer else null } override fun getName(): String { return "glideDecoder" } override fun setOutputStartTimeUs(outputStartTimeUs: Long) {} override fun flush() { if (pendingDecode != null) { pendingDecode!!.set(false) pendingDecode = null } decodeError = null inputBuffer.clear() outputBuffer.release() } override fun release() { flush() } } val glideImageDecoderFactory: ImageDecoder.Factory = object : ImageDecoder.Factory { override fun supportsFormat(format: Format): @RendererCapabilities.Capabilities Int { val isExternalImageUrl = format.sampleMimeType != null && format.sampleMimeType == MimeTypes.APPLICATION_EXTERNALLY_LOADED_IMAGE return RendererCapabilities.create( if (isExternalImageUrl) C.FORMAT_HANDLED else C.FORMAT_UNSUPPORTED_TYPE ) } override fun createImageDecoder(): ImageDecoder { return glideImageDecoder } } val player: Player = ExoPlayer.Builder(context) .setRenderersFactory( object : DefaultRenderersFactory(context) { override fun buildImageRenderers(out: ArrayList<Renderer>) { out.add( ImageRenderer(glideImageDecoderFactory, /* imageOutput= */ null)) } } ) .build()
জাভা
ImageDecoder glideImageDecoder = new ImageDecoder() { private final DecoderInputBuffer inputBuffer = new DecoderInputBuffer(BUFFER_REPLACEMENT_MODE_NORMAL); private final ImageOutputBuffer outputBuffer = new ImageOutputBuffer() { @Override public void release() { clear(); bitmap = null; } }; @Nullable private AtomicBoolean pendingDecode; @Nullable private ImageDecoderException decodeError; @Nullable @Override public DecoderInputBuffer dequeueInputBuffer() { return pendingDecode == null ? inputBuffer : null; } @Override public void queueInputBuffer(DecoderInputBuffer inputBuffer) { if (inputBuffer.isEndOfStream()) { outputBuffer.addFlag(C.BUFFER_FLAG_END_OF_STREAM); inputBuffer.clear(); return; } AtomicBoolean currentDecode = new AtomicBoolean(true); pendingDecode = currentDecode; Uri imageUri = Uri.parse( new String( inputBuffer.data.array(), inputBuffer.data.position(), inputBuffer.data.limit() - inputBuffer.data.position(), Charsets.UTF_8)); long imageTimeUs = inputBuffer.timeUs; Glide.with(context) .asBitmap() .load(imageUri) .into( new CustomTarget<Bitmap>() { @Override public void onResourceReady( Bitmap resource, @Nullable Transition<? super Bitmap> transition) { if (currentDecode.get()) { outputBuffer.timeUs = imageTimeUs; outputBuffer.bitmap = resource; pendingDecode = null; } } @Override public void onLoadFailed(@Nullable Drawable errorDrawable) { if (currentDecode.get()) { decodeError = new ImageDecoderException("Glide load failed"); } } @Override public void onLoadCleared(@Nullable Drawable placeholder) {} }); inputBuffer.clear(); } @Nullable @Override public ImageOutputBuffer dequeueOutputBuffer() throws ImageDecoderException { if (decodeError != null) { throw decodeError; } boolean hasOutput = pendingDecode == null && (outputBuffer.bitmap != null || outputBuffer.isEndOfStream()); return hasOutput ? outputBuffer : null; } @Override public String getName() { return "glideDecoder"; } @Override public void setOutputStartTimeUs(long outputStartTimeUs) {} @Override public void flush() { if (pendingDecode != null) { pendingDecode.set(false); pendingDecode = null; } decodeError = null; inputBuffer.clear(); outputBuffer.release(); } @Override public void release() { flush(); } }; ImageDecoder.Factory glideImageDecoderFactory = new ImageDecoder.Factory() { @Override public @RendererCapabilities.Capabilities int supportsFormat( Format format) { boolean isExternalImageUrl = format.sampleMimeType != null && format.sampleMimeType.equals( MimeTypes.APPLICATION_EXTERNALLY_LOADED_IMAGE); return RendererCapabilities.create( isExternalImageUrl ? C.FORMAT_HANDLED : C.FORMAT_UNSUPPORTED_TYPE); } @Override public ImageDecoder createImageDecoder() { return glideImageDecoder; } }; Player player = new ExoPlayer.Builder(context) .setRenderersFactory( new DefaultRenderersFactory(context) { @Override protected void buildImageRenderers(ArrayList<Renderer> out) { out.add( new ImageRenderer( glideImageDecoderFactory, /* imageOutput= */ null)); } }) .build();
একটি ইমেজ লোডিং লাইব্রেরি সহ ইমেজ প্রিলোডিং
প্লেব্যাকের সময়, প্লেলিস্টের পূর্ববর্তী আইটেমটি সম্পূর্ণরূপে লোড হয়ে গেলে প্লেয়ার পরবর্তী চিত্রটি প্রিলোড করার অনুরোধ করে। একটি বহিরাগত ইমেজ লোডিং লাইব্রেরি ব্যবহার করার সময়, এই প্রিলোডিং ট্রিগার করতে আপনাকে অবশ্যই একটি ExternalLoader
নির্দিষ্ট করতে হবে৷ যদি কোন প্রিলোডিং সম্ভব না হয় বা প্রয়োজন হয়, এই লোডারটি এখনও প্রদান করা প্রয়োজন, কিন্তু কিছুই করতে পারে না।
অনুরোধ করা চিত্রটি ডিস্কে আগে থেকে লোড করা হয়েছে তা নিশ্চিত করতে নিম্নলিখিত উদাহরণটি গ্লাইড ব্যবহার করে:
কোটলিন
val glidePreloader = ExternalLoader { request: LoadRequest -> GlideFutures.submit( Glide.with(context) .asFile() .apply( RequestOptions.diskCacheStrategyOf(DiskCacheStrategy.DATA) .priority(Priority.HIGH) .skipMemoryCache(true) ) .load(request.uri) ) }
জাভা
ExternalLoader glidePreloader = request -> GlideFutures.submit( Glide.with(context) .asFile() .apply( diskCacheStrategyOf(DiskCacheStrategy.DATA) .priority(Priority.HIGH) .skipMemoryCache(true)) .load(request.uri));