Description:

Paging is an Android Paging library officially launched by Google, and RecyclerView can be used to implement the Footer and Header of RecyelerView. Can be used in Java or Kotlin project, with the PagingDataAdapter to help RecyclerView to achieve paging load.

The two most important classes in Paging are PagingSource, LoadState and PagingDataAdapter:

  • PagingDataAdapter is the packaging of RecyclerView.Adapter, managing the consumption of data.Copy the code
  • PaingSource handles data paging and manages the production of data.Copy the code
  • LoadState monitors the Loading state in real time and can perform UI actions based on its value: Loading,NoLoading, Refresh(first load or Refresh),Append (next page load)Copy the code

Jetpack Compose provides a convenient API similar to Flutter. Here’s a quick example:

   TextButton(
       onClick = { retry() }, 
       modifier = Modifier .padding(20.dp) .width(80.dp) .height(30.dp), 
       shape = RoundedCornerShape(6.dp), 
       contentPadding = PaddingValues(3.dp),
       colors = textButtonColors(backgroundColor = gray300), 
       elevation = elevation( 
           defaultElevation = 2.dp,
           pressedElevation = 4.dp, 
           ),
      ) {
          Text(text = "Please try again", color = gray600) 
       }
Copy the code


#### This article mainly introduces the use of Paging3 page in JetCompose project, pull down refresh and pull up load dynamic effect. The effect is as follows:

One, Paging3 page load

  1. Introducing dependencies:

      / / the Paging 3.0
      implementation 'androidx. The paging: the paging - compose: 1.0.0 - alpha14'
      implementation "Androidx. The paging: the paging - runtime - KTX: 3.1.0 - rc01"
    Copy the code
  2. Paging implements Paging load, which is simple, quick and customizable. It manages Paging logic and exception handling internally, and Paging rules need to be defined by itself. Main code:

    
    class ExamSource(private val repository: Repository) : PagingSource<Int, Question>() {
    
        private val TAG = "--ExamSource"
    
        override fun getRefreshKey(state: PagingState<Int, Question>): Int? {
            return null
        }
    
        override suspend fun load(params: LoadParams<Int>): LoadResult<Int, Question> {
    
            return try {
                valcurrentPage = params.key ? :1
                val pageSize = params.loadSize
                Log.d(TAG, "currentPage: $currentPage")
                Log.d(TAG, "pageSize: $pageSize")
    
                // Pass in the current page number, size per page, and request data. Network requests are encapsulated in repository
                valresponseList = repository.getExamList(currentPage, pageSize = pageSize) .result? .resultData? .questionList ? : emptyList<Question>()// Load paging
                val everyPageSize = 4
                val initPageSize = 8
                / / the previous page
                val preKey = if (currentPage == 1) null else currentPage.minus(1)
                / / the next page
                var nextKey: Int? = if (currentPage == 1) {
                    initPageSize / everyPageSize
                } else {
                    currentPage.plus(1)
                }
                Log.d(TAG, "preKey: $preKey")
                Log.d(TAG, "nextKey: $nextKey")
                if (responseList.isEmpty()) {
                    nextKey = null
                }
                Log.d(TAG, "final nextKey: $nextKey")
    
                LoadResult.Page(
                    data = responseList,
                    prevKey = preKey,
                    nextKey = nextKey
                )
            } catch (e: Exception) {
                e.printStackTrace()
                LoadResult.Error(e)
            }
        }
    }
    Copy the code

Immersive status bar

// The status bar is related
implementation "Com. Google. Accompanist: accompanist - insets: 0.21.2 - beta"
implementation "Com. Google. Accompanist: accompanist insets - UI: 0.21.2 - beta"
implementation "Com. Google. Accompanist: accompanist - systemuicontroller: 0.21.2 - beta"
Copy the code
    1. Set up theWindowCompat.setDecorFitsSystemWindows(window, false)The default is false
 /** * Sets whether the decorView ADAPTS to the WindowInsetsCompat root view. * If set to false, insets of content views will not be adapted, only content views will be adapted. * please note: in the application using the setSystemUiVisibility (int) API might conflict with this method. Should stop using the setSystemUiVisibility (int). * /
 public static void setDecorFitsSystemWindows(@NonNull Window window,
             final boolean decorFitsSystemWindows) {
         if (Build.VERSION.SDK_INT >= 30) {
             Impl30.setDecorFitsSystemWindows(window, decorFitsSystemWindows);
         } else if (Build.VERSION.SDK_INT >= 16) { Impl16.setDecorFitsSystemWindows(window, decorFitsSystemWindows); }}Copy the code
    1. Status bar transparency

    Get SystemUiController’s setStatusBarColor() method to change the status bar. You can also change the bottom navigation bar color.

  setContent {
       UseComposeTheme {
            // change the status bar to transparent. Parameters: color(status bar color), darkIcons (whether they are darkIcons)
             rememberSystemUiController().setStatusBarColor(
                   Color.Transparent, darkIcons = MaterialTheme.colors.isLight
             )
 
             // Bottom navigation bar color
             rememberSystemUiController().setNavigationBarColor(
                  Color.Transparent, darkIcons = MaterialTheme.colors.isLight
             )
 
             // ...}}Copy the code
    1. Adjust to fit the status bar height

    Use the Provideo WindowinSets, wrap a layer of the display content around the provideo WindowinSets, and wrap the provideo WindowinSets under the Theme node to get the status bar height. That is, ensure that the value of statusBarsHeight() is properly obtained, and set the page Spacer of equal height to leave the height of the status bar.

     setContent {
          UseComposeTheme {
                     / / to join ProvideWindowInsets
                     ProvideWindowInsets {
                         // Change the status bar to transparent
                         rememberSystemUiController().setStatusBarColor(
                             Color.Transparent, darkIcons = MaterialTheme.colors.isLight
                         )
    
                         Surface(color = MaterialTheme.colors.background) {
                             Scaffold(
                                 modifier = Modifier.fillMaxSize()
                             ) {
                                 Column {
                                       // Fill the status bar height with white space
                                      Spacer(modifier = Modifier
                                          .statusBarsHeight()
                                          .fillMaxWidth()
                                      )
    
                                      // Your business Composable)}}}}}Copy the code
    • The effect
Before the change FitsSystemWindows :false Change the color Adaptation is highly

Three, pull down refresh

Build errors: com. Google. Accompanist: accompanist: XXX related library version are not compatible, need to rely on the same version

21:35:51. 503, 7789-7789 / com. Jesen. Driverexampaging E/AndroidRuntime: FATAL EXCEPTION: the main Process: com.jesen.driverexampaging, PID: 7789 java.lang.NoSuchMethodError: No interface method startReplaceableGroup(ILjava/lang/String;) V in class Landroidx/compose/runtime/Composer; or its super classes (declaration of 'androidx.compose.runtime.Composer' appears in /data/app/~~4FT0iYbXWuoHva-X3Y0lBg==/com.jesen.driverexampaging-4uB2hZ7cDclbzM5qgmkttA==/base.apk) at com.google.accompanist.swiperefresh.SwipeRefreshKt.rememberSwipeRefreshState(Unknown Source:5) at com.jesen.driverexampaging.common.SwipeRefreshListKt.SwipeRefreshList(SwipeRefreshList.kt:31) at com.jesen.driverexampaging.ui.composeview.RefreshExamListScreenKt.RefreshExamListScreen(RefreshExamListScreen.kt:33) at com.jesen.driverexampaging.Main2Activity$onCreate$1$1$1$1$1.invoke(Main2Activity.kt:54) at com.jesen.driverexampaging.Main2Activity$onCreate$1$1$1$1$1.invoke(Main2Activity.kt:47) at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:116) at androidx.compose.runtime.internal.ComposableLambdaImpl.invoke(ComposableLambda.jvm.kt:34) at androidx.compose.material.ScaffoldKt$ScaffoldLayout$1$1$1$bodyContentPlaceables$1.invoke(Scaffold.kt:316) at androidx.compose.material.ScaffoldKt$ScaffoldLayout$1$1$1$bodyContentPlaceables$1.invoke(Scaffold.kt:314)Copy the code
  1. Basic usage:

    • Access dependency:

      // Drop refresh
          implementation "Com. Google. Accompanist: accompanist - swiperefresh: 0.21.2 - beta"
      Copy the code
    • Set the drop-down refresh to load more and determine the status

      @Composable
      fun refreshLoadUse(viewModel: ExamViewModel) {
          // Swipe's status
          val refreshState = rememberSwipeRefreshState(isRefreshing = false)
          val collectAsLazyPagingItems = viewModel.examList.collectAsLazyPagingItems()
      
          SwipeRefresh(state = refreshState, onRefresh = {
              collectAsLazyPagingItems.refresh()
          }) {
              LazyColumn(
                  modifier = Modifier
                      .fillMaxWidth()
                      .fillMaxHeight(),
                  content = {
                      itemsIndexed(collectAsLazyPagingItems) { _, refreshData ->// Display each item
                          Box(
                              modifier = Modifier
                                  .padding(horizontal = 14.dp, vertical = 4.dp)
                                  .fillMaxWidth()
                                  .height(50.dp)
                                  .background(Color.Green, shape = RoundedCornerShape(8.dp))
                                  .border(
                                      width = 1.dp,
                                      color = Color.Red,
                                      shape = RoundedCornerShape(8.dp)
                                  )
                                  .padding(start = 10.dp), contentAlignment = Alignment.CenterStart ) { Text(text = refreshData? .data? :"")}}// Append identifies non-first page, i.e. next page or more to load
                      when (collectAsLazyPagingItems.loadState.append) {
                          is LoadState.Loading -> {
                              // Display the tail item in the load
                              item {
                                  Box(
                                      modifier = Modifier
                                          .fillMaxWidth()
                                          .height(50.dp),
                                      contentAlignment = Alignment.Center
                                  ) {
                                      Text(text = "Loading...")}}}is LoadState.Error -> {
                              // more, load error display tail item
                              item {
                                  Box(
                                      modifier = Modifier
                                          .fillMaxWidth()
                                          .height(50.dp),
                                      contentAlignment = Alignment.Center
                                  ) {
                                      Text(text = "-- loading error --")}}}}})}}Copy the code
  2. Simple packaging

    Parameter 1: LazyPagingItems wrapped request results, can be stored in the ViewModel, retrieved from the ViewMode parameter 2: listContent needs to be passed in externally needs to carry the context LazyListScope, reusableCopy the code
    / * * * the drop-down load wrap * * implementation "com. Google. Accompanist: accompanist - swiperefresh: XXX" * * /
    @Composable
    fun <T : Any> SwipeRefreshList(
        collectAsLazyPagingItems: LazyPagingItems<T>,
        listContent: LazyListScope. () - >Unit.) {
    
        val rememberSwipeRefreshState = rememberSwipeRefreshState(isRefreshing = false)
    
        SwipeRefresh(
            state = rememberSwipeRefreshState,
            onRefresh = { collectAsLazyPagingItems.refresh() }
        ) {
    
            rememberSwipeRefreshState.isRefreshing =
                collectAsLazyPagingItems.loadState.refresh is LoadState.Loading
    
            LazyColumn(
                modifier = Modifier
                    .fillMaxWidth()
                    .fillMaxHeight(),
    
                ) {
                listContent()
                collectAsLazyPagingItems.apply {
                    when {
                        loadState.append is LoadState.Loading -> {
                            // Load more, bottom loading
                            item { LoadingItem() }
                        }
                        loadState.append is LoadState.Error -> {
                            // Load more exceptions
                            item {
                                ErrorMoreRetryItem() {
                                    collectAsLazyPagingItems.retry()
                                }
                            }
                        }
                        loadState.refresh is LoadState.Error -> {
                            if (collectAsLazyPagingItems.itemCount <= 0) {
                                // If itemCount is less than 0 on refresh, the first load exception is generated
                                item {
                                    ErrorContent() {
                                        collectAsLazyPagingItems.retry()
                                    }
                                }
                            } else {
                                item {
                                    ErrorMoreRetryItem() {
                                        collectAsLazyPagingItems.retry()
                                    }
                                }
                            }
                        }
                        loadState.refresh is LoadState.Loading -> {
                            // First loading and loading
                            if (collectAsLazyPagingItems.itemCount == 0) {}}}}}}}/** * bottom load more failure handling ** /
    @Composable
    fun ErrorMoreRetryItem(retry: () -> Unit) {
        Box(modifier = Modifier.fillMaxWidth(), contentAlignment = Alignment.Center) {
            TextButton(
                onClick = { retry() },
                modifier = Modifier
                    .padding(20.dp)
                    .width(80.dp)
                    .height(30.dp),
                shape = RoundedCornerShape(6.dp),
                contentPadding = PaddingValues(3.dp),
                colors = textButtonColors(backgroundColor = gray300),
                elevation = elevation(
                    defaultElevation = 2.dp,
                    pressedElevation = 4.dp,
                ),
            ) {
                Text(text = "Please try again", color = gray600)
            }
        }
    }
    
    /** * Page load failed processing ** /
    @Composable
    fun ErrorContent(retry: () -> Unit) {
        Column(
            modifier = Modifier
                .fillMaxSize()
                .padding(top = 100.dp),
            verticalArrangement = Arrangement.Center,
            horizontalAlignment = Alignment.CenterHorizontally
        ) {
            Image(
                modifier = Modifier.padding(top = 80.dp),
                painter = painterResource(id = R.drawable.ic_default_empty),
                contentDescription = null
            )
            Text(text = "Request failed. Please check network.", modifier = Modifier.padding(8.dp))
            TextButton(
                onClick = { retry() },
                modifier = Modifier
                    .padding(20.dp)
                    .width(80.dp)
                    .height(30.dp),
                shape = RoundedCornerShape(10.dp),
                contentPadding = PaddingValues(5.dp),
                colors = textButtonColors(backgroundColor = gray300),
                elevation = elevation(
                    defaultElevation = 2.dp,
                    pressedElevation = 4.dp,
                )
                //colors = ButtonDefaults
            ) { Text(text = "Try again", color = gray700) }
        }
    }
    
    /** * bottom loading more loading... * * /
    @Composable
    fun LoadingItem(a) {
        Row(
            modifier = Modifier
                .height(34.dp)
                .fillMaxWidth()
                .padding(5.dp),
            horizontalArrangement = Arrangement.Center
        ) {
            CircularProgressIndicator(
                modifier = Modifier
                    .size(24.dp),
                color = gray600,
                strokeWidth = 2.dp
            )
            Text(
                text = "Loading...",
                color = gray600,
                modifier = Modifier
                    .fillMaxHeight()
                    .padding(start = 20.dp),
                fontSize = 18.sp,
            )
        }
    }
    Copy the code
    • Usage:

      1. List layout:

        /** * Home page list loading -- drop down refresh, load more dynamic effects ** /
        @Composable
        fun RefreshExamListScreen(
            viewModel: ExamViewModel,
            context: Context.) {
        
            val collectAsLazyPagingIDataList = viewModel.examList.collectAsLazyPagingItems()
        
            SwipeRefreshList(
                collectAsLazyPagingItems = collectAsLazyPagingIDataList
            ) {
        
                itemsIndexed(collectAsLazyPagingIDataList) { index, data ->
                    // list Item, corresponding entity data is data
                    QItemView(
                        index = index,
                        que = data,
                        onClick = { Toast.makeText(context, "ccc", Toast.LENGTH_SHORT).show() },
                    )
                }
            }
        }
        Copy the code
      2. ViewModel, including Paging3 configuration:

      class ExamViewModel : ViewModel() {
      
          val examList = Pager(
              config = PagingConfig(
                  pageSize = 4.// Number of pages per page
                  initialLoadSize = 8.// First load quantity, not required
                  prefetchDistance = 2.// The required distance from the next page)) {// This class handles paging, as mentioned earlier
              ExamSource(Repository)
          }.flow.cachedIn(viewModelScope)
      }
      Copy the code

Code examples: driving exam list https://github.com/Jesen0823/UseCompose/tree/main/DriverExamPaging