ppeper
by ppeper
3 분 소요

Categories

Tags

🚀 ViewPager2

안드로이드 어플을 개발을 하다보면 RecyclerView를 통하여 데이터 목록을 보여주는 경우가 많다. 안드로이드에서 데이터의 목록이 아니라 Pager 형식의 ViewHolder를 구현하려면 ViewPager2를 사용을 해야한다.

ViewPager와 ViewPager2

ViewPager2는 AndroidX가 발표된 이후에 새롭게 나온 ViewPager로 구글 공식문서에서 아래와 같은 기능을 지원하여 적극 사용을 권장하고 있다.

  • ViewPager2는 세로방향의 페이징을 제공하고 있다. (orientation의 속성)
  • RTL(Right To Left) 페이징을 지원한다.
    • RTL 페이징의 경우 설정된 언어에 따라 자동으로 사용 설정되지만 layoutDirection의 속성으로 수동으로 설정할 수 있다.
  • 수정 가능한 Framgent Collection을 통해 페이징을 지원하며 notifyDataSetChanged()를 호출하여 UI를 업데이트 한다.
    • 📌 앱은 런타임 시 Fragment Collection을 동적으로 수정할 수 있다.

ViewPager2 는 RecyclerView를 기반으로 만들어진 컴포넌트로 RecyclerView.AdapterFragmentStateAdapter 두 가지 방법으로 구현이 가능하다.


사용하기 (RecyclerView.Adapter)

build.gradle 설정

ViewPager2를 사용하기 위해서 build.gradle에 종속성을 추가한다.

dependencies {
    implementation "androidx.viewpager2:viewpager2:1.0.0"
}

ViewPager2 layout 설정

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">

    <androidx.viewpager2.widget.ViewPager2
        android:id="@+id/viewPager1"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="0.5"
        android:orientation="horizontal" />

</LinearLayout>

layout file에서 위와 같이 android:orientation="horizontal"로 설정이 가능하고.

Programmically로도 설정할 수 있다. viewPager2.orientation = ViewPager2.ORIENTATION_HORIZONTAL

Adapter 설정

간단하게 페이지의 title number를 보여주는 ViewPager Adapter를 만들어 준다.

data class Data(
    var title: String
)

class ViewPager2Adapter(
    private val list: ArrayList<Data>
): RecyclerView.Adapter<ViewPager2Adapter.CustomViewHolder>() {

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): CustomViewHolder {
        val binding = ItemListBinding.inflate(LayoutInflater.from(parent.context), parent, false)
        return CustomViewHolder(binding)
    }

    override fun onBindViewHolder(holder: CustomViewHolder, position: Int) {
        holder.bind(list[position])
    }

    override fun getItemCount() = list.size

    inner class CustomViewHolder(private val binding: ItemListBinding) :
        RecyclerView.ViewHolder(binding.root) {

        fun bind(data: Data) {
            with(binding) {
                textView.text = data.title
            }
        }
    }
}

Activity에서 사용

사용할 Activity에서 adpater를 붙여주어 사용한다.

class MainActivity : AppCompatActivity() {
    private lateinit var binding: ActivityMainBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(binding.root)

        val list: ArrayList<Data> = ArrayList()
        list.apply {
            add(Data("First"))
            add(Data("Second"))
            add(Data("Third"))
        }
        binding.viewPager1.adapter = ViewPager2Adapter(list)
        // binding.viewPager1.orientation = ViewPager2.ORIENTATION_HORIZONTAL
    }
}

Tablayout 사용하기

layout 파일에 TabLayout을 추가한다.

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">

    <com.google.android.material.tabs.TabLayout
        android:id="@+id/tabLayout1"
        android:layout_width="match_parent"
        android:layout_height="wrap_content" />

    ...

</LinearLayout>

TabLayoutViewPager 간의 연결은 TabLayoutMediator 를 구현한다.

TabLayoutMediator 에는 TabLayout과 ViewPager2 객체를 전달하고 attach() 함수를 호출하여 연동한다.

구현된 인터페이스는 Tab을 눌렀을때 해당하는 position마다 tab title text를 설정해 줄 수 있다.

class MainActivity : AppCompatActivity() {
        .
        .
        .
        // TabLayout 추가
        TabLayoutMediator(binding.tabLayout1, binding.viewPager1) { tab, position ->
            tab.text = "Tab ${position + 1}"
        }.attach()
    }
}

사용하기 (FragmentStateAdapter)

Fragment를 보여주도록 하는 Adapter가 바로 FragmentAdapter 이다. 이는 RecyclerView의 Adapter를 만들때 RecyclerView.Adapter를 상속 받아 만들었을때와 같은 Adapter이다.

FragmentStateAdapter 는 RecyclerView에서 Adapter에 ViewHolder를 지정하는 것과 마찬가지로, FragmentViewHolder 가 지정되어 구현 된다.

FragmentStateAdapter 를 구현하기 위해서는 getItemCount()createFragment()를 반드시 구현해야 한다.

🔔 구글 공식 문서에서 FragmentStateAdapter() 는 3종류의 생성자가 있다

    public FragmentStateAdapter(@NonNull FragmentActivity fragmentActivity) {
        this(fragmentActivity.getSupportFragmentManager(), fragmentActivity.getLifecycle());
    }

    /**
     * @param fragment if the {@link ViewPager2} lives directly in a {@link Fragment} subclass.
     *
     * @see FragmentStateAdapter#FragmentStateAdapter(FragmentActivity)
     * @see FragmentStateAdapter#FragmentStateAdapter(FragmentManager, Lifecycle)
     */
    public FragmentStateAdapter(@NonNull Fragment fragment) {
        this(fragment.getChildFragmentManager(), fragment.getLifecycle());
    }

    /**
     * @param fragmentManager of {@link ViewPager2}'s host
     * @param lifecycle of {@link ViewPager2}'s host
     *
     * @see FragmentStateAdapter#FragmentStateAdapter(FragmentActivity)
     * @see FragmentStateAdapter#FragmentStateAdapter(Fragment)
     */
    public FragmentStateAdapter(@NonNull FragmentManager fragmentManager,
            @NonNull Lifecycle lifecycle) {
        mFragmentManager = fragmentManager;
        mLifecycle = lifecycle;
        super.setHasStableIds(true);
    }

1. FragmentStateAdapter(FragmentActivity fragmentActivity)

  • 주로 ViewPager2가 Activity내에서 동작할 때 사용된다.

2. FragmentStateAdapter(Fragment fragment)

  • Fragment 내에서 동작하는 ViewPager2를 만들 때 사용한다.

3. FragmentStateAdapter(FragmentManager fragmentManager, Lifecycle lifecycle)

  • fragmentManagerlifecycle이 호출되는 것으로 보아 1, 2번의 생성자 함수에서 호출되는 함수이다.

예시에서는 CustomFragment를 두개 생성하여 FragmentActivity로 받아 사용하였다.

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity">

    ...

    <com.google.android.material.tabs.TabLayout
        android:id="@+id/tabLayout2"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>

    <androidx.viewpager2.widget.ViewPager2
        android:id="@+id/viewPager2"
        android:layout_width="match_parent"
        android:layout_height="0dp"
        android:layout_weight="0.5"
        android:orientation="horizontal" />

</LinearLayout>
class ViewPagerAdapter(fa: FragmentActivity): FragmentStateAdapter(fa){
    override fun getItemCount() = 2
    override fun createFragment(position: Int): Fragment {
        return when (position) {
            0 -> CustomFragment1()
            1 -> CustomFragment2()
            else -> throw IndexOutOfBoundsException()
        }
    }
}

Fragment 생성

Tab을 눌렀을때 Fragment를 통하여 보여주도록 CustomFragment(1,2)를 생성해 준다.

class CustomFragment1 : Fragment() {
    private lateinit var binding: FragmentCustomBinding

    override fun onCreateView(
        inflater: LayoutInflater, container: ViewGroup?,
        savedInstanceState: Bundle?
    ): View {
        binding = FragmentCustomBinding.inflate(inflater, container, false)
        return binding.root
    }
}

Activity에서 사용

Acitivty <- FragmentActivity <- AppCompatActivity로 MainActivity에서 ViewPagerAdapter(this) 로 adapter를 설정할 수 있다.

class MainActivity : AppCompatActivity() {
    private lateinit var binding: ActivityMainBinding

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = ActivityMainBinding.inflate(layoutInflater)
        setContentView(binding.root)

        ...

        binding.viewPager2.adapter = ViewPagerAdapter(this)
        // TabLayout 추가
        TabLayoutMediator(binding.tabLayout2, binding.viewPager2) { tab, position ->
            tab.text = "Fragment ${position + 1}"
        }.attach()
    }
}


References