1

早上好,

我的部分应用程序具有写下一些手写笔记的能力。我正在使用指纹示例中的片段在我的应用程序中实现“橡皮擦”:

mPaint.setXfermode(new PorterDuffXfermode(
        PorterDuff.Mode.CLEAR));
mPaint.setStrokeWidth(45);
mPaint.setStrokeCap(Paint.Cap.ROUND);

问题是每当我擦除用户已擦除的黑色路径时。一旦用户拿起他们的手指,它就消失了,但在他们擦除时它就在那里。我的朋友告诉我,PorterDuff 模式不应该实时使用,但我不相信他。这是我的更多代码,以防我做了一些愚蠢的事情:

mPaint = new Paint();
mPaint.setAntiAlias(true);
mPaint.setDither(false);
mPaint.setColor(Color.BLACK);
mPaint.setStyle(Paint.Style.STROKE);
mPaint.setStrokeJoin(Paint.Join.ROUND);
mPaint.setStrokeCap(Paint.Cap.BUTT);
mPaint.setStrokeWidth(6);
mBitmap = Bitmap.createBitmap(800, 480,Bitmap.Config.ARGB_8888);
bBitmap = Bitmap.createBitmap(800, 480,Bitmap.Config.ARGB_8888);
bBitmap.eraseColor(Color.WHITE);
mCanvas = new Canvas(mBitmap);
mPath = new Path();
mBitmapPaint = new Paint();
maxX = maxY = 100;
minX = 750;
minY = 400;

//

private void touch_up() {
    mPath.lineTo(mX, mY);
    mCanvas.drawPath(mPath, mPaint);
    mPath.reset();
}

//

protected void onDraw(Canvas canvas) {    
    canvas.drawBitmap(bBitmap, 0, 0, null);
    canvas.drawBitmap(mBitmap, 0, 0, mBitmapPaint);
    canvas.drawPath(mPath, mPaint);
}

有可能做我想让这个橡皮擦做的事情吗?有一个更好的方法吗?我的朋友真的是对的吗?

此外,我尝试将白色绘制为擦除,但它不起作用,因为有时用户正在标记下方的图像。我可以通过并使每个白色像素都透明,但这似乎效率极低。

4

5 回答 5

4

我找到了一种正确执行此操作的方法,它可以用手指擦除,而不会出现黑色笔划。这是我的解决方案:

@Override
public boolean onTouchEvent(MotionEvent event) {
    float x = event.getX();
    float y = event.getY();

    switch (event.getAction()) {
    case MotionEvent.ACTION_DOWN:
        if(isEraseMode()){
            mPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
        }
        touch_start(x, y);
        invalidate();
        break;
    case MotionEvent.ACTION_MOVE:
        touch_move(x, y);
        if(isEraseMode()){
            mPath.lineTo(mX, mY);
            mCanvas.drawPath(mPath, mPaint);
            mPath.reset();
            mPath.moveTo(x, y);            
                }
        invalidate();
        break;
    case MotionEvent.ACTION_UP:
        touch_up();
        invalidate();
        break;
    }
    return true;
}  
于 2013-10-25T23:36:03.500 回答
2

好的,所以我想出了一个解决方案。我会把这个留给遇到类似问题的人:

case MotionEvent.ACTION_DOWN:
mPaint.setStrokeWidth(25);
            mPaint.setXfermode(new PorterDuffXfermode(
                    PorterDuff.Mode.CLEAR));
            mCanvas.drawCircle(x, y, 10, mPaint);
            isErase = true;
            invalidate();
        }
        touch_start(x, y);
        invalidate();
        break;
case MotionEvent.ACTION_MOVE:   
        if(isErase)
        {
            mCanvas.drawCircle(x, y, 20, mPaint);
        }
        else{
            touch_move(x, y);
        }invalidate();
        break;
于 2012-11-27T20:55:39.567 回答
2

我也有同样的问题。

最后,我想出了解决方案。不要使用从 传递的画布onDraw(),而是使用mCanvas在构造函数中创建的画布。

于 2013-01-12T17:44:12.783 回答
1

(解决方法)它会让你变得不透明。初始化画布后,可能在onCreate您的活动中。 myCanvasView.setAlpha(0.99f);

于 2017-01-30T17:15:27.353 回答
0

我同意@Rovers,只需canvas以这种方式更改:

@Override
protected void onDraw(Canvas canvas) {
    canvas.drawBitmap(canvasBitmap, 0, 0, canvasPaint);
    if (!erase) {
        canvas.drawPath(drawPath, drawPaint);
    } else {
        drawCanvas.drawPath(drawPath, drawPaint);
    }
}

有两种 canvas intoonDraw方法 -->drawCanvascanvas.

于 2015-11-17T03:54:38.583 回答