2
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

TabbedActivityのサンプルプロジェクトをViewPager2化する

Last updated at Posted at 2021-05-03

TabLayout+ViewPagerな画面を作る場合、新規ActivityのGaralyにあるTabbedActivityが参考になります。



ただ、Android Studio 4.1.3では、タブのフォアグラウンドとバックグラウンドが同じ色になっていて、テキストもインジケータも見えないとかちょっとおかしなことになっています。
タイトル部分もToolbarと思いきやTextViewですね。どうしてでしょ?

ViewPager2にする話をしたいのですが、その前にちょっと手直ししておきます。

ViewPagerサンプルとしての手直し

レイアウトはTextViewをToolbarに置き換えておきます(FloatingActionButtonは本題から外れるので外しています)

activity_view_pager.xml
<?xml version="1.0" encoding="utf-8"?>
<androidx.coordinatorlayout.widget.CoordinatorLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    tools:context=".a.ViewPagerActivity"
    >

    <com.google.android.material.appbar.AppBarLayout
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:theme="@style/Theme.MyApplication.AppBarOverlay"
        >

        <androidx.appcompat.widget.Toolbar
            android:id="@+id/toolbar"
            android:layout_width="match_parent"
            android:layout_height="?attr/actionBarSize"
            android:background="?attr/colorPrimary"
            app:popupTheme="@style/Theme.MyApplication.PopupOverlay"
            />

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

    <androidx.viewpager.widget.ViewPager
        android:id="@+id/view_pager"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        app:layout_behavior="@string/appbar_scrolling_view_behavior"
        />
</androidx.coordinatorlayout.widget.CoordinatorLayout>

配色の問題についてはこれだけで解決します。

ActivityについてはViewBindingを使うように修正していますがほぼ変更はありません。

ViewPagerActivity.kt
class ViewPagerActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        val binding = ActivityViewPagerBinding.inflate(layoutInflater)
        setContentView(binding.root)
        setSupportActionBar(binding.toolbar)
        val sectionsPagerAdapter = SectionsPagerAdapter(this, supportFragmentManager)
        binding.viewPager.adapter = sectionsPagerAdapter
        binding.tabs.setupWithViewPager(binding.viewPager)
    }
}

ViewModelについてはそのままにしています。

PageViewModel.kt
class PageViewModel : ViewModel() {
    private val _index = MutableLiveData<Int>()
    val text: LiveData<String> = Transformations.map(_index) { "Hello world from section: $it" }

    fun setIndex(index: Int) {
        _index.value = index
    }
}

PagerAdapterについては結構大きく変えました。

FragmentPagerAdapterはFragmentManagerだけのコンストラクタはDeprecatedのため、新しいbehaviorを指定するコンストラクタを使い、BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENTを指定します。behaviorを省略した場合BEHAVIOR_SET_USER_VISIBLE_HINTを指定したのと同じ挙動になりますが、こちらの動作自体がDeprecatedです。
BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENTを指定すると、名前の通り、ユーザーに見えているFragmentだけがResume状態となるように動作します。旧来の動作では、表示されていないFragmentもResume状態となり、ユーザに見えているかどうかはsetUserVisibleHintで検出していました。この動作はDeprecatedなので今後は使わないようにしましょう。
なお、ViewPager2の動作もBEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENTと同様です。

コンストラクタではViewModelの初期設定を行っています。サンプルではなぜかPageViewModelをFragmentスコープで持ち、Fragmentで初期化するということをやっています。これだと、FragmentがリサイクルされたときにViewModelも失われるので表示状態を保持することができません。ここにあるようにActivityスコープでページのポジションをキーとして保持するようにする必要があります。

SectionsPagerAdapter.kt
private val TAB_TITLES = arrayOf(
    R.string.tab_text_1,
    R.string.tab_text_2
)

class SectionsPagerAdapter(
    private val activity: FragmentActivity, fm: FragmentManager
) : FragmentPagerAdapter(fm, BEHAVIOR_RESUME_ONLY_CURRENT_FRAGMENT) {
    init {
        repeat(count) {
            activity.pageViewModels<PageViewModel>({ it }).value.setIndex(it + 1)
        }
    }

    override fun getItem(position: Int): Fragment =
        PlaceholderFragment.newInstance(position)

    override fun getPageTitle(position: Int): CharSequence =
        activity.resources.getString(TAB_TITLES[position])

    override fun getCount(): Int = 2
}

pageViewModelsはこういう定義になっています。keyedViewModelsについては「keyを指定したViewModelの取得もlazy拡張関数で簡単にしたい」をご参考

ViewModelExtensions.kt
@MainThread
inline fun <reified VM : ViewModel> ComponentActivity.pageViewModels(
    noinline pageIndex: () -> Int,
    noinline factoryProducer: (() -> ViewModelProvider.Factory)? = null
): Lazy<VM> = keyedViewModels({ pageIndex().toString() }, factoryProducer)

@MainThread
inline fun <reified VM : ViewModel> Fragment.pageViewModels(
    noinline pageIndex: () -> Int,
    noinline factoryProducer: (() -> ViewModelProvider.Factory)? = null
): Lazy<VM> = keyedActivityViewModels({ pageIndex().toString() }, factoryProducer)

最後にFragmentです。コンストラクタでレイアウトを指定し、onCreateやonCreateViewはoverrideしません。
代わりにonViewCreatedでラベル設定を行います。
サンプルのままでは、FragmentでLiveDataをobserveするときはlifecycleOwnerはviewLifecycleOwnerを使えと怒られるので怒られないように修正。

PlaceholderFragment.kt
class PlaceholderFragment : Fragment(R.layout.fragment_view_pager) {
    private val pageViewModel: PageViewModel by pageViewModels({
        requireArguments().getInt(KEY_POSITION)
    })

    override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
        val binding = FragmentViewPagerBinding.bind(view)
        pageViewModel.text.observe(viewLifecycleOwner) {
            binding.sectionLabel.text = it
        }
    }

    companion object {
        private const val KEY_POSITION = "ARG_POSITION"

        @JvmStatic
        fun newInstance(sectionNumber: Int): PlaceholderFragment =
            PlaceholderFragment().apply {
                arguments = bundleOf(KEY_POSITION to sectionNumber)
            }
    }
}

ViewPager2化する

ここから、前項でViewPagerで実装していた部分をViewPager2に置き換えていきます。
同じ構成にするならViewModel/Fragmentは変更する必要はありません。

PagerAdapterはFragmentPagerAdapterもしくはFragmentStatePagerAdapterを使っているところを、FragmentStateAdapterに置き換えます。

getItemの代わりにcreateFragment、getCountの代わりにgetItemCountを実装します。
TabLayoutにタイトルを伝えるgetPageTitleに相当するメソッドはありません。この処理はActivityの方に移動させています。

SectionsPagerAdapter.kt
class SectionsPagerAdapter(
    activity: FragmentActivity
) : FragmentStateAdapter(activity) {
    init {
        repeat(itemCount) {
            activity.pageViewModels<PageViewModel>({ it }).value.setIndex(it + 1)
        }
    }

    override fun getItemCount(): Int = 2

    override fun createFragment(position: Int): Fragment =
        PlaceholderFragment.newInstance(position)
}

レイアウトでは、ViewPagerのところをViewPager2に置き換えます

activity_view_pager.xml
-     <androidx.viewpager.widget.ViewPager
+     <androidx.viewpager2.widget.ViewPager2

AcitivtyではTabLayoutとViewPagerの接続部分を変更します。
setupWithViewPagerは使えないので、TabLayoutMediatorを使って実装します。
こちらはtabへの反映処理を実装できるので、単にテキストを変更する以上の操作も可能ですね。

ViewPagerActivity.kt
private val TAB_TITLES = arrayOf(
    R.string.tab_text_1,
    R.string.tab_text_2
)

class ViewPagerActivity : AppCompatActivity() {
    private lateinit var binding: ActivityViewPager2Binding
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        binding = ActivityViewPager2Binding.inflate(layoutInflater)
        setContentView(binding.root)
        setSupportActionBar(binding.toolbar)
        val sectionsPagerAdapter = SectionsPagerAdapter(this)
        binding.viewPager.adapter = sectionsPagerAdapter
        TabLayoutMediator(binding.tabs, binding.viewPager) { tab, position ->
            tab.text = getString(TAB_TITLES[position])
        }.attach()
    }
}

以上が、ViewPager2に対応したTabbedActivityのサンプルになります。

2
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
2
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?