Skip to content
This repository has been archived by the owner on Feb 19, 2022. It is now read-only.

Commit

Permalink
Merge pull request #503 from FormidableLabs/downsample-zoom
Browse files Browse the repository at this point in the history
add downsample prop to VictoryZoomContainer
  • Loading branch information
boygirl authored Aug 2, 2017
2 parents 9322555 + 8d2d7de commit 5a32ceb
Show file tree
Hide file tree
Showing 2 changed files with 40 additions and 4 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -35,7 +35,7 @@
"builder-victory-component": "^5.0.1",
"d3-voronoi": "^1.1.2",
"lodash": "^4.17.4",
"victory-core": "^17.0.0"
"victory-core": "^17.1.0"
},
"devDependencies": {
"builder-victory-component-dev": "^5.0.1",
Expand Down
42 changes: 39 additions & 3 deletions src/components/containers/victory-zoom-container.js
Original file line number Diff line number Diff line change
@@ -1,8 +1,12 @@
import PropTypes from "prop-types";
import React from "react";
import { defaults, isEqual } from "lodash";
import { defaults, isEqual, get } from "lodash";
import ZoomHelpers from "./zoom-helpers";
import { VictoryContainer, VictoryClipContainer, PropTypes as CustomPropTypes } from "victory-core";
import {
VictoryContainer, VictoryClipContainer, Data, PropTypes as CustomPropTypes
} from "victory-core";

const DEFAULT_DOWNSAMPLE = 150;

export const zoomContainerMixin = (base) => class VictoryZoomContainer extends base {
static displayName = "VictoryZoomContainer";
Expand All @@ -12,6 +16,10 @@ export const zoomContainerMixin = (base) => class VictoryZoomContainer extends b
allowZoom: PropTypes.bool,
clipContainerComponent: PropTypes.element.isRequired,
dimension: PropTypes.oneOf(["x", "y"]),
downsample: PropTypes.oneOfType([
PropTypes.bool,
PropTypes.number
]),
minimumZoom: PropTypes.shape({
x: PropTypes.number,
y: PropTypes.number
Expand Down Expand Up @@ -128,6 +136,30 @@ export const zoomContainerMixin = (base) => class VictoryZoomContainer extends b
};
}

downsampleZoomData(props, childProps, domain) {
const { downsample } = props;
const rawData = get(childProps, "data");
// return undefined if downsample is not run, then default() will replace with child.props.data
if (!downsample || !rawData || !domain) { return undefined; }

// if data accessors are not used, skip calling expensive Data.formatData
const data = (childProps.x || childProps.y) ? Data.formatData(rawData, childProps) : rawData;
const maxPoints = (downsample === true) ? DEFAULT_DOWNSAMPLE : downsample;
const dimension = props.dimension || "x";

// important: assumes data is ordered by dimension
// get the start and end of the data that is in the current visible domain
let startIndex = data.findIndex((d) => d[dimension] >= domain[dimension][0]);
let endIndex = data.findIndex((d) => d[dimension] > domain[dimension][1]);
// pick one more point (if available) at each end so that VictoryLine, VictoryArea connect
if (startIndex !== 0) { startIndex -= 1; }
if (endIndex !== -1) { endIndex += 1; }

const visibleData = data.slice(startIndex, endIndex);

return Data.downsample(visibleData, maxPoints, startIndex);
}

modifyChildren(props) {
const childComponents = React.Children.toArray(props.children);

Expand Down Expand Up @@ -158,7 +190,11 @@ export const zoomContainerMixin = (base) => class VictoryZoomContainer extends b
};
}
return React.cloneElement(
child, defaults({ domain: newDomain }, child.props)
child,
defaults({
domain: newDomain,
data: this.downsampleZoomData(props, child.props, newDomain)
}, child.props)
);
});
}
Expand Down

0 comments on commit 5a32ceb

Please sign in to comment.