私はChart.jsを使用して、PHPページで円グラフを描いていますが、各スライスの値を表示するツールチップを発見しました。
[ここに画像の説明を入力]1。
しかし、その値を以下の画像のように表示したいのです。 [ここに画像の説明を入力してください]2]。
chart.jsでのやり方がわかりません。
よろしくお願いします。
私のJavascriptのコードです:
function drawPie(canvasId,data,legend){
var ctx = $("#pie-canvas-" + canvasId).get(0).getContext("2d");
var piedata = [];
$.each(data,function(i,val){
piedata.push({value:val.count,color:val.color,label:val.status});
});
var options =
{
tooltipTemplate: "<%= Math.round(circumference / 6.283 * 100) %>%",
}
var pie = new Chart(ctx).Pie(piedata,options);
if(legend)document.getElementById("legend").innerHTML = pie.generateLegend();
}
phpのコードです:
printf('<table><tr>');
echo '<td style="text-align: right;"><canvas id="pie-canvas-'
. $canvasId
. '" width="256" height="256" ></canvas></td><td style="text-align: left;width:360px;height:auto" id="legend" class="chart-legend"></td></tr></table>';
echo '<script type="text/javascript">drawPie('
. $canvasId
. ', '
. $data3
.', '
. $legend
. ');</script>';
Chart.js 2.0**以降では、Chartオブジェクトのデータが変更されています。Chart.js 2.0以降をお使いの方のために、HTML5 CanvasのfillText()
メソッドを使って、パイスライスの中にデータの値を表示する例を以下に示します。このコードはドーナツチャートでも使えます。チャートの作成時に「type: 'pie'」と「type: 'doughnut'
」の違いだけです。
Script:
ジャバスクリプト
var data = {
datasets: [{
data: [
11,
16,
7,
3,
14
],
backgroundColor: [
"#FF6384",
"#4BC0C0",
"#FFCE56",
"#E7E9ED",
"#36A2EB"
],
label: 'My dataset' // for legend
}],
labels: [
"Red",
"Green",
"Yellow",
"Grey",
"Blue"
]
};
var pieOptions = {
events: false,
animation: {
duration: 500,
easing: "easeOutQuart",
onComplete: function () {
var ctx = this.chart.ctx;
ctx.font = Chart.helpers.fontString(Chart.defaults.global.defaultFontFamily, 'normal', Chart.defaults.global.defaultFontFamily);
ctx.textAlign = 'center';
ctx.textBaseline = 'bottom';
this.data.datasets.forEach(function (dataset) {
for (var i = 0; i < dataset.data.length; i++) {
var model = dataset._meta[Object.keys(dataset._meta)[0]].data[i]._model,
total = dataset._meta[Object.keys(dataset._meta)[0]].total,
mid_radius = model.innerRadius + (model.outerRadius - model.innerRadius)/2,
start_angle = model.startAngle,
end_angle = model.endAngle,
mid_angle = start_angle + (end_angle - start_angle)/2;
var x = mid_radius * Math.cos(mid_angle);
var y = mid_radius * Math.sin(mid_angle);
ctx.fillStyle = '#fff';
if (i == 3){ // Darker text color for lighter background
ctx.fillStyle = '#444';
}
var percent = String(Math.round(dataset.data[i]/total*100)) + "%";
//Don't Display If Legend is hide or value is 0
if(dataset.data[i] != 0 && dataset._meta[0].data[i].hidden != true) {
ctx.fillText(dataset.data[i], model.x + x, model.y + y);
// Display percent in another line, line break doesn't work for fillText
ctx.fillText(percent, model.x + x, model.y + y + 15);
}
}
});
}
}
};
var pieChartCanvas = $("#pieChart");
var pieChart = new Chart(pieChartCanvas, {
type: 'pie', // or doughnut
data: data,
options: pieOptions
});
HTML
<canvas id="pieChart" width=200 height=200></canvas>
[jsFiddle][1]です。
私の知る限りでは、Chart.JSには円グラフにテキストを描くのに役立つ機能はないと思っています。しかし、だからといって、ネイティブのJavaScriptでそれを行うことができないとは限りません。以下は、円グラフの各セグメントにテキストを描画するためのコードです:
function drawSegmentValues()
{
for(var i=0; i<myPieChart.segments.length; i++)
{
// Default properties for text (size is scaled)
ctx.fillStyle="white";
var textSize = canvas.width/10;
ctx.font= textSize+"px Verdana";
// Get needed variables
var value = myPieChart.segments[i].value;
var startAngle = myPieChart.segments[i].startAngle;
var endAngle = myPieChart.segments[i].endAngle;
var middleAngle = startAngle + ((endAngle - startAngle)/2);
// Compute text location
var posX = (radius/2) * Math.cos(middleAngle) + midX;
var posY = (radius/2) * Math.sin(middleAngle) + midY;
// Text offside to middle of text
var w_offset = ctx.measureText(value).width/2;
var h_offset = textSize/4;
ctx.fillText(value, posX - w_offset, posY + h_offset);
}
}
円グラフは PieChart.segments
に格納されたセグメントの配列を持っています。これらのセグメントの startAngle
と endAngle
を見て、テキストが middleAngle
となる中間の角度を決定することができます。そして、その方向に Radius/2
だけ移動して、ラジアン単位でチャートの中央点に位置することになります。
上の例では、他にもいくつかのクリーンアップ処理が行われています。fillText()
で描かれたテキストの位置が右上にあるため、それを補正するためにオフセット値を取得する必要があります。最後に textSize
はチャート自体のサイズに基づいて決定され、チャートが大きくなればなるほどテキストも大きくなります。
[フィドル例](
)少し手を加えると、データセットの離散的な数値をグラフのパーセンタイルの数値に変更することができます。これを行うには、データセットに含まれるアイテムの「値」の合計を求め、これを「totalValue」と呼ぶ。そして、各セグメントについて、次のようにしてパーセントを求めることができる:
Math.round(myPieChart.segments[i].value/totalValue*100)+'%';
この myPieChart.segments[i].value/totalValue
というセクションは、そのセグメントがチャートに占める割合を計算するものです。例えば、現在のセグメントの値が 50
で、totalValue が 200
であったとします。この場合、セグメントを占める割合は次のようになります:50/200 => 0.25となります。あとは、これを見栄えよくするためです。0.25*100 => 25
となり、最後に%
を追加します。整数パーセントのタイルの場合、最も近い整数に丸めるのですが、精度に問題が生じることがあります。より正確さが必要な場合は、.toFixed(n)
を使って小数点以下の桁数を節約することができます。例えば、次のようにすると、必要なときに小数点以下1桁を保存することができます:
var value = myPieChart.segments[i].value/totalValue*100;
if(Math.round(value) !== value)
value = (myPieChart.segments[i].value/totalValue*100).toFixed(1);
value = value + '%';
[小数を使ったパーセンタイルの例](
2/)[整数によるパーセンタイルの例](
1/)Hung Tranの回答は完璧です。例えば、5つの要素があり、そのうちの2つが0で、残りの要素が値を持っているとすると、上記の解決策は0と0%を表示することになります。0 に等しくないことをチェックすることでフィルタリングするのがよいでしょう!
var val = dataset.data[i]; var percent = String(Math.round(val/total*100)) + "%"; >; if(val != 0){。 ctx.fillText(dataset.data[i], model.x + x, model.y + y); // パーセントを別行動で表示、改行はfillTextに効かない。 ctx.fillText(percent, model.x + x, model.y + y + 15); }です。
以下、コードを更新しました:
var data = {
datasets: [{
data: [
11,
16,
7,
3,
14
],
backgroundColor: [
"#FF6384",
"#4BC0C0",
"#FFCE56",
"#E7E9ED",
"#36A2EB"
],
label: 'My dataset' // for legend
}],
labels: [
"Red",
"Green",
"Yellow",
"Grey",
"Blue"
]
};
var pieOptions = {
events: false,
animation: {
duration: 500,
easing: "easeOutQuart",
onComplete: function () {
var ctx = this.chart.ctx;
ctx.font = Chart.helpers.fontString(Chart.defaults.global.defaultFontFamily, 'normal', Chart.defaults.global.defaultFontFamily);
ctx.textAlign = 'center';
ctx.textBaseline = 'bottom';
this.data.datasets.forEach(function (dataset) {
for (var i = 0; i < dataset.data.length; i++) {
var model = dataset._meta[Object.keys(dataset._meta)[0]].data[i]._model,
total = dataset._meta[Object.keys(dataset._meta)[0]].total,
mid_radius = model.innerRadius + (model.outerRadius - model.innerRadius)/2,
start_angle = model.startAngle,
end_angle = model.endAngle,
mid_angle = start_angle + (end_angle - start_angle)/2;
var x = mid_radius * Math.cos(mid_angle);
var y = mid_radius * Math.sin(mid_angle);
ctx.fillStyle = '#fff';
if (i == 3){ // Darker text color for lighter background
ctx.fillStyle = '#444';
}
var val = dataset.data[i];
var percent = String(Math.round(val/total*100)) + "%";
if(val != 0) {
ctx.fillText(dataset.data[i], model.x + x, model.y + y);
// Display percent in another line, line break doesn't work for fillText
ctx.fillText(percent, model.x + x, model.y + y + 15);
}
}
});
}
}
};
var pieChartCanvas = $("#pieChart");
var pieChart = new Chart(pieChartCanvas, {
type: 'pie', // or doughnut
data: data,
options: pieOptions
});