feat: dashboard CRUD for Projects + Industrial Parks, listings delete, BĐS homepage card
Some checks failed
CodeQL Analysis / CodeQL (javascript-typescript) (push) Failing after 1m15s
Deploy / Build API Image (push) Failing after 20s
Deploy / Build AI Services Image (push) Failing after 12s
E2E Tests / Playwright E2E (push) Failing after 16s
Deploy / Deploy to Staging (push) Has been skipped
Deploy / Deploy to Production (push) Has been skipped
Deploy / Smoke Test Production (push) Has been skipped
Security Scanning / Trivy Scan — AI Services Image (push) Failing after 35s
Security Scanning / Trivy Filesystem Scan (push) Failing after 30s
Backup Verification / Backup Restore Verification (push) Failing after 14m37s
Security Scanning / Trivy Scan — API Image (push) Failing after 1m4s
Security Scanning / Trivy Scan — Web Image (push) Failing after 36s
Security Scanning / Dependency Audit (pnpm) (push) Failing after 11m6s
Deploy / Build Web Image (push) Failing after 12s
Deploy / Smoke Test Staging (push) Has been skipped
Deploy / Rollback Staging (push) Has been skipped
Deploy / Rollback Production (push) Has been skipped
CI / Lint → Typecheck → Test → Build (22) (push) Failing after 8s
CI / E2E Tests (push) Has been skipped
Security Scanning / Security Gate (push) Has been cancelled

Backend — DELETE endpoints (hard delete, ADMIN or owner):
- DELETE /projects/:id (Admin) — new DeleteProjectCommand/Handler,
  repository.delete() adapter, module wiring.
- DELETE /industrial/parks/:id (Admin) — same pattern.
- DELETE /listings/:id (JWT + owner-or-Admin check in handler).

Frontend — API clients:
- lib/du-an-api.ts: add create/update/delete + CreateProjectPayload,
  UpdateProjectPayload types.
- lib/khu-cong-nghiep-api.ts: add createPark/updatePark/deletePark +
  Create/Update payload types.
- lib/listings-api.ts: add delete().

Dashboard pages — new:
- /projects (Quản lý dự án): list with filters + edit/delete actions,
  /projects/new form (sectioned Cards, zod-validated), /projects/[id]/edit
  with danger-zone delete.
- /industrial-parks (Quản lý KCN): same triad. Fix occupancy-rate display
  (percentage already 0-100, no need to *100).

Dashboard listings page:
- Add Edit/Delete row actions with confirm + useMutation; error banner
  on mutation failure. Table view gains a "Thao tác" column; list view
  gains a footer action bar below each card.

Dashboard nav:
- Catalog group: /du-an → /projects (Quản lý dự án), /khu-cong-nghiep
  → /industrial-parks (Quản lý KCN). Desktop primaryNav updated too.

Public homepage:
- Add "Bất động sản" as a 5th feature card/tab → /search, using
  listingsApi for the "Featured listings" section.
- Bump grid to lg:grid-cols-5, update features subtitle copy ("Năm/Five
  core services").

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Ho Ngoc Hai
2026-04-19 10:37:33 +07:00
parent d2488b1cc1
commit ba0bf97426
32 changed files with 2843 additions and 22 deletions

View File

@@ -0,0 +1,551 @@
'use client';
import { zodResolver } from '@hookform/resolvers/zod';
import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
import Link from 'next/link';
import { useParams, useRouter } from 'next/navigation';
import * as React from 'react';
import { useForm } from 'react-hook-form';
import { z } from 'zod';
import { Button } from '@/components/ui/button';
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
import { Input } from '@/components/ui/input';
import { Label } from '@/components/ui/label';
import { Select } from '@/components/ui/select';
import { Textarea } from '@/components/ui/textarea';
import {
industrialApi,
PARK_STATUS_LABELS,
REGION_LABELS,
type IndustrialParkStatus,
type UpdateIndustrialParkPayload,
type VietnamRegion,
} from '@/lib/khu-cong-nghiep-api';
const STATUS_OPTIONS: IndustrialParkStatus[] = [
'PLANNING',
'UNDER_CONSTRUCTION',
'OPERATIONAL',
'FULL',
];
const REGION_OPTIONS: VietnamRegion[] = ['NORTH', 'CENTRAL', 'SOUTH'];
const optionalString = z
.string()
.optional()
.transform((v) => (v && v.trim() !== '' ? v.trim() : undefined));
const optionalNonNegativeNumber = z
.string()
.optional()
.refine(
(v) => v === undefined || v === '' || (!isNaN(Number(v)) && Number(v) >= 0),
'Phải là số không âm',
);
const editSchema = z.object({
name: optionalString,
nameEn: optionalString,
slug: optionalString,
developer: optionalString,
operator: optionalString,
status: z.enum(['PLANNING', 'UNDER_CONSTRUCTION', 'OPERATIONAL', 'FULL']),
address: optionalString,
district: optionalString,
province: optionalString,
region: z.enum(['NORTH', 'CENTRAL', 'SOUTH']),
latitude: z
.string()
.optional()
.refine(
(v) => v === undefined || v === '' || (!isNaN(Number(v)) && Number(v) >= -90 && Number(v) <= 90),
'Vĩ độ từ -90 đến 90',
),
longitude: z
.string()
.optional()
.refine(
(v) =>
v === undefined || v === '' || (!isNaN(Number(v)) && Number(v) >= -180 && Number(v) <= 180),
'Kinh độ từ -180 đến 180',
),
totalAreaHa: optionalNonNegativeNumber,
leasableAreaHa: optionalNonNegativeNumber,
landRentUsdM2Year: optionalNonNegativeNumber,
rbfRentUsdM2Month: optionalNonNegativeNumber,
rbwRentUsdM2Month: optionalNonNegativeNumber,
managementFeeUsd: optionalNonNegativeNumber,
targetIndustries: optionalString,
infrastructure: optionalString,
establishedYear: optionalNonNegativeNumber,
tenantCount: optionalNonNegativeNumber,
description: optionalString,
descriptionEn: optionalString,
});
type EditFormValues = z.input<typeof editSchema>;
function infraToText(infra: Record<string, unknown> | null | undefined): string {
if (!infra) return '';
return Object.entries(infra)
.map(([k, v]) => {
if (v === true) return k;
return `${k}: ${typeof v === 'string' ? v : JSON.stringify(v)}`;
})
.join('\n');
}
function parseInfrastructure(text: string | undefined): Record<string, unknown> | undefined {
if (!text) return undefined;
const lines = text.split('\n').map((l) => l.trim()).filter(Boolean);
if (lines.length === 0) return undefined;
const out: Record<string, unknown> = {};
for (const line of lines) {
const idx = line.indexOf(':');
if (idx > 0) {
const key = line.slice(0, idx).trim();
const value = line.slice(idx + 1).trim();
if (key) out[key] = value;
} else {
out[line] = true;
}
}
return Object.keys(out).length > 0 ? out : undefined;
}
function toNumOrUndef(v: string | undefined): number | undefined {
if (v === undefined || v === '') return undefined;
const n = Number(v);
return isNaN(n) ? undefined : n;
}
export default function EditIndustrialParkPage() {
const { id } = useParams<{ id: string }>();
const router = useRouter();
const queryClient = useQueryClient();
const [error, setError] = React.useState<string | null>(null);
const {
data: park,
isLoading,
isError,
} = useQuery({
queryKey: ['admin-industrial-park', id],
queryFn: () => industrialApi.getBySlug(id),
enabled: Boolean(id),
});
const {
register,
handleSubmit,
reset,
formState: { errors },
} = useForm<EditFormValues>({
resolver: zodResolver(editSchema),
mode: 'onTouched',
defaultValues: {
status: 'PLANNING',
region: 'NORTH',
},
});
React.useEffect(() => {
if (!park) return;
reset({
name: park.name,
nameEn: park.nameEn ?? '',
slug: park.slug,
developer: park.developer,
operator: park.operator ?? '',
status: park.status,
address: park.address,
district: park.district,
province: park.province,
region: park.region,
latitude: String(park.latitude),
longitude: String(park.longitude),
totalAreaHa: String(park.totalAreaHa),
leasableAreaHa: String(park.leasableAreaHa),
landRentUsdM2Year: park.landRentUsdM2Year != null ? String(park.landRentUsdM2Year) : '',
rbfRentUsdM2Month: park.rbfRentUsdM2Month != null ? String(park.rbfRentUsdM2Month) : '',
rbwRentUsdM2Month: park.rbwRentUsdM2Month != null ? String(park.rbwRentUsdM2Month) : '',
managementFeeUsd: park.managementFeeUsd != null ? String(park.managementFeeUsd) : '',
targetIndustries: park.targetIndustries.join(', '),
infrastructure: infraToText(park.infrastructure),
establishedYear: park.establishedYear != null ? String(park.establishedYear) : '',
tenantCount: String(park.tenantCount),
description: park.description ?? '',
descriptionEn: park.descriptionEn ?? '',
});
}, [park, reset]);
const updateMutation = useMutation({
mutationFn: (payload: UpdateIndustrialParkPayload) =>
industrialApi.updatePark(park?.id ?? id, payload),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['admin-industrial-parks'] });
queryClient.invalidateQueries({ queryKey: ['admin-industrial-park', id] });
router.push('/industrial-parks');
},
onError: (err: unknown) => {
setError(err instanceof Error ? err.message : 'Có lỗi xảy ra');
},
});
const deleteMutation = useMutation({
mutationFn: () => industrialApi.deletePark(park?.id ?? id),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['admin-industrial-parks'] });
router.push('/industrial-parks');
},
onError: (err: unknown) => {
setError(err instanceof Error ? err.message : 'Không thể xoá KCN');
},
});
const onSubmit = (data: EditFormValues) => {
setError(null);
const payload: UpdateIndustrialParkPayload = {
status: data.status,
};
if (data.name) payload.name = data.name;
if (data.nameEn !== undefined) payload.nameEn = data.nameEn;
if (data.developer) payload.developer = data.developer;
if (data.operator !== undefined) payload.operator = data.operator;
const landRent = toNumOrUndef(data.landRentUsdM2Year);
if (landRent != null) payload.landRentUsdM2Year = landRent;
const rbfRent = toNumOrUndef(data.rbfRentUsdM2Month);
if (rbfRent != null) payload.rbfRentUsdM2Month = rbfRent;
const rbwRent = toNumOrUndef(data.rbwRentUsdM2Month);
if (rbwRent != null) payload.rbwRentUsdM2Month = rbwRent;
const mgmtFee = toNumOrUndef(data.managementFeeUsd);
if (mgmtFee != null) payload.managementFeeUsd = mgmtFee;
const tenantCount = toNumOrUndef(data.tenantCount);
if (tenantCount != null) payload.tenantCount = tenantCount;
if (data.targetIndustries !== undefined) {
payload.targetIndustries = data.targetIndustries
? data.targetIndustries
.split(',')
.map((s) => s.trim())
.filter(Boolean)
: [];
}
const infra = parseInfrastructure(data.infrastructure);
if (infra) payload.infrastructure = infra;
if (data.description !== undefined) payload.description = data.description;
if (data.descriptionEn !== undefined) payload.descriptionEn = data.descriptionEn;
updateMutation.mutate(payload);
};
const handleDelete = () => {
if (!park) return;
if (!window.confirm(`Xoá KCN "${park.name}"? Thao tác này không thể hoàn tác.`)) return;
deleteMutation.mutate();
};
if (isLoading) {
return (
<div className="flex min-h-[400px] items-center justify-center">
<div className="h-8 w-8 animate-spin rounded-full border-4 border-primary border-t-transparent" />
</div>
);
}
if (isError || !park) {
return (
<div className="flex min-h-[400px] flex-col items-center justify-center space-y-4">
<p className="text-destructive">Không tìm thấy KCN</p>
<Button variant="outline" onClick={() => router.push('/industrial-parks')}>
Quay lại danh sách
</Button>
</div>
);
}
return (
<div className="mx-auto max-w-4xl space-y-4 sm:space-y-6">
<div className="flex items-center justify-between gap-3">
<div>
<Link
href="/industrial-parks"
className="text-sm text-muted-foreground hover:text-foreground"
>
Danh sách KCN
</Link>
<h1 className="mt-1 text-xl font-bold sm:text-2xl">Chỉnh sửa KCN</h1>
</div>
<Link href={`/khu-cong-nghiep/${park.slug}`} target="_blank" rel="noreferrer">
<Button variant="outline">Xem trang công khai</Button>
</Link>
</div>
{error && (
<div className="rounded-md border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
{error}
<button className="ml-2 font-medium underline" onClick={() => setError(null)}>
Đóng
</button>
</div>
)}
<form onSubmit={handleSubmit(onSubmit)} className="space-y-6">
{/* Thông tin cơ bản */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Thông tin bản</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div>
<Label htmlFor="name">Tên KCN</Label>
<Input id="name" {...register('name')} />
</div>
<div>
<Label htmlFor="nameEn">Tên tiếng Anh</Label>
<Input id="nameEn" {...register('nameEn')} />
</div>
<div>
<Label htmlFor="slug">Slug</Label>
<Input id="slug" {...register('slug')} disabled />
</div>
<div>
<Label htmlFor="status">Trạng thái</Label>
<Select id="status" {...register('status')}>
{STATUS_OPTIONS.map((s) => (
<option key={s} value={s}>
{PARK_STATUS_LABELS[s]}
</option>
))}
</Select>
</div>
<div>
<Label htmlFor="developer">Chủ đu </Label>
<Input id="developer" {...register('developer')} />
</div>
<div>
<Label htmlFor="operator">Đơn vị vận hành</Label>
<Input id="operator" {...register('operator')} />
</div>
<div className="sm:col-span-2">
<Label htmlFor="description"> tả</Label>
<Textarea id="description" rows={3} {...register('description')} />
</div>
<div className="sm:col-span-2">
<Label htmlFor="descriptionEn"> tả (Tiếng Anh)</Label>
<Textarea id="descriptionEn" rows={3} {...register('descriptionEn')} />
</div>
</CardContent>
</Card>
{/* Vị trí */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Vị trí</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div className="sm:col-span-2">
<Label htmlFor="address">Đa chỉ</Label>
<Input id="address" {...register('address')} disabled />
</div>
<div>
<Label htmlFor="district">Quận/Huyện</Label>
<Input id="district" {...register('district')} disabled />
</div>
<div>
<Label htmlFor="province">Tỉnh/Thành phố</Label>
<Input id="province" {...register('province')} disabled />
</div>
<div>
<Label htmlFor="region">Vùng</Label>
<Select id="region" {...register('region')} disabled>
{REGION_OPTIONS.map((r) => (
<option key={r} value={r}>
{REGION_LABELS[r]}
</option>
))}
</Select>
</div>
<div className="grid grid-cols-2 gap-4">
<div>
<Label htmlFor="latitude"> đ</Label>
<Input
id="latitude"
type="number"
step="any"
{...register('latitude')}
disabled
/>
</div>
<div>
<Label htmlFor="longitude">Kinh đ</Label>
<Input
id="longitude"
type="number"
step="any"
{...register('longitude')}
disabled
/>
</div>
</div>
<p className="text-xs text-muted-foreground sm:col-span-2">
Thông tin vị trí đưc khoá sau khi tạo. Vui lòng tạo KCN mới nếu cần thay đi.
</p>
</CardContent>
</Card>
{/* Quy mô & giá thuê */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Quy & giá thuê</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div>
<Label htmlFor="totalAreaHa">Tổng diện tích (ha)</Label>
<Input
id="totalAreaHa"
type="number"
step="any"
{...register('totalAreaHa')}
disabled
/>
</div>
<div>
<Label htmlFor="leasableAreaHa">Diện tích cho thuê (ha)</Label>
<Input
id="leasableAreaHa"
type="number"
step="any"
{...register('leasableAreaHa')}
disabled
/>
</div>
<div>
<Label htmlFor="landRentUsdM2Year">Giá thuê đt (USD/m²/năm)</Label>
<Input
id="landRentUsdM2Year"
type="number"
step="any"
{...register('landRentUsdM2Year')}
/>
{errors.landRentUsdM2Year && (
<p className="mt-1 text-xs text-destructive">
{errors.landRentUsdM2Year.message}
</p>
)}
</div>
<div>
<Label htmlFor="rbfRentUsdM2Month">Giá thuê nhà xưởng (USD/m²/tháng)</Label>
<Input
id="rbfRentUsdM2Month"
type="number"
step="any"
{...register('rbfRentUsdM2Month')}
/>
</div>
<div>
<Label htmlFor="rbwRentUsdM2Month">Giá thuê kho (USD/m²/tháng)</Label>
<Input
id="rbwRentUsdM2Month"
type="number"
step="any"
{...register('rbwRentUsdM2Month')}
/>
</div>
<div>
<Label htmlFor="managementFeeUsd">Phí quản (USD)</Label>
<Input
id="managementFeeUsd"
type="number"
step="any"
{...register('managementFeeUsd')}
/>
</div>
<div>
<Label htmlFor="establishedYear">Năm thành lập</Label>
<Input
id="establishedYear"
type="number"
{...register('establishedYear')}
disabled
/>
</div>
<div>
<Label htmlFor="tenantCount">Số khách thuê</Label>
<Input id="tenantCount" type="number" {...register('tenantCount')} />
</div>
</CardContent>
</Card>
{/* Tiện ích & ngành nghề */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Tiện ích & ngành nghề</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div>
<Label htmlFor="targetIndustries">Ngành nghề mục tiêu</Label>
<Textarea
id="targetIndustries"
rows={2}
placeholder="Phân tách bằng dấu phẩy"
{...register('targetIndustries')}
/>
</div>
<div>
<Label htmlFor="infrastructure">Hạ tầng & tiện ích</Label>
<Textarea
id="infrastructure"
rows={4}
placeholder={'Mỗi dòng một mục, dạng "key: value"'}
{...register('infrastructure')}
/>
</div>
</CardContent>
</Card>
<div className="flex justify-end gap-3">
<Link href="/industrial-parks">
<Button type="button" variant="outline">
Huỷ
</Button>
</Link>
<Button type="submit" disabled={updateMutation.isPending}>
{updateMutation.isPending ? 'Đang lưu...' : 'Lưu thay đổi'}
</Button>
</div>
</form>
{/* Danger zone */}
<Card className="border-destructive/50">
<CardHeader>
<CardTitle className="text-lg text-destructive">Vùng nguy hiểm</CardTitle>
</CardHeader>
<CardContent className="flex items-center justify-between gap-3">
<p className="text-sm text-muted-foreground">
Xoá KCN sẽ gỡ khỏi danh sách công khai không thể hoàn tác.
</p>
<Button
type="button"
variant="destructive"
onClick={handleDelete}
disabled={deleteMutation.isPending}
>
{deleteMutation.isPending ? 'Đang xoá...' : 'Xoá KCN'}
</Button>
</CardContent>
</Card>
</div>
);
}

View File

@@ -0,0 +1,461 @@
'use client';
import { zodResolver } from '@hookform/resolvers/zod';
import { useMutation, useQueryClient } from '@tanstack/react-query';
import Link from 'next/link';
import { useRouter } from 'next/navigation';
import * as React from 'react';
import { useForm } from 'react-hook-form';
import { z } from 'zod';
import { Button } from '@/components/ui/button';
import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/card';
import { Input } from '@/components/ui/input';
import { Label } from '@/components/ui/label';
import { Select } from '@/components/ui/select';
import { Textarea } from '@/components/ui/textarea';
import {
industrialApi,
PARK_STATUS_LABELS,
REGION_LABELS,
type CreateIndustrialParkPayload,
type IndustrialParkStatus,
type VietnamRegion,
} from '@/lib/khu-cong-nghiep-api';
const STATUS_OPTIONS: IndustrialParkStatus[] = [
'PLANNING',
'UNDER_CONSTRUCTION',
'OPERATIONAL',
'FULL',
];
const REGION_OPTIONS: VietnamRegion[] = ['NORTH', 'CENTRAL', 'SOUTH'];
const optionalString = z
.string()
.optional()
.transform((v) => (v && v.trim() !== '' ? v.trim() : undefined));
const requiredString = (msg: string) => z.string().min(1, msg);
const nonNegativeNumber = (msg: string) =>
z
.string()
.min(1, msg)
.refine((v) => !isNaN(Number(v)) && Number(v) >= 0, 'Phải là số không âm');
const optionalNonNegativeNumber = z
.string()
.optional()
.refine(
(v) => v === undefined || v === '' || (!isNaN(Number(v)) && Number(v) >= 0),
'Phải là số không âm',
);
const parkFormSchema = z.object({
name: requiredString('Vui lòng nhập tên KCN'),
nameEn: optionalString,
slug: requiredString('Vui lòng nhập slug'),
developer: requiredString('Vui lòng nhập chủ đầu tư'),
operator: optionalString,
status: z.enum(['PLANNING', 'UNDER_CONSTRUCTION', 'OPERATIONAL', 'FULL']),
address: requiredString('Vui lòng nhập địa chỉ'),
district: requiredString('Vui lòng nhập quận/huyện'),
province: requiredString('Vui lòng nhập tỉnh/thành'),
region: z.enum(['NORTH', 'CENTRAL', 'SOUTH']),
latitude: z
.string()
.min(1, 'Vui lòng nhập vĩ độ')
.refine((v) => !isNaN(Number(v)) && Number(v) >= -90 && Number(v) <= 90, 'Vĩ độ từ -90 đến 90'),
longitude: z
.string()
.min(1, 'Vui lòng nhập kinh độ')
.refine(
(v) => !isNaN(Number(v)) && Number(v) >= -180 && Number(v) <= 180,
'Kinh độ từ -180 đến 180',
),
totalAreaHa: nonNegativeNumber('Vui lòng nhập tổng diện tích'),
leasableAreaHa: nonNegativeNumber('Vui lòng nhập diện tích cho thuê'),
landRentUsdM2Year: optionalNonNegativeNumber,
rbfRentUsdM2Month: optionalNonNegativeNumber,
rbwRentUsdM2Month: optionalNonNegativeNumber,
managementFeeUsd: optionalNonNegativeNumber,
targetIndustries: optionalString,
infrastructure: optionalString,
establishedYear: optionalNonNegativeNumber,
tenantCount: optionalNonNegativeNumber,
description: optionalString,
descriptionEn: optionalString,
});
type ParkFormValues = z.input<typeof parkFormSchema>;
function parseInfrastructure(text: string | undefined): Record<string, unknown> | undefined {
if (!text) return undefined;
const lines = text.split('\n').map((l) => l.trim()).filter(Boolean);
if (lines.length === 0) return undefined;
const out: Record<string, unknown> = {};
for (const line of lines) {
const idx = line.indexOf(':');
if (idx > 0) {
const key = line.slice(0, idx).trim();
const value = line.slice(idx + 1).trim();
if (key) out[key] = value;
} else {
out[line] = true;
}
}
return Object.keys(out).length > 0 ? out : undefined;
}
function toNumOrUndef(v: string | undefined): number | undefined {
if (v === undefined || v === '') return undefined;
const n = Number(v);
return isNaN(n) ? undefined : n;
}
export default function CreateIndustrialParkPage() {
const router = useRouter();
const queryClient = useQueryClient();
const [error, setError] = React.useState<string | null>(null);
const {
register,
handleSubmit,
formState: { errors },
} = useForm<ParkFormValues>({
resolver: zodResolver(parkFormSchema),
mode: 'onTouched',
defaultValues: {
status: 'PLANNING',
region: 'NORTH',
},
});
const mutation = useMutation({
mutationFn: (payload: CreateIndustrialParkPayload) => industrialApi.createPark(payload),
onSuccess: () => {
queryClient.invalidateQueries({ queryKey: ['admin-industrial-parks'] });
router.push('/industrial-parks');
},
onError: (err: unknown) => {
setError(err instanceof Error ? err.message : 'Có lỗi xảy ra');
},
});
const onSubmit = (data: ParkFormValues) => {
setError(null);
const totalArea = Number(data.totalAreaHa);
const leasableArea = Number(data.leasableAreaHa);
const industries = data.targetIndustries
? data.targetIndustries
.split(',')
.map((s) => s.trim())
.filter(Boolean)
: [];
const payload: CreateIndustrialParkPayload = {
name: data.name!,
slug: data.slug!,
developer: data.developer!,
status: data.status,
address: data.address!,
district: data.district!,
province: data.province!,
region: data.region,
latitude: Number(data.latitude),
longitude: Number(data.longitude),
totalAreaHa: totalArea,
leasableAreaHa: leasableArea,
occupancyRate: 0,
remainingAreaHa: leasableArea,
targetIndustries: industries,
};
if (data.nameEn) payload.nameEn = data.nameEn;
if (data.operator) payload.operator = data.operator;
const landRent = toNumOrUndef(data.landRentUsdM2Year);
if (landRent != null) payload.landRentUsdM2Year = landRent;
const rbfRent = toNumOrUndef(data.rbfRentUsdM2Month);
if (rbfRent != null) payload.rbfRentUsdM2Month = rbfRent;
const rbwRent = toNumOrUndef(data.rbwRentUsdM2Month);
if (rbwRent != null) payload.rbwRentUsdM2Month = rbwRent;
const mgmtFee = toNumOrUndef(data.managementFeeUsd);
if (mgmtFee != null) payload.managementFeeUsd = mgmtFee;
const year = toNumOrUndef(data.establishedYear);
if (year != null) payload.establishedYear = year;
const tenantCount = toNumOrUndef(data.tenantCount);
if (tenantCount != null) payload.tenantCount = tenantCount;
const infra = parseInfrastructure(data.infrastructure);
if (infra) payload.infrastructure = infra;
if (data.description) payload.description = data.description;
if (data.descriptionEn) payload.descriptionEn = data.descriptionEn;
mutation.mutate(payload);
};
return (
<div className="mx-auto max-w-4xl space-y-4 sm:space-y-6">
<div className="flex items-center justify-between gap-3">
<div>
<Link
href="/industrial-parks"
className="text-sm text-muted-foreground hover:text-foreground"
>
Danh sách KCN
</Link>
<h1 className="mt-1 text-xl font-bold sm:text-2xl">Thêm KCN</h1>
</div>
</div>
{error && (
<div className="rounded-md border border-destructive/50 bg-destructive/10 p-3 text-sm text-destructive">
{error}
<button className="ml-2 font-medium underline" onClick={() => setError(null)}>
Đóng
</button>
</div>
)}
<form onSubmit={handleSubmit(onSubmit)} className="space-y-6">
{/* Thông tin cơ bản */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Thông tin bản</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div>
<Label htmlFor="name">Tên KCN *</Label>
<Input id="name" {...register('name')} />
{errors.name && (
<p className="mt-1 text-xs text-destructive">{errors.name.message}</p>
)}
</div>
<div>
<Label htmlFor="nameEn">Tên tiếng Anh</Label>
<Input id="nameEn" {...register('nameEn')} />
</div>
<div>
<Label htmlFor="slug">Slug *</Label>
<Input id="slug" {...register('slug')} placeholder="vd: kcn-vsip-1" />
{errors.slug && (
<p className="mt-1 text-xs text-destructive">{errors.slug.message}</p>
)}
</div>
<div>
<Label htmlFor="status">Trạng thái *</Label>
<Select id="status" {...register('status')}>
{STATUS_OPTIONS.map((s) => (
<option key={s} value={s}>
{PARK_STATUS_LABELS[s]}
</option>
))}
</Select>
</div>
<div>
<Label htmlFor="developer">Chủ đu *</Label>
<Input id="developer" {...register('developer')} />
{errors.developer && (
<p className="mt-1 text-xs text-destructive">{errors.developer.message}</p>
)}
</div>
<div>
<Label htmlFor="operator">Đơn vị vận hành</Label>
<Input id="operator" {...register('operator')} />
</div>
<div className="sm:col-span-2">
<Label htmlFor="description"> tả</Label>
<Textarea id="description" rows={3} {...register('description')} />
</div>
<div className="sm:col-span-2">
<Label htmlFor="descriptionEn"> tả (Tiếng Anh)</Label>
<Textarea id="descriptionEn" rows={3} {...register('descriptionEn')} />
</div>
</CardContent>
</Card>
{/* Vị trí */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Vị trí</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div className="sm:col-span-2">
<Label htmlFor="address">Đa chỉ *</Label>
<Input id="address" {...register('address')} />
{errors.address && (
<p className="mt-1 text-xs text-destructive">{errors.address.message}</p>
)}
</div>
<div>
<Label htmlFor="district">Quận/Huyện *</Label>
<Input id="district" {...register('district')} />
{errors.district && (
<p className="mt-1 text-xs text-destructive">{errors.district.message}</p>
)}
</div>
<div>
<Label htmlFor="province">Tỉnh/Thành phố *</Label>
<Input id="province" {...register('province')} />
{errors.province && (
<p className="mt-1 text-xs text-destructive">{errors.province.message}</p>
)}
</div>
<div>
<Label htmlFor="region">Vùng *</Label>
<Select id="region" {...register('region')}>
{REGION_OPTIONS.map((r) => (
<option key={r} value={r}>
{REGION_LABELS[r]}
</option>
))}
</Select>
</div>
<div className="grid grid-cols-2 gap-4">
<div>
<Label htmlFor="latitude"> đ *</Label>
<Input id="latitude" type="number" step="any" {...register('latitude')} />
{errors.latitude && (
<p className="mt-1 text-xs text-destructive">{errors.latitude.message}</p>
)}
</div>
<div>
<Label htmlFor="longitude">Kinh đ *</Label>
<Input id="longitude" type="number" step="any" {...register('longitude')} />
{errors.longitude && (
<p className="mt-1 text-xs text-destructive">{errors.longitude.message}</p>
)}
</div>
</div>
</CardContent>
</Card>
{/* Quy mô & giá thuê */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Quy & giá thuê</CardTitle>
</CardHeader>
<CardContent className="grid gap-4 sm:grid-cols-2">
<div>
<Label htmlFor="totalAreaHa">Tổng diện tích (ha) *</Label>
<Input
id="totalAreaHa"
type="number"
step="any"
{...register('totalAreaHa')}
/>
{errors.totalAreaHa && (
<p className="mt-1 text-xs text-destructive">{errors.totalAreaHa.message}</p>
)}
</div>
<div>
<Label htmlFor="leasableAreaHa">Diện tích cho thuê (ha) *</Label>
<Input
id="leasableAreaHa"
type="number"
step="any"
{...register('leasableAreaHa')}
/>
{errors.leasableAreaHa && (
<p className="mt-1 text-xs text-destructive">{errors.leasableAreaHa.message}</p>
)}
</div>
<div>
<Label htmlFor="landRentUsdM2Year">Giá thuê đt (USD/m²/năm)</Label>
<Input
id="landRentUsdM2Year"
type="number"
step="any"
{...register('landRentUsdM2Year')}
/>
</div>
<div>
<Label htmlFor="rbfRentUsdM2Month">Giá thuê nhà xưởng (USD/m²/tháng)</Label>
<Input
id="rbfRentUsdM2Month"
type="number"
step="any"
{...register('rbfRentUsdM2Month')}
/>
</div>
<div>
<Label htmlFor="rbwRentUsdM2Month">Giá thuê kho (USD/m²/tháng)</Label>
<Input
id="rbwRentUsdM2Month"
type="number"
step="any"
{...register('rbwRentUsdM2Month')}
/>
</div>
<div>
<Label htmlFor="managementFeeUsd">Phí quản (USD)</Label>
<Input
id="managementFeeUsd"
type="number"
step="any"
{...register('managementFeeUsd')}
/>
</div>
<div>
<Label htmlFor="establishedYear">Năm thành lập</Label>
<Input id="establishedYear" type="number" {...register('establishedYear')} />
</div>
<div>
<Label htmlFor="tenantCount">Số khách thuê</Label>
<Input id="tenantCount" type="number" {...register('tenantCount')} />
</div>
</CardContent>
</Card>
{/* Tiện ích & ngành nghề */}
<Card>
<CardHeader>
<CardTitle className="text-lg">Tiện ích & ngành nghề</CardTitle>
</CardHeader>
<CardContent className="space-y-4">
<div>
<Label htmlFor="targetIndustries">Ngành nghề mục tiêu</Label>
<Textarea
id="targetIndustries"
rows={2}
placeholder="Phân tách bằng dấu phẩy, vd: Điện tử, Dệt may, Logistics"
{...register('targetIndustries')}
/>
</div>
<div>
<Label htmlFor="infrastructure">Hạ tầng & tiện ích</Label>
<Textarea
id="infrastructure"
rows={4}
placeholder={'Mỗi dòng một mục, dạng "key: value"\nvd:\nĐiện: 22kV\nNước: 5000 m³/ngày'}
{...register('infrastructure')}
/>
</div>
</CardContent>
</Card>
<div className="flex justify-end gap-3">
<Link href="/industrial-parks">
<Button type="button" variant="outline">
Huỷ
</Button>
</Link>
<Button type="submit" disabled={mutation.isPending}>
{mutation.isPending ? 'Đang lưu...' : 'Tạo KCN'}
</Button>
</div>
</form>
</div>
);
}

View File

@@ -0,0 +1,279 @@
'use client';
import { useMutation, useQuery, useQueryClient } from '@tanstack/react-query';
import { ExternalLink, Pencil, Trash2 } from 'lucide-react';
import Link from 'next/link';
import * as React from 'react';
import { Badge } from '@/components/ui/badge';
import { Button } from '@/components/ui/button';
import { Card, CardContent } from '@/components/ui/card';
import { Input } from '@/components/ui/input';
import { Select } from '@/components/ui/select';
import {
industrialApi,
PARK_STATUS_COLORS,
PARK_STATUS_LABELS,
REGION_LABELS,
type IndustrialParkStatus,
type SearchIndustrialParksParams,
type VietnamRegion,
} from '@/lib/khu-cong-nghiep-api';
const STATUS_OPTIONS: IndustrialParkStatus[] = [
'PLANNING',
'UNDER_CONSTRUCTION',
'OPERATIONAL',
'FULL',
];
const REGION_OPTIONS: VietnamRegion[] = ['NORTH', 'CENTRAL', 'SOUTH'];
interface FiltersState {
q: string;
province: string;
status: IndustrialParkStatus | '';
region: VietnamRegion | '';
page: number;
}
const INITIAL_FILTERS: FiltersState = {
q: '',
province: '',
status: '',
region: '',
page: 1,
};
export default function IndustrialParksListPage() {
const [filters, setFilters] = React.useState<FiltersState>(INITIAL_FILTERS);
const queryParams = React.useMemo<SearchIndustrialParksParams>(() => {
const p: SearchIndustrialParksParams = { page: filters.page, limit: 12 };
if (filters.q) p.q = filters.q;
if (filters.province) p.province = filters.province;
if (filters.status) p.status = filters.status;
if (filters.region) p.region = filters.region;
return p;
}, [filters]);
const { data: result, isLoading } = useQuery({
queryKey: ['admin-industrial-parks', queryParams],
queryFn: () => industrialApi.search(queryParams),
staleTime: 30_000,
});
const queryClient = useQueryClient();
const deleteMutation = useMutation({
mutationFn: (id: string) => industrialApi.deletePark(id),
onSuccess: () => queryClient.invalidateQueries({ queryKey: ['admin-industrial-parks'] }),
});
const handleDelete = (id: string, name: string) => {
if (!window.confirm(`Xoá KCN "${name}"? Thao tác này không thể hoàn tác.`)) return;
deleteMutation.mutate(id);
};
const resetFilters = () => setFilters(INITIAL_FILTERS);
return (
<div className="space-y-6">
{/* Header */}
<div className="flex flex-col gap-4 sm:flex-row sm:items-center sm:justify-between">
<div>
<h1 className="text-2xl font-bold">Quản KCN</h1>
<p className="text-sm text-muted-foreground">
Quản danh sách khu công nghiệp, chủ đu tình trạng lấp đy
</p>
</div>
<Link href="/industrial-parks/new">
<Button>Thêm KCN</Button>
</Link>
</div>
{/* Filters */}
<div className="flex flex-wrap items-center gap-3">
<Input
value={filters.q}
onChange={(e) => setFilters((f) => ({ ...f, q: e.target.value, page: 1 }))}
placeholder="Tìm theo tên, chủ đầu tư..."
className="w-64"
/>
<Input
value={filters.province}
onChange={(e) => setFilters((f) => ({ ...f, province: e.target.value, page: 1 }))}
placeholder="Tỉnh/Thành phố"
className="w-48"
/>
<Select
value={filters.region}
onChange={(e) =>
setFilters((f) => ({ ...f, region: e.target.value as VietnamRegion | '', page: 1 }))
}
className="w-40"
>
<option value="">Tất cả vùng</option>
{REGION_OPTIONS.map((r) => (
<option key={r} value={r}>
{REGION_LABELS[r]}
</option>
))}
</Select>
<Select
value={filters.status}
onChange={(e) =>
setFilters((f) => ({
...f,
status: e.target.value as IndustrialParkStatus | '',
page: 1,
}))
}
className="w-48"
>
<option value="">Tất cả trạng thái</option>
{STATUS_OPTIONS.map((s) => (
<option key={s} value={s}>
{PARK_STATUS_LABELS[s]}
</option>
))}
</Select>
<Button variant="outline" size="sm" onClick={resetFilters}>
Đt lại
</Button>
</div>
{deleteMutation.isError && (
<div className="flex items-center justify-between gap-2 rounded-md bg-destructive/10 px-3 py-2 text-sm text-destructive">
<p>Không thể xoá KCN. Vui lòng thử lại.</p>
<button
type="button"
onClick={() => deleteMutation.reset()}
className="text-xs underline"
>
Đóng
</button>
</div>
)}
{/* Content */}
{isLoading ? (
<div className="flex min-h-[300px] items-center justify-center">
<div className="h-8 w-8 animate-spin rounded-full border-4 border-primary border-t-transparent" />
</div>
) : !result || result.data.length === 0 ? (
<div className="flex min-h-[300px] flex-col items-center justify-center text-muted-foreground">
<p>Chưa KCN nào</p>
<Link href="/industrial-parks/new" className="mt-2">
<Button variant="outline" size="sm">
Thêm KCN đu tiên
</Button>
</Link>
</div>
) : (
<Card>
<CardContent className="p-0">
<div className="overflow-x-auto">
<table className="w-full text-sm">
<thead>
<tr className="border-b text-left">
<th className="p-3 font-medium">Tên KCN</th>
<th className="p-3 font-medium">Chủ đu </th>
<th className="p-3 font-medium">Tỉnh</th>
<th className="p-3 font-medium">Vùng</th>
<th className="p-3 font-medium text-right">Diện tích (ha)</th>
<th className="p-3 font-medium text-right">Tỉ lệ lấp đy</th>
<th className="p-3 font-medium text-center">Trạng thái</th>
<th className="p-3 font-medium text-right">Thao tác</th>
</tr>
</thead>
<tbody>
{result.data.map((park) => (
<tr
key={park.id}
className="border-b last:border-0 transition-colors hover:bg-accent/50"
>
<td className="p-3">
<Link
href={`/khu-cong-nghiep/${park.slug}`}
target="_blank"
rel="noreferrer"
className="group inline-flex items-center gap-1 font-medium hover:text-primary"
>
<span className="truncate">{park.name}</span>
<ExternalLink className="h-3 w-3 opacity-60" />
</Link>
{park.nameEn && (
<p className="truncate text-xs text-muted-foreground">{park.nameEn}</p>
)}
</td>
<td className="p-3 text-muted-foreground">{park.developer}</td>
<td className="p-3">{park.province}</td>
<td className="p-3 text-xs text-muted-foreground">
{REGION_LABELS[park.region]}
</td>
<td className="p-3 text-right">{park.totalAreaHa.toLocaleString('vi-VN')}</td>
<td className="p-3 text-right">
{park.occupancyRate.toFixed(1)}%
</td>
<td className="p-3 text-center">
<Badge className={PARK_STATUS_COLORS[park.status]} variant="outline">
{PARK_STATUS_LABELS[park.status]}
</Badge>
</td>
<td className="p-3 text-right">
<div className="flex justify-end gap-1">
<Link href={`/industrial-parks/${park.id}/edit`}>
<Button variant="ghost" size="sm" aria-label="Sửa KCN">
<Pencil className="h-3.5 w-3.5" />
</Button>
</Link>
<Button
variant="ghost"
size="sm"
aria-label="Xoá KCN"
className="text-destructive"
disabled={
deleteMutation.isPending &&
deleteMutation.variables === park.id
}
onClick={() => handleDelete(park.id, park.name)}
>
<Trash2 className="h-3.5 w-3.5" />
</Button>
</div>
</td>
</tr>
))}
</tbody>
</table>
</div>
</CardContent>
</Card>
)}
{/* Pagination */}
{result && result.totalPages > 1 && (
<div className="flex items-center justify-center gap-2">
<Button
variant="outline"
size="sm"
disabled={filters.page <= 1}
onClick={() => setFilters((f) => ({ ...f, page: f.page - 1 }))}
>
Trước
</Button>
<span className="text-sm text-muted-foreground">
Trang {result.page} / {result.totalPages}
</span>
<Button
variant="outline"
size="sm"
disabled={filters.page >= result.totalPages}
onClick={() => setFilters((f) => ({ ...f, page: f.page + 1 }))}
>
Tiếp
</Button>
</div>
)}
</div>
);
}