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