import { router, Stack, useLocalSearchParams } from "expo-router"; import { useEffect, useMemo, useState } from "react"; import { Alert, KeyboardAvoidingView, Platform, Pressable, ScrollView, StyleSheet, Text, View, } from "react-native"; import { AppBackground } from "@/components/AppBackground"; import { InvoiceEditorSectionTabs, type InvoiceEditorSection, } from "@/components/invoices/InvoiceEditorSectionTabs"; import { InvoicePdfPreview } from "@/components/invoices/InvoicePdfPreview"; import { InvoiceTotals } from "@/components/invoices/InvoiceTotals"; import { LineItemEditor, LineItemsTableHeader, type EditableLineItem } from "@/components/invoices/LineItemEditor"; import { LoadingScreen } from "@/components/LoadingScreen"; import { Button } from "@/components/ui/Button"; import { Card } from "@/components/ui/Card"; import { DateTimeField } from "@/components/ui/DateTimeField"; import { Input } from "@/components/ui/Input"; import { fonts, spacing } from "@/constants/theme"; import { useAppTheme } from "@/contexts/ThemeContext"; import { formatCurrency } from "@/lib/format"; import { getInvoiceStatus } from "@/lib/invoice-status"; import { buildPreviewPdfInput } from "@/lib/invoice-pdf-input"; import { validateLineItems } from "@/lib/form-validation"; import { ensureNotificationPermissions } from "@/lib/invoice-send-reminders"; import { useTabBarScrollPadding } from "@/lib/tab-bar-insets"; import type { ThemeColors } from "@/lib/theme-palette"; import { useThemedStyles } from "@/lib/use-themed-styles"; import { api } from "@/lib/trpc"; export default function InvoiceEditScreen() { const { colors } = useAppTheme(); const styles = useThemedStyles(createInvoiceEditStyles); const { id } = useLocalSearchParams<{ id: string }>(); const utils = api.useUtils(); const scrollPadding = useTabBarScrollPadding(); const invoiceQuery = api.invoices.getById.useQuery( { id: id ?? "" }, { enabled: Boolean(id) }, ); const [notes, setNotes] = useState(""); const [dueDate, setDueDate] = useState(() => new Date()); const [sendReminderAt, setSendReminderAt] = useState(null); const [items, setItems] = useState([]); const [section, setSection] = useState("edit"); const [error, setError] = useState(null); useEffect(() => { const invoice = invoiceQuery.data; if (!invoice) return; setNotes(invoice.notes ?? ""); setDueDate(new Date(invoice.dueDate)); setSendReminderAt(invoice.sendReminderAt ? new Date(invoice.sendReminderAt) : null); setItems( invoice.items.map((item) => ({ id: item.id, date: new Date(item.date), description: item.description, hours: String(item.hours), rate: String(item.rate), })), ); }, [invoiceQuery.data]); const updateInvoice = api.invoices.update.useMutation({ onSuccess: () => { void utils.invoices.getById.invalidate({ id: id ?? "" }); void utils.invoices.getAll.invalidate(); void utils.invoices.getAll.invalidate({ status: "draft" }); void utils.dashboard.getStats.invalidate(); Alert.alert("Saved", "Invoice updated", [ { text: "OK", onPress: () => router.back() }, ]); }, onError: (err) => setError(err.message), }); const sendInvoice = api.email.sendInvoice.useMutation({ onSuccess: (data) => { Alert.alert("Invoice sent", data.message); void utils.invoices.getById.invalidate({ id: id ?? "" }); void utils.invoices.getAll.invalidate(); void utils.dashboard.getStats.invalidate(); }, onError: (err) => Alert.alert("Could not send invoice", err.message), }); const invoice = invoiceQuery.data; const isDraft = invoice?.status === "draft"; const subtotal = useMemo( () => items.reduce((sum, item) => { const hours = Number(item.hours) || 0; const rate = Number(item.rate) || 0; return sum + hours * rate; }, 0), [items], ); const taxRate = invoice?.taxRate ?? 0; const taxAmount = subtotal * (taxRate / 100); const total = subtotal + taxAmount; const currency = invoice?.currency ?? "USD"; const lineItemsError = isDraft ? validateLineItems(items) : null; const canSave = isDraft ? !lineItemsError : true; const previewInput = useMemo(() => { if (!invoice) return null; return buildPreviewPdfInput({ invoiceNumber: invoice.invoiceNumber, invoicePrefix: invoice.invoicePrefix, businessId: invoice.businessId, clientId: invoice.clientId, issueDate: new Date(invoice.issueDate), dueDate, status: invoice.status as "draft" | "sent" | "paid", notes, taxRate, currency, items, }); }, [invoice, dueDate, notes, taxRate, currency, items]); if (!id) { return ; } if (invoiceQuery.isLoading) { return ; } if (!invoice) { return ; } const status = getInvoiceStatus(invoice); const clientEmail = invoice.client?.email?.trim() ?? ""; function promptSendInvoice() { if (!clientEmail) { Alert.alert( "No client email", "Add an email address to this client on the web app before sending invoices.", ); return; } Alert.alert( status === "draft" ? "Send invoice" : "Resend invoice", `Email this invoice to ${clientEmail}?`, [ { text: "Cancel", style: "cancel" }, { text: "Send", onPress: () => sendInvoice.mutate({ invoiceId: invoice!.id }), }, ], ); } function updateItem(index: number, patch: Partial) { setItems((prev) => prev.map((item, i) => (i === index ? { ...item, ...patch } : item))); } function addItem() { setItems((prev) => [ ...prev, { date: new Date(), description: "", hours: "1", rate: prev[prev.length - 1]?.rate ?? "0", }, ]); } function removeItem(index: number) { if (items.length <= 1) { Alert.alert("Cannot remove", "An invoice needs at least one line item."); return; } setItems((prev) => prev.filter((_, i) => i !== index)); } async function handleSave() { if (!canSave) return; setError(null); if (isDraft && sendReminderAt) { const granted = await ensureNotificationPermissions(); if (!granted) { Alert.alert( "Notifications disabled", "Turn on notifications in Settings to get reminded when it's time to send this invoice.", ); } } const parsedItems: Array<{ date: Date; description: string; hours: number; rate: number; }> = []; for (const item of items) { parsedItems.push({ date: item.date, description: item.description.trim(), hours: Number(item.hours), rate: Number(item.rate), }); } updateInvoice.mutate({ id, notes, dueDate, sendReminderAt, ...(isDraft ? { items: parsedItems, } : {}), }); } return ( {invoice.invoicePrefix} {invoice.invoiceNumber} {invoice.client?.name ?? "Client"} {section === "preview" ? ( ) : ( <> {isDraft ? ( <> {sendReminderAt ? ( setSendReminderAt(null)}> Clear send reminder ) : null} ) : null} {!isDraft ? ( Line items are locked after an invoice is sent. Mark as draft on the invoice screen to edit entries. ) : ( )} {items.map((item, index) => ( updateItem(index, patch)} onRemove={() => removeItem(index)} readOnly={!isDraft} /> ))} {isDraft ? ( + Add line ) : null} 0 ? `Tax (${taxRate}%)` : undefined} taxAmount={taxRate > 0 ? formatCurrency(taxAmount, currency) : undefined} total={formatCurrency(total, currency)} /> {lineItemsError ? {lineItemsError} : null} {error ? {error} : null}