74 lines
2.3 KiB
TypeScript
74 lines
2.3 KiB
TypeScript
|
import React, { useEffect, useRef, useState } from 'react';
|
||
|
import styled from 'styled-components';
|
||
|
import { useStoreActions, useStoreState } from 'easy-peasy';
|
||
|
import { randomInt } from '@/helpers';
|
||
|
import { CSSTransition } from 'react-transition-group';
|
||
|
|
||
|
const BarFill = styled.div`
|
||
|
${tw`h-full bg-cyan-400`};
|
||
|
transition: 250ms ease-in-out;
|
||
|
box-shadow: 0 -2px 10px 2px hsl(178, 78%, 57%);
|
||
|
`;
|
||
|
|
||
|
export default () => {
|
||
|
const interval = useRef<number>(null);
|
||
|
const timeout = useRef<number>(null);
|
||
|
const [ visible, setVisible ] = useState(false);
|
||
|
const progress = useStoreState(state => state.progress.progress);
|
||
|
const continuous = useStoreState(state => state.progress.continuous);
|
||
|
const setProgress = useStoreActions(actions => actions.progress.setProgress);
|
||
|
|
||
|
useEffect(() => {
|
||
|
return () => {
|
||
|
timeout.current && clearTimeout(timeout.current);
|
||
|
interval.current && clearInterval(interval.current);
|
||
|
};
|
||
|
}, []);
|
||
|
|
||
|
useEffect(() => {
|
||
|
setVisible((progress || 0) > 0);
|
||
|
|
||
|
if (progress === 100) {
|
||
|
// @ts-ignore
|
||
|
timeout.current = setTimeout(() => setProgress(undefined), 500);
|
||
|
}
|
||
|
}, [ progress ]);
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (!continuous) {
|
||
|
interval.current && clearInterval(interval.current);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (!progress || progress === 0) {
|
||
|
setProgress(randomInt(20, 30));
|
||
|
}
|
||
|
}, [ continuous ]);
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (continuous) {
|
||
|
interval.current && clearInterval(interval.current);
|
||
|
if ((progress || 0) >= 90) {
|
||
|
setProgress(90);
|
||
|
} else {
|
||
|
// @ts-ignore
|
||
|
interval.current = setTimeout(() => setProgress(progress + randomInt(1, 5)), 500);
|
||
|
}
|
||
|
}
|
||
|
}, [ progress, continuous ]);
|
||
|
|
||
|
return (
|
||
|
<div className={'w-full fixed'} style={{ height: '2px' }}>
|
||
|
<CSSTransition
|
||
|
timeout={250}
|
||
|
appear={true}
|
||
|
in={visible}
|
||
|
unmountOnExit={true}
|
||
|
classNames={'fade'}
|
||
|
>
|
||
|
<BarFill style={{ width: progress === undefined ? '100%' : `${progress}%` }}/>
|
||
|
</CSSTransition>
|
||
|
</div>
|
||
|
);
|
||
|
};
|