שקפים במסך הם מעברים ממסך שלם אחד לאחר, והם נפוצים בממשקי משתמש כמו
אשפי הגדרה ומצגות. בנושא הזה נסביר איך לבצע שקפים במסך באמצעות
ViewPager2
לאובייקט. אובייקטים ב-ViewPager2
יכולים להוסיף אנימציה לשקפים במסך באופן אוטומטי. לדוגמה
של שקף במסך שעובר ממסך אחד של תוכן למסך הבא:
אם אתם רוצים להתקדם ולראות דוגמה עובדתית מלאה, צפייה את האפליקציה לדוגמה הזו ב-GitHub.
כדי להשתמש ב-ViewPager2
, צריך להוסיף כמה אפליקציות
תלות של AndroidX
פרויקט. לאחר מכן מבצעים את השלבים שמפורטים בקטעים הבאים.
יצירת התצוגות
יוצרים קובץ פריסה לשימוש מאוחר יותר עבור התוכן של מקטע. צריך להגדיר גם מחרוזת לתוכן של הקטע. הדוגמה הבאה מכילה תצוגת טקסט שמציגה text:
<!-- fragment_screen_slide_page.xml --> <ScrollView xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/content" android:layout_width="match_parent" android:layout_height="match_parent" > <TextView style="?android:textAppearanceMedium" android:padding="16dp" android:lineSpacingMultiplier="1.2" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="@string/lorem_ipsum" /> </ScrollView>
יצירת המקטע
לכתוב
כיתה אחת (Fragment
)
ומחזירה את הפריסה שיצרתם
onCreateView()
. לאחר מכן אפשר ליצור מופעים של המקטע הזה בפעילות ההורה מתי שצריך
דף חדש להצגה למשתמש:
Kotlin
import androidx.fragment.app.Fragment class ScreenSlidePageFragment : Fragment() { override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View = inflater.inflate(R.layout.fragment_screen_slide_page, container, false) }
Java
import androidx.fragment.app.Fragment; ... public class ScreenSlidePageFragment extends Fragment { @Override public View onCreateView(LayoutInflater inflater, ViewGroup container, Bundle savedInstanceState) { return (ViewGroup) inflater.inflate( R.layout.fragment_screen_slide_page, container, false); } }
הוספת ViewPager2
ב-ViewPager2
אובייקטים יש תנועות החלקה מובנות למעבר בין דפים, וגם
הם מציגים אנימציות של השקפים במסך כברירת מחדל, כך שאין צורך ליצור אנימציה משלכם.
ViewPager2
שימושים
FragmentStateAdapter
אובייקטים כספקים של דפים חדשים להצגה, אז FragmentStateAdapter
משתמש
את מחלקת ה-מקטעים שיצרתם.
כדי להתחיל, צריך ליצור פריסה שמכילה אובייקט ViewPager2
:
<!-- activity_screen_slide.xml --> <androidx.viewpager2.widget.ViewPager2 xmlns:android="http://schemas.android.com/apk/res/android" android:id="@+id/pager" android:layout_width="match_parent" android:layout_height="match_parent" />
יוצרים פעילות שמבצעת את הפעולות הבאות:
- הגדרת תצוגת התוכן להיות הפריסה עם
ViewPager2
. - יוצר מחלקה שמרחיבה את המחלקה המופשטת
FragmentStateAdapter
ו מטמיעים אתcreateFragment()
כדי לספק מופעים שלScreenSlidePageFragment
כדפים חדשים. צריך להטמיע אתgetItemCount()
עבור המתאם לזימונית, שמחזירה את מספר הדפים שהמתאם יוצר. יש חמש בדוגמה. - מחבר את
FragmentStateAdapter
לאובייקטיםViewPager2
.
Kotlin
import androidx.fragment.app.Fragment import androidx.fragment.app.FragmentActivity ... /** * The number of pages (wizard steps) to show in this demo. */ private const val NUM_PAGES = 5 class ScreenSlidePagerActivity : FragmentActivity() { /** * The pager widget, which handles animation and allows swiping horizontally * to access previous and next wizard steps. */ private lateinit var viewPager: ViewPager2 override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) setContentView(R.layout.activity_screen_slide) // Instantiate a ViewPager2 and a PagerAdapter. viewPager = findViewById(R.id.pager) // The pager adapter, which provides the pages to the view pager widget. val pagerAdapter = ScreenSlidePagerAdapter(this) viewPager.adapter = pagerAdapter } override fun onBackPressed() { if (viewPager.currentItem == 0) { // If the user is currently looking at the first step, allow the system to handle // the Back button. This calls finish() on this activity and pops the back stack. super.onBackPressed() } else { // Otherwise, select the previous step. viewPager.currentItem = viewPager.currentItem - 1 } } /** * A simple pager adapter that represents 5 ScreenSlidePageFragment objects, in * sequence. */ private inner class ScreenSlidePagerAdapter(fa: FragmentActivity) : FragmentStateAdapter(fa) { override fun getItemCount(): Int = NUM_PAGES override fun createFragment(position: Int): Fragment = ScreenSlidePageFragment() } }
Java
import androidx.fragment.app.Fragment; import androidx.fragment.app.FragmentActivity; ... public class ScreenSlidePagerActivity extends FragmentActivity { /** * The number of pages (wizard steps) to show in this demo. */ private static final int NUM_PAGES = 5; /** * The pager widget, which handles animation and allows swiping horizontally to access previous * and next wizard steps. */ private ViewPager2 viewPager; /** * The pager adapter, which provides the pages to the view pager widget. */ private FragmentStateAdapter pagerAdapter; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_screen_slide); // Instantiate a ViewPager2 and a PagerAdapter. viewPager = findViewById(R.id.pager); pagerAdapter = new ScreenSlidePagerAdapter(this); viewPager.setAdapter(pagerAdapter); } @Override public void onBackPressed() { if (viewPager.getCurrentItem() == 0) { // If the user is currently looking at the first step, allow the system to handle the // Back button. This calls finish() on this activity and pops the back stack. super.onBackPressed(); } else { // Otherwise, select the previous step. viewPager.setCurrentItem(viewPager.getCurrentItem() - 1); } } /** * A simple pager adapter that represents 5 ScreenSlidePageFragment objects, in * sequence. */ private class ScreenSlidePagerAdapter extends FragmentStateAdapter { public ScreenSlidePagerAdapter(FragmentActivity fa) { super(fa); } @Override public Fragment createFragment(int position) { return new ScreenSlidePageFragment(); } @Override public int getItemCount() { return NUM_PAGES; } } }
התאמה אישית של האנימציה באמצעות PageTransformer
כדי להציג אנימציה שונה מהאנימציה שמוגדרת כברירת המחדל של שקף המסך, צריך להטמיע את
ViewPager2.PageTransformer
ולספק אותו לאובייקט ViewPager2
. הממשק חושף
method,
transformPage()
בכל נקודה במעבר של המסך, השיטה הזו מופעלת פעם אחת לכל דף גלוי -
בדרך כלל רק דף גלוי אחד — ולדפים קרובים מחוץ למסך. לדוגמה, אם הדף השלישי
גלוי והמשתמש גורר לכיוון דף רביעי, transformPage()
מופעל בדפים השני,
שלוש וארבע בכל שלב של התנועה.
בהטמעה של transformPage()
, אפשר ליצור שקף מותאם אישית
אנימציות באמצעות קביעת הדפים שצריך לשנות לפי מיקום הדף
במסך. צריך לקבל את מיקום הדף מהפרמטר position
של הפרמטר
transformPage()
.
הפרמטר position
מציין את המיקום של דף נתון ביחס
במרכז המסך. הפרמטר הזה הוא נכס דינמי שמשתנה כאשר המשתמש גולל כדי לעבור
סדרה של דפים. כאשר דף ממלא את המסך, ערך המיקום שלו הוא 0
. כאשר דף
מצויר מהצד הימני של המסך, ערך המיקום שלו הוא 1
. אם המשתמש גולל
באמצע הדף הראשון והשני, בדף הראשון מופיע -0.5
ובדף השני יש מיקום
המיקום של 0.5
. על סמך מיקום הדפים במסך, אפשר ליצור
אנימציות מותאמות אישית של שקפים על ידי הגדרת מאפייני דף באמצעות שיטות כמו
setAlpha()
,
setTranslationX()
,
או
setScaleY()
.
כאשר משתמשים בהטמעה של
PageTransformer
,
לשלוח קריאה
setPageTransformer()
בהטמעה כדי להחיל את האנימציות המותאמות אישית. לדוגמה, אם יש לך
PageTransformer
בשם ZoomOutPageTransformer
, ניתן להגדיר ערך מותאם אישית
אנימציות כמו:
Kotlin
val viewPager: ViewPager2 = findViewById(R.id.pager) ... viewPager.setPageTransformer(ZoomOutPageTransformer())
Java
ViewPager2 viewPager = findViewById(R.id.pager); ... viewPager.setPageTransformer(new ZoomOutPageTransformer());
אפשר לעיין בטרנספורמר של דף ההתרחקות
קטעי טרנספורמר של דף עומק כדי להציג דוגמאות
PageTransformer
שינוי מרחק התצוגה של דפים
כלי הטרנספורמר הזה מכווץ ומעומעם דפים כשגוללים בין דפים קרובים. כדף מתקרב למרכז, הוא צומח בחזרה לגודלו הרגיל ומתעמעם.
Kotlin
private const val MIN_SCALE = 0.85f private const val MIN_ALPHA = 0.5f class ZoomOutPageTransformer : ViewPager2.PageTransformer { override fun transformPage(view: View, position: Float) { view.apply { val pageWidth = width val pageHeight = height when { position < -1 -> { // [-Infinity,-1) // This page is way off-screen to the left. alpha = 0f } position <= 1 -> { // [-1,1] // Modify the default slide transition to shrink the page as well. val scaleFactor = Math.max(MIN_SCALE, 1 - Math.abs(position)) val vertMargin = pageHeight * (1 - scaleFactor) / 2 val horzMargin = pageWidth * (1 - scaleFactor) / 2 translationX = if (position < 0) { horzMargin - vertMargin / 2 } else { horzMargin + vertMargin / 2 } // Scale the page down (between MIN_SCALE and 1). scaleX = scaleFactor scaleY = scaleFactor // Fade the page relative to its size. alpha = (MIN_ALPHA + (((scaleFactor - MIN_SCALE) / (1 - MIN_SCALE)) * (1 - MIN_ALPHA))) } else -> { // (1,+Infinity] // This page is way off-screen to the right. alpha = 0f } } } } }
Java
public class ZoomOutPageTransformer implements ViewPager2.PageTransformer { private static final float MIN_SCALE = 0.85f; private static final float MIN_ALPHA = 0.5f; public void transformPage(View view, float position) { int pageWidth = view.getWidth(); int pageHeight = view.getHeight(); if (position < -1) { // [-Infinity,-1) // This page is way off-screen to the left. view.setAlpha(0f); } else if (position <= 1) { // [-1,1] // Modify the default slide transition to shrink the page as well. float scaleFactor = Math.max(MIN_SCALE, 1 - Math.abs(position)); float vertMargin = pageHeight * (1 - scaleFactor) / 2; float horzMargin = pageWidth * (1 - scaleFactor) / 2; if (position < 0) { view.setTranslationX(horzMargin - vertMargin / 2); } else { view.setTranslationX(-horzMargin + vertMargin / 2); } // Scale the page down (between MIN_SCALE and 1). view.setScaleX(scaleFactor); view.setScaleY(scaleFactor); // Fade the page relative to its size. view.setAlpha(MIN_ALPHA + (scaleFactor - MIN_SCALE) / (1 - MIN_SCALE) * (1 - MIN_ALPHA)); } else { // (1,+Infinity] // This page is way off-screen to the right. view.setAlpha(0f); } } }
רכיב טרנספורמר של דף עומק
שטרנספורמר זה משתמש באנימציית ברירת המחדל של השקף כדי להזיז דפים שמאלה, תוך שימוש "עומק" אנימציה של הזזת דפים ימינה. אנימציית העומק הזו מעמעמת את הדף מקטין אותו באופן לינארי.
במהלך אנימציית העומק, אנימציית ברירת המחדל (שקף במסך) עדיין מתרחשת, לכן פועל נגד השקף במסך באמצעות תרגום X שלילי. לדוגמה:
Kotlin
view.translationX = -1 * view.width * position
Java
view.setTranslationX(-1 * view.getWidth() * position);
הדוגמה הבאה ממחישה איך לבטל את אנימציית ברירת המחדל של השקף במסך בעבודה page transformer:
Kotlin
private const val MIN_SCALE = 0.75f @RequiresApi(21) class DepthPageTransformer : ViewPager2.PageTransformer { override fun transformPage(view: View, position: Float) { view.apply { val pageWidth = width when { position < -1 -> { // [-Infinity,-1) // This page is way off-screen to the left. alpha = 0f } position <= 0 -> { // [-1,0] // Use the default slide transition when moving to the left page. alpha = 1f translationX = 0f translationZ = 0f scaleX = 1f scaleY = 1f } position <= 1 -> { // (0,1] // Fade the page out. alpha = 1 - position // Counteract the default slide transition. translationX = pageWidth * -position // Move it behind the left page. translationZ = -1f // Scale the page down (between MIN_SCALE and 1). val scaleFactor = (MIN_SCALE + (1 - MIN_SCALE) * (1 - Math.abs(position))) scaleX = scaleFactor scaleY = scaleFactor } else -> { // (1,+Infinity] // This page is way off-screen to the right. alpha = 0f } } } } }
Java
@RequiresApi(21) public class DepthPageTransformer implements ViewPager2.PageTransformer { private static final float MIN_SCALE = 0.75f; public void transformPage(View view, float position) { int pageWidth = view.getWidth(); if (position < -1) { // [-Infinity,-1) // This page is way off-screen to the left. view.setAlpha(0f); } else if (position <= 0) { // [-1,0] // Use the default slide transition when moving to the left page. view.setAlpha(1f); view.setTranslationX(0f); view.setTranslationZ(0f); view.setScaleX(1f); view.setScaleY(1f); } else if (position <= 1) { // (0,1] // Fade the page out. view.setAlpha(1 - position); // Counteract the default slide transition. view.setTranslationX(pageWidth * -position); // Move it behind the left page view.setTranslationZ(-1f); // Scale the page down (between MIN_SCALE and 1). float scaleFactor = MIN_SCALE + (1 - MIN_SCALE) * (1 - Math.abs(position)); view.setScaleX(scaleFactor); view.setScaleY(scaleFactor); } else { // (1,+Infinity] // This page is way off-screen to the right. view.setAlpha(0f); } } }
מקורות מידע נוספים
מידע נוסף על ViewPager2
זמין במקורות המידע הנוספים הבאים.
דוגמיות
- ViewPager2 samples (דוגמאות ל-ViewPager2) מופעל GitHub.
סרטונים
- הפיכת הדף: מעבר ל-ViewPager2 (כנס המפתחים של Android לשנת 2019)