将 HTML 画布裁剪为其可见像素(内容)的宽度/高度?

问题描述 投票:0回答:3

可以对 HTML

canvas
元素进行内部裁剪以适应其内容吗?

例如,如果我有一个 500x500 像素的画布,其中的随机位置只有一个 10x10 像素的正方形,是否有一个函数可以通过扫描可见像素和裁剪将整个画布裁剪为 10x10?


编辑:这被标记为Javascript方法检测不透明的PNG区域的重复,但事实并非如此。该问题详细介绍了如何找到画布中不透明内容的边界,但不是如何裁剪它。我的问题的第一个词是“裁剪”,所以这就是我想关注的重点。

javascript css html canvas
3个回答
14
投票

更好的修剪功能。

虽然给定的答案有效,但它包含一个潜在的危险缺陷,创建一个新的画布而不是裁剪现有的画布,并且(链接区域搜索)效率有些低。

如果您有对画布的其他引用,创建第二个画布可能会出现问题,这很常见,因为通常有两个对画布的引用,例如

canvas
ctx.canvas
。关闭可能会使删除引用变得困难,并且如果关闭结束于某个事件,您可能永远无法删除引用。

缺陷是画布不包含像素。允许将画布设置为零大小(

canvas.width = 0; canvas.height = 0;
不会抛出错误),但某些函数不能接受零作为参数,并且会抛出错误(例如,
ctx.getImageData(0,0,ctx.canvas.width,ctx.canvas.height);
是常见做法,但如果画布没有尺寸)。由于这与调整大小没有直接关系,因此可以忽略这种潜在的崩溃并进入生产代码。

链接搜索会检查每次搜索的所有像素,当找到边缘时包含一个简单的

break
将改善搜索,平均而言仍然有更快的搜索。同时向两个方向搜索,先上下后左右,这样会减少迭代次数。您可以通过逐步执行索引来提高性能,而不是为每个像素测试计算每个像素的地址。例如
data[idx++]
data[x + y * w]

快得多

更强大的解决方案。

以下函数将使用两遍搜索从画布上裁剪透明边缘,同时考虑第一遍的结果以减少第二遍的搜索区域。

如果没有像素,它不会裁剪画布,但会返回

false
,以便可以采取行动。如果画布包含像素,它将返回
true

无需更改对画布的任何引用,因为它已就地裁剪。

// ctx is the 2d context of the canvas to be trimmed
// This function will return false if the canvas contains no or no non transparent pixels.
// Returns true if the canvas contains non transparent pixels
function trimCanvas(ctx) { // removes transparent edges
    var x, y, w, h, top, left, right, bottom, data, idx1, idx2, found, imgData;
    w = ctx.canvas.width;
    h = ctx.canvas.height;
    if (!w && !h) { return false } 
    imgData = ctx.getImageData(0, 0, w, h);
    data = new Uint32Array(imgData.data.buffer);
    idx1 = 0;
    idx2 = w * h - 1;
    found = false; 
    // search from top and bottom to find first rows containing a non transparent pixel.
    for (y = 0; y < h && !found; y += 1) {
        for (x = 0; x < w; x += 1) {
            if (data[idx1++] && !top) {  
                top = y + 1;
                if (bottom) { // top and bottom found then stop the search
                    found = true; 
                    break; 
                }
            }
            if (data[idx2--] && !bottom) { 
                bottom = h - y - 1; 
                if (top) { // top and bottom found then stop the search
                    found = true; 
                    break;
                }
            }
        }
        if (y > h - y && !top && !bottom) { return false } // image is completely blank so do nothing
    }
    top -= 1; // correct top 
    found = false;
    // search from left and right to find first column containing a non transparent pixel.
    for (x = 0; x < w && !found; x += 1) {
        idx1 = top * w + x;
        idx2 = top * w + (w - x - 1);
        for (y = top; y <= bottom; y += 1) {
            if (data[idx1] && !left) {  
                left = x + 1;
                if (right) { // if left and right found then stop the search
                    found = true; 
                    break;
                }
            }
            if (data[idx2] && !right) { 
                right = w - x - 1; 
                if (left) { // if left and right found then stop the search
                    found = true; 
                    break;
                }
            }
            idx1 += w;
            idx2 += w;
        }
    }
    left -= 1; // correct left
    if(w === right - left + 1 && h === bottom - top + 1) { return true } // no need to crop if no change in size
    w = right - left + 1;
    h = bottom - top + 1;
    ctx.canvas.width = w;
    ctx.canvas.height = h;
    ctx.putImageData(imgData, -left, -top);
    return true;            
}

2
投票

HTML canvas 元素可以在内部裁剪以适合其内容吗?

是的,使用此方法(或类似的方法)将为您提供所需的坐标。背景不必是透明的,但为了任何实际用途而必须是统一的(修改代码以适合背景)。

获取坐标后,只需使用

drawImage()
即可渲染该区域:

示例(由于问题中没有提供代码,请根据需要采用):

// obtain region here (from linked method)
var region = {
  x: x1, 
  y: y1, 
  width: x2-x1, 
  height: y2-y1
};

var croppedCanvas = document.createElement("canvas");
croppedCanvas.width = region.width;
croppedCanvas.height = region.height;

var cCtx = croppedCanvas.getContext("2d");
cCtx.drawImage(sourceCanvas, region.x, region.y, region.width, region.height, 
                             0, 0, region.width, region.height);

现在

croppedCanvas
仅包含原始画布的裁剪部分。


0
投票

简单可读版本

两次搜索似乎几乎没有必要,至少在 2024 年是这样。这是我对一个更简单、更易读的版本的看法。

示例图片

使用以下具有透明填充的 320x320 像素图像:

Example image with transparent padding

来源:我自己(随意用于任何目的)

示例代码

在此示例中,您将看到 beforeafter 之间的差异,每个图像周围都有红色边框,以显示透明图像填充结束的位置。

函数

trimImage
迭代每一行和每列像素以检测绘制像素的外部边界。然后,它将该边界内的像素重新绘制到调整大小的画布上。

<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta
      name="viewport"
      content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"
    />

    <title>Strip transparent padding</title>

    <style>
      img, canvas {
        border: 1px solid red;
        margin: 0 10px;
      }
    </style>
  </head>
  <body>
    <script type="module">
      async function trimImage(image) {
        // Create a canvas
        const canvas = document.createElement('canvas')
        const context = canvas.getContext('2d')
        document.body.appendChild(canvas)

        // Convert the image to a bitmap
        const bitmap = await createImageBitmap(image)
        const { width, height } = bitmap

        // Get pixels
        canvas.width = width
        canvas.height = height
        context.drawImage(bitmap, 0, 0)
        const { data: pixels } = context.getImageData(0, 0, width, height)
        context.clearRect(0, 0, width, height)

        // Find new bounds by ignoring transparent pixels
        const bounds = { top: height, left: width, right: 0, bottom: 0 }

        for (const row of Array(height).keys()) {
          for (const col of Array(width).keys()) {
            if (pixels[row * width * 4 + col * 4 + 3] !== 0) {
              if (row < bounds.top) bounds.top = row
              if (col < bounds.left) bounds.left = col
              if (col > bounds.right) bounds.right = col
              if (row > bounds.bottom) bounds.bottom = row
            }
          }
        }

        const newWidth = bounds.right - bounds.left
        const newHeight = bounds.bottom - bounds.top

        // Draw new image
        canvas.width = newWidth
        canvas.height = newHeight
        context.drawImage(
          bitmap,
          bounds.left,
          bounds.top,
          newWidth,
          newHeight,
          0,
          0,
          newWidth,
          newHeight,
        )
      }

      // Load the image
      const image = new Image()
      // image.src = './images/vector-die-with-transparency.webp'
      image.src = ''
      image.onload = async () => {
        document.body.append('Original:')
        document.body.append(image)
        document.body.append(document.createElement('br'))

        document.body.append('Trimmed:')
        await trimImage(image)
      }
    </script>
  </body>
</html>

注意:我必须在此片段中将图像粘贴为 base64,特别是因为我无法从跨源(堆栈溢出 cdn)加载图像,因为

getImageData()
将错误为
The canvas has been tainted by cross-origin data

在正常情况下,您应该能够使用托管在同一源上的上传的 blob 或导入的图像。如果没有,请务必检查跨源方法。

© www.soinside.com 2019 - 2024. All rights reserved.