Skip to content

Commit

Permalink
Merge pull request #503 from lifeomic/sleep-analysis-chart
Browse files Browse the repository at this point in the history
Add Sleep Data Chart
  • Loading branch information
sternetj authored Jan 2, 2024
2 parents e561388 + 59b70e9 commit e338aec
Show file tree
Hide file tree
Showing 13 changed files with 1,356 additions and 15 deletions.
2 changes: 1 addition & 1 deletion src/components/MyData/LineChart/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ import ViewShot from 'react-native-view-shot';
import { Trace, TraceLine } from './TraceLine';
import { useVictoryTheme } from '../useVictoryTheme';
import { scaleTime } from 'd3-scale';
import { Title } from './Title';
import { Title } from '../common/Title';
import { View, Dimensions } from 'react-native';
import { createStyles } from '../../BrandConfigProvider';
import { useStyles } from '../../../hooks';
Expand Down
284 changes: 284 additions & 0 deletions src/components/MyData/SleepChart/DailyChart.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,284 @@
import React, { useMemo } from 'react';
import { VictoryChart, VictoryAxis } from 'victory-native';
import { VictoryLabelProps, Tuple } from 'victory-core';
import {
eachHourOfInterval,
format,
endOfMinute,
startOfMinute,
differenceInMinutes,
} from 'date-fns';
import { View } from 'react-native';
import { createStyles, useIcons } from '../../BrandConfigProvider';
import {
ForeignObject,
G,
Circle,
Rect,
Text,
SvgProps,
CircleProps,
} from 'react-native-svg';
import { useCommonChartProps } from '../useCommonChartProps';
import uniqBy from 'lodash/unionBy';
import type { SleepChartData } from './useSleepChartData';
import ViewShot from 'react-native-view-shot';
import { t } from 'i18next';
import { useVictoryTheme } from '../useVictoryTheme';
import { useStyles } from '../../../hooks';
import { CodeableConcept } from 'fhir/r3';
import orderBy from 'lodash/orderBy';
import { Falsey } from 'lodash';
import { scaleLinear } from 'd3-scale';
import { ActivityIndicatorView } from '../../ActivityIndicatorView';

type Props = SleepChartData & {
viewShotRef: React.RefObject<ViewShot>;
};

export const DailyChart = (props: Props) => {
const { viewShotRef } = props;
const { xDomain, sleepData, isFetching } = props;
const common = useCommonChartProps();
const { sleepAnalysisTheme: theme } = useVictoryTheme();
const { styles } = useStyles(defaultStyles);

const yDomain = useMemo(
() =>
scaleLinear()
.range([0, common.height - common.padding.bottom - common.padding.top])
.domain([0, 4]),
[common],
);

const data = useMemo(
() =>
orderBy(
compact(
sleepData
.flatMap((d) =>
d.component?.map((c, i) => ({ id: `${d.id}-${i}`, ...c })),
)
.map((d) => {
if (!d?.valuePeriod?.start || !d?.valuePeriod?.end) {
return undefined;
}

const sleepType = codeToNum(d.code);
const start = startOfMinute(new Date(d.valuePeriod.start));
const end = endOfMinute(new Date(d.valuePeriod.end));

return {
...d,
x: xDomain(start),
width: xDomain(end) - xDomain(start),
y: yDomain(4 - sleepType) + common.padding.top,
height: yDomain(sleepType),
sleepType,
fill: codeToValue({
default: 'transparent',
...styles.stageColors,
})(d.code),
sleepTypeName: valToName(sleepType),
startTime: start.toLocaleTimeString(),
durationInMinutes: Math.abs(differenceInMinutes(start, end)),
};
}),
),
'sleepType',
'asc',
),
[sleepData, styles, common, xDomain, yDomain],
);

const ticks = useMemo(() => {
const [start, end] = xDomain.domain().sort((a, b) => Number(a) - Number(b));

if (start === end) {
return [];
}

return uniqBy([start, ...eachHourOfInterval({ start, end }), end], (v) =>
v.toISOString(),
);
}, [xDomain]);

return (
<View>
<ViewShot ref={viewShotRef} options={{ format: 'png' }}>
<VictoryChart
{...common}
domain={{
y: yDomain.domain() as Tuple<number>,
x: data.length ? (xDomain.domain() as Tuple<Date>) : undefined,
}}
>
<VictoryAxis
dependentAxis
tickFormat={valToName}
style={theme.dependentAxis}
/>

<G x={common.padding.left}>
{data.map((d) => (
<Rect
key={d.id}
{...d}
accessibilityLabel={t(
'sleep-analysis-stage-a11y-label',
'{{ durationInMinutes }} minutes of {{ sleepTypeName }} sleep starting at {{startTime}}',
d,
)}
/>
))}
</G>

<VictoryAxis
tickValues={ticks}
tickFormat={(v, i, a) =>
i === 0 || i + 1 === a.length ? format(v, 'hh:mm aa') : '*'
}
tickLabelComponent={<Tick enabled={!!data.length} />}
style={theme.independentAxis}
/>
</VictoryChart>
</ViewShot>

<View style={styles.loadingContainer}>
{<ActivityIndicatorView animating={isFetching} />}
</View>
</View>
);
};

function compact<T>(arr: T[]): Exclude<T, Falsey>[] {
return arr.filter((v) => !!v) as Exclude<T, Falsey>[];
}

type CodeMap<T> = Partial<Record<'light' | 'awake' | 'deep' | 'rem', T>> &
Record<'default', T>;

function codeToValue<T>(map: CodeMap<T>) {
return (coding: CodeableConcept) => {
for (const code of coding.coding ?? []) {
if (
code.system === 'http://loinc.org' &&
code.code === '93830-8' &&
map.light
) {
return map.light;
} else if (
code.system === 'http://loinc.org' &&
code.code === '93831-6' &&
map.deep
) {
return map.deep;
} else if (
code.system === 'http://loinc.org' &&
code.code === '93829-0' &&
map.rem
) {
return map.rem;
} else if (
code.system === 'http://loinc.org' &&
code.code === '93828-2' &&
map.awake
) {
return map.awake;
}
}

return map.default;
};
}

const codeToNum = codeToValue({
deep: 1,
light: 2,
rem: 3,
awake: 4,
default: 0,
});

const valToName = (value: number) =>
({
4: t('sleep-analysis-type-awake', 'Awake'),
3: t('sleep-analysis-type-rem', 'REM'),
2: t('sleep-analysis-type-light', 'Light'),
1: t('sleep-analysis-type-deep', 'Deep'),
}[value] ?? '');

type TickProps = {
text?: string;
index?: number;
enabled: boolean;
} & VictoryLabelProps;

const Tick = ({ text, index, enabled, ...props }: TickProps) => {
const { Moon, Sunrise } = useIcons();
const { styles } = useStyles(defaultStyles);
const Icon = index === 0 ? Moon : Sunrise;

if (!enabled) {
return null;
}

if (text === '*') {
return (
null && (
<Circle
{...(props as any)}
style={{ ...props.style, ...styles.tickDot }}
r={2}
/>
)
);
}

return (
<G {...(props as any)} y={(props.y ?? 0) - 5}>
<Circle {...styles.tickIconBackground} />
<ForeignObject x={-7} y={-7} width={20} height={20}>
<View style={{ width: 14, height: 14 }}>
<Icon {...styles.tickIcon} />
</View>
</ForeignObject>
<Text y={25}>{text}</Text>
</G>
);
};

const defaultStyles = createStyles('SleepAnalysisSingleDay', (theme) => ({
tickDot: {
fill: theme.colors.backdrop,
opacity: 0.4,
} as SvgProps,
tickIcon: {
color: theme.colors.onPrimary,
width: '100%',
height: '100%',
} as SvgProps,
tickIconBackground: {
fill: theme.colors.primary,
r: 10,
} as CircleProps,
stageColors: {
awake: 'hotpink',
deep: 'dodgerblue',
light: 'deepskyblue',
rem: 'deeppink',
default: 'transparent',
},
loadingContainer: {
position: 'absolute',
top: 0,
bottom: 0,
left: 0,
right: 0,
},
}));

declare module '@styles' {
interface ComponentStyles
extends ComponentNamedStyles<typeof defaultStyles> {}
}
Loading

0 comments on commit e338aec

Please sign in to comment.