mirror of
https://github.com/hex248/sprint.git
synced 2026-02-08 18:33:01 +00:00
improved timer system and overlay
This commit is contained in:
94
packages/frontend/src/components/active-timers-overlay.tsx
Normal file
94
packages/frontend/src/components/active-timers-overlay.tsx
Normal file
@@ -0,0 +1,94 @@
|
||||
import { useEffect, useMemo, useState } from "react";
|
||||
import { IssueModal } from "@/components/issue-modal";
|
||||
import { useSessionSafe } from "@/components/session-provider";
|
||||
import { TimerControls } from "@/components/timer-controls";
|
||||
import { getWorkTimeMs } from "@/components/timer-display";
|
||||
import { Button } from "@/components/ui/button";
|
||||
import { useActiveTimers, useInactiveTimers, useIssueById } from "@/lib/query/hooks";
|
||||
import { issueID } from "@/lib/utils";
|
||||
|
||||
const REFRESH_INTERVAL_MS = 10000;
|
||||
|
||||
export function ActiveTimersOverlay() {
|
||||
const session = useSessionSafe();
|
||||
const { data: activeTimers = [] } = useActiveTimers({
|
||||
refetchInterval: REFRESH_INTERVAL_MS,
|
||||
enabled: Boolean(session?.user),
|
||||
});
|
||||
const [tick, setTick] = useState(0);
|
||||
|
||||
const hasRunning = useMemo(() => activeTimers.some((timer) => timer.isRunning), [activeTimers]);
|
||||
|
||||
useEffect(() => {
|
||||
if (!hasRunning) return;
|
||||
const interval = window.setInterval(() => {
|
||||
setTick((value) => value + 1);
|
||||
}, 1000);
|
||||
return () => window.clearInterval(interval);
|
||||
}, [hasRunning]);
|
||||
|
||||
void tick;
|
||||
|
||||
if (!session?.user || activeTimers.length === 0) return null;
|
||||
|
||||
return (
|
||||
<div className="fixed bottom-[calc(1rem+env(safe-area-inset-bottom))] left-[calc(1rem+env(safe-area-inset-left))] z-50 flex flex-col gap-2">
|
||||
{activeTimers.map((timer) => (
|
||||
<ActiveTimerItem key={timer.id} timer={timer} />
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
function ActiveTimerItem({
|
||||
timer,
|
||||
}: {
|
||||
timer: {
|
||||
id: number;
|
||||
issueId: number;
|
||||
issueNumber: number;
|
||||
projectKey: string;
|
||||
timestamps: string[];
|
||||
isRunning: boolean;
|
||||
};
|
||||
}) {
|
||||
const { data: issueData } = useIssueById(timer.issueId);
|
||||
const { data: inactiveTimers = [] } = useInactiveTimers(timer.issueId, { refetchInterval: 10000 });
|
||||
const [open, setOpen] = useState(false);
|
||||
const issueKey = issueID(timer.projectKey, timer.issueNumber);
|
||||
|
||||
const inactiveWorkTimeMs = inactiveTimers.reduce(
|
||||
(total, session) => total + getWorkTimeMs(session?.timestamps),
|
||||
0,
|
||||
);
|
||||
const currentWorkTimeMs = getWorkTimeMs(timer.timestamps);
|
||||
const totalWorkTimeMs = inactiveWorkTimeMs + currentWorkTimeMs;
|
||||
|
||||
const issueKeyNode = issueData ? (
|
||||
<IssueModal
|
||||
issueData={issueData}
|
||||
open={open}
|
||||
onOpenChange={setOpen}
|
||||
trigger={
|
||||
<Button variant="link" size="none" className="text-sm tabular-nums truncate min-w-0 font-700">
|
||||
{issueKey}
|
||||
</Button>
|
||||
}
|
||||
/>
|
||||
) : (
|
||||
<Button variant="link" size="none" className="text-sm tabular-nums truncate min-w-0 font-700" disabled>
|
||||
{issueKey}
|
||||
</Button>
|
||||
);
|
||||
|
||||
return (
|
||||
<TimerControls
|
||||
issueId={timer.issueId}
|
||||
issueKey={issueKeyNode}
|
||||
timestamps={timer.timestamps}
|
||||
isRunning={timer.isRunning}
|
||||
totalTimeMs={totalWorkTimeMs}
|
||||
className="border bg-background/95 pl-2 pr-1 py-1"
|
||||
/>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user