Schema: - New Company model (name, gstNumber, address, telephone, mobile, email, invoiceAddress, isActive) - PurchaseOrder.companyId FK (optional, SET NULL on company delete) - Migration: 20260530000003_add_company Admin: - /admin/companies page with full CRUD (create, edit, deactivate, delete) - Companies table shows name, GST, contact details, status - Companies link added to Admin section of sidebar (Briefcase icon) PO forms (new / edit / import / manager-edit): - Company dropdown appears at the top of Order Information when companies exist - Pre-populated with first active company; selection persisted to DB via companyId Import form: - parseSheet() now extracts companyName from Excel row 1 (col A) - Import preview auto-matches detected company name against known companies - Shows detected name as a hint; user can override before saving Export (PDF + XLSX): - Company constants (CO_NAME, CO_ADDR, CO_TEL, INV_ADDR, INV_GST) are now derived from the linked Company record when present, falling back to the original Pelagia Marine hardcoded defaults when no company is set Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
111 lines
4.9 KiB
TypeScript
111 lines
4.9 KiB
TypeScript
"use client";
|
|
|
|
import { useState } from "react";
|
|
import { AddCompanyButton, EditCompanyButton } from "./company-form";
|
|
import { RowActionsMenu, RowActionsItem, RowActionsDestructiveItem, RowActionsSeparator } from "@/components/ui/row-actions-menu";
|
|
import { DeleteConfirmDialog } from "@/components/ui/delete-confirm-dialog";
|
|
import { ConfirmDialog } from "@/components/ui/confirm-dialog";
|
|
import { deleteCompany, toggleCompanyActive } from "./actions";
|
|
|
|
export type CompanyRow = {
|
|
id: string;
|
|
name: string;
|
|
gstNumber: string | null;
|
|
address: string | null;
|
|
telephone: string | null;
|
|
mobile: string | null;
|
|
email: string | null;
|
|
invoiceAddress: string | null;
|
|
isActive: boolean;
|
|
};
|
|
|
|
function CompanyActionsMenu({ company }: { company: CompanyRow }) {
|
|
const [editOpen, setEditOpen] = useState(false);
|
|
const [deleteOpen, setDeleteOpen] = useState(false);
|
|
const [toggleOpen, setToggleOpen] = useState(false);
|
|
|
|
return (
|
|
<>
|
|
<RowActionsMenu>
|
|
<RowActionsItem onClick={() => setEditOpen(true)}>Edit</RowActionsItem>
|
|
<RowActionsItem onClick={() => setToggleOpen(true)}>
|
|
{company.isActive ? "Deactivate" : "Activate"}
|
|
</RowActionsItem>
|
|
<RowActionsSeparator />
|
|
<RowActionsDestructiveItem onClick={() => setDeleteOpen(true)}>Delete</RowActionsDestructiveItem>
|
|
</RowActionsMenu>
|
|
<EditCompanyButton company={company} open={editOpen} onOpenChange={setEditOpen} />
|
|
<DeleteConfirmDialog
|
|
open={deleteOpen} onOpenChange={setDeleteOpen}
|
|
label={company.name} onConfirm={() => deleteCompany(company.id)}
|
|
/>
|
|
<ConfirmDialog
|
|
open={toggleOpen} onOpenChange={setToggleOpen}
|
|
title={company.isActive ? `Deactivate ${company.name}?` : `Activate ${company.name}?`}
|
|
description={company.isActive ? `${company.name} will not appear in new PO selections.` : `${company.name} will become available for new POs.`}
|
|
confirmLabel={company.isActive ? "Deactivate" : "Activate"}
|
|
onConfirm={() => toggleCompanyActive(company.id)}
|
|
/>
|
|
</>
|
|
);
|
|
}
|
|
|
|
export function CompaniesTable({ companies }: { companies: CompanyRow[] }) {
|
|
return (
|
|
<div>
|
|
<div className="mb-6 flex items-center justify-between">
|
|
<div>
|
|
<h1 className="text-2xl font-semibold text-neutral-900">Company Management</h1>
|
|
<p className="text-sm text-neutral-500 mt-0.5">Sister companies used for invoicing and purchase orders</p>
|
|
</div>
|
|
<AddCompanyButton />
|
|
</div>
|
|
|
|
<div className="rounded-lg border border-neutral-200 bg-white overflow-hidden">
|
|
<table className="w-full text-sm">
|
|
<thead className="bg-neutral-50 border-b border-neutral-200">
|
|
<tr>
|
|
<th className="px-4 py-3 text-left font-medium text-neutral-600">Company Name</th>
|
|
<th className="px-4 py-3 text-left font-medium text-neutral-600">GST Number</th>
|
|
<th className="px-4 py-3 text-left font-medium text-neutral-600">Contact</th>
|
|
<th className="px-4 py-3 text-left font-medium text-neutral-600">Status</th>
|
|
<th className="px-4 py-3 w-10"></th>
|
|
</tr>
|
|
</thead>
|
|
<tbody className="divide-y divide-neutral-100">
|
|
{companies.length === 0 && (
|
|
<tr>
|
|
<td colSpan={5} className="px-4 py-10 text-center text-neutral-400">
|
|
No companies yet. Add one to start selecting it on purchase orders.
|
|
</td>
|
|
</tr>
|
|
)}
|
|
{companies.map((c) => (
|
|
<tr key={c.id} className="hover:bg-neutral-50">
|
|
<td className="px-4 py-3">
|
|
<p className="font-medium text-neutral-900">{c.name}</p>
|
|
{c.address && <p className="text-xs text-neutral-400 mt-0.5 truncate max-w-xs">{c.address}</p>}
|
|
</td>
|
|
<td className="px-4 py-3 font-mono text-xs text-neutral-600">{c.gstNumber ?? <span className="italic text-neutral-400">—</span>}</td>
|
|
<td className="px-4 py-3 text-xs text-neutral-500 space-y-0.5">
|
|
{c.telephone && <p>☎ {c.telephone}</p>}
|
|
{c.mobile && <p>📱 {c.mobile}</p>}
|
|
{c.email && <p>✉ {c.email}</p>}
|
|
{!c.telephone && !c.mobile && !c.email && <span className="italic text-neutral-400">—</span>}
|
|
</td>
|
|
<td className="px-4 py-3">
|
|
<span className={`rounded-full px-2.5 py-0.5 text-xs font-medium ${c.isActive ? "bg-success-100 text-success-700" : "bg-neutral-100 text-neutral-500"}`}>
|
|
{c.isActive ? "Active" : "Inactive"}
|
|
</span>
|
|
</td>
|
|
<td className="px-4 py-3">
|
|
<CompanyActionsMenu company={c} />
|
|
</td>
|
|
</tr>
|
|
))}
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|