Newer
Older
/**
* Hydrograph charting module.
*/
Yan, Andrew N.
committed
const { extent } = require('d3-array');
Naab, Daniel James
committed
const { line: d3Line } = require('d3-shape');
const { createStructuredSelector } = require('reselect');
const { addSVGAccessibility, addSROnlyTable } = require('../../accessibility');
const { USWDS_MEDIUM_SCREEN, USWDS_SMALL_SCREEN, STATIC_URL } = require('../../config');
Naab, Daniel James
committed
const { dispatch, link, provide } = require('../../lib/redux');
const { Actions } = require('../../store');
const { mediaQuery } = require('../../utils');
const { audibleUI } = require('./audible');
Naab, Daniel James
committed
const { appendAxes, axesSelector } = require('./axes');
const { cursorSlider } = require('./cursor');
Bucknell, Mary S.
committed
const { lineSegmentsByParmCdSelector, currentVariableLineSegmentsSelector,
Naab, Daniel James
committed
const { CIRCLE_RADIUS, CIRCLE_RADIUS_SINGLE_PT, SPARK_LINE_DIM, layoutSelector } = require('./layout');
const { drawSimpleLegend, legendMarkerRowsSelector } = require('./legend');
const { plotSeriesSelectTable, availableTimeseriesSelector } = require('./parameters');
const { xScaleSelector, yScaleSelector, timeSeriesScalesByParmCdSelector } = require('./scales');
const { allTimeSeriesSelector, isVisibleSelector, titleSelector,
descriptionSelector, currentVariableTimeSeriesSelector, timeSeriesSelector } = require('./timeseries');
Bucknell, Mary S.
committed
const { createTooltipFocus, createTooltipText } = require('./tooltip');
const { getCurrentVariable } = require('../../selectors/timeseriesSelector');
const drawMessage = function (elem, message) {
// Set up parent element and SVG
elem.innerHTML = '';
const alertBox = elem
.append('div')
.attr('class', 'usa-alert usa-alert-warning')
.append('div')
.attr('class', 'usa-alert-body');
alertBox
.append('h3')
.attr('class', 'usa-alert-heading')
.html('Hydrograph Alert');
alertBox
.append('p')
const plotDataLine = function (elem, {visible, lines, tsKey, xScale, yScale}) {
Naab, Daniel James
committed
if (!visible) {
Naab, Daniel James
committed
for (let line of lines) {
Naab, Daniel James
committed
// If this is a single point line, then represent it as a circle.
// Otherwise, render as a line.
if (line.points.length === 1) {
elem.append('circle')
.data(line.points)
.classed('line-segment', true)
.classed('approved', line.classes.approved)
.classed('estimated', line.classes.estimated)
.attr('r', CIRCLE_RADIUS_SINGLE_PT)
.attr('cx', d => xScale(d.dateTime))
.attr('cy', d => yScale(d.value));
} else {
const tsLine = d3Line()
.x(d => xScale(d.dateTime))
.y(d => yScale(d.value));
elem.append('path')
.datum(line.points)
.classed('line-segment', true)
.classed('approved', line.classes.approved)
.classed('estimated', line.classes.estimated)
.classed(`ts-${tsKey}`, true)
.attr('d', tsLine);
}
const maskCode = line.classes.dataMask.toLowerCase();
const maskDisplayName = MASK_DESC[maskCode].replace(' ', '-').toLowerCase();
Naab, Daniel James
committed
const [xDomainStart, xDomainEnd] = extent(line.points, d => d.dateTime);
const [yRangeStart, yRangeEnd] = yScale.domain();
let maskGroup = elem.append('g')
.attr('class', `${tsKey}-mask-group`);
const xSpan = xScale(xDomainEnd) - xScale(xDomainStart);
const rectWidth = xSpan > 0 ? xSpan : 1;
maskGroup.append('rect')
.attr('x', xScale(xDomainStart))
.attr('y', yScale(yRangeEnd))
.attr('width', rectWidth)
.attr('height', Math.abs(yScale(yRangeEnd)- yScale(yRangeStart)))
.attr('class', `mask ${maskDisplayName}-mask`);
const patternId = HASH_ID[tsKey] ? `url(#${HASH_ID[tsKey]})` : '';
maskGroup.append('rect')
.attr('x', xScale(xDomainStart))
.attr('y', yScale(yRangeEnd))
.attr('width', rectWidth)
.attr('height', Math.abs(yScale(yRangeEnd) - yScale(yRangeStart)))
.attr('fill', patternId);
Naab, Daniel James
committed
}
Naab, Daniel James
committed
const plotDataLines = function (elem, {visible, tsLinesMap, tsKey, xScale, yScale}, container) {
container = container || elem.append('g');
Naab, Daniel James
committed
Naab, Daniel James
committed
const elemId = `ts-${tsKey}-group`;
container.selectAll(`#${elemId}`).remove();
const tsLineGroup = container
Naab, Daniel James
committed
.append('g')
.attr('id', elemId)
.classed('tsKey', true);
Naab, Daniel James
committed
Naab, Daniel James
committed
for (const lines of Object.values(tsLinesMap)) {
plotDataLine(tsLineGroup, {visible, lines, tsKey, xScale, yScale});
Naab, Daniel James
committed
}
Naab, Daniel James
committed
return container;
Naab, Daniel James
committed
};
let defs = elem.append('defs');
defs.append('mask')
.attr('id', 'display-mask')
.attr('maskUnits', 'userSpaceOnUse')
.append('rect')
.attr('x', '0')
.attr('y', '0')
.attr('width', '100%')
.attr('height', '100%')
.attr('fill', '#0000ff');
.attr('id', HASH_ID.current)
.attr('width', '8')
.attr('height', '8')
.attr('patternUnits', 'userSpaceOnUse')
.attr('patternTransform', 'rotate(45)')
.append('rect')
.attr('width', '4')
.attr('height', '8')
.attr('transform', 'translate(0, 0)')
.attr('mask', 'url(#display-mask)');
.attr('id', HASH_ID.compare)
.attr('width', '8')
.attr('height', '8')
.attr('patternUnits', 'userSpaceOnUse')
.attr('patternTransform', 'rotate(135)')
.append('rect')
.attr('width', '4')
.attr('height', '8')
.attr('transform', 'translate(0, 0)')
.attr('mask', 'url(#display-mask)');
const timeSeriesLegend = function(elem) {
elem.append('div')
.classed('hydrograph-container', true)
Bucknell, Mary S.
committed
.call(link(drawSimpleLegend, createStructuredSelector({
legendMarkerRows: legendMarkerRowsSelector,
layout: layoutSelector
})));
/**
* Plots the median points for a single median time series.
* @param {Object} elem
* @param {Function} xscale
* @param {Function} yscale
* @param {Number} modulo
* @param {Array} points
* @param {Boolean} showLabel
* @param {Object} variable
*/
const plotMedianPoints = function (elem, {xscale, yscale, modulo, points, showLabel, variable}) {
elem.selectAll('medianPoint')
.data(points)
.enter()
.append('circle')
.classed('median-data-series', true)
.classed(`median-modulo-${modulo}`, true)
.attr('cx', function(d) {
Naab, Daniel James
committed
return xscale(d.dateTime);
})
.attr('cy', function(d) {
Bucknell, Mary S.
committed
.on('click', dispatch(function() {
return Actions.showMedianStatsLabel(!showLabel);
}));
Bucknell, Mary S.
committed
if (showLabel) {
elem.selectAll('medianPointText')
.data(points)
.enter()
.append('text')
.text(function(d) {
return `${d.value} ${variable.unit.unitCode}`;
Naab, Daniel James
committed
return xscale(d.dateTime) + 5;
})
.attr('y', function(d) {
return yscale(d.value);
});
Bucknell, Mary S.
committed
}
/**
* Plots the median points for all median time series for the current variable.
* @param {Object} elem
* @param {Boolean} visible
* @param {Function} xscale
* @param {Function} yscale
* @param {Array} pointsList
* @param {Boolean} showLabel
* @param {Object} variable
Naab, Daniel James
committed
const plotAllMedianPoints = function (elem, {visible, xscale, yscale, seriesMap, showLabel, variable}) {
elem.select('#median-points').remove();
if (!visible) {
return;
}
const container = elem
.append('g')
.attr('id', 'median-points');
Naab, Daniel James
committed
for (const [index, seriesID] of Object.keys(seriesMap).entries()) {
const points = seriesMap[seriesID].points;
plotMedianPoints(container, {xscale, yscale, modulo: index % 6, points, showLabel, variable});
}
};
/* TODO: Please remove after WDFN-250 is implemented
const plotSROnlyTable = function (elem, {tsKey, variable, methods, visible, dataByTsID, timeSeries}) {
elem.selectAll(`#sr-only-${tsKey}`).remove();
if (!visible) {
return;
}
const container = elem.append('div')
Bucknell, Mary S.
committed
.attr('id', `sr-only-${tsKey}`)
.classed('usa-sr-only', true);
for (const seriesID of Object.keys(timeSeries)) {
const series = timeSeries[seriesID];
const method = methods[series.method].methodDescription;
let title = variable.variableName;
if (method) {
title += ` (${method})`;
}
if (tsKey === 'median') {
title = `Median ${title}`;
}
addSROnlyTable(container, {
columnNames: [title, 'Time', 'Qualifiers'],
data: dataByTsID[seriesID],
describeById: `${seriesID}-time-series-sr-desc`,
describeByText: `${seriesID} time series data in tabular format`
});
}
};
Bucknell, Mary S.
committed
elem.append('div')
.classed('timeseries-graph-title', true)
Bucknell, Mary S.
committed
.call(link((elem, title) => {
elem.html(title);
}, titleSelector));
const watermark = function (elem) {
elem.append('img')
.classed('watermark', true)
.attr('src', STATIC_URL + '/img/USGS_green_logo.svg')
Briggs, Aaron Shane
committed
.call(link(function(elem, layout) {
const transformStringSmallScreen = `matrix(0.5, 0, 0, 0.5, ${(layout.width - layout.margin.left) * .025
+ layout.margin.left - 50}, ${layout.height * .60})`;
const transformStringForAllOtherScreens = `matrix(1, 0, 0, 1, ${(layout.width - layout.margin.left) * .025
+ layout.margin.left}, ${(layout.height * .75 - (-1 * layout.height + 503) * .12)})`;
Briggs, Aaron Shane
committed
if (!mediaQuery(USWDS_SMALL_SCREEN)) {
// calculates the watermark position based on current layout dimensions
// and a conversion factor minus the area for blank space due to scaling
elem.style('transform', transformStringSmallScreen);
Briggs, Aaron Shane
committed
// adapts code for Safari browser
elem.style('-webkit-transform', transformStringSmallScreen);
Briggs, Aaron Shane
committed
// calculates the watermark position based on current layout dimensions and a conversion factor
elem.style('transform', transformStringForAllOtherScreens);
Briggs, Aaron Shane
committed
// adapts code for Safari browser
elem.style('-webkit-transform', transformStringForAllOtherScreens);
}
}, layoutSelector));
elem.call(watermark)
.append('div')
.attr('class', 'hydrograph-container')
Bucknell, Mary S.
committed
.call(createTitle)
Naab, Daniel James
committed
.call(createTooltipText)
.append('svg')
Briggs, Aaron Shane
committed
.classed('hydrograph-svg', true)
Naab, Daniel James
committed
.call(link((elem, layout) => elem.attr('viewBox', `0 0 ${layout.width + layout.margin.left + layout.margin.right} ${layout.height + layout.margin.top + layout.margin.bottom}`), layoutSelector))
Naab, Daniel James
committed
.call(link(addSVGAccessibility, createStructuredSelector({
Bucknell, Mary S.
committed
title: titleSelector,
description: descriptionSelector,
Naab, Daniel James
committed
isInteractive: () => true
})))
.call(svg => {
svg.append('g')
Naab, Daniel James
committed
.call(link((elem, layout) => elem.attr('transform', `translate(${layout.margin.left},${layout.margin.top})`), layoutSelector))
.call(link(appendAxes, axesSelector))
.call(link(plotDataLines, createStructuredSelector({
visible: isVisibleSelector('current'),
tsLinesMap: currentVariableLineSegmentsSelector('current'),
xScale: xScaleSelector('current'),
yScale: yScaleSelector,
tsKey: () => 'current'
})))
.call(link(plotDataLines, createStructuredSelector({
visible: isVisibleSelector('compare'),
tsLinesMap: currentVariableLineSegmentsSelector('compare'),
xScale: xScaleSelector('compare'),
yScale: yScaleSelector,
tsKey: () => 'compare'
})))
.call(createTooltipFocus)
.call(link(plotAllMedianPoints, createStructuredSelector({
visible: isVisibleSelector('median'),
xscale: xScaleSelector('current'),
yscale: yScaleSelector,
seriesMap: currentVariableTimeSeriesSelector('median'),
Bucknell, Mary S.
committed
showLabel: (state) => state.timeseriesState.showMedianStatsLabel
Naab, Daniel James
committed
});
.call(link(plotSROnlyTable, createStructuredSelector({
tsKey: () => 'current',
variable: currentVariableSelector,
Bucknell, Mary S.
committed
methods: getMethods,
visible: isVisibleSelector('current'),
dataByTsID: pointsTableDataSelector('current'),
timeSeries: currentVariableTimeSeriesSelector('current')
})));
elem.append('div')
.call(link(plotSROnlyTable, createStructuredSelector({
tsKey: () => 'compare',
variable: currentVariableSelector,
Bucknell, Mary S.
committed
methods: getMethods,
visible: isVisibleSelector('compare'),
dataByTsID: pointsTableDataSelector('compare'),
timeSeries: currentVariableTimeSeriesSelector('compare')
Bucknell, Mary S.
committed
})));
.call(link(plotSROnlyTable, createStructuredSelector({
tsKey: () => 'median',
variable: currentVariableSelector,
Bucknell, Mary S.
committed
methods: getMethods,
visible: isVisibleSelector('median'),
dataByTsID: pointsTableDataSelector('median'),
timeSeries: currentVariableTimeSeriesSelector('median')
Naab, Daniel James
committed
})));
/*
* Create the show last year toggle and the audible toggle for the timeseries graph.
* @param {Object} elem - D3 selection
*/
const graphControls = function(elem) {
const graphControlDiv = elem.append('ul')
.classed('usa-fieldset-inputs', true)
.classed('usa-unstyled-list', true)
.classed('graph-controls-container', true);
Bucknell, Mary S.
committed
graphControlDiv.call(link(function(elem, layout) {
if (!mediaQuery(USWDS_MEDIUM_SCREEN)) {
elem.style('padding-left', `${layout.margin.left}px`);
} else {
elem.style('padding-left', null);
}
}, layoutSelector));
graphControlDiv.append('li')
.call(audibleUI);
const compareControlDiv = graphControlDiv.append('li');
compareControlDiv.append('input')
.attr('type', 'checkbox')
.attr('id', 'last-year-checkbox')
.attr('aria-labelledby', 'last-year-label')
.attr('ga-on', 'click')
.attr('ga-event-category', 'TimeseriesGraph')
.attr('ga-event-action', 'toggleCompare')
.on('click', dispatch(function() {
return Actions.toggleTimeseries('compare', this.checked);
// Disables the checkbox if no compare time series for the current variable
Bucknell, Mary S.
committed
.call(link(function(elem, compareTimeseries) {
const exists = Object.keys(compareTimeseries) ?
Object.values(compareTimeseries).filter(tsValues => tsValues.points.length).length > 0 : false;
elem.property('disabled', !exists);
if (!exists) {
Bucknell, Mary S.
committed
elem.property('checked', false);
elem.dispatch('click');
Bucknell, Mary S.
committed
}
}, currentVariableTimeSeriesSelector('compare')))
Bucknell, Mary S.
committed
.call(link(function(elem, checked) {
Bucknell, Mary S.
committed
}, isVisibleSelector('compare')));
compareControlDiv.append('label')
.attr('id', 'last-year-label')
.attr('for', 'last-year-checkbox')
.text('Show last year');
};
Bucknell, Mary S.
committed
/**
* Modify styling to hide or display the plot area.
*
* @param elem
* @param currentTimeseries
*/
const controlGraphDisplay = function (elem, currentTimeseries) {
const seriesWithPoints = Object.values(currentTimeseries).filter(x => x.points.length > 0);
elem.attr('hidden', seriesWithPoints.length === 0 ? true : null);
};
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
const createDaterangeControls = function(elem, siteno) {
const DATE_RANGE = [{
label: 'seven-day',
name: '7 days',
period: 'P7D'
}, {
label: 'thirty-days',
name: '30 days',
period: 'P30D'
}, {
label: 'one-year',
name: '1 year',
period: 'P1Y'
}];
const container = elem.insert('ul', ':first-child')
.attr('id', 'ts-daterange-select-container')
.attr('class', 'usa-fieldset-inputs usa-unstyled-list');
const li = container.selectAll('li')
.data(DATE_RANGE)
.enter().append('li');
li.append('input')
.attr('type', 'radio')
.attr('name', 'ts-daterange-input')
.attr('id', (d) => d.label)
.attr('value', (d) => d.period)
.on('change', dispatch(function() {
return Actions.retrieveExtendedTimeseries(
siteno,
li.select('input:checked').attr('value')
);
}));
li.append('label')
.attr('for', (d) => d.label)
.text((d) => d.name);
li.select(`#${DATE_RANGE[0].label}`).attr('checked', true);
};
Bucknell, Mary S.
committed
const attachToNode = function (store, node, {siteno} = {}) {
if (!siteno) {
select(node).call(drawMessage, 'No data is available.');
return;
Bucknell, Mary S.
committed
}
Bucknell, Mary S.
committed
store.dispatch(Actions.resizeUI(window.innerWidth, node.offsetWidth));
select(node)
.call(provide(store))
.call(createDaterangeControls, siteno);
Briggs, Aaron Shane
committed
select(node).select('.graph-container')
Bucknell, Mary S.
committed
.call(link(controlGraphDisplay, timeSeriesSelector('current')()))
Bucknell, Mary S.
committed
.call(timeSeriesGraph, siteno)
Briggs, Aaron Shane
committed
.call(cursorSlider)
.append('div')
.classed('ts-legend-controls-container', true)
.call(timeSeriesLegend)
.call(graphControls);
select(node).select('.select-timeseries-container')
.call(link(plotSeriesSelectTable, createStructuredSelector({
Briggs, Aaron Shane
committed
availableTimeseries: availableTimeseriesSelector,
Bucknell, Mary S.
committed
lineSegmentsByParmCd: lineSegmentsByParmCdSelector('current')('P7D'),
timeSeriesScalesByParmCd: timeSeriesScalesByParmCdSelector('current')('P7D')(SPARK_LINE_DIM),
Briggs, Aaron Shane
committed
layout: layoutSelector
})));
select(node).select('.provisional-data-alert')
.call(link(function(elem, allTimeSeries) {
elem.attr('hidden', Object.keys(allTimeSeries).length ? null : true);
}, allTimeSeriesSelector));
Bucknell, Mary S.
committed
Bucknell, Mary S.
committed
window.onresize = function() {
store.dispatch(Actions.resizeUI(window.innerWidth, node.offsetWidth));
Bucknell, Mary S.
committed
};
store.dispatch(Actions.retrieveTimeseries(siteno));
};
module.exports = {attachToNode, timeSeriesLegend, timeSeriesGraph};