自定义view-流式布局

总是觉得需求才是促进成长的一大动力哈哈

由于本人之前是做网页开发的,对于标签的流式布局只需要在flex布局内设置几个属性就能完成了。而转了android之后却没有那么好用的属性了,所以一直都想做一个android版的流式布局
(网上一搜一大堆,但还是想要自己实现一波)

公司有一个需求:添加一个奖励标签。效果图如下:


image.png

实现思路很简单:
1.在onMearsure中遍历list,计算出控件需要占据的总高度
2.在onDraw中同理计算出每一个item的左上角位置,调用drawItem方法绘制出item

下面是源代码

import android.content.Context;
import android.graphics.Canvas;
import android.graphics.Color;
import android.graphics.Paint;
import android.graphics.Path;
import android.os.Build;
import android.support.annotation.Nullable;
import android.support.annotation.RequiresApi;
import android.util.AttributeSet;
import android.util.Log;
import android.view.View;

import java.util.ArrayList;
import java.util.List;

public class CustomFlexLayout extends View {

    private static final String TAG = "FlexLayout";

    private Paint mPaint;
    private Paint textPaint;
    private Path mPath;

    private List<String> list = new ArrayList<>();

    private int marginV = 16;//绘制的item的纵向margin值
    private int marginH = 10;//绘制的item的横向margin值
    private int paddingV = 20;//绘制的item的纵向padding值
    private int paddingH = 28;//绘制的item的横向padding值
    private int radius = 4;//绘制的item四个角的半径
    private int backColor = Color.parseColor("#f3f9ff");//绘制的item的背景颜色
    private int textColor = Color.parseColor("#007aff");//绘制的item的背景颜色
    private float textSize = 32f;

    private int height;//控件高度,在onMeasure中计算得到
    private int width;//控件宽度,在xml中设置

    private int nowWidth;
    private int nowHeight;

    private int rectHeight; //矩形高度

    //暴露给外边设置item数据的方法
    public void setList(List<String> list) {
        this.list = list;
        requestLayout();
    }

    public CustomFlexLayout(Context context) {
        this(context, null);
    }

    public CustomFlexLayout(Context context, @Nullable AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public CustomFlexLayout(Context context, @Nullable AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);

        init();
    }

    private void init() {
        mPaint = new Paint();
        mPaint.setColor(backColor);
        //设置抗锯齿
        mPaint.setAntiAlias(true);
        //设置防抖动
        mPaint.setDither(true);
        //设置填充方式
        mPaint.setStyle(Paint.Style.FILL);


        textPaint = new Paint();
        textPaint.setColor(textColor);
        //设置文本字体大小
        textPaint.setTextSize(textSize);
        //设置抗锯齿
        textPaint.setAntiAlias(true);
        //设置防抖动
        textPaint.setDither(true);
        //设置填充方式
        textPaint.setStyle(Paint.Style.FILL);

        mPath = new Path();

        rectHeight = marginV * 2 + paddingV * 2 + getFontHeight(textPaint);
        nowHeight = 0;
        nowWidth = 0;

    }

    private int getTextLength(String text) {
        return (int) textPaint.measureText(text);
    }

    private int getRectWidth(String text) {
        return marginH * 2 + paddingH * 2 + getTextLength(text);
    }

    private int getFontHeight(Paint paint) {
        Paint.FontMetrics fm = paint.getFontMetrics();
        //文字基准线的下部距离-文字基准线的上部距离 = 文字高度
        return (int) -(fm.bottom + fm.top);
    }

    @Override
    protected void onMeasure(int widthMeasureSpec, int heightMeasureSpec) {
        super.onMeasure(widthMeasureSpec, heightMeasureSpec);

        nowWidth = 0;
        nowHeight = rectHeight;
        width = getMeasuredWidth();

        int length;
        for (int i=0;i<list.size();i++) {
            length = getRectWidth(list.get(i));
            if (nowWidth + length <= width) {
                nowWidth += length;
            }else {
                nowWidth = length;
                nowHeight += rectHeight;
            }
        }
        height = nowHeight;

        setMeasuredDimension(width, height);

    }

    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    @Override
    protected void onDraw(Canvas canvas) {
        super.onDraw(canvas);
        nowWidth = 0;
        nowHeight = 0;

        for (int i=0;i<list.size();i++) {
            int length = getRectWidth(list.get(i));
            Log.i(TAG, "text:" + list.get(i) + " length:" + length);
            //还未满一行
            if (nowWidth + length <= width) {
                drawitem(list.get(i), nowWidth, nowHeight, getRectWidth(list.get(i)), canvas);
                nowWidth += length;
            }else /*满一行*/{
                nowWidth = 0;
                nowHeight += rectHeight;
                drawitem(list.get(i), nowWidth, nowHeight, getRectWidth(list.get(i)), canvas);
                nowWidth += length;
            }

        }
    }

    /**
     * 传入的参数分别为文本和矩形左上角横纵坐标,矩形宽度(矩形高度上面已经计算好)和一个画布
     */
    @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
    private void drawitem(String text, int startX, int startY, int rectWidth, Canvas canvas) {
        //Log.i(TAG, "text:" + text);
        //绘制矩形背景
        canvas.drawRoundRect(startX + marginH, startY + marginV, startX + rectWidth - marginH, startY + rectHeight - marginV, radius, radius, mPaint);
        //绘制中心文字
        canvas.drawText(text, startX + (rectWidth/2-getTextLength(text)/2),
                startY + getFontHeight(textPaint)/2 + rectHeight/2, textPaint);
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,077评论 25 707
  • 用两张图告诉你,为什么你的 App 会卡顿? - Android - 掘金 Cover 有什么料? 从这篇文章中你...
    hw1212阅读 12,714评论 2 59
  • 在Android应用中有很多标签一样的页面,这些页面是无法通过Google提供的View来完成,这个时候我们就需要...
    Wynne丶XXW阅读 517评论 1 1
  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,748评论 1 92
  • 城市变成了巨人 钢筋铁骨 水泥铺面 就连血液 也都由铁流汇成 浑身难觅几处柔软 你顶住了挤压 蜷缩在角落里 支起了...
    归农阅读 419评论 0 1