Refine purchase receipt and lot label flow

This commit is contained in:
2026-05-29 05:21:28 +07:00
parent 8f08d24a4e
commit d0bdd4bb63
7 changed files with 413 additions and 93 deletions

View File

@ -1,5 +1,6 @@
"use client";
import { Printer, Tags } from "lucide-react";
import { useEffect, useMemo, useState } from "react";
import { useLocale } from "@/components/providers/locale-provider";
@ -131,6 +132,15 @@ export function ReceiptsClient() {
setError(null);
}
async function fetchReceiptDetail(id: string) {
const response = await fetch(`/api/v1/receipts/${id}`, { cache: "no-store" });
const payload = (await response.json()) as DetailResponse<ReceiptDetail> | ApiErrorResponse;
if (!response.ok || !("data" in payload)) {
throw new Error("message" in payload ? payload.message : dict.receipts.detailError);
}
return payload.data;
}
function updateLine(index: number, patch: Partial<ReceiptLineForm>) {
setForm((current) => ({
...current,
@ -193,12 +203,25 @@ export function ReceiptsClient() {
async function openReceipt(id: string) {
setError(null);
try {
const response = await fetch(`/api/v1/receipts/${id}`, { cache: "no-store" });
const payload = (await response.json()) as DetailResponse<ReceiptDetail> | ApiErrorResponse;
if (!response.ok || !("data" in payload)) {
throw new Error(dict.receipts.detailError);
}
setSelectedReceipt(payload.data);
setSelectedReceipt(await fetchReceiptDetail(id));
} catch (err) {
setError(err instanceof Error ? err.message : dict.receipts.detailError);
}
}
async function printReceiptById(id: string) {
setError(null);
try {
printReceipt(await fetchReceiptDetail(id));
} catch (err) {
setError(err instanceof Error ? err.message : dict.receipts.detailError);
}
}
async function printLotLabelsById(id: string) {
setError(null);
try {
printLotLabels(await fetchReceiptDetail(id));
} catch (err) {
setError(err instanceof Error ? err.message : dict.receipts.detailError);
}
@ -221,9 +244,59 @@ export function ReceiptsClient() {
}
}
const submittedPurchases = useMemo(
() => purchases.filter((purchase) => purchase.status === "SUBMITTED"),
[purchases]
function printReceipt(detail: ReceiptDetail) {
setError(null);
const printWindow = window.open("", "_blank", "width=1120,height=760");
if (!printWindow) {
window.alert(locale === "id" ? "Popup cetak diblokir browser. Izinkan pop-up untuk situs ini lalu coba lagi." : "Print popup was blocked by the browser. Allow pop-ups for this site and try again.");
return;
}
try {
printWindow.document.open();
printWindow.document.write(buildReceiptPrintHtml(detail, locale));
printWindow.document.close();
} catch (err) {
if (!printWindow.closed) {
printWindow.close();
}
setError(err instanceof Error ? err.message : locale === "id" ? "Gagal menyiapkan cetak penerimaan." : "Failed to prepare receipt print.");
}
}
function printLotLabels(detail: ReceiptDetail) {
setError(null);
if (detail.generated_lots.length === 0) {
setError(locale === "id" ? "Belum ada lot yang bisa dicetak. Buat lot terlebih dahulu." : "No lot labels to print yet. Generate lots first.");
return;
}
const printWindow = window.open("", "_blank", "width=900,height=760");
if (!printWindow) {
window.alert(locale === "id" ? "Popup cetak diblokir browser. Izinkan pop-up untuk situs ini lalu coba lagi." : "Print popup was blocked by the browser. Allow pop-ups for this site and try again.");
return;
}
try {
printWindow.document.open();
printWindow.document.write(buildLotLabelsPrintHtml(detail, locale));
printWindow.document.close();
} catch (err) {
if (!printWindow.closed) {
printWindow.close();
}
setError(err instanceof Error ? err.message : locale === "id" ? "Gagal menyiapkan label lot." : "Failed to prepare lot labels.");
}
}
const receiptedPurchaseIds = useMemo(
() => new Set(items.map((item) => item.purchase.id)),
[items]
);
const receivablePurchases = useMemo(
() => purchases.filter((purchase) => purchase.status !== "CANCELLED" && !receiptedPurchaseIds.has(purchase.id)),
[purchases, receiptedPurchaseIds]
);
return (
@ -243,7 +316,7 @@ export function ReceiptsClient() {
<span className="ops-label">{dict.receipts.purchase}</span>
<select value={form.purchase_id} onChange={(event) => void handlePurchaseChange(event.target.value)} className="ops-select">
<option value="">{dict.receipts.choosePurchase}</option>
{submittedPurchases.map((purchase) => (
{receivablePurchases.map((purchase) => (
<option key={purchase.id} value={purchase.id}>
{purchase.purchase_no} · {purchase.agent?.name || dict.receipts.directPurchase}
</option>
@ -337,6 +410,24 @@ export function ReceiptsClient() {
<div className="flex gap-2">
<button type="button" onClick={() => void openReceipt(item.id)} className="ops-btn-secondary">{dict.common.detail}</button>
{item.status !== "FINALIZED" ? <button type="button" onClick={() => void generateLots(item.id)} className="ops-btn-primary">{dict.receipts.generateLots}</button> : null}
<button
type="button"
onClick={() => void printReceiptById(item.id)}
className="ops-btn-secondary"
>
<Printer className="h-4 w-4" />
{locale === "id" ? "Receipt" : "Receipt"}
</button>
{item.lot_count > 0 ? (
<button
type="button"
onClick={() => void printLotLabelsById(item.id)}
className="ops-btn-secondary"
>
<Tags className="h-4 w-4" />
{locale === "id" ? "Label lot" : "Lot labels"}
</button>
) : null}
</div>
</td>
</tr>
@ -355,7 +446,19 @@ export function ReceiptsClient() {
<h3 className="mt-2 text-2xl font-semibold text-ink">{selectedReceipt.receipt_no}</h3>
<p className="mt-2 text-sm text-slate-500">{selectedReceipt.purchase.purchase_no}</p>
</div>
{selectedReceipt.status !== "FINALIZED" ? <button type="button" onClick={() => void generateLots(selectedReceipt.id)} className="ops-btn-primary">{dict.receipts.generateLots}</button> : null}
<div className="flex flex-wrap items-center gap-2">
<button type="button" onClick={() => printReceipt(selectedReceipt)} className="ops-btn-secondary">
<Printer className="h-4 w-4" />
{locale === "id" ? "Cetak receipt" : "Print receipt"}
</button>
{selectedReceipt.generated_lots.length > 0 ? (
<button type="button" onClick={() => printLotLabels(selectedReceipt)} className="ops-btn-secondary">
<Tags className="h-4 w-4" />
{locale === "id" ? "Cetak label lot" : "Print lot labels"}
</button>
) : null}
{selectedReceipt.status !== "FINALIZED" ? <button type="button" onClick={() => void generateLots(selectedReceipt.id)} className="ops-btn-primary">{dict.receipts.generateLots}</button> : null}
</div>
</div>
<div className="mt-5 space-y-4">
{selectedReceipt.lines.map((line) => (
@ -372,7 +475,7 @@ export function ReceiptsClient() {
<div className="mt-3 space-y-2">
{selectedReceipt.generated_lots.map((lot) => (
<div key={lot.id} className="rounded border border-line/70 bg-white px-4 py-3 text-sm text-slate-600">
<span className="font-medium text-ink">{lot.lot_code}</span> · {lot.status}
<span className="font-medium text-ink">{lot.lot_code}</span> · {lot.grade?.name ?? "-"} · {formatQuantity(lot.original_qty, locale, lot.unit.code)} · {lot.status}
</div>
))}
</div>
@ -385,6 +488,178 @@ export function ReceiptsClient() {
);
}
function escapeHtml(value: string) {
return value
.replaceAll("&", "&amp;")
.replaceAll("<", "&lt;")
.replaceAll(">", "&gt;")
.replaceAll('"', "&quot;")
.replaceAll("'", "&#39;");
}
function buildReceiptPrintHtml(detail: ReceiptDetail, locale: string) {
const isIndonesian = locale === "id";
const safeReceiptNo = escapeHtml(detail.receipt_no);
const safePurchaseNo = escapeHtml(detail.purchase.purchase_no);
const safeDate = escapeHtml(new Date(detail.receipt_date).toLocaleDateString(isIndonesian ? "id-ID" : "en-US"));
const safeNotes = escapeHtml(detail.notes ?? "-");
const totalAccepted = detail.lines.reduce((sum, line) => sum + line.qty_accepted, 0);
const rows = detail.lines.map((line, index) => {
const safeGrade = escapeHtml(line.grade?.name ?? "-");
const safeWarehouse = escapeHtml(line.warehouse.name);
const safeLocation = escapeHtml(line.location?.name ?? "-");
const safeLineNotes = escapeHtml(line.notes ?? "-");
return `<tr>
<td>${index + 1}</td>
<td>${safeGrade}</td>
<td>${escapeHtml(formatQuantity(line.qty_received, locale, line.unit.code))}</td>
<td>${escapeHtml(formatQuantity(line.qty_accepted, locale, line.unit.code))}</td>
<td>${escapeHtml(formatQuantity(line.qty_rejected, locale, line.unit.code))}</td>
<td>${safeWarehouse}</td>
<td>${safeLocation}</td>
<td>${safeLineNotes}</td>
</tr>`;
}).join("");
const lotRows = detail.generated_lots.map((lot, index) => {
const safeGrade = escapeHtml(lot.grade?.name ?? "-");
return `<tr>
<td>${index + 1}</td>
<td>${escapeHtml(lot.lot_code)}</td>
<td>${safeGrade}</td>
<td>${escapeHtml(formatQuantity(lot.original_qty, locale, lot.unit.code))}</td>
<td>${escapeHtml(lot.status)}</td>
</tr>`;
}).join("");
return `<!doctype html>
<html lang="${isIndonesian ? "id" : "en"}">
<head>
<meta charset="utf-8" />
<title>${isIndonesian ? "Receipt Penerimaan" : "Receipt"} ${safeReceiptNo}</title>
<style>
body { margin: 0; background: #eef3f5; color: #182126; font-family: Arial, Helvetica, sans-serif; }
.sheet { max-width: 1120px; margin: 0 auto; padding: 24px; }
.paper { background: #fff; border: 1px solid #d9e2e7; padding: 28px; }
.top { display: flex; justify-content: space-between; gap: 24px; border-bottom: 2px solid #182126; padding-bottom: 16px; }
h1 { margin: 0; font-size: 28px; letter-spacing: 0; }
.muted { color: #64748b; font-size: 12px; line-height: 1.5; }
.meta { display: grid; grid-template-columns: repeat(4, 1fr); gap: 12px; margin: 20px 0; }
.box { border: 1px solid #d9e2e7; padding: 10px 12px; }
.label { color: #64748b; font-size: 10px; font-weight: 700; text-transform: uppercase; }
.value { margin-top: 4px; font-size: 14px; font-weight: 700; }
table { width: 100%; border-collapse: collapse; margin-top: 14px; font-size: 12px; }
th, td { border: 1px solid #d9e2e7; padding: 8px; text-align: left; vertical-align: top; }
th { background: #f1f5f9; font-size: 10px; text-transform: uppercase; color: #475569; }
h2 { margin: 24px 0 0; font-size: 16px; }
@media print {
body { background: #fff; }
.sheet { padding: 0; }
.paper { border: 0; }
}
</style>
</head>
<body onload="window.print()">
<div class="sheet">
<div class="paper">
<div class="top">
<div>
<h1>${isIndonesian ? "Receipt Penerimaan" : "Goods Receipt"}</h1>
<div class="muted">${safeReceiptNo}</div>
</div>
<div class="muted" style="text-align:right">
${isIndonesian ? "Pembelian" : "Purchase"}<br />
<strong>${safePurchaseNo}</strong>
</div>
</div>
<div class="meta">
<div class="box"><div class="label">${isIndonesian ? "Tanggal" : "Date"}</div><div class="value">${safeDate}</div></div>
<div class="box"><div class="label">${isIndonesian ? "Status" : "Status"}</div><div class="value">${escapeHtml(detail.status)}</div></div>
<div class="box"><div class="label">${isIndonesian ? "Baris" : "Lines"}</div><div class="value">${detail.lines.length}</div></div>
<div class="box"><div class="label">${isIndonesian ? "Total valid" : "Accepted total"}</div><div class="value">${escapeHtml(formatQuantity(totalAccepted, locale))}</div></div>
</div>
<div class="muted"><strong>${isIndonesian ? "Catatan" : "Notes"}:</strong> ${safeNotes}</div>
<h2>${isIndonesian ? "Baris Penerimaan" : "Receipt Lines"}</h2>
<table>
<thead><tr><th>No</th><th>Grade</th><th>${isIndonesian ? "Diterima" : "Received"}</th><th>${isIndonesian ? "Valid" : "Accepted"}</th><th>${isIndonesian ? "Ditolak" : "Rejected"}</th><th>${isIndonesian ? "Gudang" : "Warehouse"}</th><th>${isIndonesian ? "Lokasi" : "Location"}</th><th>${isIndonesian ? "Catatan" : "Notes"}</th></tr></thead>
<tbody>${rows}</tbody>
</table>
<h2>${isIndonesian ? "Lot Hasil" : "Generated Lots"}</h2>
<table>
<thead><tr><th>No</th><th>Lot</th><th>Grade</th><th>Qty</th><th>Status</th></tr></thead>
<tbody>${lotRows || `<tr><td colspan="5">${isIndonesian ? "Belum ada lot." : "No lots yet."}</td></tr>`}</tbody>
</table>
</div>
</div>
</body>
</html>`;
}
function buildLotLabelsPrintHtml(detail: ReceiptDetail, locale: string) {
const isIndonesian = locale === "id";
const labels = detail.generated_lots.map((lot) => {
const safeLotCode = escapeHtml(lot.lot_code);
const safeGrade = escapeHtml(lot.grade?.name ?? "-");
const safeWarehouse = escapeHtml(lot.warehouse.name);
const safeLocation = escapeHtml(lot.location?.name ?? "-");
const codeValue = escapeHtml(lot.qr_code_value || lot.barcode_value || lot.lot_code);
return `<section class="label">
<div class="label-head">
<div>
<div class="eyebrow">ABELBIRDNEST LOT</div>
<h2>${safeLotCode}</h2>
</div>
<div class="status">${escapeHtml(lot.status)}</div>
</div>
<div class="barcode">${codeValue}</div>
<div class="grid">
<div><span>Grade</span><strong>${safeGrade}</strong></div>
<div><span>Qty</span><strong>${escapeHtml(formatQuantity(lot.original_qty, locale, lot.unit.code))}</strong></div>
<div><span>${isIndonesian ? "Gudang" : "Warehouse"}</span><strong>${safeWarehouse}</strong></div>
<div><span>${isIndonesian ? "Lokasi" : "Location"}</span><strong>${safeLocation}</strong></div>
</div>
<div class="foot">
<span>${isIndonesian ? "Receipt" : "Receipt"} ${escapeHtml(detail.receipt_no)}</span>
<span>${escapeHtml(detail.receipt_date)}</span>
</div>
</section>`;
}).join("");
return `<!doctype html>
<html lang="${isIndonesian ? "id" : "en"}">
<head>
<meta charset="utf-8" />
<title>${isIndonesian ? "Label Lot" : "Lot Labels"} ${escapeHtml(detail.receipt_no)}</title>
<style>
body { margin: 0; background: #f1f5f9; color: #111827; font-family: Arial, Helvetica, sans-serif; }
.sheet { display: grid; grid-template-columns: repeat(2, minmax(0, 1fr)); gap: 12px; padding: 16px; }
.label { break-inside: avoid; background: #fff; border: 1.5px solid #111827; border-radius: 6px; padding: 14px; min-height: 190px; display: flex; flex-direction: column; gap: 12px; }
.label-head { display: flex; align-items: flex-start; justify-content: space-between; gap: 12px; }
.eyebrow { font-size: 9px; font-weight: 800; color: #64748b; letter-spacing: 0; }
h2 { margin: 3px 0 0; font-size: 24px; letter-spacing: 0; }
.status { border: 1px solid #111827; padding: 4px 7px; font-size: 10px; font-weight: 800; }
.barcode { border: 1px solid #cbd5e1; padding: 9px; text-align: center; font-family: "Courier New", monospace; font-size: 18px; font-weight: 800; letter-spacing: 2px; overflow-wrap: anywhere; }
.grid { display: grid; grid-template-columns: repeat(2, 1fr); gap: 8px; }
.grid div { border-top: 1px solid #e2e8f0; padding-top: 6px; }
.grid span { display: block; font-size: 9px; color: #64748b; font-weight: 800; text-transform: uppercase; }
.grid strong { display: block; margin-top: 2px; font-size: 12px; }
.foot { margin-top: auto; display: flex; justify-content: space-between; gap: 10px; color: #64748b; font-size: 10px; }
@media print {
body { background: #fff; }
.sheet { padding: 0; gap: 8mm; }
.label { border-radius: 0; min-height: 48mm; }
}
</style>
</head>
<body onload="window.print()">
<main class="sheet">${labels}</main>
</body>
</html>`;
}
function Field({ label, value, onChange, type = "text", disabled = false }: { label: string; value: string; onChange?: (value: string) => void; type?: string; disabled?: boolean }) {
return (
<label className="block">