123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222 |
- 'use strict';
- module.exports = function(Chart) {
- var helpers = Chart.helpers;
- var defaultConfig = {
- position: 'left',
- // label settings
- ticks: {
- callback: Chart.Ticks.formatters.logarithmic
- }
- };
- var LogarithmicScale = Chart.Scale.extend({
- determineDataLimits: function() {
- var me = this;
- var opts = me.options;
- var tickOpts = opts.ticks;
- var chart = me.chart;
- var data = chart.data;
- var datasets = data.datasets;
- var getValueOrDefault = helpers.getValueOrDefault;
- var isHorizontal = me.isHorizontal();
- function IDMatches(meta) {
- return isHorizontal ? meta.xAxisID === me.id : meta.yAxisID === me.id;
- }
- // Calculate Range
- me.min = null;
- me.max = null;
- me.minNotZero = null;
- if (opts.stacked) {
- var valuesPerType = {};
- helpers.each(datasets, function(dataset, datasetIndex) {
- var meta = chart.getDatasetMeta(datasetIndex);
- if (chart.isDatasetVisible(datasetIndex) && IDMatches(meta)) {
- if (valuesPerType[meta.type] === undefined) {
- valuesPerType[meta.type] = [];
- }
- helpers.each(dataset.data, function(rawValue, index) {
- var values = valuesPerType[meta.type];
- var value = +me.getRightValue(rawValue);
- if (isNaN(value) || meta.data[index].hidden) {
- return;
- }
- values[index] = values[index] || 0;
- if (opts.relativePoints) {
- values[index] = 100;
- } else {
- // Don't need to split positive and negative since the log scale can't handle a 0 crossing
- values[index] += value;
- }
- });
- }
- });
- helpers.each(valuesPerType, function(valuesForType) {
- var minVal = helpers.min(valuesForType);
- var maxVal = helpers.max(valuesForType);
- me.min = me.min === null ? minVal : Math.min(me.min, minVal);
- me.max = me.max === null ? maxVal : Math.max(me.max, maxVal);
- });
- } else {
- helpers.each(datasets, function(dataset, datasetIndex) {
- var meta = chart.getDatasetMeta(datasetIndex);
- if (chart.isDatasetVisible(datasetIndex) && IDMatches(meta)) {
- helpers.each(dataset.data, function(rawValue, index) {
- var value = +me.getRightValue(rawValue);
- if (isNaN(value) || meta.data[index].hidden) {
- return;
- }
- if (me.min === null) {
- me.min = value;
- } else if (value < me.min) {
- me.min = value;
- }
- if (me.max === null) {
- me.max = value;
- } else if (value > me.max) {
- me.max = value;
- }
- if (value !== 0 && (me.minNotZero === null || value < me.minNotZero)) {
- me.minNotZero = value;
- }
- });
- }
- });
- }
- me.min = getValueOrDefault(tickOpts.min, me.min);
- me.max = getValueOrDefault(tickOpts.max, me.max);
- if (me.min === me.max) {
- if (me.min !== 0 && me.min !== null) {
- me.min = Math.pow(10, Math.floor(helpers.log10(me.min)) - 1);
- me.max = Math.pow(10, Math.floor(helpers.log10(me.max)) + 1);
- } else {
- me.min = 1;
- me.max = 10;
- }
- }
- },
- buildTicks: function() {
- var me = this;
- var opts = me.options;
- var tickOpts = opts.ticks;
- var generationOptions = {
- min: tickOpts.min,
- max: tickOpts.max
- };
- var ticks = me.ticks = Chart.Ticks.generators.logarithmic(generationOptions, me);
- if (!me.isHorizontal()) {
- // We are in a vertical orientation. The top value is the highest. So reverse the array
- ticks.reverse();
- }
- // At this point, we need to update our max and min given the tick values since we have expanded the
- // range of the scale
- me.max = helpers.max(ticks);
- me.min = helpers.min(ticks);
- if (tickOpts.reverse) {
- ticks.reverse();
- me.start = me.max;
- me.end = me.min;
- } else {
- me.start = me.min;
- me.end = me.max;
- }
- },
- convertTicksToLabels: function() {
- this.tickValues = this.ticks.slice();
- Chart.Scale.prototype.convertTicksToLabels.call(this);
- },
- // Get the correct tooltip label
- getLabelForIndex: function(index, datasetIndex) {
- return +this.getRightValue(this.chart.data.datasets[datasetIndex].data[index]);
- },
- getPixelForTick: function(index) {
- return this.getPixelForValue(this.tickValues[index]);
- },
- getPixelForValue: function(value) {
- var me = this;
- var innerDimension;
- var pixel;
- var start = me.start;
- var newVal = +me.getRightValue(value);
- var range;
- var opts = me.options;
- var tickOpts = opts.ticks;
- if (me.isHorizontal()) {
- range = helpers.log10(me.end) - helpers.log10(start); // todo: if start === 0
- if (newVal === 0) {
- pixel = me.left;
- } else {
- innerDimension = me.width;
- pixel = me.left + (innerDimension / range * (helpers.log10(newVal) - helpers.log10(start)));
- }
- } else {
- // Bottom - top since pixels increase downward on a screen
- innerDimension = me.height;
- if (start === 0 && !tickOpts.reverse) {
- range = helpers.log10(me.end) - helpers.log10(me.minNotZero);
- if (newVal === start) {
- pixel = me.bottom;
- } else if (newVal === me.minNotZero) {
- pixel = me.bottom - innerDimension * 0.02;
- } else {
- pixel = me.bottom - innerDimension * 0.02 - (innerDimension * 0.98/ range * (helpers.log10(newVal)-helpers.log10(me.minNotZero)));
- }
- } else if (me.end === 0 && tickOpts.reverse) {
- range = helpers.log10(me.start) - helpers.log10(me.minNotZero);
- if (newVal === me.end) {
- pixel = me.top;
- } else if (newVal === me.minNotZero) {
- pixel = me.top + innerDimension * 0.02;
- } else {
- pixel = me.top + innerDimension * 0.02 + (innerDimension * 0.98/ range * (helpers.log10(newVal)-helpers.log10(me.minNotZero)));
- }
- } else {
- range = helpers.log10(me.end) - helpers.log10(start);
- innerDimension = me.height;
- pixel = me.bottom - (innerDimension / range * (helpers.log10(newVal) - helpers.log10(start)));
- }
- }
- return pixel;
- },
- getValueForPixel: function(pixel) {
- var me = this;
- var range = helpers.log10(me.end) - helpers.log10(me.start);
- var value, innerDimension;
- if (me.isHorizontal()) {
- innerDimension = me.width;
- value = me.start * Math.pow(10, (pixel - me.left) * range / innerDimension);
- } else { // todo: if start === 0
- innerDimension = me.height;
- value = Math.pow(10, (me.bottom - pixel) * range / innerDimension) / me.start;
- }
- return value;
- }
- });
- Chart.scaleService.registerScaleType('logarithmic', LogarithmicScale, defaultConfig);
- };
|