UNPKG

victory-chart

Version:
446 lines (405 loc) 20.6 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); var _omit2 = require("lodash/omit"); var _omit3 = _interopRequireDefault(_omit2); var _pick2 = require("lodash/pick"); var _pick3 = _interopRequireDefault(_pick2); var _defaults2 = require("lodash/defaults"); var _defaults3 = _interopRequireDefault(_defaults2); var _assign2 = require("lodash/assign"); var _assign3 = _interopRequireDefault(_assign2); 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 _react = require("react"); var _react2 = _interopRequireDefault(_react); var _point = require("./point"); var _point2 = _interopRequireDefault(_point); var _scale = require("../../helpers/scale"); var _scale2 = _interopRequireDefault(_scale); var _domain = require("../../helpers/domain"); var _domain2 = _interopRequireDefault(_domain); var _data = require("../../helpers/data"); var _data2 = _interopRequireDefault(_data); var _victoryCore = require("victory-core"); var _helperMethods = require("./helper-methods"); var _helperMethods2 = _interopRequireDefault(_helperMethods); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; } 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, stroke: "transparent", strokeWidth: 0 }, labels: { stroke: "transparent", fill: "#756f6a", fontFamily: "Helvetica", fontSize: 10, textAnchor: "middle", padding: 5 } }; var VictoryScatter = function (_React$Component) { _inherits(VictoryScatter, _React$Component); function VictoryScatter() { _classCallCheck(this, VictoryScatter); var _this = _possibleConstructorReturn(this, Object.getPrototypeOf(VictoryScatter).call(this)); _this.state = {}; _this.getEvents = _victoryCore.Helpers.getEvents.bind(_this); _this.getEventState = _victoryCore.Helpers.getEventState.bind(_this); return _this; } _createClass(VictoryScatter, [{ key: "getDataStyles", value: function getDataStyles(data, style) { var stylesFromData = (0, _omit3.default)(data, ["x", "y", "z", "size", "symbol", "name", "label"]); var baseDataStyle = (0, _defaults3.default)({}, stylesFromData, style); return _victoryCore.Helpers.evaluateStyle(baseDataStyle, data); } }, { key: "getLabelText", value: function getLabelText(props, datum, index) { var propsLabel = Array.isArray(props.labels) ? props.labels[index] : _victoryCore.Helpers.evaluateProp(props.labels, datum); return datum.label || propsLabel; } }, { key: "getLabelStyle", value: function getLabelStyle(labelStyle, dataProps) { var datum = dataProps.datum; var size = dataProps.size; var style = dataProps.style; var matchedStyle = (0, _pick3.default)(style, ["opacity", "fill"]); var padding = labelStyle.padding || size * 0.25; var baseLabelStyle = (0, _defaults3.default)({}, labelStyle, matchedStyle, { padding: padding }); return _victoryCore.Helpers.evaluateStyle(baseLabelStyle, datum); } }, { key: "renderData", value: function renderData(props, calculatedProps, style) { var _this2 = this; var dataEvents = this.getEvents(props.events.data, "data"); var labelEvents = this.getEvents(props.events.labels, "labels"); var scale = calculatedProps.scale; var data = calculatedProps.data; return data.map(function (datum, index) { var x = scale.x(datum.x); var y = scale.y(datum.y); var size = _helperMethods2.default.getSize(datum, props, calculatedProps); var symbol = _helperMethods2.default.getSymbol(datum, props); var dataStyle = _this2.getDataStyles(datum, style.data); var dataProps = (0, _defaults3.default)({}, _this2.getEventState(index, "data"), props.dataComponent.props, { x: x, y: y, size: size, scale: scale, datum: datum, symbol: symbol, index: index, style: dataStyle, key: "point-" + index }); var pointComponent = _react2.default.cloneElement(props.dataComponent, (0, _assign3.default)({}, dataProps, { events: _victoryCore.Helpers.getPartialEvents(dataEvents, index, dataProps) })); var text = _this2.getLabelText(props, dataProps.datum, index); if (text !== null && text !== undefined) { var labelStyle = _this2.getLabelStyle(style.labels, dataProps); var labelProps = (0, _defaults3.default)({}, _this2.getEventState(index, "labels"), props.labelComponent.props, { key: "point-label-" + index, style: labelStyle, x: x, y: y - labelStyle.padding, text: text, index: index, scale: scale, datum: dataProps.datum, textAnchor: labelStyle.textAnchor, verticalAnchor: labelStyle.verticalAnchor || "end", angle: labelStyle.angle }); var pointLabel = _react2.default.cloneElement(props.labelComponent, (0, _assign3.default)({ events: _victoryCore.Helpers.getPartialEvents(labelEvents, index, labelProps) }, labelProps)); return _react2.default.createElement( "g", { key: "point-group-" + index }, pointComponent, pointLabel ); } return pointComponent; }); } }, { key: "getCalculatedProps", value: function getCalculatedProps(props, style) { var data = _data2.default.getData(props); var range = { x: _victoryCore.Helpers.getRange(props, "x"), y: _victoryCore.Helpers.getRange(props, "y") }; var domain = { x: _domain2.default.getDomain(props, "x"), y: _domain2.default.getDomain(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 z = props.bubbleProperty || "z"; return { data: data, scale: scale, style: style, z: z }; } }, { key: "render", value: function render() { // If animating, return a `VictoryAnimation` element that will create // a new `VictoryScatter` with nearly identical props, except (1) tweened // and (2) `animate` set to null so we don't recurse forever. if (this.props.animate) { // Do less work by having `VictoryAnimation` tween only values that // make sense to tween. In the future, allow customization of animated // prop whitelist/blacklist? var whitelist = ["data", "domain", "height", "maxBubbleSize", "padding", "samples", "size", "style", "width", "x", "y"]; return _react2.default.createElement( _victoryCore.VictoryTransition, { animate: this.props.animate, animationWhitelist: whitelist }, _react2.default.createElement(VictoryScatter, this.props) ); } var style = _victoryCore.Helpers.getStyles(this.props.style, defaultStyles, "auto", "100%"); var calculatedProps = this.getCalculatedProps(this.props, style); var group = _react2.default.createElement( "g", { style: style.parent }, this.renderData(this.props, calculatedProps, 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 VictoryScatter; }(_react2.default.Component); VictoryScatter.role = "scatter"; VictoryScatter.defaultTransitions = { onExit: { duration: 600, before: function before() { return { opacity: 0 }; } }, onEnter: { duration: 600, before: function before() { return { opacity: 0 }; }, after: function after(datum) { return { opacity: datum.opacity || 1 }; } } }; VictoryScatter.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 bubbleProperty prop indicates which property of the data object should be used * to scale data points in a bubble chart */ bubbleProperty: _react.PropTypes.string, /** * The data prop specifies the data to be plotted. * Data should be in the form of an array of data points. * 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. * Other properties may be added to the data point object, such as fill, size, and symbol. * These properties will be interpreted and applied to the individual lines * @examples [{x: 1, y: 2, fill: "red"}, {x: 2, y: 3, label: "foo"}] */ data: _react.PropTypes.array, /** * The dataComponent prop takes an entire component which will be used to create points for * each datum in the chart. The new element created from the passed dataComponent will be * provided with the following properties calculated by VictoryScatter: datum, index, scale, * style, events, x, y, size, and symbol. Any of these props may be overridden by passing in * props to the supplied component, or modified or ignored within the custom component itself. * If a dataComponent is not provided, VictoryScatter will use its default Point component. */ dataComponent: _react.PropTypes.element, /** * The domain prop describes the range of values your chart will include. This prop can be * given as a array of the minimum and maximum expected values for your 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 index and namespace on the state object of VictoryScatter * i.e. `this.state[index].data = {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 VictoryScatter * if one exists. If VictoryScatter 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: () => return {data: {style: {fill: "green"}}, labels: {style: {fill: "black"}}} *}} */ 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 labelComponent prop takes in an entire label component which will be used * to create labels for each point in the scatter. The new element created from * the passed labelComponent will be supplied with the following properties: * x, y, index, datum, verticalAnchor, textAnchor, angle, style, text, and events. * any of these props may be overridden by passing in props to the supplied component, * or modified or ignored within the custom component itself. If labelComponent is omitted, * a new VictoryLabel will be created with props described above. */ labelComponent: _react.PropTypes.element, /** * The labels prop defines labels that will appear with each point in your chart. * This prop should be given as an array of values or as a function of data. * If given as an array, the number of elements in the array should be equal to * the length of the data array. Labels may also be added directly to the data object * like data={[{x: 1, y: 1, label: "first"}]}. * @examples: ["spring", "summer", "fall", "winter"], (datum) => datum.title */ labels: _react.PropTypes.oneOfType([_react.PropTypes.func, _react.PropTypes.array]), /** * The maxBubbleSize prop sets an upper limit for scaling data points in a bubble chart */ maxBubbleSize: _victoryCore.PropTypes.nonNegative, /** * 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 size prop determines how to scale each data point */ size: _react.PropTypes.oneOfType([_victoryCore.PropTypes.nonNegative, _react.PropTypes.func]), /** * 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 VictoryScatter with other components within an enclosing <svg> tag. */ standalone: _react.PropTypes.bool, /** * The style prop specifies styles for your VictoryScatter. 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. In addition to normal style properties, angle and verticalAnchor * may also be specified via the labels object, and they will be passed as props to * VictoryLabel, or any custom labelComponent. * @examples {data: {fill: "red"}, labels: {fontSize: 12}} */ style: _react.PropTypes.shape({ parent: _react.PropTypes.object, data: _react.PropTypes.object, labels: _react.PropTypes.object }), /** * The symbol prop determines which symbol should be drawn to represent data points. */ symbol: _react.PropTypes.oneOfType([_react.PropTypes.oneOf(["circle", "diamond", "plus", "square", "star", "triangleDown", "triangleUp"]), _react.PropTypes.func]), /** * 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)]) }; VictoryScatter.defaultProps = { events: {}, height: 300, padding: 50, samples: 50, scale: "linear", size: 3, standalone: true, symbol: "circle", width: 450, x: "x", y: "y", dataComponent: _react2.default.createElement(_point2.default, null), labelComponent: _react2.default.createElement(_victoryCore.VictoryLabel, null) }; VictoryScatter.getDomain = _domain2.default.getDomain.bind(_domain2.default); VictoryScatter.getData = _data2.default.getData.bind(_data2.default); exports.default = VictoryScatter;