53
52

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 5 years have passed since last update.

CYBIRDエンジニアAdvent Calendar 2015

Day 18

Google Chartsを使ってみた

Last updated at Posted at 2015-12-17

##初めに
 こんにちは。CYBIRDエンジニア Advent Calendar 2015 18日目を担当します@cy-hiroshi-chibaです。今年中途入社で、現在はデータ分析の仕事をしています。

 昨日は@doidoidonさんのCocos Studio2を触ってみたでした。アプリ作成にも興味があるので今すぐにでも試してみたくなりました。年末年始で触ってみます!!

 今日は、少し前にグラフライブラリの検討していたのですが、その中のGoogle Chartsを紹介したいと思います。

##Google Charts
 Google Chartsは様々なグラフが簡単に描け、デザイン・イベント処理などのカスタマイズが容易なライブラリです。同様のグラフライブラリは現在多く存在してますが、HighChartsamChartsのように商用利用不可のライブラリとは違い、商用利用でも無料で利用することができます。
##とりあえず使ってみる
まずヘッダにJavascriptのリンクを張ります。

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Google Charts Test</title>
<script type="text/javascript" src="https://www.google.com/jsapi"></script>
</head>
<body>

</body>
</html>

次にグラフを描画するためのHTMLタグを記述します。

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Google Charts Test</title>
<script type="text/javascript" src="https://www.google.com/jsapi"></script>
</head>
<body>
<div id="chart" style="width: 100%; height: 100%; border: 1px solid #aaa;"></div>
</body>
</html>

こんな感じです。

では実際にグラフを描画する部分を記述していきます。
使用するデータは、スマホアプリLifelogで計測した、12月8日~12月14日まで歩数、乗車時間、消費カロリーデータを使用します。

歩数 乗車時間 消費カロリー
12月 8日 7,432歩 80分 2,199kcal
12月 9日 5,474歩 54分 2,069kcal
12月10日 5,751歩 56分 2,075kcal
12月11日 6,037歩 109分 2,161kcal
12月12日 7,402歩 72分 2,359kcal
12月13日 532歩 16分 2,023kcal
12月14日 5,188歩 91分 2,093kcal

作りたいグラフををExcelで表現すると、こんな感じです。
excel_screenshot.png
※Excelで第3軸が作れなかったため、歩数と消費カロリーは主軸で共有しています。

Google Chartsでこのグラフを作成するためにはCombo Chartを使います。コードはこんな感じです。

<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8">
<title>Google Charts Test</title>
<script type="text/javascript" src="https://www.google.com/jsapi"></script>
</head>
<body>
<div id="chart" style="width: 100%; border: 1px solid #aaa;"></div>
<script>
google.load('visualization', '1', { packages : [ 'corechart' ]});
google.setOnLoadCallback(drawChart);

function drawChart(){
     // 表示するデータの設定
	var data = new google.visualization.DataTable();
	data.addColumn('string', '');
	data.addColumn('number', '歩数');
	data.addColumn('number', '乗車時間');
	data.addColumn('number', '消費カロリー');
	
	data.addRows([
	    ["12月8日",  7432,  80, 2199], 
	    ["12月9日",  5474,  54, 2069], 
	    ["12月10日", 5751,  56, 2075], 
	    ["12月11日", 6037, 109, 2161], 
	    ["12月12日", 7402,  72, 2359], 
	    ["12月13日",  532,  16, 2023], 
	    ["12月14日", 5188,  91, 2093]
	]);
	
    // グラフの設定
	var option = {
		title: 'Lifelogでとったデータ',
		width: '100%',
		height: '100%',
		series: [
			{ type: 'bars', targetAxisIndex: 0 },
			{ type: 'line', targetAxisIndex: 1 },
			{ type: 'area', targetAxisIndex: 2 }
		],
		vAxes: [
			{ title: '歩数(歩)' },
			{ title: '乗車時間(分)' }, 
			{ title: '消費カロリー(kcal)' }
		]
	};
	
	var chart = new google.visualization.ComboChart(document.getElementById('chart'));
	chart.draw(data, option);
}
</script>
</body>
</html>

表示するデータの設定部分は、DataTableのインスタンスを生成して、列の設定・データの追加をする方法のほか、CSV形式のような記述や、

var data = google.visualization.arrayToDataTable([
    [     "", "歩数", "乗車時間", "消費カロリー"]
    ["12月8日",  7432,         80,          2199], 
    ["12月9日",  5474,         54,          2069], 
    ["12月10日", 5751,         56,          2075], 
    ["12月11日", 6037,        109,          2161], 
    ["12月12日", 7402,         72,          2359], 
    ["12月13日",  532,         16,          2023], 
    ["12月14日", 5188,         91,          2093]
]);

JSON形式にも対応しています。

var data = new google.visualization.DataTable({
    "cols": [
        {"type": "string", "label": ""},
        {"type": "number", "label": "歩数"},
        {"type": "number", "label": "乗車時間"},
        {"type": "number", "label": "消費カロリー"}
    ],
    "rows": [
        {"c": [{"v": "12月8日"}, {"v": 7432}, {"v":  80}, {"v": 2199}]},
        {"c": [{"v": "12月9日"}, {"v": 5474}, {"v":  54}, {"v": 2069}]},
        {"c": [{"v": "12月10日"},{"v": 5751}, {"v":  56}, {"v": 2075}]},
        {"c": [{"v": "12月11日"},{"v": 6037}, {"v": 109}, {"v": 2161}]},
        {"c": [{"v": "12月12日"},{"v": 7402}, {"v":  72}, {"v": 2359}]},
        {"c": [{"v": "12月13日"},{"v":  532}, {"v":  16}, {"v": 2023}]},
        {"c": [{"v": "12月14日"},{"v": 5188}, {"v":  91}, {"v": 2093}]}
    ]
});

表示してみると...
google_charts1.png
右側の縦軸の表記が、重なってしまいました...

その場合はoption内の設定を変更します。
textPositionは縦軸の表記の表示位置を'none', 'out', 'in'から選択し設定することができます。

vAxes: [
   { title: '歩数(歩)' },
   { title: '乗車時間(分)', textPosition: 'in' }, 
   { title: '消費カロリー(kcal)' }
]

google_charts2.png

少々重なってはいますが、とりあえず分けることができます。

##Tooltipをカスタマイズする
デフォルトで、マウスオーバーすることでTooltipが表示され、対象日の対応する系列の情報が表示されるのですが、

google_charts3.png

これを対象日の歩数、乗車時間、消費カロリーをまとめて見たい場合は、optionに、

focusTarget: 'category'

を追加すると、
google_charts4.png

のようにできます。

また、もっと細かく設定したい場合にはHTMLで記述していきます。
手順はまず、option内にTooltipの内容をHTMLで記述するための設定を有効にします。


var option = {
   ...
   tooltip: {isHtml: true}
}

次に、HTMLを記述するための列を追加します。

var data = new google.visualization.DataTable();
    data.addColumn('string', '');
    data.addColumn({'type': 'string', 'role': 'tooltip', 'p': {'html': true}});
    data.addColumn('number', '歩数');
    ...

そして、実際に表示させる内容をHTMLで記述します。

data.addRows([
      ["12月8日", '<p>12月8日</p><table><tr><th>歩数</th> ...', 7432,  80, 2199], 
      ...
]);

とりあえず自分はこうしてみました。

google_charts5.png

また、Tooltip表示時に挿入される google-visualization-tooltipクラスを使ってスタイルを変更することもできます。

.google-visualization-tooltip {
	opacity: 0.9;  /* 透明度を変えてみる */
}

google_charts6.png

##もっとこうしたい!!
Google Chartsを使ってみて、「もっとこうしたい!!」と思ったことです。少しでも参考になればと思います。
###Windowサイズにグラフのサイズを動的に変える
 Google Chartsはoptionにサイズを100%と設定するとdivなどのブロックのサイズに合わせてグラフを作成しますが、それはページロード時のみで、Windowサイズを変更してもグラフのサイズは変わりません。

google_charts7.png

必要であれば、以下のコードを追加します。

$(window).resize(function(){
    chart.draw(data, option);
});

###HTMLで記述したTooltipの情報を動的に変更したい
 紹介したHTMLを利用したTooltipのカスタマイズでは、DataTableに直接記述しなければいけませんが、大量データを扱う場合、この方法ではHTMLだけでも膨大な情報量になるため、動的に生成できないかと考えました。

そこで、Tooltip表示時に挿入される google-visualization-tooltipクラスを使って以下の方法を試しました。

//
// DataTableに記述したTooltipに関する情報は削除します。
//
var data = new google.visualization.DataTable();
data.addColumn('string', '');
data.addColumn('number', '歩数');
data.addColumn('number', '乗車時間');
data.addColumn('number', '消費カロリー');

data.addRows([ 
     ["12月8日",  7432,  80, 2199], 
     ["12月9日",  5474,  54, 2069], 
     ["12月10日", 5751,  56, 2075], 
     ["12月11日", 6037, 109, 2161], 
     ["12月12日", 7402,  72, 2359], 
     ["12月13日",  532,  16, 2023], 
     ["12月14日", 5188,  91, 2093]
]);

// ・・・(中略)

var appendCustomTooltip = function(d){
     $(".google-visualization-tooltip").html(" ここにTooltipの内容を書く ");
}
google.visualization.events.addListener(chart, 'onmouseover', appendCustomTooltip);

すると、

google_charts8.png
できました。

が、しかし・・・
google_charts9.png
Tooltip上にマウスカーソルを合わせると元に戻ってしまします…

でも、もう少しで出来そうですね… 
解決できたら報告します。

###凡例で系列の表示/非表示を切り替えたい
 HighCharts、amChartsでは、凡例をクリックするとデータの表示・非表示を切り替える機能があります。Google Chartsではその機能はありません。これを実現してみたいのですが、

Show/hide lines in Google Chart and show grayed text in legend

を参考にしてみたところ、できました。
google_charts10.png

##まとめ
 Google Chartsはグラフが簡単に描け、自由度も高いグラフライブラリです。でも、こだわりがありすぎると多少苦労するかもしれません… 皆さんもぜひ利用してみてください!!

##最後に
明日のCYBIRDエンジニア Advent Calendar 2015は、@cy-katsuhiro-miuraさんのです!どんな内容が出てくるのでしょうか?お楽しみに!

53
52
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
53
52

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?