Android-流式标签(流式布局)

场景:

最近不是很忙, 总结一下项目中用到的流式标签

实现方案:

采取自定义ViewGroup的方式,实现view的按需摆放

实现效果图:

image

使用方式:

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 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="match_parent"
    android:orientation="vertical">

    <com.common.ui.UIFlowLayout
        android:id="@+id/uiFlowLayout"
        android:layout_width="match_parent"
        android:layout_height="wrap_content"/>
</LinearLayout>

//设置适配器
 actBinding.uiFlowLayout.setUIFlowAdapter(mFlowAdapter = new FlowAdapter());
//自定义适配器
class FlowAdapter extends UIFlowLayout.UIFlowAdapter {
        public int count = 30;

        public void setCount(int count) {
            this.count = count;
        }

        @Override
        public int getSize() {
            return count;
        }

        @NotNull
        @Override
        public View getView(@NotNull Context context, int position) {
            TextView view = new TextView(context);
            view.setText("测试 = " + position);
            view.setPadding(20, 20, 20, 20);
            view.setBackgroundResource(R.color.color_home_red);
            return view;
        }

        @Override
        public void onLabelViewClick(@NotNull View view, int position) {
            Toast.makeText(MianShiAct.this, "position = " + position, Toast.LENGTH_SHORT).show();
        }
    }

代码实现:

//api 说明
/**
     * setMaxRow 设置最大显示的行数
     * setRowGap  设置行间距
     * setColumnGap 设置列间距
     * setUIFlowAdapter  设置适配器
     * notifyDataSetChanged  刷新数据
     */

package com.common.ui

import android.content.Context
import android.util.AttributeSet
import android.view.View
import android.view.ViewGroup
import java.util.*

/**
 * <pre>
 * author : amos
 * time   : 2020/08/01 19:10
 * desc   : 流式布局
 * version: 1.0
</pre> *
 */
class UIFlowLayout @JvmOverloads constructor(
    context: Context,
    attrs: AttributeSet? = null,
    defStyleAttr: Int = 0,
    defStyleRes: Int = 0
) : ViewGroup(context, attrs, defStyleAttr, defStyleRes), IUIFlowLayoutContract {
    private var mAdapter: UIFlowAdapter? = null //标签适配器
    private val mRowViewList: MutableList<MutableList<View>> = mutableListOf() //记录每一行的view
    private val mRowHeightList: MutableList<Int> = ArrayList() //记录每一行的行高
    private var rowGap = dp2px(10f) //每行之间的间距
    private var columnGap = dp2px(10f) //每列之间的间距
    private var maxRow = 0 //能够显示的最大行数  ( maxRow <= 0 不做限制)

    fun setMaxRow(maxRow: Int): UIFlowLayout = apply {
        this.maxRow = maxRow
    }

    fun setRowGap(rowGap: Int): UIFlowLayout = apply {
        this.rowGap = if (rowGap < 0) 0 else rowGap
    }

    fun setColumnGap(columnGap: Int): UIFlowLayout = apply {
        this.columnGap = if (columnGap < 0) 0 else columnGap
    }

    /**
     * 设置标签适配器
     */
    fun setUIFlowAdapter(adapter: UIFlowAdapter) {
        mAdapter = adapter
        mAdapter!!.registerListener(this)
        parsing()
    }

    init {
        init(context, attrs)
    }

    private fun init(context: Context, attrs: AttributeSet?) {
        if (attrs != null) {
            val attributes = context.obtainStyledAttributes(attrs, R.styleable.UIFlowLayout)
            if (attributes.hasValue(R.styleable.UIFlowLayout_ui_flow_row_gap)) {
                rowGap = attributes.getDimensionPixelOffset(
                    R.styleable.UIFlowLayout_ui_flow_row_gap,
                    rowGap
                )
            }
            if (attributes.hasValue(R.styleable.UIFlowLayout_ui_flow_column_gap)) {
                columnGap = attributes.getDimensionPixelOffset(
                    R.styleable.UIFlowLayout_ui_flow_column_gap,
                    columnGap
                )
            }
            if (attributes.hasValue(R.styleable.UIFlowLayout_ui_flow_max_row_count)) {
                maxRow = attributes.getInt(R.styleable.UIFlowLayout_ui_flow_max_row_count, maxRow)
            }
            attributes.recycle()
        }
    }

    override fun onMeasure(widthMeasureSpec: Int, heightMeasureSpec: Int) {
        if (childCount == 0) {
            super.onMeasure(widthMeasureSpec, heightMeasureSpec)
            return
        }
        //父容器能够给到的最大宽
        val widthSize = MeasureSpec.getSize(widthMeasureSpec)
        //父容器的测量模式
        val widthMode = MeasureSpec.getMode(widthMeasureSpec)
        //父容器能够给到的最大高
        val heightSize = MeasureSpec.getSize(heightMeasureSpec)
        //父容器的测量模式
        val heightMode = MeasureSpec.getMode(heightMeasureSpec)

        //每一行的最大宽度
        var rowMaxWidth = 0
        //每一行的最大高度
        var rowMaxHeight = 0
        //总的最大高度
        var totalRowHeight = 0
        //总的最大宽度
        var totalRowWidth = 0
        //每个子view 的 宽和高
        var childWidth = 0

        var childHeight = 0
        mRowViewList.clear()
        mRowHeightList.clear()
        var rowChildView: MutableList<View> = ArrayList()
        var columnGapIndex = 0 //
        var i = 0
        val count = childCount
        while (i < count) {
            val childView = getChildAt(i)
            val params = childView.layoutParams as MarginLayoutParams
            measureChildWithMargins(childView, widthMeasureSpec, 0, heightMeasureSpec, 0)
            childWidth = params.leftMargin + params.rightMargin + childView.measuredWidth
            childHeight = params.topMargin + params.bottomMargin + childView.measuredHeight
            if (rowMaxWidth + childWidth > widthSize - paddingLeft - paddingRight - columnGapIndex * columnGap) { //超过一行
                totalRowWidth = Math.max(rowMaxWidth, totalRowWidth)
                //totalRowHeight += rowMaxHeight;
                mRowViewList.add(rowChildView)
                mRowHeightList.add(rowMaxHeight)
                rowChildView = ArrayList()
                rowMaxWidth = childWidth
                rowMaxHeight = childHeight
                columnGapIndex = 0
            } else { //未超过一行
                rowMaxWidth += childWidth + columnGapIndex * columnGap
                rowMaxHeight = Math.max(rowMaxHeight, childHeight)
                columnGapIndex++
            }
            rowChildView.add(childView)
            if (i == count - 1) { //最后一个
                mRowViewList.add(rowChildView)
                mRowHeightList.add(rowMaxHeight)
                totalRowWidth = Math.max(totalRowWidth, rowMaxWidth)
            }
            i++
        }
        val rowMax: Int
        rowMax = if (maxRow <= 0) { //不限制行数
            mRowHeightList.size
        } else { //限制行数
            Math.min(maxRow, mRowHeightList.size)
        }
        for (rowIndex in 0 until rowMax) {
            totalRowHeight += mRowHeightList[rowIndex]
        }
        if (rowMax > 0) {
            totalRowHeight += (rowMax - 1) * rowGap
        }
        //针对受限制的view 做一个处理
        if (rowMax < mRowViewList.size) {
            for (index in rowMax until mRowViewList.size) {
                val tempList = mRowViewList[index]
                tempList.forEach {
                    //it.visibility = GONE
                    removeView(it)
                }
            }
        }
        val resultWidth = if (widthMode == MeasureSpec.EXACTLY) widthSize else totalRowWidth
        val resultHeight = if (heightMode == MeasureSpec.EXACTLY) heightSize else totalRowHeight
        setMeasuredDimension(resultWidth, resultHeight)
    }

    override fun onLayout(changed: Boolean, l: Int, t: Int, r: Int, b: Int) {
        if (mRowViewList == null || mRowViewList.size == 0) {
            return
        }
        var left: Int
        var top: Int
        var right: Int
        var bottom: Int
        var contentLeft = paddingLeft
        var contentTop = paddingTop
        val rowCount = if (maxRow <= 0) mRowViewList.size else maxRow
        var i = 0
        val count = mRowViewList.size
        while (i < count && i < rowCount) {
            val childView = mRowViewList[i]
            for (view in childView) {
                val params = view.layoutParams as MarginLayoutParams
                left = contentLeft + params.leftMargin
                top = contentTop + params.topMargin
                right = left + view.measuredWidth
                bottom = top + view.measuredHeight
                view.layout(left, top, right, bottom)
                contentLeft = right + params.rightMargin + columnGap
            }
            contentLeft = paddingLeft
            contentTop += mRowHeightList[i] + rowGap
            i++
        }
    }

    override fun generateLayoutParams(attrs: AttributeSet): LayoutParams {
        return MarginLayoutParams(context, attrs)
    }

    override fun generateDefaultLayoutParams(): LayoutParams {
        return MarginLayoutParams(MarginLayoutParams.WRAP_CONTENT, MarginLayoutParams.WRAP_CONTENT)
    }

    override fun notifyDataSetChanged() {
        parsing()
        this.requestLayout()
    }

    /**
     * 解析数据
     */
    private fun parsing() {
        this.removeAllViews()
        mAdapter?.let {
            if (it.getSize() > 0) {
                for (index in 0 until it.getSize()) {
                    val labelView = it.getView(context, index)
                    labelView.setOnClickListener { view ->
                        it.onLabelViewClick(view, index)
                    }
                    this.addView(labelView)
                }
            }
        }
    }

    private fun dp2px(dpValue: Float): Int {
        val scale = context.resources.displayMetrics.density
        return (dpValue * scale + 0.5f).toInt()
    }

    /**
     * 动态标签的适配器
     */
    abstract class UIFlowAdapter {
        private var mObserver: IUIFlowLayoutContract? = null

        fun registerListener(listener: IUIFlowLayoutContract) {
            mObserver = listener
        }


        abstract fun getSize(): Int
        abstract fun getView(
            context: Context,
            position: Int
        ): View

        open fun onLabelViewClick(view: View, position: Int) {

        }

        open fun notifyDataSetChanged() {
            mObserver?.notifyDataSetChanged()
        }
    }
}

//contract
interface IUIFlowLayoutContract {
      fun notifyDataSetChanged()
}


//attrs.xml  自定义attrs属性
 <declare-styleable name="UIFlowLayout">
        <attr name="ui_flow_row_gap" format="dimension" />
        <attr name="ui_flow_column_gap" format="dimension" />
        <attr name="ui_flow_max_row_count" format="integer" />
    </declare-styleable>

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容