mirror of
https://github.com/remvze/moodist.git
synced 2025-12-17 08:54:13 +00:00
176 lines
4.3 KiB
TypeScript
176 lines
4.3 KiB
TypeScript
import { useState, useEffect, useRef, useMemo } from 'react';
|
|
import { FaUndo, FaPlay, FaPause } from 'react-icons/fa/index';
|
|
import { IoMdSettings } from 'react-icons/io/index';
|
|
|
|
import { Modal } from '@/components/modal';
|
|
import { Tabs } from './tabs';
|
|
import { Timer } from './timer';
|
|
import { Button } from './button';
|
|
import { Setting } from './setting';
|
|
|
|
import { useLocalStorage } from '@/hooks/use-local-storage';
|
|
import { useSoundEffect } from '@/hooks/use-sound-effect';
|
|
import { usePomodoroStore } from '@/store';
|
|
|
|
import styles from './pomodoro.module.css';
|
|
|
|
interface PomodoroProps {
|
|
onClose: () => void;
|
|
open: () => void;
|
|
show: boolean;
|
|
}
|
|
|
|
export function Pomodoro({ onClose, open, show }: PomodoroProps) {
|
|
const [showSetting, setShowSetting] = useState(false);
|
|
|
|
const [selectedTab, setSelectedTab] = useState('pomodoro');
|
|
|
|
const running = usePomodoroStore(state => state.running);
|
|
const setRunning = usePomodoroStore(state => state.setRunning);
|
|
|
|
const [timer, setTimer] = useState(0);
|
|
const interval = useRef<ReturnType<typeof setInterval> | null>(null);
|
|
|
|
const alarm = useSoundEffect('/sounds/alarm.mp3');
|
|
|
|
const defaultTimes = useMemo(
|
|
() => ({
|
|
long: 15 * 60,
|
|
pomodoro: 25 * 60,
|
|
short: 5 * 60,
|
|
}),
|
|
[],
|
|
);
|
|
|
|
const [times, setTimes] = useLocalStorage<Record<string, number>>(
|
|
'moodist-pomodoro-setting',
|
|
defaultTimes,
|
|
);
|
|
|
|
const [completions, setCompletions] = useState<Record<string, number>>({
|
|
long: 0,
|
|
pomodoro: 0,
|
|
short: 0,
|
|
});
|
|
|
|
const tabs = useMemo(
|
|
() => [
|
|
{ id: 'pomodoro', label: 'Pomodoro' },
|
|
{ id: 'short', label: 'Break' },
|
|
{ id: 'long', label: 'Long Break' },
|
|
],
|
|
[],
|
|
);
|
|
|
|
useEffect(() => {
|
|
if (running) {
|
|
if (interval.current) clearInterval(interval.current);
|
|
|
|
interval.current = setInterval(() => {
|
|
setTimer(prev => prev - 1);
|
|
}, 1000);
|
|
} else {
|
|
if (interval.current) clearInterval(interval.current);
|
|
}
|
|
}, [running]);
|
|
|
|
useEffect(() => {
|
|
if (timer <= 0 && running) {
|
|
if (interval.current) clearInterval(interval.current);
|
|
|
|
alarm.play();
|
|
|
|
setRunning(false);
|
|
setCompletions(prev => ({
|
|
...prev,
|
|
[selectedTab]: prev[selectedTab] + 1,
|
|
}));
|
|
}
|
|
}, [timer, selectedTab, running, setRunning, alarm]);
|
|
|
|
useEffect(() => {
|
|
const time = times[selectedTab] || 10;
|
|
|
|
if (interval.current) clearInterval(interval.current);
|
|
|
|
setRunning(false);
|
|
setTimer(time);
|
|
}, [selectedTab, times, setRunning]);
|
|
|
|
const toggleRunning = () => {
|
|
if (running) setRunning(false);
|
|
else if (timer <= 0) {
|
|
const time = times[selectedTab] || 10;
|
|
|
|
setTimer(time);
|
|
setRunning(true);
|
|
} else setRunning(true);
|
|
};
|
|
|
|
const restart = () => {
|
|
if (interval.current) clearInterval(interval.current);
|
|
|
|
const time = times[selectedTab] || 10;
|
|
|
|
setRunning(false);
|
|
setTimer(time);
|
|
};
|
|
|
|
return (
|
|
<>
|
|
<Modal show={show} onClose={onClose}>
|
|
<header className={styles.header}>
|
|
<h2 className={styles.title}>Pomodoro Timer</h2>
|
|
|
|
<div className={styles.button}>
|
|
<Button
|
|
icon={<IoMdSettings />}
|
|
tooltip="Change Times"
|
|
onClick={() => {
|
|
onClose();
|
|
setShowSetting(true);
|
|
}}
|
|
/>
|
|
</div>
|
|
</header>
|
|
|
|
<Tabs selectedTab={selectedTab} tabs={tabs} onSelect={setSelectedTab} />
|
|
<Timer timer={timer} />
|
|
|
|
<div className={styles.control}>
|
|
<p className={styles.completed}>
|
|
{completions[selectedTab] || 0} completed
|
|
</p>
|
|
<div className={styles.buttons}>
|
|
<Button
|
|
icon={<FaUndo />}
|
|
smallIcon
|
|
tooltip="Restart"
|
|
onClick={restart}
|
|
/>
|
|
<Button
|
|
icon={running ? <FaPause /> : <FaPlay />}
|
|
smallIcon
|
|
tooltip={running ? 'Pause' : 'Start'}
|
|
onClick={toggleRunning}
|
|
/>
|
|
</div>
|
|
</div>
|
|
</Modal>
|
|
|
|
<Setting
|
|
show={showSetting}
|
|
times={times}
|
|
onChange={times => {
|
|
setShowSetting(false);
|
|
setTimes(times);
|
|
open();
|
|
}}
|
|
onClose={() => {
|
|
setShowSetting(false);
|
|
open();
|
|
}}
|
|
/>
|
|
</>
|
|
);
|
|
}
|