Skip to content

Commit

Permalink
Allow using either moment or luxon in pluggable fashion
Browse files Browse the repository at this point in the history
  • Loading branch information
benmccann committed Jun 4, 2018
1 parent 25b7f41 commit 06efd98
Show file tree
Hide file tree
Showing 5 changed files with 176 additions and 31 deletions.
2 changes: 2 additions & 0 deletions gulpfile.js
Original file line number Diff line number Diff line change
Expand Up @@ -107,6 +107,7 @@ function buildTask() {
}

var bundled = browserify('./src/chart.js', { standalone: 'Chart' })
.ignore('luxon')
.plugin(collapse)
.bundle()
.on('error', errorHandler)
Expand All @@ -121,6 +122,7 @@ function buildTask() {
.pipe(gulp.dest(outDir));

var nonBundled = browserify('./src/chart.js', { standalone: 'Chart' })
.ignore('luxon')
.ignore('moment')
.plugin(collapse)
.bundle()
Expand Down
6 changes: 5 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,8 @@
"karma-firefox-launcher": "^1.0.1",
"karma-jasmine": "^1.1.0",
"karma-jasmine-html-reporter": "^0.2.2",
"moment": "^2.10.2",
"luxon": "^1.2.1",
"merge-stream": "^1.0.1",
"pixelmatch": "^4.0.2",
"vinyl-source-stream": "^1.1.0",
Expand All @@ -51,7 +53,9 @@
"main": "Chart.js"
},
"dependencies": {
"chartjs-color": "^2.1.0",
"chartjs-color": "^2.1.0"
},
"peerDependencies": {
"moment": "^2.10.2"
}
}
5 changes: 4 additions & 1 deletion samples/samples.js
Original file line number Diff line number Diff line change
Expand Up @@ -113,8 +113,11 @@
title: 'Line (point data)',
path: 'scales/time/line-point-data.html'
}, {
title: 'Time Series',
title: 'Time Series - Moment',
path: 'scales/time/financial.html'
}, {
title: 'Time Series - Luxon',
path: 'scales/time/financial-luxon.html'
}, {
title: 'Combo',
path: 'scales/time/combo.html'
Expand Down
101 changes: 101 additions & 0 deletions samples/scales/time/financial-luxon.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
<!doctype html>
<html>

<head>
<title>Line Chart</title>
<script src="https://moment.github.io/luxon/global/luxon.min.js"></script>
<script src="../../../dist/Chart.js"></script>
<script src="../../utils.js"></script>
<style>
canvas {
-moz-user-select: none;
-webkit-user-select: none;
-ms-user-select: none;
}
</style>
</head>

<body>
<div style="width:1000px">
<canvas id="chart1"></canvas>
</div>
<br>
<br>
Chart Type:
<select id="type">
<option value="line">Line</option>
<option value="bar">Bar</option>
</select>
<button id="update">update</button>
<script>
function randomNumber(min, max) {
return Math.random() * (max - min) + min;
}

function randomBar(date, lastClose) {
var open = randomNumber(lastClose * 0.95, lastClose * 1.05);
var close = randomNumber(open * 0.95, open * 1.05);
return {
t: date.valueOf(),
y: close
};
}

var date = luxon.DateTime.local(2017, 4, 1, 0, 0);
var data = [randomBar(date, 30)];
var labels = [date.valueOf()];
while (data.length < 60) {
date = date.plus({ days: 1 });
if (date.weekday <= 5) {
data.push(randomBar(date, data[data.length - 1].y));
labels.push(date.valueOf());
}
}

var ctx = document.getElementById('chart1').getContext('2d');
ctx.canvas.width = 1000;
ctx.canvas.height = 300;
var cfg = {
type: 'bar',
data: {
labels: labels,
datasets: [{
label: 'CHRT - Chart.js Corporation',
data: data,
type: 'line',
pointRadius: 0,
fill: false,
lineTension: 0,
borderWidth: 2
}]
},
options: {
scales: {
xAxes: [{
type: 'time',
distribution: 'series',
ticks: {
source: 'labels'
}
}],
yAxes: [{
scaleLabel: {
display: true,
labelString: 'Closing price ($)'
}
}]
}
}
};
var chart = new Chart(ctx, cfg);

document.getElementById('update').addEventListener('click', function() {
var type = document.getElementById('type').value;
chart.config.data.datasets[0].type = type;
chart.update();
});

</script>
</body>

</html>
93 changes: 64 additions & 29 deletions src/scales/scale.time.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,19 @@
/* global window: false */
'use strict';

var moment = require('moment');
moment = typeof moment === 'function' ? moment : window.moment;
var moment, luxon, DateTime;
try {
moment = require('moment');
moment = typeof moment === 'function' ? moment : window.moment;
} catch (mte) {
try {
luxon = require('luxon');
luxon = (luxon && luxon.DateTime) ? luxon : window.luxon;
DateTime = luxon && luxon.DateTime;
} catch (lxe) {
throw new Error('Chart.js - Neither Moment.js no Luxon could be found! You must include one of these date libraries to use the time scale. For more info, see https://www.chartjs.org/docs/latest/getting-started/installation.html');
}
}

var defaults = require('../core/core.defaults');
var helpers = require('../helpers/index');
Expand Down Expand Up @@ -179,11 +190,24 @@ function interpolate(table, skey, sval, tkey) {
return prev[tkey] + offset;
}

function millisToDate(value) {
return luxon ? DateTime.fromMillis(value) : moment(value);
}

/**
* Convert the given value to a moment object using the given time options.
* @see http://momentjs.com/docs/#/parsing/
*/
function momentify(value, options) {
function createDate(value, options) {
if (luxon) {
if (typeof value === 'number') {
return DateTime.fromMillis(value); // TODO(benmccann): what time zone is this value?
}
if (value instanceof Date) {
return DateTime.fromJSDate(value);
}
throw 'Input must be either a Date or milliseconds since epoch, but got ' + value;
}
var parser = options.parser;
var format = options.parser || options.format;

Expand Down Expand Up @@ -212,14 +236,18 @@ function momentify(value, options) {
return value;
}

function isValid(date) {
return luxon ? date.isValid : date.isValid();
}

function parse(input, scale) {
if (helpers.isNullOrUndef(input)) {
return null;
}

var options = scale.options.time;
var value = momentify(scale.getRightValue(input), options);
if (!value.isValid()) {
var value = createDate(scale.getRightValue(input), options);
if (!isValid(value)) {
return null;
}

Expand Down Expand Up @@ -278,7 +306,8 @@ function determineUnitForAutoTicks(minUnit, min, max, capacity) {
* Figures out what unit to format a set of ticks with
*/
function determineUnitForFormatting(ticks, minUnit, min, max) {
var duration = moment.duration(moment(max).diff(moment(min)));
var duration = luxon ? createDate(max).diff(createDate(min))
: moment.duration(moment(max).diff(moment(min)));
var ilen = UNITS.length;
var i, unit;

Expand Down Expand Up @@ -394,7 +423,7 @@ function ticksFromTimestamps(values, majorUnit) {

for (i = 0, ilen = values.length; i < ilen; ++i) {
value = values[i];
major = majorUnit ? value === +moment(value).startOf(majorUnit) : false;
major = majorUnit ? value === millisToDate(value).startOf(majorUnit).valueOf() : false;

ticks.push({
value: value,
Expand All @@ -406,18 +435,27 @@ function ticksFromTimestamps(values, majorUnit) {
}

function determineLabelFormat(data, timeOpts) {
var i, momentDate, hasTime;
var i, date, hasTime;
var ilen = data.length;

// find the label with the most parts (milliseconds, minutes, etc.)
// format all labels with the same level of detail as the most specific label
for (i = 0; i < ilen; i++) {
momentDate = momentify(data[i], timeOpts);
if (momentDate.millisecond() !== 0) {
return 'MMM D, YYYY h:mm:ss.SSS a';
}
if (momentDate.second() !== 0 || momentDate.minute() !== 0 || momentDate.hour() !== 0) {
hasTime = true;
date = createDate(data[i], timeOpts);
if (luxon) {
if (date.millisecond !== 0) {
return 'MMM D, YYYY h:mm:ss.SSS a';
}
if (date.second !== 0 || date.minute !== 0 || date.hour !== 0) {
hasTime = true;
}
} else {
if (date.millisecond() !== 0) {
return 'MMM D, YYYY h:mm:ss.SSS a';
}
if (date.second() !== 0 || date.minute() !== 0 || date.hour() !== 0) {
hasTime = true;
}
}
}
if (hasTime) {
Expand Down Expand Up @@ -463,12 +501,12 @@ module.exports = function() {
millisecond: 'h:mm:ss.SSS a', // 11:20:01.123 AM,
second: 'h:mm:ss a', // 11:20:01 AM
minute: 'h:mm a', // 11:20 AM
hour: 'hA', // 5PM
day: 'MMM D', // Sep 4
week: 'll', // Week 46, or maybe "[W]WW - YYYY" ?
month: 'MMM YYYY', // Sept 2015
quarter: '[Q]Q - YYYY', // Q3
year: 'YYYY' // 2015
hour: luxon ? 'ha' : 'hA', // 5PM
day: luxon ? 'MMM d' : 'MMM D', // Sep 4
week: luxon ? 'WW' : 'll', // Week 46, or maybe "[W]WW - YYYY" ?
month: luxon ? 'MMM yyyy' : 'MMM YYYY', // Sept 2015
quarter: luxon ? '[Q]q - yyyy' : '[Q]Q - YYYY', // Q3
year: luxon ? 'yyyy' : 'YYYY' // 2015
},
},
ticks: {
Expand All @@ -492,10 +530,6 @@ module.exports = function() {

var TimeScale = Scale.extend({
initialize: function() {
if (!moment) {
throw new Error('Chart.js - Moment.js could not be found! You must include it before Chart.js to use the time scale. Download at https://momentjs.com');
}

this.mergeTicksOptions();

Scale.prototype.initialize.call(this);
Expand Down Expand Up @@ -660,13 +694,13 @@ module.exports = function() {
label = me.getRightValue(value);
}
if (timeOpts.tooltipFormat) {
return momentify(label, timeOpts).format(timeOpts.tooltipFormat);
return createDate(label, timeOpts).format(timeOpts.tooltipFormat);
}
if (typeof label === 'string') {
return label;
}

return momentify(label, timeOpts).format(me._labelFormat);
return createDate(label, timeOpts).format(me._labelFormat);
},

/**
Expand All @@ -681,10 +715,11 @@ module.exports = function() {
var minorFormat = formats[me._unit];
var majorUnit = me._majorUnit;
var majorFormat = formats[majorUnit];
var majorTime = tick.clone().startOf(majorUnit).valueOf();
var majorTime = (luxon ? tick : tick.clone()).startOf(majorUnit).valueOf();
var majorTickOpts = options.ticks.major;
var major = majorTickOpts.enabled && majorUnit && majorFormat && time === majorTime;
var label = tick.format(formatOverride ? formatOverride : major ? majorFormat : minorFormat);
var format = formatOverride ? formatOverride : major ? majorFormat : minorFormat;
var label = luxon ? tick.toFormat(format) : tick.format(format);
var tickOpts = major ? majorTickOpts : options.ticks.minor;
var formatter = helpers.valueOrDefault(tickOpts.callback, tickOpts.userCallback);

Expand All @@ -696,7 +731,7 @@ module.exports = function() {
var i, ilen;

for (i = 0, ilen = ticks.length; i < ilen; ++i) {
labels.push(this.tickFormatFunction(moment(ticks[i].value), i, ticks));
labels.push(this.tickFormatFunction(millisToDate(ticks[i].value), i, ticks));
}

return labels;
Expand Down

0 comments on commit 06efd98

Please sign in to comment.