Определите "общий средний" цвет изображения


У меня есть изображение в формате jpg.

Мне нужно знать "общий средний" цвет изображения. На первый взгляд там можно использовать гистограмму изображения (канал RGB).

На работе я использую в основном JavaScript и PHP (немного Python), поэтому приветствовал решение на этих языках. Возможно, есть библиотека для работы с изображениями, которые решают аналогичные проблемы.

Мне не нужно динамически определять цвет изображения. Мне нужно всего один раз пройти через весь массив изображения и определите цвет каждого отдельно (эту информацию я запомню для дальнейшего использования).

Author: Samuel Liew, 2010-08-12

6 answers

Вы можете использовать PHP, чтобы получить массив цветовой палитры следующим образом:

<?php 
function colorPalette($imageFile, $numColors, $granularity = 5) 
{ 
   $granularity = max(1, abs((int)$granularity)); 
   $colors = array(); 
   $size = @getimagesize($imageFile); 
   if($size === false) 
   { 
      user_error("Unable to get image size data"); 
      return false; 
   } 
   $img = @imagecreatefromjpeg($imageFile);
   // Andres mentioned in the comments the above line only loads jpegs, 
   // and suggests that to load any file type you can use this:
   // $img = @imagecreatefromstring(file_get_contents($imageFile)); 

   if(!$img) 
   { 
      user_error("Unable to open image file"); 
      return false; 
   } 
   for($x = 0; $x < $size[0]; $x += $granularity) 
   { 
      for($y = 0; $y < $size[1]; $y += $granularity) 
      { 
         $thisColor = imagecolorat($img, $x, $y); 
         $rgb = imagecolorsforindex($img, $thisColor); 
         $red = round(round(($rgb['red'] / 0x33)) * 0x33); 
         $green = round(round(($rgb['green'] / 0x33)) * 0x33); 
         $blue = round(round(($rgb['blue'] / 0x33)) * 0x33); 
         $thisRGB = sprintf('%02X%02X%02X', $red, $green, $blue); 
         if(array_key_exists($thisRGB, $colors)) 
         { 
            $colors[$thisRGB]++; 
         } 
         else 
         { 
            $colors[$thisRGB] = 1; 
         } 
      } 
   } 
   arsort($colors); 
   return array_slice(array_keys($colors), 0, $numColors); 
} 
// sample usage: 
$palette = colorPalette('rmnp8.jpg', 10, 4); 
echo "<table>\n"; 
foreach($palette as $color) 
{ 
   echo "<tr><td style='background-color:#$color;width:2em;'>&nbsp;</td><td>#$color</td></tr>\n"; 
} 
echo "</table>\n";

, который дает вам массив, значения которого выше для того, как часто использовался этот цвет.

РЕДАКТИРОВАТЬ Комментатор спросил, как использовать это для всех файлов в каталоге, вот оно:

    if ($handle = opendir('./path/to/images')) {

        while (false !== ($file = readdir($handle))) {
           $palette = colorPalette($file, 10, 4);
           echo "<table>\n"; 
           foreach($palette as $color) { 
               echo "<tr><td style='background-color:#$color;width:2em;'>&nbsp;</td><td>#$color</td></tr>\n"; 
           } 
           echo "</table>\n";
        }
        closedir($handle);
    }

Возможно, вам не захочется делать это со слишком большим количеством файлов, но это ваш сервер.

В качестве альтернативы, если вы предпочитаете использовать Javascript Цвет Локеша-библиотека делает точно то, что вы ищете.

 67
Author: JKirchartz, 2015-06-09 14:34:19

Объединение ответа Кирхарца и Александра Хугестранда:

 function getAverage($sourceURL){

    $image = imagecreatefromjpeg($sourceURL);
    $scaled = imagescale($image, 1, 1, IMG_BICUBIC); 
    $index = imagecolorat($scaled, 0, 0);
    $rgb = imagecolorsforindex($scaled, $index); 
    $red = round(round(($rgb['red'] / 0x33)) * 0x33); 
    $green = round(round(($rgb['green'] / 0x33)) * 0x33); 
    $blue = round(round(($rgb['blue'] / 0x33)) * 0x33); 
    return sprintf('#%02X%02X%02X', $red, $green, $blue); 
 }

Проверено и проверено, возвращает шестнадцатеричную строку.

 5
Author: Johnny Rockex, 2017-05-16 09:11:33

Более коротким решением для изображения истинного цвета было бы уменьшить его до размера 1x1 пикселя и выбрать цвет в этом пикселе:

$масштаб = масштаб изображения ($img, 1, 1, IMG_BICUBIC); $meancolor = цвет изображения ($img, 0, 0);

...но я сам этого не проверял.

 2
Author: Alexander Hugestrand, 2013-07-12 12:38:20

Начните с PIL. http://www.pythonware.com/products/pil/

Откройте объект изображения. Используйте метод getdata, чтобы получить все пиксели. Усредните значения, которые вы получите обратно.

Что-то вроде этого.

Определение цвета изображения с помощью python

 1
Author: S.Lott, 2017-05-23 11:47:21
$img = glob('img/*');
foreach ($img as $key => $value) {
    $info = getimagesize($value);
    $mime = $info['mime'];
    switch ($mime) {
        case 'image/jpeg':
            $image_create_func = 'imagecreatefromjpeg';
            break;
        case 'image/png':
            $image_create_func = 'imagecreatefrompng';
            break;
        case 'image/gif':
            $image_create_func = 'imagecreatefromgif';
            break;
    }
    $avg = $image_create_func($value);
    list($width, $height) = getimagesize($value);
    $tmp = imagecreatetruecolor(1, 1);
    imagecopyresampled($tmp, $avg, 0, 0, 0, 0, 1, 1, $width, $height);
    $rgb = imagecolorat($tmp, 0, 0);
    $r = ($rgb >> 16) & 0xFF;
    $g = ($rgb >> 8) & 0xFF;
    $b = $rgb & 0xFF;
    echo '<div style="text-align:center; vertical-align: top; display:inline-block; width:100px; height:150px; margin:5px; padding:5px; background-color:rgb('.$r.','.$g.','.$b.');">';
    echo '<img style="width:auto; max-height:100%; max-width: 100%; vertical-align:middle; height:auto; margin-bottom:5px;" src="'.$value.'">';
    echo '</div>';

Вы можете получить значение среднего цвета с помощью $r, $g и $b . Повторная выборка изображения намного лучше, чем просто масштабирование!

 1
Author: antoine demacon, 2018-01-02 21:10:39

Я создал пакет composer, который предоставляет библиотеку для выбора среднего цвета из данного изображения по его пути.

Вы можете установить его, выполнив следующую команду в каталоге вашего проекта:

composer require tooleks/php-avg-color-picker

Пример использования:

<?php

use Tooleks\Php\AvgColorPicker\Gd\AvgColorPicker;

$imageAvgHexColor = (new AvgColorPicker)->getImageAvgHexByPath('/absolute/path/to/the/image.(jpg|jpeg|png|gif)');

// The `$imageAvgHexColor` variable contains the average color of the given image in HEX format (#fffff).

Смотрите документацию .

 0
Author: tooleks, 2017-05-24 17:27:31