JavaFX画布:如何在一个形状内部绘制另一个形状

3
我目前正在使用Java开发游戏,我一直在尝试找出如何在画布上绘制形状(例如圆形),并覆盖另一个形状(例如正方形),但只绘制与正方形相交的部分,类似于Photoshop中图层间的裁剪蒙版。
我尝试使用GraphicsContext.clearRect()清除底部形状不存在的区域,但这会移除背景。
下面的代码产生了这个结果:

actual result

然而,这是我想要的结果:

desired result

import javafx.animation.AnimationTimer;
import javafx.application.Application;
import javafx.scene.Group;
import javafx.scene.Scene;
import javafx.scene.canvas.Canvas;
import javafx.scene.canvas.GraphicsContext;
import javafx.scene.paint.Color;
import javafx.scene.transform.Rotate;
import javafx.stage.Stage;

public class CircleWithinSquareTest extends Application {
    @Override
    public void start(Stage stage) throws Exception {
        int width = 200;
        int height = 200;
        Canvas canvas = new Canvas(width, height);
        GraphicsContext gc = canvas.getGraphicsContext2D();

        AnimationTimer timer = new AnimationTimer() {
            final int bgCellSize = 8;
            final int x = 100;
            final int y = 100;
            double angle = 0;

            @Override
            public void handle(long now) {
                /* Draw checkered background */
                gc.setFill(Color.WHITE);
                gc.fillRect(0, 0, width, height);
                gc.setFill(Color.LIGHTGRAY);
                boolean odd = false;
                for (int y = 0; y < height; y += bgCellSize) {
                    odd = !odd;
                    for (int x = odd ? 0 : bgCellSize; x < width; x += bgCellSize * 2) {
                        gc.fillRect(x, y, bgCellSize, bgCellSize);
                    }
                }
                /* Draw square */
                gc.setFill(Color.BLUE);
                gc.fillRect(x, y, 50, 50);
                /* Draw circle */
                gc.save();
                angle += 5;
                if (angle >= 360) {
                    angle = 0;
                }
                Rotate r = new Rotate(angle, x, y);
                gc.setTransform(r.getMxx(), r.getMyx(), r.getMxy(), r.getMyy(), r.getTx(), r.getTy());
                gc.setFill(Color.RED);
                gc.fillOval(x, y, 30, 30);
                gc.restore();
            }
        };
        timer.start();

        Group root = new Group(canvas);
        Scene scene = new Scene(root);

        stage.setScene(scene);
        stage.show();
    }
}
1个回答

3

在setTransform之前,您可以使用裁剪。请添加以下代码:

gc.beginPath();
gc.rect(x, y, 50, 50);
gc.closePath();
gc.clip();

哦,哇,谢谢!我简直不敢相信我错过了这个。这比我预期的要简单得多。另外,我注意到如果我注释掉 gc.beginPath()gc.closePath(),它仍然会产生所需的结果? - Hat
1
没错,只要路径为空就可以工作。但是如果您在同一图形上下文中开始使用其他路径,请不要忘记将它们返回。 - Sergey Grinev

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接