93 lines
2.3 KiB
TypeScript
93 lines
2.3 KiB
TypeScript
// Copyright 2022 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import classNames from 'classnames';
|
|
import type { CSSProperties, ReactElement } from 'react';
|
|
import React, { useEffect, useState } from 'react';
|
|
import { animated, useSpring } from '@react-spring/web';
|
|
|
|
import type { LocalizerType } from '../../types/Util';
|
|
import { drop } from '../../util/drop';
|
|
import { TimelineDateHeader } from './TimelineDateHeader';
|
|
import { Spinner } from '../Spinner';
|
|
|
|
export type PropsType = Readonly<{
|
|
i18n: LocalizerType;
|
|
isLoading: boolean;
|
|
style?: CSSProperties;
|
|
timestamp: number;
|
|
visible: boolean;
|
|
}>;
|
|
|
|
export function TimelineFloatingHeader({
|
|
i18n,
|
|
isLoading,
|
|
style,
|
|
timestamp,
|
|
visible,
|
|
}: PropsType): ReactElement {
|
|
const [hasRendered, setHasRendered] = useState(false);
|
|
const [showSpinner, setShowSpinner] = useState(isLoading);
|
|
|
|
useEffect(() => {
|
|
setHasRendered(true);
|
|
}, []);
|
|
|
|
const [spinnerStyles, spinnerSpringRef] = useSpring(
|
|
() => ({
|
|
delay: 300,
|
|
duration: 250,
|
|
from: { opacity: 1 },
|
|
to: { opacity: 0 },
|
|
onRest: {
|
|
opacity: ({ value }) => {
|
|
if (value === 0) {
|
|
setShowSpinner(false);
|
|
}
|
|
},
|
|
},
|
|
}),
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps -- FIXME
|
|
[isLoading]
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (isLoading) {
|
|
spinnerSpringRef.stop();
|
|
spinnerSpringRef.set({ opacity: 1 });
|
|
setShowSpinner(true);
|
|
}
|
|
|
|
if (!isLoading && showSpinner) {
|
|
drop(Promise.all(spinnerSpringRef.start()));
|
|
}
|
|
|
|
if (!isLoading && !showSpinner) {
|
|
spinnerSpringRef.stop();
|
|
}
|
|
}, [isLoading, showSpinner, spinnerSpringRef]);
|
|
|
|
return (
|
|
<div
|
|
aria-level={5}
|
|
className={classNames(
|
|
'TimelineFloatingHeader',
|
|
`TimelineFloatingHeader--${
|
|
visible && hasRendered ? 'visible' : 'hidden'
|
|
}`
|
|
)}
|
|
role="heading"
|
|
style={style}
|
|
>
|
|
<TimelineDateHeader floating i18n={i18n} timestamp={timestamp} />
|
|
{showSpinner && (
|
|
<animated.div
|
|
className="TimelineFloatingHeader__spinner-container"
|
|
style={spinnerStyles}
|
|
>
|
|
<Spinner direction="on-background" size="20px" svgSize="small" />
|
|
</animated.div>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|