import { router, Stack } 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 { SelectField } from "@/components/ui/SelectField"; import { fonts, spacing } from "@/constants/theme"; import { useAppTheme } from "@/contexts/ThemeContext"; import { formatCurrency } from "@/lib/format"; import { defaultDueDate, generateInvoiceNumber } from "@/lib/invoice-number"; import { buildPreviewPdfInput } from "@/lib/invoice-pdf-input"; import { isRequiredString, isValidTaxRate, validateLineItems, } from "@/lib/form-validation"; 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 NewInvoiceScreen() { const { colors } = useAppTheme(); const styles = useThemedStyles(createNewInvoiceStyles); const utils = api.useUtils(); const scrollPadding = useTabBarScrollPadding(); const clientsQuery = api.clients.getAll.useQuery(); const [clientId, setClientId] = useState(""); const [invoiceNumber, setInvoiceNumber] = useState(generateInvoiceNumber); const [issueDate, setIssueDate] = useState(() => new Date()); const [dueDate, setDueDate] = useState(() => defaultDueDate(new Date())); const [notes, setNotes] = useState(""); const [taxRate, setTaxRate] = useState("0"); const [items, setItems] = useState([ { date: new Date(), description: "", hours: "1", rate: "0", }, ]); const [section, setSection] = useState("edit"); const [error, setError] = useState(null); const clientOptions = useMemo( () => (clientsQuery.data ?? []).map((client) => ({ label: client.name, value: client.id, })), [clientsQuery.data], ); const selectedClient = clientsQuery.data?.find((client) => client.id === clientId); const currency = selectedClient?.currency ?? "USD"; useEffect(() => { if (!selectedClient?.defaultHourlyRate) return; setItems((prev) => prev.map((item, index) => index === 0 && (item.rate === "0" || item.rate === "") ? { ...item, rate: String(selectedClient.defaultHourlyRate) } : item, ), ); }, [selectedClient?.defaultHourlyRate, selectedClient?.id]); const createInvoice = api.invoices.create.useMutation({ onSuccess: (invoice) => { void utils.invoices.getAll.invalidate(); void utils.dashboard.getStats.invalidate(); Alert.alert("Invoice created", "Your draft invoice is ready.", [ { text: "View invoice", onPress: () => router.replace(`/(app)/invoices/${invoice.id}`), }, ]); }, onError: (err) => setError(err.message), }); 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 parsedTaxRate = Number(taxRate) || 0; const taxAmount = subtotal * (parsedTaxRate / 100); const total = subtotal + taxAmount; const previewInput = useMemo( () => buildPreviewPdfInput({ invoiceNumber, clientId, issueDate, dueDate, taxRate: parsedTaxRate, currency, notes, items, }), [invoiceNumber, clientId, issueDate, dueDate, parsedTaxRate, currency, notes, items], ); const clientError = clientId ? undefined : "Select a client"; const invoiceNumberError = isRequiredString(invoiceNumber) ? undefined : "Invoice number is required"; const taxError = isValidTaxRate(taxRate) ? undefined : "Tax rate must be between 0 and 100"; const lineItemsError = validateLineItems(items); const canCreate = clientOptions.length > 0 && !clientError && !invoiceNumberError && !taxError && !lineItemsError; if (clientsQuery.isLoading) { return ; } 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)); } function handleCreate() { if (!canCreate) return; setError(null); 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), }); } createInvoice.mutate({ clientId, invoiceNumber: invoiceNumber.trim(), issueDate, dueDate, notes, taxRate: Number(taxRate), currency, items: parsedItems, status: "draft", }); } return ( {section === "preview" ? ( ) : ( <> {clientOptions.length === 0 ? ( Add a client before creating an invoice.