مانند نسخه های قبلی، Android 15 شامل تغییرات رفتاری است که ممکن است بر برنامه شما تأثیر بگذارد. تغییرات رفتاری زیر منحصراً برای برنامههایی اعمال میشود که Android 15 یا بالاتر را هدف قرار میدهند. اگر برنامه شما اندروید 15 یا بالاتر را هدف قرار می دهد، باید برنامه خود را تغییر دهید تا در صورت لزوم از این رفتارها به درستی پشتیبانی کند.
حتماً فهرستی از تغییرات رفتاری را نیز مرور کنید که بر همه برنامههای در حال اجرا در Android 15 بدون توجه به targetSdkVersion
برنامه شما تأثیر میگذارد.
عملکرد اصلی
اندروید 15 قابلیت های اصلی مختلف سیستم اندروید را اصلاح یا گسترش می دهد.
تغییرات در خدمات پیش زمینه
ما در حال انجام تغییرات زیر در سرویس های پیش زمینه با اندروید 15 هستیم.
- رفتار درنگ سرویس پیش زمینه همگام سازی داده ها
- نوع سرویس پیش زمینه پردازش رسانه جدید
- محدودیت در گیرنده های پخش
BOOT_COMPLETED
که خدمات پیش زمینه را راه اندازی می کنند - محدودیتهایی برای شروع سرویسهای پیشزمینه در زمانی که برنامه دارای مجوز
SYSTEM_ALERT_WINDOW
است
رفتار درنگ سرویس پیش زمینه همگام سازی داده ها
Android 15 introduces a new timeout behavior to dataSync
for apps targeting
Android 15 (API level 35) or higher. This behavior also applies to the new
mediaProcessing
foreground service type.
The system permits an app's dataSync
services to run for a total of 6 hours
in a 24-hour period, after which the system calls the running service's
Service.onTimeout(int, int)
method (introduced in Android
15). At this time, the service has a few seconds to call
Service.stopSelf()
. When Service.onTimeout()
is called, the
service is no longer considered a foreground service. If the service does not
call Service.stopSelf()
, the system throws an internal exception. The
exception is logged in Logcat with the following message:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type dataSync did not stop within its timeout: [component name]"
To avoid problems with this behavior change, you can do one or more of the following:
- Have your service implement the new
Service.onTimeout(int, int)
method. When your app receives the callback, make sure to callstopSelf()
within a few seconds. (If you don't stop the app right away, the system generates a failure.) - Make sure your app's
dataSync
services don't run for more than a total of 6 hours in any 24-hour period (unless the user interacts with the app, resetting the timer). - Only start
dataSync
foreground services as a result of direct user interaction; since your app is in the foreground when the service starts, your service has the full six hours after the app goes to the background. - Instead of using a
dataSync
foreground service, use an alternative API.
If your app's dataSync
foreground services have run for 6 hours in the last
24, you cannot start another dataSync
foreground service unless the user
has brought your app to the foreground (which resets the timer). If you try to
start another dataSync
foreground service, the system throws
ForegroundServiceStartNotAllowedException
with an error message like "Time limit already exhausted for foreground service
type dataSync".
Testing
To test your app's behavior, you can enable data sync timeouts even if your app
is not targeting Android 15 (as long as the app is running on an Android 15
device). To enable timeouts, run the following adb
command:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
You can also adjust the timeout period, to make it easier to test how your
app behaves when the limit is reached. To set a new timeout period, run the
following adb
command:
adb shell device_config put activity_manager data_sync_fgs_timeout_duration duration-in-milliseconds
نوع سرویس پیش زمینه پردازش رسانه جدید
اندروید 15 یک نوع سرویس پیش زمینه جدید، mediaProcessing
را معرفی می کند. این نوع سرویس برای عملیاتی مانند رمزگذاری فایل های رسانه ای مناسب است. به عنوان مثال، یک برنامه رسانه ممکن است یک فایل صوتی را دانلود کند و قبل از پخش آن را به فرمت دیگری تبدیل کند. میتوانید از سرویس پیشزمینه mediaProcessing
استفاده کنید تا مطمئن شوید که تبدیل حتی زمانی که برنامه در پسزمینه است ادامه مییابد.
این سیستم به سرویسهای mediaProcessing
یک برنامه اجازه میدهد در مجموع 6 ساعت در یک دوره 24 ساعته اجرا شوند، پس از آن سیستم سرویس سرویس در حال اجرا را متد Service.onTimeout(int, int)
(معرفی شده در Android 15) فراخوانی میکند. در این زمان، سرویس چند ثانیه فرصت دارد تا Service.stopSelf()
را فراخوانی کند. اگر سرویس Service.stopSelf()
را فراخوانی نکند، سیستم یک استثنا داخلی ایجاد می کند. استثنا با پیام زیر در Logcat وارد شده است:
Fatal Exception: android.app.RemoteServiceException: "A foreground service of
type mediaProcessing did not stop within its timeout: [component name]"
برای جلوگیری از استثناء، می توانید یکی از موارد زیر را انجام دهید:
- از سرویس خود بخواهید روش جدید
Service.onTimeout(int, int)
را پیاده سازی کند. هنگامی که برنامه شما پاسخ تماس را دریافت کرد، مطمئن شوید که در عرض چند ثانیهstopSelf()
تماس بگیرید. (اگر برنامه را فوراً متوقف نکنید، سیستم یک خرابی ایجاد می کند.) - مطمئن شوید که سرویسهای
mediaProcessing
برنامه شما در هر دوره ۲۴ ساعته در مجموع بیش از ۶ ساعت اجرا نمیشوند (مگر اینکه کاربر با برنامه تعامل داشته باشد و تایمر را بازنشانی کند). - خدمات پیش زمینه
mediaProcessing
را فقط در نتیجه تعامل مستقیم کاربر شروع کنید. از آنجایی که هنگام شروع سرویس، برنامه شما در پیش زمینه است، سرویس شما شش ساعت کامل پس از رفتن برنامه به پسزمینه است. - به جای استفاده از سرویس پیش زمینه
mediaProcessing
، از یک API جایگزین مانند WorkManager استفاده کنید.
اگر سرویسهای پیشزمینه mediaProcessing
برنامهتان در 24 ساعت گذشته به مدت 6 ساعت اجرا شده است، نمیتوانید سرویس پیشزمینه mediaProcessing
دیگری را راهاندازی کنید ، مگر اینکه کاربر برنامه شما را به پیشزمینه آورده باشد (که تایمر را بازنشانی میکند). اگر بخواهید سرویس پیشزمینه mediaProcessing
دیگری را راهاندازی کنید، سیستم ForegroundServiceStartNotAllowedException
را با پیام خطایی مانند "محدودیت زمانی برای نوع سرویس پیشزمینه mediaProcessing تمام شده است" میفرستد.
برای اطلاعات بیشتر در مورد نوع سرویس mediaProcessing
، به تغییرات در انواع خدمات پیش زمینه برای Android 15 مراجعه کنید: پردازش رسانه .
تست کردن
برای آزمایش رفتار برنامهتان، میتوانید وقفههای زمانی پردازش رسانه را فعال کنید، حتی اگر برنامه شما Android 15 را هدف قرار ندهد (تا زمانی که برنامه روی دستگاه Android 15 اجرا شود). برای فعال کردن بازه زمانی، دستور adb
زیر را اجرا کنید:
adb shell am compat enable FGS_INTRODUCE_TIME_LIMITS your-package-name
همچنین میتوانید مدت زمان وقفه را تنظیم کنید تا آزمایش نحوه عملکرد برنامهتان در زمان رسیدن به محدودیت آسانتر شود. برای تنظیم یک بازه زمانی جدید، دستور adb
زیر را اجرا کنید:
adb shell device_config put activity_manager media_processing_fgs_timeout_duration duration-in-milliseconds
محدودیت در گیرنده های پخش BOOT_COMPLETED
که خدمات پیش زمینه را راه اندازی می کنند
محدودیت های جدیدی برای گیرنده های پخش BOOT_COMPLETED
وجود دارد که خدمات پیش زمینه را راه اندازی می کنند. گیرنده های BOOT_COMPLETED
مجاز به راه اندازی انواع خدمات پیش زمینه زیر نیستند :
-
dataSync
-
camera
-
mediaPlayback
-
phoneCall
-
mediaProjection
-
microphone
(این محدودیت از اندروید 14 برایmicrophone
اعمال شده است)
اگر یک گیرنده BOOT_COMPLETED
سعی کند هر یک از آن نوع خدمات پیش زمینه را راه اندازی کند، سیستم ForegroundServiceStartNotAllowedException
پرتاب می کند.
تست کردن
برای آزمایش رفتار برنامهتان، میتوانید این محدودیتهای جدید را فعال کنید، حتی اگر برنامه شما اندروید 15 را هدف قرار ندهد (تا زمانی که برنامه روی دستگاه Android 15 اجرا شود). دستور adb
زیر را اجرا کنید:
adb shell am compat enable FGS_BOOT_COMPLETED_RESTRICTIONS your-package-name
برای ارسال یک پخش BOOT_COMPLETED
بدون راه اندازی مجدد دستگاه، دستور adb
زیر را اجرا کنید:
adb shell am broadcast -a android.intent.action.BOOT_COMPLETED your-package-name
محدودیتهایی برای شروع سرویسهای پیشزمینه در زمانی که برنامه دارای مجوز SYSTEM_ALERT_WINDOW
است
Previously, if an app held the SYSTEM_ALERT_WINDOW
permission, it could launch
a foreground service even if the app was currently in the background (as
discussed in exemptions from background start restrictions).
If an app targets Android 15, this exemption is now narrower. The app now needs
to have the SYSTEM_ALERT_WINDOW
permission and also have a visible overlay
window. That is, the app needs to first launch a
TYPE_APPLICATION_OVERLAY
window and the window
needs to be visible before you start a foreground service.
If your app attempts to start a foreground service from the background without
meeting these new requirements (and it does not have some other exemption), the
system throws ForegroundServiceStartNotAllowedException
.
If your app declares the SYSTEM_ALERT_WINDOW
permission
and launches foreground services from the background, it may be affected by this
change. If your app gets a ForegroundServiceStartNotAllowedException
, check
your app's order of operations and make sure your app already has an active
overlay window before it attempts to start a foreground service from the
background. You can check if your overlay window is currently visible
by calling View.getWindowVisibility()
, or you
can override View.onWindowVisibilityChanged()
to get notified whenever the visibility changes.
Testing
To test your app's behavior, you can enable these new restrictions even if your
app is not targeting Android 15 (as long as the app is running on an Android 15
device). To enable these new restrictions on starting foreground services
from the background, run the following adb
command:
adb shell am compat enable FGS_SAW_RESTRICTIONS your-package-name
زمانی که برنامهها میتوانند وضعیت جهانی حالت مزاحم نشوید را تغییر دهند
Apps that target Android 15 (API level 35) and higher can no longer change the
global state or policy of Do Not Disturb (DND) on a device (either by modifying
user settings, or turning off DND mode). Instead, apps must contribute an
AutomaticZenRule
, which the system combines into a global policy with the
existing most-restrictive-policy-wins scheme. Calls to existing APIs that
previously affected global state (setInterruptionFilter
,
setNotificationPolicy
) result in the creation or update of an implicit
AutomaticZenRule
, which is toggled on and off depending on the call-cycle of
those API calls.
Note that this change only affects observable behavior if the app is calling
setInterruptionFilter(INTERRUPTION_FILTER_ALL)
and expects that call to
deactivate an AutomaticZenRule
that was previously activated by their owners.
OpenJDK API تغییر می کند
اندروید 15 به کار تازه سازی کتابخانه های اصلی اندروید برای هماهنگی با ویژگی های آخرین نسخه OpenJDK LTS ادامه می دهد.
برخی از این تغییرات میتواند بر سازگاری برنامهها برای برنامههایی که Android 15 را هدف قرار میدهند (سطح API 35) تأثیر بگذارد:
تغییرات در APIهای قالببندی رشتهای : اعتبارسنجی شاخص آرگومان، پرچمها، عرض و دقت اکنون هنگام استفاده از APIهای
String.format()
وFormatter.format()
سختتر است:-
String.format(String, Object[])
-
String.format(Locale, String, Object[])
-
Formatter.format(String, Object[])
-
Formatter.format(Locale, String, Object[])
به عنوان مثال، زمانی که از شاخص آرگومان 0 استفاده می شود، استثنای زیر ایجاد می شود (
%0
در رشته قالب):IllegalFormatArgumentIndexException: Illegal format argument index = 0
در این مورد، مشکل را می توان با استفاده از شاخص آرگومان 1 (
%1
در رشته قالب) برطرف کرد.-
تغییرات در نوع مؤلفه
Arrays.asList(...).toArray()
: وقتی ازArrays.asList(...).toArray()
استفاده می کنیم، نوع جزء آرایه به دست آمده اکنون یکObject
است — نه نوع عناصر آرایه زیربنایی بنابراین کد زیر یکClassCastException
می اندازد:String[] elements = (String[]) Arrays.asList("one", "two").toArray();
برای این مورد، برای حفظ
String
به عنوان نوع مؤلفه در آرایه حاصل، میتوانید به جای آن ازCollection.toArray(Object[])
استفاده کنید:String[] elements = Arrays.asList("two", "one").toArray(new String[0]);
تغییرات در مدیریت کد زبان : هنگام استفاده از
Locale
API، کدهای زبان عبری، ییدیش و اندونزیایی دیگر به شکلهای منسوخ خود تبدیل نمیشوند (عبری:iw
، ییدیش:ji
، و اندونزیایی:in
). هنگام تعیین کد زبان برای یکی از این مناطق، به جای آن از کدهای ISO 639-1 استفاده کنید (عبری:he
، ییدیش:yi
، و اندونزیایی:id
).تغییرات در دنبالههای تصادفی int : به دنبال تغییرات ایجاد شده در https://bugs.openjdk.org/browse/JDK-8301574 ، متدهای
Random.ints()
زیر اکنون دنبالهای متفاوت از اعداد را نسبت به متدهایRandom.nextInt()
انجام دهید:به طور کلی، این تغییر نباید منجر به رفتار شکستن برنامه شود، اما کد شما نباید انتظار داشته باشد که دنباله تولید شده از متدهای
Random.ints()
مطابق باRandom.nextInt()
باشد.
بعد از اینکه compileSdk
در پیکربندی ساخت برنامه خود برای استفاده از Android 15 (سطح API 35) بهروزرسانی کردید، API جدید SequencedCollection
میتواند بر سازگاری برنامه شما تأثیر بگذارد:
برخورد با توابع پسوند
MutableList.removeFirst()
وMutableList.removeLast()
درkotlin-stdlib
نوع
List
در جاوا به نوعMutableList
در Kotlin نگاشت می شود. از آنجایی که API هایList.removeFirst()
وList.removeLast()
در اندروید 15 معرفی شده اند (سطح API 35)، کامپایلر Kotlin فراخوانی های تابع، به عنوان مثالlist.removeFirst()
به صورت ایستا بهList
API های جدید حل می کند. توابع پسوند درkotlin-stdlib
.اگر برنامهای دوباره با
compileSdk
روی35
وminSdk
روی34
یا پایینتر کامپایل شود، و سپس برنامه روی اندروید 14 و پایینتر اجرا شود، یک خطای زمان اجرا ایجاد میشود:java.lang.NoSuchMethodError: No virtual method removeFirst()Ljava/lang/Object; in class Ljava/util/ArrayList;
گزینه
NewApi
lint موجود در پلاگین Android Gradle می تواند این موارد استفاده جدید API را دریافت کند../gradlew lint
MainActivity.kt:41: Error: Call requires API level 35 (current min is 34): java.util.List#removeFirst [NewApi] list.removeFirst()برای رفع خطاهای Exception و lint در زمان اجرا، فراخوانی تابع
removeFirst()
وremoveLast()
می توان به ترتیب باremoveAt(0)
وremoveAt(list.lastIndex)
در Kotlin جایگزین کرد. اگر از Android Studio Ladybug | 2024.1.3 یا بالاتر، همچنین گزینه ای برای رفع سریع این خطاها ارائه می دهد.اگر گزینه lint غیرفعال شده است،
@SuppressLint("NewApi")
وlintOptions { disable 'NewApi' }
حذف کنید.برخورد با روش های دیگر در جاوا
روش های جدیدی به انواع موجود اضافه شده است، به عنوان مثال،
List
وDeque
. این متدهای جدید ممکن است با متدهایی با همان نام و انواع آرگومان در سایر رابطها و کلاسها سازگار نباشند. در مورد تصادم امضای روش با ناسازگاری، کامپایلرjavac
یک خطای زمان ساخت را خروجی میکند. به عنوان مثال:خطای مثال 1:
javac MyList.java
MyList.java:135: error: removeLast() in MyList cannot implement removeLast() in List public void removeLast() { ^ return type void is not compatible with Object where E is a type-variable: E extends Object declared in interface Listمثال خطای 2:
javac MyList.java
MyList.java:7: error: types Deque<Object> and List<Object> are incompatible; public class MyList implements List<Object>, Deque<Object> { both define reversed(), but with unrelated return types 1 errorمثال خطای 3:
javac MyList.java
MyList.java:43: error: types List<E#1> and MyInterface<E#2> are incompatible; public static class MyList implements List<Object>, MyInterface<Object> { class MyList inherits unrelated defaults for getFirst() from types List and MyInterface where E#1,E#2 are type-variables: E#1 extends Object declared in interface List E#2 extends Object declared in interface MyInterface 1 errorبرای رفع این خطاهای ساخت، کلاسی که این رابطها را پیادهسازی میکند، باید متد را با نوع بازگشتی سازگار لغو کند. به عنوان مثال:
@Override public Object getFirst() { return List.super.getFirst(); }
امنیت
اندروید 15 شامل تغییراتی است که امنیت سیستم را ارتقا میدهد تا از برنامهها و کاربران در برابر برنامههای مخرب محافظت کند.
فعالیت پس زمینه ایمن راه اندازی می شود
Android 15 protects users from malicious apps and gives them more control over their devices by adding changes that prevent malicious background apps from bringing other apps to the foreground, elevating their privileges, and abusing user interaction. Background activity launches have been restricted since Android 10 (API level 29).
Other changes
In addition to the restriction for UID matching, these other changes are also included:
- Change
PendingIntent
creators to block background activity launches by default. This helps prevent apps from accidentally creating aPendingIntent
that could be abused by malicious actors. - Don't bring an app to the foreground unless the
PendingIntent
sender allows it. This change aims to prevent malicious apps from abusing the ability to start activities in the background. By default, apps are not allowed to bring the task stack to the foreground unless the creator allows background activity launch privileges or the sender has background activity launch privileges. - Control how the top activity of a task stack can finish its task. If the top activity finishes a task, Android will go back to whichever task was last active. Moreover, if a non-top activity finishes its task, Android will go back to the home screen; it won't block the finish of this non-top activity.
- Prevent launching arbitrary activities from other apps into your own task. This change prevents malicious apps from phishing users by creating activities that appear to be from other apps.
- Block non-visible windows from being considered for background activity launches. This helps prevent malicious apps from abusing background activity launches to display unwanted or malicious content to users.
مقاصد امن تر
اندروید 15 اقدامات امنیتی اختیاری جدیدی را برای ایمن تر و قوی تر کردن مقاصد معرفی می کند. این تغییرات با هدف جلوگیری از آسیب پذیری های احتمالی و سوء استفاده از اهدافی است که می تواند توسط برنامه های مخرب مورد سوء استفاده قرار گیرد. دو پیشرفت اصلی برای امنیت intent در اندروید 15 وجود دارد:
- Match target-intent-filters: مقاصدی که اجزای خاصی را هدف قرار می دهند باید دقیقاً با مشخصات فیلتر هدف هدف مطابقت داشته باشند. اگر قصدی برای راهاندازی فعالیت برنامه دیگری ارسال میکنید، مؤلفه هدف هدف باید با فیلترهای هدف اعلامشده فعالیت دریافتکننده هماهنگ شود.
- Intent ها باید دارای عملکرد باشند: Intent های بدون عمل دیگر با فیلترهای intent مطابقت ندارند. این بدان معناست که مقاصد مورد استفاده برای شروع فعالیت ها یا خدمات باید دارای یک کنش به وضوح تعریف شده باشند.
برای بررسی اینکه برنامه شما چگونه به این تغییرات پاسخ می دهد، از StrictMode
در برنامه خود استفاده کنید. برای مشاهده گزارشهای دقیق درباره نقض استفاده Intent
، روش زیر را اضافه کنید:
کاتلین
fun onCreate() { StrictMode.setVmPolicy(VmPolicy.Builder() .detectUnsafeIntentLaunch() .build() ) }
جاوا
public void onCreate() { StrictMode.setVmPolicy(new VmPolicy.Builder() .detectUnsafeIntentLaunch() .build()); }
تجربه کاربری و رابط کاربری سیستم
اندروید 15 شامل تغییراتی است که برای ایجاد یک تجربه کاربری سازگارتر و بصری در نظر گرفته شده است.
ورودی پنجره تغییر می کند
در اندروید 15 دو تغییر مربوط به ورودیهای پنجره وجود دارد: لبه به لبه بهطور پیشفرض اعمال میشود و همچنین تغییراتی در پیکربندی وجود دارد، مانند پیکربندی پیشفرض نوارهای سیستم.
اجرای لبه به لبه
اگر برنامه اندروید 15 (سطح API 35) را هدف قرار دهد، برنامهها بهطور پیشفرض در دستگاههای دارای Android 15 لبه به لبه هستند.
این یک تغییر اساسی است که ممکن است بر رابط کاربری برنامه شما تأثیر منفی بگذارد. تغییرات بر حوزه های UI زیر تأثیر می گذارد:
- نوار پیمایش دسته اشاره
- به طور پیش فرض شفاف است.
- افست پایین غیرفعال است، بنابراین محتوا در پشت نوار پیمایش سیستم کشیده میشود، مگر اینکه ورودیها اعمال شوند.
-
setNavigationBarColor
وR.attr#navigationBarColor
منسوخ شدهاند و بر ناوبری اشارهای تأثیری ندارند. -
setNavigationBarContrastEnforced
وR.attr#navigationBarContrastEnforced
همچنان هیچ تاثیری بر ناوبری اشاره ای ندارند.
- ناوبری 3 دکمه ای
- Opacity به طور پیشفرض روی 80% تنظیم شده است و رنگ احتمالاً با پسزمینه پنجره مطابقت دارد.
- افست پایین غیرفعال است، بنابراین محتوا در پشت نوار پیمایش سیستم کشیده میشود مگر اینکه درجهای داخلی اعمال شوند.
-
setNavigationBarColor
وR.attr#navigationBarColor
به طور پیشفرض با پسزمینه پنجره مطابقت دارند. پس زمینه پنجره باید یک رنگ قابل ترسیم باشد تا این پیش فرض اعمال شود. این API منسوخ شده است اما همچنان بر ناوبری 3 دکمه ای تأثیر می گذارد. -
setNavigationBarContrastEnforced
وR.attr#navigationBarContrastEnforced
به طور پیشفرض درست است، که 80% پسزمینه مات را در مسیریابی 3 دکمه اضافه میکند.
- نوار وضعیت
- به طور پیش فرض شفاف است.
- افست بالا غیرفعال است، بنابراین محتوا در پشت نوار وضعیت ترسیم می شود، مگر اینکه درج ها اعمال شوند.
-
setStatusBarColor
وR.attr#statusBarColor
منسوخ شده اند و روی اندروید 15 تاثیری ندارند. -
setStatusBarContrastEnforced
وR.attr#statusBarContrastEnforced
منسوخ شدهاند اما همچنان روی اندروید 15 اثر دارند.
- برش نمایشگر
-
layoutInDisplayCutoutMode
پنجرههای غیر شناور بایدLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
باشد.SHORT_EDGES
،NEVER
وDEFAULT
بهعنوانALWAYS
تفسیر میشوند تا کاربران نوار سیاه ناشی از بریدگی نمایشگر را نبینند و لبه به لبه ظاهر شوند.
-
مثال زیر یک برنامه را قبل و بعد از هدف قرار دادن Android 15 (سطح API 35) و قبل و بعد از اعمال insets نشان می دهد.
چه باید بررسی کرد که آیا برنامه شما از قبل لبه به لبه است
اگر برنامه شما از قبل لبه به لبه است و از inset ها استفاده می کند، به جز در سناریوهای زیر، اکثراً بی تأثیر هستید. با این حال، حتی اگر فکر می کنید تحت تأثیر قرار نگرفته اید، توصیه می کنیم برنامه خود را آزمایش کنید.
- شما یک پنجره غیر شناور دارید، مانند
Activity
که به جایLAYOUT_IN_DISPLAY_CUTOUT_MODE_ALWAYS
ازSHORT_EDGES
،NEVER
یاDEFAULT
استفاده می کند. اگر برنامه شما در هنگام راهاندازی از کار میافتد، ممکن است به خاطر صفحه نمایش شما باشد. میتوانید وابستگی اصلی صفحه نمایش اسپلش را به 1.2.0-alpha01 یا بالاتر ارتقا دهید یاwindow.attributes.layoutInDisplayCutoutMode = WindowManager.LayoutInDisplayCutoutMode.always
تنظیم کنید. - ممکن است صفحه نمایش هایی با ترافیک کمتر با رابط کاربری مسدود شده وجود داشته باشد. بررسی کنید این صفحهنمایشهایی که کمتر بازدید شدهاند، رابط کاربری مسدود شده ندارند. صفحه نمایش های کم تردد عبارتند از:
- صفحات ورود یا ورود به سیستم
- صفحات تنظیمات
اگر برنامه شما از قبل لبه به لبه نیست، چه باید بررسی کنید
اگر برنامه شما در حال حاضر لبه به لبه نیست، به احتمال زیاد تحت تأثیر قرار خواهید گرفت. علاوه بر سناریوهای برنامههایی که از قبل لبه به لبه هستند، باید موارد زیر را در نظر بگیرید:
- اگر برنامه شما از Material 3 Components (
androidx.compose.material3
) در نوشتن استفاده میکند، مانندTopAppBar
،BottomAppBar
، وNavigationBar
، این مؤلفهها احتمالاً تحت تأثیر قرار نمیگیرند زیرا بهطور خودکار ورودیها را مدیریت میکنند. - اگر برنامه شما از Material 2 Components (
androidx.compose.material
) در Compose استفاده میکند، این مؤلفهها بهطور خودکار ورودیها را مدیریت نمیکنند. با این حال، می توانید به inset ها دسترسی داشته باشید و آنها را به صورت دستی اعمال کنید. در androidx.compose.material 1.6.0 و جدیدتر، از پارامترwindowInsets
برای اعمال دستی Insets برایBottomAppBar
،TopAppBar
،BottomNavigation
وNavigationRail
استفاده کنید. به همین ترتیب، از پارامترcontentWindowInsets
برایScaffold
استفاده کنید. - اگر برنامه شما از نماها و مؤلفههای مواد (
com.google.android.material
) استفاده میکند، بیشتر مؤلفههای مواد مبتنی بر نما مانندBottomNavigationView
،BottomAppBar
،NavigationRailView
یاNavigationView
، ورودیها را مدیریت میکنند و نیازی به کار اضافی ندارند. با این حال، اگر ازAppBarLayout
استفاده می کنید، بایدandroid:fitsSystemWindows="true"
را اضافه کنید. - برای ترکیبهای سفارشی، اینستها را بهصورت دستی بهعنوان بالشتک اعمال کنید. اگر محتوای شما در داخل یک
Scaffold
است، میتوانید با استفاده از مقادیرScaffold
padding از انست استفاده کنید. در غیر این صورت، با استفاده از یکی ازWindowInsets
، padding را اعمال کنید. - اگر برنامه شما از نماها و
BottomSheet
،SideSheet
یا ظروف سفارشی استفاده می کند، با استفاده ازViewCompat.setOnApplyWindowInsetsListener
padding را اعمال کنید. برایRecyclerView
، با استفاده از این شنونده، padding را اعمال کنید وclipToPadding="false"
را نیز اضافه کنید.
چه باید بررسی کرد که آیا برنامه شما باید محافظت پسزمینه سفارشی ارائه دهد
اگر برنامه شما باید محافظت پسزمینه سفارشی را برای ناوبری 3 دکمه یا نوار وضعیت ارائه دهد، برنامه شما باید با استفاده از WindowInsets.Type#tappableElement()
یک قابلیت ترکیب یا نمایش را در پشت نوار سیستم قرار دهد تا ارتفاع نوار ناوبری 3 دکمه یا WindowInsets.Type#statusBars
.
منابع لبه به لبه اضافی
برای ملاحظات بیشتر در مورد اعمال inset ها، راهنمای Edge to Edge Views و Edge to Edge Compose را ببینید.
API های منسوخ شده
API های زیر منسوخ شده اند اما غیرفعال نشده اند:
-
R.attr#enforceStatusBarContrast
-
R.attr#navigationBarColor
(برای پیمایش با 3 دکمه، با 80% آلفا) -
Window#isStatusBarContrastEnforced
-
Window#setNavigationBarColor
(برای پیمایش 3 دکمه، با 80% آلفا) -
Window#setStatusBarContrastEnforced
API های زیر منسوخ و غیرفعال شده اند:
-
R.attr#navigationBarColor
(برای پیمایش با اشاره) -
R.attr#navigationBarDividerColor
-
R.attr#statusBarColor
-
Window#setDecorFitsSystemWindows
-
Window#getNavigationBarColor
-
Window#getNavigationBarDividerColor
-
Window#getStatusBarColor
-
Window#setNavigationBarColor
(برای پیمایش با اشاره) -
Window#setNavigationBarDividerColor
-
Window#setStatusBarColor
پیکربندی پایدار
اگر برنامه شما Android 15 (سطح API 35) یا بالاتر را هدف قرار می دهد، Configuration
دیگر نوارهای سیستم را حذف نمی کند. اگر از اندازه صفحه در کلاس Configuration
برای محاسبه چیدمان استفاده می کنید، باید بسته به نیاز خود، آن را با گزینه های بهتری مانند ViewGroup
مناسب، WindowInsets
یا WindowMetricsCalculator
جایگزین کنید.
Configuration
از API 1 در دسترس بوده است. معمولاً از Activity.onConfigurationChanged
دریافت میشود. اطلاعاتی مانند تراکم، جهت و اندازه پنجره را ارائه می دهد. یکی از ویژگیهای مهم در مورد اندازههای پنجره بازگشتی از Configuration
این است که قبلاً نوارهای سیستم را حذف میکرد.
اندازه پیکربندی معمولاً برای انتخاب منبع استفاده میشود، مانند /res/layout-h500dp
، و این هنوز یک مورد استفاده معتبر است. با این حال، استفاده از آن برای محاسبه چیدمان همیشه دلسرد شده است. اگر این کار را انجام دهید، اکنون باید از آن دور شوید. شما باید بسته به مورد استفاده خود، استفاده از Configuration
را با چیزی مناسب تر جایگزین کنید.
اگر از آن برای محاسبه طرح بندی استفاده می کنید، از یک ViewGroup
مناسب مانند CoordinatorLayout
یا ConstraintLayout
استفاده کنید. اگر از آن برای تعیین ارتفاع نوار ناوبری سیستم استفاده می کنید، از WindowInsets
استفاده کنید. اگر می خواهید اندازه فعلی پنجره برنامه خود را بدانید، از computeCurrentWindowMetrics
استفاده کنید.
لیست زیر فیلدهای تحت تأثیر این تغییر را شرح می دهد:
- اندازههای
Configuration.screenWidthDp
وscreenHeightDp
دیگر نوارهای سیستم را حذف نمیکنند. -
Configuration.smallestScreenWidthDp
به طور غیر مستقیم تحت تأثیر تغییراتscreenWidthDp
وscreenHeightDp
قرار می گیرد. -
Configuration.orientation
به طور غیرمستقیم تحت تأثیر تغییراتscreenWidthDp
وscreenHeightDp
در دستگاه های نزدیک به مربع است. -
Display.getSize(Point)
به طور غیر مستقیم تحت تأثیر تغییراتConfiguration
قرار می گیرد. این در ابتدا در سطح API 30 منسوخ شد. -
Display.getMetrics()
قبلاً از سطح API 33 به این شکل کار کرده است.
صفت elegantTextHeight به طور پیش فرض درست است
برای برنامههایی که Android 15 (سطح API 35) را هدف قرار میدهند، ویژگی elegantTextHeight
TextView
بهطور پیشفرض true
میشود و فونت فشردهای را که بهطور پیشفرض استفاده میشود با برخی از اسکریپتهایی که معیارهای عمودی بزرگی دارند با فونتی که بسیار خواناتر است جایگزین میکند. فونت فشرده برای جلوگیری از شکستن طرحبندیها معرفی شد. Android 13 (سطح API 33) از بسیاری از این شکستگیها جلوگیری میکند و به طرح متن اجازه میدهد تا ارتفاع عمودی را با استفاده از ویژگی fallbackLineSpacing
کشیده شود.
در اندروید 15، فونت فشرده همچنان در سیستم باقی میماند، بنابراین برنامه شما میتواند elegantTextHeight
را روی false
تنظیم کند تا رفتار قبلی را داشته باشد، اما بعید است که در نسخههای آینده پشتیبانی شود. بنابراین، اگر برنامه شما از اسکریپت های زیر پشتیبانی می کند: عربی، لائوس، میانمار، تامیل، گجراتی، کانادا، مالایالام، اودیا، تلوگو یا تایلندی، برنامه خود را با تنظیم elegantTextHeight
روی true
تست کنید.
عرض TextView برای اشکال حروف پیچیده تغییر می کند
در نسخههای قبلی اندروید، برخی از فونتهای شکسته یا زبانهایی که شکل پیچیدهای دارند، ممکن است حروف را در ناحیه شخصیت قبلی یا بعدی بکشند. در برخی موارد، چنین حروفی در موقعیت آغاز یا پایان بریده می شدند. با شروع اندروید 15، یک TextView
عرض را برای ترسیم فضای کافی برای چنین حروفی اختصاص میدهد و به برنامهها اجازه میدهد تا برای جلوگیری از برش، بالشتکهای اضافی را در سمت چپ درخواست کنند.
از آنجایی که این تغییر بر نحوه تعیین عرض یک TextView
تأثیر میگذارد، اگر برنامه Android 15 (سطح API 35) یا بالاتر را هدف قرار دهد، TextView
به طور پیشفرض عرض بیشتری را اختصاص میدهد. می توانید این رفتار را با فراخوانی API setUseBoundsForWidth
در TextView
فعال یا غیرفعال کنید.
از آنجایی که افزودن پد سمت چپ ممکن است باعث ایجاد ناهماهنگی در طرحبندیهای موجود شود، این پد بهطور پیشفرض حتی برای برنامههایی که Android 15 یا بالاتر را هدف قرار میدهند، اضافه نمیشود. با این حال، میتوانید با فراخوانی setShiftDrawingOffsetForStartOverhang
، بالشتک اضافی برای جلوگیری از بریدن اضافه کنید.
مثالهای زیر نشان میدهند که چگونه این تغییرات میتوانند طرحبندی متن را برای برخی از فونتها و زبانها بهبود بخشند.
ارتفاع خط پیشفرض آگاه از محلی برای EditText
در نسخههای قبلی اندروید، طرحبندی متن، ارتفاع متن را به اندازه ارتفاع خط فونتی که با منطقه فعلی مطابقت دارد، افزایش میداد. به عنوان مثال، اگر محتوا به زبان ژاپنی بود، چون ارتفاع خط فونت ژاپنی کمی بزرگتر از فونت لاتین است، ارتفاع متن کمی بزرگتر می شد. با این حال، علیرغم این تفاوتها در ارتفاع خط، عنصر EditText
بدون توجه به منطقه مورد استفاده، همانطور که در تصویر زیر نشان داده شده است، اندازه یکسانی داشت:
برای برنامههایی که Android 15 (سطح API 35) را هدف قرار میدهند، اکنون یک حداقل ارتفاع خط برای EditText
محفوظ است تا با فونت مرجع برای Locale مشخصشده مطابقت داشته باشد، همانطور که در تصویر زیر نشان داده شده است:
در صورت نیاز، برنامه شما میتواند رفتار قبلی را با تعیین ویژگی useLocalePreferredLineHeightForMinimum
به false
بازیابی کند، و برنامه شما میتواند حداقل معیارهای عمودی سفارشی را با استفاده از setMinimumFontMetrics
API در Kotlin و Java تنظیم کند.
دوربین و رسانه
Android 15 تغییرات زیر را در رفتار دوربین و رسانه برای برنامه هایی که اندروید 15 یا بالاتر را هدف قرار می دهند، اعمال می کند.
محدودیت در درخواست فوکوس صوتی
Apps that target Android 15 (API level 35) must be the top app or running a
foreground service in order to request audio focus. If an app
attempts to request focus when it does not meet one of these requirements, the
call returns AUDIOFOCUS_REQUEST_FAILED
.
You can learn more about audio focus at Manage audio focus.
محدودیتهای غیر SDK بهروزرسانی شد
Android 15 شامل لیست های به روز شده از رابط های غیر SDK محدود شده بر اساس همکاری با توسعه دهندگان اندروید و آخرین آزمایش داخلی است. در صورت امکان، قبل از اینکه رابطهای غیر SDK را محدود کنیم، مطمئن میشویم که جایگزینهای عمومی در دسترس هستند.
اگر برنامه شما اندروید 15 را هدف قرار نمی دهد، برخی از این تغییرات ممکن است فوراً روی شما تأثیر نگذارند. با این حال، در حالی که بسته به سطح API هدف برنامه شما، ممکن است برنامه شما به برخی از رابطهای غیر SDK دسترسی داشته باشد، استفاده از هر روش یا فیلد غیر SDK همیشه خطر شکستن برنامه شما را بالا میبرد.
اگر مطمئن نیستید که برنامه شما از رابط های غیر SDK استفاده می کند، می توانید برنامه خود را آزمایش کنید تا متوجه شوید. اگر برنامه شما به رابطهای غیر SDK متکی است، باید برنامهریزی برای انتقال به جایگزینهای SDK را شروع کنید. با این وجود، میدانیم که برخی از برنامهها دارای موارد استفاده معتبر برای استفاده از رابطهای غیر SDK هستند. اگر نمی توانید جایگزینی برای استفاده از یک رابط غیر SDK برای یک ویژگی در برنامه خود پیدا کنید، باید یک API عمومی جدید درخواست کنید .
To learn more about the changes in this release of Android, see Updates to non-SDK interface restrictions in Android 15. To learn more about non-SDK interfaces generally, see Restrictions on non-SDK interfaces.