一、前言

很多时候会遇到使用 RecyclerView 时,要求超过一定数目的 Item 后,固定 RecyclerView 的高度,没有超过这个数目就自适应高度。这种情况更多会出现在对话框中,数量过多时不能让对话框占据整个屏幕,同时又能控制显示的 Item 个数,下面针对不同的情况可以使用不同的方法。

二、已知 Item 高度的情况下

在已知 Item 布局的高度的情况时,可以通过设置最大高度来控制显示的 Item 数目,因为 Item 高度已知,所以计算好高度就可以达到目的,下面是具体的实现:
看下效果,最大展示 3 个 Item,超过滚动展示:
iShot2020-08-0121.56.41.gif

    1. 布局界面,比较简单,就一个 RecyclerView,只要设置一个最大高度就可以了(注意:最大高度要使用layout_constraintHeight_max) ```kotlin <?xml version=”1.0” encoding=”utf-8”?>
  1. <androidx.recyclerview.widget.RecyclerView
  2. android:id="@+id/rv_recycler_fixed_one"
  3. android:layout_width="match_parent"
  4. android:layout_height="0dp"
  5. app:layout_constraintTop_toTopOf="parent"
  6. app:layout_constraintHeight_max="240dp"
  7. android:background="@color/yellow_FF9B52"
  8. />

  1. - 2. Activity 界面,简单的设置和数据设置,任意正常可用的 adapter 就可以
  2. ```java
  3. class RecyclerItemFixedActivity: AppCompatActivity() {
  4. override fun onCreate(savedInstanceState: Bundle?) {
  5. super.onCreate(savedInstanceState)
  6. setContentView(R.layout.activity_recycler_item_fixed)
  7. val list = arrayListOf<String>("测试数据一","测试数据二","测试数据三","测试数据四")
  8. val adapter by lazy { RecyclerItemFixedAdapter() }
  9. rv_recycler_fixed_one?.let {
  10. it.layoutManager = LinearLayoutManager(this)
  11. it.adapter = adapter
  12. it.addItemDecoration(DividerItemDecoration(this,LinearLayoutManager.VERTICAL))
  13. }
  14. adapter.setList(list)
  15. }
  16. }
    1. 非常普通的 Item 布局,主要是高度固定

      1. <?xml version="1.0" encoding="utf-8"?>
      2. <RelativeLayout
      3. xmlns:android="http://schemas.android.com/apk/res/android"
      4. android:layout_width="match_parent"
      5. android:layout_height="80dp"
      6. android:background="@color/green_07C0C2">
      7. <TextView
      8. android:id="@+id/tv_item_recycler_fixed_title"
      9. android:layout_width="wrap_content"
      10. android:layout_height="wrap_content"
      11. android:text="@string/app_name"
      12. android:textColor="@color/white"
      13. android:layout_centerInParent="true"/>
      14. </RelativeLayout>

      三、Item 高度无法确定时

      对于这种情况,我查了很多资料,只有动态设置 RecyclerView 方法比较有效,在设置好 Adapter 后,获取单个 Item 的高度,然后就可以用代码设置 RecyclerView 的高度。如果有更好的方法实现的话,欢迎在评论中提出。效果图和上图基本一样。实现如下:

  • 1.布局文件 ```java <?xml version=”1.0” encoding=”utf-8”?>

  1. <androidx.recyclerview.widget.RecyclerView
  2. android:id="@+id/rv_recycler_wrap_list"
  3. android:layout_width="match_parent"
  4. android:layout_height="wrap_content"
  5. app:layout_constraintTop_toTopOf="parent"
  6. />

  1. - 2. 简单的 Item文件,和上面的 Item 不同的地方就是高度不固定
  2. ```java
  3. <?xml version="1.0" encoding="utf-8"?>
  4. <RelativeLayout
  5. xmlns:android="http://schemas.android.com/apk/res/android"
  6. android:layout_width="match_parent"
  7. android:layout_height="wrap_content"
  8. android:background="@color/green_07C0C2">
  9. <TextView
  10. android:id="@+id/tv_item_recycler_wrap2_title"
  11. android:layout_width="wrap_content"
  12. android:layout_height="wrap_content"
  13. android:text="@string/app_name"
  14. android:textColor="@color/white"
  15. android:layout_centerInParent="true"
  16. android:paddingVertical="@dimen/m10"/>
  17. </RelativeLayout>
  • 3.简单的 adapter,仅仅设置一下展示数据

    1. class RecyclerWrapAdapter(resId: Int = R.layout.item_recycelr_wrap2) :
    2. BaseQuickAdapter<String, BaseViewHolder>(resId) {
    3. override fun convert(holder: BaseViewHolder, item: String) {
    4. holder.setText(R.id.tv_item_recycler_wrap2_title, item)
    5. }
    6. }
    1. activity 设置数据,最重要的是设置 RecyclerView 的高度,代码中有具体的注释

      1. class RecyclerWrapActivity: AppCompatActivity() {
      2. override fun onCreate(savedInstanceState: Bundle?) {
      3. super.onCreate(savedInstanceState)
      4. setContentView(R.layout.activity_recyclerview_wrap)
      5. val list = arrayListOf("测试数据一","测试数据二","测试数据三","测试数据四","测试数据五","测试数据六")
      6. val adapter by lazy { RecyclerWrapAdapter() }
      7. rv_recycler_wrap_list?.let {
      8. it.layoutManager = LinearLayoutManager(this)
      9. it.adapter = adapter
      10. it.addItemDecoration(DividerItemDecoration(this, LinearLayoutManager.VERTICAL))
      11. }
      12. adapter.setList(list)
      13. //参考https://www.cnblogs.com/guanxinjing/p/13037156.html
      14. //动态设置recyclerview高度
      15. rv_recycler_wrap_list?.post {
      16. //假如固定4个,超过4个才需要设置
      17. if (list.size <= 4){
      18. return@post
      19. }
      20. val itemView = rv_recycler_wrap_list.getChildAt(0)
      21. itemView?.let {
      22. val height = it.height * 4
      23. val layoutParems = rv_recycler_wrap_list.layoutParams
      24. layoutParems.height = height
      25. rv_recycler_wrap_list.layoutParams = layoutParems
      26. }
      27. }
      28. }
      29. }

      四、横向滚动时

      横向滚动时,一般是要求展示固定数目的 Item,超过数目则滚动展示,这个时候就要去动态设置 Item 布局的宽度来实现。如下图,展示 4.5个 Item,超过滚动:
      image.png
      image.png

  • 1.布局界面,和上面都一样 ```java <?xml version=”1.0” encoding=”utf-8”?>

  1. <androidx.recyclerview.widget.RecyclerView
  2. android:id="@+id/rv_recycler_wrap_list"
  3. android:layout_width="match_parent"
  4. android:layout_height="wrap_content"
  5. app:layout_constraintTop_toTopOf="parent"
  6. />

  1. - 2. activity,把方向改成横向滚动就可以了
  2. ```java
  3. class RecyclerViewWrapActivity: AppCompatActivity() {
  4. override fun onCreate(savedInstanceState: Bundle?) {
  5. super.onCreate(savedInstanceState)
  6. setContentView(R.layout.activity_recyclerview_wrap)
  7. val list = arrayListOf("测试数据一","测试数据二","测试数据三","测试数据四","测试数据五","测试数据六")
  8. val adapter by lazy { RecyclerItemWrapAdapter() }
  9. rv_recycler_wrap_list?.let {
  10. it.layoutManager = LinearLayoutManager(this,LinearLayoutManager.HORIZONTAL,false)
  11. it.adapter = adapter
  12. it.addItemDecoration(DividerItemDecoration(this, LinearLayoutManager.HORIZONTAL))
  13. }
  14. adapter.setList(list)
  15. }
  16. }

}

  1. - 4. Item 布局,高度可以固定,可以不固定,宽度 wrap_content
  2. ```java
  3. <?xml version="1.0" encoding="utf-8"?>
  4. <RelativeLayout
  5. xmlns:android="http://schemas.android.com/apk/res/android"
  6. android:layout_width="wrap_content"
  7. android:layout_height="wrap_content"
  8. android:background="@color/green_07C0C2">
  9. <TextView
  10. android:id="@+id/tv_item_recycler_wrap_title"
  11. android:layout_width="wrap_content"
  12. android:layout_height="wrap_content"
  13. android:text="@string/app_name"
  14. android:textColor="@color/white"
  15. android:layout_centerInParent="true"
  16. android:paddingVertical="@dimen/m10"
  17. />
  18. </RelativeLayout>

五、地址和参考:

Github
Android开发 在有指定数量item后固定RecyclerView高度
Android RecyclerView的item宽度保持四个半