如何修剪图像的透明度,但可以选择保持最小尺寸和位置

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

我有一张图像,其边缘周围有透明填充。我还有图像的内容边界(顶部、左侧、宽度和高度),但有时它会超出这些边界。

我正在尝试修剪所有透明像素的图像,但我可以选择保留最小边界。

例如,假设我有 10 张图像,尺寸均为 1000x1000。其中一些内容的位置为 100x100,内容大小为 100x100。有时内容有边框或其他“溢出内容”,因此内容在此之前开始。在某些图像上,边界是 {top:100, left:100, width: 100, height: 100}

如果有 10 像素边框或其他溢出内容,则边界可能是:

{top:90, left: 90, width: 120, height: 120}

如何修剪图像中的透明像素?

这是我到目前为止所拥有的:

trimImage(image) { var spritePixels = bitmapData; var spriteImage = image as HTMLImageElement; var data = spritePixels; // Uint8buffer var spriteWidth = sprite.width; var spriteHeight = sprite.height; var firstTopPixel = 0; var spritePixelLength = spritePixels.length; var newImageData = null; var newBounds = {top:0, left:0, width:0, height:0} var firstRowFound = false; var firstColumnFound = false; var firstRowWithContent = 0; var lastRowWithContent = 0; var firstColumnWithContent = 0; var lastColumnWithContent = 0; var pixel = -1; var row = 0; var column = 0; for(let i = 0; i < spritePixelLength; i += 4) { pixel++; const red = spritePixels[i]; const green = spritePixels[i + 1]; const blue = spritePixels[i + 2]; const alpha = spritePixels[i + 3]; row = Math.floor(pixel / spriteWidth); column = pixel - spriteWidth * row; //console.log("row: "+ row + " col: " + column); // ROWS // get first top row with a pixel in it if (firstRowFound==false && alpha!=0){ firstRowWithContent = row; firstRowFound = true; console.log("First non transparent pixel found at row: "); console.log("row: "+ row + " col: " + column); } // get last row with a content in it if (alpha!=0){ lastRowWithContent = row; } // COLUMNS // get first column with content in it if (firstColumnFound==false && alpha!=0) { console.log("First non transparent pixel found at column: "); console.log("row: "+ row + " col: " + column); firstColumnWithContent = column; firstColumnFound = true; } // get last column with content in it if (alpha!=0){ lastColumnWithContent = column; } } var bounds = { top: firstRowWithContent, bottom: lastRowWithContent, left: firstColumnWithContent, right: lastColumnWithContent, x: firstColumnWithContent, y: firstRowWithContent, width: lastColumnWithContent - firstColumnWithContent + 1, height: lastRowWithContent - firstRowWithContent + 1 } // bounds bottom and right might need + 1? console.log(bounds); // todo: get image at bounds return bounds; } var results = trimImage(image);


javascript image
1个回答
0
投票

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

Example image with transparent padding

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

示例代码

在此示例中,您将看到

before

after 之间的差异,每个图像都有红色边框,没有红色边框以显示透明图像填充结束的位置。 函数

trimImage

迭代每一行和每列像素以检测绘制像素的外部边界。然后它将这些像素重新绘制到画布上。

此代码使用现代风格的 JS。

<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('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.