/* * SonarQube * Copyright (C) 2009-2021 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ import * as classNames from 'classnames'; import { bisector, extent, max } from 'd3-array'; import { scaleLinear, scalePoint, scaleTime, ScaleTime } from 'd3-scale'; import { area, curveBasis, line as d3Line } from 'd3-shape'; import { flatten, isEqual, sortBy, throttle, uniq } from 'lodash'; import * as React from 'react'; import { isDefined } from '../../helpers/types'; import { Theme, ThemeConsumer } from '../theme'; import './AdvancedTimeline.css'; import './LineChart.css'; export interface Props { basisCurve?: boolean; endDate?: Date; disableZoom?: boolean; displayNewCodeLegend?: boolean; formatYTick?: (tick: number | string) => string; hideGrid?: boolean; hideXAxis?: boolean; height: number; width: number; leakPeriodDate?: Date; // used to avoid same y ticks labels maxYTicksCount: number; metricType: string; padding: number[]; selectedDate?: Date; series: T.Chart.Serie[]; showAreas?: boolean; startDate?: Date; updateSelectedDate?: (selectedDate?: Date) => void; updateTooltip?: (selectedDate?: Date, tooltipXPos?: number, tooltipIdx?: number) => void; updateZoom?: (start?: Date, endDate?: Date) => void; zoomSpeed: number; } type XScale = ScaleTime; // TODO it should be `ScaleLinear | ScalePoint | ScalePoint`, but it's super hard to make it work :'( type YScale = any; const LEGEND_LINE_HEIGHT = 16; interface State { leakLegendTextWidth?: number; maxXRange: number[]; mouseOver?: boolean; selectedDate?: Date; selectedDateXPos?: number; selectedDateIdx?: number; yScale: YScale; xScale: XScale; } export default class AdvancedTimeline extends React.PureComponent { static defaultProps = { eventSize: 8, maxYTicksCount: 4, padding: [26, 10, 50, 60], zoomSpeed: 1, }; constructor(props: Props) { super(props); const scales = this.getScales(props); const selectedDatePos = this.getSelectedDatePos(scales.xScale, props.selectedDate); this.state = { ...scales, ...selectedDatePos }; this.updateTooltipPos = throttle(this.updateTooltipPos, 40); this.handleZoomUpdate = throttle(this.handleZoomUpdate, 40); } componentDidUpdate(prevProps: Props) { let scales; let selectedDatePos; if ( this.props.metricType !== prevProps.metricType || this.props.startDate !== prevProps.startDate || this.props.endDate !== prevProps.endDate || this.props.width !== prevProps.width || this.props.padding !== prevProps.padding || this.props.height !== prevProps.height || this.props.series !== prevProps.series ) { scales = this.getScales(this.props); if (this.state.selectedDate != null) { selectedDatePos = this.getSelectedDatePos(scales.xScale, this.state.selectedDate); } } if (!isEqual(this.props.selectedDate, prevProps.selectedDate)) { const xScale = scales ? scales.xScale : this.state.xScale; selectedDatePos = this.getSelectedDatePos(xScale, this.props.selectedDate); } if (scales || selectedDatePos) { if (scales) { this.setState({ ...scales }); } if (selectedDatePos) { this.setState({ ...selectedDatePos }); } if (selectedDatePos && this.props.updateTooltip) { this.props.updateTooltip( selectedDatePos.selectedDate, selectedDatePos.selectedDateXPos, selectedDatePos.selectedDateIdx ); } } } getRatingScale = (availableHeight: number) => { return scalePoint().domain([5, 4, 3, 2, 1]).range([availableHeight, 0]); }; getLevelScale = (availableHeight: number) => { return scalePoint().domain(['ERROR', 'WARN', 'OK']).range([availableHeight, 0]); }; getYScale = (props: Props, availableHeight: number, flatData: T.Chart.Point[]): YScale => { if (props.metricType === 'RATING') { return this.getRatingScale(availableHeight); } else if (props.metricType === 'LEVEL') { return this.getLevelScale(availableHeight); } else { return scaleLinear() .range([availableHeight, 0]) .domain([0, max(flatData, (d) => Number(d.y || 0)) || 1]) .nice(); } }; getXScale = ( { startDate, endDate }: Props, availableWidth: number, flatData: T.Chart.Point[] ) => { const dateRange = extent(flatData, (d) => d.x) as [Date, Date]; const start = startDate && startDate > dateRange[0] ? startDate : dateRange[0]; const end = endDate && endDate < dateRange[1] ? endDate : dateRange[1]; const xScale: ScaleTime = scaleTime() .domain(sortBy([start, end])) .range([0, availableWidth]) .clamp(false); return { xScale, maxXRange: dateRange.map(xScale), }; }; getScales = (props: Props) => { const availableWidth = props.width - props.padding[1] - props.padding[3]; const availableHeight = props.height - props.padding[0] - props.padding[2]; const flatData = flatten(props.series.map((serie) => serie.data)); return { ...this.getXScale(props, availableWidth, flatData), yScale: this.getYScale(props, availableHeight, flatData), }; }; getSelectedDatePos = (xScale: XScale, selectedDate?: Date) => { const firstSerie = this.props.series[0]; if (selectedDate && firstSerie) { const idx = firstSerie.data.findIndex((p) => p.x.valueOf() === selectedDate.valueOf()); const xRange = sortBy(xScale.range()); const xPos = xScale(selectedDate); if (idx >= 0 && xPos >= xRange[0] && xPos <= xRange[1]) { return { selectedDate, selectedDateXPos: xScale(selectedDate), selectedDateIdx: idx, }; } } return { selectedDate: undefined, selectedDateXPos: undefined, selectedDateIdx: undefined }; }; getEventMarker = (size: number) => { const half = size / 2; return `M${half} 0 L${size} ${half} L ${half} ${size} L0 ${half} L${half} 0 L${size} ${half}`; }; handleWheel = (event: React.WheelEvent) => { event.preventDefault(); const { maxXRange, xScale } = this.state; const parentBbox = event.currentTarget.getBoundingClientRect(); const mouseXPos = (event.pageX - parentBbox.left) / parentBbox.width; const xRange = xScale.range(); const speed = event.deltaMode ? (25 / event.deltaMode) * this.props.zoomSpeed : this.props.zoomSpeed; const leftPos = xRange[0] - Math.round(speed * event.deltaY * mouseXPos); const rightPos = xRange[1] + Math.round(speed * event.deltaY * (1 - mouseXPos)); const startDate = leftPos > maxXRange[0] ? xScale.invert(leftPos) : undefined; const endDate = rightPos < maxXRange[1] ? xScale.invert(rightPos) : undefined; this.handleZoomUpdate(startDate, endDate); }; handleZoomUpdate = (startDate?: Date, endDate?: Date) => { if (this.props.updateZoom) { this.props.updateZoom(startDate, endDate); } }; handleMouseMove = (event: React.MouseEvent) => { const parentBbox = event.currentTarget.getBoundingClientRect(); this.updateTooltipPos(event.pageX - parentBbox.left); }; handleMouseEnter = () => { this.setState({ mouseOver: true }); }; handleMouseOut = () => { const { updateTooltip } = this.props; if (updateTooltip) { this.setState({ mouseOver: false, selectedDate: undefined, selectedDateXPos: undefined, selectedDateIdx: undefined, }); updateTooltip(undefined, undefined, undefined); } }; handleClick = () => { const { updateSelectedDate } = this.props; if (updateSelectedDate) { updateSelectedDate(this.state.selectedDate || undefined); } }; setLeakLegendTextWidth = (node: SVGTextElement | null) => { if (node) { this.setState({ leakLegendTextWidth: node.getBoundingClientRect().width }); } }; updateTooltipPos = (xPos: number) => { this.setState((state) => { const firstSerie = this.props.series[0]; if (state.mouseOver && firstSerie) { const { updateTooltip } = this.props; const date = state.xScale.invert(xPos); const bisectX = bisector((d) => d.x).right; let idx = bisectX(firstSerie.data, date); if (idx >= 0) { const previousPoint = firstSerie.data[idx - 1]; const nextPoint = firstSerie.data[idx]; if ( !nextPoint || (previousPoint && date.valueOf() - previousPoint.x.valueOf() <= nextPoint.x.valueOf() - date.valueOf()) ) { idx--; } const selectedDate = firstSerie.data[idx].x; const xPos = state.xScale(selectedDate); if (updateTooltip) { updateTooltip(selectedDate, xPos, idx); } return { selectedDate, selectedDateXPos: xPos, selectedDateIdx: idx }; } } return null; }); }; renderHorizontalGrid = () => { const { formatYTick } = this.props; const { xScale, yScale } = this.state; const hasTicks = typeof yScale.ticks === 'function'; let ticks: Array = hasTicks ? yScale.ticks(this.props.maxYTicksCount) : yScale.domain(); if (!ticks.length) { ticks.push(yScale.domain()[1]); } // if there are duplicated ticks, that means 4 ticks are too much for this data // so let's just use the domain values (min and max) if (formatYTick) { const formattedTicks = ticks.map((tick) => formatYTick(tick)); if (ticks.length > uniq(formattedTicks).length) { ticks = yScale.domain(); } } return ( {ticks.map((tick) => ( {formatYTick != null && ( {formatYTick(tick)} )} ))} ); }; renderXAxisTicks = () => { const { xScale, yScale } = this.state; const format = xScale.tickFormat(7); const ticks = xScale.ticks(7); const y = yScale.range()[0]; return ( {ticks.slice(0, -1).map((tick, index) => { const nextTick = index + 1 < ticks.length ? ticks[index + 1] : xScale.domain()[1]; const x = (xScale(tick) + xScale(nextTick)) / 2; return ( {format(tick)} ); })} ); }; renderNewCodeLegend = (params: { leakStart: number; leakWidth: number; theme: Theme }) => { const { leakStart, leakWidth, theme } = params; const { leakLegendTextWidth, xScale, yScale } = this.state; const yRange = yScale.range(); const xRange = xScale.range(); const legendMinWidth = (leakLegendTextWidth || 0) + theme.rawSizes.grid; const legendPadding = theme.rawSizes.grid / 2; let legendBackgroundPosition; let legendBackgroundWidth; let legendMargin; let legendPosition; let legendTextAnchor; if (leakWidth >= legendMinWidth) { legendBackgroundWidth = leakWidth; legendBackgroundPosition = leakStart; legendMargin = 0; legendPosition = legendBackgroundPosition + legendPadding; legendTextAnchor = 'start'; } else { legendBackgroundWidth = legendMinWidth; legendBackgroundPosition = xRange[xRange.length - 1] - legendBackgroundWidth; legendMargin = theme.rawSizes.grid / 2; legendPosition = xRange[xRange.length - 1] - legendPadding; legendTextAnchor = 'end'; } return ( <> new code ); }; renderLeak = () => { const { displayNewCodeLegend, leakPeriodDate } = this.props; if (!leakPeriodDate) { return null; } const { xScale, yScale } = this.state; const yRange = yScale.range(); const xRange = xScale.range(); // truncate leak to start of chart to prevent weird visual artifacts when too far left // (occurs when leak starts a long time before first analysis) const leakStart = Math.max(xScale(leakPeriodDate), xRange[0]); const leakWidth = xRange[xRange.length - 1] - leakStart; if (leakWidth < 1) { return null; } return ( {(theme) => ( <> {displayNewCodeLegend && this.renderNewCodeLegend({ leakStart, leakWidth, theme })} )} ); }; renderLines = () => { const lineGenerator = d3Line() .defined((d) => Boolean(d.y || d.y === 0)) .x((d) => this.state.xScale(d.x)) .y((d) => this.state.yScale(d.y)); if (this.props.basisCurve) { lineGenerator.curve(curveBasis); } return ( {this.props.series.map((serie, idx) => ( ))} ); }; renderDots = () => { return ( {this.props.series .map((serie, serieIdx) => serie.data .map((point, idx) => { const pointNotDefined = !point.y && point.y !== 0; const hasPointBefore = serie.data[idx - 1] && (serie.data[idx - 1].y || serie.data[idx - 1].y === 0); const hasPointAfter = serie.data[idx + 1] && (serie.data[idx + 1].y || serie.data[idx + 1].y === 0); if (pointNotDefined || hasPointBefore || hasPointAfter) { return undefined; } return ( ); }) .filter(isDefined) ) .filter((dots) => dots.length > 0)} ); }; renderAreas = () => { const areaGenerator = area() .defined((d) => Boolean(d.y || d.y === 0)) .x((d) => this.state.xScale(d.x)) .y1((d) => this.state.yScale(d.y)) .y0(this.state.yScale(0)); if (this.props.basisCurve) { areaGenerator.curve(curveBasis); } return ( {this.props.series.map((serie, idx) => ( ))} ); }; renderSelectedDate = () => { const { selectedDateIdx, selectedDateXPos, yScale } = this.state; const firstSerie = this.props.series[0]; if (selectedDateIdx == null || selectedDateXPos == null || !firstSerie) { return null; } return ( {this.props.series.map((serie, idx) => { const point = serie.data[selectedDateIdx]; if (!point || (!point.y && point.y !== 0)) { return null; } return ( ); })} ); }; renderClipPath = () => { return ( ); }; renderMouseEventsOverlay = (zoomEnabled: boolean) => { const mouseEvents: Partial> = {}; if (zoomEnabled) { mouseEvents.onWheel = this.handleWheel; } if (this.props.updateTooltip) { mouseEvents.onMouseEnter = this.handleMouseEnter; mouseEvents.onMouseMove = this.handleMouseMove; mouseEvents.onMouseOut = this.handleMouseOut; } if (this.props.updateSelectedDate) { mouseEvents.onClick = this.handleClick; } return ( ); }; render() { if (!this.props.width || !this.props.height) { return
; } const zoomEnabled = !this.props.disableZoom && this.props.updateZoom != null; const isZoomed = Boolean(this.props.startDate || this.props.endDate); return ( {zoomEnabled && this.renderClipPath()} {this.props.leakPeriodDate != null && this.renderLeak()} {!this.props.hideGrid && this.renderHorizontalGrid()} {!this.props.hideXAxis && this.renderXAxisTicks()} {this.props.showAreas && this.renderAreas()} {this.renderLines()} {this.renderDots()} {this.renderSelectedDate()} {this.renderMouseEventsOverlay(zoomEnabled)} ); } }