একটি প্রিভিউ ভিডিও ব্যবহারকারীদেরকে আপনার টিভি অ্যাপে গভীর লিঙ্ক করতে উৎসাহিত করার একটি দুর্দান্ত উপায়৷ প্রিভিউগুলি ছোট ক্লিপ থেকে সম্পূর্ণ মুভি ট্রেলার পর্যন্ত হতে পারে।
আপনি একটি পূর্বরূপ তৈরি করার সময়, এই নির্দেশিকাগুলি বিবেচনা করুন:
- একটি পূর্বরূপ বিজ্ঞাপন প্রদর্শন করবেন না. আপনি যদি ক্লায়েন্ট সাইডে বিজ্ঞাপনগুলি সেলাই করেন তবে সেগুলিকে প্রিভিউ ভিডিওতে সেলাই করবেন না৷ আপনি সার্ভার সাইডে বিজ্ঞাপন স্টিচ করলে, পূর্বরূপ দেখার জন্য একটি বিজ্ঞাপন-মুক্ত ভিডিও প্রদান করুন।
- সেরা মানের জন্য, প্রিভিউ ভিডিও 16:9 বা 4:3 হওয়া উচিত। প্রিভিউ ভিডিওগুলির প্রস্তাবিত আকারের জন্য ভিডিও প্রোগ্রামের বৈশিষ্ট্যগুলি দেখুন৷
- যখন প্রিভিউ ভিডিও এবং পোস্টার আর্টের ভিন্ন আকৃতির অনুপাত থাকে, তখন প্রিভিউ চালানোর আগে হোম স্ক্রীন পোস্টার ভিউকে ভিডিওর অ্যাসপেক্ট রেশিওতে রিসাইজ করে। ভিডিওটি লেটারবক্সযুক্ত নয়। উদাহরণস্বরূপ, পোস্টার আর্ট অনুপাত যদি
ASPECT_RATIO_MOVIE_POSTER
(1:1.441) হয় কিন্তু ভিডিও অনুপাত 16:9 হয়, পোস্টার ভিউ একটি 16:9 অঞ্চলে রূপান্তরিত হয়৷ - আপনি যখন একটি পূর্বরূপ তৈরি করেন, তখন এর বিষয়বস্তু সর্বজনীনভাবে অ্যাক্সেসযোগ্য বা DRM-এর অধীনে সুরক্ষিত হতে পারে। প্রতিটি ক্ষেত্রে বিভিন্ন পদ্ধতি প্রযোজ্য। এই পৃষ্ঠা উভয় বর্ণনা.
হোম স্ক্রিনে প্রিভিউ চালান
আপনি যদি ExoPlayer দ্বারা সমর্থিত যেকোনো ভিডিও ব্যবহার করে একটি প্রিভিউ তৈরি করেন এবং প্রিভিউটি সর্বজনীনভাবে অ্যাক্সেসযোগ্য হয়, তাহলে আপনি সরাসরি হোম স্ক্রিনে প্রিভিউ চালাতে পারেন।
আপনি যখন একটি PreviewProgram তৈরি করেন তখন নিচের উদাহরণে দেখানো হিসাবে একটি সর্বজনীনভাবে অ্যাক্সেসযোগ্য HTTPS URL সহ setPreviewVideoUri()
ব্যবহার করুন। প্রিভিউ ভিডিও বা অডিও হতে পারে।
কোটলিন
val previewVideoUrl = Uri.parse("https://www.example.com/preview.mp4") val builder = PreviewProgram.Builder() builder.setChannelId(channelId) // ... .setPreviewVideoUri(previewVideoUrl)
জাভা
Uri previewVideoUrl = Uri.parse("https://www.example.com/preview.mp4"); PreviewProgram.Builder builder = new PreviewProgram.Builder(); builder.setChannelId(channelId) // ... .setPreviewVideoUri(Uri.parse(previewVideoUrl));
একটি পৃষ্ঠের উপর পূর্বরূপ রেন্ডার
যদি আপনার ভিডিওটি DRM-সুরক্ষিত হয় বা ExoPlayer দ্বারা সমর্থিত না হয় এমন মিডিয়া টাইপ, তাহলে একটি TvInputService
ব্যবহার করুন৷ Android TV হোম স্ক্রীন onSetSurface()
কল করে আপনার পরিষেবাতে একটি Surface
পাস করে। আপনার অ্যাপটি onTune()
থেকে সরাসরি এই পৃষ্ঠে ভিডিও আঁকে।
ডাইরেক্ট সারফেস রেন্ডারিং আপনার অ্যাপকে কী রেন্ডার করা হয় এবং কীভাবে রেন্ডার করা হয় তা নিয়ন্ত্রণ করতে দেয়। আপনি চ্যানেল অ্যাট্রিবিউশনের মতো মেটাডেটা ওভারলে করতে পারেন।
ম্যানিফেস্টে আপনার TvInputService ঘোষণা করুন
আপনার অ্যাপটিকে অবশ্যই TvInputService
এর একটি বাস্তবায়ন প্রদান করতে হবে যাতে হোম স্ক্রীন আপনার প্রিভিউ রেন্ডার করতে পারে।
আপনার পরিষেবা ঘোষণায়, একটি অভিপ্রায় ফিল্টার অন্তর্ভুক্ত করুন যা অভিপ্রায়ের সাথে সম্পাদন করার জন্য TvInputService
নির্দিষ্ট করে৷ এছাড়াও একটি পৃথক XML সম্পদ হিসাবে পরিষেবা মেটাডেটা ঘোষণা করুন। পরিষেবা ঘোষণা, অভিপ্রায় ফিল্টার এবং পরিষেবা মেটাডেটা ঘোষণা নিম্নলিখিত উদাহরণে দেখানো হয়েছে:
<service android:name=".rich.PreviewInputService" android:permission="android.permission.BIND_TV_INPUT"> <!-- Required filter used by the system to launch our account service. --> <intent-filter> <action android:name="android.media.tv.TvInputService" /> </intent-filter> <!-- An XML file which describes this input. --> <meta-data android:name="android.media.tv.input" android:resource="@xml/previewinputservice" /> </service>
একটি পৃথক XML ফাইলে পরিষেবা মেটাডেটা সংজ্ঞায়িত করুন। পরিষেবা মেটাডেটা ফাইলটি আপনার অ্যাপের জন্য XML সংস্থান ডিরেক্টরিতে অবস্থিত এবং আপনার ম্যানিফেস্টে ঘোষিত সংস্থানটির নামের সাথে অবশ্যই মিলবে৷ পূর্ববর্তী উদাহরণ থেকে ম্যানিফেস্ট এন্ট্রিগুলি ব্যবহার করে, আপনি একটি খালি tv-input
ট্যাগ সহ res/xml/previewinputservice.xml
এ একটি XML ফাইল তৈরি করবেন:
<?xml version="1.0" encoding="utf-8"?>
<tv-input/>
টিভি ইনপুট ফ্রেমওয়ার্কে অবশ্যই এই ট্যাগ থাকতে হবে। যাইহোক, এটি শুধুমাত্র লাইভ চ্যানেল কনফিগার করতে ব্যবহৃত হয়। যেহেতু আপনি একটি ভিডিও রেন্ডার করছেন, ট্যাগটি খালি হওয়া উচিত৷
একটি ভিডিও ইউআরআই তৈরি করুন
আপনার পূর্বরূপ ভিডিওটি Android TV হোম স্ক্রীনের পরিবর্তে আপনার অ্যাপ দ্বারা রেন্ডার করা উচিত তা নির্দেশ করার জন্য, আপনাকে একটি PreviewProgram
জন্য একটি ভিডিও URI তৈরি করতে হবে। আপনার অ্যাপ সামগ্রীর জন্য যে শনাক্তকারী ব্যবহার করে তা দিয়ে URI শেষ হওয়া উচিত, যাতে আপনি পরে TvInputService
এ সামগ্রীটি পুনরুদ্ধার করতে পারেন।
যদি আপনার শনাক্তকারী টাইপ হয় Long
, ব্যবহার করুন TvContractCompat.buildPreviewProgramUri() :
কোটলিন
val id: Long = 1L // content identifier val componentName = new ComponentName(context, PreviewVideoInputService.class) val previewProgramVideoUri = TvContractCompat.buildPreviewProgramUri(id) .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build()
জাভা
Long id = 1L; // content identifier ComponentName componentName = new ComponentName(context, PreviewVideoInputService.class); previewProgramVideoUri = TvContractCompat.buildPreviewProgramUri(id) .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build();
যদি আপনার শনাক্তকারী টাইপ না হয় Long
, Uri.withAppendedPath()
ব্যবহার করে URI তৈরি করুন:
কোটলিন
val previewProgramVideoUri = Uri.withAppendedPath(PreviewPrograms.CONTENT_URI, "content-identifier") .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build()
জাভা
previewProgramVideoUri = Uri.withAppendedPath(PreviewPrograms.CONTENT_URI, "content-identifier") .buildUpon() .appendQueryParameter("input", TvContractCompat.buildInputId(componentName)) .build();
অ্যান্ড্রয়েড টিভি প্রিভিউ ভিডিও শুরু করতে আপনার অ্যাপটি onTune(Uri videoUri)
কল করে।
একটি পরিষেবা তৈরি করুন
নিম্নলিখিত উদাহরণটি দেখায় কিভাবে আপনার নিজস্ব PreviewInputService
তৈরি করতে TvInputService
প্রসারিত করতে হয়। নোট করুন যে পরিষেবাটি প্লেব্যাকের জন্য একটি MediaPlayer
ব্যবহার করে, তবে আপনার কোড যেকোন উপলব্ধ ভিডিও প্লেয়ার ব্যবহার করতে পারে৷
কোটলিন
import android.content.Context import android.media.MediaPlayer import android.media.tv.TvInputService import android.net.Uri import android.util.Log import android.view.Surface import java.io.IOException class PreviewVideoInputService : TvInputService() { override fun onCreateSession(inputId: String): TvInputService.Session? { return PreviewSession(this) } private inner class PreviewSession( internal var context: Context ) : TvInputService.Session(context) { internal var mediaPlayer: MediaPlayer? = MediaPlayer() override fun onRelease() { mediaPlayer?.release() mediaPlayer = null } override fun onTune(uri: Uri): Boolean { // Let the TvInputService know that the video is being loaded. notifyVideoUnavailable(VIDEO_UNAVAILABLE_REASON_TUNING) // Fetch the stream url from the TV Provider database // for content://android.media.tv/preview_program/val id = uri.lastPathSegment // Load your video in the background. retrieveYourVideoPreviewUrl(id) { videoUri -> if (videoUri == null) { Log.d(TAG, "Could not find video $id") notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN) } try { mPlayer.setDataSource(getApplicationContext(), videoUri) mPlayer.prepare() mPlayer.start() notifyVideoAvailable() } catch (IOException e) { Log.e(TAG, "Could not prepare media player", e) notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN) } } return true } override fun onSetSurface(surface: Surface?): Boolean { mediaPlayer?.setSurface(surface) return true } override fun onSetStreamVolume(volume: Float) { // The home screen may fade in and out the video's volume. // Your player should be updated accordingly. mediaPlayer?.setVolume(volume, volume) } override fun onSetCaptionEnabled(b: Boolean) { // enable/disable captions here } } companion object { private const val TAG = "PreviewInputService" } }
জাভা
import android.content.Context; import android.media.MediaPlayer; import android.media.tv.TvInputService; import android.net.Uri; import android.support.annotation.Nullable; import android.util.Log; import android.view.Surface; import java.io.IOException; public class PreviewVideoInputService extends TvInputService { private static final String TAG = "PreviewVideoInputService"; @Nullable @Override public Session onCreateSession(String inputId) { return new PreviewSession(this); } private class PreviewSession extends TvInputService.Session { private MediaPlayer mPlayer; PreviewSession(Context context) { super(context); mPlayer = new MediaPlayer(); } @Override public boolean onTune(Uri channelUri) { // Let the TvInputService know that the video is being loaded. notifyVideoUnavailable(VIDEO_UNAVAILABLE_REASON_TUNING); // Fetch the stream url from the TV Provider database // for content://android.media.tv/preview_program/String id = uri.getLastPathSegment(); // Load your video in the background. retrieveYourVideoPreviewUrl(id, new MyCallback() { public void callback(Uri videoUri) { if (videoUri == null) { Log.d(TAG, "Could not find video" + id); notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN); } try { mPlayer.setDataSource(getApplicationContext(), videoUri); mPlayer.prepare(); mPlayer.start(); notifyVideoAvailable(); } catch (IOException e) { Log.e(TAG, "Could not prepare media player", e); notifyVideoUnavailable(TvInputManager.VIDEO_UNAVAILABLE_REASON_UNKNOWN); } } }); return true; } @Override public boolean onSetSurface(@Nullable Surface surface) { if (mPlayer != null) { mPlayer.setSurface(surface); } return true; } @Override public void onRelease() { if (mPlayer != null) { mPlayer.release(); } mPlayer = null; } @Override public void onSetStreamVolume(float volume) { if (mPlayer != null) { // The home screen may fade in and out the video's volume. // Your player should be updated accordingly. mPlayer.setVolume(volume, volume); } } @Override public void onSetCaptionEnabled(boolean enabled) { // enable/disable captions here } } }