Chart.jsで作る「kintoneのダッシュボード」

※当記事の内容はサポート対象外です

こんにちは。武井です。

ダッシュボードの季節ですね。

kintoneのグラフ解析

kintoneの標準グラフにはHigh Chartsが採用されているようです。

High Chartsの特徴としては、

  • 見た目が綺麗
  • 使用に便利
  • カスタムに融通が利く

と最強らしいのですが、有料です。

一方、今回使用するChart.jsは無料チャートライブラリの中では一強の様相を呈しています。
ただし、Chart.jsはやや融通の利かない部分があります。

ダッシュボード作ってみた

201609131013

こんなん作ってみました。

詳しくは下記記事に譲りますが、
kintoneの標準機能にないグラフを厳選してChart.jsで実現したところが売りとなっております。

■Chart.jsを使ってダッシュボードを作ってみよう!
https://cybozudev.zendesk.com/hc/ja/articles/214479043

上記記事と参考画像では説明不足だったのですが、
下図のようなインタフェースも導入しているので、非常に使い勝手はよいです。

dashboard

右下部分が空白地帯でちょっと寂しいのは作業時間の関係上となります(゚゚;)

ダッシュボードをポータルに表示する

さて、kintone公式ではめったなことでDOMを操作してはいけないことになっていますので、
上記記事ではアプリ内にダッシュボードを表示していますが、
こちらでは例によってDOMをいじりまくることでポータルにダッシュボードを表示してしまう魔改造を披露していきたいと思います。

完成イメージ

portaldashboard

ぶっちゃけ使いやすさは圧倒的にこちらですね!

コードを一応すべて載せますが、アプリ版との違いは、ほぼほぼ「// 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的に考えて)。


▼複数製品も同時に無料お試し!▼