Search code examples
androidandroid-recyclerviewkotlinrecyclerview-layout

Recyclerview missing item when element's width and height is enlarged


I've been scratching my head about this problem. I have a recyclerview inside a recyclerview that shows a grid layout where you can scroll both vertically and horizontally.

I decided to enlarge the width and height of each item. Previously, width and height were: 80 x 100

enter image description here

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/book_container"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content">

    <ImageView
        android:id="@+id/inner_dropshadow"
        android:layout_width="80dp"
        android:layout_height="100dp"
        android:scaleType="fitXY"
        app:layout_constraintBottom_toTopOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:srcCompat="@drawable/bookdropshadow" />

</android.support.constraint.ConstraintLayout>

And then I enlarged it to 150x 180. And then Level 4-6 row completely disappeared no matter how much I scroll up. I only expected the recyclerview to display larger images. Why does this happen?

[EDIT] I ran a debug and it seems only 3 positions are bound in the outer adapter when the app is run on a phone but there are 4 positions when run on a tablet.

enter image description here

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/book_container"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content">

    <ImageView
        android:id="@+id/inner_dropshadow"
        android:layout_width="150dp"
        android:layout_height="180dp"
        android:scaleType="fitXY"
        app:layout_constraintBottom_toTopOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent"
        app:srcCompat="@drawable/bookdropshadow" />

</android.support.constraint.ConstraintLayout>

Please note that this also happens when I run the app on a device. In a tablet, it shows all the Levels, while on a phone, it only shows Levels 1-3.

Outer recyclerview layout

<?xml version="1.0" encoding="utf-8"?>

<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:id="@+id/mainlayout"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="@color/color_white">

    <android.support.constraint.ConstraintLayout
        android:layout_width="match_parent"
        android:layout_height="match_parent">

        <android.support.v4.widget.SwipeRefreshLayout
            android:layout_width="match_parent"
            android:layout_height="0dp"
            android:id="@+id/swiperefresh"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintTop_toTopOf="parent"
            app:layout_constraintBottom_toTopOf="@+id/toolbar">

            <ScrollView
                android:id="@+id/scrollView"
                android:layout_width="match_parent"
                android:layout_height="match_parent"
                android:overScrollMode="always">


                <android.support.constraint.ConstraintLayout
                    android:layout_width="match_parent"
                    android:layout_height="match_parent">

                    <include layout="@layout/browsebooksheader_layout"
                        android:id="@+id/bb_header"
                        app:layout_constraintBottom_toTopOf="@+id/guidelinetop"
                        app:layout_constraintEnd_toEndOf="parent"
                        app:layout_constraintStart_toStartOf="parent"
                        app:layout_constraintTop_toBottomOf="parent"/>

                    <android.support.constraint.Guideline
                        android:id="@+id/guidelinetop"
                        android:layout_width="wrap_content"
                        android:layout_height="wrap_content"
                        android:orientation="horizontal"
                        app:layout_constraintGuide_percent="0.30"/>

                    <Button
                        android:id="@+id/btn_search"
                        android:layout_width="match_parent"
                        android:layout_height="wrap_content"
                        android:gravity="center_horizontal"
                        android:background="@drawable/search_btn_border"
                        android:padding="8dp"
                        android:text="Search"
                        android:textSize="@dimen/browse_search_textsize"
                        android:textColor="@android:color/darker_gray"
                        app:layout_constraintBottom_toTopOf="@+id/recyclerview_main"
                        app:layout_constraintEnd_toEndOf="parent"
                        app:layout_constraintStart_toStartOf="parent"
                        app:layout_constraintTop_toBottomOf="@+id/bb_header" />

                    <ProgressBar
                        android:id="@+id/main_progressbar"
                        style="?android:attr/progressBarStyleSmall"
                        android:layout_marginTop="16dp"
                        android:layout_width="50dp"
                        android:layout_height="50dp"
                        android:layout_gravity="center"
                        android:layout_centerHorizontal="true"
                        android:layout_centerVertical="true"
                        android:indeterminateTint="@color/color_grey"
                        app:layout_constraintBottom_toBottomOf="parent"
                        app:layout_constraintEnd_toEndOf="parent"
                        app:layout_constraintStart_toStartOf="parent"
                        app:layout_constraintTop_toBottomOf="@+id/btn_search"/>


                    <android.support.v7.widget.RecyclerView
                        android:id="@+id/recyclerview_main"
                        android:layout_width="match_parent"
                        android:layout_height="match_parent"
                        android:nestedScrollingEnabled="false"
                        android:background="@color/color_white"
                        app:layout_constraintEnd_toEndOf="parent"
                        app:layout_constraintStart_toStartOf="parent"
                        app:layout_constraintTop_toBottomOf="@+id/btn_search"
                        app:layout_constraintBottom_toBottomOf="parent"/>

                </android.support.constraint.ConstraintLayout>
            </ScrollView>



        </android.support.v4.widget.SwipeRefreshLayout>




        <android.support.constraint.ConstraintLayout
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:id="@+id/toolbar"
            app:layout_constraintEnd_toEndOf="parent"
            app:layout_constraintTop_toBottomOf="@id/swiperefresh"
            app:layout_constraintStart_toStartOf="parent"
            app:layout_constraintBottom_toBottomOf="parent">

            <include layout="@layout/toolbar_layout_wo_learn"/>

        </android.support.constraint.ConstraintLayout>

    </android.support.constraint.ConstraintLayout>
</android.support.constraint.ConstraintLayout>

Inner recyclerview layout

<?xml version="1.0" encoding="utf-8"?>
<android.support.constraint.ConstraintLayout
    xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:app="http://schemas.android.com/apk/res-auto"
    android:layout_width="match_parent"
    android:layout_height="wrap_content">

    <TextView
        android:id="@+id/book_category"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_marginStart="8dp"
        android:layout_marginTop="8dp"
        android:text="Category Title"
        android:textColor="@color/defaultdark_color"
        android:textSize="@dimen/browse_category_textsize"
        android:fontFamily="@font/vagroundedstd_bold"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <ImageView
        android:layout_width="@dimen/browse_seeall_imagesize"
        android:layout_height="@dimen/browse_seeall_imagesize"
        android:layout_marginBottom="2dp"
        android:layout_marginTop="2dp"
        android:layout_marginEnd="8dp"
        android:id="@+id/see_all_btn"
        android:src="@drawable/seeall"
        app:layout_constraintBottom_toTopOf="@+id/inner_recyclerview"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintTop_toTopOf="parent" />

    <android.support.v7.widget.RecyclerView
        android:id="@+id/inner_recyclerview"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"
        android:nestedScrollingEnabled="false"
        android:background="@color/color_white"
        app:layout_constraintStart_toStartOf="parent"
        app:layout_constraintEnd_toEndOf="parent"
        app:layout_constraintTop_toBottomOf="@+id/book_category"/>

    <TextView
        android:id="@+id/h_line"
        android:layout_width="match_parent"
        android:layout_height="0.8dp"
        android:background="@color/color_grey"
        android:gravity="center"
        android:layout_marginTop="8dp"
        android:layout_marginBottom="8dp"
        android:alpha="0.5"
        app:layout_constraintTop_toBottomOf="@id/inner_recyclerview"/>


</android.support.constraint.ConstraintLayout>

Outer Adapter

class BrowseBooksAdapter : RecyclerView.Adapter<BrowseBooksAdapter.SimpleViewHolder>() {

    private var sortedList : SortedMap<String, List<ResourcesList>> = emptyMap<String, List<ResourcesList>>().toSortedMap()

    fun setData(sortedList : SortedMap<String, List<ResourcesList>>) {
        if (this.sortedList != sortedList) {
            this.sortedList = sortedList
            notifyDataSetChanged()
        }
    }

    override fun getItemCount(): Int {
        return sortedList.count()
    }

    override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): SimpleViewHolder {
        val view = LayoutInflater.from(parent.context).inflate(R.layout.browsebooks_layout, parent, false)
        return SimpleViewHolder(view)
    }

    override fun onBindViewHolder(holder: SimpleViewHolder, position: Int) {

        val dataList = ArrayList(sortedList.values)[position]
        holder.category_text.text = ArrayList(sortedList.keys)[position]
        holder.booksByCategory = dataList
        holder.horizontalAdapter.setData(dataList)
    }

    class SimpleViewHolder(v: View, var booksByCategory : List<ResourcesList>? = null) : RecyclerView.ViewHolder(v) {
        var recyclerView : RecyclerView = v.findViewById(R.id.inner_recyclerview)
        var category_text : TextView = v.findViewById(R.id.book_category)
        var fragmentCallback: FragmentCallback? = null
        val horizontalAdapter: BooksByCategoryAdapter


        init {
            recyclerView.layoutManager = object : LinearLayoutManager(v.context, LinearLayoutManager.HORIZONTAL, false) {
                override fun canScrollHorizontally(): Boolean {
                    return true
                }

                override fun canScrollVertically(): Boolean {
                    return false
                }
            }
            horizontalAdapter = BooksByCategoryAdapter()
            recyclerView.adapter = horizontalAdapter
            fragmentCallback = v.context as FragmentCallback
            var mLastClickTime : Long = 0


            v.see_all_btn.setOnClickListener {
                val category = v.book_category.text.toString()

                // double-clicking prevention, using threshold of 1000 ms
                if (!(SystemClock.elapsedRealtime() - mLastClickTime < 1000)){
                    mLastClickTime = SystemClock.elapsedRealtime()

                    if(fragmentCallback != null){
                        fragmentCallback!!.callBackSeeAllFragment(booksByCategory!!, category)
                    }
                }

            }
        }

    }

}

Inner Adapter

class BooksByCategoryAdapter : RecyclerView.Adapter<RecyclerView.ViewHolder>() {

private var booksByCategory: List<ResourcesList> = emptyList()

fun setData(data: List<ResourcesList>) {
    if (this.booksByCategory != data) {
        this.booksByCategory = data
        notifyDataSetChanged()
    }
}

override fun getItemCount(): Int {
    return booksByCategory.size
}


override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
    val layoutInflater = LayoutInflater.from(parent.context)
    val cellForRow = layoutInflater.inflate(R.layout.bookcontainer_layout, parent, false)
    return CustomVH(cellForRow)
}

override fun onBindViewHolder(rawHolder: RecyclerView.ViewHolder, position: Int) {
    val holder = rawHolder as CustomVH
    val booksProcessModel = BooksProcessModel()
    booksProcessModel.setItem(booksByCategory[position])
    holder.booksByCategory = booksByCategory[position]
    Picasso.get().load(booksByCategory[position].coverURI).into(holder.inner_bookcover)


}

private inner class CustomVH(v: View, var booksByCategory : ResourcesList? = null): RecyclerView.ViewHolder(v) {

    var inner_bookcover : ImageView = v.findViewById(R.id.inner_bookcover)
    var fragmentCallback: FragmentCallback? = null
    var ctx : Context = v.context
    var mLastClickTime : Long = 0

    init {
        fragmentCallback = ctx as FragmentCallback

        v.setOnClickListener {

            // double-clicking prevention, using threshold of 1000 ms
            if (!(SystemClock.elapsedRealtime() - mLastClickTime < 1000)){
                mLastClickTime = SystemClock.elapsedRealtime()

                if(fragmentCallback != null){
                    fragmentCallback!!.callBackDownloadBookFragment(booksByCategory!!)
                }
            }

        }

    }
}

class BooksProcessModel {

    private var processBooks : ResourcesList? = null

    fun setItem(books: ResourcesList) {
        this.processBooks = books
    }

    fun setVisibility(ctx: Context) : Int {

        val sp = PreferenceManager.getDefaultSharedPreferences(ctx)
        val page = sp.getString(processBooks!!.id.toString(), "")
        if (page != "") {
            return View.VISIBLE
        }
        return View.INVISIBLE
    }

}

}


Solution

  • Nevermind, figured it out. I should be using a NestedScrollView instead of a ScrollView. The missing items were still there. It's just that I couldn't scroll the nested recyclerview.