Push 1 Program Kemiskinan
This commit is contained in:
@@ -1091,3 +1091,27 @@ model LowonganPekerjaan {
|
|||||||
updatedAt DateTime @updatedAt
|
updatedAt DateTime @updatedAt
|
||||||
deletedAt DateTime @default(now())
|
deletedAt DateTime @default(now())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ========================================= PROGRAM KEMISKINAN ========================================= //
|
||||||
|
model ProgramKemiskinan {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
nama String
|
||||||
|
deskripsi String
|
||||||
|
ikonUrl String?
|
||||||
|
isActive Boolean @default(true)
|
||||||
|
// Tambahkan relasi one-to-one ke StatistikKemiskinan
|
||||||
|
statistikId String? @unique // Foreign key ke StatistikKemiskinan, unique untuk one-to-one
|
||||||
|
statistik StatistikKemiskinan? @relation(fields: [statistikId], references: [id])
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
}
|
||||||
|
|
||||||
|
model StatistikKemiskinan {
|
||||||
|
id String @id @default(uuid())
|
||||||
|
tahun Int @unique
|
||||||
|
jumlah Int
|
||||||
|
// Tidak perlu foreign key di sini jika relasi di ProgramLayanan
|
||||||
|
programKemiskinan ProgramKemiskinan?
|
||||||
|
createdAt DateTime @default(now())
|
||||||
|
updatedAt DateTime @updatedAt
|
||||||
|
}
|
||||||
|
|||||||
249
src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts
Normal file
249
src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts
Normal file
@@ -0,0 +1,249 @@
|
|||||||
|
import ApiFetch from "@/lib/api-fetch";
|
||||||
|
import { Prisma } from "@prisma/client";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
import { proxy } from "valtio";
|
||||||
|
import { z } from "zod";
|
||||||
|
|
||||||
|
const templateForm = z.object({
|
||||||
|
nama: z.string().min(1, "Nama minimal 1 karakter"),
|
||||||
|
deskripsi: z.string().min(1, "Deskripsi minimal 1 karakter"),
|
||||||
|
ikonUrl: z.string().optional(),
|
||||||
|
statistik: z.object({
|
||||||
|
tahun: z.string().min(1, "Tahun minimal 1 karakter"),
|
||||||
|
jumlah: z.string().min(1, "Jumlah minimal 1 karakter"),
|
||||||
|
})
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
const defaultForm = {
|
||||||
|
nama: "",
|
||||||
|
deskripsi: "",
|
||||||
|
ikonUrl: "",
|
||||||
|
statistik: {
|
||||||
|
tahun: "",
|
||||||
|
jumlah: ""
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const programKemiskinanState = proxy({
|
||||||
|
create: {
|
||||||
|
form: { ...defaultForm },
|
||||||
|
loading: false,
|
||||||
|
async create() {
|
||||||
|
const cek = templateForm.safeParse(programKemiskinanState.create.form);
|
||||||
|
if (!cek.success) {
|
||||||
|
const err = `[${cek.error.issues
|
||||||
|
.map((v) => `${v.path.join(".")}`)
|
||||||
|
.join("\n")}] required`;
|
||||||
|
return toast.error(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
programKemiskinanState.create.loading = true;
|
||||||
|
const res = await ApiFetch.api.ekonomi.programkemiskinan["create"].post(
|
||||||
|
programKemiskinanState.create.form
|
||||||
|
);
|
||||||
|
if (res.status === 200) {
|
||||||
|
programKemiskinanState.findMany.load();
|
||||||
|
return toast.success("success create");
|
||||||
|
}
|
||||||
|
console.log(res);
|
||||||
|
return toast.error("failed create");
|
||||||
|
} catch (error) {
|
||||||
|
console.log(error);
|
||||||
|
return toast.error("failed create");
|
||||||
|
} finally {
|
||||||
|
programKemiskinanState.create.loading = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
findMany: {
|
||||||
|
data: [] as Prisma.ProgramKemiskinanGetPayload<{
|
||||||
|
include: {
|
||||||
|
statistik: true;
|
||||||
|
};
|
||||||
|
}>[],
|
||||||
|
loading: false,
|
||||||
|
async load() {
|
||||||
|
const res = await ApiFetch.api.ekonomi.programkemiskinan[
|
||||||
|
"find-many"
|
||||||
|
].get();
|
||||||
|
if (res.status === 200) {
|
||||||
|
programKemiskinanState.findMany.data = res.data?.data ?? [];
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
findUnique: {
|
||||||
|
data: null as Prisma.ProgramKemiskinanGetPayload<{
|
||||||
|
include: {
|
||||||
|
statistik: true;
|
||||||
|
};
|
||||||
|
}> | null,
|
||||||
|
loading: false,
|
||||||
|
async load(id: string) {
|
||||||
|
try {
|
||||||
|
const res = await fetch(`/api/ekonomi/programkemiskinan/${id}`);
|
||||||
|
if (res.ok) {
|
||||||
|
const data = await res.json();
|
||||||
|
programKemiskinanState.findUnique.data = data.data ?? null;
|
||||||
|
} else {
|
||||||
|
console.error("Failed to fetch data", res.status, res.statusText);
|
||||||
|
programKemiskinanState.findUnique.data = null;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error fetching data:", error);
|
||||||
|
programKemiskinanState.findUnique.data = null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
update: {
|
||||||
|
id: "",
|
||||||
|
form: { ...defaultForm },
|
||||||
|
loading: false,
|
||||||
|
async load(id: string) {
|
||||||
|
if (!id) {
|
||||||
|
toast.warn("ID tidak valid");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await fetch(`/api/ekonomi/programkemiskinan/${id}`, {
|
||||||
|
method: "GET",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
if (!response.ok) {
|
||||||
|
throw new Error(`HTTP error! status: ${response.status}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await response.json();
|
||||||
|
|
||||||
|
if (result?.success) {
|
||||||
|
const data = result.data;
|
||||||
|
this.id = data.id;
|
||||||
|
this.form = {
|
||||||
|
nama: data.nama,
|
||||||
|
deskripsi: data.deskripsi,
|
||||||
|
ikonUrl: data.ikonUrl || "",
|
||||||
|
statistik: {
|
||||||
|
tahun: data.statistik.tahun,
|
||||||
|
jumlah: data.statistik.jumlah,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
return data; // Return the loaded data
|
||||||
|
} else {
|
||||||
|
throw new Error(result?.message || "Gagal memuat data");
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error loading program kemiskinan:", error);
|
||||||
|
toast.error(
|
||||||
|
error instanceof Error ? error.message : "Gagal memuat data"
|
||||||
|
);
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async update() {
|
||||||
|
const cek = templateForm.safeParse(programKemiskinanState.update.form);
|
||||||
|
if (!cek.success) {
|
||||||
|
const err = `[${cek.error.issues
|
||||||
|
.map((v) => `${v.path.join(".")}`)
|
||||||
|
.join("\n")}] required`;
|
||||||
|
toast.error(err);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
programKemiskinanState.update.loading = true;
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
`/api/ekonomi/programkemiskinan/${this.id}`,
|
||||||
|
{
|
||||||
|
method: "PUT",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
body: JSON.stringify({
|
||||||
|
nama: this.form.nama,
|
||||||
|
deskripsi: this.form.deskripsi,
|
||||||
|
ikonUrl: this.form.ikonUrl,
|
||||||
|
statistik: {
|
||||||
|
tahun: this.form.statistik.tahun,
|
||||||
|
jumlah: this.form.statistik.jumlah,
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorData = await response.json().catch(() => ({}));
|
||||||
|
throw new Error(
|
||||||
|
errorData.message || `HTTP error! status: ${response.status}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await response.json();
|
||||||
|
|
||||||
|
if (result.success) {
|
||||||
|
toast.success("Berhasil update program kemiskinan");
|
||||||
|
await programKemiskinanState.findMany.load(); // refresh list
|
||||||
|
return true;
|
||||||
|
} else {
|
||||||
|
throw new Error(result.message || "Gagal update program kemiskinan");
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error updating program kemiskinan:", error);
|
||||||
|
toast.error(
|
||||||
|
error instanceof Error
|
||||||
|
? error.message
|
||||||
|
: "Terjadi kesalahan saat update program kemiskinan"
|
||||||
|
);
|
||||||
|
return false;
|
||||||
|
} finally {
|
||||||
|
programKemiskinanState.update.loading = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
reset() {
|
||||||
|
programKemiskinanState.update.id = "";
|
||||||
|
programKemiskinanState.update.form = { ...defaultForm };
|
||||||
|
},
|
||||||
|
},
|
||||||
|
delete: {
|
||||||
|
loading: false,
|
||||||
|
async delete(id: string) {
|
||||||
|
if (!id) return toast.warn("ID tidak valid");
|
||||||
|
|
||||||
|
try {
|
||||||
|
programKemiskinanState.delete.loading = true;
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
`/api/ekonomi/programkemiskinan/del/${id}`,
|
||||||
|
{
|
||||||
|
method: "DELETE",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const result = await response.json();
|
||||||
|
|
||||||
|
if (response.ok && result?.success) {
|
||||||
|
toast.success(
|
||||||
|
result.message || "Program kemiskinan berhasil dihapus"
|
||||||
|
);
|
||||||
|
await programKemiskinanState.findMany.load(); // refresh list
|
||||||
|
} else {
|
||||||
|
toast.error(result?.message || "Gagal menghapus program kemiskinan");
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Gagal delete:", error);
|
||||||
|
toast.error("Terjadi kesalahan saat menghapus program kemiskinan");
|
||||||
|
} finally {
|
||||||
|
programKemiskinanState.delete.loading = false;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export default programKemiskinanState;
|
||||||
@@ -0,0 +1,126 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
|
'use client'
|
||||||
|
import colors from '@/con/colors';
|
||||||
|
import { Box, Button, Flex, Paper, Skeleton, Stack, Text } from '@mantine/core';
|
||||||
|
import { useShallowEffect } from '@mantine/hooks';
|
||||||
|
import { IconArrowBack, IconEdit, IconX } from '@tabler/icons-react';
|
||||||
|
import { useParams, useRouter } from 'next/navigation';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { useProxy } from 'valtio/utils';
|
||||||
|
import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
|
||||||
|
import programKemiskinanState from '../../../_state/ekonomi/program-kemiskinan';
|
||||||
|
|
||||||
|
|
||||||
|
function DetailProgramKemiskinan() {
|
||||||
|
const programState = useProxy(programKemiskinanState)
|
||||||
|
const [lineChart, setLineChart] = useState<any[]>([]);
|
||||||
|
const [mounted, setMounted] = useState(false);
|
||||||
|
const [modalHapus, setModalHapus] = useState(false)
|
||||||
|
const [selectedId, setSelectedId] = useState<string | null>(null)
|
||||||
|
const router = useRouter();
|
||||||
|
const params = useParams()
|
||||||
|
|
||||||
|
useShallowEffect(() => {
|
||||||
|
setMounted(true);
|
||||||
|
programState.findUnique.load(params?.id as string)
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
const handleHapus = () => {
|
||||||
|
if (selectedId) {
|
||||||
|
programState.delete.delete(selectedId)
|
||||||
|
setModalHapus(false)
|
||||||
|
setSelectedId(null)
|
||||||
|
router.push("/admin/ekonomi/program-kemiskinan")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (programState.findUnique.data) {
|
||||||
|
setLineChart([programState.findUnique.data]);
|
||||||
|
}
|
||||||
|
}, [programState.findUnique.data])
|
||||||
|
|
||||||
|
if (!programState.findUnique.data) {
|
||||||
|
return (
|
||||||
|
<Stack py={10}>
|
||||||
|
<Skeleton h={40} />
|
||||||
|
</Stack>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Box>
|
||||||
|
<Box mb={10}>
|
||||||
|
<Button variant="subtle" onClick={() => router.back()}>
|
||||||
|
<IconArrowBack color={colors['blue-button']} size={25} />
|
||||||
|
</Button>
|
||||||
|
</Box>
|
||||||
|
<Paper w={{ base: "100%", md: "50%" }} bg={colors['white-1']} p={'md'}>
|
||||||
|
<Stack>
|
||||||
|
<Text fz={"xl"} fw={"bold"}>Detail Program Kemiskinan</Text>
|
||||||
|
<Paper bg={colors['BG-trans']} p={'md'}>
|
||||||
|
<Stack gap={"xs"}>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"}>Judul Program</Text>
|
||||||
|
<Text fz={"sm"}>{programState.findUnique.data?.nama}</Text>
|
||||||
|
</Box>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"}>Deskripsi Singkat</Text>
|
||||||
|
<Text fz={"sm"} dangerouslySetInnerHTML={{ __html: programState.findUnique.data?.deskripsi }}></Text>
|
||||||
|
</Box>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"}>Ikon URL</Text>
|
||||||
|
<Text fz={"sm"}>{programState.findUnique.data?.ikonUrl}</Text>
|
||||||
|
</Box>
|
||||||
|
<Text fw={"bold"}>Statistik Jumlah Masyarakat Miskin</Text>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"}>Jumlah Masyarakat Miskin</Text>
|
||||||
|
<Text fz={"sm"}>{programState.findUnique.data?.statistik?.jumlah}</Text>
|
||||||
|
</Box>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"}>Tahun</Text>
|
||||||
|
<Text fz={"sm"}>{programState.findUnique.data?.statistik?.tahun}</Text>
|
||||||
|
</Box>
|
||||||
|
<Flex gap={"xs"} mt={10}>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
if (programState.findUnique.data) {
|
||||||
|
setSelectedId(programState.findUnique.data.id);
|
||||||
|
setModalHapus(true);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
disabled={programState.delete.loading || !programState.findUnique.data}
|
||||||
|
color={"red"}
|
||||||
|
>
|
||||||
|
<IconX size={20} />
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
onClick={() => {
|
||||||
|
if (programState.findUnique.data) {
|
||||||
|
router.push(`/admin/ekonomi/program-kemiskinan/${programState.findUnique.data.id}/edit`);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
disabled={!programState.findUnique.data}
|
||||||
|
color={"green"}
|
||||||
|
>
|
||||||
|
<IconEdit size={20} />
|
||||||
|
</Button>
|
||||||
|
</Flex>
|
||||||
|
</Stack>
|
||||||
|
</Paper>
|
||||||
|
</Stack>
|
||||||
|
</Paper>
|
||||||
|
|
||||||
|
{/* Modal Hapus */}
|
||||||
|
<ModalKonfirmasiHapus
|
||||||
|
opened={modalHapus}
|
||||||
|
onClose={() => setModalHapus(false)}
|
||||||
|
onConfirm={handleHapus}
|
||||||
|
text="Apakah anda yakin ingin menghapus program kemiskinan ini?"
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default DetailProgramKemiskinan;
|
||||||
|
|
||||||
@@ -1,45 +1,107 @@
|
|||||||
|
/* eslint-disable @typescript-eslint/no-unused-vars */
|
||||||
|
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||||
'use client'
|
'use client'
|
||||||
import colors from '@/con/colors';
|
import colors from '@/con/colors';
|
||||||
import { Box, Button, Group, Paper, Stack, Text, TextInput, Title } from '@mantine/core';
|
import { Box, Button, Group, Paper, Stack, Text, TextInput, Title } from '@mantine/core';
|
||||||
import { IconArrowBack } from '@tabler/icons-react';
|
import { IconArrowBack } from '@tabler/icons-react';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
|
import { useProxy } from 'valtio/utils';
|
||||||
|
import programKemiskinanState from '../../../_state/ekonomi/program-kemiskinan';
|
||||||
|
import CreateEditor from '../../../_com/createEditor';
|
||||||
|
import { useState } from 'react';
|
||||||
|
|
||||||
|
|
||||||
function CreateProgramKemiskinan() {
|
function CreateProgramKemiskinan() {
|
||||||
|
const programState = useProxy(programKemiskinanState)
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
const [lineChart, setLineChart] = useState<any[]>([]);
|
||||||
|
|
||||||
|
const resetForm = () => {
|
||||||
|
programState.create.form = {
|
||||||
|
nama: "",
|
||||||
|
deskripsi: "",
|
||||||
|
ikonUrl: "",
|
||||||
|
statistik: {
|
||||||
|
tahun: "",
|
||||||
|
jumlah: "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleSubmit = async () => {
|
||||||
|
const id = await programState.create.create();
|
||||||
|
if (id) {
|
||||||
|
const idStr = String(id);
|
||||||
|
await programState.findUnique.load(idStr);
|
||||||
|
if (programState.findUnique.data) {
|
||||||
|
setLineChart([programState.findUnique.data]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
resetForm()
|
||||||
|
router.push("/admin/ekonomi/program-kemiskinan")
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box>
|
<Box>
|
||||||
<Box mb={10}>
|
<Box mb={10}>
|
||||||
<Button onClick={() => router.back()} variant='subtle' color={'blue'}>
|
<Button onClick={() => router.back()} variant='subtle' color={'blue'}>
|
||||||
<IconArrowBack color={colors['blue-button']} size={25}/>
|
<IconArrowBack color={colors['blue-button']} size={25} />
|
||||||
</Button>
|
</Button>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Paper w={{base: '100%', md: '50%'}} bg={colors['white-1']} p={'md'}>
|
<Paper w={{ base: '100%', md: '50%' }} bg={colors['white-1']} p={'md'}>
|
||||||
<Stack gap={"xs"}>
|
<Stack gap={"xs"}>
|
||||||
<Title order={4}>Create Program Kemiskinan</Title>
|
<Title order={4}>Create Program Kemiskinan</Title>
|
||||||
<TextInput
|
<TextInput
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Judul Program</Text>}
|
value={programState.create.form.nama}
|
||||||
placeholder='Masukkan judul program'
|
onChange={(val) => {
|
||||||
|
programState.create.form.nama = val.target.value;
|
||||||
|
}}
|
||||||
|
label={<Text fw={"bold"} fz={"md"}>Judul Program</Text>}
|
||||||
|
placeholder='Masukkan judul program'
|
||||||
|
/>
|
||||||
|
<Box>
|
||||||
|
<Text fw={"bold"} fz={"md"}>Deskripsi</Text>
|
||||||
|
<CreateEditor
|
||||||
|
value={programState.create.form.deskripsi}
|
||||||
|
onChange={(val) => {
|
||||||
|
programState.create.form.deskripsi = val;
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Box>
|
||||||
|
<TextInput
|
||||||
|
value={programState.create.form.ikonUrl}
|
||||||
|
onChange={(val) => {
|
||||||
|
programState.create.form.ikonUrl = val.target.value;
|
||||||
|
}}
|
||||||
|
label={<Text fw={"bold"} fz={"md"}>Ikon URL</Text>}
|
||||||
|
placeholder='Masukkan ikon url'
|
||||||
|
/>
|
||||||
|
<Text fw={"bold"} fz={"md"}>Statistik Jumlah Masyarakat Miskin</Text>
|
||||||
|
<TextInput
|
||||||
|
type='number'
|
||||||
|
value={programState.create.form.statistik.jumlah}
|
||||||
|
onChange={(val) => {
|
||||||
|
programState.create.form.statistik.jumlah = val.target.value;
|
||||||
|
}}
|
||||||
|
label={<Text fw={"bold"} fz={"md"}>Jumlah Masyarakat Miskin</Text>}
|
||||||
|
placeholder='Masukkan jumlah masyarakat miskin'
|
||||||
/>
|
/>
|
||||||
<TextInput
|
<TextInput
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Deskripsi Singkat</Text>}
|
type='number'
|
||||||
placeholder='Masukkan deskripsi'
|
value={programState.create.form.statistik.tahun}
|
||||||
/>
|
onChange={(val) => {
|
||||||
<TextInput
|
programState.create.form.statistik.tahun = val.target.value;
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Jumlah Masyarakat Miskin</Text>}
|
}}
|
||||||
placeholder='Masukkan jumlah masyarakat miskin'
|
label={<Text fw={"bold"} fz={"md"}>Tahun</Text>}
|
||||||
/>
|
placeholder='Masukkan tahun'
|
||||||
<TextInput
|
|
||||||
label={<Text fw={"bold"} fz={"sm"}>Deskripsi</Text>}
|
|
||||||
placeholder='Masukkan deskripsi'
|
|
||||||
/>
|
/>
|
||||||
<Group>
|
<Group>
|
||||||
<Button bg={colors['blue-button']}>Submit</Button>
|
<Button bg={colors['blue-button']} onClick={handleSubmit}>Submit</Button>
|
||||||
</Group>
|
</Group>
|
||||||
</Stack>
|
</Stack>
|
||||||
</Paper>
|
</Paper>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,66 +0,0 @@
|
|||||||
'use client'
|
|
||||||
import colors from '@/con/colors';
|
|
||||||
import { Box, Button, Flex, Paper, Stack, Text } from '@mantine/core';
|
|
||||||
import { IconArrowBack, IconEdit, IconX } from '@tabler/icons-react';
|
|
||||||
import { useRouter } from 'next/navigation';
|
|
||||||
// import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
|
|
||||||
|
|
||||||
function DetailProgramKemiskinan() {
|
|
||||||
const router = useRouter();
|
|
||||||
return (
|
|
||||||
<Box>
|
|
||||||
<Box mb={10}>
|
|
||||||
<Button variant="subtle" onClick={() => router.back()}>
|
|
||||||
<IconArrowBack color={colors['blue-button']} size={25} />
|
|
||||||
</Button>
|
|
||||||
</Box>
|
|
||||||
<Paper w={{ base: "100%", md: "50%" }} bg={colors['white-1']} p={'md'}>
|
|
||||||
<Stack>
|
|
||||||
<Text fz={"xl"} fw={"bold"}>Detail Program Kemiskinan</Text>
|
|
||||||
|
|
||||||
<Paper bg={colors['BG-trans']} p={'md'}>
|
|
||||||
<Stack gap={"xs"}>
|
|
||||||
<Box>
|
|
||||||
<Text fw={"bold"}>Judul Program</Text>
|
|
||||||
<Text>Program A</Text>
|
|
||||||
</Box>
|
|
||||||
<Box>
|
|
||||||
<Text fw={"bold"}>Deskripsi Singkat</Text>
|
|
||||||
<Text>Deskripsi Program A</Text>
|
|
||||||
</Box>
|
|
||||||
<Box>
|
|
||||||
<Text fw={"bold"}>Jumlah Masyarakat Miskin</Text>
|
|
||||||
<Text>100</Text>
|
|
||||||
</Box>
|
|
||||||
<Box>
|
|
||||||
<Text fw={"bold"}>Deskripsi</Text>
|
|
||||||
<Text>Deskripsi Program A</Text>
|
|
||||||
</Box>
|
|
||||||
<Box>
|
|
||||||
<Flex gap={"xs"}>
|
|
||||||
<Button color="red">
|
|
||||||
<IconX size={20} />
|
|
||||||
</Button>
|
|
||||||
<Button onClick={() => router.push('/admin/ekonomi/program-kemiskinan/edit')} color="green">
|
|
||||||
<IconEdit size={20} />
|
|
||||||
</Button>
|
|
||||||
</Flex>
|
|
||||||
</Box>
|
|
||||||
</Stack>
|
|
||||||
</Paper>
|
|
||||||
</Stack>
|
|
||||||
</Paper>
|
|
||||||
|
|
||||||
{/* Modal Hapus
|
|
||||||
<ModalKonfirmasiHapus
|
|
||||||
opened={modalHapus}
|
|
||||||
onClose={() => setModalHapus(false)}
|
|
||||||
onConfirm={handleHapus}
|
|
||||||
text="Apakah anda yakin ingin menghapus potensi ini?"
|
|
||||||
/> */}
|
|
||||||
</Box>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
export default DetailProgramKemiskinan;
|
|
||||||
|
|
||||||
@@ -1,10 +1,13 @@
|
|||||||
'use client'
|
'use client'
|
||||||
import colors from '@/con/colors';
|
import colors from '@/con/colors';
|
||||||
import { Box, Button, Paper, Table, TableTbody, TableTd, TableTh, TableThead, TableTr } from '@mantine/core';
|
import { Box, Button, Paper, Skeleton, Stack, Table, TableTbody, TableTd, TableTh, TableThead, TableTr, Text } from '@mantine/core';
|
||||||
import { IconDeviceImac, IconSearch } from '@tabler/icons-react';
|
import { IconDeviceImac, IconSearch } from '@tabler/icons-react';
|
||||||
import HeaderSearch from '../../_com/header';
|
import HeaderSearch from '../../_com/header';
|
||||||
import JudulList from '../../_com/judulList';
|
import JudulList from '../../_com/judulList';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
|
import { useProxy } from 'valtio/utils';
|
||||||
|
import programKemiskinanState from '../../_state/ekonomi/program-kemiskinan';
|
||||||
|
import { useShallowEffect } from '@mantine/hooks';
|
||||||
|
|
||||||
function ProgramKemiskinan() {
|
function ProgramKemiskinan() {
|
||||||
return (
|
return (
|
||||||
@@ -14,13 +17,26 @@ function ProgramKemiskinan() {
|
|||||||
placeholder='pencarian'
|
placeholder='pencarian'
|
||||||
searchIcon={<IconSearch size={20} />}
|
searchIcon={<IconSearch size={20} />}
|
||||||
/>
|
/>
|
||||||
<ListProgramKemiskinan/>
|
<ListProgramKemiskinan />
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
function ListProgramKemiskinan() {
|
function ListProgramKemiskinan() {
|
||||||
|
const programState = useProxy(programKemiskinanState)
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
|
useShallowEffect(() => {
|
||||||
|
programState.findMany.load()
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
if (!programState.findMany.data) {
|
||||||
|
return (
|
||||||
|
<Stack py={10}>
|
||||||
|
<Skeleton h={500} />
|
||||||
|
</Stack>
|
||||||
|
)
|
||||||
|
}
|
||||||
return (
|
return (
|
||||||
<Box py={10}>
|
<Box py={10}>
|
||||||
<Paper bg={colors['white-1']} p={'md'}>
|
<Paper bg={colors['white-1']} p={'md'}>
|
||||||
@@ -35,21 +51,25 @@ function ListProgramKemiskinan() {
|
|||||||
<TableTh>Deskripsi Singkat</TableTh>
|
<TableTh>Deskripsi Singkat</TableTh>
|
||||||
<TableTh>Jumlah Masyarakat Miskin</TableTh>
|
<TableTh>Jumlah Masyarakat Miskin</TableTh>
|
||||||
<TableTh>Detail</TableTh>
|
<TableTh>Detail</TableTh>
|
||||||
</TableTr>
|
</TableTr>
|
||||||
</TableThead>
|
</TableThead>
|
||||||
<TableTbody>
|
<TableTbody>
|
||||||
<TableTr>
|
{programState.findMany.data?.map((item) => (
|
||||||
<TableTd>Program A</TableTd>
|
<TableTr key={item.id}>
|
||||||
<TableTd>Deskripsi Program A</TableTd>
|
<TableTd>{item.nama}</TableTd>
|
||||||
<TableTd>100</TableTd>
|
<TableTd>
|
||||||
<TableTd>
|
<Text fz={'sm'} dangerouslySetInnerHTML={{ __html: item.deskripsi }} />
|
||||||
<Button onClick={() => router.push('/admin/ekonomi/program-kemiskinan/detail')}>
|
</TableTd>
|
||||||
<IconDeviceImac size={20} />
|
<TableTd>{item.statistik?.jumlah}</TableTd>
|
||||||
</Button>
|
<TableTd>
|
||||||
</TableTd>
|
<Button onClick={() => router.push(`/admin/ekonomi/program-kemiskinan/${item.id}`)}>
|
||||||
</TableTr>
|
<IconDeviceImac size={20} />
|
||||||
|
</Button>
|
||||||
|
</TableTd>
|
||||||
|
</TableTr>
|
||||||
|
))}
|
||||||
</TableTbody>
|
</TableTbody>
|
||||||
</Table>
|
</Table>
|
||||||
</Paper>
|
</Paper>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ import Elysia from "elysia";
|
|||||||
import PasarDesa from "./pasar-desa";
|
import PasarDesa from "./pasar-desa";
|
||||||
import KategoriMakanan from "./kategori-makanan";
|
import KategoriMakanan from "./kategori-makanan";
|
||||||
import LowonganKerja from "./lowongan-kerja";
|
import LowonganKerja from "./lowongan-kerja";
|
||||||
|
import ProgramKemiskinan from "./program-kemiskinan";
|
||||||
|
|
||||||
const Ekonomi = new Elysia({
|
const Ekonomi = new Elysia({
|
||||||
prefix: "/api/ekonomi",
|
prefix: "/api/ekonomi",
|
||||||
@@ -10,5 +11,6 @@ const Ekonomi = new Elysia({
|
|||||||
.use(PasarDesa)
|
.use(PasarDesa)
|
||||||
.use(KategoriMakanan)
|
.use(KategoriMakanan)
|
||||||
.use(LowonganKerja)
|
.use(LowonganKerja)
|
||||||
|
.use(ProgramKemiskinan)
|
||||||
|
|
||||||
export default Ekonomi
|
export default Ekonomi
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
|
||||||
|
type FormCreate = {
|
||||||
|
nama: string;
|
||||||
|
deskripsi: string;
|
||||||
|
ikonUrl?: string; // optional karena boleh null
|
||||||
|
statistik?: {
|
||||||
|
tahun: number;
|
||||||
|
jumlah: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function programKemiskinanCreate(context: Context) {
|
||||||
|
const body = context.body as FormCreate;
|
||||||
|
|
||||||
|
const program = await prisma.programKemiskinan.create({
|
||||||
|
data: {
|
||||||
|
nama: body.nama,
|
||||||
|
deskripsi: body.deskripsi,
|
||||||
|
ikonUrl: body.ikonUrl,
|
||||||
|
statistik: body.statistik
|
||||||
|
? {
|
||||||
|
create: {
|
||||||
|
tahun: Number(body.statistik.tahun),
|
||||||
|
jumlah: Number(body.statistik.jumlah),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
statistik: true, // untuk menampilkan data relasinya juga
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success create program kemiskinan dengan relasi statistik",
|
||||||
|
data: program,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
|
||||||
|
export default async function programKemiskinanDelete(context: Context) {
|
||||||
|
const id = context.params.id as string;
|
||||||
|
|
||||||
|
await prisma.programKemiskinan.delete({
|
||||||
|
where: { id },
|
||||||
|
include: {
|
||||||
|
statistik: true,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
status: 200,
|
||||||
|
success: true,
|
||||||
|
message: "Success delete program kemiskinan",
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
|
||||||
|
export default async function programKemiskinanFindMany() {
|
||||||
|
const data = await prisma.programKemiskinan.findMany({
|
||||||
|
include: {
|
||||||
|
statistik: true, // ikut sertakan relasinya
|
||||||
|
},
|
||||||
|
orderBy: {
|
||||||
|
createdAt: "desc",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success get all program layanan",
|
||||||
|
data,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
|
||||||
|
export default async function programKemiskinanFindUnique(request: Request) {
|
||||||
|
const url = new URL(request.url);
|
||||||
|
const pathSegments = url.pathname.split('/');
|
||||||
|
const id = pathSegments[pathSegments.length - 1];
|
||||||
|
|
||||||
|
if (!id) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "ID is required",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (typeof id !== 'string') {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "ID is required",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const data = await prisma.programKemiskinan.findUnique({
|
||||||
|
where: { id },
|
||||||
|
include: {
|
||||||
|
statistik: true,
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!data) {
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Data not found",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success get program layanan",
|
||||||
|
data,
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Find by ID error:", error);
|
||||||
|
return {
|
||||||
|
success: false,
|
||||||
|
message: "Gagal mengambil data: " + (error instanceof Error ? error.message : 'Unknown error'),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,42 @@
|
|||||||
|
import Elysia, { t } from "elysia";
|
||||||
|
import programKemiskinanCreate from "./create";
|
||||||
|
import programKemiskinanDelete from "./del";
|
||||||
|
import programKemiskinanFindUnique from "./findUnique";
|
||||||
|
import programKemiskinanUpdate from "./updt";
|
||||||
|
import programKemiskinanFindMany from "./findMany";
|
||||||
|
|
||||||
|
const ProgramKemiskinan = new Elysia({
|
||||||
|
prefix: '/programkemiskinan', tags: ['Ekonomi / Program Kemiskinan']
|
||||||
|
})
|
||||||
|
.post("/create", programKemiskinanCreate, {
|
||||||
|
body: t.Object({
|
||||||
|
nama: t.String(),
|
||||||
|
deskripsi: t.String(),
|
||||||
|
ikonUrl: t.String(),
|
||||||
|
statistik: t.Object({
|
||||||
|
tahun: t.String(),
|
||||||
|
jumlah: t.String(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
.delete("/del/:id", programKemiskinanDelete)
|
||||||
|
.get("/find-many", programKemiskinanFindMany)
|
||||||
|
.get("/:id", async (context) => {
|
||||||
|
const response = await programKemiskinanFindUnique(new Request(context.request));
|
||||||
|
return response;
|
||||||
|
})
|
||||||
|
.put("/:id", async (context) => {
|
||||||
|
const response = await programKemiskinanUpdate(context);
|
||||||
|
return response;
|
||||||
|
}, {
|
||||||
|
body: t.Object({
|
||||||
|
nama: t.String(),
|
||||||
|
deskripsi: t.String(),
|
||||||
|
ikonUrl: t.String(),
|
||||||
|
statistik: t.Object({
|
||||||
|
tahun: t.String(),
|
||||||
|
jumlah: t.String(),
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
})
|
||||||
|
export default ProgramKemiskinan;
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
import prisma from "@/lib/prisma";
|
||||||
|
import { Context } from "elysia";
|
||||||
|
|
||||||
|
type FormUpdate = {
|
||||||
|
nama: string;
|
||||||
|
deskripsi: string;
|
||||||
|
ikonUrl?: string;
|
||||||
|
statistik?: {
|
||||||
|
tahun: number;
|
||||||
|
jumlah: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default async function programKemiskinanUpdate(context: Context) {
|
||||||
|
const id = context.params.id as string;
|
||||||
|
const body = context.body as FormUpdate;
|
||||||
|
|
||||||
|
const program = await prisma.programKemiskinan.update({
|
||||||
|
where: { id },
|
||||||
|
data: {
|
||||||
|
nama: body.nama,
|
||||||
|
deskripsi: body.deskripsi,
|
||||||
|
ikonUrl: body.ikonUrl,
|
||||||
|
statistik: body.statistik
|
||||||
|
? {
|
||||||
|
create: {
|
||||||
|
tahun: Number(body.statistik.tahun),
|
||||||
|
jumlah: Number(body.statistik.jumlah),
|
||||||
|
},
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
|
},
|
||||||
|
include: {
|
||||||
|
statistik: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
message: "Success update program kemiskinan",
|
||||||
|
data: program,
|
||||||
|
};
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user