Как стереть часть растрового изображения с помощью PorterDuff Xfermodes и Alpha Mask?

У меня есть альфа-маска, и я хочу использовать эту маску в качестве резины, когда пользователь касается экрана, эта часть растрового изображения должна быть стерта как форма альфа-маски. Я пробовал это, но он всегда получает весь растровое изображение как маску, а не альфа-часть. Позже пользователь сохранит растровое изображение, чтобы стертые части были прозрачными.

Вот код, который я использую:

    paintMask.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));

    Canvas canvas = new Canvas(bitmapOriginal);
    int sc = canvas.saveLayer(0, 0, width, height, null, Canvas.ALL_SAVE_FLAG);
    canvas.drawBitmap(bitmapOriginal, 0, 0, null);
    canvas.drawBitmap(bitmapMask, 0, 0, paintMask);
    canvas.restoreToCount(sc);

1 ответ

public class DrawingView extends View implements View.OnTouchListener {

    private int x = 0;
    private int y = 0;

    Bitmap bitmap;
    Path circlePath;
    Paint circlePaint;

    private final Paint paint = new Paint();
    private final Paint eraserPaint = new Paint();


    public DrawView(Context context){
        super(context);
        setFocusable(true);
        setFocusableInTouchMode(true);
        this.setOnTouchListener(this);

        // Set background
        this.setBackgroundColor(Color.CYAN);
        bp = BitmapFactory.decodeResource(getResources(), R.drawable.bg);

        // Set bitmap
        bitmap = Bitmap.createBitmap(320, 480, Bitmap.Config.ARGB_8888);
        bitmapCanvas = new Canvas();
        bitmapCanvas.setBitmap(bitmap);
        bitmapCanvas.drawColor(Color.TRANSPARENT);
        bitmapCanvas.drawBitmap(bp, 0, 0, null);

        circlePath = new Path();
        circlePaint = new Paint();
        circlePaint.setAntiAlias(true);
        circlePaint.setColor(Color.BLUE);
        circlePaint.setStyle(Paint.Style.STROKE);
        circlePaint.setStrokeJoin(Paint.Join.MITER);
        circlePaint.setStrokeWidth(4f);

        // Set eraser paint properties
        eraserPaint.setAlpha(0);
        eraserPaint.setStrokeJoin(Paint.Join.ROUND);
        eraserPaint.setStrokeCap(Paint.Cap.ROUND);
        eraserPaint.setXfermode(new PorterDuffXfermode(PorterDuff.Mode.CLEAR));
        eraserPaint.setAntiAlias(true);

    }

    @Override
    public void onDraw(Canvas canvas) {

        canvas.drawBitmap(bitmap, 0, 0, paint);
        bitmapCanvas.drawCircle(x, y, 30, eraserPaint);

        canvas.drawPath(circlePath, circlePaint);
    }

    public boolean onTouch(View view, MotionEvent event) {
        x = (int) event.getX();
        y = (int) event.getY();

        bitmapCanvas.drawCircle(x, y, 30, eraserPaint);

        circlePath.reset();
        circlePath.addCircle(x, y, 30, Path.Direction.CW);

        int ac=event.getAction();
        switch(ac){
            case MotionEvent.ACTION_UP:
                Toast.makeText(MainActivity.this, String.valueOf(x), Toast.LENGTH_SHORT).show();
                circlePath.reset();
                break;
        }
        invalidate();
        return true;
    }
}

и в вашем файле макета...

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:background="#00ff00"
tools:context=".MainActivity">
<DrawingView
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:id="@+id/ln1"
    android:orientation="vertical"
    /></LinearLayout>

надеюсь, это поможет

Другие вопросы по тегам