标签流式布局

展示多个标签,一行满后自动换行,可以设置行间距和标签的左右间距
效果图
这里写图片描述

布局文件

    <com.rsd.library.widget.RFlowLayout
        android:layout_width="wrap_content"
        app:fTextBg="@drawable/border"
        app:fLineSpace="10dp"
        app:fTagSpace="10dp"
        app:fTextSize="23sp"
        android:layout_height="wrap_content"/>

继承viewgroup,重写起onmeasure计算wrap宽高,重写onlayout计算每个子view的位置
首先onMeasure

protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        int widthMode = MeasureSpec.getMode(widthMeasureSpec);
        int widthSize = MeasureSpec.getSize(widthMeasureSpec);
        int heightMode = MeasureSpec.getMode(heightMeasureSpec);
        int heightSize = MeasureSpec.getSize(heightMeasureSpec);
        int width = 0, height = getPaddingTop() + getPaddingBottom();
        int lineWidth = getPaddingLeft() + getPaddingRight(), lineHeight = 0;
        View child;
        int space = 0;
        for (int i = 0; i < getChildCount(); i++) {
            child = getChildAt(i);
            measureChild(child, widthMeasureSpec, heightMeasureSpec);
            int cWidth = child.getMeasuredWidth();
            int cHeight = child.getMeasuredHeight();
            if (lineWidth + cWidth + space > widthSize) {
                space = 0;
                //如果再加一个标签宽度不够就换行
                width = Math.max(lineWidth, cWidth); //总行宽改变
                lineWidth = cWidth + getPaddingLeft() + getPaddingRight();//新行宽
                height += lineHeight;//总行高增加
                if (i != 0) {
                    cHeight += lineSpace;
                }
                lineHeight = cHeight; //新行高
            } else {
                space = tagSpace;
                if (i != 0) {
                    cWidth += tagSpace;
                }
                lineWidth += cWidth; //行宽增加
                lineHeight = Math.max(lineHeight, cHeight); //改变该行高度
            }
            if (i == getChildCount() - 1) {
                width = Math.max(width, lineWidth);
                height += lineHeight;
            }
        }
        if (widthMode == MeasureSpec.EXACTLY) {
            width = widthSize;
        }
        if (heightMode == MeasureSpec.EXACTLY) {
            height = heightSize;

        }
        setMeasuredDimension(width, height);
    }
protected void onLayout(boolean changed, int l, int t, int r, int b) {
        int left = getPaddingLeft(), top = getPaddingTop();//每个view的左上开始位置
        int ml = 0, mt = 0;//每个子view的左上边距,正常是两个space,但第一行和每行第一个为0
        int width = getWidth();
        int lineHeight = 0;
        View child;
        for (int i = 0; i < getChildCount(); i++) {
            child = getChildAt(i);
            if (left + child.getMeasuredWidth() + tagSpace + getPaddingRight() > width) {//换行
                left = getPaddingLeft();
                ml = 0;//每行第一个没有左边距
                top += lineHeight;
                lineHeight = child.getMeasuredHeight();
                if (i == 0) {
                    mt = 0;//第一行没有上边据,其余的加上行间距
                } else {
                    mt = lineSpace;
                }
                lineHeight += mt;
            } else {
                if (i == 0) {//如果第一个没有左边距
                    ml = 0;
                } else {
                    ml = tagSpace;
                }
                lineHeight = Math.max(lineHeight, child.getMeasuredHeight() + mt);
            }
            child.layout(left + ml, top + mt,
                    left + ml + child.getMeasuredWidth(),
                    top + mt + child.getMeasuredHeight());
            left += child.getMeasuredWidth() + ml;
        }
    }

然后就是设置标签内容,点击事件,标签我就是添加了一个个textview

public void setTags(String[] tags) {
        removeAllViews();
        for (int i = 0; i < tags.length; i++) {
            TextView textView = new TextView(getContext());
            textView.setTextSize(textSize);
            textView.setTextColor(textColor);
            textView.setGravity(Gravity.CENTER);
            if (textBg != -1) {
                textView.setBackgroundResource(textBg);
            }
            textView.setText(tags[i]);
            textView.setTag(i);
            textView.setOnClickListener(this);
            addView(textView);
        }
        setSelectIndex(0);
    }

    //设置选中某个tag
    public void setSelectIndex(int index) {
        if (index == oldIndex) {
            return;
        }
        //改变上次选中的状态,默认选中是0
        TextView view = (TextView) getChildAt(oldIndex);
        view.setTextColor(textColor);
        if (textBg != -1) {
            view.setBackgroundResource(textBg);
        }
        //当前选中的
        view = (TextView) getChildAt(index);
        view.setTextColor(selectTextColor);
        if (selectTextBg != -1) {
            view.setBackgroundResource(selectTextBg);
        }
        oldIndex = index;
    }

    public void setOnItemClickListener(OnItemClickListener onItemClickListener) {
        this.onItemClickListener = onItemClickListener;
    }

    @Override
    public void onClick(View v) {
        int position = (int) v.getTag();
        if (onItemClickListener != null) {
            setSelectIndex(position);
            onItemClickListener.onTagClick(position);
        }
    }

    public interface OnItemClickListener {
        void onTagClick(int position);
    }
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Android中的流式布局是一种常用的布局方式,它可以根据内容的大小和数量自动调整控件的位置和大小,使得界面能够自适应屏幕的宽度。通常在需要展示多个标签、图片或文字等的场景下使用。 流式布局的特点是将内容按照先后顺序从左到右排列,当一行的宽度不足以容纳下一个控件时,会自动换行。这种布局方式能够节省空间,提高界面的可读性和美观性。 在Android中,可以使用FlowLayout这个第三方库来实现流式布局。使用FlowLayout的步骤如下:首先在项目的build.gradle文件中添加依赖,然后在布局文件中将根布局设置为FlowLayout,并在其中添加需要展示的控件,可以通过调整控件的属性来定义布局的样式和排列方式。 流式布局可以动态调整控件的位置和大小,可以通过设置权重来控制每个控件在水平方向上的占比,也可以设置边距来调整控件之间的间隔。另外,流式布局还可以为每个控件设置点击事件和长按事件,方便实现更丰富的交互效果。 总之,流式布局是一种灵活且强大的布局方式,可以有效地解决多个控件在界面上排列不下或排列不美观的问题,同时也能够提高界面的可读性和用户体验。在开发Android应用时,如果遇到需要展示多个标签、图片或文字等的场景,流式布局是一个很好的选择。 ### 回答2: Android流式布局是一种灵活的布局方式,用于在屏幕上动态自适应地显示一系列视图。它能够根据子视图的大小和屏幕大小自动调整子视图的位置和宽度。这种布局方式适用于显示不规则大小的子视图,尤其适用于显示标签、图片、标签云等。 Android流式布局可以通过使用LinearLayout或GridLayout来实现。在LinearLayout中,可以设置orientation属性为horizontal或vertical来实现水平或垂直流式布局。在GridLayout中,可以通过设置列数来控制每行显示的子视图数量。 Android流式布局的优点是可以根据屏幕的大小和方向自动调整子视图的布局,使得页面在不同设备上都能够良好地显示。同时,它也提供了更好的用户体验,因为用户可以在不同屏幕上以不同的方式查看和交互。 然而,Android流式布局也存在一些限制。由于其自适应特性,子视图的大小和位置可能会受到限制。此外,较复杂的布局可能会导致性能问题,因为在布局过程中需要进行多次测量和计算。因此,在使用流式布局时,需要谨慎处理子视图的大小和数量,以提高性能并避免布局过于复杂。 总结来说,Android流式布局是一种灵活而自适应的布局方式,适用于显示不规则大小的子视图。它可以根据屏幕的大小和方向自动调整子视图的布局,并提供更好的用户体验。然而,需要注意处理子视图的大小和数量,以提高性能并避免布局过于复杂。 ### 回答3: Android流式布局(Flow Layout)是一种动态适应屏幕宽度的布局方式,主要用于解决在屏幕上按行排列多个子视图的问题。 在传统的线性布局中,如果视图超出屏幕宽度,就会自动换行,但是每一行只会放置一个子视图。而在流式布局中,子视图会根据屏幕宽度自动换行,并且每一行可以放置多个子视图,适应屏幕不同宽度的设备。 流式布局的使用非常方便,只需要将子视图添加到流式布局中即可。它提供了一些属性来控制子视图在布局中的排列方式,比如子视图之间的间距、子视图的对齐方式等。此外,流式布局还可以通过设置权重属性,实现子视图的均匀分布或者按比例分布。 流式布局在一些场景下非常有用,比如在标签云、瀑布流展示等需要动态调整子视图排列的情况下。相比于其他布局方式,流式布局可以更好地利用屏幕空间,提高用户体验。 总之,Android流式布局是一种动态适应屏幕宽度的布局方式,可以方便地排列多个子视图,并提供了一些属性来控制子视图的排列方式和样式。它的使用简单灵活,适用于多种场景,可以有效地提高用户界面的可用性和美观性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值