40

Dribbble.com が画像内の優勢な色を検出する機能を複製しようとしています。下の画像では、左側の画像の 8 つの主要な色を示す Dribbble.com のスクリーンショットを見ることができます。画像の実際のページは次のとおりですhttp://dribbble.com/shots/528033-Fresh-Easy?list=following

PHP でこれを実行できるようにする必要があります。必要な色を取得したら、それらをデータベースに保存するので、ページの読み込みごとに処理を実行する必要はありません。

画像からこれらの色を取得する方法について調査した結果、画像をピクセルごとに調べて、最も多く発生する色を保存するだけだと言う人もいました。それ以上のことがあり、最も頻繁に存在する色を取得しても、望ましい効果が得られないと言う人もいます. 彼らは、画像/色を量子化する必要があると言います(私はこの時点で迷っています)。

下の Dribble ショットの下の画像は、同じことを行う Javascript ライブラリです。そのページは、ここで見ることができます

そのページのソースを表示すると、という名前の Javascript ファイルがquantize.jsあり、結果が非​​常に良好であることがわかります。だから私はそのJavascriptライブラリができることを望んでいますが、PHPとGD / ImageMagickを使っています

ここに画像の説明を入力


PHPで画像の色と数を返すこの関数を見つけましたが、結果は上記のJavascriptバージョンとドリブルの結果とは異なります

/**
 * Returns the colors of the image in an array, ordered in descending order, where the keys are the colors, and the values are the count of the color.
 *
 * @return array
 */
function Get_Color()
{
    if (isset($this->image))
    {
        $PREVIEW_WIDTH    = 150;  //WE HAVE TO RESIZE THE IMAGE, BECAUSE WE ONLY NEED THE MOST SIGNIFICANT COLORS.
        $PREVIEW_HEIGHT   = 150;
        $size = GetImageSize($this->image);
        $scale=1;
        if ($size[0]>0)
        $scale = min($PREVIEW_WIDTH/$size[0], $PREVIEW_HEIGHT/$size[1]);
        if ($scale < 1)
        {
            $width = floor($scale*$size[0]);
            $height = floor($scale*$size[1]);
        }
        else
        {
            $width = $size[0];
            $height = $size[1];
        }
        $image_resized = imagecreatetruecolor($width, $height);
        if ($size[2]==1)
        $image_orig=imagecreatefromgif($this->image);
        if ($size[2]==2)
        $image_orig=imagecreatefromjpeg($this->image);
        if ($size[2]==3)
        $image_orig=imagecreatefrompng($this->image);
        imagecopyresampled($image_resized, $image_orig, 0, 0, 0, 0, $width, $height, $size[0], $size[1]); //WE NEED NEAREST NEIGHBOR RESIZING, BECAUSE IT DOESN'T ALTER THE COLORS
        $im = $image_resized;
        $imgWidth = imagesx($im);
        $imgHeight = imagesy($im);
        for ($y=0; $y < $imgHeight; $y++)
        {
            for ($x=0; $x < $imgWidth; $x++)
            {
                $index = imagecolorat($im,$x,$y);
                $Colors = imagecolorsforindex($im,$index);
                $Colors['red']=intval((($Colors['red'])+15)/32)*32;    //ROUND THE COLORS, TO REDUCE THE NUMBER OF COLORS, SO THE WON'T BE ANY NEARLY DUPLICATE COLORS!
                $Colors['green']=intval((($Colors['green'])+15)/32)*32;
                $Colors['blue']=intval((($Colors['blue'])+15)/32)*32;
                if ($Colors['red']>=256)
                $Colors['red']=240;
                if ($Colors['green']>=256)
                $Colors['green']=240;
                if ($Colors['blue']>=256)
                $Colors['blue']=240;
                $hexarray[]=substr("0".dechex($Colors['red']),-2).substr("0".dechex($Colors['green']),-2).substr("0".dechex($Colors['blue']),-2);
            }
        }
        $hexarray=array_count_values($hexarray);
        natsort($hexarray);
        $hexarray=array_reverse($hexarray,true);
        return $hexarray;

    }
    else die("You must enter a filename! (\$image parameter)");
}

それで、PHPでそのようなタスクを行う方法を誰かが知っているかどうか尋ねていますか? おそらく、あなたが知っている何かがすでに存在するか、これを行うための一歩を踏み出すためのヒントをいただければ幸いです

4

7 に答える 7

48

これがまさにPHPで探しているものです: https://github.com/thephpleague/color-extractor

例 :

use League\ColorExtractor\Palette;

$palette = Palette::fromFilename('some/image.png');

$topEightColors = $palette->getMostUsedColors(8);
于 2014-02-18T07:30:44.213 に答える
18

これは、画像の主な色を取得するための私の簡単な方法です

$image=imagecreatefromjpeg('image.jpg');
$thumb=imagecreatetruecolor(1,1);
imagecopyresampled($thumb,$image,0,0,0,0,1,1,imagesx($image),imagesy($image));
$mainColor=strtoupper(dechex(imagecolorat($thumb,0,0)));
echo $mainColor;
于 2016-03-30T23:44:17.467 に答える
8

画像を縮小する必要があり、画像の主要な色が得られます。パレットに 4 色が必要な場合は、約 に縮小し8x8、6 色を約に 縮小12x8します...

imagecopyresized縮小された画像の場合、すべてのピクセルをチェックして配列に保存しますimagecolorat($image,px,py)

これを試してみてください

<?php

// EXAMPLE PICTURE
$url='https://www.nordoff-robbins.org.uk/sites/default/files/google.jpg';

//var_dump(getColorPallet($url));

echoColors(getColorPallet($url));


function echoColors($pallet){ // OUTPUT COLORSBAR
    foreach ($pallet as $key=>$val)
        echo '<div style="display:inline-block;width:50px;height:20px;background:#'.$val.'"> </div>';
}

function getColorPallet($imageURL, $palletSize=[16,8]){ // GET PALLET FROM IMAGE PLAY WITH INPUT PALLET SIZE
    // SIMPLE CHECK INPUT VALUES
    if(!$imageURL) return false;

    // IN THIS EXEMPLE WE CREATE PALLET FROM JPG IMAGE
    $img = imagecreatefromjpeg($imageURL);

    // SCALE DOWN IMAGE
    $imgSizes=getimagesize($imageURL);

    $resizedImg=imagecreatetruecolor($palletSize[0],$palletSize[1]);

    imagecopyresized($resizedImg, $img , 0, 0 , 0, 0, $palletSize[0], $palletSize[1], $imgSizes[0], $imgSizes[1]);

    imagedestroy($img);

    //CHECK IMAGE
    /*header("Content-type: image/png");
    imagepng($resizedImg);
    die();*/

    //GET COLORS IN ARRAY
    $colors=[];

    for($i=0;$i<$palletSize[1];$i++)
        for($j=0;$j<$palletSize[0];$j++)
            $colors[]=dechex(imagecolorat($resizedImg,$j,$i));

    imagedestroy($resizedImg);

    //REMOVE DUPLICATES
    $colors= array_unique($colors);

    return $colors;

}
?>

私にぴったりです。

于 2016-12-08T16:38:02.830 に答える
3

リンク先のページには GitHub のソース コードへのリンクがあるため、彼らがどのように行っているかを正確に知りたい場合は、PHP でソースを複製できます。

彼らのやり方とあなたのやり方の大きな違いは、クラスタリングを使って色を見つけていることです。保存時に色を丸める代わりに、すべての生の色を配列に保存しています。次に、クラスター内のポイントとクラスター内の色の数の比率が最も高いクラスターが見つかるまで、この配列をループします。この中心点は、最も一般的な色です。パレットは次に高いクラスターのセットによって定義され、クラスターのほぼ完全なオーバーラップを防ぐためのいくつかのロジックがあります。

于 2012-04-24T03:16:03.023 に答える