Android 圓角跟啤、圓形 ImageView 實(shí)現(xiàn)

一、 特點(diǎn)

  • 基于AppCompatImageView擴(kuò)展
  • 支持圓角起趾、圓形顯示
  • 可繪制邊框诗舰,圓形時(shí)可繪制內(nèi)外兩層邊框
  • 支持邊框不覆蓋圖片
  • 可繪制遮罩
  • ......

二、基本原理

我們要實(shí)現(xiàn)的圖片控件繼承自AppCompatImageView阳掐,它是ImageView的子類始衅,但提供了更好的兼容性冷蚂,我們?cè)诖嘶A(chǔ)上添加了若干自定義的屬性和方法以實(shí)現(xiàn)最終的 NiceImageView

public class NiceImageView extends AppCompatImageView {
    ......
}

要實(shí)圓角或者圓形的顯示效果,就是對(duì)圖片顯示的內(nèi)容區(qū)域進(jìn)行“裁剪”汛闸,只顯示指定的區(qū)域即可蝙茶。如何做呢?

一種比較直接的辦法是這樣的诸老,由于圖片是被繪制在畫(huà)布上的隆夯,所以用canvasclipPath()方法先將畫(huà)布裁剪成指定形狀,這樣就能讓圖片按指定形狀顯示了别伏,重新draw()方法即可:

    @Override
    public void draw(Canvas canvas) {
        canvas.save();
        canvas.clipPath(path);
        super.draw(canvas);
        canvas.restore();
    }

這樣使用src蹄衷、background屬性給ImageView設(shè)置顯示的圖片都能達(dá)到預(yù)期的顯示效果。但是由于clipPath()方法不支持抗鋸齒厘肮,圖片邊緣會(huì)有明顯的毛糙感愧口,體驗(yàn)并不理想,所以需要尋找其它方法类茂。

另一種方法是使用圖像的 Alpha 合成模式耍属,即
PorterDuff 來(lái)實(shí)現(xiàn),官方文檔巩检。這里我們使用其中的DST_IN模式厚骗。整個(gè)過(guò)程就是先繪制目標(biāo)圖像,也就是圖片兢哭;再繪制原圖像领舰,即一個(gè)圓角矩形或者圓形,這樣最終目標(biāo)圖像只顯示和原圖像重合的區(qū)域迟螺。

xfermode = new PorterDuffXfermode(PorterDuff.Mode.DST_IN);
    @Override
    protected void onDraw(Canvas canvas) {
        // 使用離屏緩存冲秽,新建一個(gè)srcRectF區(qū)域大小的圖層
        canvas.saveLayer(srcRectF, null, Canvas.ALL_SAVE_FLAG);
        // ImageView自身的繪制流程,即繪制圖片
        super.onDraw(canvas);
        // 給path添加一個(gè)圓角矩形或者圓形
        if (isCircle) {
            path.addCircle(width / 2.0f, height / 2.0f, radius, Path.Direction.CCW);
        } else {
            path.addRoundRect(srcRectF, srcRadii, Path.Direction.CCW);
        }
        paint.setAntiAlias(true);
        // 畫(huà)筆為填充模式
        paint.setStyle(Paint.Style.FILL);
        // 設(shè)置混合模式
        paint.setXfermode(xfermode);
        // 繪制path
        canvas.drawPath(path, paint);
        // 清除Xfermode
        paint.setXfermode(null);
        // 恢復(fù)畫(huà)布狀態(tài)
        canvas.restore();
    }

到這里就實(shí)現(xiàn)了顯示為圓角或者圓形了煮仇。但是需要通過(guò)src屬性或者對(duì)應(yīng)的方法來(lái)設(shè)置圖片劳跃,否則不能達(dá)到預(yù)期效果。

三浙垫、繪制邊框

繪制邊框就相對(duì)容易理解了刨仑,只需要繪制一個(gè)指定樣式的圓角矩形或者圓形即可:

    private void drawBorders(Canvas canvas) {
        if (isCircle) {
            if (borderWidth > 0) {
                drawCircleBorder(canvas, borderWidth, borderColor, radius - borderWidth / 2.0f);
            }
            if (innerBorderWidth > 0) {
                drawCircleBorder(canvas, innerBorderWidth, innerBorderColor, radius - borderWidth - innerBorderWidth / 2.0f);
            }
        } else {
            if (borderWidth > 0) {
                drawRectFBorder(canvas, borderWidth, borderColor, borderRectF, borderRadii);
            }
        }
    }

    private void drawCircleBorder(Canvas canvas, int borderWidth, int borderColor, float radius) {
        initBorderPaint(borderWidth, borderColor);
        path.addCircle(width / 2.0f, height / 2.0f, radius, Path.Direction.CCW);
        canvas.drawPath(path, paint);
    }

    private void drawRectFBorder(Canvas canvas, int borderWidth, int borderColor, RectF rectF, float[] radii) {
        initBorderPaint(borderWidth, borderColor);
        path.addRoundRect(rectF, radii, Path.Direction.CCW);
        canvas.drawPath(path, paint);
    }

    private void initBorderPaint(int borderWidth, int borderColor) {
        path.reset();
        // 設(shè)置畫(huà)筆為描邊模式
        paint.setStyle(Paint.Style.STROKE);
        // 描邊寬度
        paint.setStrokeWidth(borderWidth);
        // 描邊顏色
        paint.setColor(borderColor);
    }

當(dāng)圖片顯示為圓形時(shí),還可以繪制一個(gè)內(nèi)邊框夹姥,但圓角矩形的話由于圓角大小的問(wèn)題杉武,目前只能設(shè)置一個(gè)邊框咯。

但是有個(gè)問(wèn)題辙售,繪制的邊框會(huì)覆蓋在圖片上轻抱,如果邊框太寬會(huì)導(dǎo)致圖片的可見(jiàn)區(qū)域變小了,影像顯示效果旦部,像這樣祈搜,左下角的花盆不見(jiàn)了:



那么如何讓邊框不覆蓋在圖片上呢较店?可以在 Alpha 合成繪制前先將畫(huà)布縮小一定比例,最后再繪制邊框容燕,這樣問(wèn)題就解決了梁呈。

    @Override
    protected void onDraw(Canvas canvas) {
        canvas.saveLayer(srcRectF, null, Canvas.ALL_SAVE_FLAG);
        // 縮小畫(huà)布
        if (!isCoverSrc) {
            float sx = 1.0f * (width - 2 * borderWidth - 2 * innerBorderWidth) / width;
            float sy = 1.0f * (height - 2 * borderWidth - 2 * innerBorderWidth) / height;
            // 縮小畫(huà)布,使圖片內(nèi)容不被border蘸秘、padding覆蓋
            canvas.scale(sx, sy, width / 2.0f, height / 2.0f);
        }
        ......
        canvas.restore();
        // 繪制邊框
        drawBorders(canvas);
    }

縮放后的ImageView顯示區(qū)域的寬高就是原寬官卡、高分別減去2倍的邊框?qū)挾龋@樣縮小的比例也就顯而易見(jiàn)了醋虏。效果如下寻咒,左下角的花盆出來(lái)了:


四、繪制遮罩

遮罩可以理解為一層帶透明度的顏色颈嚼,遮罩默認(rèn)不繪制毛秘,當(dāng)制定了遮罩顏色時(shí)才會(huì)繪制,實(shí)現(xiàn)很簡(jiǎn)單:

    @Override
    protected void onDraw(Canvas canvas) {
        ......      
        // 繪制遮罩
        if (maskColor != 0) {
            paint.setColor(maskColor);
            canvas.drawPath(path, paint);
        }
        canvas.restore();
        drawBorders(canvas);
    }

例如加一個(gè)透明度30%的紅色遮罩后的效果:


核心的實(shí)現(xiàn)邏輯就這些了粘舟,剩下的就是自定義屬性和方法了奈嘿,有興趣的可以看源碼系羞,都很簡(jiǎn)單,希望對(duì)你有所幫助吧轻庆!

更多細(xì)節(jié)及用法見(jiàn)GitHub:https://github.com/SheHuan/NiceImageView

五旬薯、其它

如果你需要實(shí)現(xiàn)類似釘釘?shù)膱A形組合頭像晰骑,例如:


可以先生成對(duì)應(yīng)的Bitmap,并用圓形的 NiceImageView 顯示即可绊序。如何生成組合Bitmap可以參考這里:CombineBitmap硕舆。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市骤公,隨后出現(xiàn)的幾起案子抚官,更是在濱河造成了極大的恐慌,老刑警劉巖阶捆,帶你破解...
    沈念sama閱讀 210,914評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件凌节,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡洒试,警方通過(guò)查閱死者的電腦和手機(jī)倍奢,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,935評(píng)論 2 383
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)垒棋,“玉大人卒煞,你說(shuō)我怎么就攤上這事〉鸺埽” “怎么了畔裕?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,531評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵衣撬,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我扮饶,道長(zhǎng)淮韭,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,309評(píng)論 1 282
  • 正文 為了忘掉前任贴届,我火速辦了婚禮靠粪,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘毫蚓。我一直安慰自己占键,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,381評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布元潘。 她就那樣靜靜地躺著畔乙,像睡著了一般。 火紅的嫁衣襯著肌膚如雪翩概。 梳的紋絲不亂的頭發(fā)上牲距,一...
    開(kāi)封第一講書(shū)人閱讀 49,730評(píng)論 1 289
  • 那天,我揣著相機(jī)與錄音钥庇,去河邊找鬼牍鞠。 笑死,一個(gè)胖子當(dāng)著我的面吹牛评姨,可吹牛的內(nèi)容都是我干的难述。 我是一名探鬼主播,決...
    沈念sama閱讀 38,882評(píng)論 3 404
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼吐句,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼胁后!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起嗦枢,我...
    開(kāi)封第一講書(shū)人閱讀 37,643評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤攀芯,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后文虏,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體侣诺,經(jīng)...
    沈念sama閱讀 44,095評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,448評(píng)論 2 325
  • 正文 我和宋清朗相戀三年择葡,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了紧武。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,566評(píng)論 1 339
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡敏储,死狀恐怖阻星,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤妥箕,帶...
    沈念sama閱讀 34,253評(píng)論 4 328
  • 正文 年R本政府宣布滥酥,位于F島的核電站,受9級(jí)特大地震影響畦幢,放射性物質(zhì)發(fā)生泄漏坎吻。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,829評(píng)論 3 312
  • 文/蒙蒙 一宇葱、第九天 我趴在偏房一處隱蔽的房頂上張望瘦真。 院中可真熱鬧,春花似錦黍瞧、人聲如沸诸尽。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,715評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)您机。三九已至,卻和暖如春年局,著一層夾襖步出監(jiān)牢的瞬間际看,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,945評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工矢否, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留仲闽,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,248評(píng)論 2 360
  • 正文 我出身青樓兴喂,卻偏偏與公主長(zhǎng)得像蔼囊,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子衣迷,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,440評(píng)論 2 348

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