PorterDuffXfermode 實(shí)現(xiàn)刮刮卡效果

今天閑來(lái)無(wú)事做楚堤,就想著鞏固一下PorterDuffXfermode 的使用吧,先是做了圓形和圓角矩形頭像效果含懊,然后就做了個(gè)刮刮卡效果身冬。如下圖所示

20160509220814491.png

當(dāng)然 PorterDuffXfermode的詳細(xì)使用在這里我就不在強(qiáng)調(diào)了,可以去 Google 一下岔乔,下面主要來(lái)介紹一下效果的實(shí)現(xiàn)原理酥筝。

其實(shí)這個(gè)效果十分的簡(jiǎn)單,主要就是 PorterDuffXfermode 模式的設(shè)置和 Paint 的屬性設(shè)置雏门,下面來(lái)看看核心代碼

mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.testpicture);
        mDstBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mSrcBitmap.getHeight(), Bitmap.Config.ARGB_8888);
        mPath = new Path();

        paint = new Paint();
        paint.setAntiAlias(true);
        paint.setAlpha(0);
        paint.setStyle(Paint.Style.STROKE);
        paint.setStrokeWidth(50);
        paint.setStrokeCap(Paint.Cap.ROUND);
        paint.setStrokeJoin(Paint.Join.ROUND);
        paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));
        mCanvas = new Canvas(mDstBitmap);
        mCanvas.drawColor(Color.GRAY);

看看 paint 的屬性設(shè)置嘿歌,占了代碼的 70%,看看都是哪些效果茁影,

首先設(shè)置防鋸齒效果宙帝,將畫(huà)筆設(shè)置成 stroke模式并且設(shè)置寬度為 50,這么設(shè)置有什么用呢募闲?這就決定了我們每次滑動(dòng)步脓,路徑的寬度是多少。

至于 StrokeCapStrokeJoin 的屬性就是設(shè)置讓我們的畫(huà)筆點(diǎn)下去和滑動(dòng)的時(shí)候造成的效果更圓滑,仔細(xì)看看上面的效果演示沪编,是不是路徑的兩端都是圓弧形狀。

PorterDuffXfermode 設(shè)置成PorterDuff.Mode.DST_IN模式年扩,然后調(diào)用Canvas.drawColor()方法,這個(gè)時(shí)候?qū)嶋H上我們就是完成繪制一個(gè)和圖像相同大小的灰色區(qū)域蚁廓,也就是我們將要刮去的部分。

這個(gè)時(shí)候就很奇怪了厨幻,只是在 mDstBitmap上進(jìn)行繪制相嵌,我們并沒(méi)有 mSrcBitmap ,更別提進(jìn)行什么操作了况脆,怎么實(shí)現(xiàn)效果呢饭宾?實(shí)際上這個(gè)時(shí)候我們可以將mSrcBitmap虛擬化出來(lái),就當(dāng)它是存在的格了,而且是一個(gè)透明圖看铆,這也就是為什么我們將 畫(huà)筆的 Alpha設(shè)置為0,這樣我們每次手指滑動(dòng)的時(shí)候?qū)嶋H上就是在動(dòng)態(tài)增加 mSrcBitmap 的面積盛末,那么我們的mDstBitmap的灰色區(qū)域也就被擦除了弹惦。

下面看看OnDraw() 方法

 @Override
    protected void onDraw(Canvas canvas) {
        canvas.drawBitmap(mBitmap, 0, 0, null);
        canvas.drawBitmap(mDstBitmap, 0, 0, null);
    }

這個(gè)時(shí)候一定要畫(huà)mBitmap再畫(huà) mDstBitmap ,這樣的話就像是 圖形我已經(jīng)畫(huà)出來(lái)了悄但,不過(guò)被 mDstBitmap擋住了棠隐,這樣當(dāng)我們滑動(dòng)的時(shí)候,不斷擦除 mDstBitmap檐嚣,也就將圖形顯示出來(lái)了助泽。

下面粘貼出源碼,實(shí)際上嚎京,我們還可以測(cè)試圖的寬度和長(zhǎng)度嗡贺,對(duì)當(dāng)前屏幕進(jìn)行適配,以免出現(xiàn)圖形大于屏幕挖藏,部分圖形無(wú)法顯示的問(wèn)題暑刃,實(shí)現(xiàn)起來(lái)也不是很復(fù)雜,所以在此就不講述膜眠。

public class XferModeView extends View {

    private Bitmap mBitmap;
    private Bitmap mDstBitmap;
    private Canvas mCanvas;
    private Path mPath;
    private Paint paint;

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

    public XferModeView(Context context, AttributeSet attrs) {
        this(context, attrs, 0);
    }

    public XferModeView(Context context, AttributeSet attrs, int defStyleAttr) {
        super(context, attrs, defStyleAttr);
        initView();
    }

    private void initView() {

        mBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.testpicture);
        mDstBitmap = Bitmap.createBitmap(mBitmap.getWidth(), mBitmap.getHeight(), Bitmap.Config.ARGB_8888);
        mPath = new Path();

        paint = new Paint();
        paint.setAntiAlias(true);
        paint.setAlpha(0);
        paint.setStyle(Paint.Style.STROKE);
        paint.setStrokeWidth(50);
        paint.setStrokeCap(Paint.Cap.ROUND);
        paint.setStrokeJoin(Paint.Join.ROUND);
        paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));
        mCanvas = new Canvas(mDstBitmap);
        mCanvas.drawColor(Color.GRAY);
    }

    @Override
    protected void onDraw(Canvas canvas) {
        canvas.drawBitmap(mBitmap, 0, 0, null);
        canvas.drawBitmap(mDstBitmap, 0, 0, null);
    }

    @Override
    public boolean onTouchEvent(MotionEvent event) {

        float x = event.getX();
        float y = event.getY();

        switch (event.getAction()) {
            case MotionEvent.ACTION_DOWN:
                mPath.reset();
                mPath.moveTo(x, y);
                break;
            case MotionEvent.ACTION_MOVE:
                mPath.lineTo(x, y);
                break;
        }

        mCanvas.drawPath(mPath, paint);
        invalidate();
        return true;
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末岩臣,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子宵膨,更是在濱河造成了極大的恐慌架谎,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,376評(píng)論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件辟躏,死亡現(xiàn)場(chǎng)離奇詭異谷扣,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén)会涎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)裹匙,“玉大人,你說(shuō)我怎么就攤上這事末秃「乓常” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,966評(píng)論 0 347
  • 文/不壞的土叔 我叫張陵练慕,是天一觀的道長(zhǎng)惰匙。 經(jīng)常有香客問(wèn)我,道長(zhǎng)铃将,這世上最難降的妖魔是什么项鬼? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,432評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮劲阎,結(jié)果婚禮上绘盟,老公的妹妹穿的比我還像新娘。我一直安慰自己悯仙,他們只是感情好奥此,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,519評(píng)論 6 385
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著雁比,像睡著了一般稚虎。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上偎捎,一...
    開(kāi)封第一講書(shū)人閱讀 49,792評(píng)論 1 290
  • 那天蠢终,我揣著相機(jī)與錄音,去河邊找鬼茴她。 笑死寻拂,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的丈牢。 我是一名探鬼主播祭钉,決...
    沈念sama閱讀 38,933評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼己沛!你這毒婦竟也來(lái)了慌核?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,701評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤申尼,失蹤者是張志新(化名)和其女友劉穎垮卓,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體师幕,經(jīng)...
    沈念sama閱讀 44,143評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡粟按,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,488評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片灭将。...
    茶點(diǎn)故事閱讀 38,626評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡疼鸟,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出庙曙,到底是詐尸還是另有隱情愚臀,我是刑警寧澤,帶...
    沈念sama閱讀 34,292評(píng)論 4 329
  • 正文 年R本政府宣布矾利,位于F島的核電站,受9級(jí)特大地震影響馋袜,放射性物質(zhì)發(fā)生泄漏男旗。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,896評(píng)論 3 313
  • 文/蒙蒙 一欣鳖、第九天 我趴在偏房一處隱蔽的房頂上張望察皇。 院中可真熱鬧,春花似錦泽台、人聲如沸什荣。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,742評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)稻爬。三九已至,卻和暖如春蜕依,著一層夾襖步出監(jiān)牢的瞬間桅锄,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,977評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工样眠, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留友瘤,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,324評(píng)論 2 360
  • 正文 我出身青樓檐束,卻偏偏與公主長(zhǎng)得像辫秧,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子被丧,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,494評(píng)論 2 348

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