自定義view-流式布局

總是覺得需求才是促進(jìn)成長的一大動力哈哈

由于本人之前是做網(wǎng)頁開發(fā)的绢片,對于標(biāo)簽的流式布局只需要在flex布局內(nèi)設(shè)置幾個屬性就能完成了滤馍。而轉(zhuǎn)了android之后卻沒有那么好用的屬性了,所以一直都想做一個android版的流式布局
(網(wǎng)上一搜一大堆底循,但還是想要自己實現(xiàn)一波)

公司有一個需求:添加一個獎勵標(biāo)簽纪蜒。效果圖如下:


image.png

實現(xiàn)思路很簡單:
1.在onMearsure中遍歷list,計算出控件需要占據(jù)的總高度
2.在onDraw中同理計算出每一個item的左上角位置此叠,調(diào)用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中設(shè)置

    private int nowWidth;
    private int nowHeight;

    private int rectHeight; //矩形高度

    //暴露給外邊設(shè)置item數(shù)據(jù)的方法
    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);
        //設(shè)置抗鋸齒
        mPaint.setAntiAlias(true);
        //設(shè)置防抖動
        mPaint.setDither(true);
        //設(shè)置填充方式
        mPaint.setStyle(Paint.Style.FILL);


        textPaint = new Paint();
        textPaint.setColor(textColor);
        //設(shè)置文本字體大小
        textPaint.setTextSize(textSize);
        //設(shè)置抗鋸齒
        textPaint.setAntiAlias(true);
        //設(shè)置防抖動
        textPaint.setDither(true);
        //設(shè)置填充方式
        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();
        //文字基準(zhǔn)線的下部距離-文字基準(zhǔn)線的上部距離 = 文字高度
        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;
            }

        }
    }

    /**
     * 傳入的參數(shù)分別為文本和矩形左上角橫縱坐標(biāo)灭袁,矩形寬度(矩形高度上面已經(jīng)計算好)和一個畫布
     */
    @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);
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末猬错,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子茸歧,更是在濱河造成了極大的恐慌倦炒,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,406評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件软瞎,死亡現(xiàn)場離奇詭異逢唤,居然都是意外死亡,警方通過查閱死者的電腦和手機涤浇,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,732評論 3 393
  • 文/潘曉璐 我一進(jìn)店門鳖藕,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人只锭,你說我怎么就攤上這事著恩。” “怎么了?”我有些...
    開封第一講書人閱讀 163,711評論 0 353
  • 文/不壞的土叔 我叫張陵喉誊,是天一觀的道長邀摆。 經(jīng)常有香客問我,道長伍茄,這世上最難降的妖魔是什么栋盹? 我笑而不...
    開封第一講書人閱讀 58,380評論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮敷矫,結(jié)果婚禮上贞盯,老公的妹妹穿的比我還像新娘。我一直安慰自己沪饺,他們只是感情好躏敢,可當(dāng)我...
    茶點故事閱讀 67,432評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著整葡,像睡著了一般件余。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上遭居,一...
    開封第一講書人閱讀 51,301評論 1 301
  • 那天啼器,我揣著相機與錄音,去河邊找鬼俱萍。 笑死端壳,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的枪蘑。 我是一名探鬼主播损谦,決...
    沈念sama閱讀 40,145評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼岳颇!你這毒婦竟也來了照捡?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,008評論 0 276
  • 序言:老撾萬榮一對情侶失蹤话侧,失蹤者是張志新(化名)和其女友劉穎栗精,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體瞻鹏,經(jīng)...
    沈念sama閱讀 45,443評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡悲立,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,649評論 3 334
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了新博。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片薪夕。...
    茶點故事閱讀 39,795評論 1 347
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖叭披,靈堂內(nèi)的尸體忽然破棺而出寥殖,到底是詐尸還是另有隱情玩讳,我是刑警寧澤涩蜘,帶...
    沈念sama閱讀 35,501評論 5 345
  • 正文 年R本政府宣布嚼贡,位于F島的核電站,受9級特大地震影響同诫,放射性物質(zhì)發(fā)生泄漏粤策。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,119評論 3 328
  • 文/蒙蒙 一误窖、第九天 我趴在偏房一處隱蔽的房頂上張望叮盘。 院中可真熱鬧,春花似錦霹俺、人聲如沸柔吼。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,731評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽愈魏。三九已至,卻和暖如春想际,著一層夾襖步出監(jiān)牢的瞬間培漏,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,865評論 1 269
  • 我被黑心中介騙來泰國打工胡本, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留牌柄,地道東北人。 一個月前我還...
    沈念sama閱讀 47,899評論 2 370
  • 正文 我出身青樓侧甫,卻偏偏與公主長得像珊佣,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子披粟,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,724評論 2 354

推薦閱讀更多精彩內(nèi)容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,107評論 25 707
  • 用兩張圖告訴你彩扔,為什么你的 App 會卡頓? - Android - 掘金 Cover 有什么料? 從這篇文章中你...
    hw1212閱讀 12,723評論 2 59
  • 在Android應(yīng)用中有很多標(biāo)簽一樣的頁面僻爽,這些頁面是無法通過Google提供的View來完成虫碉,這個時候我們就需要...
    Wynne丶XXW閱讀 517評論 1 1
  • 問答題47 /72 常見瀏覽器兼容性問題與解決方案? 參考答案 (1)瀏覽器兼容問題一:不同瀏覽器的標(biāo)簽?zāi)J(rèn)的外補...
    _Yfling閱讀 13,751評論 1 92
  • 城市變成了巨人 鋼筋鐵骨 水泥鋪面 就連血液 也都由鐵流匯成 渾身難覓幾處柔軟 你頂住了擠壓 蜷縮在角落里 支起了...
    歸農(nóng)閱讀 419評論 0 1