import ChartPage, {WithChart} from '~/components/ChartPage'; import {NextI18NextPage, useTranslation} from '~/utils/i18n'; import React, {useCallback, useEffect, useMemo, useState} from 'react'; import {Run, StepInfo, Tag, TimeType} from '~/resource/pr-curve'; import {borderColor, rem} from '~/utils/style'; import {AsideSection} from '~/components/Aside'; import Content from '~/components/Content'; import Error from '~/components/Error'; import Field from '~/components/Field'; import PRCurveChart from '~/components/PRCurvePage/PRCurveChart'; import Preloader from '~/components/Preloader'; import RunAside from '~/components/RunAside'; import StepSlider from '~/components/PRCurvePage/StepSlider'; import TimeModeSelect from '~/components/TimeModeSelect'; import Title from '~/components/Title'; import {cycleFetcher} from '~/utils/fetch'; import queryString from 'query-string'; import styled from 'styled-components'; import {useRunningRequest} from '~/hooks/useRequest'; import useTagFilter from '~/hooks/useTagFilter'; const StepSliderWrapper = styled.div` max-height: 30vh; overflow: auto; overflow-x: hidden; overflow-y: auto; flex-shrink: 0; > ${AsideSection}:last-child { padding-bottom: ${rem(20)}; margin-bottom: 0; } + .run-section { border-top: 1px solid ${borderColor}; margin-top: 0; padding-top: ${rem(20)}; } &:empty + .run-section { border-top: none; } `; const PRCurve: NextI18NextPage = () => { const {t} = useTranslation(['pr-curve', 'common']); const [running, setRunning] = useState(true); const {runs, tags, runsInTags, selectedRuns, onChangeRuns, loadingRuns, loadingTags} = useTagFilter( 'pr-curve', running ); const [indexes, setIndexes] = useState>({}); const onChangeIndexes = useCallback( (run: string, index: number) => setIndexes(indexes => ({ ...indexes, [run]: index })), [] ); useEffect( () => setIndexes(indexes => runsInTags.reduce((m, c) => { if (indexes[c.label] != null) { m[c.label] = indexes[c.label]; } return m; }, {}) ), [runsInTags] ); const {data: stepInfo} = useRunningRequest( runsInTags.map(run => `/pr-curve/steps?${queryString.stringify({run: run.label})}`), !!running, (...urls) => cycleFetcher(urls) ); const runWithInfo = useMemo( () => runsInTags.map((run, i) => ({ ...run, index: indexes[run.label] ?? (stepInfo?.[i].length ?? 1) - 1, steps: stepInfo?.[i].map(j => j[1]) ?? [], wallTimes: stepInfo?.[i].map(j => Math.floor(j[0] * 1000)) ?? [], relatives: stepInfo?.[i].map(j => (j[0] - stepInfo[i][0][0]) * 1000) ?? [] })), [runsInTags, stepInfo, indexes] ); const [timeType, setTimeType] = useState(TimeType.Step); const prCurveTags = useMemo( () => tags.map(tag => ({ ...tag, runs: tag.runs.map(run => ({ ...run, index: 0, steps: [] as Run['steps'], wallTimes: [] as Run['wallTimes'], relatives: [] as Run['relatives'], ...runWithInfo.find(r => r.label === run.label) })) })), [tags, runWithInfo] ); const aside = useMemo( () => runs.length ? ( {runWithInfo.map(run => ( onChangeIndexes(run.label, index)} /> ))} ) : null, [t, onChangeRuns, running, runs, selectedRuns, timeType, runWithInfo, onChangeIndexes] ); const withChart = useCallback>( ({label, runs, ...args}) => , [running] ); return ( <> {t('common:pr-curve')} {!loadingRuns && !runs.length ? ( ) : ( )} ); }; PRCurve.getInitialProps = () => ({ namespacesRequired: ['pr-curve', 'common'] }); export default PRCurve;