להחליק בין מקטעים באמצעות ViewPager2

רוצה לנסות את שיטת הכתיבה?
'Jetpack פיתוח נייטיב' היא ערכת הכלים המומלצת לממשק המשתמש ל-Android. איך משתמשים בזימונית ב'כתיבה'.

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

איור 1. אנימציה של שקף במסך.
 

אם אתם רוצים להתקדם ולראות דוגמה עובדתית מלאה, צפייה את האפליקציה לדוגמה הזו ב-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

שינוי מרחק התצוגה של דפים

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

איור 2. דוגמה אחת (ZoomOutPageTransformer).
 

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);
        }
    }
}

רכיב טרנספורמר של דף עומק

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

איור 3. דוגמה אחת (DepthPageTransformer).
 

במהלך אנימציית העומק, אנימציית ברירת המחדל (שקף במסך) עדיין מתרחשת, לכן פועל נגד השקף במסך באמצעות תרגום 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 זמין במקורות המידע הנוספים הבאים.

דוגמיות

סרטונים