mirror of
https://github.com/soconnor0919/beenvoice.git
synced 2025-12-13 09:34:44 -05:00
The changes add row click functionality and consistent action button styling across data tables. Main updates: - Add `onRowClick` handler to make rows clickable and navigate to details pages - Add `data-action-button` attribute to exclude action buttons from row click - Fix TypeScript errors and types
441 lines
13 KiB
TypeScript
441 lines
13 KiB
TypeScript
"use client";
|
|
|
|
import * as React from "react";
|
|
import { Button } from "~/components/ui/button";
|
|
import { Input } from "~/components/ui/input";
|
|
import { Label } from "~/components/ui/label";
|
|
import { DatePicker } from "~/components/ui/date-picker";
|
|
import { NumberInput } from "~/components/ui/number-input";
|
|
import {
|
|
Trash2,
|
|
Plus,
|
|
GripVertical,
|
|
ChevronUp,
|
|
ChevronDown,
|
|
} from "lucide-react";
|
|
import { cn } from "~/lib/utils";
|
|
|
|
interface InvoiceItem {
|
|
id: string;
|
|
date: Date;
|
|
description: string;
|
|
hours: number;
|
|
rate: number;
|
|
amount: number;
|
|
}
|
|
|
|
interface InvoiceLineItemsProps {
|
|
items: InvoiceItem[];
|
|
onAddItem: () => void;
|
|
onRemoveItem: (index: number) => void;
|
|
onUpdateItem: (
|
|
index: number,
|
|
field: string,
|
|
value: string | number | Date,
|
|
) => void;
|
|
onMoveUp: (index: number) => void;
|
|
onMoveDown: (index: number) => void;
|
|
className?: string;
|
|
}
|
|
|
|
interface LineItemRowProps {
|
|
item: InvoiceItem;
|
|
index: number;
|
|
canRemove: boolean;
|
|
onRemove: (index: number) => void;
|
|
onUpdate: (
|
|
index: number,
|
|
field: string,
|
|
value: string | number | Date,
|
|
) => void;
|
|
onMoveUp: (index: number) => void;
|
|
onMoveDown: (index: number) => void;
|
|
isFirst: boolean;
|
|
isLast: boolean;
|
|
}
|
|
|
|
function LineItemRow({
|
|
item,
|
|
index,
|
|
canRemove,
|
|
onRemove,
|
|
onUpdate,
|
|
}: LineItemRowProps) {
|
|
return (
|
|
<>
|
|
{/* Desktop Layout - Table Row */}
|
|
<tr className="group hover:bg-muted/20 hidden transition-colors lg:table-row">
|
|
{/* Drag Handle */}
|
|
<td className="w-6 p-2 text-center align-top">
|
|
<GripVertical className="text-muted-foreground mt-1 h-4 w-4 cursor-grab" />
|
|
</td>
|
|
|
|
{/* Main Content */}
|
|
<td className="p-2" colSpan={5}>
|
|
{/* Description */}
|
|
<div className="mb-3">
|
|
<Input
|
|
value={item.description}
|
|
onChange={(e) => onUpdate(index, "description", e.target.value)}
|
|
placeholder="Describe the work performed..."
|
|
className="w-full border-0 bg-transparent py-0 pr-0 pl-2 text-sm font-medium focus-visible:ring-0"
|
|
/>
|
|
</div>
|
|
|
|
{/* Controls Row */}
|
|
<div className="flex items-center gap-3">
|
|
{/* Date */}
|
|
<DatePicker
|
|
date={item.date}
|
|
onDateChange={(date) =>
|
|
onUpdate(index, "date", date ?? new Date())
|
|
}
|
|
size="sm"
|
|
className="h-9 w-28"
|
|
/>
|
|
|
|
{/* Hours */}
|
|
<NumberInput
|
|
value={item.hours}
|
|
onChange={(value) => onUpdate(index, "hours", value)}
|
|
min={0}
|
|
step={0.25}
|
|
width="auto"
|
|
className="h-9 w-28"
|
|
/>
|
|
|
|
{/* Rate */}
|
|
<NumberInput
|
|
value={item.rate}
|
|
onChange={(value) => onUpdate(index, "rate", value)}
|
|
min={0}
|
|
step={1}
|
|
prefix="$"
|
|
width="auto"
|
|
className="h-9 w-28"
|
|
/>
|
|
|
|
{/* Amount */}
|
|
<div className="ml-auto">
|
|
<span className="text-primary font-semibold">
|
|
${(item.hours * item.rate).toFixed(2)}
|
|
</span>
|
|
</div>
|
|
|
|
{/* Actions */}
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="sm"
|
|
onClick={() => onRemove(index)}
|
|
className={cn(
|
|
"text-muted-foreground h-8 w-8 p-0 transition-colors hover:text-red-500",
|
|
!canRemove && "cursor-not-allowed opacity-50",
|
|
)}
|
|
disabled={!canRemove}
|
|
aria-label="Remove item"
|
|
>
|
|
<Trash2 className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
|
|
{/* Tablet Layout - Condensed Row */}
|
|
<tr className="group hover:bg-muted/20 hidden transition-colors md:table-row lg:hidden">
|
|
{/* Drag Handle */}
|
|
<td className="w-6 p-2 text-center align-top">
|
|
<GripVertical className="text-muted-foreground mt-1 h-4 w-4 cursor-grab" />
|
|
</td>
|
|
|
|
{/* Main Content - Description on top, inputs below */}
|
|
<td className="p-3" colSpan={6}>
|
|
{/* Description */}
|
|
<div className="mb-3">
|
|
<Input
|
|
value={item.description}
|
|
onChange={(e) => onUpdate(index, "description", e.target.value)}
|
|
placeholder="Describe the work performed..."
|
|
className="w-full pl-3 text-sm font-medium"
|
|
/>
|
|
</div>
|
|
|
|
{/* Controls Row - Date/Hours/Rate break to separate rows on smaller screens */}
|
|
<div className="flex flex-wrap items-center gap-2 sm:gap-3">
|
|
<DatePicker
|
|
date={item.date}
|
|
onDateChange={(date) =>
|
|
onUpdate(index, "date", date ?? new Date())
|
|
}
|
|
size="sm"
|
|
className="h-9 w-full sm:w-28"
|
|
/>
|
|
<NumberInput
|
|
value={item.hours}
|
|
onChange={(value) => onUpdate(index, "hours", value)}
|
|
min={0}
|
|
step={0.25}
|
|
width="full"
|
|
className="h-9 w-1/2 sm:w-28"
|
|
/>
|
|
<NumberInput
|
|
value={item.rate}
|
|
onChange={(value) => onUpdate(index, "rate", value)}
|
|
min={0}
|
|
step={1}
|
|
prefix="$"
|
|
width="full"
|
|
className="h-9 w-1/2 sm:w-28"
|
|
/>
|
|
|
|
{/* Amount and Actions - inline with controls on larger screens */}
|
|
<div className="mt-3 flex items-center justify-between sm:mt-0 sm:ml-auto sm:gap-3">
|
|
<span className="text-primary font-semibold">
|
|
${(item.hours * item.rate).toFixed(2)}
|
|
</span>
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="sm"
|
|
onClick={() => onRemove(index)}
|
|
className={cn(
|
|
"text-muted-foreground h-8 w-8 p-0 transition-colors hover:text-red-500",
|
|
!canRemove && "cursor-not-allowed opacity-50",
|
|
)}
|
|
disabled={!canRemove}
|
|
aria-label="Remove item"
|
|
>
|
|
<Trash2 className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
</>
|
|
);
|
|
}
|
|
|
|
function MobileLineItem({
|
|
item,
|
|
index,
|
|
canRemove,
|
|
onRemove,
|
|
onUpdate,
|
|
onMoveUp,
|
|
onMoveDown,
|
|
isFirst,
|
|
isLast,
|
|
}: LineItemRowProps) {
|
|
return (
|
|
<div className="bg-card space-y-3 rounded-lg border p-4 md:hidden">
|
|
{/* Description */}
|
|
<div className="space-y-1">
|
|
<Label className="text-muted-foreground text-xs">Description</Label>
|
|
<Input
|
|
value={item.description}
|
|
onChange={(e) => onUpdate(index, "description", e.target.value)}
|
|
placeholder="Describe the work performed..."
|
|
className="pl-3 text-sm"
|
|
/>
|
|
</div>
|
|
|
|
{/* Date */}
|
|
<div className="space-y-1">
|
|
<Label className="text-muted-foreground text-xs">Date</Label>
|
|
<DatePicker
|
|
date={item.date}
|
|
onDateChange={(date) => onUpdate(index, "date", date ?? new Date())}
|
|
size="sm"
|
|
/>
|
|
</div>
|
|
|
|
{/* Hours and Rate in a row */}
|
|
<div className="grid grid-cols-2 gap-3">
|
|
<div className="space-y-1">
|
|
<Label className="text-muted-foreground text-xs">Hours</Label>
|
|
<NumberInput
|
|
value={item.hours}
|
|
onChange={(value) => onUpdate(index, "hours", value)}
|
|
min={0}
|
|
step={0.25}
|
|
width="full"
|
|
/>
|
|
</div>
|
|
<div className="space-y-1">
|
|
<Label className="text-muted-foreground text-xs">Rate</Label>
|
|
<NumberInput
|
|
value={item.rate}
|
|
onChange={(value) => onUpdate(index, "rate", value)}
|
|
min={0}
|
|
step={1}
|
|
prefix="$"
|
|
width="full"
|
|
/>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Bottom section with controls, item name, and total */}
|
|
<div className="flex items-center justify-between border-t pt-2">
|
|
<div className="flex items-center gap-2">
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="sm"
|
|
onClick={() => onMoveUp(index)}
|
|
className={cn(
|
|
"h-8 w-8 p-0 transition-colors",
|
|
isFirst
|
|
? "text-muted-foreground/50 cursor-not-allowed"
|
|
: "text-muted-foreground hover:text-foreground",
|
|
)}
|
|
disabled={isFirst}
|
|
aria-label="Move up"
|
|
>
|
|
<ChevronUp className="h-4 w-4" />
|
|
</Button>
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="sm"
|
|
onClick={() => onMoveDown(index)}
|
|
className={cn(
|
|
"h-8 w-8 p-0 transition-colors",
|
|
isLast
|
|
? "text-muted-foreground/50 cursor-not-allowed"
|
|
: "text-muted-foreground hover:text-foreground",
|
|
)}
|
|
disabled={isLast}
|
|
aria-label="Move down"
|
|
>
|
|
<ChevronDown className="h-4 w-4" />
|
|
</Button>
|
|
<Button
|
|
type="button"
|
|
variant="ghost"
|
|
size="sm"
|
|
onClick={() => onRemove(index)}
|
|
className={cn(
|
|
"text-muted-foreground h-8 w-8 p-0 transition-colors hover:text-red-500",
|
|
!canRemove && "cursor-not-allowed opacity-50",
|
|
)}
|
|
disabled={!canRemove}
|
|
aria-label="Remove item"
|
|
>
|
|
<Trash2 className="h-4 w-4" />
|
|
</Button>
|
|
</div>
|
|
<div className="flex-1 px-3 text-center">
|
|
<span className="text-muted-foreground block text-sm font-medium">
|
|
<span className="hidden sm:inline">Item </span>
|
|
<span className="sm:hidden">#</span>
|
|
{index + 1}
|
|
</span>
|
|
</div>
|
|
<div className="flex flex-col items-end">
|
|
<span className="text-muted-foreground text-xs">Total</span>
|
|
<span className="text-primary text-lg font-bold">
|
|
${(item.hours * item.rate).toFixed(2)}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export function InvoiceLineItems({
|
|
items,
|
|
onAddItem,
|
|
onRemoveItem,
|
|
onUpdateItem,
|
|
onMoveUp,
|
|
onMoveDown,
|
|
className,
|
|
}: InvoiceLineItemsProps) {
|
|
const canRemoveItems = items.length > 1;
|
|
|
|
return (
|
|
<div className={cn("space-y-2", className)}>
|
|
{/* Desktop and Tablet Table */}
|
|
<div className="hidden md:block">
|
|
<div className="overflow-hidden rounded-lg border">
|
|
<table className="w-full">
|
|
{/* Desktop Header */}
|
|
<thead className="bg-muted/30 hidden lg:table-header-group">
|
|
<tr>
|
|
<th className="w-6 p-2"></th>
|
|
<th
|
|
className="text-muted-foreground p-2 text-left text-xs font-medium"
|
|
colSpan={5}
|
|
>
|
|
Invoice Items
|
|
</th>
|
|
</tr>
|
|
</thead>
|
|
|
|
{/* Tablet Header */}
|
|
<thead className="bg-muted/30 md:table-header-group lg:hidden">
|
|
<tr>
|
|
<th className="w-6 p-2"></th>
|
|
<th
|
|
className="text-muted-foreground p-2 text-left text-xs font-medium"
|
|
colSpan={6}
|
|
>
|
|
Invoice Items
|
|
</th>
|
|
</tr>
|
|
</thead>
|
|
|
|
<tbody className="divide-y">
|
|
{items.map((item, index) => (
|
|
<LineItemRow
|
|
key={item.id}
|
|
item={item}
|
|
index={index}
|
|
canRemove={canRemoveItems}
|
|
onRemove={onRemoveItem}
|
|
onUpdate={onUpdateItem}
|
|
onMoveUp={onMoveUp}
|
|
onMoveDown={onMoveDown}
|
|
isFirst={index === 0}
|
|
isLast={index === items.length - 1}
|
|
/>
|
|
))}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
|
|
{/* Mobile Cards */}
|
|
<div className="space-y-2 md:hidden">
|
|
{items.map((item, index) => (
|
|
<MobileLineItem
|
|
key={item.id}
|
|
item={item}
|
|
index={index}
|
|
canRemove={canRemoveItems}
|
|
onRemove={onRemoveItem}
|
|
onUpdate={onUpdateItem}
|
|
onMoveUp={onMoveUp}
|
|
onMoveDown={onMoveDown}
|
|
isFirst={index === 0}
|
|
isLast={index === items.length - 1}
|
|
/>
|
|
))}
|
|
</div>
|
|
|
|
{/* Add Item Button */}
|
|
<div className="px-3 pt-3">
|
|
<Button
|
|
type="button"
|
|
variant="outline"
|
|
onClick={onAddItem}
|
|
className="w-full"
|
|
>
|
|
<Plus className="mr-2 h-4 w-4" />
|
|
Add Another Item
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|