רבים מממשקי ה-API ליצירת אנימציות מקבלים פרמטרים לצורך התאמה אישית של ההתנהגות שלהם.
התאמה אישית של אנימציות באמצעות הפרמטר AnimationSpec
רוב ממשקי ה-API ליצירת אנימציה מאפשרים למפתחים להתאים אישית את מפרטי האנימציה באמצעות פרמטר AnimationSpec
אופציונלי.
val alpha: Float by animateFloatAsState( targetValue = if (enabled) 1f else 0.5f, // Configure the animation duration and easing. animationSpec = tween(durationMillis = 300, easing = FastOutSlowInEasing), label = "alpha" )
יש סוגים שונים של AnimationSpec
ליצירת סוגים שונים של אנימציה.
יצירת אנימציה מבוססת-פיזיקה באמצעות spring
spring
יוצר אנימציה מבוססת-פיזיקה בין ערכי ההתחלה והסיום. הוא מקבל 2 פרמטרים: dampingRatio
ו-stiffness
.
dampingRatio
מגדיר את מידת ההתנגדות של האביב. ערך ברירת המחדל הוא Spring.DampingRatioNoBouncy
.
איור 1. הגדרת יחסי דעיכה שונים של קפיצים.
stiffness
מגדיר את מהירות התנועה של קפיץ לכיוון ערך הסיום. ערך ברירת המחדל הוא Spring.StiffnessMedium
.
איור 2. הגדרת קשיחות שונה של קפיץ
val value by animateFloatAsState( targetValue = 1f, animationSpec = spring( dampingRatio = Spring.DampingRatioHighBouncy, stiffness = Spring.StiffnessMedium ), label = "spring spec" )
האפשרות spring
יכולה לטפל בהפרעות בצורה חלקה יותר מאשר הסוגים של AnimationSpec
שמבוססים על משך זמן, כי היא מבטיחה את המשכיות המהירות כשערך היעד משתנה במהלך אנימציות. spring
משמש כברירת המחדל של AnimationSpec ב-API רבים של אנימציה, כמו animate*AsState
ו-updateTransition
.
לדוגמה, אם מחילים הגדרת spring
על האנימציה הבאה שמופעל על ידי מגע המשתמש, כשמפריעים לאנימציה במהלך התקדמותה, אפשר לראות שהשימוש ב-tween
לא מגיב בצורה חלקה כמו השימוש ב-spring
.
איור 3. הגדרת מפרט tween
לעומת spring
לאנימציה והפרעה לה.
אנימציה בין ערכי ההתחלה והסיום באמצעות עקומת הפסקה עם tween
tween
מפעיל אנימציה בין ערכי ההתחלה והסיום במהלך durationMillis
שצוין באמצעות עקומת האטה. tween
הוא קיצור של המילה between (בין), כי הוא בין שני ערכים.
אפשר גם לציין את הערך delayMillis
כדי לדחות את תחילת האנימציה.
val value by animateFloatAsState( targetValue = 1f, animationSpec = tween( durationMillis = 300, delayMillis = 50, easing = LinearOutSlowInEasing ), label = "tween delay" )
מידע נוסף זמין במאמר החלקה.
אנימציה לערכים ספציפיים במועדים מסוימים באמצעות keyframes
keyframes
מפעיל אנימציה על סמך ערכי קובץ ה-snapshot שצוינו בחותמות זמן שונות במהלך משך האנימציה. בכל רגע נתון, ערך האנימציה יתבצע אינטרפולציה בין שני ערכים של נקודות מפתח. לכל אחד מ-keyframes האלה אפשר לציין ערך Easing כדי לקבוע את עקומת הביניים.
אפשר לציין את הערכים ב-0 אלפיות שנייה ובמשך הזמן. אם לא מציינים את הערכים האלה, הם מוגדרים כברירת מחדל לערכי ההתחלה והסיום של האנימציה, בהתאמה.
val value by animateFloatAsState( targetValue = 1f, animationSpec = keyframes { durationMillis = 375 0.0f at 0 using LinearOutSlowInEasing // for 0-15 ms 0.2f at 15 using FastOutLinearInEasing // for 15-75 ms 0.4f at 75 // ms 0.4f at 225 // ms }, label = "keyframe" )
אנימציה חלקה בין תמונות מפתח באמצעות keyframesWithSplines
כדי ליצור אנימציה שמתבססת על עקומה חלקה במעבר בין ערכים, אפשר להשתמש במפרטי אנימציה מסוג keyframesWithSplines
במקום במפרטי אנימציה מסוג keyframes
.
val offset by animateOffsetAsState( targetValue = Offset(300f, 300f), animationSpec = keyframesWithSpline { durationMillis = 6000 Offset(0f, 0f) at 0 Offset(150f, 200f) atFraction 0.5f Offset(0f, 100f) atFraction 0.7f } )
פריימים מרכזיים שמבוססים על ספליינים שימושיים במיוחד לתנועה של פריטים במסך ב-2D.
בסרטונים הבאים מוצגים ההבדלים בין keyframes
ל-keyframesWithSpline
, תוך שימוש באותה קבוצה של קואורדינטות x, y שצריכות להגדיר עיגול.
keyframes
|
keyframesWithSplines
|
---|---|
כפי שאפשר לראות, מפתחות ה-keyframe שמבוססים על spline מציעים מעברים חלקים יותר בין הנקודות, כי הם משתמשים בקו עקומה של Bezier כדי ליצור אנימציה חלקה בין הפריטים. המפרט הזה שימושי ליצירת אנימציה מוגדרת מראש. עם זאת, אם אתם עובדים עם נקודות שמנוהלות על ידי משתמשים, עדיף להשתמש ב-springs כדי להשיג חלקות דומה בין הנקודות, כי הן ניתנות להפרעה.
איך חוזרים על אנימציה באמצעות repeatable
repeatable
מפעילה אנימציה שמבוססת על משך זמן (כמו tween
או keyframes
) שוב ושוב עד שהיא מגיעה למספר החזרות שצוין. אפשר להעביר את הפרמטר repeatMode
כדי לציין אם האנימציה צריכה לחזור על עצמה מההתחלה (RepeatMode.Restart
) או מהסוף (RepeatMode.Reverse
).
val value by animateFloatAsState( targetValue = 1f, animationSpec = repeatable( iterations = 3, animation = tween(durationMillis = 300), repeatMode = RepeatMode.Reverse ), label = "repeatable spec" )
איך חוזרים על אנימציה ללא הגבלה באמצעות infiniteRepeatable
infiniteRepeatable
דומה ל-repeatable
, אבל היא חוזרת על עצמה במשך מספר אינסופי של חזרות.
val value by animateFloatAsState( targetValue = 1f, animationSpec = infiniteRepeatable( animation = tween(durationMillis = 300), repeatMode = RepeatMode.Reverse ), label = "infinite repeatable" )
בבדיקות שמשתמשות ב-ComposeTestRule
, אנימציות שמשתמשות ב-infiniteRepeatable
לא מופעלות. הרכיב יומר באמצעות הערך הראשוני של כל ערך מונפש.
מעבר מיידי לערך הסיום באמצעות snap
snap
הוא AnimationSpec
מיוחד שמעביר את הערך באופן מיידי לערך הסופי. אפשר לציין את הערך delayMillis
כדי לעכב את תחילת האנימציה.
val value by animateFloatAsState( targetValue = 1f, animationSpec = snap(delayMillis = 50), label = "snap spec" )
הגדרת פונקציית העברה בהתאמה אישית
פעולות AnimationSpec
שמבוססות על משך זמן (כמו tween
או keyframes
) משתמשות ב-Easing
כדי לשנות את החלק של האנימציה. כך הערך שמופיע באנימציה יכול להאיץ ולהאט, במקום לנוע בקצב קבוע. השבר הוא ערך בין 0 (התחלה) ל-1.0 (סיום) שמציין את הנקודה הנוכחית באנימציה.
למעשה, העקומה היא פונקציה שמקבלת ערך שבר בין 0 ל-1.0 ומחזירה ערך של float. הערך המוחזר יכול להיות מחוץ לגבול כדי לייצג חריגה מעל או מתחת לערך היעד. אפשר ליצור עקומת Easing בהתאמה אישית כמו הקוד שבהמשך.
val CustomEasing = Easing { fraction -> fraction * fraction } @Composable fun EasingUsage() { val value by animateFloatAsState( targetValue = 1f, animationSpec = tween( durationMillis = 300, easing = CustomEasing ), label = "custom easing" ) // …… }
ב-Compose יש כמה פונקציות Easing
מובנות שמכסות את רוב תרחישי השימוש.
למידע נוסף על בחירת פונקציית העברה בהתאם לתרחיש, ראו מהירות – Material Design.
FastOutSlowInEasing
LinearOutSlowInEasing
FastOutLinearEasing
LinearEasing
CubicBezierEasing
- מידע נוסף
אנימציה של סוגי נתונים מותאמים אישית על ידי המרה אל AnimationVector
וממנו
רוב ממשקי ה-API ליצירת אנימציות ב-Compose תומכים ב-Float
, ב-Color
, ב-Dp
ובסוגי נתונים בסיסיים אחרים כערכים של אנימציה כברירת מחדל, אבל לפעמים צריך ליצור אנימציה של סוגי נתונים אחרים, כולל סוגי נתונים מותאמים אישית. במהלך האנימציה, כל ערך שמשתנה מיוצג כ-AnimationVector
. הערך מומר ל-AnimationVector
ולהפך באמצעות TwoWayConverter
תואם, כדי שמערכת האנימציה המרכזית תוכל לטפל בהם באופן אחיד. לדוגמה, Int
מיוצג כ-AnimationVector1D
שמכיל ערך נקודה צפה יחיד.
TwoWayConverter
עבור Int
נראה כך:
val IntToVector: TwoWayConverter<Int, AnimationVector1D> = TwoWayConverter({ AnimationVector1D(it.toFloat()) }, { it.value.toInt() })
Color
הוא למעשה קבוצה של 4 ערכים, אדום, ירוק, כחול ואלפא, ולכן Color
מומר ל-AnimationVector4D
שמכיל 4 ערכים מסוג float. כך, כל סוג נתונים שמשמש באנימציות מומר ל-AnimationVector1D
, ל-AnimationVector2D
, ל-AnimationVector3D
או ל-AnimationVector4D
, בהתאם למאפייני המימדים שלו. כך אפשר ליצור אנימציה של רכיבים שונים באובייקט בנפרד, לכל אחד מהם עם מעקב מהירות משלו. אפשר לגשת לממירים מובנים של סוגי נתונים בסיסיים באמצעות ממירים כמו Color.VectorConverter
או Dp.VectorConverter
.
אם רוצים להוסיף תמיכה בסוג נתונים חדש כערך מונפש, אפשר ליצור TwoWayConverter
משלכם ולספק אותו ל-API. לדוגמה, אפשר להשתמש ב-animateValueAsState
כדי ליצור אנימציה לסוג הנתונים המותאם אישית באופן הבא:
data class MySize(val width: Dp, val height: Dp) @Composable fun MyAnimation(targetSize: MySize) { val animSize: MySize by animateValueAsState( targetSize, TwoWayConverter( convertToVector = { size: MySize -> // Extract a float value from each of the `Dp` fields. AnimationVector2D(size.width.value, size.height.value) }, convertFromVector = { vector: AnimationVector2D -> MySize(vector.v1.dp, vector.v2.dp) } ), label = "size" ) }
הרשימה הבאה כוללת כמה VectorConverter
מובנים:
Color.VectorConverter
Dp.VectorConverter
Offset.VectorConverter
Int.VectorConverter
Float.VectorConverter
IntSize.VectorConverter
מומלץ עבורך
- הערה: טקסט הקישור מוצג כש-JavaScript מושבת
- אנימציות מבוססות-ערך
- פיתוח קוד איטרטיבי {:#iterative-code-dev }
- אנימציות ב'כתיבה'