<?php //變量定義,畫橢圓弧時的角度大小 define("ANGLELENGTH", 10); /** * 繪製圖片 * @param $title 3D圖的標題 * @param $dataArr 顯示的數據數組 * @param $labelArr 對應數據的標籤分類數組 * @param $colorArr 對應繪圖顏色的數組 * @param $a 畫布的基準寬度 * @param $b 畫布的基準高度 * @param $v 3D柱的高度 * @param $font 字體大小 * @return 繪製成功的圖片訪問路徑 */ function drawPieImg($title, $dataArr, $labelArr, $colorArr, $a = 250, $b = 120, $v = 15, $font = 10) { //餅狀圖開始的位置 x y $ox = 5 + $a; $oy = 5 + $b; // 返回指定自體一個字符寬度的像素值 $fw = imagefontwidth($font); $fh = imagefontheight($font); $n = count($dataArr); //計算數組長度 $w = 10 + $a * 2; $h = 10 + $b * 2 + $v + ($fh + 2) * $n; //建立畫板 $img = imagecreate($w, $h); //轉RGB爲索引色 各類傳入的值 for ($i = 0; $i < $n; $i++) { $colorArr[$i] = drawIndexColor($img, $colorArr[$i]); //爲圖像$img分配顏色 } //背景的顏色 $clrbk = imagecolorallocate($img, 0xff, 0xff, 0xff); $clrt = imagecolorallocate($img, 0x00, 0x00, 0x00); //填充背景色 imagefill($img, 0, 0, $clrbk); //求和 計算出總的數值 $tot = 0; for ($i = 0; $i < $n; $i++) { $tot += $dataArr[$i]; } //每一個分類的起始角度大小 $sd = 0; //每一個分類所佔據的角度大小 $ed = 0; $ly = 10 + $b * 2 + $v; for ($i = 0; $i < $n; $i++) { //上次的結束角度賦值給本次的起始的角度 $sd = $ed; $ed += $dataArr[$i] / $tot * 360; //畫3d扇面 draw3DSector($img, $ox, $oy + 20, $a, $b, $v, $sd, $ed, $colorArr[$i]); //畫標籤 imagefilledrectangle($img, 5, $ly, 5 + $fw, $ly + $fh, $colorArr[$i]); imagerectangle($img, 5, $ly, 5 + $fw, $ly + $fh, $clrt); //中文轉碼 $str = $labelArr[$i]; imagettftext($img, $font, 0, 5 + 2 * $fw, $ly + 13, $clrt, "/usr/local/apache243/htdocs/GDdemo/simsun.ttc", $str . ":" . $dataArr[$i] . "(" . (round(10000 * ($dataArr[$i] / $tot)) / 100) . "%)"); $ly += $fh + 2; } //繪製圖片標題 imagettftext($img, 15, 0, 5, 15, $clrt, "/usr/local/apache243/htdocs/GDdemo/simsun.ttc", $title); //輸出圖形 //header("Content-type: image/png"); //輸出生成的圖片 $imgFileName = "./" . time() . ".png"; imagepng($img, $imgFileName); return $imgFileName; } /** * 繪製3d扇面 * @param imagecreate $img 畫布 * @param int $ox 餅狀圖距離x邊框的距離 * @param int $oy 餅狀圖距離x邊框的距離 * @param int $a 畫布的基準寬度 * @param int $b 畫布的基準高度 * @param int $v 3d柱面高度 * @param int $sd 起始的角度 * @param int $ed 結束的角度 * @param $clr 顏色 */ function draw3DSector($img, $ox, $oy, $a, $b, $v, $sd, $ed, $clr) { drawSector($img, $ox, $oy, $a, $b, $sd, $ed, $clr); //起始的角度小於180度 if ($sd < 180) { list($red, $green, $blue) = drawDarkColor($img, $clr); //爲圖像分配顏色 這是比較暗的顏色 $clr = imagecolorallocate($img, $red, $green, $blue); //若是結束的角度大於180度 if ($ed > 180) { $ed = 180; } //求出角度對應橢圓的點 list($sx, $sy) = getExy($a, $b, $sd); $sx += $ox; $sy += $oy; list($ex, $ey) = getExy($a, $b, $ed); $ex += $ox; $ey += $oy; //繪製豎着的直線 imageline($img, $sx, $sy, $sx, $sy + $v, $clr); //繪製結束點的豎着的直線 imageline($img, $ex, $ey, $ex, $ey + $v, $clr); // drawArc($img, $ox, $oy + $v, $a, $b, $sd, $ed, $clr); list($sx, $sy) = getExy($a, $b, ($sd + $ed) / 2); $sy += $oy + $v / 2; $sx += $ox; imagefill($img, $sx, $sy, $clr); } } /** * 繪製橢圓弧 */ function drawArc($img, $ox, $oy, $a, $b, $sd, $ed, $clr) { $n = ANGLELENGTH > 0 ? ceil(($ed - $sd) / ANGLELENGTH) : -1; $d = $sd; list($x0, $y0) = getExy($a, $b, $d); for ($i = 0; $i < $n; $i++) { $d = ($d + ANGLELENGTH) > $ed ? $ed : ($d + ANGLELENGTH); // list() 函數用數組中的元素爲一組變量賦值 list($x, $y) = getExy($a, $b, $d); imageline($img, $x0 + $ox, $y0 + $oy, $x + $ox, $y + $oy, $clr); $x0 = $x; $y0 = $y; } } /** * 繪製扇面 */ function drawSector($img, $ox, $oy, $a, $b, $sd, $ed, $clr) { //ceil 返回大於這個輸得最小整數 $n = ANGLELENGTH > 0 ? ceil(($ed - $sd) / ANGLELENGTH) : -1; $d = $sd; //list() 函數用數組中的元素爲一組變量賦值 list($x0, $y0) = getExy($a, $b, $d); imageline($img, $x0 + $ox, $y0 + $oy, $ox, $oy, $clr); for ($i = 0; $i < $n; $i++) { $d = ($d + ANGLELENGTH) > $ed ? $ed : ($d + ANGLELENGTH); list($x, $y) = getExy($a, $b, $d); imageline($img, $x0 + $ox, $y0 + $oy, $x + $ox, $y + $oy, $clr); $x0 = $x; $y0 = $y; } imageline($img, $x0 + $ox, $y0 + $oy, $ox, $oy, $clr); list($x, $y) = getExy($a / 2, $b / 2, ($d + $sd) / 2); imagefill($img, $x + $ox, $y + $oy, $clr); } /** * 根據$clr顏色獲取對應的柱的陰影色 * @param $img 圖像 * @param $clr 顏色 * @return rgb顏色數組 */ function drawDarkColor($img, $clr) { //Array //( // [red] => 226 // [green] => 222 // [blue] => 252 // [alpha] => 0 //) $rgb = imagecolorsforindex($img, $clr); return array($rgb["red"] / 2, $rgb["green"] / 2, $rgb["blue"] / 2); } /** * 求角度$d對應的橢圓上的點座標 * * @param $a 橫座標 * @param $b 縱座標 * @param $d 角度 * @return 對應橢圓點座標 */ function getExy($a, $b, $d) { //將角度轉換爲弧度 2π=360度 $d = deg2rad($d); //這個地方沒有看明白 return array(round($a * cos($d)), round($b * sin($d))); } /** * 爲圖像分配RGB索引色 */ function drawIndexColor($img, $clr) { $red = ($clr >> 16) & 0xff; $green = ($clr >> 8) & 0xff; $blue = ($clr) & 0xff; //imagecolorallocate — 爲一幅圖像分配顏色 return imagecolorallocate($img, $red, $green, $blue); } //測試示例 $title = "動物園動物種類分佈狀況"; $dataArr = array(20, 10, 20, 20, 10, 20, 30, 10, 11, 32, 12); //測試數據數組 $labelArr = array("大象", "長頸鹿", "鱷魚", "鴕鳥", "老虎", "獅子", "猴子", "斑馬", "野驢", "羚羊", "猩猩"); //標籤 $colorArr = array(0x99ff00, 0x312520, 0x801dae, 0x25f8cb, 0x808080, 0xa29b7c, 0xbce672, 0x44cef6, 0xc2ccd0, 0x8d4bbb, 0x2e4e7e, 0x50616d, 0x845a33, 0x622a1d, 0xff6666, 0xe4c6d0, 0x8c4356, 0xa29b7c, 0x30dff3, 0xfffbf0, 0xedd1d8, 0x801dae, 0xb0a4e3, 0x0099ff, 0x003371, 0x60281e, 0x8c4356, 0x4c221b, 0xff99ff, 0xffff99, 0x0aa344, 0x99ffff, 0xc0ebd7, 0x1bd1a5, 0xc32136, 0xff3333, 0x009999, 0xfff143, 0xf9906f, 0xbe002f, 0x8d4bbb, 0x815476); //對應顏色數組 $result = drawPieImg($title, $dataArr, $labelArr, $colorArr); echo "<img src=" . $result . " mce_src=" . $result . ">"; ?>