Search code examples
javascriptcsshtmlcanvas

Cropping an HTML canvas to the width/height of its visible pixels (content)?


Can an HTML canvas element be internally cropped to fit its content?

For example, if I have a 500x500 pixel canvas with only a 10x10 pixel square at a random location inside it, is there a function which will crop the entire canvas to 10x10 by scanning for visible pixels and cropping?


Edit: this was marked as a duplicate of Javascript Method to detect area of a PNG that is not transparent but it's not. That question details how to find the bounds of non-transparent content in the canvas, but not how to crop it. The first word of my question is "cropping" so that's what I'd like to focus on.


Solution

  • Simple readable version

    A two-pass search seems hardly necessary, at least in 2024. Here's my take at a much simpler and more readable version.

    Example image

    Using the following image of 320x320 pixels with transparent padding:

    Example image with transparent padding

    Source: myself (feel free to use for any purpose)

    Example code

    In this example, you'll see the difference between before and after, each image with a red border around it to show where the transparent image padding ends.

    The function trimImage iterates over every row and column of pixels to detect the outer boundaries of the painted pixels. It then redraws the pixels within that boundary onto a resized canvas.

    <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>

    Note: I had to paste in the image as base64 in this snippet specifically because I couldn't load an image from cross origin (stack overflow cdn) because getImageData() will error as The canvas has been tainted by cross-origin data.

    Under normal circumstances you should be able to use the uploaded blob or imported image that's hosted on the same origin. If not, be sure to check out cross-origin method.