Как обрезать прозрачность изображения, но при необходимости сохранить минимальный размер и положение

У меня есть изображение с прозрачным отступом по краям. У меня также есть границы содержимого изображения (сверху, слева, ширина и высота), но иногда оно выходит за эти границы.

Я пытаюсь обрезать изображение всех прозрачных пикселей, НО, при необходимости, я хочу сохранить минимальные границы.

Например, предположим, что у меня есть 10 изображений размером 1000x1000. На некоторых из них контент позиционируется размером 100х100, а размер контента составляет 100х100. Иногда контент имеет рамку или другой переполненный контент, поэтому контент начинается раньше. На некоторых изображениях границы будут

{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.info("row: "+ row + " col: " + column);
      
      // ROWS
      // get first top row with a pixel in it
      if (firstRowFound==false && alpha!=0){
         firstRowWithContent = row;
         firstRowFound = true;
         console.info("First non transparent pixel found at row: ");
         console.info("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.info("First non transparent pixel found at column: ");
         console.info("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.info(bounds);
   // todo: get image at bounds
  
   return bounds;
}

var results = trimImage(image);

должна ли функция автоматически обнаруживать прозрачные части? и сохранить только минимальный прямоугольник видимого содержимого?

IT goldman 25.04.2024 21:46

@ITgoldman да. правильно. Думаю, я понял это. я сейчас тестирую

1.21 gigawatts 26.04.2024 01:01
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
Поведение ключевого слова "this" в стрелочной функции в сравнении с нормальной функцией
В JavaScript одним из самых запутанных понятий является поведение ключевого слова "this" в стрелочной и обычной функциях.
Концепция локализации и ее применение в приложениях React ⚡️
Концепция локализации и ее применение в приложениях React ⚡️
Локализация - это процесс адаптации приложения к различным языкам и культурным требованиям. Это позволяет пользователям получить опыт, соответствующий...
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
Улучшение производительности загрузки с помощью Google Tag Manager и атрибута Defer
В настоящее время производительность загрузки веб-сайта имеет решающее значение не только для удобства пользователей, но и для ранжирования в...
Безумие обратных вызовов в javascript [JS]
Безумие обратных вызовов в javascript [JS]
Здравствуйте! Юный падаван 🚀. Присоединяйся ко мне, чтобы разобраться в одной из самых запутанных концепций, когда вы начинаете изучать мир...
Система управления парковками с использованием HTML, CSS и JavaScript
Система управления парковками с использованием HTML, CSS и JavaScript
Веб-сайт по управлению парковками был создан с использованием HTML, CSS и JavaScript. Это простой сайт, ничего вычурного. Основная цель -...
JavaScript Вопросы с множественным выбором и ответы
JavaScript Вопросы с множественным выбором и ответы
Если вы ищете платформу, которая предоставляет вам бесплатный тест JavaScript MCQ (Multiple Choice Questions With Answers) для оценки ваших знаний,...
1
3
125
1
Перейти к ответу Данный вопрос помечен как решенный

Ответы 1

Ответ принят как подходящий

Пример изображения

Используем следующее изображение размером 320x320 пикселей с прозрачным заполнением:

Источник: я (можно использовать в любых целях)

Пример кода

В этом примере вы увидите разницу между «до» и «после», каждое изображение окружено красной рамкой, чтобы показать, где заканчивается прозрачное поле изображения.

Функция 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.

В обычных обстоятельствах вы сможете использовать загруженный большой двоичный объект или импортированное изображение, размещенное в том же источнике. Если нет, обязательно ознакомьтесь с методом перекрестного происхождения.

Другие вопросы по теме