Skip to content

Commit

Permalink
Refactor the data flow inside of Scaler
Browse files Browse the repository at this point in the history
Clean up the internal data flow inside of the Scaler component. This
refactoring had a few main goals:

  1. Fully-own the responsibility of providing scaled data downstream
     alongside the Series and Collection items.
  2. Increase performance by using better checks to detect when the
     incoming Series and Collections objects were externally changed.
  3. Stop exposing domainsByItemId / subDomainsByItemId to the
     downstream rendering components, in deference to Series and
     Collection objects directly.
  4. Get closer toward enabling per-Series timeDomains / timeSubDomains
     as documented in #301.

The end result of this is that the (internal) API has nontrivially
changed. Instead of Scaler exposing maps of domainsByItemId and
subDomainByItemId, it will simply pass through the series and
collections arrays, but they are now all guaranteed to be
fully-populated with domain and subDomain information. This means that
rendering components can simply render using the timeDomain,
timeSubDomain, xDomain, xSubDomain, yDomain, and ySubDomain domains
(which are guaranteed to be present) and don't need to concern
themselves with how they were populated (user/state, props, calculated
from data, or placeholders).

This does not yet fully enable separate timeDomain / timeSubdomain
throughout the library, but it removes almost all blockers. The final
piece to this puzzle is in DataProvider -- it needs to learn a one more
trick in order to support this (it currently only knows about one
timeSubDomain). That cleanup will be saved for a future PR.
  • Loading branch information
Evan Charlton committed May 30, 2019
1 parent 2cf0d8b commit 5aec1d8
Show file tree
Hide file tree
Showing 20 changed files with 1,071 additions and 710 deletions.
23 changes: 4 additions & 19 deletions src/components/AxisCollection/YAxis.js
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,7 @@ import PropTypes from 'prop-types';
import { createYScale } from '../../utils/scale-helpers';
import GriffPropTypes, { singleSeriePropType } from '../../utils/proptypes';
import AxisPlacement from '../AxisPlacement';
import ScalerContext from '../../context/Scaler';
import ZoomRect from '../ZoomRect';
import Axes from '../../utils/Axes';
import { withDisplayName } from '../../utils/displayName';

const propTypes = {
zoomable: PropTypes.bool,
Expand All @@ -23,9 +20,6 @@ const propTypes = {
tickFormatter: PropTypes.func.isRequired,
defaultColor: PropTypes.string,
ticks: PropTypes.number,

// These are populated by Griff.
subDomainsByItemId: GriffPropTypes.subDomainsByItemId.isRequired,
};

const defaultProps = {
Expand All @@ -39,8 +33,6 @@ const defaultProps = {
ticks: 0,
};

const getItem = (series, collection) => series || collection;

const getLineProps = ({
strokeWidth,
tickSizeInner,
Expand Down Expand Up @@ -156,16 +148,15 @@ const YAxis = ({
onMouseEnter,
onMouseLeave,
series,
subDomainsByItemId,
tickFormatter,
ticks,
width,
yAxisPlacement,
zoomable,
}) => {
const item = getItem(series, collection);
const item = series || collection;
const color = item.color || defaultColor;
const scale = createYScale(Axes.y(subDomainsByItemId[item.id]), height);
const scale = createYScale(item.ySubDomain, height);
const axis = d3.axisRight(scale);
const tickFontSize = 14;
const strokeWidth = 2;
Expand Down Expand Up @@ -243,7 +234,7 @@ const YAxis = ({
width={width}
height={height}
zoomAxes={{ y: true }}
itemIds={[getItem(series, collection).id]}
itemIds={[(series || collection).id]}
/>
)}
</g>
Expand All @@ -253,10 +244,4 @@ const YAxis = ({
YAxis.propTypes = propTypes;
YAxis.defaultProps = defaultProps;

export default withDisplayName('YAxis', props => (
<ScalerContext.Consumer>
{({ subDomainsByItemId }) => (
<YAxis {...props} subDomainsByItemId={subDomainsByItemId} />
)}
</ScalerContext.Consumer>
));
export default YAxis;
139 changes: 107 additions & 32 deletions src/components/ContextChart/index.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
import React from 'react';
import React, { useMemo } from 'react';
import PropTypes from 'prop-types';
import { SizeMe } from 'react-sizeme';
import DataContext from '../../context/Data';
import ScalerContext from '../../context/Scaler';
import LineCollection from '../LineCollection';
import XAxis from '../XAxis';
Expand All @@ -11,9 +12,8 @@ import AxisPlacement from '../AxisPlacement';
import { multiFormat } from '../../utils/multiFormat';
import Axes from '../../utils/Axes';
import { createYScale, createXScale } from '../../utils/scale-helpers';
import { firstResolvedDomain } from '../Scaler';
import { calculateDomainFromData } from '../DataProvider';
import { withDisplayName } from '../../utils/displayName';
import { calculateDomains } from '../Scaler';

const propTypes = {
height: PropTypes.number,
Expand All @@ -25,9 +25,9 @@ const propTypes = {
xAxisPlacement: GriffPropTypes.axisPlacement,

// These are all provided by Griff.
domainsByItemId: GriffPropTypes.domainsByItemId.isRequired,
collections: GriffPropTypes.collections.isRequired,
unscaledSeries: GriffPropTypes.multipleSeries.isRequired,
series: GriffPropTypes.multipleSeries.isRequired,
subDomainsByItemId: GriffPropTypes.subDomainsByItemId.isRequired,
updateDomains: GriffPropTypes.updateDomains.isRequired,
width: PropTypes.number,
};
Expand Down Expand Up @@ -76,12 +76,55 @@ const renderXAxis = (position, xAxis, { xAxisPlacement }) => {
return null;
};

const domainToString = domain => {
if (!domain) {
return 'undefined';
}
if (!Array.isArray(domain)) {
return 'not-an-array';
}
return `[${domain[0]},${domain[1]}]`;
};

const dataRange = item => {
const { data, timeAccessor } = item;
if (!data) {
return 'undefined';
}
if (!Array.isArray(data)) {
return 'not-an-array';
}
if (!timeAccessor) {
return 'inaccessible';
}
if (data.length === 0) {
return 'no-data';
}
return `[${timeAccessor(data[0])},${timeAccessor(data[data.length - 1])}]`;
};

// A helper function to provide checksum-ish hashes to React.useMemo so that we
// can only recompute the domains when relevant information changes.
const getDomainHashes = (...items) =>
items.map(itemGroup =>
itemGroup
.map(
item =>
`${item.id}: { data: { ${(item.data || []).length} [${dataRange(
item
)}] }, collectionId: ${item.collectionId}, yDomain: ${domainToString(
item.yDomain
)}, ySubDomain: ${domainToString(item.ySubDomain)} }`
)
.join(', ')
);

const ContextChart = ({
annotations: propsAnnotations,
domainsByItemId,
height: propsHeight,
unscaledSeries,
collections,
series,
subDomainsByItemId,
updateDomains,
width,
xAxisFormatter,
Expand All @@ -93,19 +136,48 @@ const ContextChart = ({
return null;
}

const getYScale = (s, height) => {
const domain =
firstResolvedDomain(
s.yDomain,
Axes.y(domainsByItemId[s.collectionId || s.id])
) ||
calculateDomainFromData(s.data, s.yAccessor, s.y0Accessor, s.y1Accessor);
const reconciledDomains = useMemo(() => {
// First things first: figure out what domain each series wants to have.
const domainsByItemId = {};
unscaledSeries.forEach(s => {
const { collectionId, id } = s;

const domain = s.yDomain || calculateDomains(s).y;

domainsByItemId[id] = domain;

if (collectionId) {
const collectedDomain = domainsByItemId[collectionId] || [
Number.MAX_SAFE_INTEGER,
Number.MIN_SAFE_INTEGER,
];
domainsByItemId[collectionId] = [
Math.min(domain[0], collectedDomain[0]),
Math.max(domain[1], collectedDomain[1]),
];
}
});

// Do another pass over it to update the collected items' domains.
unscaledSeries.forEach(s => {
const { id, collectionId } = s;
if (!collectionId) {
return;
}

domainsByItemId[id] = domainsByItemId[collectionId];
});

return domainsByItemId;
}, getDomainHashes(unscaledSeries, collections));

const getYScale = (seriesIndex, height) => {
const scaled = series[seriesIndex];
const domain = reconciledDomains[scaled.id];
return createYScale(domain, height);
};

const firstItemId = series[0].id;
const timeDomain = Axes.time(domainsByItemId[firstItemId]);
const timeSubDomain = Axes.time(subDomainsByItemId[firstItemId]);
const { timeDomain, timeSubDomain } = series[0];
const height = getChartHeight({
height: propsHeight,
xAxisHeight,
Expand Down Expand Up @@ -144,7 +216,6 @@ const ContextChart = ({
yScalerFactory={getYScale}
scaleY={false}
scaleX={false}
subDomainsByItemId={subDomainsByItemId}
/>
<Brush
width={width}
Expand All @@ -170,20 +241,24 @@ ContextChart.propTypes = propTypes;
ContextChart.defaultProps = defaultProps;

export default withDisplayName('ContextChart', props => (
<ScalerContext.Consumer>
{({ domainsByItemId, subDomainsByItemId, updateDomains, series }) => (
<SizeMe monitorWidth>
{({ size }) => (
<ContextChart
width={size.width}
series={series}
{...props}
subDomainsByItemId={subDomainsByItemId}
domainsByItemId={domainsByItemId}
updateDomains={updateDomains}
/>
<DataContext.Consumer>
{({ series: unscaledSeries }) => (
<ScalerContext.Consumer>
{({ updateDomains, series, collections }) => (
<SizeMe monitorWidth>
{({ size }) => (
<ContextChart
width={size.width}
unscaledSeries={unscaledSeries}
collections={collections}
series={series}
{...props}
updateDomains={updateDomains}
/>
)}
</SizeMe>
)}
</SizeMe>
</ScalerContext.Consumer>
)}
</ScalerContext.Consumer>
</DataContext.Consumer>
));
Loading

0 comments on commit 5aec1d8

Please sign in to comment.