react-vis
Version:
Data visualization library based on React and d3.
861 lines (766 loc) • 27.8 kB
JavaScript
var _slicedToArray = function () { function sliceIterator(arr, i) { var _arr = []; var _n = true; var _d = false; var _e = undefined; try { for (var _i = arr[Symbol.iterator](), _s; !(_n = (_s = _i.next()).done); _n = true) { _arr.push(_s.value); if (i && _arr.length === i) break; } } catch (err) { _d = true; _e = err; } finally { try { if (!_n && _i["return"]) _i["return"](); } finally { if (_d) throw _e; } } return _arr; } return function (arr, i) { if (Array.isArray(arr)) { return arr; } else if (Symbol.iterator in Object(arr)) { return sliceIterator(arr, i); } else { throw new TypeError("Invalid attempt to destructure non-iterable instance"); } }; }();
var _extends = Object.assign || function (target) { for (var i = 1; i < arguments.length; i++) { var source = arguments[i]; for (var key in source) { if (Object.prototype.hasOwnProperty.call(source, key)) { target[key] = source[key]; } } } return target; };
var _SCALE_FUNCTIONS;
function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length); i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
// Copyright (c) 2016 - 2017 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import { scaleLinear, scalePoint, scaleOrdinal, scaleLog, scaleTime, scaleUtc } from 'd3-scale';
import { extent } from 'd3-array';
import { set } from 'd3-collection';
import { hsl } from 'd3-color';
import PropTypes from 'prop-types';
import { warning } from './react-utils';
import { getUniquePropertyValues, addValueToArray } from './data-utils';
/**
* Linear scale name.
* @type {string}
* @const
*/
var LINEAR_SCALE_TYPE = 'linear';
/**
* Ordinal scale name.
* @type {string}
* @const
*/
var ORDINAL_SCALE_TYPE = 'ordinal';
/**
* Category scale.
* @type {string}
* @const
*/
var CATEGORY_SCALE_TYPE = 'category';
/**
* Literal scale.
* Differs slightly from d3's identity scale in that it does not coerce value
* into numbers, it simply returns exactly what you give it
* @type {string}
* @const
*/
var LITERAL_SCALE_TYPE = 'literal';
/**
* Log scale name.
* @type {string}
* @const
*/
var LOG_SCALE_TYPE = 'log';
/**
* Time scale name.
* @type {string}
* @const
*/
var TIME_SCALE_TYPE = 'time';
/**
* Time UTC scale name.
* @type {string}
* @const
*/
var TIME_UTC_SCALE_TYPE = 'time-utc';
/**
* Scale functions that are supported in the library.
* @type {Object}
* @const
*/
var SCALE_FUNCTIONS = (_SCALE_FUNCTIONS = {}, _defineProperty(_SCALE_FUNCTIONS, LINEAR_SCALE_TYPE, scaleLinear), _defineProperty(_SCALE_FUNCTIONS, ORDINAL_SCALE_TYPE, scalePoint), _defineProperty(_SCALE_FUNCTIONS, CATEGORY_SCALE_TYPE, scaleOrdinal), _defineProperty(_SCALE_FUNCTIONS, LITERAL_SCALE_TYPE, literalScale), _defineProperty(_SCALE_FUNCTIONS, LOG_SCALE_TYPE, scaleLog), _defineProperty(_SCALE_FUNCTIONS, TIME_SCALE_TYPE, scaleTime), _defineProperty(_SCALE_FUNCTIONS, TIME_UTC_SCALE_TYPE, scaleUtc), _SCALE_FUNCTIONS);
/**
* Attrs for which a scale can be set up at XYPlot level
* @type {Array}
* @const
*/
var XYPLOT_ATTR = ['color', 'fill', 'opacity', 'stroke'];
/**
* Title case a given string
* @param {String} str Array of values.
* @returns {String} titlecased string
*/
function toTitleCase(str) {
return '' + str[0].toUpperCase() + str.slice(1);
}
/**
* Find the smallest distance between the values on a given scale and return
* the index of the element, where the smallest distance was found.
* It returns the first occurrence of i where
* `scale(value[i]) - scale(value[i - 1])` is minimal
* @param {Array} values Array of values.
* @param {Object} scaleObject Scale object.
* @returns {number} Index of an element where the smallest distance was found.
* @private
*/
export function _getSmallestDistanceIndex(values, scaleObject) {
var scaleFn = getScaleFnFromScaleObject(scaleObject);
var result = 0;
if (scaleFn) {
var nextValue = void 0;
var currentValue = scaleFn(values[0]);
var distance = Infinity;
var nextDistance = void 0;
for (var i = 1; i < values.length; i++) {
nextValue = scaleFn(values[i]);
nextDistance = Math.abs(nextValue - currentValue);
if (nextDistance < distance) {
distance = nextDistance;
result = i;
}
currentValue = nextValue;
}
}
return result;
}
/**
* Crate a scale function from the scale object.
* @param {Object} scaleObject Scale object.
- scaleObject.domain {Array}
- scaleObject.range {Array}
- scaleObject.type {string}
- scaleObject.attr {string}
* @returns {*} Scale function.
* @private
*/
export function getScaleFnFromScaleObject(scaleObject) {
if (!scaleObject) {
return null;
}
var type = scaleObject.type,
domain = scaleObject.domain,
range = scaleObject.range;
var modDomain = domain[0] === domain[1] ? domain[0] === 0 ? [-1, 0] : [-domain[0], domain[0]] : domain;
if (type === LITERAL_SCALE_TYPE) {
return literalScale(range[0]);
}
var scale = SCALE_FUNCTIONS[type]().domain(modDomain).range(range);
if (type === ORDINAL_SCALE_TYPE) {
scale.padding(0.5);
}
return scale;
}
/**
* Get the domain from the array of data.
* @param {Array} allData All data.
* @param {function} accessor - accessor for main value.
* @param {function} accessor0 - accessor for the naught value.
* @param {string} type Scale type.
* @returns {Array} Domain.
* @private
*/
export function getDomainByAccessor(allData, accessor, accessor0, type) {
var domain = void 0;
// Collect both attr and available attr0 values from the array of data.
var values = allData.reduce(function (data, d) {
var value = accessor(d);
var value0 = accessor0(d);
if (_isDefined(value)) {
data.push(value);
}
if (_isDefined(value0)) {
data.push(value0);
}
return data;
}, []);
if (!values.length) {
return [];
}
// Create proper domain depending on the type of the scale.
if (type !== ORDINAL_SCALE_TYPE && type !== CATEGORY_SCALE_TYPE) {
domain = extent(values);
} else {
domain = set(values).values();
}
return domain;
}
/**
* Create custom scale object from the value. When the scale is created from
* this object, it should return the same value all time.
* @param {string} attr Attribute.
* @param {*} value Value.
* @param {string} type - the type of scale being used
* @param {function} accessor - the accessor function
* @param {function} accessor0 - the accessor function for the potential naught value
* @returns {Object} Custom scale object.
* @private
*/
function _createScaleObjectForValue(attr, value, type, accessor, accessor0) {
if (type === LITERAL_SCALE_TYPE) {
return {
type: LITERAL_SCALE_TYPE,
domain: [],
range: [value],
distance: 0,
attr: attr,
baseValue: undefined,
isValue: true,
accessor: accessor,
accessor0: accessor0
};
}
if (typeof value === 'undefined') {
return null;
}
return {
type: CATEGORY_SCALE_TYPE,
range: [value],
domain: [],
distance: 0,
attr: attr,
baseValue: undefined,
isValue: true,
accessor: accessor,
accessor0: accessor0
};
}
/**
* Create a regular scale object for a further use from the existing parameters.
* @param {Array} domain - Domain.
* @param {Array} range - Range.
* @param {string} type - Type.
* @param {number} distance - Distance.
* @param {string} attr - Attribute.
* @param {number} baseValue - Base value.
* @param {function} accessor - Attribute accesor
* @param {function} accessor0 - Attribute accesor for potential naught value
* @returns {Object} Scale object.
* @private
*/
function _createScaleObjectForFunction(_ref) {
var domain = _ref.domain,
range = _ref.range,
type = _ref.type,
distance = _ref.distance,
attr = _ref.attr,
baseValue = _ref.baseValue,
accessor = _ref.accessor,
accessor0 = _ref.accessor0;
return {
domain: domain,
range: range,
type: type,
distance: distance,
attr: attr,
baseValue: baseValue,
isValue: false,
accessor: accessor,
accessor0: accessor0
};
}
/**
* Get scale object from props. E. g. object like {xRange, xDomain, xDistance,
* xType} is transformed into {range, domain, distance, type}.
* @param {Object} props Props.
* @param {string} attr Attribute.
* @returns {*} Null or an object with the scale.
* @private
*/
function _collectScaleObjectFromProps(props, attr) {
var value = props[attr],
fallbackValue = props['_' + attr + 'Value'],
range = props[attr + 'Range'],
_props$ = props[attr + 'Distance'],
distance = _props$ === undefined ? 0 : _props$,
baseValue = props[attr + 'BaseValue'],
_props$2 = props[attr + 'Type'],
type = _props$2 === undefined ? LINEAR_SCALE_TYPE : _props$2,
noFallBack = props[attr + 'NoFallBack'],
_props$3 = props['get' + toTitleCase(attr)],
accessor = _props$3 === undefined ? function (d) {
return d[attr];
} : _props$3,
_props$4 = props['get' + toTitleCase(attr) + '0'],
accessor0 = _props$4 === undefined ? function (d) {
return d[attr + '0'];
} : _props$4;
var domain = props[attr + 'Domain'];
// Return value-based scale if the value is assigned.
if (!noFallBack && typeof value !== 'undefined') {
return _createScaleObjectForValue(attr, value, props[attr + 'Type'], accessor, accessor0);
}
// Pick up the domain from the properties and create a new one if it's not
// available.
if (typeof baseValue !== 'undefined') {
domain = addValueToArray(domain, baseValue);
}
// Make sure that the minimum necessary properties exist.
if (!range || !domain || !domain.length) {
// Try to use the fallback value if it is available.
return _createScaleObjectForValue(attr, fallbackValue, props[attr + 'Type'], accessor, accessor0);
}
return _createScaleObjectForFunction({
domain: domain,
range: range,
type: type,
distance: distance,
attr: attr,
baseValue: baseValue,
accessor: accessor,
accessor0: accessor0
});
}
/**
* Compute left domain adjustment for the given values.
* @param {Array} values Array of values.
* @returns {number} Domain adjustment.
* @private
*/
function _computeLeftDomainAdjustment(values) {
if (values.length > 1) {
return (values[1] - values[0]) / 2;
}
if (values.length === 1) {
return values[0] - 0.5;
}
return 0;
}
/**
* Compute right domain adjustment for the given values.
* @param {Array} values Array of values.
* @returns {number} Domain adjustment.
* @private
*/
function _computeRightDomainAdjustment(values) {
if (values.length > 1) {
return (values[values.length - 1] - values[values.length - 2]) / 2;
}
if (values.length === 1) {
return values[0] - 0.5;
}
return 0;
}
/**
* Compute distance for the given values.
* @param {Array} values Array of values.
* @param {Array} domain Domain.
* @param {number} bestDistIndex Index of a best distance found.
* @param {function} scaleFn Scale function.
* @returns {number} Domain adjustment.
* @private
*/
function _computeScaleDistance(values, domain, bestDistIndex, scaleFn) {
if (values.length > 1) {
// Avoid zero indexes.
var i = Math.max(bestDistIndex, 1);
return Math.abs(scaleFn(values[i]) - scaleFn(values[i - 1]));
}
if (values.length === 1) {
return Math.abs(scaleFn(domain[1]) - scaleFn(domain[0]));
}
return 0;
}
/**
* Normilize array of values with a single value.
* @param {Array} arr Array of data.
* @param {Array} values Array of values.
* @param {string} attr Attribute.
* @param {string} type Type.
* @private
*/
function _normalizeValues(data, values, accessor0, type) {
if (type === TIME_SCALE_TYPE && values.length === 1) {
var attr0 = accessor0(data[0]);
return [attr0].concat(_toConsumableArray(values));
}
return values;
}
/**
* Get the distance, the smallest and the largest value of the domain.
* @param {Array} data Array of data for the single series.
* @param {Object} scaleObject Scale object.
* @returns {{domain0: number, domainN: number, distance: number}} Result.
* @private
*/
export function _getScaleDistanceAndAdjustedDomain(data, scaleObject) {
var domain = scaleObject.domain,
type = scaleObject.type,
accessor = scaleObject.accessor,
accessor0 = scaleObject.accessor0;
var uniqueValues = getUniquePropertyValues(data, accessor);
// Fix time scale if a data has only one value.
var values = _normalizeValues(data, uniqueValues, accessor0, type);
var index = _getSmallestDistanceIndex(values, scaleObject);
var adjustedDomain = [].concat(domain);
adjustedDomain[0] -= _computeLeftDomainAdjustment(values);
adjustedDomain[domain.length - 1] += _computeRightDomainAdjustment(values);
// Fix log scale if it's too small.
if (type === LOG_SCALE_TYPE && domain[0] <= 0) {
adjustedDomain[0] = Math.min(domain[1] / 10, 1);
}
var adjustedScaleFn = getScaleFnFromScaleObject(_extends({}, scaleObject, {
domain: adjustedDomain
}));
var distance = _computeScaleDistance(values, adjustedDomain, index, adjustedScaleFn);
return {
domain0: adjustedDomain[0],
domainN: adjustedDomain[adjustedDomain.length - 1],
distance: distance
};
}
/**
* Returns true if scale adjustments are possible for a given scale.
* @param {Object} props Props.
* @param {Object} scaleObject Scale object.
* @returns {boolean} True if scale adjustments possible.
* @private
*/
function _isScaleAdjustmentPossible(props, scaleObject) {
var attr = scaleObject.attr;
var _props$_adjustBy = props._adjustBy,
adjustBy = _props$_adjustBy === undefined ? [] : _props$_adjustBy,
_props$_adjustWhat = props._adjustWhat,
adjustWhat = _props$_adjustWhat === undefined ? [] : _props$_adjustWhat;
// The scale cannot be adjusted if there's no attributes to adjust, no
// suitable values
return adjustWhat.length && adjustBy.length && adjustBy.indexOf(attr) !== -1;
}
/**
* Adjust continuous scales (e.g. 'linear', 'log' and 'time') by adding the
* space from the left and right of them and by computing the best distance.
* @param {Object} props Props.
* @param {Object} scaleObject Scale object.
* @returns {*} Scale object with adjustments.
* @private
*/
function _adjustContinuousScale(props, scaleObject) {
var allSeriesData = props._allData,
_props$_adjustWhat2 = props._adjustWhat,
adjustWhat = _props$_adjustWhat2 === undefined ? [] : _props$_adjustWhat2;
// Assign the initial values.
var domainLength = scaleObject.domain.length;
var domain = scaleObject.domain;
var scaleDomain0 = domain[0];
var scaleDomainN = domain[domainLength - 1];
var scaleDistance = scaleObject.distance;
// Find the smallest left position of the domain, the largest right position
// of the domain and the best distance for them.
allSeriesData.forEach(function (data, index) {
if (adjustWhat.indexOf(index) === -1) {
return;
}
if (data && data.length) {
var _getScaleDistanceAndA = _getScaleDistanceAndAdjustedDomain(data, scaleObject),
domain0 = _getScaleDistanceAndA.domain0,
domainN = _getScaleDistanceAndA.domainN,
distance = _getScaleDistanceAndA.distance;
scaleDomain0 = Math.min(scaleDomain0, domain0);
scaleDomainN = Math.max(scaleDomainN, domainN);
scaleDistance = Math.max(scaleDistance, distance);
}
});
scaleObject.domain = [scaleDomain0].concat(_toConsumableArray(domain.slice(1, -1)), [scaleDomainN]);
scaleObject.distance = scaleDistance;
return scaleObject;
}
/**
* Get an adjusted scale. Suitable for 'category' and 'ordinal' scales.
* @param {Object} scaleObject Scale object.
* @returns {*} Scale object with adjustments.
* @private
*/
export function _adjustCategoricalScale(scaleObject) {
var scaleFn = getScaleFnFromScaleObject(scaleObject);
var domain = scaleObject.domain,
range = scaleObject.range;
if (domain.length > 1) {
scaleObject.distance = Math.abs(scaleFn(domain[1]) - scaleFn(domain[0]));
} else {
scaleObject.distance = Math.abs(range[1] - range[0]);
}
return scaleObject;
}
/**
* Retrieve a scale object or a value from the properties passed.
* @param {Object} props Object of props.
* @param {string} attr Attribute.
* @returns {*} Scale object, value or null.
*/
export function getScaleObjectFromProps(props, attr) {
// Create the initial scale object.
var scaleObject = _collectScaleObjectFromProps(props, attr);
if (!scaleObject) {
return null;
}
// Make sure if it's possible to add space to the scale object. If not,
// return the object immediately.
if (!_isScaleAdjustmentPossible(props, scaleObject)) {
return scaleObject;
}
var type = scaleObject.type;
// Depending on what type the scale is, apply different adjustments. Distances
// for the ordinal and category scales are even, equal domains cannot be
// adjusted.
if (type === ORDINAL_SCALE_TYPE || type === CATEGORY_SCALE_TYPE) {
return _adjustCategoricalScale(scaleObject);
}
return _adjustContinuousScale(props, scaleObject);
}
/**
* Get d3 scale for a given prop.
* @param {Object} props Props.
* @param {string} attr Attribute.
* @returns {function} d3 scale function.
*/
export function getAttributeScale(props, attr) {
var scaleObject = getScaleObjectFromProps(props, attr);
return getScaleFnFromScaleObject(scaleObject);
}
/**
* Get the value of `attr` from the object.
* @param {Object} d - data Object.
* @param {Function} accessor - accessor function.
* @returns {*} Value of the point.
* @private
*/
function _getAttrValue(d, accessor) {
return accessor(d.data ? d.data : d);
}
function _isDefined(value) {
return typeof value !== 'undefined';
}
/*
* Adds a percentage of padding to a given domain
* @param {Array} domain X or Y domain to pad.
* @param {Number} padding Percentage of padding to add to domain
* @returns {Array} Padded Domain
*/
function _padDomain(domain, padding) {
if (!domain) {
return domain;
}
if (isNaN(parseFloat(domain[0])) || isNaN(parseFloat(domain[1]))) {
return domain;
}
var _domain = _slicedToArray(domain, 2),
min = _domain[0],
max = _domain[1];
var domainPadding = (max - min) * (padding * 0.01);
return [min - domainPadding, max + domainPadding];
}
/**
* Get prop functor (either a value or a function) for a given attribute.
* @param {Object} props Series props.
* @param {Function} accessor - Property accessor.
* @returns {*} Function or value.
*/
export function getAttributeFunctor(props, attr) {
var scaleObject = getScaleObjectFromProps(props, attr);
if (scaleObject) {
var scaleFn = getScaleFnFromScaleObject(scaleObject);
return function (d) {
return scaleFn(_getAttrValue(d, scaleObject.accessor));
};
}
return null;
}
/**
* Get the functor which extracts value form [attr]0 property. Use baseValue if
* no attr0 property for a given object is defined. Fall back to domain[0] if no
* base value is available.
* @param {Object} props Object of props.
* @param {string} attr Attribute name.
* @returns {*} Function which returns value or null if no values available.
*/
export function getAttr0Functor(props, attr) {
var scaleObject = getScaleObjectFromProps(props, attr);
if (scaleObject) {
var domain = scaleObject.domain;
var _scaleObject$baseValu = scaleObject.baseValue,
baseValue = _scaleObject$baseValu === undefined ? domain[0] : _scaleObject$baseValu;
var scaleFn = getScaleFnFromScaleObject(scaleObject);
return function (d) {
var value = _getAttrValue(d, scaleObject.accessor0);
return scaleFn(_isDefined(value) ? value : baseValue);
};
}
return null;
}
/**
* Tries to get the string|number value of the attr and falls back to
* a fallback property in case if the value is a scale.
* @param {Object} props Series props.
* @param {string} attr Property name.
* @returns {*} Function or value.
*/
export function getAttributeValue(props, attr) {
var scaleObject = getScaleObjectFromProps(props, attr);
if (scaleObject) {
if (!scaleObject.isValue && props['_' + attr + 'Value'] === undefined) {
warning('[React-vis] Cannot use data defined ' + attr + ' for this ' + 'series type. Using fallback value instead.');
}
return props['_' + attr + 'Value'] || scaleObject.range[0];
}
return null;
}
/**
* Get prop types by the attribute.
* @param {string} attr Attribute.
* @returns {Object} Object of xDomain, xRange, xType, xDistance and _xValue,
* where x is an attribute passed to the function.
*/
export function getScalePropTypesByAttribute(attr) {
var _ref2;
return _ref2 = {}, _defineProperty(_ref2, '_' + attr + 'Value', PropTypes.any), _defineProperty(_ref2, attr + 'Domain', PropTypes.array), _defineProperty(_ref2, 'get' + toTitleCase(attr), PropTypes.func), _defineProperty(_ref2, 'get' + toTitleCase(attr) + '0', PropTypes.func), _defineProperty(_ref2, attr + 'Range', PropTypes.array), _defineProperty(_ref2, attr + 'Type', PropTypes.oneOf(Object.keys(SCALE_FUNCTIONS))), _defineProperty(_ref2, attr + 'Distance', PropTypes.number), _defineProperty(_ref2, attr + 'BaseValue', PropTypes.any), _ref2;
}
/**
* Extract the list of scale properties from the entire props object.
* @param {Object} props Props.
* @param {Array<String>} attributes Array of attributes for the given
* components (for instance, `['x', 'y', 'color']`).
* @returns {Object} Collected props.
*/
export function extractScalePropsFromProps(props, attributes) {
var result = {};
Object.keys(props).forEach(function (key) {
// this filtering is critical for extracting the correct accessors!
var attr = attributes.find(function (a) {
// width
var isPlainSet = key.indexOf(a) === 0;
// Ex: _data
var isUnderscoreSet = key.indexOf('_' + a) === 0;
// EX: getX
var usesGet = key.indexOf('get' + toTitleCase(a)) === 0;
return isPlainSet || isUnderscoreSet || usesGet;
});
if (!attr) {
return;
}
result[key] = props[key];
});
return result;
}
/**
* Extract the missing scale props from the given data and return them as
* an object.
* @param {Object} props Props.
* @param {Array} data Array of all data.
* @param {Array<String>} attributes Array of attributes for the given
* components (for instance, `['x', 'y', 'color']`).
* @returns {Object} Collected props.
*/
export function getMissingScaleProps(props, data, attributes) {
var result = {};
// Make sure that the domain is set pad it if specified
attributes.forEach(function (attr) {
if (!props['get' + toTitleCase(attr)]) {
result['get' + toTitleCase(attr)] = function (d) {
return d[attr];
};
}
if (!props['get' + toTitleCase(attr) + '0']) {
result['get' + toTitleCase(attr) + '0'] = function (d) {
return d[attr + '0'];
};
}
if (!props[attr + 'Domain']) {
result[attr + 'Domain'] = getDomainByAccessor(data, props['get' + toTitleCase(attr)] || result['get' + toTitleCase(attr)], props['get' + toTitleCase(attr) + '0'] || result['get' + toTitleCase(attr) + '0'], props[attr + 'Type']);
if (props[attr + 'Padding']) {
result[attr + 'Domain'] = _padDomain(result[attr + 'Domain'], props[attr + 'Padding']);
}
}
});
return result;
}
/**
* Return a d3 scale that returns the literal value that was given to it
* @returns {function} literal scale.
*/
export function literalScale(defaultValue) {
function scale(d) {
if (d === undefined) {
return defaultValue;
}
return d;
}
function response() {
return scale;
}
scale.domain = response;
scale.range = response;
scale.unknown = response;
scale.copy = response;
return scale;
}
export function getFontColorFromBackground(background) {
if (background) {
return hsl(background).l > 0.57 ? '#222' : '#fff';
}
return null;
}
/**
* Creates fallback values for series from scales defined at XYPlot level.
* @param {Object} props Props of the XYPlot object.
* @param {Array<Object>} children Array of components, children of XYPlot
* @returns {Array<Object>} Collected props.
*/
export function getXYPlotValues(props, children) {
var XYPlotScales = XYPLOT_ATTR.reduce(function (prev, attr) {
var domain = props[attr + 'Domain'],
range = props[attr + 'Range'],
type = props[attr + 'Type'];
if (domain && range && type) {
return _extends({}, prev, _defineProperty({}, attr, SCALE_FUNCTIONS[type]().domain(domain).range(range)));
}
return prev;
}, {});
return children.map(function (child) {
return XYPLOT_ATTR.reduce(function (prev, attr) {
if (child.props && child.props[attr] !== undefined) {
var scaleInput = child.props[attr];
var scale = XYPlotScales[attr];
var fallbackValue = scale ? scale(scaleInput) : scaleInput;
return _extends({}, prev, _defineProperty({}, '_' + attr + 'Value', fallbackValue));
}
return prev;
}, {});
});
}
var OPTIONAL_SCALE_PROPS = ['Padding'];
var OPTIONAL_SCALE_PROPS_REGS = OPTIONAL_SCALE_PROPS.map(function (str) {
return new RegExp(str + '$', 'i');
});
/**
* Get the list of optional scale-related settings for XYPlot
* mostly just used to find padding properties
* @param {Object} props Object of props.
* @returns {Object} Optional Props.
* @private
*/
export function getOptionalScaleProps(props) {
return Object.keys(props).reduce(function (acc, prop) {
var propIsNotOptional = OPTIONAL_SCALE_PROPS_REGS.every(function (reg) {
return !prop.match(reg);
});
if (propIsNotOptional) {
return acc;
}
acc[prop] = props[prop];
return acc;
}, {});
}
export default {
extractScalePropsFromProps: extractScalePropsFromProps,
getAttributeScale: getAttributeScale,
getAttributeFunctor: getAttributeFunctor,
getAttr0Functor: getAttr0Functor,
getAttributeValue: getAttributeValue,
getDomainByAccessor: getDomainByAccessor,
getFontColorFromBackground: getFontColorFromBackground,
getMissingScaleProps: getMissingScaleProps,
getOptionalScaleProps: getOptionalScaleProps,
getScaleObjectFromProps: getScaleObjectFromProps,
getScalePropTypesByAttribute: getScalePropTypesByAttribute,
getXYPlotValues: getXYPlotValues,
literalScale: literalScale
};