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 { LineItemEditor, 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 { 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 [expandedIndex, setExpandedIndex] = useState(0); 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 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() { const nextIndex = items.length; setItems((prev) => [ ...prev, { date: new Date(), description: "", hours: "1", rate: prev[prev.length - 1]?.rate ?? "0", }, ]); setExpandedIndex(nextIndex); } 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)); setExpandedIndex((current) => { if (current === null) return null; if (current === index) return null; return current > index ? current - 1 : current; }); } 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 ( {clientOptions.length === 0 ? ( Add a client before creating an invoice.