diff options
Diffstat (limited to 'lib')
| -rw-r--r-- | lib/general-contracts/detail/general-contract-approval-request-dialog.tsx | 5 | ||||
| -rw-r--r-- | lib/general-contracts/utils.ts | 63 | ||||
| -rw-r--r-- | lib/pq/service.ts | 2 | ||||
| -rw-r--r-- | lib/rfq-last/approval-actions.ts | 26 | ||||
| -rw-r--r-- | lib/rfq-last/service.ts | 2 | ||||
| -rw-r--r-- | lib/rfq-last/vendor-response/editor/vendor-response-editor.tsx | 44 | ||||
| -rw-r--r-- | lib/techsales-rfq/approval-actions.ts | 71 |
7 files changed, 203 insertions, 10 deletions
diff --git a/lib/general-contracts/detail/general-contract-approval-request-dialog.tsx b/lib/general-contracts/detail/general-contract-approval-request-dialog.tsx index db0901cb..04054369 100644 --- a/lib/general-contracts/detail/general-contract-approval-request-dialog.tsx +++ b/lib/general-contracts/detail/general-contract-approval-request-dialog.tsx @@ -245,7 +245,10 @@ export function ContractApprovalRequestDialog({ if (contractData) { summary.basicInfo = { ...summary.basicInfo, - externalYardEntry: contractData.externalYardEntry || 'N' + externalYardEntry: contractData.externalYardEntry || 'N', + vendorCountry: (contractData as any)?.vendorCountry || summary.basicInfo.vendorCountry, + vendorName: (contractData as any)?.vendorName || summary.basicInfo.vendorName, + vendorCode: (contractData as any)?.vendorCode || summary.basicInfo.vendorCode, } } } catch { diff --git a/lib/general-contracts/utils.ts b/lib/general-contracts/utils.ts index 5bbb5980..1262dc4d 100644 --- a/lib/general-contracts/utils.ts +++ b/lib/general-contracts/utils.ts @@ -170,6 +170,51 @@ export function mapContractDataToTemplateVariables(contractSummary: ContractSumm ).join('\n') : '' + // PDFTron 템플릿 루프용 데이터 ({{#storageList}} ... {{/storageList}} 사용) + const storageList = storageItems.map(item => ({ + project: (item.projectName || item.projectCode || '').toString().trim(), + poNumber: (item.poNumber || '').toString().trim(), + hullNumber: (item.hullNumber || '').toString().trim(), + remainingAmount: formatCurrency(item.remainingAmount), + })) + + // 일반 견적 품목 루프용 데이터 ({{#itemsList}} ... {{/itemsList}} 사용) + const itemsList = (items || []).map((item, idx) => { + const quantityRaw = item.quantity ?? item.qty ?? '' + const unitPriceRaw = item.contractUnitPrice ?? item.unitPrice ?? '' + const amountRaw = + item.contractAmount ?? + (Number(quantityRaw) * Number(unitPriceRaw)) + + const quantityNum = Number(quantityRaw) + const hasQuantity = !isNaN(quantityNum) + const unitPriceNum = Number(unitPriceRaw) + const hasUnitPrice = !isNaN(unitPriceNum) + const hasAmountCalc = !isNaN(amountRaw as number) + + const amount = hasAmountCalc ? formatCurrency(amountRaw) : '' + + return { + no: idx + 1, // NO + hullNumber: (item.projectCode || basicInfo.projectCode || '').toString().trim(), // 호선번호 + shipType: (item.projectName || basicInfo.projectName || '').toString().trim(), // 선종/선형 + exportCountry: (basicInfo.vendorCountry || basicInfo.country || '').toString().trim(), // 수출국 + itemName: (item.itemInfo || item.description || item.itemCode || '').toString().trim(), // 품목 + unit: (item.quantityUnit || '').toString().trim(), // 단위 + unitPrice: hasUnitPrice ? formatCurrency(unitPriceNum) : formatCurrency(unitPriceRaw), // 단가 + amount, // 금액 + remark: (item.remark || item.remarks || item.note || '').toString().trim(), // 비고 + // 보존용 기존 필드 + itemCode: (item.itemCode || item.itemInfo || '').toString().trim(), + quantity: hasQuantity ? quantityNum : (quantityRaw ?? ''), + } + }) + + // 루프 미지원 템플릿을 위한 품목 텍스트 fallback + const itemsTableText = itemsList.length > 0 + ? itemsList.map(i => `${i.no}. ${i.hullNumber || '-'} / ${i.shipType || '-'} / ${i.exportCountry || '-'} / ${i.itemName || '-'} / 단위:${i.unit || '-'} / 단가:${i.unitPrice || '-'} / 금액:${i.amount || '-'} / 비고:${i.remark || '-'}`).join('\n') + : '' + // ═══════════════════════════════════════════════════════════════ // 변수 매핑 시작 @@ -296,7 +341,13 @@ export function mapContractDataToTemplateVariables(contractSummary: ContractSumm // ---------------------------------- storageTableText: storageTableText, // {{storageTableText}} (fallback) // PDFTron에서 배열을 받아 테이블 루프를 돌릴 수 있다면 아래 키를 사용 - storageList: storageItems, + storageList, + + // ---------------------------------- + // 일반 견적 품목 루프 (템플릿 표에 {{#itemsList}} 사용) + // ---------------------------------- + itemsList, + itemsTableText, } // 3. 모든 키를 순회하며 undefined나 null을 빈 문자열로 변환 (안전장치) @@ -306,5 +357,13 @@ export function mapContractDataToTemplateVariables(contractSummary: ContractSumm } }) - return variables + // 4. PDF 템플릿에서 추출한 {{ }} 변수명이 공백을 포함할 수 있어 trim 처리 후 매핑 + const normalizedVariables: Record<string, any> = {} + Object.entries(variables).forEach(([key, value]) => { + const trimmedKey = key.trim() + const trimmedValue = typeof value === 'string' ? value.trim() : value + normalizedVariables[trimmedKey] = trimmedValue + }) + + return normalizedVariables } diff --git a/lib/pq/service.ts b/lib/pq/service.ts index 15e71c4d..ea8a389c 100644 --- a/lib/pq/service.ts +++ b/lib/pq/service.ts @@ -2641,7 +2641,7 @@ export async function approvePQAction({ await db .update(vendorPQSubmissions) .set({ - status: "APPROVED", + status: "QM_APPROVED", approvedAt: currentDate, updatedAt: currentDate, }) diff --git a/lib/rfq-last/approval-actions.ts b/lib/rfq-last/approval-actions.ts index be435931..2f9d0843 100644 --- a/lib/rfq-last/approval-actions.ts +++ b/lib/rfq-last/approval-actions.ts @@ -8,6 +8,7 @@ import { ApprovalSubmissionSaga } from '@/lib/approval'; import { mapRfqSendToTemplateVariables } from './approval-handlers'; +import { prepareEmailAttachments } from './service'; interface RfqSendApprovalData { // RFQ 기본 정보 @@ -95,7 +96,27 @@ export async function requestRfqSendWithApproval(data: RfqSendApprovalData) { applicationReason: data.applicationReason, }); - // 3. 결재 상신용 payload 구성 + // 3. Knox 상신용 첨부파일 준비 (실제 파일 객체로 변환) + const emailAttachments = await prepareEmailAttachments(data.rfqId, data.attachmentIds); + type PreparedAttachment = { + filename?: string | null; + content: BlobPart; + contentType?: string | null; + }; + const knoxAttachments = (emailAttachments as PreparedAttachment[]) + .filter((att): att is PreparedAttachment => Boolean(att && att.content)) + .map( + (att) => + new File([att.content], att.filename || 'attachment', { + type: att.contentType || 'application/octet-stream', + }) + ); + + if (knoxAttachments.length === 0) { + throw new Error('상신할 첨부파일을 준비하지 못했습니다.'); + } + + // 4. 결재 상신용 payload 구성 // ⚠️ cronjob 환경에서 실행되므로 currentUser 정보를 포함해야 함 const approvalPayload = { rfqId: data.rfqId, @@ -113,7 +134,7 @@ export async function requestRfqSendWithApproval(data: RfqSendApprovalData) { }, }; - // 4. Saga로 결재 상신 + // 5. Saga로 결재 상신 const saga = new ApprovalSubmissionSaga( 'rfq_send_with_attachments', // 핸들러 키 approvalPayload, // 결재 승인 후 실행될 데이터 @@ -128,6 +149,7 @@ export async function requestRfqSendWithApproval(data: RfqSendApprovalData) { epId: data.currentUser.epId, email: data.currentUser.email, }, + attachments: knoxAttachments, } ); diff --git a/lib/rfq-last/service.ts b/lib/rfq-last/service.ts index 68cfdac7..23f5f63a 100644 --- a/lib/rfq-last/service.ts +++ b/lib/rfq-last/service.ts @@ -3349,7 +3349,7 @@ async function getProjectInfo(projectId: number) { return project; } -async function prepareEmailAttachments(rfqId: number, attachmentIds: number[]) { +export async function prepareEmailAttachments(rfqId: number, attachmentIds: number[]) { const attachments = await db .select({ attachment: rfqLastAttachments, diff --git a/lib/rfq-last/vendor-response/editor/vendor-response-editor.tsx b/lib/rfq-last/vendor-response/editor/vendor-response-editor.tsx index 8c70b8dd..18fc5d50 100644 --- a/lib/rfq-last/vendor-response/editor/vendor-response-editor.tsx +++ b/lib/rfq-last/vendor-response/editor/vendor-response-editor.tsx @@ -9,6 +9,14 @@ import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/com import { Button } from "@/components/ui/button" import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs" import { Badge } from "@/components/ui/badge" +import { + Dialog, + DialogContent, + DialogDescription, + DialogFooter, + DialogHeader, + DialogTitle, +} from "@/components/ui/dialog" import { toast } from "sonner" import RfqInfoHeader from "./rfq-info-header" import CommercialTermsForm from "./commercial-terms-form" @@ -130,6 +138,7 @@ export default function VendorResponseEditor({ const [deletedAttachments, setDeletedAttachments] = useState<any[]>([]) const [uploadProgress, setUploadProgress] = useState(0) // 추가 const [currencyDecimalPlaces, setCurrencyDecimalPlaces] = useState<number>(2) // 통화별 소수점 자리수 + const [confirmOpen, setConfirmOpen] = useState(false) console.log(existingResponse,"existingResponse") @@ -682,7 +691,7 @@ export default function VendorResponseEditor({ <Button type="button" variant="default" - onClick={() => handleFormSubmit(true)} // 직접 핸들러 호출 + onClick={() => setConfirmOpen(true)} // 제출 전 확인 다이얼로그 disabled={loading || !allContractsSigned || isSubmitted || activeTab !== 'attachments'} > {!allContractsSigned ? ( @@ -714,6 +723,39 @@ export default function VendorResponseEditor({ </Button> </div> + {/* 최종 제출 확인 다이얼로그 */} + <Dialog open={confirmOpen} onOpenChange={setConfirmOpen}> + <DialogContent> + <DialogHeader> + <DialogTitle>최종 제출</DialogTitle> + <DialogDescription> + 최종 제출하시겠습니까? + </DialogDescription> + </DialogHeader> + <DialogFooter className="flex gap-2 sm:justify-end"> + <Button + type="button" + variant="outline" + onClick={() => setConfirmOpen(false)} + disabled={loading} + > + 취소 + </Button> + <Button + type="button" + variant="default" + onClick={() => { + setConfirmOpen(false) + handleFormSubmit(true) + }} + disabled={loading || !allContractsSigned || isSubmitted || activeTab !== 'attachments'} + > + 제출하기 + </Button> + </DialogFooter> + </DialogContent> + </Dialog> + </div> </form> </FormProvider> diff --git a/lib/techsales-rfq/approval-actions.ts b/lib/techsales-rfq/approval-actions.ts index cf914592..c85a950f 100644 --- a/lib/techsales-rfq/approval-actions.ts +++ b/lib/techsales-rfq/approval-actions.ts @@ -97,7 +97,13 @@ export async function requestTechSalesRfqSendWithApproval(data: TechSalesRfqSend applicationReason: data.applicationReason, }); - // 5. 결재 상신용 payload 구성 + // 5. Knox 상신용 첨부파일 준비 + const knoxAttachments = await prepareKnoxDrmAttachments(data.drmAttachmentIds); + if (knoxAttachments.length === 0) { + throw new Error('상신할 DRM 첨부파일을 준비하지 못했습니다.'); + } + + // 6. 결재 상신용 payload 구성 const approvalPayload = { rfqId: data.rfqId, rfqCode: data.rfqCode, @@ -112,7 +118,7 @@ export async function requestTechSalesRfqSendWithApproval(data: TechSalesRfqSend }, }; - // 6. Saga로 결재 상신 + // 7. Saga로 결재 상신 const saga = new ApprovalSubmissionSaga( 'tech_sales_rfq_send_with_drm', // 핸들러 키 approvalPayload, // 결재 승인 후 실행될 데이터 @@ -127,6 +133,7 @@ export async function requestTechSalesRfqSendWithApproval(data: TechSalesRfqSend epId: data.currentUser.epId, email: data.currentUser.email, }, + attachments: knoxAttachments, } ); @@ -171,6 +178,59 @@ function getTechSalesRevalidationPath(rfqType: "SHIP" | "TOP" | "HULL"): string return "/evcp/budgetary-tech-sales-ship"; } } + +/** + * Knox 상신용 DRM 첨부파일을 File 객체로 준비 + */ +async function prepareKnoxDrmAttachments(attachmentIds: number[]): Promise<File[]> { + if (!attachmentIds || attachmentIds.length === 0) return []; + + const db = (await import('@/db/db')).default; + const { techSalesAttachments } = await import('@/db/schema/techSales'); + const { inArray } = await import('drizzle-orm'); + + const attachments = await db.query.techSalesAttachments.findMany({ + where: inArray(techSalesAttachments.id, attachmentIds), + columns: { + id: true, + filePath: true, + originalFileName: true, + fileName: true, + fileType: true, + }, + }); + + const baseUrl = process.env.NEXT_PUBLIC_BASE_URL || process.env.NEXT_PUBLIC_URL; + const files: File[] = []; + + for (const attachment of attachments) { + if (!attachment.filePath || !baseUrl) { + console.error('[TechSales RFQ Approval] 첨부파일 경로나 BASE_URL이 없습니다.', attachment.id); + continue; + } + + const fileUrl = `${baseUrl}${attachment.filePath}`; + const response = await fetch(fileUrl); + + if (!response.ok) { + console.error(`[TechSales RFQ Approval] 첨부파일 다운로드 실패: ${fileUrl} (status: ${response.status})`); + continue; + } + + const blob = await response.blob(); + const file = new File( + [blob], + attachment.originalFileName || attachment.fileName || 'attachment', + { + type: attachment.fileType || blob.type || 'application/octet-stream', + } + ); + + files.push(file); + } + + return files; +} /** * 기술영업 RFQ DRM 첨부 해제 결재 상신 * @@ -214,6 +274,12 @@ export async function requestRfqResendWithDrmApproval(data: { applicationReason: data.applicationReason, }); + // DRM 첨부파일을 Knox 상신용 File 객체로 준비 + const knoxAttachments = await prepareKnoxDrmAttachments(data.drmAttachmentIds); + if (knoxAttachments.length === 0) { + throw new Error('상신할 DRM 첨부파일을 준비하지 못했습니다.'); + } + // 결재 payload 구성 const approvalPayload = { rfqId: data.rfqId, @@ -244,6 +310,7 @@ export async function requestRfqResendWithDrmApproval(data: { epId: data.currentUser.epId, email: data.currentUser.email, }, + attachments: knoxAttachments, } ); |
