6

我有一个ImageView用于绘制圆角图像的子类。代码基于this answer,如下:

public class ImageViewRoundedCorners extends ImageView {
    ...
    @Override
    protected void onDraw(Canvas canvas) {
        Bitmap scaledBitmap = Bitmap.createBitmap(getMeasuredWidth(),
                                                  getMeasuredHeight(),
                                                  Bitmap.Config.ARGB_8888); 
        Canvas scaledCanvas = new Canvas(scaledBitmap);
        super.onDraw(scaledCanvas); 
        drawRoundedCornerBitmap(canvas, scaledBitmap, 
                                getMeasuredWidth(), getMeasuredHeight());

        scaledBitmap.recycle();
    }

    protected void drawRoundedCornerBitmap(Canvas outputCanvas, Bitmap input, int w, int h) {
        Bitmap output = Bitmap.createBitmap(w, h, Config.ARGB_8888);
        Canvas canvas = new Canvas(output);

        mPaint.reset();
        mPaint.setAntiAlias(true);
        canvas.drawARGB(0, 0, 0, 0);

        mPaint.setStyle(Paint.Style.FILL);
        canvas.drawPath(mClipPath, mPaint);

        mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
        canvas.drawBitmap(input, 0, 0, mPaint);

        outputCanvas.drawBitmap(output, 0, 0, null);
    }
}

使用此代码,图像被绘制成适当的圆角。为了避免前两行的分配drawRoundedCornerBitmap,我想直接绘制到outputCanvas,这是最初传递给的画布onDraw。新的实现如下所示:

protected void drawRoundedCornerBitmap(...) {
    mPaint.reset();
    mPaint.setAntiAlias(true);
    outputCanvas.drawARGB(0, 0, 0, 0);

    mPaint.setStyle(Paint.Style.FILL);
    outputCanvas.drawPath(mClipPath, mPaint);

    mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.SRC_IN));
    outputCanvas.drawBitmap(input, 0, 0, mPaint);
}

出于某种原因,此代码似乎忽略了 Porter-Duff 模式,而只是使用正常(非圆角)角绘制图像。为什么会这样?绘制到Bitmap使原始代码工作的中间体有什么意义?

4

1 回答 1

0

创建一个可绘制的 Romain Guy 已经为您完成了这项工作。我们不是链接工厂,但他的博客文章对其进行了广泛的解释,并提供了一种有效的方法。圆角

真正的基本原则是创建一个BitmapShader并将其附加到一个对象,该对象以自定义方式Paint绘制,您只需将其应用于.DrawableDrawableImageView

使用 drawable 意味着 Image 只在画布上绘制一次,这意味着只绘制一次图像,然后 ImageView 所做的只是缩放 drawable。

于 2013-01-28T15:19:47.517 回答