forked from github/plane
Compare commits
1 Commits
preview
...
feat/timer
Author | SHA1 | Date | |
---|---|---|---|
|
72ab77fa6f |
154
apps/app/hooks/use-stopwatch.tsx
Normal file
154
apps/app/hooks/use-stopwatch.tsx
Normal file
@ -0,0 +1,154 @@
|
||||
import { useState, useEffect, useCallback, useMemo } from "react";
|
||||
|
||||
// hooks
|
||||
import useLocalStorage from "hooks/use-local-storage";
|
||||
|
||||
type TimeSlot = {
|
||||
start_time: Date | string;
|
||||
end_time: Date | null | string;
|
||||
type: "START" | "PAUSE" | "RESUME" | "STOP";
|
||||
actual_time: number;
|
||||
};
|
||||
|
||||
type Props = {
|
||||
key: string | null;
|
||||
};
|
||||
|
||||
const defaultValue: {
|
||||
actual_time_spent: number;
|
||||
time_slots: TimeSlot[];
|
||||
} = {
|
||||
actual_time_spent: 0,
|
||||
time_slots: [],
|
||||
};
|
||||
|
||||
const useStopwatch = (props: Props) => {
|
||||
const { key } = props;
|
||||
|
||||
const { storedValue, setValue } = useLocalStorage(key ?? "", defaultValue);
|
||||
|
||||
const [elapsed, setElapsed] = useState(0);
|
||||
|
||||
const start = useCallback(() => {
|
||||
setValue({
|
||||
actual_time_spent: 0,
|
||||
time_slots: [
|
||||
{
|
||||
start_time: new Date(),
|
||||
end_time: null,
|
||||
type: "START",
|
||||
actual_time: 0,
|
||||
},
|
||||
],
|
||||
});
|
||||
}, [setValue]);
|
||||
|
||||
const stop = useCallback(() => {
|
||||
if (!storedValue || storedValue.time_slots.length <= 0) return;
|
||||
|
||||
const lastTimeSlot = storedValue.time_slots[storedValue.time_slots.length - 1];
|
||||
|
||||
if (lastTimeSlot.type === "STOP") return;
|
||||
|
||||
const newTimeSlot = {
|
||||
start_time: lastTimeSlot.start_time,
|
||||
end_time: new Date(),
|
||||
type: "STOP" as const,
|
||||
actual_time:
|
||||
lastTimeSlot.type === "PAUSE"
|
||||
? lastTimeSlot.actual_time
|
||||
: lastTimeSlot.actual_time +
|
||||
(new Date().getTime() - new Date(lastTimeSlot.start_time).getTime()),
|
||||
};
|
||||
|
||||
setValue({
|
||||
actual_time_spent: storedValue.actual_time_spent + newTimeSlot.actual_time,
|
||||
time_slots: [...storedValue.time_slots, newTimeSlot],
|
||||
});
|
||||
}, [setValue, storedValue]);
|
||||
|
||||
const pause = useCallback(() => {
|
||||
if (!storedValue || storedValue.time_slots.length <= 0) return;
|
||||
|
||||
const lastTimeSlot = storedValue.time_slots[storedValue.time_slots.length - 1];
|
||||
|
||||
if (lastTimeSlot.type === "PAUSE") return;
|
||||
|
||||
const newTimeSlot = {
|
||||
start_time: lastTimeSlot.start_time,
|
||||
end_time: new Date(),
|
||||
type: "PAUSE" as const,
|
||||
actual_time:
|
||||
lastTimeSlot.actual_time +
|
||||
(new Date().getTime() - new Date(lastTimeSlot.start_time).getTime()),
|
||||
};
|
||||
|
||||
setValue({
|
||||
actual_time_spent: storedValue.actual_time_spent + newTimeSlot.actual_time,
|
||||
time_slots: [...storedValue.time_slots, newTimeSlot],
|
||||
});
|
||||
}, [setValue, storedValue]);
|
||||
|
||||
const resume = useCallback(() => {
|
||||
if (!storedValue || storedValue.time_slots.length <= 0) return;
|
||||
|
||||
const lastTimeSlot = storedValue.time_slots[storedValue.time_slots.length - 1];
|
||||
|
||||
if (lastTimeSlot.type === "RESUME") return;
|
||||
|
||||
const newTimeSlot = {
|
||||
start_time: new Date(),
|
||||
end_time: null,
|
||||
type: "RESUME" as const,
|
||||
actual_time: lastTimeSlot.actual_time,
|
||||
};
|
||||
|
||||
setValue({
|
||||
actual_time_spent: storedValue.actual_time_spent,
|
||||
time_slots: [...storedValue.time_slots, newTimeSlot],
|
||||
});
|
||||
}, [setValue, storedValue]);
|
||||
|
||||
const reset = useCallback(() => {
|
||||
setElapsed(0);
|
||||
setValue(defaultValue);
|
||||
}, [setValue]);
|
||||
|
||||
const isRunning = useMemo(() => {
|
||||
if (!storedValue || storedValue.time_slots.length <= 0) return false;
|
||||
|
||||
const lastTimeSlot = storedValue.time_slots[storedValue.time_slots.length - 1];
|
||||
|
||||
return lastTimeSlot.type === "START" || lastTimeSlot.type === "RESUME";
|
||||
}, [storedValue]);
|
||||
|
||||
useEffect(() => {
|
||||
if (!storedValue || storedValue.time_slots.length <= 0) return;
|
||||
|
||||
const lastTimeSlot = storedValue.time_slots[storedValue.time_slots.length - 1];
|
||||
|
||||
if (lastTimeSlot.type === "START" || lastTimeSlot.type === "RESUME") {
|
||||
const interval = setInterval(() => {
|
||||
setElapsed(
|
||||
lastTimeSlot.actual_time +
|
||||
(new Date().getTime() - new Date(lastTimeSlot.start_time).getTime())
|
||||
);
|
||||
}, 1000);
|
||||
|
||||
return () => clearInterval(interval);
|
||||
} else setElapsed(lastTimeSlot.actual_time);
|
||||
}, [storedValue, setElapsed]);
|
||||
|
||||
return {
|
||||
elapsed,
|
||||
isRunning,
|
||||
start,
|
||||
stop,
|
||||
isStopped: storedValue?.time_slots[storedValue?.time_slots.length - 1]?.type === "STOP",
|
||||
reset,
|
||||
pause,
|
||||
resume,
|
||||
} as const;
|
||||
};
|
||||
|
||||
export default useStopwatch;
|
Loading…
Reference in New Issue
Block a user