@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,562 @@
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 { Textarea } from '@/components/ui/textarea';
53
+ import { useApp, useQuery } from '@hed-hog/next-app-provider';
54
+ import { zodResolver } from '@hookform/resolvers/zod';
55
+ import { Pencil, Plus, Trash2 } from 'lucide-react';
56
+ import { useTranslations } from 'next-intl';
57
+ import { useEffect, useState } from 'react';
58
+ import { useForm } from 'react-hook-form';
59
+ import { z } from 'zod';
60
+
61
+ type Contract = {
62
+ id: number;
63
+ contact_id: number;
64
+ name: string;
65
+ description?: string | null;
66
+ starts_at: string;
67
+ ends_at?: string | null;
68
+ total_seats?: number | null;
69
+ status: string;
70
+ signed_at?: string | null;
71
+ };
72
+
73
+ type ListResponse<T> = {
74
+ data?: T[];
75
+ total?: number;
76
+ };
77
+
78
+ const schema = z.object({
79
+ contact_id: z.coerce.number().min(1),
80
+ name: z.string().trim().min(1),
81
+ description: z.string().optional(),
82
+ starts_at: z.string().min(1),
83
+ ends_at: z.string().optional(),
84
+ total_seats: z.coerce.number().min(0).optional(),
85
+ status: z.enum(['draft', 'active', 'expired', 'terminated']),
86
+ signed_at: z.string().optional(),
87
+ });
88
+
89
+ type ContractFormValues = z.infer<typeof schema>;
90
+
91
+ const badgeClass = (value: string) => {
92
+ if (['active', 'paid', 'authorized', 'approved'].includes(value)) {
93
+ return 'bg-green-100 text-green-800';
94
+ }
95
+ if (['pending', 'open', 'draft'].includes(value)) {
96
+ return 'bg-yellow-100 text-yellow-800';
97
+ }
98
+ if (['failed', 'canceled', 'void', 'terminated', 'expired'].includes(value)) {
99
+ return 'bg-red-100 text-red-800';
100
+ }
101
+ if (['paused', 'inactive'].includes(value)) {
102
+ return 'bg-gray-100 text-gray-800';
103
+ }
104
+ return 'bg-slate-100 text-slate-800';
105
+ };
106
+
107
+ export default function BillingContractsPage() {
108
+ const t = useTranslations('BillingContractsPage');
109
+ const { request, currentLocaleCode, showToastHandler } = useApp();
110
+ const [search, setSearch] = useState('');
111
+ const [page, setPage] = useState(1);
112
+ const [pageSize, setPageSize] = useState(12);
113
+ const [sheetOpen, setSheetOpen] = useState(false);
114
+ const [editingItem, setEditingItem] = useState<Contract | null>(null);
115
+ const [deleteId, setDeleteId] = useState<number | null>(null);
116
+
117
+ const form = useForm<ContractFormValues>({
118
+ resolver: zodResolver(schema),
119
+ defaultValues: {
120
+ contact_id: 0,
121
+ name: '',
122
+ description: '',
123
+ starts_at: '',
124
+ ends_at: '',
125
+ total_seats: 1,
126
+ status: 'draft',
127
+ signed_at: '',
128
+ },
129
+ });
130
+
131
+ useEffect(() => {
132
+ if (!sheetOpen) {
133
+ return;
134
+ }
135
+
136
+ if (editingItem) {
137
+ form.reset({
138
+ contact_id: editingItem.contact_id,
139
+ name: editingItem.name,
140
+ description: editingItem.description ?? '',
141
+ starts_at: editingItem.starts_at.slice(0, 10),
142
+ ends_at: editingItem.ends_at?.slice(0, 10) ?? '',
143
+ total_seats: editingItem.total_seats ?? 1,
144
+ status: (editingItem.status as ContractFormValues['status']) ?? 'draft',
145
+ signed_at: editingItem.signed_at?.slice(0, 10) ?? '',
146
+ });
147
+ return;
148
+ }
149
+
150
+ form.reset({
151
+ contact_id: 0,
152
+ name: '',
153
+ description: '',
154
+ starts_at: '',
155
+ ends_at: '',
156
+ total_seats: 1,
157
+ status: 'draft',
158
+ signed_at: '',
159
+ });
160
+ }, [editingItem, form, sheetOpen]);
161
+
162
+ const { data, refetch } = useQuery<{ items: Contract[]; total: number }>({
163
+ queryKey: ['billing-contracts', currentLocaleCode, page, pageSize, search],
164
+ queryFn: async () => {
165
+ const response = await request<ListResponse<Contract>>({
166
+ url: '/billing/contracts',
167
+ method: 'GET',
168
+ params: { page, pageSize, search: search.trim() || undefined },
169
+ });
170
+
171
+ const payload = response.data as ListResponse<Contract> | Contract[];
172
+ if (Array.isArray(payload)) {
173
+ return { items: payload, total: payload.length };
174
+ }
175
+
176
+ return {
177
+ items: payload.data ?? [],
178
+ total: payload.total ?? payload.data?.length ?? 0,
179
+ };
180
+ },
181
+ placeholderData: (old) => old,
182
+ });
183
+
184
+ const items = data?.items ?? [];
185
+ const totalItems = data?.total ?? 0;
186
+
187
+ const formatDate = (value?: string | null) => {
188
+ if (!value) return '-';
189
+ return new Intl.DateTimeFormat('pt-BR', { dateStyle: 'short' }).format(
190
+ new Date(value)
191
+ );
192
+ };
193
+
194
+ const onSubmit = async (values: ContractFormValues) => {
195
+ try {
196
+ const payload = {
197
+ contact_id: values.contact_id,
198
+ name: values.name,
199
+ description: values.description?.trim() || undefined,
200
+ starts_at: values.starts_at,
201
+ ends_at: values.ends_at || undefined,
202
+ total_seats: values.total_seats,
203
+ status: values.status,
204
+ signed_at: values.signed_at || undefined,
205
+ };
206
+
207
+ if (editingItem) {
208
+ await request({
209
+ url: `/billing/contracts/${editingItem.id}`,
210
+ method: 'PATCH',
211
+ data: payload,
212
+ });
213
+ } else {
214
+ await request({
215
+ url: '/billing/contracts',
216
+ method: 'POST',
217
+ data: payload,
218
+ });
219
+ }
220
+
221
+ showToastHandler?.('success', t('messages.saveSuccess'));
222
+ setSheetOpen(false);
223
+ setEditingItem(null);
224
+ form.reset();
225
+ await refetch();
226
+ } catch {
227
+ showToastHandler?.('error', t('messages.saveError'));
228
+ }
229
+ };
230
+
231
+ const onDelete = async () => {
232
+ if (!deleteId) return;
233
+
234
+ try {
235
+ await request({
236
+ url: `/billing/contracts/${deleteId}`,
237
+ method: 'DELETE',
238
+ });
239
+ showToastHandler?.('success', t('messages.deleteSuccess'));
240
+ setDeleteId(null);
241
+ await refetch();
242
+ } catch {
243
+ showToastHandler?.('error', t('messages.deleteError'));
244
+ }
245
+ };
246
+
247
+ return (
248
+ <Page>
249
+ <PageHeader
250
+ title={t('title')}
251
+ description={t('description')}
252
+ breadcrumbs={[
253
+ { label: t('breadcrumbs.home'), href: '/' },
254
+ { label: t('breadcrumbs.billing'), href: '/billing' },
255
+ { label: t('breadcrumbs.contracts') },
256
+ ]}
257
+ actions={[
258
+ {
259
+ label: t('actions.create'),
260
+ onClick: () => {
261
+ setEditingItem(null);
262
+ setSheetOpen(true);
263
+ },
264
+ icon: <Plus className="size-4" />,
265
+ },
266
+ ]}
267
+ />
268
+
269
+ <SearchBar
270
+ searchQuery={search}
271
+ onSearchChange={(value) => {
272
+ setSearch(value);
273
+ setPage(1);
274
+ }}
275
+ onSearch={() => {
276
+ setPage(1);
277
+ void refetch();
278
+ }}
279
+ placeholder={t('filters.searchPlaceholder')}
280
+ />
281
+
282
+ <div className="overflow-x-auto rounded-md border">
283
+ <Table>
284
+ <TableHeader>
285
+ <TableRow>
286
+ <TableHead>{t('table.columns.name')}</TableHead>
287
+ <TableHead>{t('table.columns.contactId')}</TableHead>
288
+ <TableHead>{t('table.columns.status')}</TableHead>
289
+ <TableHead>{t('table.columns.totalSeats')}</TableHead>
290
+ <TableHead>{t('table.columns.startsAt')}</TableHead>
291
+ <TableHead>{t('table.columns.endsAt')}</TableHead>
292
+ <TableHead className="w-[120px] text-right">
293
+ {t('table.columns.actions')}
294
+ </TableHead>
295
+ </TableRow>
296
+ </TableHeader>
297
+ <TableBody>
298
+ {items.length === 0 && (
299
+ <TableRow>
300
+ <TableCell
301
+ colSpan={7}
302
+ className="text-center text-muted-foreground"
303
+ >
304
+ {t('table.empty')}
305
+ </TableCell>
306
+ </TableRow>
307
+ )}
308
+ {items.map((item) => (
309
+ <TableRow key={item.id}>
310
+ <TableCell className="font-medium">{item.name}</TableCell>
311
+ <TableCell>{item.contact_id}</TableCell>
312
+ <TableCell>
313
+ <Badge className={badgeClass(item.status)}>
314
+ {item.status}
315
+ </Badge>
316
+ </TableCell>
317
+ <TableCell>{item.total_seats ?? '-'}</TableCell>
318
+ <TableCell>{formatDate(item.starts_at)}</TableCell>
319
+ <TableCell>{formatDate(item.ends_at)}</TableCell>
320
+ <TableCell>
321
+ <div className="flex justify-end gap-2">
322
+ <Button
323
+ variant="outline"
324
+ size="icon"
325
+ onClick={() => {
326
+ setEditingItem(item);
327
+ setSheetOpen(true);
328
+ }}
329
+ >
330
+ <Pencil className="size-4" />
331
+ </Button>
332
+ <Button
333
+ variant="destructive"
334
+ size="icon"
335
+ onClick={() => setDeleteId(item.id)}
336
+ >
337
+ <Trash2 className="size-4" />
338
+ </Button>
339
+ </div>
340
+ </TableCell>
341
+ </TableRow>
342
+ ))}
343
+ </TableBody>
344
+ </Table>
345
+ </div>
346
+
347
+ <PaginationFooter
348
+ currentPage={page}
349
+ pageSize={pageSize}
350
+ totalItems={totalItems}
351
+ onPageChange={setPage}
352
+ onPageSizeChange={(nextSize) => {
353
+ setPageSize(nextSize);
354
+ setPage(1);
355
+ }}
356
+ />
357
+
358
+ <Sheet
359
+ open={sheetOpen}
360
+ onOpenChange={(open) => {
361
+ setSheetOpen(open);
362
+ if (!open) {
363
+ setEditingItem(null);
364
+ }
365
+ }}
366
+ >
367
+ <SheetContent className="w-full sm:max-w-xl">
368
+ <SheetHeader>
369
+ <SheetTitle>
370
+ {editingItem ? t('sheet.editTitle') : t('sheet.createTitle')}
371
+ </SheetTitle>
372
+ <SheetDescription>{t('sheet.description')}</SheetDescription>
373
+ </SheetHeader>
374
+
375
+ <Form {...form}>
376
+ <form
377
+ className="space-y-4 p-4"
378
+ onSubmit={form.handleSubmit(onSubmit)}
379
+ >
380
+ <div className="grid grid-cols-2 gap-4">
381
+ <FormField
382
+ control={form.control}
383
+ name="contact_id"
384
+ render={({ field }) => (
385
+ <FormItem>
386
+ <FormLabel>{t('form.contactId')}</FormLabel>
387
+ <FormControl>
388
+ <Input
389
+ type="number"
390
+ min={1}
391
+ value={field.value}
392
+ onChange={(event) =>
393
+ field.onChange(Number(event.target.value))
394
+ }
395
+ />
396
+ </FormControl>
397
+ <FormMessage />
398
+ </FormItem>
399
+ )}
400
+ />
401
+ <FormField
402
+ control={form.control}
403
+ name="name"
404
+ render={({ field }) => (
405
+ <FormItem>
406
+ <FormLabel>{t('form.name')}</FormLabel>
407
+ <FormControl>
408
+ <Input {...field} />
409
+ </FormControl>
410
+ <FormMessage />
411
+ </FormItem>
412
+ )}
413
+ />
414
+ </div>
415
+
416
+ <FormField
417
+ control={form.control}
418
+ name="description"
419
+ render={({ field }) => (
420
+ <FormItem>
421
+ <FormLabel>{t('form.description')}</FormLabel>
422
+ <FormControl>
423
+ <Textarea {...field} value={field.value ?? ''} />
424
+ </FormControl>
425
+ <FormMessage />
426
+ </FormItem>
427
+ )}
428
+ />
429
+
430
+ <div className="grid grid-cols-3 gap-4">
431
+ <FormField
432
+ control={form.control}
433
+ name="starts_at"
434
+ render={({ field }) => (
435
+ <FormItem>
436
+ <FormLabel>{t('form.startsAt')}</FormLabel>
437
+ <FormControl>
438
+ <Input type="date" {...field} />
439
+ </FormControl>
440
+ <FormMessage />
441
+ </FormItem>
442
+ )}
443
+ />
444
+ <FormField
445
+ control={form.control}
446
+ name="ends_at"
447
+ render={({ field }) => (
448
+ <FormItem>
449
+ <FormLabel>{t('form.endsAt')}</FormLabel>
450
+ <FormControl>
451
+ <Input
452
+ type="date"
453
+ {...field}
454
+ value={field.value ?? ''}
455
+ />
456
+ </FormControl>
457
+ <FormMessage />
458
+ </FormItem>
459
+ )}
460
+ />
461
+ <FormField
462
+ control={form.control}
463
+ name="signed_at"
464
+ render={({ field }) => (
465
+ <FormItem>
466
+ <FormLabel>{t('form.signedAt')}</FormLabel>
467
+ <FormControl>
468
+ <Input
469
+ type="date"
470
+ {...field}
471
+ value={field.value ?? ''}
472
+ />
473
+ </FormControl>
474
+ <FormMessage />
475
+ </FormItem>
476
+ )}
477
+ />
478
+ </div>
479
+
480
+ <div className="grid grid-cols-2 gap-4">
481
+ <FormField
482
+ control={form.control}
483
+ name="total_seats"
484
+ render={({ field }) => (
485
+ <FormItem>
486
+ <FormLabel>{t('form.totalSeats')}</FormLabel>
487
+ <FormControl>
488
+ <Input
489
+ type="number"
490
+ min={0}
491
+ value={field.value ?? 0}
492
+ onChange={(event) =>
493
+ field.onChange(Number(event.target.value))
494
+ }
495
+ />
496
+ </FormControl>
497
+ <FormMessage />
498
+ </FormItem>
499
+ )}
500
+ />
501
+ <FormField
502
+ control={form.control}
503
+ name="status"
504
+ render={({ field }) => (
505
+ <FormItem>
506
+ <FormLabel>{t('form.status')}</FormLabel>
507
+ <Select
508
+ value={field.value}
509
+ onValueChange={field.onChange}
510
+ >
511
+ <FormControl>
512
+ <SelectTrigger className="w-full">
513
+ <SelectValue />
514
+ </SelectTrigger>
515
+ </FormControl>
516
+ <SelectContent>
517
+ <SelectItem value="draft">draft</SelectItem>
518
+ <SelectItem value="active">active</SelectItem>
519
+ <SelectItem value="expired">expired</SelectItem>
520
+ <SelectItem value="terminated">terminated</SelectItem>
521
+ </SelectContent>
522
+ </Select>
523
+ <FormMessage />
524
+ </FormItem>
525
+ )}
526
+ />
527
+ </div>
528
+
529
+ <Button
530
+ type="submit"
531
+ className="w-full"
532
+ disabled={form.formState.isSubmitting}
533
+ >
534
+ {t('actions.save')}
535
+ </Button>
536
+ </form>
537
+ </Form>
538
+ </SheetContent>
539
+ </Sheet>
540
+
541
+ <AlertDialog
542
+ open={deleteId !== null}
543
+ onOpenChange={(open) => !open && setDeleteId(null)}
544
+ >
545
+ <AlertDialogContent>
546
+ <AlertDialogHeader>
547
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
548
+ <AlertDialogDescription>
549
+ {t('deleteDialog.description')}
550
+ </AlertDialogDescription>
551
+ </AlertDialogHeader>
552
+ <AlertDialogFooter>
553
+ <AlertDialogCancel>{t('deleteDialog.cancel')}</AlertDialogCancel>
554
+ <AlertDialogAction onClick={onDelete}>
555
+ {t('deleteDialog.confirm')}
556
+ </AlertDialogAction>
557
+ </AlertDialogFooter>
558
+ </AlertDialogContent>
559
+ </AlertDialog>
560
+ </Page>
561
+ );
562
+ }