| 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);};
 |