LoginSignup
2
2

More than 3 years have passed since last update.

[Android]Epoxy と Paging を組み合わせたサンプルと解説

Last updated at Posted at 2020-08-22

はじめに

Epoxyと Paging を組み合わせてページ毎にデータを読み込むリストを作成してみたいと思います。
アーキテクチャは Google が推奨している MVVM を利用して、次の手順で作成を進めます。

No タイトル
Step 0 必要なライブラリをセットアップする
Step 1 Retorift でページ毎のデータを取得する
Step 2 PagedList でページ毎のデータを管理する
Step 3 EpoxyRecyclerView で表示する EpoxyModel を作成する
Step 4 EpoxyRecyclerView で表示を制御する EpoxyController を作成する
Step 5 作成したクラスの動作を確認する

完成イメージは次のような感じになります。

Image from Gyazo

アプリの構成

  • PagedList がページ毎のデータの管理をする。
  • PagedList を生成するには DataSource・DataSource.Factory が必要になる。
  • PagedList を EpoxyRecyclerView で表示するには PagedListEpoxyController が必要になる。
  • ページごとのデータ管理は PageList が中心となって行う、
    そのため PagedList の生成や表示に必要となる周辺のクラスを実装していく必要がある。
    それらの作成するクラスをまとめると次の表や図のような感じになる。
分類 名称 説明
View PagedListEpoxyController EpoxyRecyclerView に PagedList を表示するための Controller クラスの実装
ViewModel LiveData<PagedList<Item>> ページ毎のデータ管理を行うためのクラスの実装
Model PageKeyedDataSource PagedListを生成するのに必要となるクラスの実装
Model Data Source Factory PagedListを生成するのに必要となるクラスの実装
Model Service Class ページ毎のデータ取得するクラスの実装

Image from Gyazo

Step 0 必要なライブラリをセットアップする

アプリケーションの作成に必要となる、Epoxy・Retrofit・Gson などのライブラリをインストールする。
これらのライブラリを使うには kapt や databinding、 Java1.8 を有効にする必要があるのでこちらも設定しておきます。

ライブラリ バージョン 説明
Epoxy 3.11.0 RecyclerView ライブラリ
Retrofit 2.9.0 HTTP クライアントライブラリ
Gson 2.8.5 JSONデータをJavaオブジェクトの相互変換ライブラリ
Paging 2.1.2 Paging ライブラリ
apply plugin: 'kotlin-kapt'

android {
    
    
    compileOptions {
        sourceCompatibility JavaVersion.VERSION_1_8
        targetCompatibility JavaVersion.VERSION_1_8
    }

    kotlinOptions {
        jvmTarget = "1.8"
    }

    dataBinding {
        enabled = true
    }
    
    
}

dependencies {
    
    
    def epoxy_version = "3.11.0"
    implementation "com.airbnb.android:epoxy:$epoxy_version"
    implementation "com.airbnb.android:epoxy-databinding:${epoxy_version}"
    implementation "com.airbnb.android:epoxy-paging:${epoxy_version}"
    kapt "com.airbnb.android:epoxy-processor:$epoxy_version"

    def retrofit_version ="2.9.0"
    implementation "com.squareup.retrofit2:retrofit:${retrofit_version}"
    implementation "com.squareup.retrofit2:converter-gson:${retrofit_version}"

    def gson_version = "2.8.5"
    implementation "com.google.code.gson:gson:${gson_version}"

    def paging_version = "2.1.2"
    implementation "androidx.paging:paging-runtime:$paging_version" 
    
    
}

あとは Retrofit でネットワークを使って通信を行うので AndroidManifest.xml にて android.permission.INTERNET の権限を追加しておきます。

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="jp.kaleidot725.sample">

    <uses-permission android:name="android.permission.INTERNET" />

    <application
        android:allowBackup="true"
        android:icon="@mipmap/ic_launcher"
        android:label="@string/app_name"
        android:roundIcon="@mipmap/ic_launcher_round"
        android:supportsRtl="true"
        android:theme="@style/AppTheme">
        <activity android:name=".ui.MainActivity">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>
    </application>

</manifest>

Step 1 Retorift でページ毎のデータを取得する

今回はページ単位の情報を取得する API として Qiita の記事一覧の取得 API ( GET /api/v2/items ) を利用します。HTTP Client には Retrofit を利用します。なので次のクラスを作成して API を呼び出せるようにしていきます。

役割 クラス名 役割
Entity Item Qiitaの記事データを表すデータクラス
Service QiitaService Qiita API を利用するためのサービスクラス
Data Source ItemDataSource QiitaService を利用してページ毎のデータを取得するクラス
Data Source Factory ItemDataSourceFactory ItemDataSource を生成するクラス

Item
Qiitaの記事データを表すデータクラスを宣言します。Item には Group や Tag や User といった情報も含まれるのであわせて定義していきます。API で利用するデータクラスはJSON To Kotlin Classを利用すると JSON を入力するだけで自動生成してくれるので楽です。

data class Item(
    val body: String, val editing: Boolean, val comments_count: Int, val created_at: String,
    val group: Group, val id: String, val likes_count: Int, val page_views_count: Int, val `private`: Boolean,
    val reactions_count: Int, val rendered_body: String, val tags: List<Tag>, val title: String, val updated_at: String,
    val url: String, val user: User
)

data class Group(
    val created_at: String, val id: Int, val name: String,
    val `private`: Boolean, val updated_at: String, val url_name: String
)

data class Tag(val name: String, val versions: List<String>)

data class User(
    val description: String, val facebook_id: String, val followees_count: Int,
    val followers_count: Int, val github_login_name: String, val id: String, val items_count: Int,
    val linkedin_id: String, val location: String, val name: String, val organization: String, val permanent_id: Int,
    val profile_image_url: String, val team_only: Boolean, val twitter_screen_name: String, val website_url: String
)

QiitaService

Retrofit で Qiita API の GET /api/v2/items を利用できるようにします。今回はページ毎に取得したいので page や per_page のクエリを追加しておきます。Retrofit の実装方法の詳細については公式ドキュメント を閲覧してください。

interface QiitaService {
    @GET("/api/v2/items")
    fun getItems(@Query("page")page: Int, @Query("per_page") perPage: Int): Call<List<Item>>
}

Step2 PagedList でページ毎のデータを管理する

PagedListの生成方法

ページ毎に API からデータを取得する場合は PagedList を生成します。 PagedList の生成は LivePagedListBuilder で行いますが、そのときに DataSource と DataSource.Factory が必要になるので実装します。

ItemDataSource

DataSource で実際にどのようなデータをページ毎に生成するか決めます。今回は RecyclerView が表示されたら 1ページ目のデータ、末尾に到達したら次のページのデータを表示するように実装してみます。(データ取得には先程作成した、Qiitaの記事一覧取得処理を利用します。)

// API呼び出しをしているので、本来であればここで例外の対処を記述する必要がありますが省略しています。
class ItemDataSource(private val service: QiitaService) : PageKeyedDataSource<Int, Item>() {
    override fun loadBefore(params: LoadParams<Int>, callback: LoadCallback<Int, Item>) {}

    // API呼び出しをしているので、本来であればここで例外の対処を記述する必要がありますが省略しています。
    override fun loadInitial(params: LoadInitialParams<Int>, callback: LoadInitialCallback<Int, Item>) {
        // 1 ページ目のデータを取得する
        val page = 1

        // 1 ページに表示するデータ数
        val perPage = params.requestedLoadSize

        // ページに表示するデータを取得する
        val items = service.getItems(page, perPage).execute().body() ?: emptyList()

        // 次に表示するページの番号を計算する
        val nextPage = page + 1

        // 取得したデータ、次に表示するページの番号を結果として返す
        callback.onResult(items , null, nextPage)
    }

    override fun loadAfter(params: LoadParams<Int>, callback: LoadCallback<Int, Item>) {
        // params.key には 前の loadInitial や loadAfter の呼び出しで返した nextPage が格納されている
        val page = params.key // 1 ページ目のデータを取得する

        // params.requestedLoadSize には 1ページに表示するデータ数が格納されている。
        val perPage = params.requestedLoadSize

        // ページに表示するデータを取得する
        val items = service.getItems(page, perPage).execute().body() ?: emptyList()

        // 次に表示するページの番号を計算する
        val nextPage = page + 1

        // 取得したデータ、次に表示するページの番号を結果として返す
        callback.onResult(items, nextPage)
    }
}

ItemDataSourceFactory

DataSource にデータ取得の処理を記述しましたが、DataSource.Factory を通して生成するような仕組みになっています。実際に LivePagedListBuilder で利用するのは DataSource.Factory になりますので定義してやります。

class ItemDataSourceFactory(service: QiitaService) : DataSource.Factory<Int, Item>() {
    private val source = ItemDataSource(service)

    override fun create(): DataSource<Int, Item> {
        return source
    }
}

MainViewModel

ここまで準備できればあとは PagedList を作成するだけです。作成した ItemDataSourceFactory を DI して LivePagedListBuilder に渡します。そして build してやれば PagedList が作成されます。これらのオブジェクトの作成は次のように MainViewModel を行うことにします。

class MainViewModel(private val itemDataSourcefactory: ItemDataSourceFactory): ViewModel() { 
    private val config = PagedList.Config.Builder().setInitialLoadSizeHint(10).setPageSize(10).build()
    val entities: LiveData<PagedList<Item>> = LivePagedListBuilder(factory, config).build()
}

Step 3 EpoxyRecyclerView で表示する EpoxyModel を作成する

EpoxyRecyclerView で表示する EpoxyModel を作成していきます。
次のような TextView を持つ CardView というシンプルなレイアウトを今回は定義します。

<?xml version="1.0" encoding="utf-8"?>
<layout 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">

    <data>

        <variable
            name="title"
            type="String" />

    </data>

    <androidx.cardview.widget.CardView
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:layout_margin="8dp"
        app:cardCornerRadius="4dp">

        <TextView
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:padding="8dp"
            android:text="@{title}"
            android:textSize="24sp"
            app:layout_constraintBottom_toBottomOf="parent"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toTopOf="parent"
            tools:text="TEST TEST TEST TEST TEST" />

    </androidx.cardview.widget.CardView>
</layout>

あと package-info.java というファイルをどこでもいいので作成して、EpoxyDataBindingLayouts に先程作成してレイアウトファイルの名称を記述してやります。このファイルを記述することで Epoxy が定義したレイアウトファイルから自動的に EpoxyModel を生成してくれるようになります。

@EpoxyDataBindingLayouts({R.layout.layout_epoxy_item } )
package jp.kaleidot725.sample;

import com.airbnb.epoxy.EpoxyDataBindingLayouts;

Step4 EpoxyRecyclerView で表示を制御する EpoxyController を作成する

Epoxy で Paging をしたいときには PagedListEpoxyController を継承したクラスを作成する必要があります。PagedListEpoxyController では buildItemModel にてページごとに表示する EpoxyModel を生成するようになっています。なので buildItemModel で Step3 で定義した EpoxyModel を生成して返すようにしてやります。

class ItemPagedListController : PagedListEpoxyController<Item>() {
    override fun buildItemModel(currentPosition: Int, item: Item?): EpoxyModel<*> {
        requireNotNull(item)
        return LayoutEpoxyItemBindingModel_().apply {
            id(item.id)
            title(item.title)
        }
    }
}

Step 5 作成したクラスの動作を確認する

あとは今まで作成したクラスを初期化してセットアップしてやるだけです。
次の定義で QiitaService・ItemDataSourceFactory・MainViewModel を生成できるようにしてやります。

Koin

object ViewModelFactory {
    private val retrofit get() = Retrofit.Builder()
            .baseUrl("https://qiita.com/")
            .addConverterFactory(GsonConverterFactory.create())
            .build()

    private val service = retrofit.create(QiitaService::class.java)

    val mainViewModel: MainViewModel get() = MainViewModel(ItemDataSourceFactory(service))
}

そして MainActivity にて ViewModel や EpoxyRecyclerView のセットアップをしてやれば OK です。

MainAcitivity

class MainActivity : AppCompatActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        DataBindingUtil.setContentView<ActivityMainBinding>(this, R.layout.activity_main).also { binding ->
            val controller = ItemPagedListController()
            binding.epoxyRecyclerView.adapter = controller.adapter

            val layoutManager = LinearLayoutManager(applicationContext).apply { orientation = RecyclerView.VERTICAL }
            binding.epoxyRecyclerView.layoutManager = layoutManager

            val viewModel = ViewModelFactory.mainViewModel
            binding.viewModel = viewModel
            viewModel.entities.observe(this, Observer { controller.submitList(it) })
        }
    }
}

activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<layout>
    <data>
        <variable
            name="viewModel"
            type="jp.kaleidot725.sample.ui.MainViewModel" />
    </data>

    <FrameLayout 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"
        tools:context=".ui.MainActivity">

        <com.airbnb.epoxy.EpoxyRecyclerView
            android:id="@+id/epoxy_recycler_view"
            android:layout_width="match_parent"
            android:layout_height="match_parent" />
    </FrameLayout>
</layout>

おわりに

Epoxy だと PagedListEpoxyController を継承した Controller を実装してやれば Paging に対応できる感じになっているみたいですね。Paging の実装の部分に関しては特に通常の RecyclerView を使っていた実装と変わらずそのまま使えるみたいですね。

2
2
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
2