feat:emergency issue

This commit is contained in:
Anang Yusman 2025-06-10 12:55:36 +08:00
parent c064026779
commit d38bb005a1
12 changed files with 608 additions and 153 deletions

View File

@ -122,10 +122,22 @@ const useTableColumns = () => {
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent className="p-0" align="end">
<DetailSettingTracking id={row.original.id} isDetail={true} />
<UpdateSettingTracking id={row.original.id} isUpdate={true} />
<Link
href={`/admin/settings/setting-tracking/detail/${row.original.id}`}
>
<DropdownMenuItem className="p-2 border-b text-default-700 group focus:bg-default focus:text-primary-foreground rounded-none">
<Eye className="w-4 h-4 me-1.5" />
View
</DropdownMenuItem>
</Link>
<Link
href={`/admin/settings/setting-tracking/update/${row.original.id}`}
>
<DropdownMenuItem className="p-2 border-b text-default-700 group focus:bg-default focus:text-primary-foreground rounded-none">
<SquarePen className="w-4 h-4 me-1.5" />
Edit
</DropdownMenuItem>
</Link>
<DropdownMenuItem
onClick={() => handleDeleteMedia(row.original.id)}
className="p-2 border-b text-destructive bg-destructive/30 focus:bg-destructive focus:text-destructive-foreground rounded-none"

View File

@ -1,141 +1,141 @@
"use client";
// "use client";
import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog";
import { z } from "zod";
import { useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import {
Form,
FormControl,
FormDescription,
FormField,
FormItem,
FormLabel,
FormMessage,
} from "@/components/ui/form";
import { useRouter } from "@/i18n/routing";
import { Input } from "@/components/ui/input";
import { Checkbox } from "@/components/ui/checkbox";
import { getUserRoles, postCategory } from "@/service/settings/settings";
import { Fragment, useEffect, useState } from "react";
import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group";
import { Icon } from "@iconify/react/dist/iconify.js";
import { Textarea } from "@/components/ui/textarea";
import { close, error, loading } from "@/config/swal";
import { useToast } from "@/components/ui/use-toast";
import { stringify } from "querystring";
import { useDropzone } from "react-dropzone";
import { CloudUpload } from "lucide-react";
import Image from "next/image";
import { Upload } from "tus-js-client";
import { getCookiesDecrypt } from "@/lib/utils";
import Cookies from "js-cookie";
import { useTranslations } from "next-intl";
// import { Button } from "@/components/ui/button";
// import {
// Dialog,
// DialogContent,
// DialogFooter,
// DialogHeader,
// DialogTitle,
// DialogTrigger,
// } from "@/components/ui/dialog";
// import { z } from "zod";
// import { useForm } from "react-hook-form";
// import { zodResolver } from "@hookform/resolvers/zod";
// import {
// Form,
// FormControl,
// FormDescription,
// FormField,
// FormItem,
// FormLabel,
// FormMessage,
// } from "@/components/ui/form";
// import { useRouter } from "@/i18n/routing";
// import { Input } from "@/components/ui/input";
// import { Checkbox } from "@/components/ui/checkbox";
// import { getUserRoles, postCategory } from "@/service/settings/settings";
// import { Fragment, useEffect, useState } from "react";
// import { RadioGroup, RadioGroupItem } from "@/components/ui/radio-group";
// import { Icon } from "@iconify/react/dist/iconify.js";
// import { Textarea } from "@/components/ui/textarea";
// import { close, error, loading } from "@/config/swal";
// import { useToast } from "@/components/ui/use-toast";
// import { stringify } from "querystring";
// import { useDropzone } from "react-dropzone";
// import { CloudUpload } from "lucide-react";
// import Image from "next/image";
// import { Upload } from "tus-js-client";
// import { getCookiesDecrypt } from "@/lib/utils";
// import Cookies from "js-cookie";
// import { useTranslations } from "next-intl";
const wilayahList = [
{ id: "mabes", label: "Mabes" },
{ id: "polda", label: "Polda" },
{ id: "satker", label: "Satker" },
];
// const wilayahList = [
// { id: "mabes", label: "Mabes" },
// { id: "polda", label: "Polda" },
// { id: "satker", label: "Satker" },
// ];
const jumlahList = [5, 10, 15, 20, 25, 30];
// const jumlahList = [5, 10, 15, 20, 25, 30];
export default function CreateSettingTracking() {
const t = useTranslations("Menu");
const [isOpen, setIsOpen] = useState(false);
// export default function CreateSettingTracking() {
// const t = useTranslations("Menu");
// const [isOpen, setIsOpen] = useState(false);
const form = useForm({
defaultValues: {
wilayah: [] as string[],
jumlah: [] as number[],
},
});
// const form = useForm({
// defaultValues: {
// wilayah: [] as string[],
// jumlah: [] as number[],
// },
// });
const onSubmit = (values: any) => {
console.log("Submitted values:", values);
setIsOpen(false);
};
// const onSubmit = (values: any) => {
// console.log("Submitted values:", values);
// setIsOpen(false);
// };
return (
<Dialog open={isOpen} onOpenChange={setIsOpen}>
<DialogTrigger asChild>
<Button onClick={() => setIsOpen(true)}>Tambah Setting Tracking</Button>
</DialogTrigger>
// return (
// <Dialog open={isOpen} onOpenChange={setIsOpen}>
// <DialogTrigger asChild>
// <Button onClick={() => setIsOpen(true)}>Tambah Setting Tracking</Button>
// </DialogTrigger>
<DialogContent className="sm:max-w-md">
<DialogHeader>
<DialogTitle>Add Setting Tracking Berita Harian</DialogTitle>
</DialogHeader>
// <DialogContent className="sm:max-w-md">
// <DialogHeader>
// <DialogTitle>Add Setting Tracking Berita Harian</DialogTitle>
// </DialogHeader>
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
{/* Wilayah */}
<FormField
control={form.control}
name="wilayah"
render={({ field }) => (
<FormItem>
<FormLabel>Wilayah</FormLabel>
<div className="flex gap-4">
{wilayahList.map((item) => (
<div key={item.id} className="flex items-center gap-2">
<Checkbox
checked={field.value.includes(item.id)}
onCheckedChange={(checked) => {
const updated = checked
? [...field.value, item.id]
: field.value.filter((val) => val !== item.id);
field.onChange(updated);
}}
/>
<label className="text-sm">{item.label}</label>
</div>
))}
</div>
</FormItem>
)}
/>
// <Form {...form}>
// <form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
// {/* Wilayah */}
// <FormField
// control={form.control}
// name="wilayah"
// render={({ field }) => (
// <FormItem>
// <FormLabel>Wilayah</FormLabel>
// <div className="flex gap-4">
// {wilayahList.map((item) => (
// <div key={item.id} className="flex items-center gap-2">
// <Checkbox
// checked={field.value.includes(item.id)}
// onCheckedChange={(checked) => {
// const updated = checked
// ? [...field.value, item.id]
// : field.value.filter((val) => val !== item.id);
// field.onChange(updated);
// }}
// />
// <label className="text-sm">{item.label}</label>
// </div>
// ))}
// </div>
// </FormItem>
// )}
// />
<FormField
control={form.control}
name="jumlah"
render={({ field }) => (
<FormItem>
<FormLabel>Jumlah Tracking Berita Harian</FormLabel>
<div className="flex gap-4 flex-wrap">
{jumlahList.map((num) => (
<div key={num} className="flex items-center gap-2">
<Checkbox
checked={field.value.includes(num)}
onCheckedChange={(checked) => {
const updated = checked
? [...field.value, num]
: field.value.filter((val) => val !== num);
field.onChange(updated);
}}
/>
<label className="text-sm">{num}</label>
</div>
))}
</div>
</FormItem>
)}
/>
// <FormField
// control={form.control}
// name="jumlah"
// render={({ field }) => (
// <FormItem>
// <FormLabel>Jumlah Tracking Berita Harian</FormLabel>
// <div className="flex gap-4 flex-wrap">
// {jumlahList.map((num) => (
// <div key={num} className="flex items-center gap-2">
// <Checkbox
// checked={field.value.includes(num)}
// onCheckedChange={(checked) => {
// const updated = checked
// ? [...field.value, num]
// : field.value.filter((val) => val !== num);
// field.onChange(updated);
// }}
// />
// <label className="text-sm">{num}</label>
// </div>
// ))}
// </div>
// </FormItem>
// )}
// />
<DialogFooter>
<Button type="submit">Tambah Setting</Button>
</DialogFooter>
</form>
</Form>
</DialogContent>
</Dialog>
);
}
// <DialogFooter>
// <Button type="submit">Tambah Setting</Button>
// </DialogFooter>
// </form>
// </Form>
// </DialogContent>
// </Dialog>
// );
// }

View File

@ -49,8 +49,8 @@ import {
PopoverContent,
PopoverTrigger,
} from "@/components/ui/popover";
import CreateSettingTracking from "./create";
import useTableColumns from "./column";
import { UploadIcon } from "lucide-react";
const AdminSettingTrackingTable = () => {
const router = useRouter();
@ -190,7 +190,15 @@ const AdminSettingTrackingTable = () => {
return (
<div className="w-full overflow-x-auto bg-white p-4 rounded-sm space-y-3">
<div className="flex items-end justify-between">
<CreateSettingTracking />
{/* <CreateSettingTracking /> */}
<div className="flex-none">
<Link href={"/admin/settings/setting-tracking/create"}>
<Button color="primary" className="text-white" size="md">
<UploadIcon size={18} className="mr-2" />
Tambah Setting Tracking
</Button>
</Link>
</div>
<Popover>
<PopoverTrigger asChild>
<Button size="md" variant="outline">

View File

@ -0,0 +1,15 @@
import CreateSettingTracking from "@/components/form/media-tracking/setting-tracking-form";
import SiteBreadcrumb from "@/components/site-breadcrumb";
const SettingTrackingCreatePage = () => {
return (
<div>
<SiteBreadcrumb />
<div className="space-y-4">
<CreateSettingTracking />
</div>
</div>
);
};
export default SettingTrackingCreatePage;

View File

@ -0,0 +1,15 @@
import DetailSettingTracking from "@/components/form/media-tracking/setting-tracking-detail-form";
import SiteBreadcrumb from "@/components/site-breadcrumb";
const SettingTrackingDetailPage = () => {
return (
<div>
<SiteBreadcrumb />
<div className="space-y-4">
<DetailSettingTracking />
</div>
</div>
);
};
export default SettingTrackingDetailPage;

View File

@ -0,0 +1,15 @@
import UpdateSettingTracking from "@/components/form/media-tracking/setting-tracking-update-form";
import SiteBreadcrumb from "@/components/site-breadcrumb";
const SettingTrackingUpdatePage = () => {
return (
<div>
<SiteBreadcrumb />
<div className="space-y-4">
<UpdateSettingTracking />
</div>
</div>
);
};
export default SettingTrackingUpdatePage;

View File

@ -51,10 +51,8 @@ import { ChevronDownIcon } from "lucide-react";
import { getOperatorUser } from "@/service/management-user/management-user";
const taskSchema = z.object({
title: z.string().min(1, { message: "Judul diperlukan" }),
description: z.string().min(2, {
message: "Narasi Penugasan harus lebih dari 2 karakter.",
}),
message: z.string().optional(),
description: z.string().optional(),
});
export type taskDetail = {
@ -509,7 +507,7 @@ export default function FormQuestionsReply() {
<Label>Judul</Label>
<Controller
control={control}
name="title"
name="message"
render={({ field }) => (
<Input
size="md"
@ -605,6 +603,7 @@ export default function FormQuestionsReply() {
);
}
}}
disabled
/>
<span>{op.label}</span>
</label>

View File

@ -0,0 +1,110 @@
"use client";
import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog";
import { z } from "zod";
import { Controller, useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import {
Form,
FormControl,
FormDescription,
FormField,
FormItem,
FormLabel,
FormMessage,
} from "@/components/ui/form";
import { Input } from "@/components/ui/input";
import { Checkbox } from "@/components/ui/checkbox";
import { Fragment, useEffect, useState } from "react";
import { useTranslations } from "next-intl";
import { Card } from "@/components/ui/card";
const wilayahList = [
{ id: "mabes", label: "Mabes" },
{ id: "polda", label: "Polda" },
{ id: "satker", label: "Satker" },
];
const jumlahList = [5, 10, 15, 20, 25, 30];
export default function DetailSettingTracking() {
const t = useTranslations("Menu");
const [isOpen, setIsOpen] = useState(false);
const form = useForm({
defaultValues: {
wilayah: [] as string[],
jumlah: [] as number[],
},
});
const onSubmit = (values: any) => {
console.log("Submitted values:", values);
setIsOpen(false);
};
return (
<>
<Card className="px-3 py-3">
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
{/* Wilayah */}
<FormField
control={form.control}
name="wilayah"
render={({ field }) => (
<FormItem>
<FormLabel>Wilayah</FormLabel>
<div className="flex gap-4">
{wilayahList.map((item) => (
<div key={item.id} className="flex items-center gap-2">
<Checkbox
checked={field.value.includes(item.id)}
onCheckedChange={(checked) => {
const updated = checked
? [...field.value, item.id]
: field.value.filter((val) => val !== item.id);
field.onChange(updated);
}}
/>
<label className="text-sm">{item.label}</label>
</div>
))}
</div>
</FormItem>
)}
/>
<FormField
control={form.control}
name="jumlah"
render={({ field }) => (
<FormItem>
<FormLabel>Jumlah Tracking Berita Harian</FormLabel>
<div className="flex gap-4 flex-wrap">
<Input
size={"md"}
type="number"
placeholder="Masukan Nama Iklan"
/>
</div>
</FormItem>
)}
/>
<div className="flex items-end justify-end">
<Button type="submit">Tambah Setting</Button>
</div>
</form>
</Form>
</Card>
</>
);
}

View File

@ -0,0 +1,110 @@
"use client";
import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog";
import { z } from "zod";
import { Controller, useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import {
Form,
FormControl,
FormDescription,
FormField,
FormItem,
FormLabel,
FormMessage,
} from "@/components/ui/form";
import { Input } from "@/components/ui/input";
import { Checkbox } from "@/components/ui/checkbox";
import { Fragment, useEffect, useState } from "react";
import { useTranslations } from "next-intl";
import { Card } from "@/components/ui/card";
const wilayahList = [
{ id: "mabes", label: "Mabes" },
{ id: "polda", label: "Polda" },
{ id: "satker", label: "Satker" },
];
const jumlahList = [5, 10, 15, 20, 25, 30];
export default function CreateSettingTracking() {
const t = useTranslations("Menu");
const [isOpen, setIsOpen] = useState(false);
const form = useForm({
defaultValues: {
wilayah: [] as string[],
jumlah: [] as number[],
},
});
const onSubmit = (values: any) => {
console.log("Submitted values:", values);
setIsOpen(false);
};
return (
<>
<Card className="px-3 py-3">
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
{/* Wilayah */}
<FormField
control={form.control}
name="wilayah"
render={({ field }) => (
<FormItem>
<FormLabel>Wilayah</FormLabel>
<div className="flex gap-4">
{wilayahList.map((item) => (
<div key={item.id} className="flex items-center gap-2">
<Checkbox
checked={field.value.includes(item.id)}
onCheckedChange={(checked) => {
const updated = checked
? [...field.value, item.id]
: field.value.filter((val) => val !== item.id);
field.onChange(updated);
}}
/>
<label className="text-sm">{item.label}</label>
</div>
))}
</div>
</FormItem>
)}
/>
<FormField
control={form.control}
name="jumlah"
render={({ field }) => (
<FormItem>
<FormLabel>Jumlah Tracking Berita Harian</FormLabel>
<div className="flex gap-4 flex-wrap">
<Input
size={"md"}
type="number"
placeholder="Masukan Nama Iklan"
/>
</div>
</FormItem>
)}
/>
<div className="flex items-end justify-end">
<Button type="submit">Tambah Setting</Button>
</div>
</form>
</Form>
</Card>
</>
);
}

View File

@ -0,0 +1,110 @@
"use client";
import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog";
import { z } from "zod";
import { Controller, useForm } from "react-hook-form";
import { zodResolver } from "@hookform/resolvers/zod";
import {
Form,
FormControl,
FormDescription,
FormField,
FormItem,
FormLabel,
FormMessage,
} from "@/components/ui/form";
import { Input } from "@/components/ui/input";
import { Checkbox } from "@/components/ui/checkbox";
import { Fragment, useEffect, useState } from "react";
import { useTranslations } from "next-intl";
import { Card } from "@/components/ui/card";
const wilayahList = [
{ id: "mabes", label: "Mabes" },
{ id: "polda", label: "Polda" },
{ id: "satker", label: "Satker" },
];
const jumlahList = [5, 10, 15, 20, 25, 30];
export default function UpdateSettingTracking() {
const t = useTranslations("Menu");
const [isOpen, setIsOpen] = useState(false);
const form = useForm({
defaultValues: {
wilayah: [] as string[],
jumlah: [] as number[],
},
});
const onSubmit = (values: any) => {
console.log("Submitted values:", values);
setIsOpen(false);
};
return (
<>
<Card className="px-3 py-3">
<Form {...form}>
<form onSubmit={form.handleSubmit(onSubmit)} className="space-y-4">
{/* Wilayah */}
<FormField
control={form.control}
name="wilayah"
render={({ field }) => (
<FormItem>
<FormLabel>Wilayah</FormLabel>
<div className="flex gap-4">
{wilayahList.map((item) => (
<div key={item.id} className="flex items-center gap-2">
<Checkbox
checked={field.value.includes(item.id)}
onCheckedChange={(checked) => {
const updated = checked
? [...field.value, item.id]
: field.value.filter((val) => val !== item.id);
field.onChange(updated);
}}
/>
<label className="text-sm">{item.label}</label>
</div>
))}
</div>
</FormItem>
)}
/>
<FormField
control={form.control}
name="jumlah"
render={({ field }) => (
<FormItem>
<FormLabel>Jumlah Tracking Berita Harian</FormLabel>
<div className="flex gap-4 flex-wrap">
<Input
size={"md"}
type="number"
placeholder="Masukan Nama Iklan"
/>
</div>
</FormItem>
)}
/>
<div className="flex items-end justify-end">
<Button type="submit">Tambah Setting</Button>
</div>
</form>
</Form>
</Card>
</>
);
}

View File

@ -1,4 +1,6 @@
export type DetailTicket = {
title: string;
description: string;
commentFromUserId: string;
assignedTeams: string;
message: string;
@ -19,7 +21,10 @@ export type DetailTicket = {
feedUrl: string;
recommendationName: string;
link: string;
uploadFiles?: string;
uploadFiles?: {
fileUrl: string;
fileName: string;
}[];
description: string;
};
};

View File

@ -1,4 +1,3 @@
// InfoLainnyaModal.tsx
import {
Dialog,
DialogContent,
@ -6,6 +5,7 @@ import {
DialogTitle,
} from "@/components/ui/dialog";
import { DetailTicket } from "./info-lainnya-types";
import { useState } from "react";
interface InfoLainnyaModalProps {
open: boolean;
@ -18,6 +18,34 @@ export default function InfoLainnyaModal({
onClose,
data,
}: InfoLainnyaModalProps) {
const files = data?.uploadFiles || [];
const [currentIndex, setCurrentIndex] = useState(0);
const handlePrev = () => {
setCurrentIndex((prev) => (prev > 0 ? prev - 1 : prev));
};
const handleNext = () => {
setCurrentIndex((prev) => (prev < files.length - 1 ? prev + 1 : prev));
};
const currentFile = files[currentIndex];
const isImage = (fileUrl: string) =>
/\.(jpeg|jpg|png|gif|bmp|webp)$/i.test(fileUrl.toLowerCase());
const getIframeUrl = (fileUrl: string): string => {
const lower = fileUrl.toLowerCase();
// Dokumen ditampilkan melalui Google Docs Viewer
if (/\.(pdf|doc|docx|xls|xlsx|ppt|pptx)$/i.test(lower)) {
return `https://docs.google.com/viewer?url=${encodeURIComponent(
fileUrl
)}&embedded=true`;
}
return fileUrl;
};
return (
<Dialog open={open} onOpenChange={onClose}>
<DialogContent size="md">
@ -41,11 +69,11 @@ export default function InfoLainnyaModal({
<div>:{data?.recommendationName}</div>
<div className="font-medium">Link Pendukung</div>
<div>
<div className="flex">
:
<a
href={data?.link}
className="text-blue-600"
className="text-blue-600 flex w-[100px]"
target="_blank"
rel="noopener noreferrer"
>
@ -53,15 +81,43 @@ export default function InfoLainnyaModal({
</a>
</div>
{data?.uploadFiles && (
{files.length > 0 && (
<>
<div className="font-medium">Lampiran</div>
<div>
<div className="font-medium col-span-2">Lampiran</div>
<div className="col-span-2 flex flex-col items-center space-y-2 w-full">
{isImage(currentFile?.fileUrl || "") ? (
<img
src={data?.uploadFiles}
alt="Lampiran"
className="max-w-xs"
src={currentFile?.fileUrl}
alt={`Lampiran ${currentIndex + 1}`}
className="max-h-[300px] w-auto object-contain border rounded"
/>
) : (
<iframe
src={getIframeUrl(currentFile?.fileUrl || "")}
title={`Lampiran ${currentIndex + 1}`}
className="w-full max-w-2xl h-[300px] border rounded"
onError={(e) => {
(e.target as HTMLIFrameElement).style.display = "none";
}}
/>
)}
<div className="flex gap-2">
<button
onClick={handlePrev}
disabled={currentIndex === 0}
className="px-2 py-1 border rounded disabled:opacity-50"
>
&lt;
</button>
<button
onClick={handleNext}
disabled={currentIndex === files.length - 1}
className="px-2 py-1 border rounded disabled:opacity-50"
>
&gt;
</button>
</div>
</div>
</>
)}