victory-chart
Version:
Chart Component for Victory
387 lines (352 loc) • 17.4 kB
JavaScript
"use strict";
Object.defineProperty(exports, "__esModule", {
value: true
});
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 _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
var _last = require("lodash/last");
var _last2 = _interopRequireDefault(_last);
var _assign = require("lodash/assign");
var _assign2 = _interopRequireDefault(_assign);
var _react = require("react");
var _react2 = _interopRequireDefault(_react);
var _data = require("../../helpers/data");
var _data2 = _interopRequireDefault(_data);
var _domain = require("../../helpers/domain");
var _domain2 = _interopRequireDefault(_domain);
var _scale = require("../../helpers/scale");
var _scale2 = _interopRequireDefault(_scale);
var _victoryCore = require("victory-core");
var _area = require("./area");
var _area2 = _interopRequireDefault(_area);
var _areaLabel = require("./area-label");
var _areaLabel2 = _interopRequireDefault(_areaLabel);
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
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 _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
function _possibleConstructorReturn(self, call) { if (!self) { throw new ReferenceError("this hasn't been initialised - super() hasn't been called"); } return call && (typeof call === "object" || typeof call === "function") ? call : self; }
function _inherits(subClass, superClass) { if (typeof superClass !== "function" && superClass !== null) { throw new TypeError("Super expression must either be null or a function, not " + typeof superClass); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, enumerable: false, writable: true, configurable: true } }); if (superClass) Object.setPrototypeOf ? Object.setPrototypeOf(subClass, superClass) : subClass.__proto__ = superClass; }
var defaultStyles = {
data: {
fill: "#756f6a",
opacity: 1
},
labels: {
fontSize: 12,
padding: 4,
fill: "black"
}
};
var VictoryArea = function (_React$Component) {
_inherits(VictoryArea, _React$Component);
function VictoryArea() {
_classCallCheck(this, VictoryArea);
return _possibleConstructorReturn(this, Object.getPrototypeOf(VictoryArea).apply(this, arguments));
}
_createClass(VictoryArea, [{
key: "componentWillMount",
value: function componentWillMount() {
this.state = {
dataState: {},
labelsState: {}
};
}
}, {
key: "getBaseline",
value: function getBaseline(calculatedProps) {
var data = calculatedProps.data;
var domain = calculatedProps.domain;
var minY = Math.min.apply(Math, _toConsumableArray(domain.y)) > 0 ? Math.min.apply(Math, _toConsumableArray(domain.y)) : 0;
return data.map(function (datum) {
var y0 = datum.yOffset || minY;
return (0, _assign2.default)({ y0: y0 }, datum);
});
}
}, {
key: "renderArea",
value: function renderArea(props, calculatedProps) {
var scale = calculatedProps.scale;
var style = calculatedProps.style;
var interpolation = props.interpolation;
var events = props.events;
var label = props.label;
var labelComponent = props.labelComponent;
var dataComponent = props.dataComponent;
var getBoundEvents = _victoryCore.Helpers.getEvents.bind(this);
var dataEvents = getBoundEvents(events.data, "data");
var data = this.getBaseline(calculatedProps);
var areaProps = (0, _assign2.default)({ scale: scale, interpolation: interpolation, data: data, events: dataEvents, style: style.data }, this.state.dataState[0]);
var areaComponent = _react2.default.cloneElement(dataComponent, areaProps);
if (label) {
var lastData = (0, _last2.default)(data);
var position = {
x: scale.x.call(this, lastData.x),
y: scale.y.call(this, lastData.y + lastData.y0)
};
return _react2.default.createElement(
"g",
null,
areaComponent,
_react2.default.createElement(_areaLabel2.default, _extends({
style: style.labels,
data: data,
events: getBoundEvents(events.labels, "labels"),
position: position,
labelText: label,
labelComponent: labelComponent
}, this.state.labelsState[0]))
);
}
return areaComponent;
}
}, {
key: "renderData",
value: function renderData(props, style) {
var data = _data2.default.getData(props);
var range = {
x: _victoryCore.Helpers.getRange(props, "x"),
y: _victoryCore.Helpers.getRange(props, "y")
};
var padding = _victoryCore.Helpers.getPadding(props);
var domain = {
x: _domain2.default.getDomainWithZero(props, "x"),
y: _domain2.default.getDomainWithZero(props, "y")
};
var scale = {
x: _scale2.default.getBaseScale(props, "x").domain(domain.x).range(range.x),
y: _scale2.default.getBaseScale(props, "y").domain(domain.y).range(range.y)
};
var calculatedProps = {
style: style, data: data, domain: domain, scale: scale, padding: padding
};
return this.renderArea(props, calculatedProps);
}
}, {
key: "render",
value: function render() {
if (this.props.animate) {
var whitelist = ["data", "domain", "height", "padding", "style", "width"];
return _react2.default.createElement(
_victoryCore.VictoryTransition,
{ animate: this.props.animate, animationWhitelist: whitelist },
_react2.default.createElement(VictoryArea, this.props)
);
}
var style = _victoryCore.Helpers.getStyles(this.props.style, defaultStyles, "auto", "100%");
var group = _react2.default.createElement(
"g",
{ style: style.parent },
this.renderData(this.props, style)
);
return this.props.standalone ? _react2.default.createElement(
"svg",
_extends({
style: style.parent,
viewBox: "0 0 " + this.props.width + " " + this.props.height
}, this.props.events.parent),
group
) : group;
}
}]);
return VictoryArea;
}(_react2.default.Component);
VictoryArea.role = "area";
VictoryArea.defaultTransitions = {
onExit: {
duration: 500,
before: function before() {
return { y: 0, yOffset: 0 };
}
},
onEnter: {
duration: 500,
before: function before() {
return { y: 0, yOffset: 0, xOffset: 0 };
},
after: function after(datum) {
return { y: datum.y, yOffset: datum.yOffset, xOffset: datum.xOffset };
}
}
};
VictoryArea.propTypes = {
/**
* The animate prop specifies props for VictoryAnimation to use. The animate prop should
* also be used to specify enter and exit transition configurations with the `onExit`
* and `onEnter` namespaces respectively.
* @examples {duration: 500, onEnd: () => {}, onEnter: {duration: 500, before: () => ({y: 0})})}
*/
animate: _react.PropTypes.object,
/**
* The categories prop specifies how categorical data for a chart should be ordered.
* This prop should be given as an array of string values, or an object with
* these arrays of values specified for x and y. If this prop is not set,
* categorical data will be plotted in the order it was given in the data array
* @examples ["dogs", "cats", "mice"]
*/
categories: _react.PropTypes.oneOfType([_react.PropTypes.arrayOf(_react.PropTypes.string), _react.PropTypes.shape({
x: _react.PropTypes.arrayOf(_react.PropTypes.string),
y: _react.PropTypes.arrayOf(_react.PropTypes.string)
})]),
/**
* The data prop specifies the data to be plotted. Data should be in the form of an array
* of data points, or an array of arrays of data points for multiple datasets.
* Each data point may be any format you wish (depending on the `x` and `y` accessor props),
* but by default, an object with x and y properties is expected.
* @examples [{x: 1, y: 2}, {x: 2, y: 3}], [[1, 2], [2, 3]],
* [[{x: "a", y: 1}, {x: "b", y: 2}], [{x: "a", y: 2}, {x: "b", y: 3}]]
*/
data: _react.PropTypes.array,
/**
* The dataComponent prop takes an entire, HTML-complete data component which will be used to
* create an area. The new element created from the passed dataComponent will be provided
* with the following properties calculated by VictoryArea: a scale object, an array of
* modified data objects (including x, y, and calculated y0), interpolation, style, and events
* If a dataComponent is not provided, VictoryArea will use its default Area component.
*/
dataComponent: _react.PropTypes.element,
/**
* The domain prop describes the range of values your bar chart will cover. This prop can be
* given as a array of the minimum and maximum expected values for your bar chart,
* or as an object that specifies separate arrays for x and y.
* If this prop is not provided, a domain will be calculated from data, or other
* available information.
* @examples [-1, 1], {x: [0, 100], y: [0, 1]}
*/
domain: _react.PropTypes.oneOfType([_victoryCore.PropTypes.domain, _react.PropTypes.shape({
x: _victoryCore.PropTypes.domain,
y: _victoryCore.PropTypes.domain
})]),
/**
* The events prop attaches arbitrary event handlers to data and label elements
* Event handlers are called with their corresponding events, corresponding component props,
* and their index in the data array, and event name. The return value of event handlers
* will be stored by unique index on the state object of VictoryArea
* i.e. `this.state.dataState[dataIndex] = {style: {fill: "red"}...}`, and will be
* applied by index to the appropriate child component. Event props on the
* parent namespace are just spread directly on to the top level svg of VictoryArea
* if one exists. If VictoryArea is set up to render g elements i.e. when it is
* rendered within chart, or when `standalone={false}` parent events will not be applied.
*
* @examples {data: {
* onClick: () => onClick: () => return {style: {fill: "green"}}
*}}
*/
events: _react.PropTypes.shape({
data: _react.PropTypes.object,
labels: _react.PropTypes.object,
parent: _react.PropTypes.object
}),
/**
* The height props specifies the height the svg viewBox of the chart container.
* This value should be given as a number of pixels
*/
height: _victoryCore.PropTypes.nonNegative,
/**
* The interpolation prop determines how data points should be connected
* when plotting a line
*/
interpolation: _react.PropTypes.oneOf(["basis", "basisClosed", "basisOpen", "bundle", "cardinal", "cardinalClosed", "cardinalOpen", "catmullRom", "catmullRomClosed", "catmullRomOpen", "linear", "linearClosed", "monotoneX", "monotoneY", "natural", "radial", "step", "stepAfter", "stepBefore"]),
/**
* The label prop defines labels that will appear at the edge of each area. This prop
* should be given as a string
*/
label: _react.PropTypes.string,
/**
* The labelComponent prop takes in an entire, HTML-complete label
* component which will be used to create labels for each area in the
* chart. The new element created from the passed labelComponent will have
* children preserved, or provided via the labels array, textAnchor, and verticalAnchor
* preserved or default values provided by defaults; and styles filled out with defaults
* provided by the style prop, and dataAttributes prop. If labelComponent is omitted,
* but a labels array is specified, a new VictoryLabel will be created.
*/
labelComponent: _react.PropTypes.element,
/**
* The padding props specifies the amount of padding in number of pixels between
* the edge of the chart and any rendered child components. This prop can be given
* as a number or as an object with padding specified for top, bottom, left
* and right.
*/
padding: _react.PropTypes.oneOfType([_react.PropTypes.number, _react.PropTypes.shape({
top: _react.PropTypes.number,
bottom: _react.PropTypes.number,
left: _react.PropTypes.number,
right: _react.PropTypes.number
})]),
/**
* The samples prop specifies how many individual points to plot when plotting
* y as a function of x. Samples is ignored if x props are provided instead.
*/
samples: _victoryCore.PropTypes.nonNegative,
/**
* The scale prop determines which scales your chart should use. This prop can be
* given as a string specifying a supported scale ("linear", "time", "log", "sqrt"),
* as a d3 scale function, or as an object with scales specified for x and y
* @exampes d3Scale.time(), {x: "linear", y: "log"}
*/
scale: _react.PropTypes.oneOfType([_victoryCore.PropTypes.scale, _react.PropTypes.shape({
x: _victoryCore.PropTypes.scale,
y: _victoryCore.PropTypes.scale
})]),
/**
* The standalone prop determines whether the component will render a standalone svg
* or a <g> tag that will be included in an external svg. Set standalone to false to
* compose VictoryBar with other components within an enclosing <svg> tag.
*/
standalone: _react.PropTypes.bool,
/**
* The style prop specifies styles for your VictoryArea. Any valid inline style properties
* will be applied. Height, width, and padding should be specified via the height,
* width, and padding props, as they are used to calculate the alignment of
* components within chart.
* @examples {data: {fill: "red"}, labels: {fontSize: 12}}
*/
style: _react.PropTypes.shape({
parent: _react.PropTypes.object,
data: _react.PropTypes.object,
labels: _react.PropTypes.object
}),
/**
* The width props specifies the width of the svg viewBox of the chart container
* This value should be given as a number of pixels
*/
width: _victoryCore.PropTypes.nonNegative,
/**
* The x prop specifies how to access the X value of each data point.
* If given as a function, it will be run on each data point, and returned value will be used.
* If given as an integer, it will be used as an array index for array-type data points.
* If given as a string, it will be used as a property key for object-type data points.
* If given as an array of strings, or a string containing dots or brackets,
* it will be used as a nested object property path (for details see Lodash docs for _.get).
* If `null` or `undefined`, the data value will be used as is (identity function/pass-through).
* @examples 0, 'x', 'x.value.nested.1.thing', 'x[2].also.nested', null, d => Math.sin(d)
*/
x: _react.PropTypes.oneOfType([_react.PropTypes.func, _victoryCore.PropTypes.allOfType([_victoryCore.PropTypes.integer, _victoryCore.PropTypes.nonNegative]), _react.PropTypes.string, _react.PropTypes.arrayOf(_react.PropTypes.string)]),
/**
* The y prop specifies how to access the Y value of each data point.
* If given as a function, it will be run on each data point, and returned value will be used.
* If given as an integer, it will be used as an array index for array-type data points.
* If given as a string, it will be used as a property key for object-type data points.
* If given as an array of strings, or a string containing dots or brackets,
* it will be used as a nested object property path (for details see Lodash docs for _.get).
* If `null` or `undefined`, the data value will be used as is (identity function/pass-through).
* @examples 0, 'y', 'y.value.nested.1.thing', 'y[2].also.nested', null, d => Math.sin(d)
*/
y: _react.PropTypes.oneOfType([_react.PropTypes.func, _victoryCore.PropTypes.allOfType([_victoryCore.PropTypes.integer, _victoryCore.PropTypes.nonNegative]), _react.PropTypes.string, _react.PropTypes.arrayOf(_react.PropTypes.string), _react.PropTypes.arrayOf(_react.PropTypes.func)])
};
VictoryArea.defaultProps = {
dataComponent: _react2.default.createElement(_area2.default, null),
events: {},
height: 300,
padding: 50,
scale: "linear",
samples: 50,
standalone: true,
interpolation: "linear",
width: 450,
x: "x",
y: "y"
};
VictoryArea.getDomain = _domain2.default.getDomainWithZero.bind(_domain2.default);
VictoryArea.getData = _data2.default.getData.bind(_data2.default);
exports.default = VictoryArea;