Android Canvas使用layer實(shí)現(xiàn)圓形頭像

一蝙砌、前言

在開發(fā)中遇到這么一個(gè)問題。我們app注冊時(shí)候使用了微博的頭像阅悍,但是目前微博用戶初始頭像(未曾設(shè)置過頭像的時(shí)候)返回的是gif格式附鸽,而在我們的某個(gè)地方的頭像顯示控件SimpleDraweeView中需要使用圓形特性,但是的Fresco使用中g(shù)if圖無法實(shí)現(xiàn)圓形文虏。網(wǎng)上也有方案侣诺,當(dāng)背景不是純色,F(xiàn)resco解決gif不能轉(zhuǎn)圓形問題的終極方案氧秘,思路或許可行年鸳,但是太過麻煩。然后就考慮使用自定義view實(shí)現(xiàn)丸相。過程中發(fā)現(xiàn)一個(gè)簡單的圓形方案搔确,過程用到Canvas的layer知識,在此學(xué)習(xí)記錄下來。

二膳算、Canvas與Layer

  • Canvas座硕,即畫布。Canvas有一系列簡單的操作方法提供涕蜂,有 平移(translate)华匾,旋轉(zhuǎn)(rotate),縮放(scale),錯(cuò)切(skew)机隙,裁剪(clip)蜘拉,保存(save),保存圖層(saveLayer)有鹿,恢復(fù)(restore),恢復(fù)到指定狀態(tài)(restoreToCount)等旭旭。
    關(guān)于操作,在這篇文章中講的很透徹了葱跋,我就不一一累述您机。Canvas的變換
  • Layer,即圖層年局。詳細(xì)了解际看,參考Canvas的saveLayer理解

三、圓形頭像實(shí)現(xiàn)思路

利用Xfermode實(shí)現(xiàn)只繪制自己想要的區(qū)域矢否,即圓形部分仲闽。更多在 Android-PorterDuffXfermode的正確使用方式

package com.zii.canvas.layer;

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.graphics.Canvas;
import android.graphics.Paint;
import android.graphics.PorterDuff;
import android.graphics.PorterDuffXfermode;
import android.graphics.RectF;
import android.support.annotation.UiThread;
import android.util.AttributeSet;

import com.facebook.drawee.view.SimpleDraweeView;
import com.zii.canvas.R;

public class GenderAvatarView extends SimpleDraweeView {

    private Paint mDstPaint;
    private Paint mSrcPaint;
    private Paint mOverlayPaint;

    private Bitmap mBitmapFemale;
    private Bitmap mBitmapMale;
    private Bitmap mBitmapGender;

    private RectF mRoundRectF;
    private boolean mIsMale;

    public GenderAvatarView(Context context, AttributeSet attrs) {
        super(context, attrs);
        init();
    }

    public GenderAvatarView(Context context, AttributeSet attrs, int defStyle) {
        super(context, attrs, defStyle);
        init();
    }

    private void init() {
        mDstPaint = new Paint();
        mDstPaint.setAntiAlias(true);

        mSrcPaint = new Paint();
        mSrcPaint.setAntiAlias(true);
        mSrcPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));//重點(diǎn)

        mOverlayPaint = new Paint();
        mOverlayPaint.setAntiAlias(true);

        mBitmapFemale = BitmapFactory.decodeResource(getContext().getResources(), R.drawable.ic_setting_gender_lady);
        mBitmapMale = BitmapFactory.decodeResource(getContext().getResources(), R.drawable.ic_setting_gender_man);

        mRoundRectF = new RectF();

    }

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

    @Override
    protected void onSizeChanged(int w, int h, int oldw, int oldh) {
        super.onSizeChanged(w, h, oldw, oldh);

        mBitmapGender = mIsMale ? mBitmapMale : mBitmapFemale;
        mBitmapGender = zoomBitmap(mBitmapGender, w, h);
    }

    @Override
    protected void onLayout(boolean changed, int left, int top, int right, int bottom) {
        super.onLayout(changed, left, top, right, bottom);

        mRoundRectF.set(0, 0, getWidth(), getHeight());
    }

    @Override
    protected void onDraw(Canvas canvas) {
        canvas.saveLayer(mRoundRectF, mDstPaint, Canvas.ALL_SAVE_FLAG);
        canvas.drawRoundRect(mRoundRectF, mRoundRectF.centerX(), mRoundRectF.centerY(), mDstPaint);//dst,圓形區(qū)域

        canvas.saveLayer(mRoundRectF, mSrcPaint, Canvas.ALL_SAVE_FLAG);//paint使用src_in模式僵朗,與上一次相交保留src顯示
        super.onDraw(canvas);//src赖欣,頭像
        canvas.restore();

        canvas.drawBitmap(mBitmapGender, 0, 0, mOverlayPaint);//overlay
    }

    private Bitmap zoomBitmap(Bitmap src, int dstWidth, int dstHeight) {
        if (src == null || (src.getWidth() == dstWidth && src.getHeight() == dstHeight)) {
            return src;
        }

        return Bitmap.createScaledBitmap(src, dstWidth, dstHeight, true);
    }

    @UiThread
    public void setGender(boolean isMale) {
        this.mIsMale = isMale;
        invalidate();
    }
}

四、后記

emmm验庙,很簡單顶吮,是不是?同時(shí)也解決了fresco的SimpleDraweeView對gif不實(shí)現(xiàn)圓形的特性粪薛,同時(shí)根據(jù)我自己需要增加性別圈圈的覆蓋物了悴了。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市违寿,隨后出現(xiàn)的幾起案子湃交,更是在濱河造成了極大的恐慌,老刑警劉巖藤巢,帶你破解...
    沈念sama閱讀 217,277評論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件搞莺,死亡現(xiàn)場離奇詭異,居然都是意外死亡掂咒,警方通過查閱死者的電腦和手機(jī)才沧,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,689評論 3 393
  • 文/潘曉璐 我一進(jìn)店門迈喉,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人温圆,你說我怎么就攤上這事挨摸。” “怎么了捌木?”我有些...
    開封第一講書人閱讀 163,624評論 0 353
  • 文/不壞的土叔 我叫張陵油坝,是天一觀的道長。 經(jīng)常有香客問我刨裆,道長澈圈,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,356評論 1 293
  • 正文 為了忘掉前任帆啃,我火速辦了婚禮瞬女,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘努潘。我一直安慰自己诽偷,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,402評論 6 392
  • 文/花漫 我一把揭開白布疯坤。 她就那樣靜靜地躺著报慕,像睡著了一般。 火紅的嫁衣襯著肌膚如雪压怠。 梳的紋絲不亂的頭發(fā)上眠冈,一...
    開封第一講書人閱讀 51,292評論 1 301
  • 那天,我揣著相機(jī)與錄音菌瘫,去河邊找鬼蜗顽。 笑死,一個(gè)胖子當(dāng)著我的面吹牛雨让,可吹牛的內(nèi)容都是我干的雇盖。 我是一名探鬼主播,決...
    沈念sama閱讀 40,135評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼栖忠,長吁一口氣:“原來是場噩夢啊……” “哼崔挖!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起娃闲,我...
    開封第一講書人閱讀 38,992評論 0 275
  • 序言:老撾萬榮一對情侶失蹤虚汛,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后皇帮,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,429評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡蛋辈,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,636評論 3 334
  • 正文 我和宋清朗相戀三年属拾,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了将谊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,785評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡渐白,死狀恐怖尊浓,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情纯衍,我是刑警寧澤栋齿,帶...
    沈念sama閱讀 35,492評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站襟诸,受9級特大地震影響瓦堵,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜歌亲,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,092評論 3 328
  • 文/蒙蒙 一菇用、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧陷揪,春花似錦惋鸥、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,723評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至飞蚓,卻和暖如春滤港,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背玷坠。 一陣腳步聲響...
    開封第一講書人閱讀 32,858評論 1 269
  • 我被黑心中介騙來泰國打工蜗搔, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人八堡。 一個(gè)月前我還...
    沈念sama閱讀 47,891評論 2 370
  • 正文 我出身青樓樟凄,卻偏偏與公主長得像,于是被迫代替她去往敵國和親兄渺。 傳聞我的和親對象是個(gè)殘疾皇子缝龄,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,713評論 2 354

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

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,099評論 25 707
  • 在iOS中隨處都可以看到絢麗的動(dòng)畫效果,實(shí)現(xiàn)這些動(dòng)畫的過程并不復(fù)雜挂谍,今天將帶大家一窺iOS動(dòng)畫全貌叔壤。在這里你可以看...
    F麥子閱讀 5,110評論 5 13
  • 在iOS中隨處都可以看到絢麗的動(dòng)畫效果,實(shí)現(xiàn)這些動(dòng)畫的過程并不復(fù)雜口叙,今天將帶大家一窺ios動(dòng)畫全貌炼绘。在這里你可以看...
    每天刷兩次牙閱讀 8,488評論 6 30
  • 【Android 自定義View之繪圖】 基礎(chǔ)圖形的繪制 一、Paint與Canvas 繪圖需要兩個(gè)工具妄田,筆和紙俺亮。...
    Rtia閱讀 11,667評論 5 34
  • 此刻驮捍,在回家的公交車上。比較幸運(yùn)脚曾,沒有等多久东且,而且沒有太擠。 心里升起不太舒服的感覺本讥,從昨天下午發(fā)出"眾籌"消息到...
    作家阿紫閱讀 229評論 2 0