流式布局展示评论标签

前言

电商应用开发时用户评论功能是一个很重要的模块,通过用户的反馈知道服务存在的不足,进而为改进工作流程提供依据。用户的评论会被企业使用大数据处理抽取出一些基本的评论标签,这些标签就能够展示整体评论上方供用户了解其他客户对商家或货物的整体看法。现在就来简单的使用流式布局实现展示用户评价标签。

效果

这里写图片描述

实现过程

流式布局的特点是能够根据用户提供的数据按照从左到右,从上到下的方式展示控件,如果在一行展示不下就会空下余下的位置并切到下一行展示。很显然Android自带的五大布局无法完成这样复杂的实现逻辑,需要自定义实现这种流式布局方法。

首先需要定义每个标签之前的横向和竖向的间距,有了间距之后就不会感到所有的标签都聚集在一起的拥挤感了。在attrs.xml文件中定义如下的自定义属性:

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <declare-styleable name="FlowLayout">
        <attr name="vertical_margin" format="dimension" />
        <attr name="horizontal_margin" format="dimension" />
    </declare-styleable>
</resources>

在FlowLayout自定义布局的构造方法里解析这两个数值:

public FlowLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
    super(context, attrs, defStyleAttr);
    if (attrs != null) {
        TypedArray array = context.obtainStyledAttributes(attrs, R.styleable.FlowLayout);
        mVerticalMargin = array.getDimensionPixelOffset(R.styleable.FlowLayout_vertical_margin, CommonUtils.dp2px(5));
        mHorizontalMargin = array.getDimensionPixelOffset(R.styleable.FlowLayout_horizontal_margin, CommonUtils.dp2px(5));
        array.recycle();
    }
    init();
}

接下来当用户设置标签数据的时候为这些标签数据添加对应的展示控件并初始化。

public void setData(List<String> data) {
    // 清空之前的数据
    mData.clear();
    mData.addAll(data);
    // 清空之前的控件
    removeAllViews();
    // 清空之前的行控件记录和高度记录
    mLineViews.clear();
    mHeights.clear();

    int count = data.size();
    for (int i = 0; i < count; i++) {
        String str = data.get(i);
        View view = mInflater.inflate(R.layout.item_comment, this, false);
        TextView textView = (TextView) view.findViewById(R.id.text);
        textView.setText(str);
        addView(view);
    }
}

在流式布局里定义一下几个简单的变量来记录控件的拜访位置:

// 展示的标签数据
private List<String> mData = new ArrayList<>();

// 每行展示的标签View,mLineViews.size()代表展示多少行,
// 里面的每个List<View>表示当前行展示的标签
private List<List<View>> mLineViews = new ArrayList<>();
// 每一行展示的高度
private List<Integer> mHeights = new ArrayList<>();

private LayoutInflater mInflater;

// 标签之间的横向和竖向边距
private int mVerticalMargin = CommonUtils.dp2px(5);
private int mHorizontalMargin = CommonUtils.dp2px(5);

前面添加了所有标签控件接下来开始在onMeasure测量期间判断每一行展示的标签控件和每行的展示高度,通过这些计算最终能够获取整个流式布局的高度和宽度。

@Override
protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
    // 获取父控件传递进来的测量数据
    int widthSize = MeasureSpec.getSize(widthMeasureSpec);
    int widthMode = MeasureSpec.getMode(widthMeasureSpec);
    int heightSize = MeasureSpec.getSize(heightMeasureSpec);
    int heightMode = MeasureSpec.getMode(heightMeasureSpec);

    int width = 0, height = 0, lineWidth = 0, lineHeight = 0;
    // 如果父控件要求必须是指定的widthSize,就按照widthSize来设置流式布局的宽度
    if (widthMode == MeasureSpec.EXACTLY) {
        width = widthSize;
    } else {
        // 否则就认为是wrap_content,使用屏幕宽度和指定最大值里最小的
        width = Math.min(widthSize, CommonUtils.getScreenWidth());
    }

    // 清空每行展示标签和每行高度
    mLineViews.clear();
    mHeights.clear();

    int count = getChildCount();
    int maxLineWidth = width - getPaddingLeft() - getPaddingRight();
    List<View> lineViews = new ArrayList<>();
    mLineViews.add(lineViews);

    // 遍历所有标签控件
    for (int i = 0; i < count; i++) {
        View child = getChildAt(i);
        // 测量子控件的高度
        measureChild(child, widthMeasureSpec, heightMeasureSpec);
        // 如果当前行的宽度加上标签子控件的宽度大于一行的最大宽度,换行
        if (lineWidth + child.getMeasuredWidth() + mHorizontalMargin > maxLineWidth) {
            // 保存当前行的高度
            mHeights.add(lineHeight);
            // 为下一行新增保存下一行标签的容器
            lineViews = new ArrayList<>();
            mLineViews.add(lineViews);
            lineViews.add(child);

            // 下一行目前测量到的标签所占据的宽度
            lineWidth = child.getMeasuredWidth() + mHorizontalMargin;
            // 下一行测量到的高度
            lineHeight = child.getMeasuredHeight() + mVerticalMargin;
        } else {
            // 如果当前行测量到的标签还没有到换行的情况,将当前标签控件方法当前行
            lineViews.add(child);

            // 如果当前的标签高度超过目前测量的行高,行高改成较大的值
            if (lineHeight < child.getMeasuredHeight() + mVerticalMargin) {
                lineHeight = child.getMeasuredHeight() + mVerticalMargin;
            }

            // 增加目前测量到的标签占据宽度值
            lineWidth += child.getMeasuredWidth() + mHorizontalMargin;

            // 如果是最后一个控件,由于不会在做任何测量,需要把最后一行的高度保存下来
            if (i == count - 1) {
                mHeights.add(lineHeight);
            }
        }
    }


    // 根据前面测量每行的高度,相加起来获取展示标签需要的高度
    int maxHeight = getPaddingBottom() + getPaddingTop();
    for (int i = 0; i < mHeights.size(); i++) {
        maxHeight += mHeights.get(i);
    }

    if (heightMode == MeasureSpec.EXACTLY) {
        height = heightSize;
    } else {
        height = Math.min(maxHeight, heightSize);
    }

    // 设置流式布局的高度
    setMeasuredDimension(width, height);
}

测量完成之后需要使用每行的标签控件mLineViews布局,一行一行从前向后,从上到下开始做布局。

@Override
protected void onLayout(boolean changed, int l, int t, int r, int b) {
    if (changed) {
        // x开始位置从paddingLeft开始,y从paddingTop开始
        int x = getPaddingLeft(), y = getPaddingTop(), count = mLineViews.size(), lineHeight;
        // 遍历所有行
        for (int i = 0; i < count; i++) {
            List<View> viewList = mLineViews.get(i);
            lineHeight = mHeights.get(i);

            // 遍历每行里的每一个标签
            for (int j = 0; j < viewList.size(); j++) {
                // 放置标签控件
                View child = viewList.get(j);
                child.layout(x, y, x + child.getMeasuredWidth(), y + child.getMeasuredHeight());
                x += child.getMeasuredWidth() + mHorizontalMargin;
            }

            // 下一行x从paddingLeft开始,y需要加上当前行的行高
            x = getPaddingLeft();
            y += lineHeight;
        }
    }
}

通过这些简单的测量和布局操作就能够实现简单的流式布局,至于标签的选中点击事件等其实就是简单的添加背景和设置Click事件,这里就不再赘述。

  • 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、付费专栏及课程。

余额充值