import { addMonths, addWeeks, addYears, endOfMonth, endOfWeek, endOfYear, format, startOfMonth, startOfWeek, startOfYear, subMonths, subWeeks, subYears, } from 'date-fns' import { IChartDataset } from '@/types/chart' import { ISport } from '@/types/sports' import { IStatisticsChartData, IStatisticsDateParams, TStatisticsDatasetKeys, TStatisticsDatasets, TStatisticsFromApi, } from '@/types/statistics' import { incrementDate, getStartDate } from '@/utils/dates' import { sportColors } from '@/utils/sports' import { convertStatsDistance } from '@/utils/units' const dateFormats: Record> = { week: { api: 'yyyy-MM-dd', chart: 'dd/MM/yyyy', }, month: { api: 'yyyy-MM', chart: 'MM/yyyy', }, year: { api: 'yyyy', chart: 'yyyy', }, } export const datasetKeys: TStatisticsDatasetKeys[] = [ 'average_speed', 'nb_workouts', 'total_duration', 'total_distance', 'total_ascent', 'total_descent', ] export const getDateKeys = ( params: IStatisticsDateParams, weekStartingMonday: boolean ): Date[] => { const days = [] for ( let day = getStartDate(params.duration, params.start, weekStartingMonday); day <= params.end; day = incrementDate(params.duration, day) ) { days.push(day) } return days } const getStatisticsChartDataset = ( sportLabel: string, color: string, isLineChart = false ): IChartDataset => { const dataset: IChartDataset = { label: sportLabel, backgroundColor: [color], data: [], } if (isLineChart) { dataset.type = 'line' dataset.borderColor = [color] dataset.spanGaps = true } return dataset } export const getDatasets = (displayedSports: ISport[]): TStatisticsDatasets => { const datasets: TStatisticsDatasets = { average_speed: [], nb_workouts: [], total_distance: [], total_duration: [], total_ascent: [], total_descent: [], } displayedSports.map((sport) => { const color = sport.color ? sport.color : sportColors[sport.label] datasets.average_speed.push( getStatisticsChartDataset(sport.label, color, true) ) datasets.nb_workouts.push(getStatisticsChartDataset(sport.label, color)) datasets.total_distance.push(getStatisticsChartDataset(sport.label, color)) datasets.total_duration.push(getStatisticsChartDataset(sport.label, color)) datasets.total_ascent.push(getStatisticsChartDataset(sport.label, color)) datasets.total_descent.push(getStatisticsChartDataset(sport.label, color)) }) return datasets } export const convertStatsValue = ( datasetKey: TStatisticsDatasetKeys, value: number, useImperialUnits: boolean ): number => { switch (datasetKey) { case 'average_speed': case 'total_distance': case 'total_ascent': case 'total_descent': return convertStatsDistance( ['average_speed', 'total_distance'].includes(datasetKey) ? 'km' : 'm', value, useImperialUnits ) default: case 'nb_workouts': case 'total_duration': return value } } export const formatStats = ( params: IStatisticsDateParams, weekStartingMonday: boolean, sports: ISport[], displayedSportsId: number[], apiStats: TStatisticsFromApi, useImperialUnits: boolean ): IStatisticsChartData => { const dayKeys = getDateKeys(params, weekStartingMonday) const dateFormat = dateFormats[params.duration] const displayedSports = sports.filter((sport) => displayedSportsId.includes(sport.id) ) const labels: string[] = [] const datasets = getDatasets(displayedSports) const sportsId: Record = {} displayedSports.map( (displayedSport) => (sportsId[displayedSport.label] = displayedSport.id) ) dayKeys.map((key) => { const date: string = format(key, dateFormat.api) const label: string = format(key, dateFormat.chart) labels.push(label) datasetKeys.map((datasetKey) => { datasets[datasetKey].map((dataset) => { dataset.data.push( apiStats !== {} && date in apiStats && sportsId[dataset.label] in apiStats[date] ? convertStatsValue( datasetKey, apiStats[date][sportsId[dataset.label]][datasetKey], useImperialUnits ) : datasetKey === 'average_speed' ? null : 0 ) }) }) }) return { labels, datasets, } } export const getStatsDateParams = ( date: Date, timeFrame: string, weekStartingMonday: boolean ): IStatisticsDateParams => { const weekStartsOn = weekStartingMonday ? 1 : 0 const start = timeFrame === 'year' ? startOfYear(subYears(date, 9)) : timeFrame === 'week' ? startOfWeek(subMonths(date, 2), { weekStartsOn }) : startOfMonth(subMonths(date, 11)) // month const end = timeFrame === 'year' ? endOfYear(date) : timeFrame === 'week' ? endOfWeek(date, { weekStartsOn }) : endOfMonth(date) // month return { duration: timeFrame, end, start, } } export const updateChartParams = ( chartParams: IStatisticsDateParams, backward: boolean, weekStartingMonday: boolean ): IStatisticsDateParams => { const { duration, start, end } = chartParams const weekStartsOn = weekStartingMonday ? 1 : 0 return { duration, end: duration === 'year' ? endOfYear(backward ? subYears(end, 1) : addYears(end, 1)) : duration === 'week' ? endOfWeek(backward ? subWeeks(end, 1) : addWeeks(end, 1), { weekStartsOn, }) : endOfMonth(backward ? subMonths(end, 1) : addMonths(end, 1)), start: duration === 'year' ? startOfYear(backward ? subYears(start, 1) : addYears(start, 1)) : duration === 'week' ? startOfWeek(backward ? subWeeks(start, 1) : addWeeks(start, 1), { weekStartsOn, }) : startOfMonth(backward ? subMonths(start, 1) : addMonths(start, 1)), } }