Files
nick-tracker/packages/frontend/src/pages/InboxPage.tsx
Debian 64b8e0d80c feat: implement Phase 1 Foundation for AutoScheduler GTD System
- Initialize npm workspaces monorepo (backend, frontend, shared-types)
- Scaffold NestJS backend with modules: Auth, Users, Tasks, Projects, Inbox, Health
- Create React frontend with Vite, TailwindCSS, Radix UI
- Implement TypeORM entities: User, InboxItem, Task, Project
- Add JWT authentication with Passport.js and bcrypt
- Build Inbox capture API (POST /inbox, GET /inbox, POST /inbox/:id/process)
- Create Inbox UI with quick-add form and GTD processing workflow modal
- Configure Docker Compose stack (postgres, redis, backend, frontend)
- Add health check endpoint with database/Redis status
- Write unit tests for auth and inbox services

Phase 1 features complete:
- GTD Inbox Capture: Manual tasks via web form quick-add
- GTD Processing Workflow: Interactive inbox processing interface

Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
2026-01-11 08:42:54 +00:00

129 lines
4.2 KiB
TypeScript

import { useState } from 'react';
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
import { Plus, ArrowRight, Trash2 } from 'lucide-react';
import { inboxApi } from '../lib/api';
import { Button } from '../components/ui/Button';
import { Input } from '../components/ui/Input';
import { Card, CardContent } from '../components/ui/Card';
import { ProcessModal } from '../components/inbox/ProcessModal';
import type { InboxItemResponseDto } from '@nick-tracker/shared-types';
export function InboxPage() {
const queryClient = useQueryClient();
const [newItem, setNewItem] = useState('');
const [selectedItem, setSelectedItem] = useState<InboxItemResponseDto | null>(null);
const { data: items = [], isLoading } = useQuery({
queryKey: ['inbox'],
queryFn: async () => {
const response = await inboxApi.getAll();
return response.data as InboxItemResponseDto[];
},
});
const createMutation = useMutation({
mutationFn: (content: string) => inboxApi.create(content),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['inbox'] });
setNewItem('');
},
});
const deleteMutation = useMutation({
mutationFn: (id: string) => inboxApi.delete(id),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['inbox'] });
},
});
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
if (newItem.trim()) {
createMutation.mutate(newItem.trim());
}
};
const handleProcess = (item: InboxItemResponseDto) => {
setSelectedItem(item);
};
const handleCloseModal = () => {
setSelectedItem(null);
queryClient.invalidateQueries({ queryKey: ['inbox'] });
};
return (
<div className="p-8">
<div className="max-w-3xl mx-auto">
<div className="mb-8">
<h1 className="text-3xl font-bold mb-2">Inbox</h1>
<p className="text-muted-foreground">
Capture everything on your mind. Process items to clarify what they mean.
</p>
</div>
<form onSubmit={handleSubmit} className="mb-8 flex gap-2">
<Input
value={newItem}
onChange={(e) => setNewItem(e.target.value)}
placeholder="What's on your mind?"
className="flex-1"
/>
<Button type="submit" disabled={createMutation.isPending || !newItem.trim()}>
<Plus className="h-4 w-4 mr-2" />
Add
</Button>
</form>
{isLoading ? (
<div className="text-center text-muted-foreground py-8">Loading...</div>
) : items.length === 0 ? (
<Card>
<CardContent className="py-12 text-center">
<p className="text-muted-foreground">
Your inbox is empty. Add items above to capture thoughts and tasks.
</p>
</CardContent>
</Card>
) : (
<div className="space-y-3">
{items.map((item) => (
<Card key={item.id}>
<CardContent className="py-4 flex items-center justify-between">
<div className="flex-1">
<p className="font-medium">{item.content}</p>
<p className="text-sm text-muted-foreground">
{new Date(item.createdAt).toLocaleDateString()}
</p>
</div>
<div className="flex gap-2">
<Button
variant="outline"
size="sm"
onClick={() => handleProcess(item)}
>
<ArrowRight className="h-4 w-4 mr-2" />
Process
</Button>
<Button
variant="ghost"
size="icon"
onClick={() => deleteMutation.mutate(item.id)}
>
<Trash2 className="h-4 w-4 text-destructive" />
</Button>
</div>
</CardContent>
</Card>
))}
</div>
)}
</div>
{selectedItem && (
<ProcessModal item={selectedItem} onClose={handleCloseModal} />
)}
</div>
);
}