@lodashventure/medusa-parcel-shipping 0.3.13 → 0.3.14

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.
@@ -0,0 +1,2064 @@
1
+ import { jsxs, jsx, Fragment } from "react/jsx-runtime";
2
+ import { defineWidgetConfig, defineRouteConfig } from "@medusajs/admin-sdk";
3
+ import { Container, Heading, Button, Text, Badge, Drawer, Label, Select, Input, Switch, toast, Table, Prompt, Textarea } from "@medusajs/ui";
4
+ import { useState, useEffect } from "react";
5
+ import { Package, AlertCircle, Truck, Clock, Plus, Edit, Trash } from "lucide-react";
6
+ import { MapPin, ArchiveBox, CurrencyDollarSolid, HandTruck } from "@medusajs/icons";
7
+ import "@medusajs/admin-shared";
8
+ const OrderShippingQuoteWidget = ({ data }) => {
9
+ const [quote, setQuote] = useState(null);
10
+ const [loading, setLoading] = useState(false);
11
+ const [error, setError] = useState(null);
12
+ const order = data;
13
+ const fetchShippingQuote = async () => {
14
+ setLoading(true);
15
+ setError(null);
16
+ try {
17
+ const items = order.items.map((item) => {
18
+ var _a, _b, _c, _d, _e, _f, _g, _h, _i, _j, _k, _l, _m, _n, _o, _p, _q, _r, _s;
19
+ return {
20
+ sku: ((_a = item.variant) == null ? void 0 : _a.sku) || item.title,
21
+ width: ((_b = item.variant) == null ? void 0 : _b.width) || ((_d = (_c = item.variant) == null ? void 0 : _c.product) == null ? void 0 : _d.width) || 10,
22
+ length: ((_e = item.variant) == null ? void 0 : _e.length) || ((_g = (_f = item.variant) == null ? void 0 : _f.product) == null ? void 0 : _g.length) || 10,
23
+ height: ((_h = item.variant) == null ? void 0 : _h.height) || ((_j = (_i = item.variant) == null ? void 0 : _i.product) == null ? void 0 : _j.height) || 10,
24
+ weight: ((_k = item.variant) == null ? void 0 : _k.weight) || ((_m = (_l = item.variant) == null ? void 0 : _l.product) == null ? void 0 : _m.weight) || 0.1,
25
+ quantity: item.quantity,
26
+ attributes: {
27
+ fragile: ((_o = (_n = item.variant) == null ? void 0 : _n.metadata) == null ? void 0 : _o.fragile) || false,
28
+ noStack: ((_q = (_p = item.variant) == null ? void 0 : _p.metadata) == null ? void 0 : _q.noStack) || false,
29
+ stackable: (_s = (_r = item.variant) == null ? void 0 : _r.metadata) == null ? void 0 : _s.stackable
30
+ }
31
+ };
32
+ });
33
+ const shippingAddress = order.shipping_address;
34
+ const payload = {
35
+ items,
36
+ shipping_address: {
37
+ address_line_1: (shippingAddress == null ? void 0 : shippingAddress.address_1) || "",
38
+ address_line_2: (shippingAddress == null ? void 0 : shippingAddress.address_2) || "",
39
+ city: (shippingAddress == null ? void 0 : shippingAddress.city) || "",
40
+ province: (shippingAddress == null ? void 0 : shippingAddress.province) || "",
41
+ postcode: (shippingAddress == null ? void 0 : shippingAddress.postal_code) || "",
42
+ country: (shippingAddress == null ? void 0 : shippingAddress.country_code) || "TH"
43
+ },
44
+ allow_split_boxes: true
45
+ };
46
+ const response = await fetch("/store/quote", {
47
+ method: "POST",
48
+ headers: {
49
+ "Content-Type": "application/json",
50
+ "x-publishable-api-key": "pk_99d0d47048eaf52697f42d149a7f58c1661652292b9e1ea1445f519cd16dc071"
51
+ },
52
+ body: JSON.stringify(payload)
53
+ });
54
+ if (!response.ok) {
55
+ const errorData = await response.json();
56
+ throw new Error(errorData.message || "Failed to fetch quote");
57
+ }
58
+ const data2 = await response.json();
59
+ setQuote(data2);
60
+ } catch (err) {
61
+ setError(err.message);
62
+ } finally {
63
+ setLoading(false);
64
+ }
65
+ };
66
+ const getServiceName = (carrierType, serviceCode) => {
67
+ const serviceNames = {
68
+ COMPANY_FLEET_MESSENGER_3H: "ส่งด่วน 3 ชั่วโมง",
69
+ COMPANY_TRUCK_SAME_DAY: "ส่งด่วนภายในวัน",
70
+ PRIVATE_CARRIER_STANDARD_3_5D: "EMS ไปรษณีย์ (3-5 วัน)",
71
+ PRIVATE_CARRIER_EXPRESS_1_2D: "Express (1-2 วัน)"
72
+ };
73
+ return serviceNames[`${carrierType}_${serviceCode}`] || `${carrierType} - ${serviceCode}`;
74
+ };
75
+ const formatETA = (option) => {
76
+ if (option.eta_hours_min && option.eta_hours_max) {
77
+ return `${option.eta_hours_min}-${option.eta_hours_max} ชั่วโมง`;
78
+ }
79
+ if (option.eta_days_min && option.eta_days_max) {
80
+ return `${option.eta_days_min}-${option.eta_days_max} วัน`;
81
+ }
82
+ return "N/A";
83
+ };
84
+ return /* @__PURE__ */ jsxs(Container, { className: "divide-y p-0", children: [
85
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between px-6 py-4", children: [
86
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-x-2", children: [
87
+ /* @__PURE__ */ jsx(Package, { className: "text-ui-fg-subtle" }),
88
+ /* @__PURE__ */ jsx(Heading, { level: "h2", children: "ตัวเลือกการจัดส่ง" })
89
+ ] }),
90
+ /* @__PURE__ */ jsx(
91
+ Button,
92
+ {
93
+ size: "small",
94
+ onClick: fetchShippingQuote,
95
+ isLoading: loading,
96
+ disabled: !order.shipping_address,
97
+ children: "คำนวณค่าจัดส่ง"
98
+ }
99
+ )
100
+ ] }),
101
+ !order.shipping_address && /* @__PURE__ */ jsx("div", { className: "px-6 py-4", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 text-ui-fg-subtle", children: [
102
+ /* @__PURE__ */ jsx(AlertCircle, {}),
103
+ /* @__PURE__ */ jsx(Text, { size: "small", children: "กรุณาเพิ่มที่อยู่จัดส่งก่อนคำนวณค่าจัดส่ง" })
104
+ ] }) }),
105
+ error && /* @__PURE__ */ jsx("div", { className: "px-6 py-4", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 text-red-500", children: [
106
+ /* @__PURE__ */ jsx(AlertCircle, {}),
107
+ /* @__PURE__ */ jsx(Text, { size: "small", children: error })
108
+ ] }) }),
109
+ quote && /* @__PURE__ */ jsxs("div", { className: "px-6 py-4 space-y-4", children: [
110
+ /* @__PURE__ */ jsxs("div", { className: "rounded-lg border p-4", children: [
111
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2 mb-2", children: [
112
+ /* @__PURE__ */ jsx(Package, { className: "text-ui-fg-subtle", size: 20 }),
113
+ /* @__PURE__ */ jsxs(Text, { weight: "plus", size: "small", children: [
114
+ "กล่องที่แนะนำ: ",
115
+ quote.best_box.name
116
+ ] })
117
+ ] }),
118
+ /* @__PURE__ */ jsxs("div", { className: "grid grid-cols-2 gap-2 text-sm text-ui-fg-subtle", children: [
119
+ /* @__PURE__ */ jsxs("div", { children: [
120
+ "ขนาด: ",
121
+ quote.best_box.width_cm,
122
+ " × ",
123
+ quote.best_box.length_cm,
124
+ " ×",
125
+ " ",
126
+ quote.best_box.height_cm,
127
+ " cm"
128
+ ] }),
129
+ /* @__PURE__ */ jsxs("div", { children: [
130
+ "น้ำหนักรวม: ",
131
+ quote.packing.totalWeightKg.toFixed(2),
132
+ " kg"
133
+ ] }),
134
+ /* @__PURE__ */ jsxs("div", { children: [
135
+ "ความสูงที่ใช้: ",
136
+ quote.packing.usedHeight,
137
+ " cm"
138
+ ] }),
139
+ /* @__PURE__ */ jsxs("div", { children: [
140
+ "ว่างเหลือ: ",
141
+ quote.packing.headroom,
142
+ " cm"
143
+ ] })
144
+ ] })
145
+ ] }),
146
+ /* @__PURE__ */ jsxs("div", { className: "space-y-3", children: [
147
+ /* @__PURE__ */ jsx(Text, { weight: "plus", size: "small", children: "ตัวเลือกการจัดส่ง" }),
148
+ quote.shipping_options.map((option, index) => /* @__PURE__ */ jsxs(
149
+ "div",
150
+ {
151
+ className: `rounded-lg border p-4 ${!option.available ? "opacity-50 bg-ui-bg-subtle" : "bg-ui-bg-base"}`,
152
+ children: [
153
+ /* @__PURE__ */ jsxs("div", { className: "flex items-start justify-between mb-2", children: [
154
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-2", children: [
155
+ /* @__PURE__ */ jsx(Truck, { className: "text-ui-fg-subtle", size: 20 }),
156
+ /* @__PURE__ */ jsx(Text, { weight: "plus", children: getServiceName(option.carrier_type, option.service_code) }),
157
+ option.available ? /* @__PURE__ */ jsx(Badge, { color: "green", size: "small", children: "พร้อมใช้งาน" }) : /* @__PURE__ */ jsx(Badge, { color: "red", size: "small", children: "ไม่พร้อมใช้งาน" })
158
+ ] }),
159
+ /* @__PURE__ */ jsxs(Text, { weight: "plus", size: "large", children: [
160
+ "฿",
161
+ option.price.toFixed(2)
162
+ ] })
163
+ ] }),
164
+ /* @__PURE__ */ jsx("div", { className: "grid grid-cols-2 gap-2 text-sm text-ui-fg-subtle mb-2", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1", children: [
165
+ /* @__PURE__ */ jsx(Clock, { size: 16 }),
166
+ /* @__PURE__ */ jsxs("span", { children: [
167
+ "เวลาจัดส่ง: ",
168
+ formatETA(option)
169
+ ] })
170
+ ] }) }),
171
+ !option.available && option.reason && /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-1 text-sm text-red-500 mb-2", children: [
172
+ /* @__PURE__ */ jsx(AlertCircle, { size: 16 }),
173
+ /* @__PURE__ */ jsx("span", { children: option.reason })
174
+ ] }),
175
+ /* @__PURE__ */ jsxs("div", { className: "mt-3 pt-3 border-t space-y-1 text-sm", children: [
176
+ /* @__PURE__ */ jsxs("div", { className: "flex justify-between text-ui-fg-subtle", children: [
177
+ /* @__PURE__ */ jsx("span", { children: "ค่ากล่อง:" }),
178
+ /* @__PURE__ */ jsxs("span", { children: [
179
+ "฿",
180
+ option.price_breakdown.box_cost.toFixed(2)
181
+ ] })
182
+ ] }),
183
+ /* @__PURE__ */ jsxs("div", { className: "flex justify-between text-ui-fg-subtle", children: [
184
+ /* @__PURE__ */ jsx("span", { children: "ค่าวัสดุห่อหุ้ม:" }),
185
+ /* @__PURE__ */ jsxs("span", { children: [
186
+ "฿",
187
+ option.price_breakdown.packaging_materials_cost.toFixed(
188
+ 2
189
+ )
190
+ ] })
191
+ ] }),
192
+ /* @__PURE__ */ jsxs("div", { className: "flex justify-between text-ui-fg-subtle", children: [
193
+ /* @__PURE__ */ jsx("span", { children: "ค่าจัดส่ง:" }),
194
+ /* @__PURE__ */ jsxs("span", { children: [
195
+ "฿",
196
+ option.price_breakdown.shipping_cost.toFixed(2)
197
+ ] })
198
+ ] }),
199
+ /* @__PURE__ */ jsxs("div", { className: "flex justify-between font-semibold pt-1 border-t", children: [
200
+ /* @__PURE__ */ jsx("span", { children: "รวมทั้งหมด:" }),
201
+ /* @__PURE__ */ jsxs("span", { children: [
202
+ "฿",
203
+ option.price_breakdown.total.toFixed(2)
204
+ ] })
205
+ ] })
206
+ ] })
207
+ ]
208
+ },
209
+ index
210
+ ))
211
+ ] })
212
+ ] })
213
+ ] });
214
+ };
215
+ defineWidgetConfig({
216
+ zone: "order.details.after"
217
+ });
218
+ const TH_PROVINCES = [
219
+ "กรุงเทพมหานคร",
220
+ "กระบี่",
221
+ "กาญจนบุรี",
222
+ "กาฬสินธุ์",
223
+ "กำแพงเพชร",
224
+ "ขอนแก่น",
225
+ "จันทบุรี",
226
+ "ฉะเชิงเทรา",
227
+ "ชลบุรี",
228
+ "ชัยนาท",
229
+ "ชัยภูมิ",
230
+ "ชุมพร",
231
+ "เชียงราย",
232
+ "เชียงใหม่",
233
+ "ตรัง",
234
+ "ตราด",
235
+ "ตาก",
236
+ "นครนายก",
237
+ "นครปฐม",
238
+ "นครพนม",
239
+ "นครราชสีมา",
240
+ "นครศรีธรรมราช",
241
+ "นครสวรรค์",
242
+ "นนทบุรี",
243
+ "นราธิวาส",
244
+ "น่าน",
245
+ "บึงกาฬ",
246
+ "บุรีรัมย์",
247
+ "ปทุมธานี",
248
+ "ประจวบคีรีขันธ์",
249
+ "ปราจีนบุรี",
250
+ "ปัตตานี",
251
+ "พระนครศรีอยุธยา",
252
+ "พะเยา",
253
+ "พังงา",
254
+ "พัทลุง",
255
+ "พิจิตร",
256
+ "พิษณุโลก",
257
+ "เพชรบุรี",
258
+ "เพชรบูรณ์",
259
+ "แพร่",
260
+ "ภูเก็ต",
261
+ "มหาสารคาม",
262
+ "มุกดาหาร",
263
+ "แม่ฮ่องสอน",
264
+ "ยโสธร",
265
+ "ยะลา",
266
+ "ร้อยเอ็ด",
267
+ "ระนอง",
268
+ "ระยอง",
269
+ "ราชบุรี",
270
+ "ลพบุรี",
271
+ "ลำปาง",
272
+ "ลำพูน",
273
+ "เลย",
274
+ "ศรีสะเกษ",
275
+ "สกลนคร",
276
+ "สงขลา",
277
+ "สตูล",
278
+ "สมุทรปราการ",
279
+ "สมุทรสงคราม",
280
+ "สมุทรสาคร",
281
+ "สระแก้ว",
282
+ "สระบุรี",
283
+ "สิงห์บุรี",
284
+ "สุโขทัย",
285
+ "สุพรรณบุรี",
286
+ "สุราษฎร์ธานี",
287
+ "สุรินทร์",
288
+ "หนองคาย",
289
+ "หนองบัวลำภู",
290
+ "อ่างทอง",
291
+ "อำนาจเจริญ",
292
+ "อุดรธานี",
293
+ "อุตรดิตถ์",
294
+ "อุทัยธานี",
295
+ "อุบลราชธานี"
296
+ ];
297
+ const ServiceAreaModal = ({ area, onClose }) => {
298
+ const [formData, setFormData] = useState({
299
+ kind: "PROVINCE",
300
+ value: "",
301
+ active: true
302
+ });
303
+ const [isSaving, setIsSaving] = useState(false);
304
+ useEffect(() => {
305
+ if (area) {
306
+ setFormData({
307
+ kind: area.kind,
308
+ value: area.value,
309
+ active: area.active
310
+ });
311
+ }
312
+ }, [area]);
313
+ const handleSubmit = async (e) => {
314
+ e.preventDefault();
315
+ const errors = [];
316
+ const trimmedValue = formData.value.trim();
317
+ if (!trimmedValue) {
318
+ errors.push("กรุณากรอกค่า");
319
+ }
320
+ if (formData.kind === "PROVINCE") {
321
+ if (!TH_PROVINCES.includes(trimmedValue)) {
322
+ errors.push("จังหวัดไม่ถูกต้อง กรุณาตรวจสอบการสะกดชื่อจังหวัด");
323
+ }
324
+ } else if (formData.kind === "POSTCODE_PREFIX") {
325
+ if (!/^\d{1,5}$/.test(trimmedValue)) {
326
+ errors.push("รหัสไปรษณีย์ต้องเป็นตัวเลข 1-5 หลัก");
327
+ }
328
+ }
329
+ if (errors.length > 0) {
330
+ toast.error("ข้อมูลไม่ถูกต้อง", {
331
+ description: errors.join(", ")
332
+ });
333
+ return;
334
+ }
335
+ setIsSaving(true);
336
+ try {
337
+ const payload = {
338
+ kind: formData.kind,
339
+ value: trimmedValue,
340
+ active: formData.active
341
+ };
342
+ const url = area ? `/admin/service-areas/${area.id}` : "/admin/service-areas";
343
+ const method = area ? "PUT" : "POST";
344
+ const response = await fetch(url, {
345
+ method,
346
+ headers: { "Content-Type": "application/json" },
347
+ credentials: "include",
348
+ body: JSON.stringify(payload)
349
+ });
350
+ if (response.ok) {
351
+ toast.success("สำเร็จ", {
352
+ description: area ? "แก้ไขพื้นที่บริการแล้ว" : "สร้างพื้นที่บริการใหม่แล้ว"
353
+ });
354
+ onClose();
355
+ } else {
356
+ const error = await response.json();
357
+ throw new Error(error.message || "Failed to save");
358
+ }
359
+ } catch (error) {
360
+ toast.error("ข้อผิดพลาด", {
361
+ description: error instanceof Error ? error.message : "ไม่สามารถบันทึกข้อมูลได้"
362
+ });
363
+ } finally {
364
+ setIsSaving(false);
365
+ }
366
+ };
367
+ return /* @__PURE__ */ jsx(Drawer, { open: true, onOpenChange: (open) => !open && onClose(), children: /* @__PURE__ */ jsxs(Drawer.Content, { children: [
368
+ /* @__PURE__ */ jsx(Drawer.Header, { children: /* @__PURE__ */ jsx(Drawer.Title, { children: area ? "แก้ไขพื้นที่บริการ" : "สร้างพื้นที่บริการใหม่" }) }),
369
+ /* @__PURE__ */ jsx(Drawer.Body, { children: /* @__PURE__ */ jsxs("form", { onSubmit: handleSubmit, className: "flex flex-col gap-y-4", children: [
370
+ /* @__PURE__ */ jsxs("div", { children: [
371
+ /* @__PURE__ */ jsx(Label, { htmlFor: "kind", children: "ประเภท *" }),
372
+ /* @__PURE__ */ jsxs(
373
+ Select,
374
+ {
375
+ value: formData.kind,
376
+ onValueChange: (value) => setFormData({ ...formData, kind: value, value: "" }),
377
+ required: true,
378
+ children: [
379
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "เลือกประเภท" }) }),
380
+ /* @__PURE__ */ jsxs(Select.Content, { children: [
381
+ /* @__PURE__ */ jsx(Select.Item, { value: "PROVINCE", children: "จังหวัด" }),
382
+ /* @__PURE__ */ jsx(Select.Item, { value: "POSTCODE_PREFIX", children: "รหัสไปรษณีย์" })
383
+ ] })
384
+ ]
385
+ }
386
+ )
387
+ ] }),
388
+ /* @__PURE__ */ jsxs("div", { children: [
389
+ /* @__PURE__ */ jsx(Label, { htmlFor: "value", children: formData.kind === "PROVINCE" ? "ชื่อจังหวัด *" : "รหัสไปรษณีย์ (1-5 หลัก) *" }),
390
+ formData.kind === "PROVINCE" ? /* @__PURE__ */ jsxs(
391
+ Select,
392
+ {
393
+ value: formData.value,
394
+ onValueChange: (value) => setFormData({ ...formData, value }),
395
+ required: true,
396
+ children: [
397
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "เลือกจังหวัด" }) }),
398
+ /* @__PURE__ */ jsx(Select.Content, { children: TH_PROVINCES.map((province) => /* @__PURE__ */ jsx(Select.Item, { value: province, children: province }, province)) })
399
+ ]
400
+ }
401
+ ) : /* @__PURE__ */ jsx(
402
+ Input,
403
+ {
404
+ id: "value",
405
+ type: "text",
406
+ value: formData.value,
407
+ onChange: (e) => setFormData({ ...formData, value: e.target.value }),
408
+ placeholder: "เช่น 10, 102, 10200",
409
+ maxLength: 5,
410
+ pattern: "\\d{1,5}",
411
+ required: true
412
+ }
413
+ ),
414
+ formData.kind === "POSTCODE_PREFIX" && /* @__PURE__ */ jsx("p", { className: "text-xs text-ui-fg-subtle mt-1", children: 'ใส่รหัส 1-5 หลักเพื่อครอบคลุมหลายพื้นที่ เช่น "10" = กรุงเทพฯ ทั้งหมด' })
415
+ ] }),
416
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-x-2", children: [
417
+ /* @__PURE__ */ jsx(
418
+ Switch,
419
+ {
420
+ id: "active",
421
+ checked: formData.active,
422
+ onCheckedChange: (checked) => setFormData({ ...formData, active: checked })
423
+ }
424
+ ),
425
+ /* @__PURE__ */ jsx(Label, { htmlFor: "active", children: "เปิดใช้งาน" })
426
+ ] }),
427
+ /* @__PURE__ */ jsxs(Drawer.Footer, { children: [
428
+ /* @__PURE__ */ jsx(
429
+ Button,
430
+ {
431
+ type: "button",
432
+ variant: "secondary",
433
+ onClick: onClose,
434
+ disabled: isSaving,
435
+ children: "ยกเลิก"
436
+ }
437
+ ),
438
+ /* @__PURE__ */ jsx(Button, { type: "submit", disabled: isSaving, children: isSaving ? "กำลังบันทึก..." : "บันทึก" })
439
+ ] })
440
+ ] }) })
441
+ ] }) });
442
+ };
443
+ const AREA_KINDS = [
444
+ { value: "ALL", label: "ทั้งหมด" },
445
+ { value: "PROVINCE", label: "จังหวัด" },
446
+ { value: "POSTCODE_PREFIX", label: "รหัสไปรษณีย์" }
447
+ ];
448
+ const ServiceAreasTable = () => {
449
+ const [areas, setAreas] = useState([]);
450
+ const [filteredAreas, setFilteredAreas] = useState([]);
451
+ const [searchTerm, setSearchTerm] = useState("");
452
+ const [kindFilter, setKindFilter] = useState("ALL");
453
+ const [isModalOpen, setIsModalOpen] = useState(false);
454
+ const [editingArea, setEditingArea] = useState(null);
455
+ const [isLoading, setIsLoading] = useState(false);
456
+ const [deleteAreaId, setDeleteAreaId] = useState(null);
457
+ const [deleteAreaValue, setDeleteAreaValue] = useState("");
458
+ useEffect(() => {
459
+ fetchAreas();
460
+ }, []);
461
+ useEffect(() => {
462
+ let filtered = areas;
463
+ if (kindFilter !== "ALL") {
464
+ filtered = filtered.filter((area) => area.kind === kindFilter);
465
+ }
466
+ if (searchTerm) {
467
+ filtered = filtered.filter(
468
+ (area) => area.value.toLowerCase().includes(searchTerm.toLowerCase())
469
+ );
470
+ }
471
+ setFilteredAreas(filtered);
472
+ }, [searchTerm, kindFilter, areas]);
473
+ const fetchAreas = async () => {
474
+ setIsLoading(true);
475
+ try {
476
+ const response = await fetch("/admin/service-areas", {
477
+ credentials: "include"
478
+ });
479
+ const data = await response.json();
480
+ console.log("Fetched service areas:", data);
481
+ setAreas(data.service_areas || []);
482
+ } catch (error) {
483
+ toast.error("ข้อผิดพลาด", {
484
+ description: "ไม่สามารถโหลดข้อมูลพื้นที่บริการได้"
485
+ });
486
+ } finally {
487
+ setIsLoading(false);
488
+ }
489
+ };
490
+ const handleToggleActive = async (area) => {
491
+ try {
492
+ const response = await fetch(`/admin/service-areas/${area.id}`, {
493
+ method: "PUT",
494
+ headers: { "Content-Type": "application/json" },
495
+ credentials: "include",
496
+ body: JSON.stringify({ ...area, active: !area.active })
497
+ });
498
+ if (response.ok) {
499
+ toast.success("สำเร็จ", {
500
+ description: `${area.active ? "ปิดใช้งาน" : "เปิดใช้งาน"}พื้นที่ ${area.value} แล้ว`
501
+ });
502
+ fetchAreas();
503
+ } else {
504
+ throw new Error("Failed to update");
505
+ }
506
+ } catch (error) {
507
+ toast.error("ข้อผิดพลาด", {
508
+ description: "ไม่สามารถอัพเดทสถานะได้"
509
+ });
510
+ }
511
+ };
512
+ const handleDelete = async () => {
513
+ if (!deleteAreaId) return;
514
+ try {
515
+ const response = await fetch(`/admin/service-areas/${deleteAreaId}`, {
516
+ method: "DELETE",
517
+ credentials: "include"
518
+ });
519
+ if (response.ok) {
520
+ toast.success("สำเร็จ", {
521
+ description: `ลบพื้นที่ ${deleteAreaValue} แล้ว`
522
+ });
523
+ fetchAreas();
524
+ } else {
525
+ throw new Error("Failed to delete");
526
+ }
527
+ } catch (error) {
528
+ toast.error("ข้อผิดพลาด", {
529
+ description: "ไม่สามารถลบพื้นที่ได้"
530
+ });
531
+ } finally {
532
+ setDeleteAreaId(null);
533
+ setDeleteAreaValue("");
534
+ }
535
+ };
536
+ const handleEdit = (area) => {
537
+ setEditingArea(area);
538
+ setIsModalOpen(true);
539
+ };
540
+ const handleCreateNew = () => {
541
+ setEditingArea(null);
542
+ setIsModalOpen(true);
543
+ };
544
+ const handleModalClose = () => {
545
+ setIsModalOpen(false);
546
+ setEditingArea(null);
547
+ fetchAreas();
548
+ };
549
+ const openDeletePrompt = (id, value) => {
550
+ setDeleteAreaId(id);
551
+ setDeleteAreaValue(value);
552
+ };
553
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
554
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
555
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between gap-x-4", children: [
556
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-x-4 flex-1", children: [
557
+ /* @__PURE__ */ jsx(
558
+ Input,
559
+ {
560
+ type: "search",
561
+ placeholder: "ค้นหาจังหวัดหรือรหัสไปรษณีย์...",
562
+ value: searchTerm,
563
+ onChange: (e) => setSearchTerm(e.target.value),
564
+ className: "max-w-md"
565
+ }
566
+ ),
567
+ /* @__PURE__ */ jsxs(Select, { value: kindFilter, onValueChange: setKindFilter, children: [
568
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "ประเภท" }) }),
569
+ /* @__PURE__ */ jsx(Select.Content, { children: AREA_KINDS.map((kind) => /* @__PURE__ */ jsx(Select.Item, { value: kind.value, children: kind.label }, kind.value)) })
570
+ ] })
571
+ ] }),
572
+ /* @__PURE__ */ jsxs(Button, { onClick: handleCreateNew, children: [
573
+ /* @__PURE__ */ jsx(Plus, {}),
574
+ "สร้างพื้นที่ใหม่"
575
+ ] })
576
+ ] }),
577
+ /* @__PURE__ */ jsxs(Table, { children: [
578
+ /* @__PURE__ */ jsx(Table.Header, { children: /* @__PURE__ */ jsxs(Table.Row, { children: [
579
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ประเภท" }),
580
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ค่า" }),
581
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "สถานะ" }),
582
+ /* @__PURE__ */ jsx(Table.HeaderCell, { className: "text-right", children: "การจัดการ" })
583
+ ] }) }),
584
+ /* @__PURE__ */ jsx(Table.Body, { children: isLoading ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "กำลังโหลด..." }) }) : filteredAreas.length === 0 ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "ไม่พบข้อมูลพื้นที่บริการ" }) }) : filteredAreas.map((area) => /* @__PURE__ */ jsxs(Table.Row, { children: [
585
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Badge, { color: area.kind === "PROVINCE" ? "blue" : "purple", children: area.kind === "PROVINCE" ? "จังหวัด" : "รหัสไปรษณีย์" }) }),
586
+ /* @__PURE__ */ jsx(Table.Cell, { children: area.value }),
587
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(
588
+ Badge,
589
+ {
590
+ color: area.active ? "green" : "grey",
591
+ className: "cursor-pointer",
592
+ onClick: () => handleToggleActive(area),
593
+ children: area.active ? "เปิดใช้งาน" : "ปิดใช้งาน"
594
+ }
595
+ ) }),
596
+ /* @__PURE__ */ jsx(Table.Cell, { className: "text-right", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
597
+ /* @__PURE__ */ jsx(
598
+ Button,
599
+ {
600
+ variant: "transparent",
601
+ size: "small",
602
+ onClick: () => handleEdit(area),
603
+ children: /* @__PURE__ */ jsx(Edit, {})
604
+ }
605
+ ),
606
+ /* @__PURE__ */ jsx(
607
+ Button,
608
+ {
609
+ variant: "transparent",
610
+ size: "small",
611
+ onClick: () => openDeletePrompt(area.id, area.value),
612
+ children: /* @__PURE__ */ jsx(Trash, {})
613
+ }
614
+ )
615
+ ] }) })
616
+ ] }, area.id)) })
617
+ ] })
618
+ ] }),
619
+ isModalOpen && /* @__PURE__ */ jsx(ServiceAreaModal, { area: editingArea, onClose: handleModalClose }),
620
+ /* @__PURE__ */ jsx(
621
+ Prompt,
622
+ {
623
+ variant: "confirmation",
624
+ open: !!deleteAreaId,
625
+ onOpenChange: () => {
626
+ setDeleteAreaId(null);
627
+ setDeleteAreaValue("");
628
+ },
629
+ children: /* @__PURE__ */ jsxs(Prompt.Content, { children: [
630
+ /* @__PURE__ */ jsxs(Prompt.Header, { children: [
631
+ /* @__PURE__ */ jsx(Prompt.Title, { children: "ลบพื้นที่บริการ" }),
632
+ /* @__PURE__ */ jsxs(Prompt.Description, { children: [
633
+ 'คุณต้องการลบพื้นที่ "',
634
+ deleteAreaValue,
635
+ '" ใช่หรือไม่? การดำเนินการนี้ไม่สามารถย้อนกลับได้'
636
+ ] })
637
+ ] }),
638
+ /* @__PURE__ */ jsxs(Prompt.Footer, { children: [
639
+ /* @__PURE__ */ jsx(Prompt.Cancel, { children: "ยกเลิก" }),
640
+ /* @__PURE__ */ jsx(Prompt.Action, { onClick: handleDelete, children: "ลบ" })
641
+ ] })
642
+ ] })
643
+ }
644
+ )
645
+ ] });
646
+ };
647
+ const AreasPage = () => {
648
+ return /* @__PURE__ */ jsx(Container, { children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
649
+ /* @__PURE__ */ jsx("div", { className: "flex items-center justify-between", children: /* @__PURE__ */ jsx(Heading, { level: "h1", children: "พื้นที่บริการ" }) }),
650
+ /* @__PURE__ */ jsx(ServiceAreasTable, {})
651
+ ] }) });
652
+ };
653
+ const config$3 = defineRouteConfig({
654
+ icon: MapPin,
655
+ label: "พื้นที่บริการ"
656
+ });
657
+ const ParcelBoxModal = ({ box, onClose }) => {
658
+ const [formData, setFormData] = useState({
659
+ name: "",
660
+ width_cm: "",
661
+ length_cm: "",
662
+ height_cm: "",
663
+ max_weight_kg: "",
664
+ price_thb: "",
665
+ active: true
666
+ });
667
+ const [isSaving, setIsSaving] = useState(false);
668
+ useEffect(() => {
669
+ if (box) {
670
+ setFormData({
671
+ name: box.name,
672
+ width_cm: box.width_cm.toString(),
673
+ length_cm: box.length_cm.toString(),
674
+ height_cm: box.height_cm.toString(),
675
+ max_weight_kg: box.max_weight_kg.toString(),
676
+ price_thb: box.price_thb.toString(),
677
+ active: box.active
678
+ });
679
+ }
680
+ }, [box]);
681
+ const handleSubmit = async (e) => {
682
+ e.preventDefault();
683
+ const errors = [];
684
+ if (!formData.name.trim()) {
685
+ errors.push("กรุณากรอกชื่อกล่อง");
686
+ }
687
+ const width = parseFloat(formData.width_cm);
688
+ const length = parseFloat(formData.length_cm);
689
+ const height = parseFloat(formData.height_cm);
690
+ const maxWeight = parseFloat(formData.max_weight_kg);
691
+ const price = parseFloat(formData.price_thb);
692
+ if (isNaN(width) || width <= 0) {
693
+ errors.push("ความกว้างต้องเป็นตัวเลขมากกว่า 0");
694
+ }
695
+ if (isNaN(length) || length <= 0) {
696
+ errors.push("ความยาวต้องเป็นตัวเลขมากกว่า 0");
697
+ }
698
+ if (isNaN(height) || height <= 0) {
699
+ errors.push("ความสูงต้องเป็นตัวเลขมากกว่า 0");
700
+ }
701
+ if (isNaN(maxWeight) || maxWeight <= 0) {
702
+ errors.push("น้ำหนักสูงสุดต้องเป็นตัวเลขมากกว่า 0");
703
+ }
704
+ if (isNaN(price) || price < 0) {
705
+ errors.push("ราคาต้องเป็นตัวเลขไม่ติดลบ");
706
+ }
707
+ if (errors.length > 0) {
708
+ toast.error("ข้อมูลไม่ถูกต้อง", {
709
+ description: errors.join(", ")
710
+ });
711
+ return;
712
+ }
713
+ setIsSaving(true);
714
+ try {
715
+ const payload = {
716
+ name: formData.name.trim(),
717
+ width_cm: width,
718
+ length_cm: length,
719
+ height_cm: height,
720
+ max_weight_kg: maxWeight,
721
+ price_thb: price,
722
+ active: formData.active
723
+ };
724
+ const url = box ? `/admin/boxes/${box.id}` : "/admin/boxes";
725
+ const method = box ? "PUT" : "POST";
726
+ const response = await fetch(url, {
727
+ method,
728
+ headers: { "Content-Type": "application/json" },
729
+ credentials: "include",
730
+ body: JSON.stringify(payload)
731
+ });
732
+ if (response.ok) {
733
+ toast.success("สำเร็จ", {
734
+ description: box ? "แก้ไขกล่องพัสดุแล้ว" : "สร้างกล่องพัสดุใหม่แล้ว"
735
+ });
736
+ onClose();
737
+ } else {
738
+ const error = await response.json();
739
+ throw new Error(error.message || "Failed to save");
740
+ }
741
+ } catch (error) {
742
+ toast.error("ข้อผิดพลาด", {
743
+ description: error instanceof Error ? error.message : "ไม่สามารถบันทึกข้อมูลได้"
744
+ });
745
+ } finally {
746
+ setIsSaving(false);
747
+ }
748
+ };
749
+ return /* @__PURE__ */ jsx(Drawer, { open: true, onOpenChange: (open) => !open && onClose(), children: /* @__PURE__ */ jsxs(Drawer.Content, { children: [
750
+ /* @__PURE__ */ jsx(Drawer.Header, { children: /* @__PURE__ */ jsx(Drawer.Title, { children: box ? "แก้ไขกล่องพัสดุ" : "สร้างกล่องพัสดุใหม่" }) }),
751
+ /* @__PURE__ */ jsx(Drawer.Body, { children: /* @__PURE__ */ jsxs("form", { onSubmit: handleSubmit, className: "flex flex-col gap-y-4", children: [
752
+ /* @__PURE__ */ jsxs("div", { children: [
753
+ /* @__PURE__ */ jsx(Label, { htmlFor: "name", children: "ชื่อกล่อง *" }),
754
+ /* @__PURE__ */ jsx(
755
+ Input,
756
+ {
757
+ id: "name",
758
+ value: formData.name,
759
+ onChange: (e) => setFormData({ ...formData, name: e.target.value }),
760
+ placeholder: "เช่น กล่อง S, กล่อง M",
761
+ required: true
762
+ }
763
+ )
764
+ ] }),
765
+ /* @__PURE__ */ jsxs("div", { className: "grid grid-cols-3 gap-x-4", children: [
766
+ /* @__PURE__ */ jsxs("div", { children: [
767
+ /* @__PURE__ */ jsx(Label, { htmlFor: "width", children: "ความกว้าง (cm) *" }),
768
+ /* @__PURE__ */ jsx(
769
+ Input,
770
+ {
771
+ id: "width",
772
+ type: "number",
773
+ step: "0.1",
774
+ value: formData.width_cm,
775
+ onChange: (e) => setFormData({ ...formData, width_cm: e.target.value }),
776
+ placeholder: "20",
777
+ required: true
778
+ }
779
+ )
780
+ ] }),
781
+ /* @__PURE__ */ jsxs("div", { children: [
782
+ /* @__PURE__ */ jsx(Label, { htmlFor: "length", children: "ความยาว (cm) *" }),
783
+ /* @__PURE__ */ jsx(
784
+ Input,
785
+ {
786
+ id: "length",
787
+ type: "number",
788
+ step: "0.1",
789
+ value: formData.length_cm,
790
+ onChange: (e) => setFormData({ ...formData, length_cm: e.target.value }),
791
+ placeholder: "30",
792
+ required: true
793
+ }
794
+ )
795
+ ] }),
796
+ /* @__PURE__ */ jsxs("div", { children: [
797
+ /* @__PURE__ */ jsx(Label, { htmlFor: "height", children: "ความสูง (cm) *" }),
798
+ /* @__PURE__ */ jsx(
799
+ Input,
800
+ {
801
+ id: "height",
802
+ type: "number",
803
+ step: "0.1",
804
+ value: formData.height_cm,
805
+ onChange: (e) => setFormData({ ...formData, height_cm: e.target.value }),
806
+ placeholder: "15",
807
+ required: true
808
+ }
809
+ )
810
+ ] })
811
+ ] }),
812
+ /* @__PURE__ */ jsxs("div", { children: [
813
+ /* @__PURE__ */ jsx(Label, { htmlFor: "maxWeight", children: "น้ำหนักสูงสุด (kg) *" }),
814
+ /* @__PURE__ */ jsx(
815
+ Input,
816
+ {
817
+ id: "maxWeight",
818
+ type: "number",
819
+ step: "0.1",
820
+ value: formData.max_weight_kg,
821
+ onChange: (e) => setFormData({ ...formData, max_weight_kg: e.target.value }),
822
+ placeholder: "0.5",
823
+ required: true
824
+ }
825
+ )
826
+ ] }),
827
+ /* @__PURE__ */ jsxs("div", { children: [
828
+ /* @__PURE__ */ jsx(Label, { htmlFor: "price", children: "ราคา (THB) *" }),
829
+ /* @__PURE__ */ jsx(
830
+ Input,
831
+ {
832
+ id: "price",
833
+ type: "number",
834
+ step: "0.01",
835
+ value: formData.price_thb,
836
+ onChange: (e) => setFormData({ ...formData, price_thb: e.target.value }),
837
+ placeholder: "10.00",
838
+ required: true
839
+ }
840
+ )
841
+ ] }),
842
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-x-2", children: [
843
+ /* @__PURE__ */ jsx(
844
+ Switch,
845
+ {
846
+ id: "active",
847
+ checked: formData.active,
848
+ onCheckedChange: (checked) => setFormData({ ...formData, active: checked })
849
+ }
850
+ ),
851
+ /* @__PURE__ */ jsx(Label, { htmlFor: "active", children: "เปิดใช้งาน" })
852
+ ] }),
853
+ /* @__PURE__ */ jsxs(Drawer.Footer, { children: [
854
+ /* @__PURE__ */ jsx(Button, { type: "button", variant: "secondary", onClick: onClose, disabled: isSaving, children: "ยกเลิก" }),
855
+ /* @__PURE__ */ jsx(Button, { type: "submit", disabled: isSaving, children: isSaving ? "กำลังบันทึก..." : "บันทึก" })
856
+ ] })
857
+ ] }) })
858
+ ] }) });
859
+ };
860
+ const ParcelBoxesTable = () => {
861
+ const [boxes, setBoxes] = useState([]);
862
+ const [filteredBoxes, setFilteredBoxes] = useState([]);
863
+ const [searchTerm, setSearchTerm] = useState("");
864
+ const [isModalOpen, setIsModalOpen] = useState(false);
865
+ const [editingBox, setEditingBox] = useState(null);
866
+ const [isLoading, setIsLoading] = useState(false);
867
+ const [deleteBoxId, setDeleteBoxId] = useState(null);
868
+ const [deleteBoxName, setDeleteBoxName] = useState("");
869
+ useEffect(() => {
870
+ fetchBoxes();
871
+ }, []);
872
+ useEffect(() => {
873
+ const filtered = boxes.filter(
874
+ (box) => box.name.toLowerCase().includes(searchTerm.toLowerCase())
875
+ );
876
+ setFilteredBoxes(filtered);
877
+ }, [searchTerm, boxes]);
878
+ const fetchBoxes = async () => {
879
+ setIsLoading(true);
880
+ try {
881
+ const response = await fetch("/admin/boxes", {
882
+ credentials: "include"
883
+ });
884
+ const data = await response.json();
885
+ setBoxes(data.boxes || []);
886
+ } catch (error) {
887
+ toast.error("ข้อผิดพลาด", {
888
+ description: "ไม่สามารถโหลดข้อมูลกล่องพัสดุได้"
889
+ });
890
+ } finally {
891
+ setIsLoading(false);
892
+ }
893
+ };
894
+ const handleToggleActive = async (box) => {
895
+ try {
896
+ const response = await fetch(`/admin/boxes/${box.id}`, {
897
+ method: "PUT",
898
+ headers: { "Content-Type": "application/json" },
899
+ credentials: "include",
900
+ body: JSON.stringify({ ...box, active: !box.active })
901
+ });
902
+ if (response.ok) {
903
+ toast.success("สำเร็จ", {
904
+ description: `${box.active ? "ปิดใช้งาน" : "เปิดใช้งาน"}กล่อง ${box.name} แล้ว`
905
+ });
906
+ fetchBoxes();
907
+ } else {
908
+ throw new Error("Failed to update");
909
+ }
910
+ } catch (error) {
911
+ toast.error("ข้อผิดพลาด", {
912
+ description: "ไม่สามารถอัพเดทสถานะได้"
913
+ });
914
+ }
915
+ };
916
+ const handleDelete = async () => {
917
+ if (!deleteBoxId) return;
918
+ try {
919
+ const response = await fetch(`/admin/boxes/${deleteBoxId}`, {
920
+ method: "DELETE",
921
+ credentials: "include"
922
+ });
923
+ if (response.ok) {
924
+ toast.success("สำเร็จ", {
925
+ description: `ลบกล่อง ${deleteBoxName} แล้ว`
926
+ });
927
+ fetchBoxes();
928
+ } else {
929
+ throw new Error("Failed to delete");
930
+ }
931
+ } catch (error) {
932
+ toast.error("ข้อผิดพลาด", {
933
+ description: "ไม่สามารถลบกล่องได้"
934
+ });
935
+ } finally {
936
+ setDeleteBoxId(null);
937
+ setDeleteBoxName("");
938
+ }
939
+ };
940
+ const handleEdit = (box) => {
941
+ setEditingBox(box);
942
+ setIsModalOpen(true);
943
+ };
944
+ const handleCreateNew = () => {
945
+ setEditingBox(null);
946
+ setIsModalOpen(true);
947
+ };
948
+ const handleModalClose = () => {
949
+ setIsModalOpen(false);
950
+ setEditingBox(null);
951
+ fetchBoxes();
952
+ };
953
+ const openDeletePrompt = (id, name) => {
954
+ setDeleteBoxId(id);
955
+ setDeleteBoxName(name);
956
+ };
957
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
958
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
959
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between gap-x-4", children: [
960
+ /* @__PURE__ */ jsx(
961
+ Input,
962
+ {
963
+ type: "search",
964
+ placeholder: "ค้นหาชื่อกล่อง...",
965
+ value: searchTerm,
966
+ onChange: (e) => setSearchTerm(e.target.value),
967
+ className: "max-w-md"
968
+ }
969
+ ),
970
+ /* @__PURE__ */ jsxs(Button, { onClick: handleCreateNew, children: [
971
+ /* @__PURE__ */ jsx(Plus, {}),
972
+ "สร้างกล่องใหม่"
973
+ ] })
974
+ ] }),
975
+ /* @__PURE__ */ jsxs(Table, { children: [
976
+ /* @__PURE__ */ jsx(Table.Header, { children: /* @__PURE__ */ jsxs(Table.Row, { children: [
977
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ชื่อ" }),
978
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ขนาด (กว้าง×ยาว×สูง cm)" }),
979
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "น้ำหนักสูงสุด (kg)" }),
980
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ราคา (THB)" }),
981
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "สถานะ" }),
982
+ /* @__PURE__ */ jsx(Table.HeaderCell, { className: "text-right", children: "การจัดการ" })
983
+ ] }) }),
984
+ /* @__PURE__ */ jsx(Table.Body, { children: isLoading ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "กำลังโหลด..." }) }) : filteredBoxes.length === 0 ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "ไม่พบข้อมูลกล่องพัสดุ" }) }) : filteredBoxes.map((box) => /* @__PURE__ */ jsxs(Table.Row, { children: [
985
+ /* @__PURE__ */ jsx(Table.Cell, { children: box.name }),
986
+ /* @__PURE__ */ jsxs(Table.Cell, { children: [
987
+ box.width_cm,
988
+ " × ",
989
+ box.length_cm,
990
+ " × ",
991
+ box.height_cm
992
+ ] }),
993
+ /* @__PURE__ */ jsx(Table.Cell, { children: box.max_weight_kg }),
994
+ /* @__PURE__ */ jsx(Table.Cell, { children: box.price_thb.toFixed(2) }),
995
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(
996
+ Badge,
997
+ {
998
+ color: box.active ? "green" : "grey",
999
+ className: "cursor-pointer",
1000
+ onClick: () => handleToggleActive(box),
1001
+ children: box.active ? "เปิดใช้งาน" : "ปิดใช้งาน"
1002
+ }
1003
+ ) }),
1004
+ /* @__PURE__ */ jsx(Table.Cell, { className: "text-right", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
1005
+ /* @__PURE__ */ jsx(
1006
+ Button,
1007
+ {
1008
+ variant: "transparent",
1009
+ size: "small",
1010
+ onClick: () => handleEdit(box),
1011
+ children: /* @__PURE__ */ jsx(Edit, {})
1012
+ }
1013
+ ),
1014
+ /* @__PURE__ */ jsx(
1015
+ Button,
1016
+ {
1017
+ variant: "transparent",
1018
+ size: "small",
1019
+ onClick: () => openDeletePrompt(box.id, box.name),
1020
+ children: /* @__PURE__ */ jsx(Trash, {})
1021
+ }
1022
+ )
1023
+ ] }) })
1024
+ ] }, box.id)) })
1025
+ ] })
1026
+ ] }),
1027
+ isModalOpen && /* @__PURE__ */ jsx(ParcelBoxModal, { box: editingBox, onClose: handleModalClose }),
1028
+ /* @__PURE__ */ jsx(
1029
+ Prompt,
1030
+ {
1031
+ variant: "confirmation",
1032
+ open: !!deleteBoxId,
1033
+ onOpenChange: () => {
1034
+ setDeleteBoxId(null);
1035
+ setDeleteBoxName("");
1036
+ },
1037
+ children: /* @__PURE__ */ jsxs(Prompt.Content, { children: [
1038
+ /* @__PURE__ */ jsxs(Prompt.Header, { children: [
1039
+ /* @__PURE__ */ jsx(Prompt.Title, { children: "ลบกล่องพัสดุ" }),
1040
+ /* @__PURE__ */ jsxs(Prompt.Description, { children: [
1041
+ 'คุณต้องการลบกล่อง "',
1042
+ deleteBoxName,
1043
+ '" ใช่หรือไม่? การดำเนินการนี้ไม่สามารถย้อนกลับได้'
1044
+ ] })
1045
+ ] }),
1046
+ /* @__PURE__ */ jsxs(Prompt.Footer, { children: [
1047
+ /* @__PURE__ */ jsx(Prompt.Cancel, { children: "ยกเลิก" }),
1048
+ /* @__PURE__ */ jsx(Prompt.Action, { onClick: handleDelete, children: "ลบ" })
1049
+ ] })
1050
+ ] })
1051
+ }
1052
+ )
1053
+ ] });
1054
+ };
1055
+ const BoxesPage = () => {
1056
+ return /* @__PURE__ */ jsx(Container, { children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
1057
+ /* @__PURE__ */ jsx("div", { className: "flex items-center justify-between", children: /* @__PURE__ */ jsx(Heading, { level: "h1", children: "กล่องพัสดุ" }) }),
1058
+ /* @__PURE__ */ jsx(ParcelBoxesTable, {})
1059
+ ] }) });
1060
+ };
1061
+ const config$2 = defineRouteConfig({
1062
+ icon: ArchiveBox,
1063
+ label: "กล่องพัสดุ"
1064
+ });
1065
+ const CATEGORIES$1 = [
1066
+ { value: "PACKAGING", label: "วัสดุหีบห่อ" },
1067
+ { value: "PROTECTION", label: "วัสดุป้องกัน" },
1068
+ { value: "FILLING", label: "วัสดุเติมช่องว่าง" },
1069
+ { value: "TAPE", label: "เทป/กาว" },
1070
+ { value: "LABEL", label: "ฉลาก/สติกเกอร์" },
1071
+ { value: "OTHER", label: "อื่นๆ" }
1072
+ ];
1073
+ const MaterialCostModal = ({
1074
+ materialCost,
1075
+ onClose
1076
+ }) => {
1077
+ const [formData, setFormData] = useState({
1078
+ name: "",
1079
+ unit: "",
1080
+ cost_per_unit: "",
1081
+ currency: "THB",
1082
+ category: "",
1083
+ description: "",
1084
+ active: true
1085
+ });
1086
+ const [isSaving, setIsSaving] = useState(false);
1087
+ useEffect(() => {
1088
+ if (materialCost) {
1089
+ setFormData({
1090
+ name: materialCost.name,
1091
+ unit: materialCost.unit,
1092
+ cost_per_unit: materialCost.cost_per_unit.toString(),
1093
+ currency: materialCost.currency,
1094
+ category: materialCost.category || "",
1095
+ description: materialCost.description || "",
1096
+ active: materialCost.active
1097
+ });
1098
+ }
1099
+ }, [materialCost]);
1100
+ const handleSubmit = async (e) => {
1101
+ e.preventDefault();
1102
+ const errors = [];
1103
+ if (!formData.name.trim()) {
1104
+ errors.push("กรุณากรอกชื่อวัสดุ");
1105
+ }
1106
+ if (!formData.unit.trim()) {
1107
+ errors.push("กรุณากรอกหน่วย");
1108
+ }
1109
+ const costPerUnit = parseFloat(formData.cost_per_unit);
1110
+ if (isNaN(costPerUnit) || costPerUnit < 0) {
1111
+ errors.push("ต้นทุนต่อหน่วยต้องเป็นตัวเลขไม่ติดลบ");
1112
+ }
1113
+ if (errors.length > 0) {
1114
+ toast.error("ข้อมูลไม่ถูกต้อง", {
1115
+ description: errors.join(", ")
1116
+ });
1117
+ return;
1118
+ }
1119
+ setIsSaving(true);
1120
+ try {
1121
+ const payload = {
1122
+ name: formData.name.trim(),
1123
+ unit: formData.unit.trim(),
1124
+ cost_per_unit: costPerUnit,
1125
+ currency: formData.currency,
1126
+ category: formData.category || void 0,
1127
+ description: formData.description.trim() || void 0,
1128
+ active: formData.active
1129
+ };
1130
+ const url = materialCost ? `/admin/material-costs/${materialCost.id}` : "/admin/material-costs";
1131
+ const method = materialCost ? "PUT" : "POST";
1132
+ const response = await fetch(url, {
1133
+ method,
1134
+ headers: { "Content-Type": "application/json" },
1135
+ credentials: "include",
1136
+ body: JSON.stringify(payload)
1137
+ });
1138
+ if (response.ok) {
1139
+ toast.success("สำเร็จ", {
1140
+ description: materialCost ? "แก้ไขรายการต้นทุนวัสดุแล้ว" : "สร้างรายการต้นทุนวัสดุใหม่แล้ว"
1141
+ });
1142
+ onClose();
1143
+ } else {
1144
+ const error = await response.json();
1145
+ throw new Error(error.message || "Failed to save");
1146
+ }
1147
+ } catch (error) {
1148
+ toast.error("ข้อผิดพลาด", {
1149
+ description: error instanceof Error ? error.message : "ไม่สามารถบันทึกข้อมูลได้"
1150
+ });
1151
+ } finally {
1152
+ setIsSaving(false);
1153
+ }
1154
+ };
1155
+ return /* @__PURE__ */ jsx(Drawer, { open: true, onOpenChange: (open) => !open && onClose(), children: /* @__PURE__ */ jsxs(Drawer.Content, { children: [
1156
+ /* @__PURE__ */ jsx(Drawer.Header, { children: /* @__PURE__ */ jsx(Drawer.Title, { children: materialCost ? "แก้ไขรายการต้นทุนวัสดุ" : "สร้างรายการต้นทุนวัสดุใหม่" }) }),
1157
+ /* @__PURE__ */ jsx(Drawer.Body, { children: /* @__PURE__ */ jsxs("form", { onSubmit: handleSubmit, className: "flex flex-col gap-y-4", children: [
1158
+ /* @__PURE__ */ jsxs("div", { children: [
1159
+ /* @__PURE__ */ jsx(Label, { htmlFor: "name", children: "ชื่อวัสดุ *" }),
1160
+ /* @__PURE__ */ jsx(
1161
+ Input,
1162
+ {
1163
+ id: "name",
1164
+ value: formData.name,
1165
+ onChange: (e) => setFormData({ ...formData, name: e.target.value }),
1166
+ placeholder: "เช่น พลาสติกกันกระแทก, กล่องกระดาษ",
1167
+ required: true
1168
+ }
1169
+ )
1170
+ ] }),
1171
+ /* @__PURE__ */ jsxs("div", { children: [
1172
+ /* @__PURE__ */ jsx(Label, { htmlFor: "category", children: "หมวดหมู่ (ไม่บังคับ)" }),
1173
+ /* @__PURE__ */ jsxs(
1174
+ Select,
1175
+ {
1176
+ value: formData.category || void 0,
1177
+ onValueChange: (value) => setFormData({ ...formData, category: value }),
1178
+ children: [
1179
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "เลือกหมวดหมู่ (ไม่บังคับ)" }) }),
1180
+ /* @__PURE__ */ jsx(Select.Content, { children: CATEGORIES$1.map((category) => /* @__PURE__ */ jsx(Select.Item, { value: category.value, children: category.label }, category.value)) })
1181
+ ]
1182
+ }
1183
+ )
1184
+ ] }),
1185
+ /* @__PURE__ */ jsxs("div", { className: "grid grid-cols-2 gap-x-4", children: [
1186
+ /* @__PURE__ */ jsxs("div", { children: [
1187
+ /* @__PURE__ */ jsx(Label, { htmlFor: "unit", children: "หน่วย *" }),
1188
+ /* @__PURE__ */ jsx(
1189
+ Input,
1190
+ {
1191
+ id: "unit",
1192
+ value: formData.unit,
1193
+ onChange: (e) => setFormData({ ...formData, unit: e.target.value }),
1194
+ placeholder: "เช่น ชิ้น, เมตร, กิโลกรัม",
1195
+ required: true
1196
+ }
1197
+ )
1198
+ ] }),
1199
+ /* @__PURE__ */ jsxs("div", { children: [
1200
+ /* @__PURE__ */ jsx(Label, { htmlFor: "costPerUnit", children: "ต้นทุนต่อหน่วย *" }),
1201
+ /* @__PURE__ */ jsx(
1202
+ Input,
1203
+ {
1204
+ id: "costPerUnit",
1205
+ type: "number",
1206
+ step: "0.01",
1207
+ value: formData.cost_per_unit,
1208
+ onChange: (e) => setFormData({ ...formData, cost_per_unit: e.target.value }),
1209
+ placeholder: "0.00",
1210
+ required: true
1211
+ }
1212
+ )
1213
+ ] })
1214
+ ] }),
1215
+ /* @__PURE__ */ jsxs("div", { children: [
1216
+ /* @__PURE__ */ jsx(Label, { htmlFor: "currency", children: "สกุลเงิน" }),
1217
+ /* @__PURE__ */ jsx(
1218
+ Input,
1219
+ {
1220
+ id: "currency",
1221
+ value: formData.currency,
1222
+ onChange: (e) => setFormData({ ...formData, currency: e.target.value }),
1223
+ placeholder: "THB"
1224
+ }
1225
+ )
1226
+ ] }),
1227
+ /* @__PURE__ */ jsxs("div", { children: [
1228
+ /* @__PURE__ */ jsx(Label, { htmlFor: "description", children: "คำอธิบาย" }),
1229
+ /* @__PURE__ */ jsx(
1230
+ Textarea,
1231
+ {
1232
+ id: "description",
1233
+ value: formData.description,
1234
+ onChange: (e) => setFormData({ ...formData, description: e.target.value }),
1235
+ placeholder: "คำอธิบายเพิ่มเติมเกี่ยวกับวัสดุนี้...",
1236
+ rows: 3
1237
+ }
1238
+ )
1239
+ ] }),
1240
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-x-2", children: [
1241
+ /* @__PURE__ */ jsx(
1242
+ Switch,
1243
+ {
1244
+ id: "active",
1245
+ checked: formData.active,
1246
+ onCheckedChange: (checked) => setFormData({ ...formData, active: checked })
1247
+ }
1248
+ ),
1249
+ /* @__PURE__ */ jsx(Label, { htmlFor: "active", children: "เปิดใช้งาน" })
1250
+ ] }),
1251
+ /* @__PURE__ */ jsxs(Drawer.Footer, { children: [
1252
+ /* @__PURE__ */ jsx(
1253
+ Button,
1254
+ {
1255
+ type: "button",
1256
+ variant: "secondary",
1257
+ onClick: onClose,
1258
+ disabled: isSaving,
1259
+ children: "ยกเลิก"
1260
+ }
1261
+ ),
1262
+ /* @__PURE__ */ jsx(Button, { type: "submit", disabled: isSaving, children: isSaving ? "กำลังบันทึก..." : "บันทึก" })
1263
+ ] })
1264
+ ] }) })
1265
+ ] }) });
1266
+ };
1267
+ const CATEGORIES = [
1268
+ { value: "ALL", label: "ทั้งหมด" },
1269
+ { value: "PACKAGING", label: "วัสดุหีบห่อ" },
1270
+ { value: "PROTECTION", label: "วัสดุป้องกัน" },
1271
+ { value: "FILLING", label: "วัสดุเติมช่องว่าง" },
1272
+ { value: "TAPE", label: "เทป/กาว" },
1273
+ { value: "LABEL", label: "ฉลาก/สติกเกอร์" },
1274
+ { value: "OTHER", label: "อื่นๆ" }
1275
+ ];
1276
+ const MaterialCostsTable = () => {
1277
+ const [materialCosts, setMaterialCosts] = useState([]);
1278
+ const [filteredMaterialCosts, setFilteredMaterialCosts] = useState([]);
1279
+ const [searchTerm, setSearchTerm] = useState("");
1280
+ const [categoryFilter, setCategoryFilter] = useState("ALL");
1281
+ const [isModalOpen, setIsModalOpen] = useState(false);
1282
+ const [editingMaterialCost, setEditingMaterialCost] = useState(null);
1283
+ const [isLoading, setIsLoading] = useState(false);
1284
+ const [deleteMaterialCostId, setDeleteMaterialCostId] = useState(null);
1285
+ const [deleteMaterialCostName, setDeleteMaterialCostName] = useState("");
1286
+ useEffect(() => {
1287
+ fetchMaterialCosts();
1288
+ }, []);
1289
+ useEffect(() => {
1290
+ let filtered = materialCosts;
1291
+ if (categoryFilter !== "ALL") {
1292
+ filtered = filtered.filter((mc) => mc.category === categoryFilter);
1293
+ }
1294
+ if (searchTerm) {
1295
+ filtered = filtered.filter(
1296
+ (mc) => mc.name.toLowerCase().includes(searchTerm.toLowerCase())
1297
+ );
1298
+ }
1299
+ setFilteredMaterialCosts(filtered);
1300
+ }, [searchTerm, categoryFilter, materialCosts]);
1301
+ const fetchMaterialCosts = async () => {
1302
+ setIsLoading(true);
1303
+ try {
1304
+ const response = await fetch("/admin/material-costs", {
1305
+ credentials: "include"
1306
+ });
1307
+ const data = await response.json();
1308
+ setMaterialCosts(data.material_costs || []);
1309
+ } catch (error) {
1310
+ toast.error("ข้อผิดพลาด", {
1311
+ description: "ไม่สามารถโหลดข้อมูลต้นทุนวัสดุได้"
1312
+ });
1313
+ } finally {
1314
+ setIsLoading(false);
1315
+ }
1316
+ };
1317
+ const handleToggleActive = async (materialCost) => {
1318
+ try {
1319
+ const response = await fetch(`/admin/material-costs/${materialCost.id}`, {
1320
+ method: "PUT",
1321
+ headers: { "Content-Type": "application/json" },
1322
+ credentials: "include",
1323
+ body: JSON.stringify({ ...materialCost, active: !materialCost.active })
1324
+ });
1325
+ if (response.ok) {
1326
+ toast.success("สำเร็จ", {
1327
+ description: `${materialCost.active ? "ปิดใช้งาน" : "เปิดใช้งาน"}วัสดุ ${materialCost.name} แล้ว`
1328
+ });
1329
+ fetchMaterialCosts();
1330
+ } else {
1331
+ throw new Error("Failed to update");
1332
+ }
1333
+ } catch (error) {
1334
+ toast.error("ข้อผิดพลาด", {
1335
+ description: "ไม่สามารถอัพเดทสถานะได้"
1336
+ });
1337
+ }
1338
+ };
1339
+ const handleDelete = async () => {
1340
+ if (!deleteMaterialCostId) return;
1341
+ try {
1342
+ const response = await fetch(
1343
+ `/admin/material-costs/${deleteMaterialCostId}`,
1344
+ {
1345
+ method: "DELETE",
1346
+ credentials: "include"
1347
+ }
1348
+ );
1349
+ if (response.ok) {
1350
+ toast.success("สำเร็จ", {
1351
+ description: `ลบวัสดุ ${deleteMaterialCostName} แล้ว`
1352
+ });
1353
+ fetchMaterialCosts();
1354
+ } else {
1355
+ throw new Error("Failed to delete");
1356
+ }
1357
+ } catch (error) {
1358
+ toast.error("ข้อผิดพลาด", {
1359
+ description: "ไม่สามารถลบวัสดุได้"
1360
+ });
1361
+ } finally {
1362
+ setDeleteMaterialCostId(null);
1363
+ setDeleteMaterialCostName("");
1364
+ }
1365
+ };
1366
+ const handleEdit = (materialCost) => {
1367
+ setEditingMaterialCost(materialCost);
1368
+ setIsModalOpen(true);
1369
+ };
1370
+ const handleCreateNew = () => {
1371
+ setEditingMaterialCost(null);
1372
+ setIsModalOpen(true);
1373
+ };
1374
+ const handleModalClose = () => {
1375
+ setIsModalOpen(false);
1376
+ setEditingMaterialCost(null);
1377
+ fetchMaterialCosts();
1378
+ };
1379
+ const openDeletePrompt = (id, name) => {
1380
+ setDeleteMaterialCostId(id);
1381
+ setDeleteMaterialCostName(name);
1382
+ };
1383
+ const getCategoryLabel = (category) => {
1384
+ var _a;
1385
+ if (!category) return "-";
1386
+ return ((_a = CATEGORIES.find((c) => c.value === category)) == null ? void 0 : _a.label) || category;
1387
+ };
1388
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
1389
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
1390
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between gap-x-4", children: [
1391
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-x-4 flex-1", children: [
1392
+ /* @__PURE__ */ jsx(
1393
+ Input,
1394
+ {
1395
+ type: "search",
1396
+ placeholder: "ค้นหาชื่อวัสดุ...",
1397
+ value: searchTerm,
1398
+ onChange: (e) => setSearchTerm(e.target.value),
1399
+ className: "max-w-md"
1400
+ }
1401
+ ),
1402
+ /* @__PURE__ */ jsxs(Select, { value: categoryFilter, onValueChange: setCategoryFilter, children: [
1403
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "หมวดหมู่" }) }),
1404
+ /* @__PURE__ */ jsx(Select.Content, { children: CATEGORIES.map((category) => /* @__PURE__ */ jsx(Select.Item, { value: category.value, children: category.label }, category.value)) })
1405
+ ] })
1406
+ ] }),
1407
+ /* @__PURE__ */ jsxs(Button, { onClick: handleCreateNew, children: [
1408
+ /* @__PURE__ */ jsx(Plus, {}),
1409
+ "สร้างรายการใหม่"
1410
+ ] })
1411
+ ] }),
1412
+ /* @__PURE__ */ jsxs(Table, { children: [
1413
+ /* @__PURE__ */ jsx(Table.Header, { children: /* @__PURE__ */ jsxs(Table.Row, { children: [
1414
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ชื่อ" }),
1415
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "หมวดหมู่" }),
1416
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "หน่วย" }),
1417
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ต้นทุนต่อหน่วย" }),
1418
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "คำอธิบาย" }),
1419
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "สถานะ" }),
1420
+ /* @__PURE__ */ jsx(Table.HeaderCell, { className: "text-right", children: "การจัดการ" })
1421
+ ] }) }),
1422
+ /* @__PURE__ */ jsx(Table.Body, { children: isLoading ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "กำลังโหลด..." }) }) : filteredMaterialCosts.length === 0 ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "ไม่พบข้อมูลต้นทุนวัสดุ" }) }) : filteredMaterialCosts.map((materialCost) => /* @__PURE__ */ jsxs(Table.Row, { children: [
1423
+ /* @__PURE__ */ jsx(Table.Cell, { children: materialCost.name }),
1424
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Badge, { color: "blue", children: getCategoryLabel(materialCost.category) }) }),
1425
+ /* @__PURE__ */ jsx(Table.Cell, { children: materialCost.unit }),
1426
+ /* @__PURE__ */ jsxs(Table.Cell, { children: [
1427
+ materialCost.cost_per_unit.toFixed(2),
1428
+ " ",
1429
+ materialCost.currency
1430
+ ] }),
1431
+ /* @__PURE__ */ jsx(Table.Cell, { className: "max-w-xs truncate", children: materialCost.description || "-" }),
1432
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(
1433
+ Badge,
1434
+ {
1435
+ color: materialCost.active ? "green" : "grey",
1436
+ className: "cursor-pointer",
1437
+ onClick: () => handleToggleActive(materialCost),
1438
+ children: materialCost.active ? "เปิดใช้งาน" : "ปิดใช้งาน"
1439
+ }
1440
+ ) }),
1441
+ /* @__PURE__ */ jsx(Table.Cell, { className: "text-right", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
1442
+ /* @__PURE__ */ jsx(
1443
+ Button,
1444
+ {
1445
+ variant: "transparent",
1446
+ size: "small",
1447
+ onClick: () => handleEdit(materialCost),
1448
+ children: /* @__PURE__ */ jsx(Edit, {})
1449
+ }
1450
+ ),
1451
+ /* @__PURE__ */ jsx(
1452
+ Button,
1453
+ {
1454
+ variant: "transparent",
1455
+ size: "small",
1456
+ onClick: () => openDeletePrompt(materialCost.id, materialCost.name),
1457
+ children: /* @__PURE__ */ jsx(Trash, {})
1458
+ }
1459
+ )
1460
+ ] }) })
1461
+ ] }, materialCost.id)) })
1462
+ ] })
1463
+ ] }),
1464
+ isModalOpen && /* @__PURE__ */ jsx(
1465
+ MaterialCostModal,
1466
+ {
1467
+ materialCost: editingMaterialCost,
1468
+ onClose: handleModalClose
1469
+ }
1470
+ ),
1471
+ /* @__PURE__ */ jsx(
1472
+ Prompt,
1473
+ {
1474
+ variant: "confirmation",
1475
+ open: !!deleteMaterialCostId,
1476
+ onOpenChange: () => {
1477
+ setDeleteMaterialCostId(null);
1478
+ setDeleteMaterialCostName("");
1479
+ },
1480
+ children: /* @__PURE__ */ jsxs(Prompt.Content, { children: [
1481
+ /* @__PURE__ */ jsxs(Prompt.Header, { children: [
1482
+ /* @__PURE__ */ jsx(Prompt.Title, { children: "ลบรายการต้นทุนวัสดุ" }),
1483
+ /* @__PURE__ */ jsxs(Prompt.Description, { children: [
1484
+ 'คุณต้องการลบวัสดุ "',
1485
+ deleteMaterialCostName,
1486
+ '" ใช่หรือไม่? การดำเนินการนี้ไม่สามารถย้อนกลับได้'
1487
+ ] })
1488
+ ] }),
1489
+ /* @__PURE__ */ jsxs(Prompt.Footer, { children: [
1490
+ /* @__PURE__ */ jsx(Prompt.Cancel, { children: "ยกเลิก" }),
1491
+ /* @__PURE__ */ jsx(Prompt.Action, { onClick: handleDelete, children: "ลบ" })
1492
+ ] })
1493
+ ] })
1494
+ }
1495
+ )
1496
+ ] });
1497
+ };
1498
+ const MaterialCostsPage = () => {
1499
+ return /* @__PURE__ */ jsx(Container, { children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
1500
+ /* @__PURE__ */ jsx("div", { className: "flex items-center justify-between", children: /* @__PURE__ */ jsx(Heading, { level: "h1", children: "ต้นทุนวัสดุ" }) }),
1501
+ /* @__PURE__ */ jsx(MaterialCostsTable, {})
1502
+ ] }) });
1503
+ };
1504
+ const config$1 = defineRouteConfig({
1505
+ icon: CurrencyDollarSolid,
1506
+ label: "ต้นทุนวัสดุ"
1507
+ });
1508
+ const CARRIER_TYPES$1 = [
1509
+ { value: "COMPANY_FLEET", label: "บริษัทรถส่งของ (Messenger)" },
1510
+ { value: "COMPANY_TRUCK", label: "รถบริษัท (Same Day)" },
1511
+ { value: "PRIVATE_CARRIER", label: "ขนส่งเอกชน (EMS/Express)" }
1512
+ ];
1513
+ const SERVICE_CODES$1 = [
1514
+ { value: "MESSENGER_3H", label: "ส่งด่วน 3 ชม.", usesHours: true },
1515
+ { value: "SAME_DAY", label: "ส่งด่วนภายในวัน", usesHours: true },
1516
+ { value: "STANDARD_3_5D", label: "EMS 3-5 วัน", usesHours: false },
1517
+ { value: "EXPRESS_1_2D", label: "Express 1-2 วัน", usesHours: false }
1518
+ ];
1519
+ const ShippingRateModal = ({ rate, onClose }) => {
1520
+ const [formData, setFormData] = useState({
1521
+ carrier_type: "COMPANY_FLEET",
1522
+ service_code: "MESSENGER_3H",
1523
+ max_weight_kg: "",
1524
+ price: "",
1525
+ currency: "THB",
1526
+ priority: "0",
1527
+ eta_hours_min: "",
1528
+ eta_hours_max: "",
1529
+ eta_days_min: "",
1530
+ eta_days_max: "",
1531
+ active: true
1532
+ });
1533
+ const [isSaving, setIsSaving] = useState(false);
1534
+ const selectedService = SERVICE_CODES$1.find((s) => s.value === formData.service_code);
1535
+ const usesHours = (selectedService == null ? void 0 : selectedService.usesHours) ?? false;
1536
+ useEffect(() => {
1537
+ var _a, _b, _c, _d;
1538
+ if (rate) {
1539
+ setFormData({
1540
+ carrier_type: rate.carrier_type,
1541
+ service_code: rate.service_code,
1542
+ max_weight_kg: rate.max_weight_kg.toString(),
1543
+ price: rate.price.toString(),
1544
+ currency: rate.currency,
1545
+ priority: (rate.priority ?? 0).toString(),
1546
+ eta_hours_min: ((_a = rate.eta_hours_min) == null ? void 0 : _a.toString()) || "",
1547
+ eta_hours_max: ((_b = rate.eta_hours_max) == null ? void 0 : _b.toString()) || "",
1548
+ eta_days_min: ((_c = rate.eta_days_min) == null ? void 0 : _c.toString()) || "",
1549
+ eta_days_max: ((_d = rate.eta_days_max) == null ? void 0 : _d.toString()) || "",
1550
+ active: rate.active
1551
+ });
1552
+ }
1553
+ }, [rate]);
1554
+ const handleSubmit = async (e) => {
1555
+ e.preventDefault();
1556
+ const errors = [];
1557
+ const maxWeight = parseFloat(formData.max_weight_kg);
1558
+ const price = parseFloat(formData.price);
1559
+ const priority = parseInt(formData.priority);
1560
+ if (isNaN(maxWeight) || maxWeight <= 0) {
1561
+ errors.push("น้ำหนักสูงสุดต้องเป็นตัวเลขมากกว่า 0");
1562
+ }
1563
+ if (isNaN(price) || price < 0) {
1564
+ errors.push("ราคาต้องเป็นตัวเลขไม่ติดลบ");
1565
+ }
1566
+ if (isNaN(priority)) {
1567
+ errors.push("ลำดับความสำคัญต้องเป็นตัวเลข");
1568
+ }
1569
+ let etaHoursMin;
1570
+ let etaHoursMax;
1571
+ let etaDaysMin;
1572
+ let etaDaysMax;
1573
+ if (usesHours) {
1574
+ etaHoursMin = formData.eta_hours_min ? parseFloat(formData.eta_hours_min) : void 0;
1575
+ etaHoursMax = formData.eta_hours_max ? parseFloat(formData.eta_hours_max) : void 0;
1576
+ if (etaHoursMin === void 0 || etaHoursMax === void 0) {
1577
+ errors.push("กรุณากรอก ETA เป็นชั่วโมงให้ครบ");
1578
+ } else if (etaHoursMin > etaHoursMax) {
1579
+ errors.push("ETA ชั่วโมงต่ำสุดต้องน้อยกว่าหรือเท่ากับสูงสุด");
1580
+ }
1581
+ } else {
1582
+ etaDaysMin = formData.eta_days_min ? parseFloat(formData.eta_days_min) : void 0;
1583
+ etaDaysMax = formData.eta_days_max ? parseFloat(formData.eta_days_max) : void 0;
1584
+ if (etaDaysMin === void 0 || etaDaysMax === void 0) {
1585
+ errors.push("กรุณากรอก ETA เป็นวันให้ครบ");
1586
+ } else if (etaDaysMin > etaDaysMax) {
1587
+ errors.push("ETA วันต่ำสุดต้องน้อยกว่าหรือเท่ากับสูงสุด");
1588
+ }
1589
+ }
1590
+ if (errors.length > 0) {
1591
+ toast.error("ข้อมูลไม่ถูกต้อง", {
1592
+ description: errors.join(", ")
1593
+ });
1594
+ return;
1595
+ }
1596
+ setIsSaving(true);
1597
+ try {
1598
+ const payload = {
1599
+ carrier_type: formData.carrier_type,
1600
+ service_code: formData.service_code,
1601
+ max_weight_kg: maxWeight,
1602
+ price,
1603
+ currency: formData.currency,
1604
+ priority,
1605
+ active: formData.active
1606
+ };
1607
+ if (usesHours) {
1608
+ payload.eta_hours_min = etaHoursMin;
1609
+ payload.eta_hours_max = etaHoursMax;
1610
+ } else {
1611
+ payload.eta_days_min = etaDaysMin;
1612
+ payload.eta_days_max = etaDaysMax;
1613
+ }
1614
+ const url = rate ? `/admin/shipping-rates/${rate.id}` : "/admin/shipping-rates";
1615
+ const method = rate ? "PUT" : "POST";
1616
+ const response = await fetch(url, {
1617
+ method,
1618
+ headers: { "Content-Type": "application/json" },
1619
+ credentials: "include",
1620
+ body: JSON.stringify(payload)
1621
+ });
1622
+ if (response.ok) {
1623
+ toast.success("สำเร็จ", {
1624
+ description: rate ? "แก้ไขอัตราค่าขนส่งแล้ว" : "สร้างอัตราค่าขนส่งใหม่แล้ว"
1625
+ });
1626
+ onClose();
1627
+ } else {
1628
+ const error = await response.json();
1629
+ throw new Error(error.message || "Failed to save");
1630
+ }
1631
+ } catch (error) {
1632
+ toast.error("ข้อผิดพลาด", {
1633
+ description: error instanceof Error ? error.message : "ไม่สามารถบันทึกข้อมูลได้"
1634
+ });
1635
+ } finally {
1636
+ setIsSaving(false);
1637
+ }
1638
+ };
1639
+ return /* @__PURE__ */ jsx(Drawer, { open: true, onOpenChange: (open) => !open && onClose(), children: /* @__PURE__ */ jsxs(Drawer.Content, { children: [
1640
+ /* @__PURE__ */ jsx(Drawer.Header, { children: /* @__PURE__ */ jsx(Drawer.Title, { children: rate ? "แก้ไขอัตราค่าขนส่ง" : "สร้างอัตราค่าขนส่งใหม่" }) }),
1641
+ /* @__PURE__ */ jsx(Drawer.Body, { children: /* @__PURE__ */ jsxs("form", { onSubmit: handleSubmit, className: "flex flex-col gap-y-4", children: [
1642
+ /* @__PURE__ */ jsxs("div", { children: [
1643
+ /* @__PURE__ */ jsx(Label, { htmlFor: "carrier", children: "ประเภทขนส่ง *" }),
1644
+ /* @__PURE__ */ jsxs(
1645
+ Select,
1646
+ {
1647
+ value: formData.carrier_type,
1648
+ onValueChange: (value) => setFormData({ ...formData, carrier_type: value }),
1649
+ children: [
1650
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "เลือกประเภทขนส่ง" }) }),
1651
+ /* @__PURE__ */ jsx(Select.Content, { children: CARRIER_TYPES$1.map((carrier) => /* @__PURE__ */ jsx(Select.Item, { value: carrier.value, children: carrier.label }, carrier.value)) })
1652
+ ]
1653
+ }
1654
+ )
1655
+ ] }),
1656
+ /* @__PURE__ */ jsxs("div", { children: [
1657
+ /* @__PURE__ */ jsx(Label, { htmlFor: "service", children: "บริการจัดส่ง *" }),
1658
+ /* @__PURE__ */ jsxs(
1659
+ Select,
1660
+ {
1661
+ value: formData.service_code,
1662
+ onValueChange: (value) => setFormData({
1663
+ ...formData,
1664
+ service_code: value,
1665
+ eta_hours_min: "",
1666
+ eta_hours_max: "",
1667
+ eta_days_min: "",
1668
+ eta_days_max: ""
1669
+ }),
1670
+ children: [
1671
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "เลือกบริการ" }) }),
1672
+ /* @__PURE__ */ jsx(Select.Content, { children: SERVICE_CODES$1.map((service) => /* @__PURE__ */ jsx(Select.Item, { value: service.value, children: service.label }, service.value)) })
1673
+ ]
1674
+ }
1675
+ )
1676
+ ] }),
1677
+ /* @__PURE__ */ jsxs("div", { className: "grid grid-cols-2 gap-x-4", children: [
1678
+ /* @__PURE__ */ jsxs("div", { children: [
1679
+ /* @__PURE__ */ jsx(Label, { htmlFor: "maxWeight", children: "น้ำหนักสูงสุด (kg) *" }),
1680
+ /* @__PURE__ */ jsx(
1681
+ Input,
1682
+ {
1683
+ id: "maxWeight",
1684
+ type: "number",
1685
+ step: "0.1",
1686
+ min: "0",
1687
+ value: formData.max_weight_kg,
1688
+ onChange: (e) => setFormData({ ...formData, max_weight_kg: e.target.value }),
1689
+ placeholder: "เช่น 3",
1690
+ required: true
1691
+ }
1692
+ )
1693
+ ] }),
1694
+ /* @__PURE__ */ jsxs("div", { children: [
1695
+ /* @__PURE__ */ jsx(Label, { htmlFor: "price", children: "ราคา (THB) *" }),
1696
+ /* @__PURE__ */ jsx(
1697
+ Input,
1698
+ {
1699
+ id: "price",
1700
+ type: "number",
1701
+ step: "0.01",
1702
+ min: "0",
1703
+ value: formData.price,
1704
+ onChange: (e) => setFormData({ ...formData, price: e.target.value }),
1705
+ placeholder: "เช่น 80",
1706
+ required: true
1707
+ }
1708
+ )
1709
+ ] })
1710
+ ] }),
1711
+ /* @__PURE__ */ jsxs("div", { children: [
1712
+ /* @__PURE__ */ jsx(Label, { htmlFor: "priority", children: "ลำดับความสำคัญ" }),
1713
+ /* @__PURE__ */ jsx(
1714
+ Input,
1715
+ {
1716
+ id: "priority",
1717
+ type: "number",
1718
+ value: formData.priority,
1719
+ onChange: (e) => setFormData({ ...formData, priority: e.target.value }),
1720
+ placeholder: "0"
1721
+ }
1722
+ ),
1723
+ /* @__PURE__ */ jsx("p", { className: "text-xs text-ui-fg-subtle mt-1", children: "ตัวเลขสูงกว่าจะถูกเสนอให้เลือกก่อน (ค่าเริ่มต้น 0)" })
1724
+ ] }),
1725
+ /* @__PURE__ */ jsxs("div", { children: [
1726
+ /* @__PURE__ */ jsxs(Label, { children: [
1727
+ usesHours ? "ETA (ชั่วโมง)" : "ETA (วัน)",
1728
+ " *"
1729
+ ] }),
1730
+ /* @__PURE__ */ jsx("div", { className: "grid grid-cols-2 gap-x-4 mt-2", children: usesHours ? /* @__PURE__ */ jsxs(Fragment, { children: [
1731
+ /* @__PURE__ */ jsx(
1732
+ Input,
1733
+ {
1734
+ type: "number",
1735
+ placeholder: "ต่ำสุด",
1736
+ value: formData.eta_hours_min,
1737
+ onChange: (e) => setFormData({ ...formData, eta_hours_min: e.target.value }),
1738
+ required: true
1739
+ }
1740
+ ),
1741
+ /* @__PURE__ */ jsx(
1742
+ Input,
1743
+ {
1744
+ type: "number",
1745
+ placeholder: "สูงสุด",
1746
+ value: formData.eta_hours_max,
1747
+ onChange: (e) => setFormData({ ...formData, eta_hours_max: e.target.value }),
1748
+ required: true
1749
+ }
1750
+ )
1751
+ ] }) : /* @__PURE__ */ jsxs(Fragment, { children: [
1752
+ /* @__PURE__ */ jsx(
1753
+ Input,
1754
+ {
1755
+ type: "number",
1756
+ placeholder: "ต่ำสุด",
1757
+ value: formData.eta_days_min,
1758
+ onChange: (e) => setFormData({ ...formData, eta_days_min: e.target.value }),
1759
+ required: true
1760
+ }
1761
+ ),
1762
+ /* @__PURE__ */ jsx(
1763
+ Input,
1764
+ {
1765
+ type: "number",
1766
+ placeholder: "สูงสุด",
1767
+ value: formData.eta_days_max,
1768
+ onChange: (e) => setFormData({ ...formData, eta_days_max: e.target.value }),
1769
+ required: true
1770
+ }
1771
+ )
1772
+ ] }) })
1773
+ ] }),
1774
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center gap-x-2", children: [
1775
+ /* @__PURE__ */ jsx(
1776
+ Switch,
1777
+ {
1778
+ id: "active",
1779
+ checked: formData.active,
1780
+ onCheckedChange: (checked) => setFormData({ ...formData, active: checked })
1781
+ }
1782
+ ),
1783
+ /* @__PURE__ */ jsx(Label, { htmlFor: "active", children: "เปิดใช้งาน" })
1784
+ ] }),
1785
+ /* @__PURE__ */ jsxs(Drawer.Footer, { children: [
1786
+ /* @__PURE__ */ jsx(Button, { type: "button", variant: "secondary", onClick: onClose, disabled: isSaving, children: "ยกเลิก" }),
1787
+ /* @__PURE__ */ jsx(Button, { type: "submit", disabled: isSaving, children: isSaving ? "กำลังบันทึก..." : "บันทึก" })
1788
+ ] })
1789
+ ] }) })
1790
+ ] }) });
1791
+ };
1792
+ const CARRIER_TYPES = [
1793
+ { value: "ALL", label: "ทั้งหมด" },
1794
+ { value: "COMPANY_FLEET", label: "บริษัทรถส่งของ (Messenger)" },
1795
+ { value: "COMPANY_TRUCK", label: "รถบริษัท (Same Day)" },
1796
+ { value: "PRIVATE_CARRIER", label: "ขนส่งเอกชน (EMS/Express)" }
1797
+ ];
1798
+ const SERVICE_CODES = [
1799
+ { value: "ALL", label: "ทั้งหมด" },
1800
+ { value: "MESSENGER_3H", label: "ส่งด่วน 3 ชม." },
1801
+ { value: "SAME_DAY", label: "ส่งด่วนภายในวัน" },
1802
+ { value: "STANDARD_3_5D", label: "EMS 3-5 วัน" },
1803
+ { value: "EXPRESS_1_2D", label: "Express 1-2 วัน" }
1804
+ ];
1805
+ const ShippingRatesTable = () => {
1806
+ const [rates, setRates] = useState([]);
1807
+ const [filteredRates, setFilteredRates] = useState([]);
1808
+ const [carrierFilter, setCarrierFilter] = useState("ALL");
1809
+ const [serviceFilter, setServiceFilter] = useState("ALL");
1810
+ const [isModalOpen, setIsModalOpen] = useState(false);
1811
+ const [editingRate, setEditingRate] = useState(null);
1812
+ const [isLoading, setIsLoading] = useState(false);
1813
+ const [deleteRateId, setDeleteRateId] = useState(null);
1814
+ useEffect(() => {
1815
+ fetchRates();
1816
+ }, []);
1817
+ useEffect(() => {
1818
+ let filtered = rates;
1819
+ if (carrierFilter !== "ALL") {
1820
+ filtered = filtered.filter((rate) => rate.carrier_type === carrierFilter);
1821
+ }
1822
+ if (serviceFilter !== "ALL") {
1823
+ filtered = filtered.filter((rate) => rate.service_code === serviceFilter);
1824
+ }
1825
+ setFilteredRates(filtered);
1826
+ }, [carrierFilter, serviceFilter, rates]);
1827
+ const fetchRates = async () => {
1828
+ setIsLoading(true);
1829
+ try {
1830
+ const response = await fetch("/admin/shipping-rates", {
1831
+ credentials: "include"
1832
+ });
1833
+ const data = await response.json();
1834
+ setRates(data.rates || []);
1835
+ } catch (error) {
1836
+ toast.error("ข้อผิดพลาด", {
1837
+ description: "ไม่สามารถโหลดข้อมูลอัตราค่าขนส่งได้"
1838
+ });
1839
+ } finally {
1840
+ setIsLoading(false);
1841
+ }
1842
+ };
1843
+ const handleDelete = async () => {
1844
+ if (!deleteRateId) return;
1845
+ try {
1846
+ const response = await fetch(`/admin/shipping-rates/${deleteRateId}`, {
1847
+ method: "DELETE",
1848
+ credentials: "include"
1849
+ });
1850
+ if (response.ok) {
1851
+ toast.success("สำเร็จ", {
1852
+ description: "ลบอัตราค่าขนส่งแล้ว"
1853
+ });
1854
+ fetchRates();
1855
+ } else {
1856
+ throw new Error("Failed to delete");
1857
+ }
1858
+ } catch (error) {
1859
+ toast.error("ข้อผิดพลาด", {
1860
+ description: "ไม่สามารถลบอัตราค่าขนส่งได้"
1861
+ });
1862
+ } finally {
1863
+ setDeleteRateId(null);
1864
+ }
1865
+ };
1866
+ const handleEdit = (rate) => {
1867
+ setEditingRate(rate);
1868
+ setIsModalOpen(true);
1869
+ };
1870
+ const handleCreateNew = () => {
1871
+ setEditingRate(null);
1872
+ setIsModalOpen(true);
1873
+ };
1874
+ const handleModalClose = () => {
1875
+ setIsModalOpen(false);
1876
+ setEditingRate(null);
1877
+ fetchRates();
1878
+ };
1879
+ const getCarrierLabel = (type) => {
1880
+ var _a;
1881
+ return ((_a = CARRIER_TYPES.find((c) => c.value === type)) == null ? void 0 : _a.label) || type;
1882
+ };
1883
+ const getServiceLabel = (code) => {
1884
+ var _a;
1885
+ return ((_a = SERVICE_CODES.find((s) => s.value === code)) == null ? void 0 : _a.label) || code;
1886
+ };
1887
+ const formatETA = (rate) => {
1888
+ if (rate.eta_hours_min !== null && rate.eta_hours_min !== void 0 && rate.eta_hours_max !== null && rate.eta_hours_max !== void 0) {
1889
+ return `${rate.eta_hours_min}-${rate.eta_hours_max} ชม.`;
1890
+ }
1891
+ if (rate.eta_days_min !== null && rate.eta_days_min !== void 0 && rate.eta_days_max !== null && rate.eta_days_max !== void 0) {
1892
+ return `${rate.eta_days_min}-${rate.eta_days_max} วัน`;
1893
+ }
1894
+ return "-";
1895
+ };
1896
+ return /* @__PURE__ */ jsxs(Fragment, { children: [
1897
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
1898
+ /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-between gap-x-4", children: [
1899
+ /* @__PURE__ */ jsxs("div", { className: "flex gap-x-4 items-end", children: [
1900
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-1", children: [
1901
+ /* @__PURE__ */ jsx("label", { className: "text-sm font-medium text-ui-fg-subtle", children: "ประเภทขนส่ง" }),
1902
+ /* @__PURE__ */ jsxs(Select, { value: carrierFilter, onValueChange: setCarrierFilter, children: [
1903
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "ทั้งหมด" }) }),
1904
+ /* @__PURE__ */ jsx(Select.Content, { children: CARRIER_TYPES.map((type) => /* @__PURE__ */ jsx(Select.Item, { value: type.value, children: type.label }, type.value)) })
1905
+ ] })
1906
+ ] }),
1907
+ /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-1", children: [
1908
+ /* @__PURE__ */ jsx("label", { className: "text-sm font-medium text-ui-fg-subtle", children: "บริการจัดส่ง" }),
1909
+ /* @__PURE__ */ jsxs(Select, { value: serviceFilter, onValueChange: setServiceFilter, children: [
1910
+ /* @__PURE__ */ jsx(Select.Trigger, { children: /* @__PURE__ */ jsx(Select.Value, { placeholder: "ทั้งหมด" }) }),
1911
+ /* @__PURE__ */ jsx(Select.Content, { children: SERVICE_CODES.map((service) => /* @__PURE__ */ jsx(Select.Item, { value: service.value, children: service.label }, service.value)) })
1912
+ ] })
1913
+ ] })
1914
+ ] }),
1915
+ /* @__PURE__ */ jsxs(Button, { onClick: handleCreateNew, children: [
1916
+ /* @__PURE__ */ jsx(Plus, {}),
1917
+ "สร้างอัตราใหม่"
1918
+ ] })
1919
+ ] }),
1920
+ /* @__PURE__ */ jsxs(Table, { children: [
1921
+ /* @__PURE__ */ jsx(Table.Header, { children: /* @__PURE__ */ jsxs(Table.Row, { children: [
1922
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ประเภทขนส่ง" }),
1923
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "บริการ" }),
1924
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "น้ำหนัก ≤ (kg)" }),
1925
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ราคา (THB)" }),
1926
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "ETA" }),
1927
+ /* @__PURE__ */ jsx(Table.HeaderCell, { children: "สถานะ" }),
1928
+ /* @__PURE__ */ jsx(Table.HeaderCell, { className: "text-right", children: "การจัดการ" })
1929
+ ] }) }),
1930
+ /* @__PURE__ */ jsx(Table.Body, { children: isLoading ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "กำลังโหลด..." }) }) : filteredRates.length === 0 ? /* @__PURE__ */ jsx(Table.Row, { children: /* @__PURE__ */ jsx(Table.Cell, { className: "text-center", children: "ไม่พบข้อมูลอัตราค่าขนส่ง" }) }) : filteredRates.map((rate) => /* @__PURE__ */ jsxs(Table.Row, { children: [
1931
+ /* @__PURE__ */ jsx(Table.Cell, { children: getCarrierLabel(rate.carrier_type) }),
1932
+ /* @__PURE__ */ jsx(Table.Cell, { children: getServiceLabel(rate.service_code) }),
1933
+ /* @__PURE__ */ jsxs(Table.Cell, { children: [
1934
+ "≤ ",
1935
+ rate.max_weight_kg,
1936
+ " kg"
1937
+ ] }),
1938
+ /* @__PURE__ */ jsx(Table.Cell, { children: rate.price.toFixed(2) }),
1939
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Badge, { color: "blue", children: formatETA(rate) }) }),
1940
+ /* @__PURE__ */ jsx(Table.Cell, { children: /* @__PURE__ */ jsx(Badge, { color: rate.active ? "green" : "grey", children: rate.active ? "เปิดใช้งาน" : "ปิดใช้งาน" }) }),
1941
+ /* @__PURE__ */ jsx(Table.Cell, { className: "text-right", children: /* @__PURE__ */ jsxs("div", { className: "flex items-center justify-end gap-x-2", children: [
1942
+ /* @__PURE__ */ jsx(
1943
+ Button,
1944
+ {
1945
+ variant: "transparent",
1946
+ size: "small",
1947
+ onClick: () => handleEdit(rate),
1948
+ children: /* @__PURE__ */ jsx(Edit, {})
1949
+ }
1950
+ ),
1951
+ /* @__PURE__ */ jsx(
1952
+ Button,
1953
+ {
1954
+ variant: "transparent",
1955
+ size: "small",
1956
+ onClick: () => setDeleteRateId(rate.id),
1957
+ children: /* @__PURE__ */ jsx(Trash, {})
1958
+ }
1959
+ )
1960
+ ] }) })
1961
+ ] }, rate.id)) })
1962
+ ] })
1963
+ ] }),
1964
+ isModalOpen && /* @__PURE__ */ jsx(ShippingRateModal, { rate: editingRate, onClose: handleModalClose }),
1965
+ /* @__PURE__ */ jsx(
1966
+ Prompt,
1967
+ {
1968
+ variant: "confirmation",
1969
+ open: !!deleteRateId,
1970
+ onOpenChange: () => setDeleteRateId(null),
1971
+ children: /* @__PURE__ */ jsxs(Prompt.Content, { children: [
1972
+ /* @__PURE__ */ jsxs(Prompt.Header, { children: [
1973
+ /* @__PURE__ */ jsx(Prompt.Title, { children: "ลบอัตราค่าขนส่ง" }),
1974
+ /* @__PURE__ */ jsx(Prompt.Description, { children: "คุณต้องการลบอัตราค่าขนส่งนี้ใช่หรือไม่? การดำเนินการนี้ไม่สามารถย้อนกลับได้" })
1975
+ ] }),
1976
+ /* @__PURE__ */ jsxs(Prompt.Footer, { children: [
1977
+ /* @__PURE__ */ jsx(Prompt.Cancel, { children: "ยกเลิก" }),
1978
+ /* @__PURE__ */ jsx(Prompt.Action, { onClick: handleDelete, children: "ลบ" })
1979
+ ] })
1980
+ ] })
1981
+ }
1982
+ )
1983
+ ] });
1984
+ };
1985
+ const RatesPage = () => {
1986
+ return /* @__PURE__ */ jsx(Container, { children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col gap-y-4", children: [
1987
+ /* @__PURE__ */ jsx("div", { className: "flex items-center justify-between", children: /* @__PURE__ */ jsx(Heading, { level: "h1", children: "อัตราค่าขนส่ง" }) }),
1988
+ /* @__PURE__ */ jsx(ShippingRatesTable, {})
1989
+ ] }) });
1990
+ };
1991
+ const config = defineRouteConfig({
1992
+ icon: HandTruck,
1993
+ label: "อัตราค่าขนส่ง"
1994
+ });
1995
+ const widgetModule = { widgets: [
1996
+ {
1997
+ Component: OrderShippingQuoteWidget,
1998
+ zone: ["order.details.after"]
1999
+ }
2000
+ ] };
2001
+ const routeModule = {
2002
+ routes: [
2003
+ {
2004
+ Component: AreasPage,
2005
+ path: "/areas"
2006
+ },
2007
+ {
2008
+ Component: BoxesPage,
2009
+ path: "/boxes"
2010
+ },
2011
+ {
2012
+ Component: MaterialCostsPage,
2013
+ path: "/material-costs"
2014
+ },
2015
+ {
2016
+ Component: RatesPage,
2017
+ path: "/rates"
2018
+ }
2019
+ ]
2020
+ };
2021
+ const menuItemModule = {
2022
+ menuItems: [
2023
+ {
2024
+ label: config$3.label,
2025
+ icon: config$3.icon,
2026
+ path: "/areas",
2027
+ nested: void 0
2028
+ },
2029
+ {
2030
+ label: config$2.label,
2031
+ icon: config$2.icon,
2032
+ path: "/boxes",
2033
+ nested: void 0
2034
+ },
2035
+ {
2036
+ label: config$1.label,
2037
+ icon: config$1.icon,
2038
+ path: "/material-costs",
2039
+ nested: void 0
2040
+ },
2041
+ {
2042
+ label: config.label,
2043
+ icon: config.icon,
2044
+ path: "/rates",
2045
+ nested: void 0
2046
+ }
2047
+ ]
2048
+ };
2049
+ const formModule = { customFields: {} };
2050
+ const displayModule = {
2051
+ displays: {}
2052
+ };
2053
+ const i18nModule = { resources: {} };
2054
+ const plugin = {
2055
+ widgetModule,
2056
+ routeModule,
2057
+ menuItemModule,
2058
+ formModule,
2059
+ displayModule,
2060
+ i18nModule
2061
+ };
2062
+ export {
2063
+ plugin as default
2064
+ };