@hed-hog/operations 0.0.297 → 0.0.299
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/hedhog/frontend/app/_components/collaborator-details-screen.tsx.ejs +310 -310
- package/hedhog/frontend/app/_components/collaborator-form-screen.tsx.ejs +631 -631
- package/hedhog/frontend/app/_components/contract-details-screen.tsx.ejs +132 -132
- package/hedhog/frontend/app/_components/contract-form-screen.tsx.ejs +558 -558
- package/hedhog/frontend/app/_components/project-details-screen.tsx.ejs +291 -291
- package/hedhog/frontend/app/_components/project-form-screen.tsx.ejs +689 -689
- package/hedhog/frontend/app/_lib/api.ts.ejs +32 -32
- package/hedhog/frontend/app/_lib/hooks/use-operations-access.ts.ejs +44 -44
- package/hedhog/frontend/app/_lib/types.ts.ejs +360 -360
- package/hedhog/frontend/app/_lib/utils/format.ts.ejs +129 -129
- package/hedhog/frontend/app/_lib/utils/forms.ts.ejs +14 -14
- package/hedhog/frontend/app/approvals/page.tsx.ejs +386 -386
- package/hedhog/frontend/app/collaborators/[id]/edit/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/collaborators/[id]/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/collaborators/new/page.tsx.ejs +5 -5
- package/hedhog/frontend/app/collaborators/page.tsx.ejs +261 -261
- package/hedhog/frontend/app/contracts/[id]/edit/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/contracts/[id]/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/contracts/new/page.tsx.ejs +17 -17
- package/hedhog/frontend/app/contracts/page.tsx.ejs +262 -262
- package/hedhog/frontend/app/page.tsx.ejs +319 -319
- package/hedhog/frontend/app/projects/[id]/edit/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/projects/[id]/page.tsx.ejs +11 -11
- package/hedhog/frontend/app/projects/new/page.tsx.ejs +5 -5
- package/hedhog/frontend/app/projects/page.tsx.ejs +236 -236
- package/hedhog/frontend/app/schedule-adjustments/page.tsx.ejs +418 -418
- package/hedhog/frontend/app/team/page.tsx.ejs +339 -339
- package/hedhog/frontend/app/time-off/page.tsx.ejs +328 -328
- package/hedhog/frontend/app/timesheets/page.tsx.ejs +636 -636
- package/hedhog/frontend/messages/en.json +648 -648
- package/hedhog/frontend/messages/pt.json +647 -647
- package/package.json +4 -4
|
@@ -1,132 +1,132 @@
|
|
|
1
|
-
'use client';
|
|
2
|
-
|
|
3
|
-
import { EmptyState, Page } from '@/components/entity-list';
|
|
4
|
-
import { Button } from '@/components/ui/button';
|
|
5
|
-
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
|
6
|
-
import {
|
|
7
|
-
Table,
|
|
8
|
-
TableBody,
|
|
9
|
-
TableCell,
|
|
10
|
-
TableHead,
|
|
11
|
-
TableHeader,
|
|
12
|
-
TableRow,
|
|
13
|
-
} from '@/components/ui/table';
|
|
14
|
-
import { useApp, useQuery } from '@hed-hog/next-app-provider';
|
|
15
|
-
import { Download, FileText, Pencil } from 'lucide-react';
|
|
16
|
-
import Link from 'next/link';
|
|
17
|
-
import { useTranslations } from 'next-intl';
|
|
18
|
-
import { OperationsHeader } from './operations-header';
|
|
19
|
-
import { SectionCard } from './section-card';
|
|
20
|
-
import { StatusBadge } from './status-badge';
|
|
21
|
-
import { fetchOperations } from '../_lib/api';
|
|
22
|
-
import { useOperationsAccess } from '../_lib/hooks/use-operations-access';
|
|
23
|
-
import type { OperationsContractDetails } from '../_lib/types';
|
|
24
|
-
import {
|
|
25
|
-
formatCurrency,
|
|
26
|
-
formatDate,
|
|
27
|
-
formatDateTime,
|
|
28
|
-
formatEnumLabel,
|
|
29
|
-
getStatusBadgeClass,
|
|
30
|
-
} from '../_lib/utils/format';
|
|
31
|
-
|
|
32
|
-
function downloadBase64File(fileName: string, mimeType: string, base64: string) {
|
|
33
|
-
const href = `data:${mimeType};base64,${base64}`;
|
|
34
|
-
const link = document.createElement('a');
|
|
35
|
-
link.href = href;
|
|
36
|
-
link.download = fileName;
|
|
37
|
-
link.click();
|
|
38
|
-
}
|
|
39
|
-
|
|
40
|
-
export function ContractDetailsScreen({ contractId }: { contractId: number }) {
|
|
41
|
-
const t = useTranslations('operations.ContractDetailsPage');
|
|
42
|
-
const commonT = useTranslations('operations.Common');
|
|
43
|
-
const { request, currentLocaleCode } = useApp();
|
|
44
|
-
const access = useOperationsAccess();
|
|
45
|
-
|
|
46
|
-
const { data: contract, refetch } = useQuery<OperationsContractDetails>({
|
|
47
|
-
queryKey: ['operations-contract-details', currentLocaleCode, contractId],
|
|
48
|
-
queryFn: () =>
|
|
49
|
-
fetchOperations<OperationsContractDetails>(request, `/operations/contracts/${contractId}`),
|
|
50
|
-
});
|
|
51
|
-
|
|
52
|
-
if (!contract) {
|
|
53
|
-
return (
|
|
54
|
-
<Page>
|
|
55
|
-
<OperationsHeader title={t('title')} description={t('description')} current={t('breadcrumb')} />
|
|
56
|
-
<EmptyState
|
|
57
|
-
icon={<FileText className="size-12" />}
|
|
58
|
-
title={commonT('states.emptyTitle')}
|
|
59
|
-
description={t('notFound')}
|
|
60
|
-
actionLabel={commonT('actions.refresh')}
|
|
61
|
-
onAction={() => void refetch()}
|
|
62
|
-
/>
|
|
63
|
-
</Page>
|
|
64
|
-
);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
const currentPdf = contract.documents.find((document) => document.isCurrent);
|
|
68
|
-
|
|
69
|
-
return (
|
|
70
|
-
<Page>
|
|
71
|
-
<OperationsHeader
|
|
72
|
-
title={contract.name}
|
|
73
|
-
description={t('description')}
|
|
74
|
-
current={t('breadcrumb')}
|
|
75
|
-
actions={
|
|
76
|
-
<div className="flex gap-2">
|
|
77
|
-
{currentPdf?.fileContentBase64 ? (
|
|
78
|
-
<Button variant="outline" size="sm" onClick={() => downloadBase64File(currentPdf.fileName, currentPdf.mimeType, currentPdf.fileContentBase64 || '')}>
|
|
79
|
-
<Download className="size-4" />
|
|
80
|
-
{t('actions.downloadPdf')}
|
|
81
|
-
</Button>
|
|
82
|
-
) : null}
|
|
83
|
-
{access.isDirector ? (
|
|
84
|
-
<Button size="sm" asChild>
|
|
85
|
-
<Link href={`/operations/contracts/${contract.id}/edit`}>
|
|
86
|
-
<Pencil className="size-4" />
|
|
87
|
-
{commonT('actions.edit')}
|
|
88
|
-
</Link>
|
|
89
|
-
</Button>
|
|
90
|
-
) : null}
|
|
91
|
-
</div>
|
|
92
|
-
}
|
|
93
|
-
/>
|
|
94
|
-
|
|
95
|
-
<Tabs defaultValue="overview">
|
|
96
|
-
<TabsList className="flex-wrap">
|
|
97
|
-
<TabsTrigger value="overview">{t('tabs.overview')}</TabsTrigger>
|
|
98
|
-
<TabsTrigger value="parties">{t('tabs.parties')}</TabsTrigger>
|
|
99
|
-
<TabsTrigger value="signatures">{t('tabs.signatures')}</TabsTrigger>
|
|
100
|
-
<TabsTrigger value="financials">{t('tabs.financials')}</TabsTrigger>
|
|
101
|
-
<TabsTrigger value="documents">{t('tabs.documents')}</TabsTrigger>
|
|
102
|
-
<TabsTrigger value="revisions">{t('tabs.revisions')}</TabsTrigger>
|
|
103
|
-
<TabsTrigger value="history">{t('tabs.history')}</TabsTrigger>
|
|
104
|
-
</TabsList>
|
|
105
|
-
|
|
106
|
-
<TabsContent value="overview">
|
|
107
|
-
<SectionCard title={t('sections.overview')}>
|
|
108
|
-
<dl className="grid gap-3 text-sm md:grid-cols-3">
|
|
109
|
-
<div><dt className="text-muted-foreground">{commonT('labels.contract')}</dt><dd className="font-medium">{contract.name}</dd></div>
|
|
110
|
-
<div><dt className="text-muted-foreground">{t('labels.origin')}</dt><dd className="font-medium">{formatEnumLabel(contract.originType)}</dd></div>
|
|
111
|
-
<div><dt className="text-muted-foreground">{t('labels.mainParty')}</dt><dd className="font-medium">{contract.mainRelatedPartyName || commonT('labels.notAvailable')}</dd></div>
|
|
112
|
-
<div><dt className="text-muted-foreground">{t('labels.contractType')}</dt><dd className="font-medium">{formatEnumLabel(contract.contractType)}</dd></div>
|
|
113
|
-
<div><dt className="text-muted-foreground">{t('labels.signatureStatus')}</dt><dd className="font-medium"><StatusBadge label={formatEnumLabel(contract.signatureStatus)} className={getStatusBadgeClass(contract.signatureStatus)} /></dd></div>
|
|
114
|
-
<div><dt className="text-muted-foreground">{t('labels.activeState')}</dt><dd className="font-medium">{contract.isActive ? t('labels.active') : t('labels.inactive')}</dd></div>
|
|
115
|
-
<div><dt className="text-muted-foreground">{commonT('labels.startDate')}</dt><dd className="font-medium">{formatDate(contract.startDate)}</dd></div>
|
|
116
|
-
<div><dt className="text-muted-foreground">{commonT('labels.endDate')}</dt><dd className="font-medium">{formatDate(contract.endDate)}</dd></div>
|
|
117
|
-
<div><dt className="text-muted-foreground">{commonT('labels.status')}</dt><dd className="font-medium"><StatusBadge label={formatEnumLabel(contract.status)} className={getStatusBadgeClass(contract.status)} /></dd></div>
|
|
118
|
-
</dl>
|
|
119
|
-
{contract.contentHtml ? <div className="prose prose-sm mt-4 max-w-none rounded-lg border p-4" dangerouslySetInnerHTML={{ __html: contract.contentHtml }} /> : null}
|
|
120
|
-
</SectionCard>
|
|
121
|
-
</TabsContent>
|
|
122
|
-
|
|
123
|
-
<TabsContent value="parties"><SectionCard title={t('sections.parties')}>{contract.parties.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.mainParty')}</TableHead><TableHead>{t('labels.partyRole')}</TableHead><TableHead>{t('labels.partyType')}</TableHead><TableHead>{t('labels.documentNumber')}</TableHead></TableRow></TableHeader><TableBody>{contract.parties.map((party) => <TableRow key={party.id}><TableCell>{party.displayName}</TableCell><TableCell>{formatEnumLabel(party.partyRole)}</TableCell><TableCell>{formatEnumLabel(party.partyType)}</TableCell><TableCell>{party.documentNumber || commonT('labels.notAvailable')}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noParties')}</p>}</SectionCard></TabsContent>
|
|
124
|
-
<TabsContent value="signatures"><SectionCard title={t('sections.signatures')}>{contract.signatures.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.signer')}</TableHead><TableHead>{commonT('labels.status')}</TableHead><TableHead>{t('labels.signedAt')}</TableHead></TableRow></TableHeader><TableBody>{contract.signatures.map((signature) => <TableRow key={signature.id}><TableCell>{signature.signerName}</TableCell><TableCell><StatusBadge label={formatEnumLabel(signature.status)} className={getStatusBadgeClass(signature.status)} /></TableCell><TableCell>{formatDate(signature.signedAt)}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noSignatures')}</p>}</SectionCard></TabsContent>
|
|
125
|
-
<TabsContent value="financials"><SectionCard title={t('sections.financials')}><dl className="grid gap-3 text-sm md:grid-cols-4"><div><dt className="text-muted-foreground">{t('labels.value')}</dt><dd className="font-medium">{formatCurrency(contract.valueAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.payment')}</dt><dd className="font-medium">{formatCurrency(contract.paymentAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.revenue')}</dt><dd className="font-medium">{formatCurrency(contract.revenueAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.fine')}</dt><dd className="font-medium">{formatCurrency(contract.fineAmount ?? 0)}</dd></div></dl></SectionCard></TabsContent>
|
|
126
|
-
<TabsContent value="documents"><SectionCard title={t('sections.documents')}>{contract.documents.length ? <div className="space-y-3">{contract.documents.map((document) => <div key={document.id} className="flex items-center justify-between rounded-lg border px-4 py-3"><div><div className="font-medium">{document.fileName}</div><div className="text-xs text-muted-foreground">{formatEnumLabel(document.documentType)} • {formatDateTime(document.createdAt)}</div></div>{document.fileContentBase64 ? <Button variant="outline" size="sm" onClick={() => downloadBase64File(document.fileName, document.mimeType, document.fileContentBase64 || '')}><Download className="size-4" />{t('actions.download')}</Button> : null}</div>)}</div> : <p className="text-sm text-muted-foreground">{t('states.noDocuments')}</p>}</SectionCard></TabsContent>
|
|
127
|
-
<TabsContent value="revisions"><SectionCard title={t('sections.revisions')}>{contract.revisions.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.revision')}</TableHead><TableHead>{commonT('labels.status')}</TableHead><TableHead>{commonT('labels.startDate')}</TableHead></TableRow></TableHeader><TableBody>{contract.revisions.map((revision) => <TableRow key={revision.id}><TableCell>{revision.title}</TableCell><TableCell><StatusBadge label={formatEnumLabel(revision.status)} className={getStatusBadgeClass(revision.status)} /></TableCell><TableCell>{formatDate(revision.effectiveDate)}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noRevisions')}</p>}</SectionCard></TabsContent>
|
|
128
|
-
<TabsContent value="history"><SectionCard title={t('sections.history')}>{contract.history.length ? <div className="space-y-3">{contract.history.map((item) => <div key={item.id} className="rounded-lg border px-4 py-3 text-sm"><div className="font-medium">{formatEnumLabel(item.action)}</div><div className="text-muted-foreground">{formatDateTime(item.createdAt)}</div><div>{item.note || commonT('labels.noNotes')}</div></div>)}</div> : <p className="text-sm text-muted-foreground">{t('states.noHistory')}</p>}</SectionCard></TabsContent>
|
|
129
|
-
</Tabs>
|
|
130
|
-
</Page>
|
|
131
|
-
);
|
|
132
|
-
}
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { EmptyState, Page } from '@/components/entity-list';
|
|
4
|
+
import { Button } from '@/components/ui/button';
|
|
5
|
+
import { Tabs, TabsContent, TabsList, TabsTrigger } from '@/components/ui/tabs';
|
|
6
|
+
import {
|
|
7
|
+
Table,
|
|
8
|
+
TableBody,
|
|
9
|
+
TableCell,
|
|
10
|
+
TableHead,
|
|
11
|
+
TableHeader,
|
|
12
|
+
TableRow,
|
|
13
|
+
} from '@/components/ui/table';
|
|
14
|
+
import { useApp, useQuery } from '@hed-hog/next-app-provider';
|
|
15
|
+
import { Download, FileText, Pencil } from 'lucide-react';
|
|
16
|
+
import Link from 'next/link';
|
|
17
|
+
import { useTranslations } from 'next-intl';
|
|
18
|
+
import { OperationsHeader } from './operations-header';
|
|
19
|
+
import { SectionCard } from './section-card';
|
|
20
|
+
import { StatusBadge } from './status-badge';
|
|
21
|
+
import { fetchOperations } from '../_lib/api';
|
|
22
|
+
import { useOperationsAccess } from '../_lib/hooks/use-operations-access';
|
|
23
|
+
import type { OperationsContractDetails } from '../_lib/types';
|
|
24
|
+
import {
|
|
25
|
+
formatCurrency,
|
|
26
|
+
formatDate,
|
|
27
|
+
formatDateTime,
|
|
28
|
+
formatEnumLabel,
|
|
29
|
+
getStatusBadgeClass,
|
|
30
|
+
} from '../_lib/utils/format';
|
|
31
|
+
|
|
32
|
+
function downloadBase64File(fileName: string, mimeType: string, base64: string) {
|
|
33
|
+
const href = `data:${mimeType};base64,${base64}`;
|
|
34
|
+
const link = document.createElement('a');
|
|
35
|
+
link.href = href;
|
|
36
|
+
link.download = fileName;
|
|
37
|
+
link.click();
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
export function ContractDetailsScreen({ contractId }: { contractId: number }) {
|
|
41
|
+
const t = useTranslations('operations.ContractDetailsPage');
|
|
42
|
+
const commonT = useTranslations('operations.Common');
|
|
43
|
+
const { request, currentLocaleCode } = useApp();
|
|
44
|
+
const access = useOperationsAccess();
|
|
45
|
+
|
|
46
|
+
const { data: contract, refetch } = useQuery<OperationsContractDetails>({
|
|
47
|
+
queryKey: ['operations-contract-details', currentLocaleCode, contractId],
|
|
48
|
+
queryFn: () =>
|
|
49
|
+
fetchOperations<OperationsContractDetails>(request, `/operations/contracts/${contractId}`),
|
|
50
|
+
});
|
|
51
|
+
|
|
52
|
+
if (!contract) {
|
|
53
|
+
return (
|
|
54
|
+
<Page>
|
|
55
|
+
<OperationsHeader title={t('title')} description={t('description')} current={t('breadcrumb')} />
|
|
56
|
+
<EmptyState
|
|
57
|
+
icon={<FileText className="size-12" />}
|
|
58
|
+
title={commonT('states.emptyTitle')}
|
|
59
|
+
description={t('notFound')}
|
|
60
|
+
actionLabel={commonT('actions.refresh')}
|
|
61
|
+
onAction={() => void refetch()}
|
|
62
|
+
/>
|
|
63
|
+
</Page>
|
|
64
|
+
);
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
const currentPdf = contract.documents.find((document) => document.isCurrent);
|
|
68
|
+
|
|
69
|
+
return (
|
|
70
|
+
<Page>
|
|
71
|
+
<OperationsHeader
|
|
72
|
+
title={contract.name}
|
|
73
|
+
description={t('description')}
|
|
74
|
+
current={t('breadcrumb')}
|
|
75
|
+
actions={
|
|
76
|
+
<div className="flex gap-2">
|
|
77
|
+
{currentPdf?.fileContentBase64 ? (
|
|
78
|
+
<Button variant="outline" size="sm" onClick={() => downloadBase64File(currentPdf.fileName, currentPdf.mimeType, currentPdf.fileContentBase64 || '')}>
|
|
79
|
+
<Download className="size-4" />
|
|
80
|
+
{t('actions.downloadPdf')}
|
|
81
|
+
</Button>
|
|
82
|
+
) : null}
|
|
83
|
+
{access.isDirector ? (
|
|
84
|
+
<Button size="sm" asChild>
|
|
85
|
+
<Link href={`/operations/contracts/${contract.id}/edit`}>
|
|
86
|
+
<Pencil className="size-4" />
|
|
87
|
+
{commonT('actions.edit')}
|
|
88
|
+
</Link>
|
|
89
|
+
</Button>
|
|
90
|
+
) : null}
|
|
91
|
+
</div>
|
|
92
|
+
}
|
|
93
|
+
/>
|
|
94
|
+
|
|
95
|
+
<Tabs defaultValue="overview">
|
|
96
|
+
<TabsList className="flex-wrap">
|
|
97
|
+
<TabsTrigger value="overview">{t('tabs.overview')}</TabsTrigger>
|
|
98
|
+
<TabsTrigger value="parties">{t('tabs.parties')}</TabsTrigger>
|
|
99
|
+
<TabsTrigger value="signatures">{t('tabs.signatures')}</TabsTrigger>
|
|
100
|
+
<TabsTrigger value="financials">{t('tabs.financials')}</TabsTrigger>
|
|
101
|
+
<TabsTrigger value="documents">{t('tabs.documents')}</TabsTrigger>
|
|
102
|
+
<TabsTrigger value="revisions">{t('tabs.revisions')}</TabsTrigger>
|
|
103
|
+
<TabsTrigger value="history">{t('tabs.history')}</TabsTrigger>
|
|
104
|
+
</TabsList>
|
|
105
|
+
|
|
106
|
+
<TabsContent value="overview">
|
|
107
|
+
<SectionCard title={t('sections.overview')}>
|
|
108
|
+
<dl className="grid gap-3 text-sm md:grid-cols-3">
|
|
109
|
+
<div><dt className="text-muted-foreground">{commonT('labels.contract')}</dt><dd className="font-medium">{contract.name}</dd></div>
|
|
110
|
+
<div><dt className="text-muted-foreground">{t('labels.origin')}</dt><dd className="font-medium">{formatEnumLabel(contract.originType)}</dd></div>
|
|
111
|
+
<div><dt className="text-muted-foreground">{t('labels.mainParty')}</dt><dd className="font-medium">{contract.mainRelatedPartyName || commonT('labels.notAvailable')}</dd></div>
|
|
112
|
+
<div><dt className="text-muted-foreground">{t('labels.contractType')}</dt><dd className="font-medium">{formatEnumLabel(contract.contractType)}</dd></div>
|
|
113
|
+
<div><dt className="text-muted-foreground">{t('labels.signatureStatus')}</dt><dd className="font-medium"><StatusBadge label={formatEnumLabel(contract.signatureStatus)} className={getStatusBadgeClass(contract.signatureStatus)} /></dd></div>
|
|
114
|
+
<div><dt className="text-muted-foreground">{t('labels.activeState')}</dt><dd className="font-medium">{contract.isActive ? t('labels.active') : t('labels.inactive')}</dd></div>
|
|
115
|
+
<div><dt className="text-muted-foreground">{commonT('labels.startDate')}</dt><dd className="font-medium">{formatDate(contract.startDate)}</dd></div>
|
|
116
|
+
<div><dt className="text-muted-foreground">{commonT('labels.endDate')}</dt><dd className="font-medium">{formatDate(contract.endDate)}</dd></div>
|
|
117
|
+
<div><dt className="text-muted-foreground">{commonT('labels.status')}</dt><dd className="font-medium"><StatusBadge label={formatEnumLabel(contract.status)} className={getStatusBadgeClass(contract.status)} /></dd></div>
|
|
118
|
+
</dl>
|
|
119
|
+
{contract.contentHtml ? <div className="prose prose-sm mt-4 max-w-none rounded-lg border p-4" dangerouslySetInnerHTML={{ __html: contract.contentHtml }} /> : null}
|
|
120
|
+
</SectionCard>
|
|
121
|
+
</TabsContent>
|
|
122
|
+
|
|
123
|
+
<TabsContent value="parties"><SectionCard title={t('sections.parties')}>{contract.parties.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.mainParty')}</TableHead><TableHead>{t('labels.partyRole')}</TableHead><TableHead>{t('labels.partyType')}</TableHead><TableHead>{t('labels.documentNumber')}</TableHead></TableRow></TableHeader><TableBody>{contract.parties.map((party) => <TableRow key={party.id}><TableCell>{party.displayName}</TableCell><TableCell>{formatEnumLabel(party.partyRole)}</TableCell><TableCell>{formatEnumLabel(party.partyType)}</TableCell><TableCell>{party.documentNumber || commonT('labels.notAvailable')}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noParties')}</p>}</SectionCard></TabsContent>
|
|
124
|
+
<TabsContent value="signatures"><SectionCard title={t('sections.signatures')}>{contract.signatures.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.signer')}</TableHead><TableHead>{commonT('labels.status')}</TableHead><TableHead>{t('labels.signedAt')}</TableHead></TableRow></TableHeader><TableBody>{contract.signatures.map((signature) => <TableRow key={signature.id}><TableCell>{signature.signerName}</TableCell><TableCell><StatusBadge label={formatEnumLabel(signature.status)} className={getStatusBadgeClass(signature.status)} /></TableCell><TableCell>{formatDate(signature.signedAt)}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noSignatures')}</p>}</SectionCard></TabsContent>
|
|
125
|
+
<TabsContent value="financials"><SectionCard title={t('sections.financials')}><dl className="grid gap-3 text-sm md:grid-cols-4"><div><dt className="text-muted-foreground">{t('labels.value')}</dt><dd className="font-medium">{formatCurrency(contract.valueAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.payment')}</dt><dd className="font-medium">{formatCurrency(contract.paymentAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.revenue')}</dt><dd className="font-medium">{formatCurrency(contract.revenueAmount ?? 0)}</dd></div><div><dt className="text-muted-foreground">{t('labels.fine')}</dt><dd className="font-medium">{formatCurrency(contract.fineAmount ?? 0)}</dd></div></dl></SectionCard></TabsContent>
|
|
126
|
+
<TabsContent value="documents"><SectionCard title={t('sections.documents')}>{contract.documents.length ? <div className="space-y-3">{contract.documents.map((document) => <div key={document.id} className="flex items-center justify-between rounded-lg border px-4 py-3"><div><div className="font-medium">{document.fileName}</div><div className="text-xs text-muted-foreground">{formatEnumLabel(document.documentType)} • {formatDateTime(document.createdAt)}</div></div>{document.fileContentBase64 ? <Button variant="outline" size="sm" onClick={() => downloadBase64File(document.fileName, document.mimeType, document.fileContentBase64 || '')}><Download className="size-4" />{t('actions.download')}</Button> : null}</div>)}</div> : <p className="text-sm text-muted-foreground">{t('states.noDocuments')}</p>}</SectionCard></TabsContent>
|
|
127
|
+
<TabsContent value="revisions"><SectionCard title={t('sections.revisions')}>{contract.revisions.length ? <div className="overflow-x-auto rounded-md border"><Table><TableHeader><TableRow><TableHead>{t('labels.revision')}</TableHead><TableHead>{commonT('labels.status')}</TableHead><TableHead>{commonT('labels.startDate')}</TableHead></TableRow></TableHeader><TableBody>{contract.revisions.map((revision) => <TableRow key={revision.id}><TableCell>{revision.title}</TableCell><TableCell><StatusBadge label={formatEnumLabel(revision.status)} className={getStatusBadgeClass(revision.status)} /></TableCell><TableCell>{formatDate(revision.effectiveDate)}</TableCell></TableRow>)}</TableBody></Table></div> : <p className="text-sm text-muted-foreground">{t('states.noRevisions')}</p>}</SectionCard></TabsContent>
|
|
128
|
+
<TabsContent value="history"><SectionCard title={t('sections.history')}>{contract.history.length ? <div className="space-y-3">{contract.history.map((item) => <div key={item.id} className="rounded-lg border px-4 py-3 text-sm"><div className="font-medium">{formatEnumLabel(item.action)}</div><div className="text-muted-foreground">{formatDateTime(item.createdAt)}</div><div>{item.note || commonT('labels.noNotes')}</div></div>)}</div> : <p className="text-sm text-muted-foreground">{t('states.noHistory')}</p>}</SectionCard></TabsContent>
|
|
129
|
+
</Tabs>
|
|
130
|
+
</Page>
|
|
131
|
+
);
|
|
132
|
+
}
|