@hed-hog/billing 0.0.2

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.
Files changed (232) hide show
  1. package/dist/adapters/billing-gateway-registry.d.ts +8 -0
  2. package/dist/adapters/billing-gateway-registry.d.ts.map +1 -0
  3. package/dist/adapters/billing-gateway-registry.js +33 -0
  4. package/dist/adapters/billing-gateway-registry.js.map +1 -0
  5. package/dist/adapters/mercadopago.adapter.d.ts +44 -0
  6. package/dist/adapters/mercadopago.adapter.d.ts.map +1 -0
  7. package/dist/adapters/mercadopago.adapter.js +68 -0
  8. package/dist/adapters/mercadopago.adapter.js.map +1 -0
  9. package/dist/adapters/pagarme.adapter.d.ts +44 -0
  10. package/dist/adapters/pagarme.adapter.d.ts.map +1 -0
  11. package/dist/adapters/pagarme.adapter.js +68 -0
  12. package/dist/adapters/pagarme.adapter.js.map +1 -0
  13. package/dist/adapters/payment-gateway.adapter.d.ts +15 -0
  14. package/dist/adapters/payment-gateway.adapter.d.ts.map +1 -0
  15. package/dist/adapters/payment-gateway.adapter.js +3 -0
  16. package/dist/adapters/payment-gateway.adapter.js.map +1 -0
  17. package/dist/adapters/stripe.adapter.d.ts +44 -0
  18. package/dist/adapters/stripe.adapter.d.ts.map +1 -0
  19. package/dist/adapters/stripe.adapter.js +69 -0
  20. package/dist/adapters/stripe.adapter.js.map +1 -0
  21. package/dist/billing-contracts.controller.d.ts +22 -0
  22. package/dist/billing-contracts.controller.d.ts.map +1 -0
  23. package/dist/billing-contracts.controller.js +84 -0
  24. package/dist/billing-contracts.controller.js.map +1 -0
  25. package/dist/billing-coupons.controller.d.ts +22 -0
  26. package/dist/billing-coupons.controller.d.ts.map +1 -0
  27. package/dist/billing-coupons.controller.js +84 -0
  28. package/dist/billing-coupons.controller.js.map +1 -0
  29. package/dist/billing-dashboard.controller.d.ts +13 -0
  30. package/dist/billing-dashboard.controller.d.ts.map +1 -0
  31. package/dist/billing-dashboard.controller.js +36 -0
  32. package/dist/billing-dashboard.controller.js.map +1 -0
  33. package/dist/billing-entitlements.controller.d.ts +19 -0
  34. package/dist/billing-entitlements.controller.d.ts.map +1 -0
  35. package/dist/billing-entitlements.controller.js +62 -0
  36. package/dist/billing-entitlements.controller.js.map +1 -0
  37. package/dist/billing-gateways.controller.d.ts +8 -0
  38. package/dist/billing-gateways.controller.d.ts.map +1 -0
  39. package/dist/billing-gateways.controller.js +50 -0
  40. package/dist/billing-gateways.controller.js.map +1 -0
  41. package/dist/billing-invoices.controller.d.ts +18 -0
  42. package/dist/billing-invoices.controller.d.ts.map +1 -0
  43. package/dist/billing-invoices.controller.js +61 -0
  44. package/dist/billing-invoices.controller.js.map +1 -0
  45. package/dist/billing-offers.controller.d.ts +22 -0
  46. package/dist/billing-offers.controller.d.ts.map +1 -0
  47. package/dist/billing-offers.controller.js +84 -0
  48. package/dist/billing-offers.controller.js.map +1 -0
  49. package/dist/billing-orders.controller.d.ts +19 -0
  50. package/dist/billing-orders.controller.d.ts.map +1 -0
  51. package/dist/billing-orders.controller.js +62 -0
  52. package/dist/billing-orders.controller.js.map +1 -0
  53. package/dist/billing-payments.controller.d.ts +17 -0
  54. package/dist/billing-payments.controller.d.ts.map +1 -0
  55. package/dist/billing-payments.controller.js +51 -0
  56. package/dist/billing-payments.controller.js.map +1 -0
  57. package/dist/billing-prices.controller.d.ts +22 -0
  58. package/dist/billing-prices.controller.d.ts.map +1 -0
  59. package/dist/billing-prices.controller.js +84 -0
  60. package/dist/billing-prices.controller.js.map +1 -0
  61. package/dist/billing-products.controller.d.ts +22 -0
  62. package/dist/billing-products.controller.d.ts.map +1 -0
  63. package/dist/billing-products.controller.js +84 -0
  64. package/dist/billing-products.controller.js.map +1 -0
  65. package/dist/billing-refunds.controller.d.ts +16 -0
  66. package/dist/billing-refunds.controller.d.ts.map +1 -0
  67. package/dist/billing-refunds.controller.js +41 -0
  68. package/dist/billing-refunds.controller.js.map +1 -0
  69. package/dist/billing-subscriptions.controller.d.ts +22 -0
  70. package/dist/billing-subscriptions.controller.d.ts.map +1 -0
  71. package/dist/billing-subscriptions.controller.js +92 -0
  72. package/dist/billing-subscriptions.controller.js.map +1 -0
  73. package/dist/billing-webhooks.controller.d.ts +16 -0
  74. package/dist/billing-webhooks.controller.d.ts.map +1 -0
  75. package/dist/billing-webhooks.controller.js +41 -0
  76. package/dist/billing-webhooks.controller.js.map +1 -0
  77. package/dist/billing.module.d.ts +3 -0
  78. package/dist/billing.module.d.ts.map +1 -0
  79. package/dist/billing.module.js +61 -0
  80. package/dist/billing.module.js.map +1 -0
  81. package/dist/billing.service.d.ts +169 -0
  82. package/dist/billing.service.d.ts.map +1 -0
  83. package/dist/billing.service.js +290 -0
  84. package/dist/billing.service.js.map +1 -0
  85. package/dist/dto/create-contract.dto.d.ts +11 -0
  86. package/dist/dto/create-contract.dto.d.ts.map +1 -0
  87. package/dist/dto/create-contract.dto.js +52 -0
  88. package/dist/dto/create-contract.dto.js.map +1 -0
  89. package/dist/dto/create-coupon.dto.d.ts +12 -0
  90. package/dist/dto/create-coupon.dto.d.ts.map +1 -0
  91. package/dist/dto/create-coupon.dto.js +60 -0
  92. package/dist/dto/create-coupon.dto.js.map +1 -0
  93. package/dist/dto/create-entitlement.dto.d.ts +12 -0
  94. package/dist/dto/create-entitlement.dto.d.ts.map +1 -0
  95. package/dist/dto/create-entitlement.dto.js +54 -0
  96. package/dist/dto/create-entitlement.dto.js.map +1 -0
  97. package/dist/dto/create-offer.dto.d.ts +7 -0
  98. package/dist/dto/create-offer.dto.d.ts.map +1 -0
  99. package/dist/dto/create-offer.dto.js +35 -0
  100. package/dist/dto/create-offer.dto.js.map +1 -0
  101. package/dist/dto/create-order.dto.d.ts +12 -0
  102. package/dist/dto/create-order.dto.d.ts.map +1 -0
  103. package/dist/dto/create-order.dto.js +65 -0
  104. package/dist/dto/create-order.dto.js.map +1 -0
  105. package/dist/dto/create-price.dto.d.ts +15 -0
  106. package/dist/dto/create-price.dto.d.ts.map +1 -0
  107. package/dist/dto/create-price.dto.js +76 -0
  108. package/dist/dto/create-price.dto.js.map +1 -0
  109. package/dist/dto/create-product.dto.d.ts +9 -0
  110. package/dist/dto/create-product.dto.d.ts.map +1 -0
  111. package/dist/dto/create-product.dto.js +45 -0
  112. package/dist/dto/create-product.dto.js.map +1 -0
  113. package/dist/dto/create-subscription.dto.d.ts +14 -0
  114. package/dist/dto/create-subscription.dto.d.ts.map +1 -0
  115. package/dist/dto/create-subscription.dto.js +67 -0
  116. package/dist/dto/create-subscription.dto.js.map +1 -0
  117. package/dist/dto/update-contract.dto.d.ts +6 -0
  118. package/dist/dto/update-contract.dto.d.ts.map +1 -0
  119. package/dist/dto/update-contract.dto.js +9 -0
  120. package/dist/dto/update-contract.dto.js.map +1 -0
  121. package/dist/dto/update-coupon.dto.d.ts +6 -0
  122. package/dist/dto/update-coupon.dto.d.ts.map +1 -0
  123. package/dist/dto/update-coupon.dto.js +9 -0
  124. package/dist/dto/update-coupon.dto.js.map +1 -0
  125. package/dist/dto/update-offer.dto.d.ts +6 -0
  126. package/dist/dto/update-offer.dto.d.ts.map +1 -0
  127. package/dist/dto/update-offer.dto.js +9 -0
  128. package/dist/dto/update-offer.dto.js.map +1 -0
  129. package/dist/dto/update-order.dto.d.ts +6 -0
  130. package/dist/dto/update-order.dto.d.ts.map +1 -0
  131. package/dist/dto/update-order.dto.js +9 -0
  132. package/dist/dto/update-order.dto.js.map +1 -0
  133. package/dist/dto/update-price.dto.d.ts +6 -0
  134. package/dist/dto/update-price.dto.d.ts.map +1 -0
  135. package/dist/dto/update-price.dto.js +9 -0
  136. package/dist/dto/update-price.dto.js.map +1 -0
  137. package/dist/dto/update-product.dto.d.ts +6 -0
  138. package/dist/dto/update-product.dto.d.ts.map +1 -0
  139. package/dist/dto/update-product.dto.js +9 -0
  140. package/dist/dto/update-product.dto.js.map +1 -0
  141. package/dist/dto/update-subscription.dto.d.ts +6 -0
  142. package/dist/dto/update-subscription.dto.d.ts.map +1 -0
  143. package/dist/dto/update-subscription.dto.js +9 -0
  144. package/dist/dto/update-subscription.dto.js.map +1 -0
  145. package/dist/index.d.ts +32 -0
  146. package/dist/index.d.ts.map +1 -0
  147. package/dist/index.js +48 -0
  148. package/dist/index.js.map +1 -0
  149. package/hedhog/data/menu.yaml +284 -0
  150. package/hedhog/data/role.yaml +7 -0
  151. package/hedhog/data/route.yaml +422 -0
  152. package/hedhog/frontend/app/_lib/billing-mocks.ts.ejs +270 -0
  153. package/hedhog/frontend/app/contracts/page.tsx.ejs +562 -0
  154. package/hedhog/frontend/app/coupons/page.tsx.ejs +669 -0
  155. package/hedhog/frontend/app/entitlements/page.tsx.ejs +526 -0
  156. package/hedhog/frontend/app/gateways/page.tsx.ejs +308 -0
  157. package/hedhog/frontend/app/invoices/page.tsx.ejs +179 -0
  158. package/hedhog/frontend/app/offers/page.tsx.ejs +483 -0
  159. package/hedhog/frontend/app/orders/page.tsx.ejs +424 -0
  160. package/hedhog/frontend/app/page.tsx.ejs +186 -0
  161. package/hedhog/frontend/app/payments/page.tsx.ejs +187 -0
  162. package/hedhog/frontend/app/prices/page.tsx.ejs +704 -0
  163. package/hedhog/frontend/app/products/page.tsx.ejs +568 -0
  164. package/hedhog/frontend/app/refunds/page.tsx.ejs +174 -0
  165. package/hedhog/frontend/app/reports/page.tsx.ejs +177 -0
  166. package/hedhog/frontend/app/subscriptions/page.tsx.ejs +283 -0
  167. package/hedhog/frontend/app/webhooks/page.tsx.ejs +172 -0
  168. package/hedhog/frontend/messages/en.json +551 -0
  169. package/hedhog/frontend/messages/pt.json +563 -0
  170. package/hedhog/table/billing_contract.yaml +37 -0
  171. package/hedhog/table/billing_contract_seat.yaml +28 -0
  172. package/hedhog/table/billing_coupon.yaml +42 -0
  173. package/hedhog/table/billing_entitlement.yaml +41 -0
  174. package/hedhog/table/billing_entitlement_event.yaml +20 -0
  175. package/hedhog/table/billing_invoice.yaml +60 -0
  176. package/hedhog/table/billing_invoice_item.yaml +25 -0
  177. package/hedhog/table/billing_offer.yaml +22 -0
  178. package/hedhog/table/billing_offer_price.yaml +23 -0
  179. package/hedhog/table/billing_order.yaml +50 -0
  180. package/hedhog/table/billing_order_item.yaml +35 -0
  181. package/hedhog/table/billing_payment.yaml +66 -0
  182. package/hedhog/table/billing_payment_method.yaml +49 -0
  183. package/hedhog/table/billing_payment_provider.yaml +21 -0
  184. package/hedhog/table/billing_price.yaml +53 -0
  185. package/hedhog/table/billing_product.yaml +31 -0
  186. package/hedhog/table/billing_product_item.yaml +24 -0
  187. package/hedhog/table/billing_provider_event.yaml +31 -0
  188. package/hedhog/table/billing_refund.yaml +41 -0
  189. package/hedhog/table/billing_seat_allocation.yaml +42 -0
  190. package/hedhog/table/billing_subscription.yaml +66 -0
  191. package/hedhog/table/billing_subscription_event.yaml +20 -0
  192. package/hedhog/table/billing_subscription_item.yaml +29 -0
  193. package/package.json +37 -0
  194. package/src/adapters/billing-gateway-registry.ts +23 -0
  195. package/src/adapters/mercadopago.adapter.ts +66 -0
  196. package/src/adapters/pagarme.adapter.ts +66 -0
  197. package/src/adapters/payment-gateway.adapter.ts +14 -0
  198. package/src/adapters/stripe.adapter.ts +67 -0
  199. package/src/billing-contracts.controller.ts +46 -0
  200. package/src/billing-coupons.controller.ts +46 -0
  201. package/src/billing-dashboard.controller.ts +14 -0
  202. package/src/billing-entitlements.controller.ts +34 -0
  203. package/src/billing-gateways.controller.ts +19 -0
  204. package/src/billing-invoices.controller.ts +32 -0
  205. package/src/billing-offers.controller.ts +46 -0
  206. package/src/billing-orders.controller.ts +33 -0
  207. package/src/billing-payments.controller.ts +20 -0
  208. package/src/billing-prices.controller.ts +46 -0
  209. package/src/billing-products.controller.ts +46 -0
  210. package/src/billing-refunds.controller.ts +15 -0
  211. package/src/billing-subscriptions.controller.ts +49 -0
  212. package/src/billing-webhooks.controller.ts +15 -0
  213. package/src/billing.module.ts +48 -0
  214. package/src/billing.service.ts +391 -0
  215. package/src/dto/create-contract.dto.ts +30 -0
  216. package/src/dto/create-coupon.dto.ts +37 -0
  217. package/src/dto/create-entitlement.dto.ts +31 -0
  218. package/src/dto/create-offer.dto.ts +17 -0
  219. package/src/dto/create-order.dto.ts +42 -0
  220. package/src/dto/create-price.dto.ts +50 -0
  221. package/src/dto/create-product.dto.ts +25 -0
  222. package/src/dto/create-subscription.dto.ts +42 -0
  223. package/src/dto/update-contract.dto.ts +4 -0
  224. package/src/dto/update-coupon.dto.ts +4 -0
  225. package/src/dto/update-offer.dto.ts +4 -0
  226. package/src/dto/update-order.dto.ts +4 -0
  227. package/src/dto/update-price.dto.ts +4 -0
  228. package/src/dto/update-product.dto.ts +4 -0
  229. package/src/dto/update-subscription.dto.ts +4 -0
  230. package/src/index.ts +32 -0
  231. package/src/language/en.json +8 -0
  232. package/src/language/pt.json +8 -0
@@ -0,0 +1,526 @@
1
+ 'use client';
2
+
3
+ import {
4
+ Page,
5
+ PageHeader,
6
+ PaginationFooter,
7
+ SearchBar,
8
+ } from '@/components/entity-list';
9
+ import {
10
+ AlertDialog,
11
+ AlertDialogAction,
12
+ AlertDialogCancel,
13
+ AlertDialogContent,
14
+ AlertDialogDescription,
15
+ AlertDialogFooter,
16
+ AlertDialogHeader,
17
+ AlertDialogTitle,
18
+ } from '@/components/ui/alert-dialog';
19
+ import { Badge } from '@/components/ui/badge';
20
+ import { Button } from '@/components/ui/button';
21
+ import {
22
+ Form,
23
+ FormControl,
24
+ FormField,
25
+ FormItem,
26
+ FormLabel,
27
+ FormMessage,
28
+ } from '@/components/ui/form';
29
+ import { Input } from '@/components/ui/input';
30
+ import {
31
+ Select,
32
+ SelectContent,
33
+ SelectItem,
34
+ SelectTrigger,
35
+ SelectValue,
36
+ } from '@/components/ui/select';
37
+ import {
38
+ Sheet,
39
+ SheetContent,
40
+ SheetDescription,
41
+ SheetHeader,
42
+ SheetTitle,
43
+ } from '@/components/ui/sheet';
44
+ import {
45
+ Table,
46
+ TableBody,
47
+ TableCell,
48
+ TableHead,
49
+ TableHeader,
50
+ TableRow,
51
+ } from '@/components/ui/table';
52
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
53
+ import { zodResolver } from '@hookform/resolvers/zod';
54
+ import { Plus, Trash2 } from 'lucide-react';
55
+ import { useTranslations } from 'next-intl';
56
+ import { useState } from 'react';
57
+ import { useForm } from 'react-hook-form';
58
+ import { z } from 'zod';
59
+
60
+ type Entitlement = {
61
+ id: number;
62
+ contact_id: number;
63
+ source_type: string;
64
+ target_type: string;
65
+ access_type: string;
66
+ status: string;
67
+ starts_at: string | null;
68
+ ends_at: string | null;
69
+ };
70
+
71
+ type ListResponse<T> = {
72
+ data?: T[];
73
+ total?: number;
74
+ };
75
+
76
+ const schema = z.object({
77
+ contact_id: z.coerce.number().min(1),
78
+ source_type: z.string().trim().min(1),
79
+ source_id: z.coerce.number().min(1),
80
+ target_type: z.string().trim().min(1),
81
+ target_id: z.coerce.number().min(1),
82
+ access_type: z.string().trim().min(1),
83
+ starts_at: z.string().optional(),
84
+ ends_at: z.string().optional(),
85
+ status: z.enum(['active', 'revoked', 'expired']),
86
+ });
87
+
88
+ type EntitlementFormValues = z.infer<typeof schema>;
89
+
90
+ const badgeClass = (value: string) => {
91
+ if (['active', 'paid', 'authorized', 'approved'].includes(value)) {
92
+ return 'bg-green-100 text-green-800';
93
+ }
94
+ if (['pending', 'open', 'draft'].includes(value)) {
95
+ return 'bg-yellow-100 text-yellow-800';
96
+ }
97
+ if (['failed', 'canceled', 'void', 'revoked'].includes(value)) {
98
+ return 'bg-red-100 text-red-800';
99
+ }
100
+ if (['paused', 'inactive'].includes(value)) {
101
+ return 'bg-gray-100 text-gray-800';
102
+ }
103
+ return 'bg-slate-100 text-slate-800';
104
+ };
105
+
106
+ export default function BillingEntitlementsPage() {
107
+ const t = useTranslations('BillingEntitlementsPage');
108
+ const { request, currentLocaleCode, showToastHandler } = useApp();
109
+ const [search, setSearch] = useState('');
110
+ const [page, setPage] = useState(1);
111
+ const [pageSize, setPageSize] = useState(12);
112
+ const [sheetOpen, setSheetOpen] = useState(false);
113
+ const [deleteId, setDeleteId] = useState<number | null>(null);
114
+
115
+ const form = useForm<EntitlementFormValues>({
116
+ resolver: zodResolver(schema),
117
+ defaultValues: {
118
+ contact_id: 0,
119
+ source_type: 'product',
120
+ source_id: 0,
121
+ target_type: 'feature',
122
+ target_id: 0,
123
+ access_type: 'full',
124
+ starts_at: '',
125
+ ends_at: '',
126
+ status: 'active',
127
+ },
128
+ });
129
+
130
+ const { data, refetch } = useQuery<{ items: Entitlement[]; total: number }>({
131
+ queryKey: [
132
+ 'billing-entitlements',
133
+ currentLocaleCode,
134
+ page,
135
+ pageSize,
136
+ search,
137
+ ],
138
+ queryFn: async () => {
139
+ const response = await request<ListResponse<Entitlement>>({
140
+ url: '/billing/entitlements',
141
+ method: 'GET',
142
+ params: { page, pageSize, search: search.trim() || undefined },
143
+ });
144
+
145
+ const payload = response.data as
146
+ | ListResponse<Entitlement>
147
+ | Entitlement[];
148
+ if (Array.isArray(payload)) {
149
+ return { items: payload, total: payload.length };
150
+ }
151
+
152
+ return {
153
+ items: payload.data ?? [],
154
+ total: payload.total ?? payload.data?.length ?? 0,
155
+ };
156
+ },
157
+ placeholderData: (old) => old,
158
+ });
159
+
160
+ const items = data?.items ?? [];
161
+ const totalItems = data?.total ?? 0;
162
+
163
+ const onSubmit = async (values: EntitlementFormValues) => {
164
+ try {
165
+ await request({
166
+ url: '/billing/entitlements',
167
+ method: 'POST',
168
+ data: {
169
+ ...values,
170
+ starts_at: values.starts_at || undefined,
171
+ ends_at: values.ends_at || undefined,
172
+ },
173
+ });
174
+
175
+ showToastHandler?.('success', t('messages.createSuccess'));
176
+ setSheetOpen(false);
177
+ form.reset();
178
+ await refetch();
179
+ } catch {
180
+ showToastHandler?.('error', t('messages.createError'));
181
+ }
182
+ };
183
+
184
+ const onDelete = async () => {
185
+ if (!deleteId) return;
186
+
187
+ try {
188
+ await request({
189
+ url: `/billing/entitlements/${deleteId}`,
190
+ method: 'DELETE',
191
+ });
192
+
193
+ showToastHandler?.('success', t('messages.deleteSuccess'));
194
+ setDeleteId(null);
195
+ await refetch();
196
+ } catch {
197
+ showToastHandler?.('error', t('messages.deleteError'));
198
+ }
199
+ };
200
+
201
+ const formatDate = (value: string | null) => {
202
+ if (!value) return '-';
203
+ return new Intl.DateTimeFormat('pt-BR', { dateStyle: 'short' }).format(
204
+ new Date(value)
205
+ );
206
+ };
207
+
208
+ return (
209
+ <Page>
210
+ <PageHeader
211
+ title={t('title')}
212
+ description={t('description')}
213
+ breadcrumbs={[
214
+ { label: t('breadcrumbs.home'), href: '/' },
215
+ { label: t('breadcrumbs.billing'), href: '/billing' },
216
+ { label: t('breadcrumbs.entitlements') },
217
+ ]}
218
+ actions={[
219
+ {
220
+ label: t('actions.create'),
221
+ onClick: () => setSheetOpen(true),
222
+ icon: <Plus className="size-4" />,
223
+ },
224
+ ]}
225
+ />
226
+
227
+ <SearchBar
228
+ searchQuery={search}
229
+ onSearchChange={(value) => {
230
+ setSearch(value);
231
+ setPage(1);
232
+ }}
233
+ onSearch={() => {
234
+ setPage(1);
235
+ void refetch();
236
+ }}
237
+ placeholder={t('filters.searchPlaceholder')}
238
+ />
239
+
240
+ <div className="overflow-x-auto rounded-md border">
241
+ <Table>
242
+ <TableHeader>
243
+ <TableRow>
244
+ <TableHead>{t('table.columns.contactId')}</TableHead>
245
+ <TableHead>{t('table.columns.sourceType')}</TableHead>
246
+ <TableHead>{t('table.columns.targetType')}</TableHead>
247
+ <TableHead>{t('table.columns.accessType')}</TableHead>
248
+ <TableHead>{t('table.columns.status')}</TableHead>
249
+ <TableHead>{t('table.columns.startsAt')}</TableHead>
250
+ <TableHead>{t('table.columns.endsAt')}</TableHead>
251
+ <TableHead className="w-[100px] text-right">
252
+ {t('table.columns.actions')}
253
+ </TableHead>
254
+ </TableRow>
255
+ </TableHeader>
256
+ <TableBody>
257
+ {items.length === 0 && (
258
+ <TableRow>
259
+ <TableCell
260
+ colSpan={8}
261
+ className="text-center text-muted-foreground"
262
+ >
263
+ {t('table.empty')}
264
+ </TableCell>
265
+ </TableRow>
266
+ )}
267
+ {items.map((item) => (
268
+ <TableRow key={item.id}>
269
+ <TableCell>{item.contact_id}</TableCell>
270
+ <TableCell>{item.source_type}</TableCell>
271
+ <TableCell>{item.target_type}</TableCell>
272
+ <TableCell>
273
+ <Badge className={badgeClass(item.access_type)}>
274
+ {item.access_type}
275
+ </Badge>
276
+ </TableCell>
277
+ <TableCell>
278
+ <Badge className={badgeClass(item.status)}>
279
+ {item.status}
280
+ </Badge>
281
+ </TableCell>
282
+ <TableCell>{formatDate(item.starts_at)}</TableCell>
283
+ <TableCell>{formatDate(item.ends_at)}</TableCell>
284
+ <TableCell className="text-right">
285
+ <Button
286
+ variant="destructive"
287
+ size="icon"
288
+ onClick={() => setDeleteId(item.id)}
289
+ >
290
+ <Trash2 className="size-4" />
291
+ </Button>
292
+ </TableCell>
293
+ </TableRow>
294
+ ))}
295
+ </TableBody>
296
+ </Table>
297
+ </div>
298
+
299
+ <PaginationFooter
300
+ currentPage={page}
301
+ pageSize={pageSize}
302
+ totalItems={totalItems}
303
+ onPageChange={setPage}
304
+ onPageSizeChange={(nextSize) => {
305
+ setPageSize(nextSize);
306
+ setPage(1);
307
+ }}
308
+ />
309
+
310
+ <Sheet open={sheetOpen} onOpenChange={setSheetOpen}>
311
+ <SheetContent className="w-full sm:max-w-xl">
312
+ <SheetHeader>
313
+ <SheetTitle>{t('sheet.createTitle')}</SheetTitle>
314
+ <SheetDescription>{t('sheet.description')}</SheetDescription>
315
+ </SheetHeader>
316
+
317
+ <Form {...form}>
318
+ <form
319
+ className="space-y-4 p-4"
320
+ onSubmit={form.handleSubmit(onSubmit)}
321
+ >
322
+ <div className="grid grid-cols-2 gap-4">
323
+ <FormField
324
+ control={form.control}
325
+ name="contact_id"
326
+ render={({ field }) => (
327
+ <FormItem>
328
+ <FormLabel>{t('form.contactId')}</FormLabel>
329
+ <FormControl>
330
+ <Input
331
+ type="number"
332
+ min={1}
333
+ value={field.value}
334
+ onChange={(event) =>
335
+ field.onChange(Number(event.target.value))
336
+ }
337
+ />
338
+ </FormControl>
339
+ <FormMessage />
340
+ </FormItem>
341
+ )}
342
+ />
343
+
344
+ <FormField
345
+ control={form.control}
346
+ name="access_type"
347
+ render={({ field }) => (
348
+ <FormItem>
349
+ <FormLabel>{t('form.accessType')}</FormLabel>
350
+ <FormControl>
351
+ <Input {...field} />
352
+ </FormControl>
353
+ <FormMessage />
354
+ </FormItem>
355
+ )}
356
+ />
357
+ </div>
358
+
359
+ <div className="grid grid-cols-2 gap-4">
360
+ <FormField
361
+ control={form.control}
362
+ name="source_type"
363
+ render={({ field }) => (
364
+ <FormItem>
365
+ <FormLabel>{t('form.sourceType')}</FormLabel>
366
+ <FormControl>
367
+ <Input {...field} />
368
+ </FormControl>
369
+ <FormMessage />
370
+ </FormItem>
371
+ )}
372
+ />
373
+ <FormField
374
+ control={form.control}
375
+ name="source_id"
376
+ render={({ field }) => (
377
+ <FormItem>
378
+ <FormLabel>{t('form.sourceId')}</FormLabel>
379
+ <FormControl>
380
+ <Input
381
+ type="number"
382
+ min={1}
383
+ value={field.value}
384
+ onChange={(event) =>
385
+ field.onChange(Number(event.target.value))
386
+ }
387
+ />
388
+ </FormControl>
389
+ <FormMessage />
390
+ </FormItem>
391
+ )}
392
+ />
393
+ </div>
394
+
395
+ <div className="grid grid-cols-2 gap-4">
396
+ <FormField
397
+ control={form.control}
398
+ name="target_type"
399
+ render={({ field }) => (
400
+ <FormItem>
401
+ <FormLabel>{t('form.targetType')}</FormLabel>
402
+ <FormControl>
403
+ <Input {...field} />
404
+ </FormControl>
405
+ <FormMessage />
406
+ </FormItem>
407
+ )}
408
+ />
409
+ <FormField
410
+ control={form.control}
411
+ name="target_id"
412
+ render={({ field }) => (
413
+ <FormItem>
414
+ <FormLabel>{t('form.targetId')}</FormLabel>
415
+ <FormControl>
416
+ <Input
417
+ type="number"
418
+ min={1}
419
+ value={field.value}
420
+ onChange={(event) =>
421
+ field.onChange(Number(event.target.value))
422
+ }
423
+ />
424
+ </FormControl>
425
+ <FormMessage />
426
+ </FormItem>
427
+ )}
428
+ />
429
+ </div>
430
+
431
+ <div className="grid grid-cols-3 gap-4">
432
+ <FormField
433
+ control={form.control}
434
+ name="starts_at"
435
+ render={({ field }) => (
436
+ <FormItem>
437
+ <FormLabel>{t('form.startsAt')}</FormLabel>
438
+ <FormControl>
439
+ <Input
440
+ type="date"
441
+ {...field}
442
+ value={field.value ?? ''}
443
+ />
444
+ </FormControl>
445
+ <FormMessage />
446
+ </FormItem>
447
+ )}
448
+ />
449
+ <FormField
450
+ control={form.control}
451
+ name="ends_at"
452
+ render={({ field }) => (
453
+ <FormItem>
454
+ <FormLabel>{t('form.endsAt')}</FormLabel>
455
+ <FormControl>
456
+ <Input
457
+ type="date"
458
+ {...field}
459
+ value={field.value ?? ''}
460
+ />
461
+ </FormControl>
462
+ <FormMessage />
463
+ </FormItem>
464
+ )}
465
+ />
466
+ <FormField
467
+ control={form.control}
468
+ name="status"
469
+ render={({ field }) => (
470
+ <FormItem>
471
+ <FormLabel>{t('form.status')}</FormLabel>
472
+ <Select
473
+ value={field.value}
474
+ onValueChange={field.onChange}
475
+ >
476
+ <FormControl>
477
+ <SelectTrigger className="w-full">
478
+ <SelectValue />
479
+ </SelectTrigger>
480
+ </FormControl>
481
+ <SelectContent>
482
+ <SelectItem value="active">active</SelectItem>
483
+ <SelectItem value="revoked">revoked</SelectItem>
484
+ <SelectItem value="expired">expired</SelectItem>
485
+ </SelectContent>
486
+ </Select>
487
+ <FormMessage />
488
+ </FormItem>
489
+ )}
490
+ />
491
+ </div>
492
+
493
+ <Button
494
+ type="submit"
495
+ className="w-full"
496
+ disabled={form.formState.isSubmitting}
497
+ >
498
+ {t('actions.save')}
499
+ </Button>
500
+ </form>
501
+ </Form>
502
+ </SheetContent>
503
+ </Sheet>
504
+
505
+ <AlertDialog
506
+ open={deleteId !== null}
507
+ onOpenChange={(open) => !open && setDeleteId(null)}
508
+ >
509
+ <AlertDialogContent>
510
+ <AlertDialogHeader>
511
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
512
+ <AlertDialogDescription>
513
+ {t('deleteDialog.description')}
514
+ </AlertDialogDescription>
515
+ </AlertDialogHeader>
516
+ <AlertDialogFooter>
517
+ <AlertDialogCancel>{t('deleteDialog.cancel')}</AlertDialogCancel>
518
+ <AlertDialogAction onClick={onDelete}>
519
+ {t('deleteDialog.confirm')}
520
+ </AlertDialogAction>
521
+ </AlertDialogFooter>
522
+ </AlertDialogContent>
523
+ </AlertDialog>
524
+ </Page>
525
+ );
526
+ }