按下键盘时隐藏和显示背景图像,保留覆盖元素

发布于 2025-01-10 22:13:56 字数 1338 浏览 0 评论 0原文

我想在单击鼠标时在图像上绘制椭圆,当我按下键盘时,交替隐藏和显示下面的图像,而不清理椭圆。

我正在使用 createGraphics() 来存储图像数据,并使用remove(),因此当按下键盘时,我会看到图像消失,但它不起作用,

这是我尝试执行的操作的草图:

let isMouseBeeingPressed = false;
let img;
let isShow = true
let bufferImg;

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  loadImageBuffer();

}

function loadImageBuffer() {
  bufferImg = createGraphics(400, 400);
  bufferImg.image(img, 0, 0);
  image(bufferImg, 0, 0);

}


function draw() {
    if(isMouseBeeingPressed) {
      stroke(0, 0, 0, 50);
      fill(255);
      ellipse(mouseX, mouseY, 20);
    }
}

function keyPressed() {
  
  if(isShow) {
    bufferImg.remove();
  } else {
    image(bufferImg, 0, 0);
  }
  
   
  console.log('isShow:', isShow);
  return isShow = !isShow;
}

function mousePressed() {
  isMouseBeeingPressed = true;
}

function mouseReleased() {
  isMouseBeeingPressed = false;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

知道如何实现这一目标吗?

I would like to paint ellipses over an image when clicking the mouse, and when I press the keyboard, hide and show the underneath image alternatively, without cleaning the ellipses.

I'm using createGraphics() to store the image data, and remove() so when the keyboard is pressed I spect the image disappear but it doesn't work

Here is a sketch of what I trying to do:

let isMouseBeeingPressed = false;
let img;
let isShow = true
let bufferImg;

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  loadImageBuffer();

}

function loadImageBuffer() {
  bufferImg = createGraphics(400, 400);
  bufferImg.image(img, 0, 0);
  image(bufferImg, 0, 0);

}


function draw() {
    if(isMouseBeeingPressed) {
      stroke(0, 0, 0, 50);
      fill(255);
      ellipse(mouseX, mouseY, 20);
    }
}

function keyPressed() {
  
  if(isShow) {
    bufferImg.remove();
  } else {
    image(bufferImg, 0, 0);
  }
  
   
  console.log('isShow:', isShow);
  return isShow = !isShow;
}

function mousePressed() {
  isMouseBeeingPressed = true;
}

function mouseReleased() {
  isMouseBeeingPressed = false;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

Any idea of how to achieve this?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

橘亓 2025-01-17 22:13:56

最好的方法可能是将椭圆绘制到单独的缓冲区 (p5.Graphics),然后根据需要将其绘制在图像或空白背景的顶部。另一种方法是记录数组中每个椭圆的位置,以便可以重新绘制它们。然而,后一种方法将使用更多的内存,并且在绘制许多椭圆之后打开和关闭图像将有明显的延迟。

方法#1(通过p5.Graphics渲染)

let img;
let graphics;
let isShow = true;

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  image(img, 0, 0);
  graphics = createGraphics(width, height);
}

function draw() {
  if (mouseIsPressed) {
    graphics.stroke(0, 0, 0, 50);
    graphics.fill(255);
    graphics.ellipse(mouseX, mouseY, 20);
    
    background(255);
    if (isShow) {
      image(img, 0, 0);
    }
    image(graphics, 0, 0);
  }
}

function keyPressed(e) {
  isShow = !isShow;
  background(255);
  if (isShow) {
    image(img, 0, 0);
  }
  
  image(graphics, 0, 0);
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

更新:关于此方法需要注意的一件事是,每当按下鼠标时,都必须重新绘制整个场景(白色背景、图像(如果适用)和前景)。其原因是抗锯齿和透明度。当您在图形缓冲区上绘制椭圆时,由于抗锯齿,边缘将具有一些部分透明的像素。如果您重复将缓冲区绘制为覆盖层而不重新绘制其后面的内容,那么部分透明的像素将变得越来越不透明,直到它们被出售为黑色。这将导致椭圆的外边缘稍厚且像素化程度更高。

方法#2(椭圆位置数组)

let img;
let isShow = true
let positions = [];

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  image(img, 0, 0);
}


function draw() {
  if (mouseIsPressed) {
    stroke(0, 0, 0, 50);
    fill(255);
    ellipse(mouseX, mouseY, 20);
    positions.push([mouseX, mouseY]);
  }
}

function keyPressed() {
  isShow = !isShow;
  background(255);
  if (isShow) {
    image(img, 0, 0);
  }
  stroke(0, 0, 0, 50);
  fill(255);
  for (const [x, y] of positions) {
    ellipse(x, y, 20);
  }
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

The best approach is probably to draw the ellipses to a separate buffer (p5.Graphics), and then draw that on top of the image or blank background as needed. An alternative approach would be to record the position of each ellipse in an array so that they can be redrawn. However, the latter approach will use more memory and switching the image on and off will have a noticeable delay after many ellipses have been drawn.

Approach #1 (render via p5.Graphics)

let img;
let graphics;
let isShow = true;

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  image(img, 0, 0);
  graphics = createGraphics(width, height);
}

function draw() {
  if (mouseIsPressed) {
    graphics.stroke(0, 0, 0, 50);
    graphics.fill(255);
    graphics.ellipse(mouseX, mouseY, 20);
    
    background(255);
    if (isShow) {
      image(img, 0, 0);
    }
    image(graphics, 0, 0);
  }
}

function keyPressed(e) {
  isShow = !isShow;
  background(255);
  if (isShow) {
    image(img, 0, 0);
  }
  
  image(graphics, 0, 0);
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

Update: one thing to note about this approach is that whenever the mouse is pressed, the entire scene must be redrawn (white background, image if applicable, and foreground). The reason for this is antialiasing and transparency. When you draw an ellipse on the graphics buffer the edges will have some partially transparent pixels due to antialiasing. If you repeatedly draw the buffer as an overlay without redrawing what is behind it then the partially transparent pixels will become less and less transparent until they are sold black. This will cause your ellipses to have a slightly thicker and more pixelated outer edge.

Approach #2 (Array of ellipse positions)

let img;
let isShow = true
let positions = [];

function preload() {
  img = loadImage(
    "https://images.pexels.com/photos/20787/pexels-photo.jpg?auto=compress&cs=tinysrgb&dpr=1&w=500"
  );
}

function setup() {
  createCanvas(500, 500);
  background(255);
  image(img, 0, 0);
}


function draw() {
  if (mouseIsPressed) {
    stroke(0, 0, 0, 50);
    fill(255);
    ellipse(mouseX, mouseY, 20);
    positions.push([mouseX, mouseY]);
  }
}

function keyPressed() {
  isShow = !isShow;
  background(255);
  if (isShow) {
    image(img, 0, 0);
  }
  stroke(0, 0, 0, 50);
  fill(255);
  for (const [x, y] of positions) {
    ellipse(x, y, 20);
  }
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/p5.js/1.4.0/p5.min.js"></script>

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文