241 lines
6.2 KiB
TypeScript
241 lines
6.2 KiB
TypeScript
import {
|
|
addMonths,
|
|
addWeeks,
|
|
addYears,
|
|
endOfMonth,
|
|
endOfWeek,
|
|
endOfYear,
|
|
format,
|
|
startOfMonth,
|
|
startOfWeek,
|
|
startOfYear,
|
|
subMonths,
|
|
subWeeks,
|
|
subYears,
|
|
} from 'date-fns'
|
|
|
|
import createI18n from '@/i18n'
|
|
import { IChartDataset } from '@/types/chart'
|
|
import { ISport } from '@/types/sports'
|
|
import {
|
|
IStatisticsChartData,
|
|
IStatisticsDateParams,
|
|
TStatisticsDatasetKeys,
|
|
TStatisticsDatasets,
|
|
TStatisticsFromApi,
|
|
} from '@/types/statistics'
|
|
import { incrementDate, getStartDate, getDateFormat } from '@/utils/dates'
|
|
import { localeFromLanguage } from '@/utils/locales'
|
|
import { sportColors } from '@/utils/sports'
|
|
import { convertStatsDistance } from '@/utils/units'
|
|
|
|
const { locale } = createI18n.global
|
|
|
|
const dateFormats: Record<string, Record<string, string>> = {
|
|
week: {
|
|
api: 'yyyy-MM-dd',
|
|
chart: 'MM/dd/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,
|
|
userDateFormat: string
|
|
): 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<string, number> = {}
|
|
displayedSports.map(
|
|
(displayedSport) => (sportsId[displayedSport.label] = displayedSport.id)
|
|
)
|
|
|
|
dayKeys.map((key) => {
|
|
const date: string = format(key, dateFormat.api)
|
|
const label: string = format(
|
|
key,
|
|
params.duration === 'week'
|
|
? getDateFormat(userDateFormat, locale.value)
|
|
: dateFormat.chart,
|
|
{ locale: localeFromLanguage[locale.value] }
|
|
)
|
|
labels.push(label)
|
|
datasetKeys.map((datasetKey) => {
|
|
datasets[datasetKey].map((dataset) => {
|
|
dataset.data.push(
|
|
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)),
|
|
}
|
|
}
|