Created
September 20, 2018 20:31
-
-
Save Sirelon/c2928e39c0a7c8c4fd4738b45acd3069 to your computer and use it in GitHub Desktop.
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.sirelon | |
import android.content.Context | |
import android.graphics.Canvas | |
import android.graphics.Rect | |
import android.graphics.drawable.Drawable | |
import android.util.Log | |
import android.view.View | |
import android.widget.LinearLayout | |
import androidx.recyclerview.widget.RecyclerView | |
class ControlFlowItemDivider( | |
context: Context, | |
orientation: Int, | |
/** | |
* Adapter position COULD be NEGATIVE!!! | |
* return true if item of such position should be decorated, false - otherwise | |
*/ | |
private val shouldDraw: ((adapterPosition: Int) -> Boolean)? | |
) : RecyclerView.ItemDecoration() { | |
val HORIZONTAL = LinearLayout.HORIZONTAL | |
val VERTICAL = LinearLayout.VERTICAL | |
private val TAG = "DividerItem" | |
private val ATTRS = intArrayOf(android.R.attr.listDivider) | |
private var mDivider: Drawable? = null | |
/** | |
* Current orientation. Either [.HORIZONTAL] or [.VERTICAL]. | |
*/ | |
private var mOrientation: Int = 0 | |
private val mBounds = Rect() | |
/** | |
* Creates a divider [RecyclerView.ItemDecoration] that can be used with a | |
* [LinearLayoutManager]. | |
* | |
* @param context Current context, it will be used to access resources. | |
* @param orientation Divider orientation. Should be [.HORIZONTAL] or [.VERTICAL]. | |
*/ | |
init { | |
val a = context.obtainStyledAttributes(ATTRS) | |
mDivider = a.getDrawable(0) | |
if (mDivider == null) { | |
Log.w( | |
TAG, | |
"@android:attr/listDivider was not set in the theme used for this " + "DividerItemDecoration. Please set that attribute all call setDrawable()" | |
) | |
} | |
a.recycle() | |
setOrientation(orientation) | |
} | |
/** | |
* Sets the orientation for this divider. This should be called if | |
* [RecyclerView.LayoutManager] changes orientation. | |
* | |
* @param orientation [.HORIZONTAL] or [.VERTICAL] | |
*/ | |
fun setOrientation(orientation: Int) { | |
if (orientation != HORIZONTAL && orientation != VERTICAL) { | |
throw IllegalArgumentException( | |
"Invalid orientation. It should be either HORIZONTAL or VERTICAL" | |
) | |
} | |
mOrientation = orientation | |
} | |
/** | |
* Sets the [Drawable] for this divider. | |
* | |
* @param drawable Drawable that should be used as a divider. | |
*/ | |
fun setDrawable(drawable: Drawable) { | |
mDivider = drawable | |
} | |
override fun onDraw(c: Canvas, parent: RecyclerView, state: RecyclerView.State) { | |
if (parent.layoutManager == null || mDivider == null) { | |
return | |
} | |
if (mOrientation == VERTICAL) { | |
drawVertical(c, parent) | |
} else { | |
drawHorizontal(c, parent) | |
} | |
} | |
private fun drawVertical(canvas: Canvas, parent: RecyclerView) { | |
canvas.save() | |
val left: Int | |
val right: Int | |
if (parent.clipToPadding) { | |
left = parent.paddingLeft | |
right = parent.width - parent.paddingRight | |
canvas.clipRect( | |
left, parent.paddingTop, right, | |
parent.height - parent.paddingBottom | |
) | |
} else { | |
left = 0 | |
right = parent.width | |
} | |
val childCount = parent.childCount | |
for (i in 0 until childCount) { | |
val child = parent.getChildAt(i) | |
val adapterPosition = parent.layoutManager?.getPosition(child) ?: -1 | |
if (shouldDraw?.invoke(adapterPosition) == false) { | |
continue | |
} | |
parent.getDecoratedBoundsWithMargins(child, mBounds) | |
val bottom = mBounds.bottom + Math.round(child.translationY) | |
val top = bottom - mDivider!!.intrinsicHeight | |
mDivider!!.setBounds(left, top, right, bottom) | |
mDivider!!.draw(canvas) | |
} | |
canvas.restore() | |
} | |
private fun drawHorizontal(canvas: Canvas, parent: RecyclerView) { | |
canvas.save() | |
val top: Int | |
val bottom: Int | |
if (parent.clipToPadding) { | |
top = parent.paddingTop | |
bottom = parent.height - parent.paddingBottom | |
canvas.clipRect( | |
parent.paddingLeft, top, | |
parent.width - parent.paddingRight, bottom | |
) | |
} else { | |
top = 0 | |
bottom = parent.height | |
} | |
val childCount = parent.childCount | |
for (i in 0 until childCount) { | |
val child = parent.getChildAt(i) | |
val adapterPosition = parent.layoutManager?.getPosition(child) ?: -1 | |
if (shouldDraw?.invoke(adapterPosition) == false) { | |
continue | |
} | |
parent.layoutManager!!.getDecoratedBoundsWithMargins(child, mBounds) | |
val right = mBounds.right + Math.round(child.translationX) | |
val left = right - mDivider!!.intrinsicWidth | |
mDivider!!.setBounds(left, top, right, bottom) | |
mDivider!!.draw(canvas) | |
} | |
canvas.restore() | |
} | |
override fun getItemOffsets( | |
outRect: Rect, view: View, parent: RecyclerView, | |
state: RecyclerView.State | |
) { | |
if (mDivider == null) { | |
outRect.set(0, 0, 0, 0) | |
return | |
} | |
if (mOrientation == VERTICAL) { | |
outRect.set(0, 0, 0, mDivider!!.intrinsicHeight) | |
} else { | |
outRect.set(0, 0, mDivider!!.intrinsicWidth, 0) | |
} | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment