@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,669 @@
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 { InputMoney } from '@/components/ui/input-money';
31
+ import {
32
+ Select,
33
+ SelectContent,
34
+ SelectItem,
35
+ SelectTrigger,
36
+ SelectValue,
37
+ } from '@/components/ui/select';
38
+ import {
39
+ Sheet,
40
+ SheetContent,
41
+ SheetDescription,
42
+ SheetHeader,
43
+ SheetTitle,
44
+ } from '@/components/ui/sheet';
45
+ import {
46
+ Table,
47
+ TableBody,
48
+ TableCell,
49
+ TableHead,
50
+ TableHeader,
51
+ TableRow,
52
+ } from '@/components/ui/table';
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 Coupon = {
62
+ id: number;
63
+ code: string;
64
+ name: string;
65
+ discount_type: string;
66
+ discount_value: number;
67
+ currency: string;
68
+ uses_count: number;
69
+ max_uses: number | null;
70
+ status: string;
71
+ starts_at: string | null;
72
+ ends_at: string | null;
73
+ };
74
+
75
+ type ListResponse<T> = {
76
+ data?: T[];
77
+ total?: number;
78
+ };
79
+
80
+ const schema = z.object({
81
+ code: z.string().trim().min(1),
82
+ name: z.string().trim().min(1),
83
+ discount_type: z.enum(['percent', 'fixed']),
84
+ discount_value: z.coerce.number().min(0),
85
+ currency: z.string().default('BRL'),
86
+ max_uses: z.coerce.number().min(1).optional(),
87
+ starts_at: z.string().optional(),
88
+ ends_at: z.string().optional(),
89
+ status: z.enum(['active', 'inactive', 'draft']),
90
+ });
91
+
92
+ type CouponFormValues = z.infer<typeof schema>;
93
+
94
+ const formatCurrency = (cents: number, currency = 'BRL') =>
95
+ new Intl.NumberFormat('pt-BR', { style: 'currency', currency }).format(
96
+ cents / 100
97
+ );
98
+
99
+ const badgeClass = (value: string) => {
100
+ if (['active', 'paid', 'authorized', 'approved'].includes(value)) {
101
+ return 'bg-green-100 text-green-800';
102
+ }
103
+ if (['pending', 'open', 'draft'].includes(value)) {
104
+ return 'bg-yellow-100 text-yellow-800';
105
+ }
106
+ if (['failed', 'canceled', 'void'].includes(value)) {
107
+ return 'bg-red-100 text-red-800';
108
+ }
109
+ if (['paused', 'inactive'].includes(value)) {
110
+ return 'bg-gray-100 text-gray-800';
111
+ }
112
+ return 'bg-slate-100 text-slate-800';
113
+ };
114
+
115
+ const mapDiscountTypeToApi = (value: CouponFormValues['discount_type']) => {
116
+ if (value === 'percent') {
117
+ return 'percentage';
118
+ }
119
+
120
+ return 'fixed';
121
+ };
122
+
123
+ const mapDiscountTypeFromApi = (
124
+ value: string
125
+ ): CouponFormValues['discount_type'] => {
126
+ if (value === 'percentage') {
127
+ return 'percent';
128
+ }
129
+
130
+ return 'fixed';
131
+ };
132
+
133
+ export default function BillingCouponsPage() {
134
+ const t = useTranslations('BillingCouponsPage');
135
+ const { request, currentLocaleCode, showToastHandler } = useApp();
136
+ const [search, setSearch] = useState('');
137
+ const [statusFilter, setStatusFilter] = useState('all');
138
+ const [page, setPage] = useState(1);
139
+ const [pageSize, setPageSize] = useState(12);
140
+ const [sheetOpen, setSheetOpen] = useState(false);
141
+ const [editingCoupon, setEditingCoupon] = useState<Coupon | null>(null);
142
+ const [deleteId, setDeleteId] = useState<number | null>(null);
143
+
144
+ const form = useForm<CouponFormValues>({
145
+ resolver: zodResolver(schema),
146
+ defaultValues: {
147
+ code: '',
148
+ name: '',
149
+ discount_type: 'percent',
150
+ discount_value: 0,
151
+ currency: 'BRL',
152
+ max_uses: 1,
153
+ starts_at: '',
154
+ ends_at: '',
155
+ status: 'active',
156
+ },
157
+ });
158
+
159
+ useEffect(() => {
160
+ if (!sheetOpen) {
161
+ return;
162
+ }
163
+
164
+ if (editingCoupon) {
165
+ form.reset({
166
+ code: editingCoupon.code,
167
+ name: editingCoupon.name,
168
+ discount_type: mapDiscountTypeFromApi(editingCoupon.discount_type),
169
+ discount_value: editingCoupon.discount_value,
170
+ currency: editingCoupon.currency,
171
+ max_uses: editingCoupon.max_uses ?? 1,
172
+ starts_at: editingCoupon.starts_at?.slice(0, 10) ?? '',
173
+ ends_at: editingCoupon.ends_at?.slice(0, 10) ?? '',
174
+ status:
175
+ (editingCoupon.status as CouponFormValues['status']) ?? 'active',
176
+ });
177
+ return;
178
+ }
179
+
180
+ form.reset({
181
+ code: '',
182
+ name: '',
183
+ discount_type: 'percent',
184
+ discount_value: 0,
185
+ currency: 'BRL',
186
+ max_uses: 1,
187
+ starts_at: '',
188
+ ends_at: '',
189
+ status: 'active',
190
+ });
191
+ }, [editingCoupon, form, sheetOpen]);
192
+
193
+ const { data, refetch } = useQuery<{ items: Coupon[]; total: number }>({
194
+ queryKey: [
195
+ 'billing-coupons',
196
+ currentLocaleCode,
197
+ search,
198
+ statusFilter,
199
+ page,
200
+ pageSize,
201
+ ],
202
+ queryFn: async () => {
203
+ const response = await request<ListResponse<Coupon>>({
204
+ url: '/billing/coupons',
205
+ method: 'GET',
206
+ params: {
207
+ page,
208
+ pageSize,
209
+ search: search.trim() || undefined,
210
+ status: statusFilter !== 'all' ? statusFilter : undefined,
211
+ },
212
+ });
213
+
214
+ const payload = response.data as ListResponse<Coupon> | Coupon[];
215
+ if (Array.isArray(payload)) {
216
+ return { items: payload, total: payload.length };
217
+ }
218
+
219
+ return {
220
+ items: payload.data ?? [],
221
+ total: payload.total ?? payload.data?.length ?? 0,
222
+ };
223
+ },
224
+ placeholderData: (old) => old,
225
+ });
226
+
227
+ const items = data?.items ?? [];
228
+ const totalItems = data?.total ?? 0;
229
+
230
+ const onSubmit = async (values: CouponFormValues) => {
231
+ try {
232
+ const payload = {
233
+ code: values.code,
234
+ name: values.name,
235
+ discount_type: mapDiscountTypeToApi(values.discount_type),
236
+ discount_value: values.discount_value,
237
+ currency: values.currency,
238
+ max_uses: values.max_uses,
239
+ starts_at: values.starts_at || undefined,
240
+ ends_at: values.ends_at || undefined,
241
+ status: values.status,
242
+ };
243
+
244
+ if (editingCoupon) {
245
+ await request({
246
+ url: `/billing/coupons/${editingCoupon.id}`,
247
+ method: 'PATCH',
248
+ data: payload,
249
+ });
250
+ } else {
251
+ await request({
252
+ url: '/billing/coupons',
253
+ method: 'POST',
254
+ data: payload,
255
+ });
256
+ }
257
+
258
+ showToastHandler?.('success', t('messages.saveSuccess'));
259
+ setSheetOpen(false);
260
+ setEditingCoupon(null);
261
+ form.reset();
262
+ await refetch();
263
+ } catch {
264
+ showToastHandler?.('error', t('messages.saveError'));
265
+ }
266
+ };
267
+
268
+ const onDelete = async () => {
269
+ if (!deleteId) {
270
+ return;
271
+ }
272
+
273
+ try {
274
+ await request({
275
+ url: `/billing/coupons/${deleteId}`,
276
+ method: 'DELETE',
277
+ });
278
+ showToastHandler?.('success', t('messages.deleteSuccess'));
279
+ setDeleteId(null);
280
+ await refetch();
281
+ } catch {
282
+ showToastHandler?.('error', t('messages.deleteError'));
283
+ }
284
+ };
285
+
286
+ return (
287
+ <Page>
288
+ <PageHeader
289
+ title={t('title')}
290
+ description={t('description')}
291
+ breadcrumbs={[
292
+ { label: t('breadcrumbs.home'), href: '/' },
293
+ { label: t('breadcrumbs.billing'), href: '/billing' },
294
+ { label: t('breadcrumbs.coupons') },
295
+ ]}
296
+ actions={[
297
+ {
298
+ label: t('actions.create'),
299
+ onClick: () => {
300
+ setEditingCoupon(null);
301
+ setSheetOpen(true);
302
+ },
303
+ icon: <Plus className="size-4" />,
304
+ },
305
+ ]}
306
+ />
307
+
308
+ <SearchBar
309
+ searchQuery={search}
310
+ onSearchChange={(value) => {
311
+ setSearch(value);
312
+ setPage(1);
313
+ }}
314
+ onSearch={() => {
315
+ setPage(1);
316
+ void refetch();
317
+ }}
318
+ placeholder={t('filters.searchPlaceholder')}
319
+ filters={{
320
+ value: statusFilter,
321
+ options: [
322
+ { label: t('filters.all'), value: 'all' },
323
+ { label: 'active', value: 'active' },
324
+ { label: 'inactive', value: 'inactive' },
325
+ { label: 'draft', value: 'draft' },
326
+ ],
327
+ onChange: (value) => {
328
+ setStatusFilter(value);
329
+ setPage(1);
330
+ },
331
+ placeholder: t('filters.status'),
332
+ }}
333
+ />
334
+
335
+ <div className="overflow-x-auto rounded-md border">
336
+ <Table>
337
+ <TableHeader>
338
+ <TableRow>
339
+ <TableHead>{t('table.columns.code')}</TableHead>
340
+ <TableHead>{t('table.columns.name')}</TableHead>
341
+ <TableHead>{t('table.columns.discountType')}</TableHead>
342
+ <TableHead>{t('table.columns.discountValue')}</TableHead>
343
+ <TableHead>{t('table.columns.uses')}</TableHead>
344
+ <TableHead>{t('table.columns.status')}</TableHead>
345
+ <TableHead className="w-[120px] text-right">
346
+ {t('table.columns.actions')}
347
+ </TableHead>
348
+ </TableRow>
349
+ </TableHeader>
350
+ <TableBody>
351
+ {items.length === 0 && (
352
+ <TableRow>
353
+ <TableCell
354
+ colSpan={7}
355
+ className="text-center text-muted-foreground"
356
+ >
357
+ {t('table.empty')}
358
+ </TableCell>
359
+ </TableRow>
360
+ )}
361
+ {items.map((item) => (
362
+ <TableRow key={item.id}>
363
+ <TableCell className="font-medium">{item.code}</TableCell>
364
+ <TableCell>{item.name}</TableCell>
365
+ <TableCell>
366
+ <Badge className={badgeClass(item.discount_type)}>
367
+ {item.discount_type}
368
+ </Badge>
369
+ </TableCell>
370
+ <TableCell>
371
+ {item.discount_type === 'fixed'
372
+ ? formatCurrency(
373
+ Math.round(item.discount_value),
374
+ item.currency
375
+ )
376
+ : `${item.discount_value}%`}
377
+ </TableCell>
378
+ <TableCell>
379
+ {item.uses_count}/{item.max_uses ?? '-'}
380
+ </TableCell>
381
+ <TableCell>
382
+ <Badge className={badgeClass(item.status)}>
383
+ {item.status}
384
+ </Badge>
385
+ </TableCell>
386
+ <TableCell>
387
+ <div className="flex justify-end gap-2">
388
+ <Button
389
+ variant="outline"
390
+ size="icon"
391
+ onClick={() => {
392
+ setEditingCoupon(item);
393
+ setSheetOpen(true);
394
+ }}
395
+ >
396
+ <Pencil className="size-4" />
397
+ </Button>
398
+ <Button
399
+ variant="destructive"
400
+ size="icon"
401
+ onClick={() => setDeleteId(item.id)}
402
+ >
403
+ <Trash2 className="size-4" />
404
+ </Button>
405
+ </div>
406
+ </TableCell>
407
+ </TableRow>
408
+ ))}
409
+ </TableBody>
410
+ </Table>
411
+ </div>
412
+
413
+ <PaginationFooter
414
+ currentPage={page}
415
+ pageSize={pageSize}
416
+ totalItems={totalItems}
417
+ onPageChange={setPage}
418
+ onPageSizeChange={(nextSize) => {
419
+ setPageSize(nextSize);
420
+ setPage(1);
421
+ }}
422
+ />
423
+
424
+ <Sheet
425
+ open={sheetOpen}
426
+ onOpenChange={(open) => {
427
+ setSheetOpen(open);
428
+ if (!open) {
429
+ setEditingCoupon(null);
430
+ }
431
+ }}
432
+ >
433
+ <SheetContent className="w-full sm:max-w-xl">
434
+ <SheetHeader>
435
+ <SheetTitle>
436
+ {editingCoupon ? t('sheet.editTitle') : t('sheet.createTitle')}
437
+ </SheetTitle>
438
+ <SheetDescription>{t('sheet.description')}</SheetDescription>
439
+ </SheetHeader>
440
+ <Form {...form}>
441
+ <form
442
+ className="space-y-4 p-4"
443
+ onSubmit={form.handleSubmit(onSubmit)}
444
+ >
445
+ <div className="grid grid-cols-2 gap-4">
446
+ <FormField
447
+ control={form.control}
448
+ name="code"
449
+ render={({ field }) => (
450
+ <FormItem>
451
+ <FormLabel>{t('form.code')}</FormLabel>
452
+ <FormControl>
453
+ <Input
454
+ placeholder={t('form.codePlaceholder')}
455
+ {...field}
456
+ />
457
+ </FormControl>
458
+ <FormMessage />
459
+ </FormItem>
460
+ )}
461
+ />
462
+ <FormField
463
+ control={form.control}
464
+ name="name"
465
+ render={({ field }) => (
466
+ <FormItem>
467
+ <FormLabel>{t('form.name')}</FormLabel>
468
+ <FormControl>
469
+ <Input
470
+ placeholder={t('form.namePlaceholder')}
471
+ {...field}
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="discount_type"
484
+ render={({ field }) => (
485
+ <FormItem>
486
+ <FormLabel>{t('form.discountType')}</FormLabel>
487
+ <Select
488
+ value={field.value}
489
+ onValueChange={field.onChange}
490
+ >
491
+ <FormControl>
492
+ <SelectTrigger className="w-full">
493
+ <SelectValue />
494
+ </SelectTrigger>
495
+ </FormControl>
496
+ <SelectContent>
497
+ <SelectItem value="percent">percent</SelectItem>
498
+ <SelectItem value="fixed">fixed</SelectItem>
499
+ </SelectContent>
500
+ </Select>
501
+ <FormMessage />
502
+ </FormItem>
503
+ )}
504
+ />
505
+
506
+ <FormField
507
+ control={form.control}
508
+ name="currency"
509
+ render={({ field }) => (
510
+ <FormItem>
511
+ <FormLabel>{t('form.currency')}</FormLabel>
512
+ <FormControl>
513
+ <Input {...field} value={field.value ?? 'BRL'} />
514
+ </FormControl>
515
+ <FormMessage />
516
+ </FormItem>
517
+ )}
518
+ />
519
+ </div>
520
+
521
+ <FormField
522
+ control={form.control}
523
+ name="discount_value"
524
+ render={({ field }) => (
525
+ <FormItem>
526
+ <FormLabel>{t('form.discountValue')}</FormLabel>
527
+ <FormControl>
528
+ {form.watch('discount_type') === 'fixed' ? (
529
+ <InputMoney
530
+ value={field.value / 100}
531
+ onValueChange={(value) =>
532
+ field.onChange(Math.round((value ?? 0) * 100))
533
+ }
534
+ />
535
+ ) : (
536
+ <Input
537
+ type="number"
538
+ min={0}
539
+ max={100}
540
+ step="0.01"
541
+ {...field}
542
+ onChange={(event) =>
543
+ field.onChange(Number(event.target.value))
544
+ }
545
+ />
546
+ )}
547
+ </FormControl>
548
+ <FormMessage />
549
+ </FormItem>
550
+ )}
551
+ />
552
+
553
+ <div className="grid grid-cols-3 gap-4">
554
+ <FormField
555
+ control={form.control}
556
+ name="max_uses"
557
+ render={({ field }) => (
558
+ <FormItem>
559
+ <FormLabel>{t('form.maxUses')}</FormLabel>
560
+ <FormControl>
561
+ <Input
562
+ type="number"
563
+ min={1}
564
+ {...field}
565
+ value={field.value ?? 1}
566
+ onChange={(event) =>
567
+ field.onChange(Number(event.target.value))
568
+ }
569
+ />
570
+ </FormControl>
571
+ <FormMessage />
572
+ </FormItem>
573
+ )}
574
+ />
575
+
576
+ <FormField
577
+ control={form.control}
578
+ name="starts_at"
579
+ render={({ field }) => (
580
+ <FormItem>
581
+ <FormLabel>{t('form.startsAt')}</FormLabel>
582
+ <FormControl>
583
+ <Input
584
+ type="date"
585
+ {...field}
586
+ value={field.value ?? ''}
587
+ />
588
+ </FormControl>
589
+ <FormMessage />
590
+ </FormItem>
591
+ )}
592
+ />
593
+
594
+ <FormField
595
+ control={form.control}
596
+ name="ends_at"
597
+ render={({ field }) => (
598
+ <FormItem>
599
+ <FormLabel>{t('form.endsAt')}</FormLabel>
600
+ <FormControl>
601
+ <Input
602
+ type="date"
603
+ {...field}
604
+ value={field.value ?? ''}
605
+ />
606
+ </FormControl>
607
+ <FormMessage />
608
+ </FormItem>
609
+ )}
610
+ />
611
+ </div>
612
+
613
+ <FormField
614
+ control={form.control}
615
+ name="status"
616
+ render={({ field }) => (
617
+ <FormItem>
618
+ <FormLabel>{t('form.status')}</FormLabel>
619
+ <Select value={field.value} onValueChange={field.onChange}>
620
+ <FormControl>
621
+ <SelectTrigger className="w-full">
622
+ <SelectValue />
623
+ </SelectTrigger>
624
+ </FormControl>
625
+ <SelectContent>
626
+ <SelectItem value="active">active</SelectItem>
627
+ <SelectItem value="inactive">inactive</SelectItem>
628
+ <SelectItem value="draft">draft</SelectItem>
629
+ </SelectContent>
630
+ </Select>
631
+ <FormMessage />
632
+ </FormItem>
633
+ )}
634
+ />
635
+
636
+ <Button
637
+ type="submit"
638
+ className="w-full"
639
+ disabled={form.formState.isSubmitting}
640
+ >
641
+ {t('actions.save')}
642
+ </Button>
643
+ </form>
644
+ </Form>
645
+ </SheetContent>
646
+ </Sheet>
647
+
648
+ <AlertDialog
649
+ open={deleteId !== null}
650
+ onOpenChange={(open) => !open && setDeleteId(null)}
651
+ >
652
+ <AlertDialogContent>
653
+ <AlertDialogHeader>
654
+ <AlertDialogTitle>{t('deleteDialog.title')}</AlertDialogTitle>
655
+ <AlertDialogDescription>
656
+ {t('deleteDialog.description')}
657
+ </AlertDialogDescription>
658
+ </AlertDialogHeader>
659
+ <AlertDialogFooter>
660
+ <AlertDialogCancel>{t('deleteDialog.cancel')}</AlertDialogCancel>
661
+ <AlertDialogAction onClick={onDelete}>
662
+ {t('deleteDialog.confirm')}
663
+ </AlertDialogAction>
664
+ </AlertDialogFooter>
665
+ </AlertDialogContent>
666
+ </AlertDialog>
667
+ </Page>
668
+ );
669
+ }