API های android.media.projection
معرفی شده در Android 5 (سطح API 21) به شما امکان می دهند محتویات نمایشگر دستگاه را به عنوان یک جریان رسانه ضبط کنید که می توانید آن را پخش، ضبط یا پخش کنید به دستگاه های دیگر مانند تلویزیون.
اندروید 14 (سطح API 34) اشتراکگذاری صفحه برنامه را معرفی میکند که به کاربران امکان میدهد بدون توجه به حالت پنجره، یک پنجره برنامه را به جای کل صفحه دستگاه به اشتراک بگذارند. اشتراکگذاری صفحه برنامه، نوار وضعیت، نوار پیمایش، اعلانها و سایر عناصر رابط کاربری سیستم را از نمایشگر اشتراکگذاری شده مستثنی میکند—حتی زمانی که از اشتراکگذاری صفحه برنامه برای ضبط یک برنامه در تمام صفحه استفاده میشود. فقط محتویات برنامه انتخاب شده به اشتراک گذاشته می شود.
اشتراکگذاری صفحه برنامه، حریم خصوصی کاربر را تضمین میکند، بهرهوری کاربر را افزایش میدهد و چندوظیفگی را با امکان اجرای چندین برنامه به کاربران افزایش میدهد، اما اشتراکگذاری محتوا را فقط به یک برنامه محدود میکند.
سه نمایش نمایش
یک رسانه نمایش محتویات نمایشگر دستگاه یا پنجره برنامه را می گیرد و سپس تصویر گرفته شده را به یک صفحه نمایش مجازی که تصویر را روی یک Surface
نمایش می دهد، نمایش می دهد.
این برنامه Surface
با استفاده از MediaRecorder
، SurfaceTexture
، یا ImageReader
ارائه میکند، که محتویات نمایشگر گرفته شده را مصرف میکند و شما را قادر میسازد تا تصاویر رندر شده روی Surface
را در زمان واقعی مدیریت کنید. می توانید تصاویر را به عنوان ضبط ذخیره کنید یا آنها را به تلویزیون یا دستگاه دیگری ارسال کنید.
نمایش واقعی
با دریافت رمزی که به برنامه شما امکان می دهد محتویات نمایشگر دستگاه یا پنجره برنامه را ضبط کند، جلسه نمایش رسانه را شروع کنید. توکن با نمونه ای از کلاس MediaProjection
نمایش داده می شود.
از متد getMediaProjection()
سرویس سیستم MediaProjectionManager
برای ایجاد یک نمونه MediaProjection
هنگام شروع یک فعالیت جدید استفاده کنید. فعالیت را با یک intent از متد createScreenCaptureIntent()
شروع کنید تا عملیات ضبط صفحه را مشخص کنید:
کاتلین
val mediaProjectionManager = getSystemService(MediaProjectionManager::class.java) var mediaProjection : MediaProjection
val startMediaProjection = registerForActivityResult( StartActivityForResult() ) { result -> if (result.resultCode == RESULT_OK) { mediaProjection = mediaProjectionManager .getMediaProjection(result.resultCode, result.data!!) } }
startMediaProjection.launch(mediaProjectionManager.createScreenCaptureIntent())
جاوا
final MediaProjectionManager mediaProjectionManager = getSystemService(MediaProjectionManager.class); final MediaProjection[] mediaProjection = new MediaProjection[1];
ActivityResultLauncherstartMediaProjection = registerForActivityResult( new StartActivityForResult(), result -> { if (result.getResultCode() == Activity.RESULT_OK) { mediaProjection[0] = mediaProjectionManager .getMediaProjection(result.getResultCode(), result.getData()); } } );
startMediaProjection.launch(mediaProjectionManager.createScreenCaptureIntent());
نمایش مجازی
مرکز نمایش رسانه ای نمایش مجازی است که با فراخوانی createVirtualDisplay()
در یک نمونه MediaProjection
ایجاد می کنید:
کاتلین
virtualDisplay = mediaProjection.createVirtualDisplay( "ScreenCapture", width, height, screenDensity, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, surface, null, null)
جاوا
virtualDisplay = mediaProjection.createVirtualDisplay( "ScreenCapture", width, height, screenDensity, DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR, surface, null, null);
پارامترهای width
و height
ابعاد نمایش مجازی را مشخص می کنند. برای بدست آوردن مقادیر عرض و ارتفاع، از API های WindowMetrics
معرفی شده در اندروید 11 (سطح API 30) استفاده کنید. (برای جزئیات، به بخش اندازه طرح ریزی رسانه مراجعه کنید.)
سطح
سطح پخش رسانه را اندازه کنید تا خروجی با وضوح مناسب تولید شود. برای پخش صفحه نمایش به تلویزیون یا مانیتور کامپیوتر، سطح را بزرگ (با وضوح کم) و برای ضبط صفحه نمایش دستگاه کوچک (با وضوح بالا) کنید.
از Android 12L (سطح API 32)، هنگام رندر کردن محتوای ضبط شده روی سطح، سیستم محتوا را به طور یکنواخت مقیاس می کند و نسبت تصویر را حفظ می کند، به طوری که هر دو بعد محتوا (عرض و ارتفاع) برابر یا کمتر از متن مربوطه باشند. ابعاد سطح سپس محتوای گرفته شده روی سطح متمرکز می شود.
رویکرد مقیاسبندی Android 12L با به حداکثر رساندن اندازه تصویر سطح و حصول اطمینان از نسبت تصویر مناسب، پخش صفحه نمایش را به تلویزیون و سایر نمایشگرهای بزرگ بهبود میبخشد.
مجوز خدمات پیش زمینه
اگر برنامه شما Android 14 یا بالاتر را هدف قرار میدهد، مانیفست برنامه باید شامل یک اعلامیه مجوز برای نوع سرویس پیشزمینه mediaProjection
باشد:
<manifest ...>
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
<uses-permission android:name="android.permission.FOREGROUND_SERVICE_MEDIA_PROJECTION" />
<application ...>
<service
android:name=".MyMediaProjectionService"
android:foregroundServiceType="mediaProjection"
android:exported="false">
</service>
</application>
</manifest>
سرویس پخش رسانه را با فراخوانی به startForeground()
شروع کنید.
اگر نوع سرویس پیش زمینه را در تماس مشخص نکنید، نوع به صورت پیش فرض یک عدد صحیح بیتی از انواع سرویس های پیش زمینه تعریف شده در مانیفست است. اگر مانیفست هیچ نوع سرویسی را مشخص نکند، سیستم MissingForegroundServiceTypeException
را پرتاب می کند.
رضایت کاربر
قبل از هر جلسه نمایش رسانه، برنامه شما باید رضایت کاربر را درخواست کند. یک جلسه یک فراخوانی واحد برای createVirtualDisplay()
است. یک توکن MediaProjection
باید فقط یک بار برای برقراری تماس استفاده شود.
در اندروید 14 یا بالاتر، اگر برنامه شما یکی از موارد زیر را انجام دهد، متد createVirtualDisplay()
یک SecurityException
ایجاد میکند:
- یک نمونه
Intent
را که ازcreateScreenCaptureIntent()
بهgetMediaProjection()
ارسال می کند بیش از یک بار ارسال می کند. - بیش از یک بار در همان نمونه
MediaProjection
،createVirtualDisplay()
فراخوانی می کند
اندازه طرح ریزی رسانه
یک نمایش رسانه ای می تواند کل صفحه نمایش دستگاه یا یک پنجره برنامه را بدون در نظر گرفتن حالت پنجره ضبط کند.
اندازه اولیه
با نمایش رسانه تمام صفحه، برنامه شما باید اندازه صفحه نمایش دستگاه را تعیین کند. در اشتراکگذاری صفحه برنامه، تا زمانی که کاربر منطقه عکسبرداری را انتخاب نکرده باشد، برنامه شما نمیتواند اندازه نمایشگر گرفته شده را تعیین کند. بنابراین، اندازه اولیه هر نمایش رسانه ای اندازه صفحه نمایش دستگاه است.
از متد getMaximumWindowMetrics()
WindowManager
getMaximumWindowMetrics برای برگرداندن یک شی WindowMetrics
برای صفحه دستگاه استفاده کنید، حتی اگر برنامه میزبان نمایش رسانه در حالت چند پنجره ای باشد و تنها بخشی از نمایشگر را اشغال کند.
برای سازگاری تا سطح API 14، از روش WindowMetricsCalculator
computeMaximumWindowMetrics()
از کتابخانه Jetpack WindowManager
استفاده کنید.
برای دریافت عرض و ارتفاع نمایشگر دستگاه، متد getBounds()
WindowMetrics
فراخوانی کنید.
اندازه تغییر می کند
وقتی دستگاه چرخانده میشود یا کاربر پنجره برنامه را بهعنوان منطقه عکسبرداری در اشتراکگذاری صفحه برنامه انتخاب میکند، اندازه نمایش رسانه میتواند تغییر کند. اگر محتوای ضبطشده اندازهای متفاوت از حداکثر معیارهای پنجره بهدستآمده در هنگام راهاندازی طرحدهی رسانهای باشد، نمایش رسانه ممکن است جعبه نامه باشد.
برای اطمینان از اینکه نمایش رسانه دقیقاً با اندازه محتوای ضبط شده برای هر منطقه ضبط شده و چرخش دستگاه مطابقت دارد، از پاسخ تماس onCapturedContentResize()
برای تغییر اندازه عکس استفاده کنید. (برای اطلاعات بیشتر به بخش سفارشی سازی که در ادامه می آید مراجعه کنید).
سفارشی سازی
برنامه شما میتواند تجربه کاربر پیشبینی رسانه را با APIهای MediaProjection.Callback
زیر سفارشی کند:
onCapturedContentVisibilityChanged()
: برنامه میزبان (برنامهای که پخش رسانه را شروع کرد) را قادر میسازد تا محتوای اشتراکگذاری شده را نشان دهد یا پنهان کند.از این پاسخ تماس برای سفارشی کردن رابط کاربری برنامه خود بر اساس قابل مشاهده بودن منطقه ضبط شده برای کاربر استفاده کنید. به عنوان مثال، اگر برنامه شما برای کاربر قابل مشاهده است و محتوای ضبط شده را در رابط کاربری برنامه نمایش می دهد، و برنامه ضبط شده نیز برای کاربر قابل مشاهده است (همانطور که در این پاسخ تماس نشان داده شده است)، کاربر همان محتوا را دو بار می بیند. از پاسخ تماس برای به روز رسانی رابط کاربری برنامه خود استفاده کنید تا محتوای گرفته شده را پنهان کنید و فضای طرح بندی برنامه خود را برای محتوای دیگر آزاد کنید.
onCapturedContentResize()
: برنامه میزبان را قادر میسازد تا اندازه نمایش رسانه را بر روی صفحه نمایش مجازی وSurface
نمایش رسانه را بر اساس اندازه منطقه نمایش گرفته شده تغییر دهد.هر زمان که محتوای ضبطشده (یک پنجره برنامه یا نمایش کامل دستگاه) اندازهگیری میشود (به دلیل چرخش دستگاه یا ورود برنامه ضبطشده به حالت پنجرهگیری متفاوت). از این API برای تغییر اندازه نمایشگر مجازی و سطح استفاده کنید تا اطمینان حاصل کنید که نسبت تصویر با محتوای ضبط شده مطابقت دارد و عکس در جعبه نامه نیست.
بازیابی منابع
برنامه شما باید فراخوانی MediaProjection
onStop()
را ثبت کند تا زمانی که جلسه نمایش رسانه متوقف شد و نامعتبر شد، مطلع شود. هنگامی که جلسه متوقف می شود، برنامه شما باید منابعی را که در اختیار دارد، مانند صفحه نمایش مجازی و سطح نمایش، آزاد کند. یک جلسه پخش رسانه متوقف شده دیگر نمی تواند یک نمایشگر مجازی جدید ایجاد کند، حتی اگر برنامه شما قبلاً یک نمایش مجازی برای آن طرح رسانه ایجاد نکرده باشد.
هنگامی که پخش رسانه ای پایان می یابد، سیستم تماس برگشتی را فراخوانی می کند. این فسخ ممکن است به دلایل مختلفی رخ دهد، مانند:
- کاربر جلسه را با استفاده از رابط کاربری برنامه یا تراشه نوار وضعیت نمایش رسانه سیستم متوقف می کند
- صفحه در حال قفل شدن است
- یک جلسه دیگر طرح رسانه ای شروع می شود
- روند برنامه از بین می رود
اگر برنامه شما پاسخ تماس را ثبت نمی کند، هر فراخوانی برای createVirtualDisplay()
IllegalStateException
ارسال می کند.
انصراف دهید
اندروید 14 یا بالاتر به طور پیشفرض اشتراکگذاری صفحه برنامه را فعال میکند. هر جلسه نمایش رسانه به کاربران این امکان را می دهد که یک پنجره برنامه یا کل صفحه نمایش را به اشتراک بگذارند.
برنامه شما میتواند با فراخوانی متد createScreenCaptureIntent(MediaProjectionConfig)
با آرگومان MediaProjectionConfig
که از فراخوانی به createConfigForDefaultDisplay()
بازگردانده میشود، از اشتراکگذاری صفحه برنامه انصراف دهد.
فراخوانی برای createScreenCaptureIntent(MediaProjectionConfig)
با آرگومان MediaProjectionConfig
که از فراخوانی به createConfigForUserChoice()
بازگردانده شده است، همان رفتار پیشفرض است، یعنی فراخوانی به createScreenCaptureIntent()
.
برنامه های قابل تغییر اندازه
همیشه برنامه های پخش رسانه خود را قابل تغییر اندازه کنید ( resizeableActivity="true"
). برنامههای قابل تغییر اندازه از تغییرات پیکربندی دستگاه و حالت چند پنجرهای پشتیبانی میکنند ( به پشتیبانی چند پنجرهای مراجعه کنید).
اگر برنامه شما قابل تغییر اندازه نیست، باید محدوده نمایش را از یک زمینه پنجره پرس و جو کند و از getMaximumWindowMetrics()
برای بازیابی WindowMetrics
حداکثر ناحیه نمایش در دسترس برنامه استفاده کند:
کاتلین
val windowContext = context.createWindowContext(context.display!!, WindowManager.LayoutParams.TYPE_APPLICATION, null) val projectionMetrics = windowContext.getSystemService(WindowManager::class.java) .maximumWindowMetrics
جاوا
Context windowContext = context.createWindowContext(context.getDisplay(), WindowManager.LayoutParams.TYPE_APPLICATION, null); WindowMetrics projectionMetrics = windowContext.getSystemService(WindowManager.class) .getMaximumWindowMetrics();
تراشه نوار وضعیت و توقف خودکار
بهرهبرداریهای پیشبینی صفحه، دادههای خصوصی کاربر مانند اطلاعات مالی را در معرض دید کاربران قرار میدهند، زیرا کاربران متوجه نمیشوند که صفحه دستگاهشان به اشتراک گذاشته میشود.
Android 15 (سطح API 35) و بالاتر، یک تراشه نوار وضعیت را نمایش میدهد که بزرگ و برجسته است تا کاربران را در مورد هر گونه نمایش صفحه نمایش در حال پیشرفت هشدار دهد. کاربران میتوانند روی تراشه ضربه بزنند تا از اشتراکگذاری، پخش یا ضبط صفحه نمایش خود جلوگیری کنند. همچنین، با قفل شدن صفحه نمایش دستگاه، نمایش صفحه به طور خودکار متوقف می شود.
با شروع اشتراکگذاری صفحه، ارسال محتوا، یا ضبط، در دسترس بودن تراشه نوار وضعیت نمایش رسانه را آزمایش کنید. تراشه باید در نوار وضعیت ظاهر شود.
برای اطمینان از اینکه برنامهتان منابع را آزاد میکند و رابط کاربری خود را بهروزرسانی میکند، وقتی پیشبینی صفحه توسط تعامل کاربر با تراشه نوار وضعیت یا فعالسازی صفحه قفل متوقف میشود، موارد زیر را انجام دهید:
یک نمونه از
MediaProjection.Callback
ایجاد کنید.متد callback
onStop()
را پیاده سازی کنید. این روش زمانی فراخوانی می شود که نمایش صفحه متوقف شود. منابعی که برنامه شما در اختیار دارد را آزاد کنید و در صورت نیاز رابط کاربری برنامه را بهروزرسانی کنید.
برای آزمایش پاسخ تماس، روی تراشه نوار وضعیت ضربه بزنید یا صفحه دستگاه را قفل کنید تا نمایش صفحه متوقف شود. بررسی کنید که متد onStop()
فراخوانی شده باشد و برنامه شما همانطور که در نظر گرفته شده پاسخ دهد.
منابع اضافی
برای اطلاعات بیشتر درباره پخش رسانه، به ضبط ویدیو و پخش صدا مراجعه کنید.