הגדרת חלונות מוטמעים

כדי לאפשר לאפליקציה שליטה מלאה במיקום שבו התוכן מוצג, צריך לבצע את שלבי ההגדרה הבאים. אם לא תבצעו את השלבים האלה, יכול להיות שהאפליקציה תצייר צבעים שחורים או אחידים מאחורי ממשק המשתמש של המערכת, או שלא תבצע אנימציה באופן סינכרוני עם המקלדת הווירטואלית.

  1. כדי לאכוף תצוגה מקצה לקצה ב-Android מגרסה 15 ואילך, צריך לטרגט ל-Android 15 (רמת API‏ 35) ומעלה. האפליקציה מוצגת מאחורי ממשק המשתמש של המערכת. אפשר לשנות את ממשק המשתמש של האפליקציה על ידי טיפול בשוליים הפנימיים.
  2. אופציונלי: אפשר לקרוא ל-enableEdgeToEdge() ב-Activity.onCreate(), כדי שהאפליקציה תוצג מקצה לקצה בגרסאות קודמות של Android.
  3. מגדירים את android:windowSoftInputMode="adjustResize" ברשומה של AndroidManifest.xml בפעילות. ההגדרה הזו מאפשרת לאפליקציה לקבל את הגודל של מקלדת ה-IME של התוכנה כהזחות פנימיות, וכך לעזור לכם להחיל את הפריסה והריווח המתאימים כשמקלדת ה-IME מופיעה ונעלמת באפליקציה.

    <!-- In your AndroidManifest.xml file: -->
    <activity
      android:name=".ui.MainActivity"
      android:label="@string/app_name"
      android:windowSoftInputMode="adjustResize"
      android:theme="@style/Theme.MyApplication"
      android:exported="true">
    

שימוש בממשקי API של כתיבה

אחרי שהפעילות משתלטת על הטיפול בכל השוליים הפנימיים, אפשר להשתמש בממשקי API של Compose כדי לוודא שהתוכן לא מוסתר ושהאלמנטים האינטראקטיביים לא חופפים ל-UI של המערכת. ממשקי ה-API האלה מסנכרנים גם את פריסת האפליקציה עם שינויים במיקום.

לדוגמה, זו השיטה הכי בסיסית להחלת השוליים הפנימיים על התוכן של כל האפליקציה:

override fun onCreate(savedInstanceState: Bundle?) {
    super.onCreate(savedInstanceState)

    enableEdgeToEdge()

    setContent {
        Box(Modifier.safeDrawingPadding()) {
            // the rest of the app
        }
    }
}

קטע הקוד הזה מחיל את safeDrawing שוליים פנימיים של חלון כריפוד מסביב לכל התוכן של האפליקציה. אמנם זה מבטיח שרכיבים שאפשר לקיים איתם אינטראקציה לא יחפפו את ממשק המשתמש של המערכת, אבל זה גם אומר שאף חלק מהאפליקציה לא יוצג מאחורי ממשק המשתמש של המערכת כדי ליצור אפקט מקצה לקצה. כדי להשתמש בכל החלון, צריך לכוונן את המיקום של השוליים הפנימיים במסך או ברכיב.

כל סוגי החלונות הקטנים האלה מונפשים אוטומטית באמצעות אנימציות של IME שהועברו לאחור ל-API 21. כתוצאה מכך, גם כל הפריסות שמשתמשות בערכי השוליים האלה מונפשות באופן אוטומטי כשהערכים משתנים.

יש שתי דרכים עיקריות להשתמש בסוגי השוליים הפנימיים האלה כדי להתאים את הפריסות של Composable: משנים את מאפייני הריווח ומשנים את מאפייני הגודל של השוליים הפנימיים.

ערכים לשינוי מרווחים פנימיים

Modifier.windowInsetsPadding(windowInsets: WindowInsets) מחיל את השוליים הפנימיים של החלון שצוינו כריפוד, בדיוק כמו ש-Modifier.padding היה עושה. לדוגמה, Modifier.windowInsetsPadding(WindowInsets.safeDrawing) מוסיף את השוליים של האזור הבטוח לציור כריפוד בכל 4 הצדדים.

יש גם כמה שיטות מובנות לשימוש בסוגי ההזחה הנפוצים ביותר. ‫Modifier.safeDrawingPadding() היא אחת מהשיטות האלה, והיא שוות ערך ל-Modifier.windowInsetsPadding(WindowInsets.safeDrawing). יש משנים דומים לסוגים האחרים של שוליים פנימיים.

שינוי גודל של תוספת

המשנים הבאים מחילים כמות של שוליים פנימיים של חלון על ידי הגדרת הגודל של הרכיב להיות הגודל של השוליים הפנימיים:

Modifier.windowInsetsStartWidth(windowInsets: WindowInsets)

החלת הצד ההתחלתי של windowInsets כרוחב (כמו Modifier.width)

Modifier.windowInsetsEndWidth(windowInsets: WindowInsets)

החלת הצד הסופי של windowInsets כרוחב (כמו Modifier.width)

Modifier.windowInsetsTopHeight(windowInsets: WindowInsets)

החלת הצד העליון של windowInsets כגובה (כמו Modifier.height)

Modifier.windowInsetsBottomHeight(windowInsets: WindowInsets)

הגובה מוגדר לפי הצד התחתון של windowInsets (כמו Modifier.height)

המשנים האלה שימושיים במיוחד כשרוצים לשנות את הגודל של Spacer כך שיתפוס את המקום של השוליים הפנימיים:

LazyColumn(
    Modifier.imePadding()
) {
    // Other content
    item {
        Spacer(
            Modifier.windowInsetsBottomHeight(
                WindowInsets.systemBars
            )
        )
    }
}

צריכה של מוצרים מוטמעים

המשנים של שוליים פנימיים (windowInsetsPadding ועוזרים כמו safeDrawingPadding) צורכים באופן אוטומטי את החלק של השוליים הפנימיים שמוחל כשוליים פנימיים. כשמסתכלים על עץ הקומפוזיציה, רואים ששוליים פנימיים מוזחים (inset) מוטמעים (nested) ומוזחים (inset) בגודל שונה. הם יודעים שחלק מהשוליים הפנימיים כבר נצרכו על ידי שוליים פנימיים מוזחים חיצוניים, ולכן הם לא משתמשים באותו חלק של השוליים הפנימיים יותר מפעם אחת, כדי שלא יהיה יותר מדי רווח נוסף.

גם משני גודל של שוליים פנימיים מונעים שימוש באותו חלק של השוליים הפנימיים יותר מפעם אחת, אם כבר נעשה שימוש בשוליים הפנימיים. עם זאת, מכיוון שהן משנות את הגודל שלהן ישירות, הן לא צורכות את השוליים הפנימיים בעצמן.

כתוצאה מכך, שינויים ב-padding של רכיבים מוטמעים משנים אוטומטית את כמות ה-padding שמוחלת על כל רכיב composable.

אם נסתכל על אותה דוגמה של LazyColumn כמו קודם, נראה שהגודל של LazyColumn משתנה על ידי שינוי המאפיין imePadding. בתוך LazyColumn, הפריט האחרון מוגדר בגובה של החלק התחתון של סרגלי המערכת:

LazyColumn(
    Modifier.imePadding()
) {
    // Other content
    item {
        Spacer(
            Modifier.windowInsetsBottomHeight(
                WindowInsets.systemBars
            )
        )
    }
}

כשסוגרים את ה-IME, לא מוחל ריווח על שינוי הגודל imePadding(), כי אין גובה ל-IME. מכיוון שהמגדיר imePadding() לא מוסיף ריווח פנימי, לא נעשה שימוש בשום שוליים פנימיים, והגובה של Spacer יהיה הגודל של הצד התחתון של סרגלי המערכת.

כשמקלדת ה-IME נפתחת, המרווחים הפנימיים של ה-IME מונפשים בהתאם לגודל של ה-IME, והמשנה imePadding() מתחיל להחיל ריווח פנימי בתחתית כדי לשנות את הגודל של LazyColumn בזמן שה-IME נפתח. כשמשנה המאפיין imePadding() מתחיל להחיל ריווח פנימי תחתון, הוא גם מתחיל להשתמש באותו ריווח שוליים. לכן, הגובה של Spacer מתחיל לרדת, כי המרווח של סרגלי המערכת כבר הוחל על ידי משנה המאפיינים imePadding(). אם משתמשים במגדיר imePadding() כדי להחיל ריווח פנימי תחתון שגדול יותר מסרגלי המערכת, הגובה של Spacer יהיה אפס.

כשמקלדת ה-IME נסגרת, השינויים מתרחשים בסדר הפוך: Spacer מתחיל להתרחב מגובה אפס ברגע ש-imePadding() מוחל על פחות מהצד התחתון של סרגלי המערכת, עד שבסופו של דבר Spacer תואם לגובה של הצד התחתון של סרגלי המערכת ברגע שמקלדת ה-IME מסיימת את האנימציה שלה.

איור 2. עמודה עצלה מקצה לקצה עם TextField.

ההתנהגות הזו מתאפשרת באמצעות תקשורת בין כל משני windowInsetsPadding, ויש עוד כמה דרכים להשפיע עליה.

Modifier.consumeWindowInsets(insets: WindowInsets) גם צורך שוליים פנימיים, כמו Modifier.windowInsetsPadding, אבל הוא לא משתמש בשוליים הפנימיים שנצרכו כמרווח. השימוש הזה שימושי בשילוב עם משני הגודל inset, כדי לציין לאחים שחלק מסוים מהשוליים הפנימיים כבר נוצל:

Column(Modifier.verticalScroll(rememberScrollState())) {
    Spacer(Modifier.windowInsetsTopHeight(WindowInsets.systemBars))

    Column(
        Modifier.consumeWindowInsets(
            WindowInsets.systemBars.only(WindowInsetsSides.Vertical)
        )
    ) {
        // content
        Spacer(Modifier.windowInsetsBottomHeight(WindowInsets.ime))
    }

    Spacer(Modifier.windowInsetsBottomHeight(WindowInsets.systemBars))
}

הפונקציה Modifier.consumeWindowInsets(paddingValues: PaddingValues) מתנהגת באופן דומה מאוד לגרסה עם הארגומנט WindowInsets, אבל היא מקבלת את הארגומנט PaddingValues לשימוש. הדבר שימושי כדי להודיע לילדים מתי יש ריווח או מרווחים שנוצרו באמצעות מנגנון אחר ולא באמצעות משני הריווח הפנימי, כמו Modifier.padding רגיל או מרווחים בגובה קבוע:

Column(Modifier.padding(16.dp).consumeWindowInsets(PaddingValues(16.dp))) {
    // content
    Spacer(Modifier.windowInsetsBottomHeight(WindowInsets.ime))
}

במקרים שבהם נדרשים ערכי השוליים הגולמיים של החלון ללא שימוש, אפשר להשתמש ישירות בערכים של WindowInsets או להשתמש ב-WindowInsets.asPaddingValues() כדי להחזיר PaddingValues של השוליים שלא מושפעים מהשימוש. עם זאת, בגלל האזהרות הבאות, מומלץ להשתמש בשינויי הריפוד של שולי החלון ובשינויי הגודל של שולי החלון בכל מקום שאפשר.

שוליים פנימיים ושלבים ב-Jetpack פיתוח נייטיב

‫Compose משתמשת בממשקי ה-API הבסיסיים של AndroidX כדי לעדכן ולהנפיש את השוליים הפנימיים, שמשתמשים בממשקי ה-API הבסיסיים של הפלטפורמה לניהול השוליים הפנימיים. בגלל אופן הפעולה של הפלטפורמה, יש קשר מיוחד בין המרווחים הפנימיים לבין השלבים של Jetpack Compose.

הערך של insets מתעדכן אחרי שלב היצירה, אבל לפני שלב הפריסה. כלומר, כשקוראים את הערך של המרווחים הפנימיים בהרכבה, בדרך כלל משתמשים בערך של המרווחים הפנימיים שמאחר בפְרֵים אחד. המשנים המובנים שמתוארים בדף הזה נועדו לעכב את השימוש בערכי השוליים עד לשלב הפריסה, כדי להבטיח שהשימוש בערכי השוליים יתבצע באותו פריים שבו הם מתעדכנים.