Chart.jsで作る「kintoneのダッシュボード」
※当記事の内容はサポート対象外です
こんにちは。武井です。
ダッシュボードの季節ですね。
目次
kintoneのグラフ解析
kintoneの標準グラフにはHigh Chartsが採用されているようです。
High Chartsの特徴としては、
- 見た目が綺麗
- 使用に便利
- カスタムに融通が利く
と最強らしいのですが、有料です。
一方、今回使用するChart.jsは無料チャートライブラリの中では一強の様相を呈しています。
ただし、Chart.jsはやや融通の利かない部分があります。
ダッシュボード作ってみた
こんなん作ってみました。
詳しくは下記記事に譲りますが、
kintoneの標準機能にないグラフを厳選してChart.jsで実現したところが売りとなっております。
■Chart.jsを使ってダッシュボードを作ってみよう!
https://cybozudev.zendesk.com/hc/ja/articles/214479043
上記記事と参考画像では説明不足だったのですが、
下図のようなインタフェースも導入しているので、非常に使い勝手はよいです。
右下部分が空白地帯でちょっと寂しいのは作業時間の関係上となります(゚゚;)
ダッシュボードをポータルに表示する
さて、kintone公式ではめったなことでDOMを操作してはいけないことになっていますので、
上記記事ではアプリ内にダッシュボードを表示していますが、
こちらでは例によってDOMをいじりまくることでポータルにダッシュボードを表示してしまう魔改造を披露していきたいと思います。
完成イメージ
ぶっちゃけ使いやすさは圧倒的にこちらですね!
コードを一応すべて載せますが、アプリ版との違いは、ほぼほぼ「// URLが変更される場合のチャート書き直し処理」からcreateSlider関数までの部分のみです。
(function() {
"use strict";
// 販売実績アプリID
var SALES_APPID = 70;
// 試用申し込み管理アプリID
var TRIAL_APPID = 71;
var oneTimeFlg = true;
var chart1, chart2, chart3;
var onceFlg = true;
/* 【参考】ドーナツグラフの各項目を常に表示
Chart.pluginService.register({
beforeRender: function (chart) {
if (chart.config.options.showAllTooltips) {
chart.pluginTooltips = [];
chart.config.data.datasets.forEach(function (dataset, i) {
chart.getDatasetMeta(i).data.forEach(function (sector, j) {
chart.pluginTooltips.push(new Chart.Tooltip({
_chart: chart.chart,
_chartInstance: chart,
_data: chart.data,
_options: chart.options.tooltips,
_active: [sector]
}, chart));
});
});
chart.options.tooltips.enabled = false;
}
},
afterDraw: function (chart, easing) {
if (chart.config.options.showAllTooltips) {
if (!chart.allTooltipsOnce) {
if (easing !== 1)
return;
chart.allTooltipsOnce = true;
}
chart.options.tooltips.enabled = true;
Chart.helpers.each(chart.pluginTooltips, function (tooltip) {
tooltip.initialize();
tooltip.update();
tooltip.pivot();
tooltip.transition(easing).draw();
});
chart.options.tooltips.enabled = false;
}
}
});*/
// URLが変更される場合のチャート書き直し処理
window.onhashchange = firstFunc;
// ポータルを最初に読み込んだ時のチャート表示処理
window.onload = firstFunc;
function firstFunc() {
if (onceFlg) {
if (location.href.indexOf("/#/portal") > 0) {
onceFlg = false;
var portalInterval = setInterval(function () {
if ($(".ocean-portal-announcement .gaia-argoui-widget-menu").length > 0) {
clearInterval(portalInterval);
$(".ocean-portal-announcement .gaia-argoui-widget-menu").css("display", "block");
$(".ocean-portal-announcement .gaia-argoui-widget-menu").append('<div id="slider" style="margin-left:30px"></div><br /><p id="p1" style="margin-left:30px;">■売上高(前年同月比)</p><div id="canvas1Div" style="margin-left:30px;"><canvas id="canvas1"></canvas></div><br /><p id="p2" style="margin-left:30px;">■売上高・試用申込数</p><div id="canvas2Div" style="margin-left:30px;"><canvas id="canvas2"></canvas></div><br /><p id="p3" style="margin-left:30px;">■製品別売上高</p><div id="canvas3Div" style="margin-left:30px"><canvas id="canvas3"></canvas></div>');
createCompareGraph();
// お知らせを更新した場合等、URLが変わらずに画面が更新されるパターンでアプリ一覧を表示し直す処理
var rePortalInterval = setInterval(function () {
if ($("#slider").length === 0) {
clearInterval(rePortalInterval);
oneTimeFlg = true;
firstFunc();
}
}, 500);
}
}, 100);
}
}
}
function createSlider() {
$("#slider").dateRangeSlider();
$("#slider").dateRangeSlider(
"option",
{
bounds: {
min: moment().add(-5, 'years').startOf("month").toDate(),
max: moment().startOf("month").toDate()
},
step: {
months: 1
},
range: {
min: {months: 11},
max: {months: 11}
},
formatter: function(val) {
return moment(val).format("YYYY年M月");
}
}
).bind("userValuesChanged", function(e, data) {
// スライダーを動かした場合
createCompareGraph(data.values.min, data.values.max);
});
reset();
oneTimeFlg = false;
}
function reset() {
// スライダーの初期位置
$("#slider").dateRangeSlider("values", moment().add(-1, 'years').startOf("month").toDate(), moment().startOf("month").toDate());
// スライダーの位置を微調整
$("#slider").css("margin-bottom", "20px");
}
function createCompareGraph(min, max) {
//setLoading();
var maxMonth, maxMonth2, minMonth, minMonth2, elevenMonthsBefore, tenMonthsBefore, nineMonthsBefore, eightMonthsBefore, sevenMonthsBefore, sixMonthsBefore, fiveMonthsBefore, fourMonthsBefore, threeMonthsBefore, twoMonthsBefore, oneMonthBefore, thisMonth;
if (max) {
maxMonth = '"' + moment(max).endOf("month").format('YYYY-MM-DD') + '"';
maxMonth2 = moment(max).add(-12, "months").endOf("month").format('YYYY-MM-DD');
elevenMonthsBefore = moment(max).add(-11, 'months').format("M月");
tenMonthsBefore = moment(max).add(-10, 'months').format("M月");
nineMonthsBefore = moment(max).add(-9, 'months').format("M月");
eightMonthsBefore = moment(max).add(-8, 'months').format("M月");
sevenMonthsBefore = moment(max).add(-7, 'months').format("M月");
sixMonthsBefore = moment(max).add(-6, 'months').format("M月");
fiveMonthsBefore = moment(max).add(-5, 'months').format("M月");
fourMonthsBefore = moment(max).add(-4, 'months').format("M月");
threeMonthsBefore = moment(max).add(-3, 'months').format("M月");
twoMonthsBefore = moment(max).add(-2, 'months').format("M月");
oneMonthBefore = moment(max).add(-1, 'months').format("M月");
thisMonth = moment(max).format("M月");
} else {
maxMonth = "THIS_MONTH()";
maxMonth2 = moment().add(-12, "months").endOf("month").format('YYYY-MM-DD');
elevenMonthsBefore = moment().add(-11, 'months').format("M月");
tenMonthsBefore = moment().add(-10, 'months').format("M月");
nineMonthsBefore = moment().add(-9, 'months').format("M月");
eightMonthsBefore = moment().add(-8, 'months').format("M月");
sevenMonthsBefore = moment().add(-7, 'months').format("M月");
sixMonthsBefore = moment().add(-6, 'months').format("M月");
fiveMonthsBefore = moment().add(-5, 'months').format("M月");
fourMonthsBefore = moment().add(-4, 'months').format("M月");
threeMonthsBefore = moment().add(-3, 'months').format("M月");
twoMonthsBefore = moment().add(-2, 'months').format("M月");
oneMonthBefore = moment().add(-1, 'months').format("M月");
thisMonth = moment().format("M月");
}
if (min) {
minMonth = moment(min).format('YYYY-MM-DD');
minMonth2 = moment(min).add(-12, "months").startOf("month").format('YYYY-MM-DD');
} else {
minMonth = moment().add(-11, "months").startOf("month").format('YYYY-MM-DD');
minMonth2 = moment().add(-23, "months").startOf("month").format('YYYY-MM-DD');
}
// 今月から一年前までのレコード取得
fetchRecords(SALES_APPID, '日付 <= ' + maxMonth + ' and 日付 >= "' + minMonth + '" order by 日付 asc').then(function(canvas1Rec) {
var data = [];
data[elevenMonthsBefore] = 0;
data[tenMonthsBefore] = 0;
data[nineMonthsBefore] = 0;
data[eightMonthsBefore] = 0;
data[sevenMonthsBefore] = 0;
data[sixMonthsBefore] = 0;
data[fiveMonthsBefore] = 0;
data[fourMonthsBefore] = 0;
data[threeMonthsBefore] = 0;
data[twoMonthsBefore] = 0;
data[oneMonthBefore] = 0;
data[thisMonth] = 0;
for (var i = 0; i < canvas1Rec.length; i++) {
var month;
var date = canvas1Rec[i].日付.value.split("-");
if (date[1].charAt(0) === "0") {
month = date[1].charAt(1) + "月";
} else {
month = date[1] + "月";
}
data[month] += parseInt(canvas1Rec[i].販売額.value, 10);
}
var eventuallyThisYearData = [data[elevenMonthsBefore], data[tenMonthsBefore], data[nineMonthsBefore], data[eightMonthsBefore], data[sevenMonthsBefore], data[sixMonthsBefore], data[fiveMonthsBefore], data[fourMonthsBefore], data[threeMonthsBefore], data[twoMonthsBefore], data[oneMonthBefore], data[thisMonth]];
fetchRecords(SALES_APPID, '日付 <= "' + maxMonth2 + '" and 日付 >= "' + minMonth2 + '" order by 日付 asc').then(function(canvas1Rec2) {
fetchRecords(TRIAL_APPID, '日付 <= ' + maxMonth + ' and 日付 >= "' + minMonth + '" order by 日付 asc').then(function(canvas2Rec) {
data = refreshData(data);
// 前年同月比グラフのデータ作成
for (var j = 0; j < canvas1Rec2.length; j++) {
var month2;
var date2 = canvas1Rec2[j].日付.value.split("-");
if (date2[1].charAt(0) === "0") {
month2 = date2[1].charAt(1) + "月";
} else {
month2 = date2[1] + "月";
}
data[month2] += parseInt(canvas1Rec2[j].販売額.value, 10);
}
var eventuallyLastYearData = [data[elevenMonthsBefore], data[tenMonthsBefore], data[nineMonthsBefore], data[eightMonthsBefore], data[sevenMonthsBefore], data[sixMonthsBefore], data[fiveMonthsBefore], data[fourMonthsBefore], data[threeMonthsBefore], data[twoMonthsBefore], data[oneMonthBefore], data[thisMonth]];
data = refreshData(data);
// ミックスドグラフのデータ作成
for (var k = 0; k < canvas2Rec.length; k++) {
var month3;
var date3 = canvas2Rec[k].日付.value.split("-");
if (date3[1].charAt(0) === "0") {
month3 = date3[1].charAt(1) + "月";
} else {
month3 = date3[1] + "月";
}
data[month3] += 1;
}
var eventuallyTrialData = [data[elevenMonthsBefore], data[tenMonthsBefore], data[nineMonthsBefore], data[eightMonthsBefore], data[sevenMonthsBefore], data[sixMonthsBefore], data[fiveMonthsBefore], data[fourMonthsBefore], data[threeMonthsBefore], data[twoMonthsBefore], data[oneMonthBefore], data[thisMonth]];
// ドーナツグラフのデータ作成
var prodData = [];
var prodLabels = [];
for (var l = 0; l < canvas1Rec.length; l++) {
var prod = canvas1Rec[l].製品名.value;
if (typeof (prodData[prod]) === "undefined") {
prodData[prod] = 0;
prodLabels.push(prod);
}
prodData[prod] += parseInt(canvas1Rec[l].販売額.value, 10);
}
var eventuallyByProductData = [];
// 製品背景色
var prodBGColorMST = {
プリントクリエイター: "rgba(54, 162, 235, 0.4)",
フォームクリエイター: "rgba(255, 99, 132, 0.4)",
kBackup: "rgba(0, 100, 0, 0.4)",
kViewer: "rgba(255, 60, 255, 0.4)",
タイムスタンプ: "rgba(255, 255, 0, 0.4)",
メールワイズ: "rgba(255, 112, 0, 0.4)",
ガルーン: "rgba(60, 60, 120, 0.4)",
kintone: "rgba(255, 0, 0, 0.4)",
安否確認: "rgba(0, 255, 255, 0.4)"
};
// 【参考】製品縁取り色
/*var prodBDColorMST = {
プリントクリエイター:"rgba(54, 162, 235, 1)",
フォームクリエイター:"rgba(255,99,132,1)",
kBackup:"rgba(0, 170, 0, 1)",
kViewer:"rgba(255, 170, 255, 1)",
タイムスタンプ:"rgba(255, 255, 0, 1)",
メールワイズ:"rgba(255, 112, 0, 1)",
ガルーン:"rgba(100, 0, 0, 1)",
kintone:"rgba(255, 184, 184, 1)",
安否確認:"rgba(0, 255, 255, 1)"
};*/
var prodBGColor = [];
//var prodBDColor = [];
// 降順にソート
prodLabels.sort(function(a, b) {
if (prodData[a] > prodData[b]) {
return -1;
}
if (prodData[a] < prodData[b]) {
return 1;
}
return 0;
});
for (var m = 0; m < prodLabels.length; m++) {
eventuallyByProductData.push(prodData[prodLabels[m]]);
prodBGColor.push(prodBGColorMST[prodLabels[m]]);
// 【参考】縁取り色をつける場合
//prodBDColor.push(prodBDColorMST[prodLabels[m]]);
// 【参考】ランダムで配色する場合の処理
//prodBGColor.push(dynamicColors());
//prodBDColor.push(dynamicColors());
}
// 前年同月比グラフを描画
var ctx = document.getElementById("canvas1");
if (chart1) {
chart1.destroy();
}
chart1 = new Chart(ctx, {
type: 'bar',
data: {
labels: [elevenMonthsBefore, tenMonthsBefore, nineMonthsBefore, eightMonthsBefore, sevenMonthsBefore, sixMonthsBefore, fiveMonthsBefore, fourMonthsBefore, threeMonthsBefore, twoMonthsBefore, oneMonthBefore, thisMonth],
datasets: [
{
label: '前年同月売上',
data: eventuallyLastYearData,
backgroundColor: "rgba(54, 162, 235, 0.2)",
borderColor: 'rgba(54, 162, 235, 1)',
borderWidth: 1
},
{
label: min ? moment(min).format('YYYY年M月~') + moment(max).format('YYYY年M月売上') : moment().add(-11, "months").format('YYYY年M月~') + moment().format('YYYY年M月売上'),
data: eventuallyThisYearData,
backgroundColor: 'rgba(255, 99, 132, 0.2)',
borderColor: 'rgba(255,99,132,1)',
borderWidth: 1
}
]
},
options: {
scales: {
yAxes: [{
display: true,
scaleLabel: {
display: true,
labelString: '売上',
fontFamily: 'monospace'
},
ticks: {
beginAtZero: true,
maxTicksLimit: 8,
callback: function(value) {
return value + '万円';
}
}
}]
},
tooltips: {
enabled: true,
mode: 'single',
callbacks: {
title: function(tooltipItems, titleData) {
var val = $("#slider").dateRangeSlider("values");
var min2 = moment(val.min);
var tmp = tooltipItems[0].xLabel.split("月");
var first = moment(min2.format("YYYY-") + tmp[0] + '-1', 'YYYY-M-D');
if (first.isBetween(min2, moment(val.max), null, '[]')) {
if (tooltipItems[0].datasetIndex === 0) {
return min2.add(-1, 'years').format("YYYY年") + tmp[0] + '月';
}
return min2.format("YYYY年") + tmp[0] + '月';
}
if (tooltipItems[0].datasetIndex === 0) {
return min2.format("YYYY年") + tmp[0] + '月';
}
return min2.add(1, 'years').format("YYYY年") + tmp[0] + '月';
},
label: function(tooltipItems, data3) {
return tooltipItems.yLabel + '万円';
}
}
}
}
});
// mixedグラフを描画
var ctx2 = document.getElementById("canvas2");
if (chart2) {
chart2.destroy();
}
chart2 = new Chart(ctx2, {
type: 'bar',
data: {
labels: [elevenMonthsBefore, tenMonthsBefore, nineMonthsBefore, eightMonthsBefore, sevenMonthsBefore, sixMonthsBefore, fiveMonthsBefore, fourMonthsBefore, threeMonthsBefore, twoMonthsBefore, oneMonthBefore, thisMonth],
datasets: [
{
yAxisID: "y-axis-1",
type: 'line',
label: '試用申込数',
data: eventuallyTrialData,
backgroundColor: "rgba(54, 162, 235, 0.2)",
borderColor: 'rgba(54, 162, 235, 1)',
borderWidth: 1
},
{
yAxisID: "y-axis-0",
type: 'bar',
label: min ? moment(min).format('YYYY年M月~') + moment(max).format('YYYY年M月売上') : moment().add(-11, "months").format('YYYY年M月~') + moment().format('YYYY年M月売上'),
data: eventuallyThisYearData,
backgroundColor: 'rgba(255, 99, 132, 0.2)',
borderColor: 'rgba(255,99,132,1)',
borderWidth: 1
}
]
},
options: {
scales: {
yAxes: [{
position: "left",
id: "y-axis-0",
display: true,
scaleLabel: {
display: true,
labelString: '売上',
fontFamily: 'monospace'
},
ticks: {
beginAtZero: true,
maxTicksLimit: 8,
callback: function(value) {
return value + '万円';
}
}
}, {
position: "right",
id: "y-axis-1",
display: true,
scaleLabel: {
display: true,
labelString: '試用申込数',
fontFamily: 'monospace'
},
ticks: {
beginAtZero: true,
maxTicksLimit: 6
}
}]
},
tooltips: {
enabled: true,
mode: 'single',
callbacks: {
title: function(tooltipItems, data4) {
var val = $("#slider").dateRangeSlider("values");
var min3 = moment(val.min);
var tmp = tooltipItems[0].xLabel.split("月");
var first = moment(min3.format("YYYY-") + tmp[0] + '-1', 'YYYY-M-D');
if (first.isBetween(min3, moment(val.max), null, '[]')) {
return min3.format("YYYY年") + tmp[0] + '月';
}
return min3.add(1, 'years').format("YYYY年") + tmp[0] + '月';
},
label: function(tooltipItems, data5) {
if (tooltipItems.datasetIndex === 0) {
return '試用申込数:' + tooltipItems.yLabel;
}
return '売上:' + tooltipItems.yLabel + '万円';
}
}
}
}
});
// ドーナツグラフを描画
var ctx3 = document.getElementById("canvas3");
if (chart3) {
chart3.destroy();
}
chart3 = new Chart(ctx3, {
type: 'doughnut',
data: {
labels: prodLabels,
datasets: [
{
label: min ? moment(min).format('YYYY年M月~') + moment(max).format('YYYY年M月売上') : moment().add(-11, "months").format('YYYY年M月~') + moment().format('YYYY年M月売上'),
data: eventuallyByProductData,
backgroundColor: prodBGColor,
//borderColor: prodBDColor,
borderWidth: 1
}
]
},
options: {
showAllTooltips: true,
animation: {
animateScale: true
},
tooltips: {
callbacks: {
title: function(tooltipItems, data6) {
return data6.labels[tooltipItems[0].index];
},
label: function(tooltipItems, data7) {
var sum = 0;
for (var n = 0; n < data7.datasets[0].data.length; n++) {
sum += data7.datasets[0].data[n];
}
var per = Math.round((data7.datasets[0].data[tooltipItems.index] / sum * 100));
return per + '% ' + data7.datasets[0].data[tooltipItems.index] + '万円';
}
}
}
}
});
// グラフのデフォルトフォントサイズ
Chart.defaults.global.defaultFontSize = 14;
// 期間指定スライダーの横幅調整
$("#slider").css("width", "90%");
// スライダーの作成は画面読み込み時のみ
if (oneTimeFlg) {
createSlider();
}
onceFlg = true;
//removeLoading();
});
});
});
}
// data配列の中身を消す関数
function refreshData(data) {
for (var key in data) {
data[key] = 0;
}
return data;
}
// 全レコード取得関数
function fetchRecords(appId, query, opt_offset, opt_limit, opt_records) {
var offset = opt_offset || 0;
var limit = opt_limit || 100;
var allRecords = opt_records || [];
var params = {
app: appId,
query: query + ' limit ' + limit + ' offset ' + offset
};
return kintone.api('/k/v1/records', 'GET', params).then(function(resp) {
allRecords = allRecords.concat(resp.records);
if (resp.records.length === limit) {
return fetchRecords(appId, query, offset + limit, limit, allRecords);
}
return allRecords;
});
}
// ローディング画面を出す関数
// function setLoading() {
// var $body = $('body');
// $body.css('width', '100%');
//
// var $loading = $('<div>').attr('id', 'loading').attr('class', 'loading')
// .attr('style', 'width: 100%; height: 100%; position:absolute;' +
// ' top:0; left:0; text-align:center; background-color:#666666; opacity:0.6; z-index: 9;');
// var $div = $('<div>').attr('id', 'imgBox').attr('style', 'width: 100%; height: 100%;');
// var $img = $('<img>').attr('src', '');
// $loading.append($div.append($img));
// $body.append($loading);
//
// $('#imgBox').attr('style', 'margin-top: ' + Math.floor($('#loading').height() / 2) + 'px;');
//
// $body.css('position', 'fixed');
//
// var autoRemoveCnt = 0;
// var autoRemoveInterval = setInterval(function() {
// autoRemoveCnt++;
// if (autoRemoveCnt > 9) {
// clearInterval(autoRemoveInterval);
// removeLoading();
// }
// }, 1000);
// }
// ローディング画面を消す関数
// function removeLoading() {
// var $loading = $('.loading');
// $loading.remove();
//
// var $body = $('body');
// $body.css('position', '');
// }
// 【参考】ドーナツグラフの各製品色をランダムに決定する関数
// function dynamicColors() {
// var r = Math.floor(Math.random() * 255);
// var g = Math.floor(Math.random() * 255);
// var b = Math.floor(Math.random() * 255);
// return "rgb(" + r + "," + g + "," + b + ")";
// }
})();
感想
他のデベロッパーサイト記事と比して私の記事のコード量に圧倒されますね(゜o ゜;)
Chart.jsではできないことが割と多いので、「いかにChart.jsを要件に寄せるか」ではなく「いかに要件をChart.jsに寄せるか」が問われます(SIer的に考えて)。


