// 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 (