// Copyright (c) 2022 Uber Technologies, Inc.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import React, {useMemo} from 'react';
import {
HorizontalGridLines,
LineSeries,
XYPlot,
CustomSVGSeries,
Hint,
YAxis,
MarkSeries,
LineSeriesPoint,
RVNearestXData
} from 'react-vis';
import {LineChart} from 'reducers';
import styled from 'styled-components';
import {datetimeFormatter} from 'utils/data-utils';
const LineChartWrapper = styled.div`
.rv-xy-plot {
/* important for rendering hint */
position: relative;
}
.rv-xy-plot__inner {
/* important to show axis */
overflow: visible;
}
.rv-xy-plot__grid-lines__line {
stroke: ${props => props.theme.histogramFillOutRange};
stroke-dasharray: 1px 4px;
}
.rv-xy-plot__axis__tick__text {
font-size: 9px;
fill: ${props => props.theme.textColor};
}
`;
const StyledHint = styled.div`
background-color: #d3d8e0;
border-radius: 2px;
color: ${props => props.theme.textColorLT};
font-size: 9px;
margin: 4px;
padding: 3px 6px;
pointer-events: none;
user-select: none;
`;
interface HintContentProps {
x: number;
y: number;
format: (ts: number) => string;
}
const HintContent = ({x, y, format}: HintContentProps) => (
{format(x)}
{y}
);
export interface HoverDP {
x: number;
y: number;
color?: string | number;
opacity?: string | number;
stroke?: string | number;
fill?: string | number;
size?: string | number;
}
interface LineChartProps {
brushComponent?: any;
brushing?: boolean;
color?: string;
enableChartHover?: boolean;
height: number;
hoveredDP?: HoverDP | null;
isEnlarged?: boolean;
lineChart?: LineChart;
margin: {top?: number; bottom?: number; left?: number; right?: number};
onMouseMove: (datapoint: LineSeriesPoint | null, data?: RVNearestXData) => void;
width: number;
timezone?: string | null;
timeFormat?: string;
}
const MARGIN = {top: 0, bottom: 0, left: 0, right: 0};
function LineChartFactory() {
const LineChart = ({
brushComponent,
brushing,
color,
enableChartHover,
height,
hoveredDP,
isEnlarged,
lineChart,
margin,
onMouseMove,
width,
timezone,
timeFormat
}: LineChartProps) => {
const {series, yDomain} = lineChart || {};
const brushData = useMemo(() => {
return series && series[0] && series[0].x && yDomain && yDomain[1]
? [{x: series[0].x, y: yDomain[1], customComponent: () => brushComponent}]
: [];
}, [series, yDomain, brushComponent]);
const hintFormatter = useMemo(() => datetimeFormatter(timezone)(timeFormat), [
timezone,
timeFormat
]);
return (
{
onMouseMove(null);
}}
>
{isEnlarged && }
{hoveredDP && enableChartHover && !brushing ? (
) : null}
);
};
return LineChart;
}
export default LineChartFactory;