Android關(guān)于裁剪圖片透明區(qū)域的算法

最近項(xiàng)目中遇到這么一個需求,需要裁剪掉圖片的透明區(qū)域。找了很久,最后確定,只能通過自己讀取Bitmap的像素點(diǎn)來讀取圖片的邊界來裁剪。下面記錄一下過程。

原圖如下

PorterDuffXfermode

最開始想的是使用PorterDuffXfermode來處理,因?yàn)檫@種方式其實(shí)很快的,但是,雖然這種方式可以用來處理圖片,但是無法滿足獲取圖片邊界的需求。
代碼如下:

    public static Bitmap cutStickerBitmap(String name, Context context, Bitmap bitmap) {
        Calendar calendar = Calendar.getInstance();
        Bitmap bit = Bitmap.createBitmap(bitmap.getWidth(), bitmap.getHeight(), Bitmap.Config.ARGB_8888);
        bit.eraseColor(Color.WHITE);
        Canvas canvas = new Canvas(bit);

        Paint paint = new Paint();
        paint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.DST_IN));

        canvas.drawBitmap(bitmap, 0, 0, paint);
        Paint paintRect = new Paint();
        paintRect.setColor(Color.RED);
        paintRect.setAntiAlias(true);
        paintRect.setStyle(Paint.Style.STROKE);
        paintRect.setStrokeWidth(4);
        Rect rect = new Rect();
        //畫一個框表示邊界
        rect.set(0, 0, bitmap.getWidth(), bitmap.getHeight());
        canvas.drawRect(rect, paintRect);
        Log.i("xx", " 運(yùn)行時間 cutStickerBitmap " + (Calendar.getInstance().getTimeInMillis() - calendar.getTimeInMillis()));
        return bit;
    }

代碼處理的時間,可以看到時間很快,可惜不符合需求:



處理的結(jié)果如下:

自己寫算法讀像素

關(guān)于這個算法,其實(shí),我寫過兩個版本。 自己寫算法的話,需要自己評估你自己的需求和結(jié)果。一開始,為了防止圖形是不連續(xù)的,我采取的思路是從圖片四周來獲取邊界。思路如圖:



這么做的確是可以保證百分百正確的,但是這么做有一個問題,那就是因?yàn)槊看蔚玫降膱D片分辨率不一樣,在有的時候處理一次需要10s以上,這實(shí)在是太慢了。

后來仔細(xì)觀察之后發(fā)現(xiàn),其實(shí)不需要那么嚴(yán)格的,因?yàn)榈玫降乃械膱D形都是連續(xù)的,所以新的算法就是從中間開始處理的。



沿著中線,先向右邊獲取邊界,再沿著中線從左邊獲取邊界,同時為了加快速度,采取的一次讀兩個像素的方式,如果要求嚴(yán)謹(jǐn)可以改成每次讀一個像素,算法代碼如下。

    //處理貼紙
    public static Bitmap cutStickerBitmap(Bitmap bitmap) {
        Calendar calendar = Calendar.getInstance();
        int top = 0;
        int left = 0;
        int right = bitmap.getWidth() - 1;
        int bottom = 0;

        for (int w = bitmap.getWidth() / 2; w < bitmap.getWidth(); w += 2) {
            int h = 0;
            for (h = 0; h < bitmap.getHeight(); h+=2) {
                int color = bitmap.getPixel(w, h);
                if (color != 0) {
                    if (top == 0) {
                        top = h;
                    }
                    top = Math.min(top, h);
                    break;
                }
            }

            if (h >= bitmap.getHeight() - 1) {
                right = w;
                break;
            }
        }

        for (int w = bitmap.getWidth() / 2 - 1; w >= 0; w -= 2) {
            int h = 0;
            for (h = 0; h < bitmap.getHeight(); h+=2) {
                int color = bitmap.getPixel(w, h);
                if (color != 0) {
                    if (top == 0) {
                        top = h;
                    }
                    top = Math.min(top, h);
                    break;
                }
            }

            if (h >= bitmap.getHeight() - 1) {
                left = w;
                break;
            }
        }

        for (int w = left; w <= right; w++) {
            for (int h = bitmap.getHeight() - 1; h >= top; h--) {
                int color = bitmap.getPixel(w, h);
                if (color != 0) {
                    bottom = Math.max(bottom, h);
                    break;
                }
            }
        }

        //對邊界值做一次判斷,保證嚴(yán)謹(jǐn)
        int x = (int) Math.max(left  , 0f);
        int y = (int) Math.max(top  , 0f);
        int w = (int) Math.min(right - left  , bitmap.getWidth() - x);
        int h = (int) Math.min(bottom - top  , bitmap.getHeight() - y);
        if (x + w > bitmap.getWidth()) {
            x = 0;
            w = bitmap.getWidth();
        }

        if (y + h > bitmap.getHeight()) {
            y = 0;
            h = bitmap.getHeight();
        }
        Log.i("xx", " 運(yùn)行時間 cutStickerBitmap " + (Calendar.getInstance().getTimeInMillis() - calendar.getTimeInMillis()));

        return Bitmap.createBitmap(bitmap, x, y, w, h);
    }

代碼處理的時間如下:
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容