diff --git a/.gemini/hooks/telegram-notify.ts b/.gemini/hooks/telegram-notify.ts
new file mode 100755
index 00000000..063d337e
--- /dev/null
+++ b/.gemini/hooks/telegram-notify.ts
@@ -0,0 +1,65 @@
+#!/usr/bin/env bun
+import { readFileSync } from "node:fs";
+
+// Fungsi untuk mencari string terpanjang dalam objek (biasanya balasan AI)
+function findLongestString(obj: any): string {
+ let longest = "";
+ const search = (item: any) => {
+ if (typeof item === "string") {
+ if (item.length > longest.length) longest = item;
+ } else if (Array.isArray(item)) {
+ item.forEach(search);
+ } else if (item && typeof item === "object") {
+ Object.values(item).forEach(search);
+ }
+ };
+ search(obj);
+ return longest;
+}
+
+async function run() {
+ try {
+ const inputRaw = readFileSync(0, "utf-8");
+ if (!inputRaw) return;
+ const input = JSON.parse(inputRaw);
+
+ // DEBUG: Lihat struktur asli di console terminal (stderr)
+ console.error("DEBUG KEYS:", Object.keys(input));
+
+ const BOT_TOKEN = process.env.BOT_TOKEN;
+ const CHAT_ID = process.env.CHAT_ID;
+
+ const sessionId = input.session_id || "unknown";
+
+ // Cari teks secara otomatis di seluruh objek JSON
+ let finalText = findLongestString(input.response || input);
+
+ if (!finalText || finalText.length < 5) {
+ finalText =
+ "Teks masih gagal diekstraksi. Struktur: " +
+ Object.keys(input).join(", ");
+ }
+
+ const message =
+ `✅ *Gemini Task Selesai*\n\n` +
+ `🆔 Session: \`${sessionId}\` \n\n` +
+ `🧠 Output:\n${finalText.substring(0, 3500)}`;
+
+ await fetch(`https://api.telegram.org/bot${BOT_TOKEN}/sendMessage`, {
+ method: "POST",
+ headers: { "Content-Type": "application/json" },
+ body: JSON.stringify({
+ chat_id: CHAT_ID,
+ text: message,
+ parse_mode: "Markdown",
+ }),
+ });
+
+ process.stdout.write(JSON.stringify({ status: "continue" }));
+ } catch (err) {
+ console.error("Hook Error:", err);
+ process.stdout.write(JSON.stringify({ status: "continue" }));
+ }
+}
+
+run();
diff --git a/.gemini/settings.json b/.gemini/settings.json
new file mode 100644
index 00000000..ed736356
--- /dev/null
+++ b/.gemini/settings.json
@@ -0,0 +1,17 @@
+{
+ "hooks": {
+ "AfterAgent": [
+ {
+ "matcher": "*",
+ "hooks": [
+ {
+ "name": "telegram-notify",
+ "type": "command",
+ "command": "bun $GEMINI_PROJECT_DIR/.gemini/hooks/telegram-notify.ts",
+ "timeout": 10000
+ }
+ ]
+ }
+ ]
+ }
+}
diff --git a/.gitignore b/.gitignore
index ebd64b35..2f3afc79 100644
--- a/.gitignore
+++ b/.gitignore
@@ -31,6 +31,9 @@ yarn-error.log*
# env
.env*
+# QC
+QC
+
# vercel
.vercel
diff --git a/.vscode/extensions.json b/.vscode/extensions.json
new file mode 100644
index 00000000..7e257db9
--- /dev/null
+++ b/.vscode/extensions.json
@@ -0,0 +1,3 @@
+{
+ "recommendations": []
+}
\ No newline at end of file
diff --git a/AGENTS.md b/AGENTS.md
new file mode 100644
index 00000000..dc9d86dc
--- /dev/null
+++ b/AGENTS.md
@@ -0,0 +1,167 @@
+# AGENTS.md
+
+This file contains essential information for agentic coding agents working in the desa-darmasaba repository.
+
+## Project Overview
+
+Desa Darmasaba is a Next.js 15 application for village management services in Badung, Bali. It uses:
+- **Framework**: Next.js 15 with App Router
+- **Language**: TypeScript with strict mode
+- **Styling**: Mantine UI components with custom CSS
+- **Backend**: Elysia.js API server integrated with Next.js
+- **Database**: PostgreSQL with Prisma ORM
+- **State Management**: Jotai for global state
+- **Authentication**: JWT with iron-session
+
+## Build Commands
+
+```bash
+# Development
+npm run dev
+
+# Production build
+npm run build
+
+# Start production server
+npm start
+
+# Database seeding
+bun run prisma/seed.ts
+
+# Linting (ESLint)
+npx eslint .
+
+# Type checking
+npx tsc --noEmit
+
+# Prisma operations
+npx prisma generate
+npx prisma db push
+npx prisma studio
+```
+
+## Running Tests
+
+Currently no test framework is configured. When adding tests:
+- Set up test scripts in package.json
+- Consider Jest or Vitest for unit testing
+- Use Playwright for E2E testing
+- Update this section with specific test commands
+
+## Code Style Guidelines
+
+### Imports
+- Use absolute imports with `@/` alias (configured in tsconfig.json)
+- Group imports: external libraries first, then internal modules
+- Keep import statements organized and remove unused imports
+
+```typescript
+// External libraries
+import { useState } from 'react'
+import { Button, Stack } from '@mantine/core'
+
+// Internal modules
+import ApiFetch from '@/lib/api-fetch'
+import { MyComponent } from '@/components/my-component'
+```
+
+### TypeScript Configuration
+- Strict mode enabled (`"strict": true`)
+- Target: ES2017
+- Module resolution: bundler
+- Path alias: `@/*` maps to `./src/*`
+
+### Naming Conventions
+- **Components**: PascalCase (e.g., `UploadImage.tsx`)
+- **Files**: kebab-case for utilities (e.g., `api-fetch.ts`)
+- **Variables/Functions**: camelCase
+- **Constants**: UPPER_SNAKE_CASE
+- **Database Models**: PascalCase (Prisma convention)
+
+### Error Handling
+- Use try-catch blocks for async operations
+- Implement proper error boundaries in React components
+- Log errors appropriately without exposing sensitive data
+- Use Zod for runtime validation and type safety
+
+### API Structure
+- Backend uses Elysia.js with TypeScript
+- API routes are in `src/app/api/[[...slugs]]/` directory
+- Use treaty client for type-safe API calls
+- Follow RESTful conventions for endpoints
+- Include proper HTTP status codes and error responses
+
+### Database Operations
+- Use Prisma client from `@/lib/prisma.ts`
+- Database connection includes graceful shutdown handling
+- Use transactions for complex operations
+- Implement proper error handling for database queries
+
+### Component Guidelines
+- Use functional components with hooks
+- Implement proper prop types with TypeScript interfaces
+- Use Mantine components for UI consistency
+- Follow atomic design principles when possible
+- Add loading states and error states for async operations
+
+### State Management
+- Use Jotai atoms for global state
+- Keep local state in components when possible
+- Use React Query (SWR) for server state caching
+- Implement optimistic updates for better UX
+
+### Styling
+- Primary: Mantine UI components
+- Use Mantine theme system for customization
+- Custom CSS should be minimal and scoped
+- Follow responsive design principles
+- Use semantic HTML5 elements
+
+### Environment Variables
+- Use `.env.local` for development
+- Prefix public variables with `NEXT_PUBLIC_`
+- Never commit environment files to version control
+- Use proper typing for environment variables
+
+### File Organization
+```
+src/
+├── app/ # Next.js app router pages
+├── components/ # Reusable React components
+├── lib/ # Utility functions and configurations
+├── state/ # Jotai atoms and state management
+├── types/ # TypeScript type definitions
+└── con/ # Constants and static data
+```
+
+### Security Practices
+- Validate all user inputs with Zod schemas
+- Use JWT tokens for authentication
+- Implement proper CORS configuration
+- Never expose database credentials or API keys
+- Use HTTPS in production
+- Implement rate limiting for sensitive endpoints
+
+### Performance Considerations
+- Use Next.js Image optimization
+- Implement proper caching strategies
+- Use React.memo for expensive components
+- Optimize bundle size with dynamic imports
+- Use Prisma query optimization
+
+## Development Workflow
+
+1. Always run type checking before committing: `npx tsc --noEmit`
+2. Run linting to catch style issues: `npx eslint .`
+3. Test database changes with `npx prisma db push`
+4. Use the integrated Swagger docs at `/api/docs` for API testing
+5. Check environment variables are properly configured
+6. Verify responsive design on different screen sizes
+
+## Important Notes
+
+- The application uses a custom Elysia.js server integrated with Next.js API routes
+- Image uploads are handled through `/api/upl-img-single` endpoint
+- Database seeding is done with Bun runtime
+- The app supports Indonesian locale (id_ID) for SEO and content
+- CORS is configured to allow cross-origin requests during development
\ No newline at end of file
diff --git a/GEMINI.md b/GEMINI.md
new file mode 100644
index 00000000..4c8370da
--- /dev/null
+++ b/GEMINI.md
@@ -0,0 +1,62 @@
+# Project: Desa Darmasaba
+
+## Project Overview
+
+The `desa-darmasaba` project is a Next.js (version 15+) application developed with TypeScript. It serves as an official platform for Desa Darmasaba (a village in Badung, Bali), offering various public services, news, and detailed village profiles.
+
+**Key Technologies:**
+
+* **Frontend Framework:** Next.js (v15+) with React (v19+)
+* **Language:** TypeScript
+* **UI Library:** Mantine UI
+* **Database ORM:** Prisma (v6+)
+* **Database:** PostgreSQL (as configured in `prisma/schema.prisma`)
+* **API Framework:** Elysia (used for API routes, as seen in dependencies)
+* **State Management:** Potentially Jotai and Valtio (listed in dependencies)
+* **Image Processing:** Sharp
+* **Package Manager:** Likely Bun, given `bun.lockb` and the `prisma:seed` script.
+
+The application architecture follows the Next.js App Router structure, with comprehensive data models defined in `prisma/schema.prisma` covering various domains like public information, health, security, economy, innovation, environment, and education. It also includes configurations for image handling and caching.
+
+## Building and Running
+
+This project uses `bun` as the package manager. Ensure Bun is installed to run these commands.
+
+* **Install Dependencies:**
+ ```bash
+ bun install
+ ```
+
+* **Development Server:**
+ Runs the Next.js development server.
+ ```bash
+ bun run dev
+ ```
+
+* **Build for Production:**
+ Builds the Next.js application for production deployment.
+ ```bash
+ bun run build
+ ```
+
+* **Start Production Server:**
+ Starts the Next.js application in production mode.
+ ```bash
+ bun run start
+ ```
+
+* **Database Seeding:**
+ Executes the Prisma seeding script to populate the database.
+ ```bash
+ bun run prisma:seed
+ ```
+
+## Development Conventions
+
+* **Coding Language:** TypeScript is strictly enforced.
+* **Frontend Framework:** Next.js App Router for page and component structuring.
+* **UI/UX:** Adherence to Mantine UI component library for consistent styling and user experience.
+* **Database Interaction:** Prisma ORM is used for all database operations, with a PostgreSQL database.
+* **Linting:** ESLint is configured with `next/core-web-vitals` and `next/typescript` to maintain code quality and adherence to Next.js and TypeScript best practices.
+* **Styling:** PostCSS is used, with `postcss-preset-mantine` and `postcss-simple-vars` defining Mantine-specific breakpoints and other CSS variables.
+* **Imports:** Absolute imports are configured using `@/*` which resolves to the `src/` directory.
diff --git a/QWEN.md b/QWEN.md
new file mode 100644
index 00000000..21d07dbf
--- /dev/null
+++ b/QWEN.md
@@ -0,0 +1,232 @@
+# Desa Darmasaba - Village Management System
+
+## Project Overview
+
+Desa Darmasaba is a comprehensive Next.js 15 application designed for village management services in Darmasaba, Badung, Bali. The application serves as a digital platform for government services, public information, and community engagement. It features multiple sections including PPID (Public Information Disclosure), health services, security, education, environment, economy, innovation, and more.
+
+### Key Technologies
+- **Framework**: Next.js 15 with App Router
+- **Language**: TypeScript with strict mode
+- **Styling**: Mantine UI components with custom CSS
+- **Backend**: Elysia.js API server integrated with Next.js
+- **Database**: PostgreSQL with Prisma ORM
+- **State Management**: Valtio for global state
+- **Authentication**: JWT with iron-session
+
+### Architecture
+The application follows a modular architecture with:
+- A main frontend built with Next.js and Mantine UI
+- An integrated Elysia.js API server for backend operations
+- Prisma ORM for database interactions
+- File storage integration with Seafile
+- Multiple domain-specific modules (PPID, health, security, education, etc.)
+
+## Building and Running
+
+### Prerequisites
+- Node.js (with Bun runtime)
+- PostgreSQL database
+- Seafile server for file storage
+
+### Setup Instructions
+1. Install dependencies:
+ ```bash
+ bun install
+ ```
+
+2. Set up environment variables in `.env.local`:
+ ```
+ DATABASE_URL=your_postgresql_connection_string
+ SEAFILE_TOKEN=your_seafile_token
+ SEAFILE_REPO_ID=your_seafile_repo_id
+ SEAFILE_BASE_URL=your_seafile_base_url
+ SEAFILE_PUBLIC_SHARE_TOKEN=your_seafile_public_share_token
+ SEAFILE_URL=your_seafile_api_url
+ WIBU_UPLOAD_DIR=your_upload_directory
+ ```
+
+3. Generate Prisma client:
+ ```bash
+ bunx prisma generate
+ ```
+
+4. Push database schema:
+ ```bash
+ bunx prisma db push
+ ```
+
+5. Seed the database:
+ ```bash
+ bun run prisma/seed.ts
+ ```
+
+6. Run the development server:
+ ```bash
+ bun run dev
+ ```
+
+### Available Scripts
+- `bun run dev` - Start development server
+- `bun run build` - Build for production
+- `bun run start` - Start production server
+- `bun run prisma/seed.ts` - Run database seeding
+- `bunx prisma generate` - Generate Prisma client
+- `bunx prisma db push` - Push schema changes to database
+- `bunx prisma studio` - Open Prisma Studio GUI
+
+## Development Conventions
+
+### Code Structure
+```
+src/
+├── app/ # Next.js app router pages
+│ ├── admin/ # Admin dashboard pages
+│ ├── api/ # API routes with Elysia.js
+│ ├── darmasaba/ # Public-facing village pages
+│ └── ...
+├── con/ # Constants and configuration
+├── hooks/ # React hooks
+├── lib/ # Utility functions and configurations
+├── middlewares/ # Next.js middleware
+├── state/ # Global state management
+├── store/ # Additional state management
+├── types/ # TypeScript type definitions
+└── utils/ # Utility functions
+```
+
+### Import Conventions
+- Use absolute imports with `@/` alias (configured in tsconfig.json)
+- Group imports: external libraries first, then internal modules
+- Keep import statements organized and remove unused imports
+
+```typescript
+// External libraries
+import { useState } from 'react'
+import { Button, Stack } from '@mantine/core'
+
+// Internal modules
+import ApiFetch from '@/lib/api-fetch'
+import { MyComponent } from '@/components/my-component'
+```
+
+### TypeScript Configuration
+- Strict mode enabled (`"strict": true`)
+- Target: ES2017
+- Module resolution: bundler
+- Path alias: `@/*` maps to `./src/*`
+
+### Naming Conventions
+- **Components**: PascalCase (e.g., `UploadImage.tsx`)
+- **Files**: kebab-case for utilities (e.g., `api-fetch.ts`)
+- **Variables/Functions**: camelCase
+- **Constants**: UPPER_SNAKE_CASE
+- **Database Models**: PascalCase (Prisma convention)
+
+### Error Handling
+- Use try-catch blocks for async operations
+- Implement proper error boundaries in React components
+- Log errors appropriately without exposing sensitive data
+- Use Zod for runtime validation and type safety
+
+### API Structure
+- Backend uses Elysia.js with TypeScript
+- API routes are in `src/app/api/[[...slugs]]/` directory
+- Use treaty client for type-safe API calls
+- Follow RESTful conventions for endpoints
+- Include proper HTTP status codes and error responses
+
+### Database Operations
+- Use Prisma client from `@/lib/prisma.ts`
+- Database connection includes graceful shutdown handling
+- Use transactions for complex operations
+- Implement proper error handling for database queries
+
+### Component Guidelines
+- Use functional components with hooks
+- Implement proper prop types with TypeScript interfaces
+- Use Mantine components for UI consistency
+- Follow atomic design principles when possible
+- Add loading states and error states for async operations
+
+### State Management
+- Use Valtio proxies for global state
+- Keep local state in components when possible
+- Use SWR for server state caching
+- Implement optimistic updates for better UX
+
+### Styling
+- Primary: Mantine UI components
+- Use Mantine theme system for customization
+- Custom CSS should be minimal and scoped
+- Follow responsive design principles
+- Use semantic HTML5 elements
+
+### Security Practices
+- Validate all user inputs with Zod schemas
+- Use JWT tokens for authentication
+- Implement proper CORS configuration
+- Never expose database credentials or API keys
+- Use HTTPS in production
+- Implement rate limiting for sensitive endpoints
+
+### Performance Considerations
+- Use Next.js Image optimization
+- Implement proper caching strategies
+- Use React.memo for expensive components
+- Optimize bundle size with dynamic imports
+- Use Prisma query optimization
+
+## Domain Modules
+
+The application is organized into several domain modules:
+
+1. **PPID (Public Information Disclosure)**: Profile, structure, information requests, legal basis
+2. **Health**: Health facilities, programs, emergency response, disease information
+3. **Security**: Community security, emergency contacts, crime prevention
+4. **Education**: Schools, scholarships, educational programs
+5. **Economy**: Local markets, BUMDes, employment data
+6. **Environment**: Environmental data, conservation, waste management
+7. **Innovation**: Digital services, innovation programs
+8. **Culture**: Village traditions, music, cultural preservation
+
+Each module has its own section in both the admin panel and public-facing areas.
+
+## File Storage Integration
+
+The application integrates with Seafile for file storage, with specific handling for:
+- Images and documents
+- Public sharing capabilities
+- CDN URL generation
+- Batch processing of assets
+
+## Testing
+
+Currently no formal test framework is configured. When adding tests:
+- Consider Jest or Vitest for unit testing
+- Use Playwright for E2E testing
+- Update this section with specific test commands
+
+## Deployment
+
+The application includes deployment scripts in the `NOTE.md` file that outline:
+- Automated deployment with GitHub API integration
+- Environment-specific configurations
+- PM2 process management
+- Release management with versioning
+
+## Troubleshooting
+
+Common issues and solutions:
+- **API endpoints returning 404**: Check that environment variables are properly configured
+- **Database connection errors**: Verify DATABASE_URL in environment variables
+- **File upload issues**: Ensure Seafile integration is properly configured
+- **Build failures**: Run `bunx prisma generate` before building
+
+## Development Workflow
+
+1. Always run type checking before committing: `bunx tsc --noEmit`
+2. Run linting to catch style issues: `bun run eslint .`
+3. Test database changes with `bunx prisma db push`
+4. Use the integrated Swagger docs at `/api/docs` for API testing
+5. Check environment variables are properly configured
+6. Verify responsive design on different screen sizes
\ No newline at end of file
diff --git a/__tests__/api/fileStorage.test.ts b/__tests__/api/fileStorage.test.ts
new file mode 100644
index 00000000..ca12c5a2
--- /dev/null
+++ b/__tests__/api/fileStorage.test.ts
@@ -0,0 +1,30 @@
+import { describe, it, expect } from 'vitest';
+import ApiFetch from '@/lib/api-fetch';
+
+describe('FileStorage API', () => {
+ it('should fetch a list of files from /api/fileStorage/findMany', async () => {
+ const response = await ApiFetch.api.fileStorage.findMany.get();
+
+ expect(response.status).toBe(200);
+
+ const responseBody = response.data as any;
+
+ expect(responseBody.data).toBeInstanceOf(Array);
+ expect(responseBody.data.length).toBe(2);
+ expect(responseBody.data[0].name).toBe('file1.jpg');
+ });
+
+ it('should create a file using /api/fileStorage/create', async () => {
+ const mockFile = new File(['hello'], 'hello.png', { type: 'image/png' });
+ const response = await ApiFetch.api.fileStorage.create.post({
+ file: mockFile,
+ name: 'hello.png',
+ });
+
+ expect(response.status).toBe(200);
+ const responseBody = response.data as any;
+
+ expect(responseBody.data.realName).toBe('hello.png');
+ expect(responseBody.data.id).toBe('3');
+ });
+});
diff --git a/__tests__/e2e/homepage.spec.ts b/__tests__/e2e/homepage.spec.ts
new file mode 100644
index 00000000..57b02234
--- /dev/null
+++ b/__tests__/e2e/homepage.spec.ts
@@ -0,0 +1,11 @@
+import { test, expect } from '@playwright/test';
+
+test('homepage has correct title and content', async ({ page }) => {
+ await page.goto('/');
+
+ // Wait for the redirect to /darmasaba
+ await page.waitForURL('/darmasaba');
+
+ // Check for the main heading
+ await expect(page.getByText('DARMASABA', { exact: true })).toBeVisible();
+});
diff --git a/__tests__/mocks/handlers.ts b/__tests__/mocks/handlers.ts
new file mode 100644
index 00000000..2854bf86
--- /dev/null
+++ b/__tests__/mocks/handlers.ts
@@ -0,0 +1,43 @@
+import { http, HttpResponse } from 'msw';
+
+export const handlers = [
+ http.get('http://localhost:3000/api/fileStorage/findMany', () => {
+ return HttpResponse.json({
+ data: [
+ { id: '1', name: 'file1.jpg', url: '/uploads/file1.jpg' },
+ { id: '2', name: 'file2.png', url: '/uploads/file2.png' },
+ ],
+ meta: {
+ page: 1,
+ limit: 10,
+ total: 2,
+ totalPages: 1,
+ },
+ });
+ }),
+ http.post('http://localhost:3000/api/fileStorage/create', async ({ request }) => {
+ const data = await request.formData();
+ const file = data.get('file') as File;
+ const name = data.get('name') as string;
+
+ if (!file) {
+ return new HttpResponse(null, { status: 400 });
+ }
+
+ return HttpResponse.json({
+ data: {
+ id: '3',
+ name: 'generated-nanoid',
+ path: `/uploads/generated-nanoid`,
+ link: `/uploads/generated-nanoid`,
+ realName: name,
+ mimeType: file.type,
+ category: "uncategorized",
+ isActive: true,
+ createdAt: new Date(),
+ updatedAt: new Date(),
+ deletedAt: null,
+ }
+ });
+ }),
+];
diff --git a/__tests__/mocks/server.ts b/__tests__/mocks/server.ts
new file mode 100644
index 00000000..e52fee0a
--- /dev/null
+++ b/__tests__/mocks/server.ts
@@ -0,0 +1,4 @@
+import { setupServer } from 'msw/node';
+import { handlers } from './handlers';
+
+export const server = setupServer(...handlers);
diff --git a/__tests__/setup.ts b/__tests__/setup.ts
new file mode 100644
index 00000000..83d8b89c
--- /dev/null
+++ b/__tests__/setup.ts
@@ -0,0 +1,7 @@
+import '@testing-library/jest-dom';
+import { server } from './mocks/server';
+import { beforeAll, afterEach, afterAll } from 'vitest';
+
+beforeAll(() => server.listen({ onUnhandledRequest: 'bypass' }));
+afterEach(() => server.resetHandlers());
+afterAll(() => server.close());
diff --git a/bun.lockb b/bun.lockb
index e571b0b1..7f208dd4 100755
Binary files a/bun.lockb and b/bun.lockb differ
diff --git a/darkMode.md b/darkMode.md
new file mode 100644
index 00000000..713ac5db
--- /dev/null
+++ b/darkMode.md
@@ -0,0 +1,169 @@
+# 🌙 Dark Mode Design Specification
+## Admin Darmasaba – Dashboard & CMS
+
+Dokumen ini mendefinisikan standar **Dark Mode UI** agar:
+- nyaman di mata
+- konsisten
+- tidak flat
+- tetap profesional untuk aplikasi pemerintahan
+
+---
+
+## 🎨 Color Palette (Dark Mode)
+
+### Background Layers
+| Layer | Token | Warna | Fungsi |
+|------|------|------|------|
+| Base | `--bg-base` | `#0B1220` | Background utama aplikasi |
+| App | `--bg-app` | `#0F172A` | Area kerja utama |
+| Card | `--bg-card` | `#162235` | Card / container |
+| Surface | `--bg-surface` | `#1E2A3D` | Table header, tab, input |
+
+---
+
+### Border & Divider
+| Token | Warna | Catatan |
+|-----|------|--------|
+| `--border-default` | `#2A3A52` | Border utama |
+| `--border-soft` | `#22314A` | Divider halus |
+
+> ❗ Hindari border terlalu tipis (`opacity < 20%`)
+
+---
+
+### Text Colors
+| Jenis | Token | Warna |
+|-----|------|------|
+| Primary | `--text-primary` | `#E5E7EB` |
+| Secondary | `--text-secondary` | `#9CA3AF` |
+| Muted | `--text-muted` | `#6B7280` |
+| Inverse | `--text-inverse` | `#020617` |
+
+---
+
+### Accent & Action
+| Fungsi | Warna |
+|------|------|
+| Primary Action | `#3B82F6` |
+| Hover | `#2563EB` |
+| Active | `#1D4ED8` |
+| Link | `#60A5FA` |
+
+---
+
+### Status Colors
+| Status | Warna |
+|------|------|
+| Success | `#22C55E` |
+| Warning | `#FACC15` |
+| Error | `#EF4444` |
+| Info | `#38BDF8` |
+
+---
+
+## 🧱 Layout Rules
+
+### Sidebar
+- Background: `--bg-app`
+- Active menu:
+ - Background: `rgba(59,130,246,0.15)`
+ - Text: Primary
+ - Indicator: kiri (2–3px accent bar)
+- Hover:
+ - Background: `rgba(255,255,255,0.04)`
+
+---
+
+### Header / Topbar
+- Background: `linear-gradient(#0F172A → #0B1220)`
+- Border bawah wajib (`--border-soft`)
+- Icon:
+ - Default: muted
+ - Hover: primary
+
+---
+
+## 📦 Card & Section
+
+### Card
+- Background: `--bg-card`
+- Border: `--border-default`
+- Radius: 12–16px
+- Jangan pakai shadow hitam
+
+### Section Header
+- Font weight lebih besar
+- Text: primary
+- Spacing jelas dari konten
+
+---
+
+## 📊 Table (Dark Mode Friendly)
+
+### Table Header
+- Background: `--bg-surface`
+- Text: secondary
+- Font weight: medium
+
+### Table Row
+- Default: transparent
+- Hover:
+ - Background: `rgba(255,255,255,0.03)`
+- Divider antar row wajib terlihat
+
+### Link di Table
+- Warna link **lebih terang dari text**
+- Hover underline
+
+---
+
+## 🔘 Button Rules
+
+### Primary Button
+- Background: Primary Action
+- Text: Inverse
+- Hover: darker shade
+
+### Secondary Button
+- Background: transparent
+- Border: `--border-default`
+- Text: primary
+
+### Icon Button
+- Default: muted
+- Hover: primary + bg soft
+
+---
+
+## 🧭 Tab Navigation
+
+- Inactive:
+ - Text: muted
+- Active:
+ - Background: `rgba(59,130,246,0.15)`
+ - Text: primary
+ - Icon ikut berubah
+
+---
+
+## 🌗 Dark vs Light Mode Rule
+- Layout, spacing, typography **HARUS SAMA**
+- Yang boleh beda:
+ - warna
+ - border intensity
+ - background layer
+
+> ❌ Jangan ganti struktur UI antara dark & light
+
+---
+
+## ✅ Dark Mode Checklist
+- [ ] Kontras teks terbaca
+- [ ] Active state jelas
+- [ ] Hover terasa hidup
+- [ ] Tidak flat
+- [ ] Tidak silau
+
+---
+
+Dokumen ini adalah **single source of truth** untuk Dark Mode.
\ No newline at end of file
diff --git a/gambar.ttx b/gambar.ttx
new file mode 100644
index 00000000..6fa9e10e
--- /dev/null
+++ b/gambar.ttx
@@ -0,0 +1,99 @@
+type DirItem = {
+ type: "file" | "dir";
+ name: string;
+ path: string;
+ size?: number;
+};
+
+// type FileDownloadResponse = {
+// url: string;
+// };
+
+// const TOKEN = "20a19f4a04032215d50ce53292e6abdd38b9f806";
+// const REPO_ID = "8814bfe1-30d5-4e77-ab36-3122fa59a022";
+// const DIR_TARGET = "image";
+
+// const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com/api2";
+
+const TOKEN = process.env.SEAFILE_TOKEN!;
+const REPO_ID = process.env.SEAFILE_REPO_ID!;
+
+// ⛔ PENTING: RELATIVE PATH (tanpa slash depan)
+const DIR_TARGET = "asset-web";
+
+const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com/api2";
+
+const headers = {
+ Authorization: `Token ${TOKEN}`,
+};
+
+/**
+ * Ambil list file di directory
+ */
+async function getDirItems(): Promise {
+ const res = await fetch(
+ `${BASE_URL}/repos/${REPO_ID}/dir/?p=${DIR_TARGET}`,
+ { headers }
+ );
+
+ if (!res.ok) {
+ throw new Error(`Failed get dir items: ${res.statusText}`);
+ }
+
+ return res.json();
+}
+
+/**
+ * Ambil download URL file
+ */
+async function getDownloadUrl(filePath: string): Promise {
+
+
+ const res = await fetch(
+ `${BASE_URL}/repos/${REPO_ID}/file/?p=${encodeURIComponent(filePath)}`,
+ { headers }
+ );
+
+ if (!res.ok) {
+ throw new Error(`Failed get file url: ${res.statusText}`);
+ }
+
+ const data = await res.json();
+
+ return data;
+}
+
+/**
+ * Ambil semua download URL dari target dir
+ */
+async function getAllDownloadUrls() {
+ const items = await getDirItems();
+
+ const files = items.filter((item) => item.type === "file");
+
+ const results = await Promise.all(
+ files.map(async (file) => {
+ const filePath = `${DIR_TARGET}/${file.name}`;
+ const url = await getDownloadUrl(filePath);
+ return {
+ name: file.name,
+ path: filePath,
+ downloadUrl: url,
+ };
+ })
+ );
+
+ return results;
+}
+
+// contoh eksekusi
+(async () => {
+ try {
+ console.log("ambil gambar")
+ const urls = await getAllDownloadUrls();
+ await Bun.write("list_image2.json", JSON.stringify(urls))
+ console.log("selesai !")
+ } catch (err) {
+ console.error(err);
+ }
+})();
diff --git a/package.json b/package.json
index 472efe96..61958b92 100644
--- a/package.json
+++ b/package.json
@@ -3,9 +3,12 @@
"version": "0.1.5",
"private": true,
"scripts": {
- "dev": "bun --bun next dev",
- "build": "bun --bun next build",
- "start": "bun --bun next start"
+ "dev": "next dev",
+ "build": "next build",
+ "start": "next start",
+ "test:api": "vitest run",
+ "test:e2e": "playwright test",
+ "test": "bun run test:api && bun run test:e2e"
},
"prisma": {
"seed": "bun run prisma/seed.ts"
@@ -19,6 +22,7 @@
"@elysiajs/static": "^1.3.0",
"@elysiajs/stream": "^1.1.0",
"@elysiajs/swagger": "^1.2.0",
+ "@emotion/react": "^11.14.0",
"@mantine/carousel": "^7.16.2",
"@mantine/charts": "^7.17.1",
"@mantine/core": "^7.17.4",
@@ -26,6 +30,7 @@
"@mantine/dropzone": "^8.1.1",
"@mantine/form": "^8.1.0",
"@mantine/hooks": "^7.17.4",
+ "@mantine/modals": "^8.3.6",
"@mantine/tiptap": "^7.17.4",
"@paljs/types": "^8.1.0",
"@prisma/client": "^6.3.1",
@@ -43,20 +48,26 @@
"@types/bun": "^1.2.2",
"@types/leaflet": "^1.9.20",
"@types/lodash": "^4.17.16",
+ "@types/mime-types": "^3.0.1",
"@types/nodemailer": "^7.0.2",
"add": "^2.0.6",
"adm-zip": "^0.5.16",
"animate.css": "^4.1.1",
+ "async-mutex": "^0.5.0",
"bcryptjs": "^3.0.2",
"bun": "^1.2.2",
"chart.js": "^4.4.8",
"classnames": "^2.5.1",
+ "cli-progress": "^3.12.0",
"colors": "^1.4.0",
+ "date-fns": "^4.1.0",
"dayjs": "^1.11.13",
+ "dompurify": "^3.3.1",
"dotenv": "^17.2.3",
"elysia": "^1.3.5",
- "embla-carousel-autoplay": "^8.5.2",
- "embla-carousel-react": "^7.1.0",
+ "embla-carousel": "^8.6.0",
+ "embla-carousel-autoplay": "^8.6.0",
+ "embla-carousel-react": "^8.6.0",
"extract-zip": "^2.0.1",
"form-data": "^4.0.2",
"framer-motion": "^12.23.5",
@@ -68,6 +79,7 @@
"leaflet": "^1.9.4",
"list": "^2.0.19",
"lodash": "^4.17.21",
+ "mime-types": "^3.0.2",
"motion": "^12.4.1",
"nanoid": "^5.1.5",
"next": "^15.5.2",
@@ -80,6 +92,7 @@
"prisma": "^6.3.1",
"react": "^19.0.0",
"react-dom": "^19.0.0",
+ "react-exif-orientation-img": "^0.1.5",
"react-international-phone": "^4.6.0",
"react-leaflet": "^5.0.0",
"react-simple-toasts": "^6.1.0",
@@ -97,16 +110,24 @@
},
"devDependencies": {
"@eslint/eslintrc": "^3",
+ "@playwright/test": "^1.58.2",
+ "@testing-library/jest-dom": "^6.9.1",
+ "@types/cli-progress": "^3.11.6",
+ "@types/dompurify": "^3.2.0",
"@types/jsonwebtoken": "^9.0.10",
"@types/node": "^20",
"@types/react": "^19",
"@types/react-dom": "^19",
+ "@vitest/ui": "^4.0.18",
"eslint": "^9",
"eslint-config-next": "15.1.6",
+ "jsdom": "^28.0.0",
+ "msw": "^2.12.9",
"parcel": "^2.6.2",
"postcss": "^8.5.1",
"postcss-preset-mantine": "^1.17.0",
"postcss-simple-vars": "^7.0.1",
- "typescript": "^5"
+ "typescript": "^5",
+ "vitest": "^4.0.18"
}
}
diff --git a/playwright-report/data/3262232ac1998014dfaa14b6734778979a7c99c4.md b/playwright-report/data/3262232ac1998014dfaa14b6734778979a7c99c4.md
new file mode 100644
index 00000000..c0cf174c
--- /dev/null
+++ b/playwright-report/data/3262232ac1998014dfaa14b6734778979a7c99c4.md
@@ -0,0 +1,208 @@
+# Page snapshot
+
+```yaml
+- generic [active] [ref=e1]:
+ - generic [ref=e2]:
+ - generic [ref=e7]:
+ - button "Darmasaba Logo" [ref=e8] [cursor=pointer]:
+ - img "Darmasaba Logo" [ref=e10]
+ - button "PPID" [ref=e11] [cursor=pointer]:
+ - generic [ref=e13]: PPID
+ - button "Desa" [ref=e14] [cursor=pointer]:
+ - generic [ref=e16]: Desa
+ - button "Kesehatan" [ref=e17] [cursor=pointer]:
+ - generic [ref=e19]: Kesehatan
+ - button "Keamanan" [ref=e20] [cursor=pointer]:
+ - generic [ref=e22]: Keamanan
+ - button "Ekonomi" [ref=e23] [cursor=pointer]:
+ - generic [ref=e25]: Ekonomi
+ - button "Inovasi" [ref=e26] [cursor=pointer]:
+ - generic [ref=e28]: Inovasi
+ - button "Lingkungan" [ref=e29] [cursor=pointer]:
+ - generic [ref=e31]: Lingkungan
+ - button "Pendidikan" [ref=e32] [cursor=pointer]:
+ - generic [ref=e34]: Pendidikan
+ - button "Musik" [ref=e35] [cursor=pointer]:
+ - generic [ref=e37]: Musik
+ - button [ref=e38] [cursor=pointer]:
+ - img [ref=e40]
+ - generic [ref=e46]:
+ - generic [ref=e51]:
+ - generic [ref=e52]:
+ - generic [ref=e53]:
+ - img "Logo Darmasaba" [ref=e55]
+ - img "Logo Pudak" [ref=e57]
+ - generic [ref=e63]:
+ - generic [ref=e65]:
+ - generic [ref=e66]:
+ - img [ref=e67]
+ - paragraph [ref=e71]: Jam Operasional
+ - generic [ref=e72]:
+ - generic [ref=e74]: Buka
+ - paragraph [ref=e75]: 07:30 - 15:30
+ - generic [ref=e77]:
+ - generic [ref=e78]:
+ - img [ref=e79]
+ - paragraph [ref=e82]: Hari Ini
+ - generic [ref=e83]:
+ - paragraph [ref=e84]: Status Kantor
+ - paragraph [ref=e85]: Sedang Beroperasi
+ - paragraph [ref=e95]: Bagikan ide, kritik, atau saran Anda untuk mendukung pembangunan desa. Semua lebih mudah dengan fitur interaktif yang kami sediakan.
+ - generic [ref=e102]:
+ - generic [ref=e103]: Browser Anda tidak mendukung video.
+ - generic [ref=e106]:
+ - heading "Penghargaan Desa" [level=2] [ref=e107]
+ - paragraph [ref=e110]: Sedang memuat data penghargaan...
+ - button "Lihat semua penghargaan" [ref=e111] [cursor=pointer]:
+ - generic [ref=e112]:
+ - paragraph [ref=e114]: Lihat Semua Penghargaan
+ - img [ref=e116]
+ - generic [ref=e119]:
+ - generic [ref=e121]:
+ - heading "Layanan" [level=1] [ref=e122]
+ - paragraph [ref=e123]: Layanan adalah fitur yang membantu warga desa mengakses berbagai kebutuhan administrasi, informasi, dan bantuan secara cepat, mudah, dan transparan. Dengan fitur ini, semua layanan desa ada dalam genggaman Anda!
+ - link "Detail" [ref=e125] [cursor=pointer]:
+ - /url: /darmasaba/desa/layanan
+ - generic [ref=e127]: Detail
+ - separator [ref=e129]
+ - generic [ref=e130]:
+ - generic [ref=e131]:
+ - paragraph [ref=e132]: Potensi Desa
+ - paragraph [ref=e133]: Jelajahi berbagai potensi dan peluang yang dimiliki desa. Fitur ini membantu warga maupun pemerintah desa dalam merencanakan dan mengembangkan program berbasis kekuatan lokal.
+ - paragraph [ref=e136]: Sedang memuat potensi desa...
+ - button "Lihat Semua Potensi" [ref=e139] [cursor=pointer]:
+ - generic [ref=e140]:
+ - generic [ref=e141]: Lihat Semua Potensi
+ - img [ref=e143]
+ - separator [ref=e146]
+ - generic [ref=e147]:
+ - generic [ref=e148]:
+ - paragraph [ref=e150]: Desa Anti Korupsi
+ - paragraph [ref=e151]: Desa antikorupsi mendorong pemerintahan jujur dan transparan. Keuangan desa dikelola secara terbuka dengan melibatkan warga dalam pengawasan anggaran, sehingga digunakan tepat sasaran dan sesuai kebutuhan masyarakat.
+ - link "Selengkapnya" [ref=e153] [cursor=pointer]:
+ - /url: /darmasaba/desa-anti-korupsi/detail
+ - generic [ref=e155]: Selengkapnya
+ - paragraph [ref=e158]: Memuat Data...
+ - generic [ref=e166]:
+ - heading "SDGs Desa" [level=1] [ref=e168]
+ - paragraph [ref=e169]: SDGs Desa adalah upaya desa untuk menciptakan pembangunan yang maju, inklusif, dan berkelanjutan melalui 17 tujuan mulai dari pengentasan kemiskinan, pendidikan, kesehatan, hingga pelestarian lingkungan.
+ - generic [ref=e170]:
+ - generic [ref=e171]:
+ - img [ref=e172]
+ - paragraph [ref=e175]: Data SDGs Desa belum tersedia
+ - link "Jelajahi Semua Tujuan SDGs Desa" [ref=e177] [cursor=pointer]:
+ - /url: /darmasaba/sdgs-desa
+ - paragraph [ref=e180]: Jelajahi Semua Tujuan SDGs Desa
+ - generic [ref=e181]:
+ - generic [ref=e183]:
+ - heading "APBDes" [level=1] [ref=e184]
+ - paragraph [ref=e185]: Transparansi APBDes Darmasaba adalah langkah nyata menuju tata kelola desa yang bersih, terbuka, dan bertanggung jawab.
+ - link "Lihat Semua Data" [ref=e187] [cursor=pointer]:
+ - /url: /darmasaba/apbdes
+ - generic [ref=e189]: Lihat Semua Data
+ - generic [ref=e191]:
+ - paragraph [ref=e193]: Pilih Tahun APBDes
+ - generic [ref=e194]:
+ - textbox "Pilih Tahun APBDes" [ref=e195]:
+ - /placeholder: Pilih tahun
+ - generic:
+ - img
+ - paragraph [ref=e197]: Tidak ada data APBDes untuk tahun yang dipilih.
+ - generic [ref=e202]:
+ - heading "Prestasi Desa" [level=1] [ref=e203]
+ - paragraph [ref=e204]: Kami bangga dengan pencapaian desa hingga saat ini. Semoga prestasi ini menjadi inspirasi untuk terus berkarya dan berinovasi demi kemajuan bersama.
+ - link "Lihat Semua Prestasi" [ref=e205] [cursor=pointer]:
+ - /url: /darmasaba/prestasi-desa
+ - generic [ref=e207]: Lihat Semua Prestasi
+ - button [ref=e211] [cursor=pointer]:
+ - img [ref=e214]
+ - button [ref=e219] [cursor=pointer]:
+ - img [ref=e221]
+ - generic [ref=e225]:
+ - contentinfo [ref=e228]:
+ - generic [ref=e230]:
+ - generic [ref=e231]:
+ - heading "Komitmen Layanan Kami" [level=2] [ref=e232]
+ - generic [ref=e233]:
+ - generic [ref=e234]:
+ - paragraph [ref=e235]: "1. Transparansi:"
+ - paragraph [ref=e236]: Pengelolaan dana desa dilakukan secara terbuka agar masyarakat dapat memahami dan memantau penggunaan anggaran.
+ - generic [ref=e237]:
+ - paragraph [ref=e238]: "2. Profesionalisme:"
+ - paragraph [ref=e239]: Layanan desa diberikan secara cepat, adil, dan profesional demi kepuasan masyarakat.
+ - generic [ref=e240]:
+ - paragraph [ref=e241]: "3. Partisipasi:"
+ - paragraph [ref=e242]: Masyarakat dilibatkan aktif dalam pengambilan keputusan demi pembangunan desa yang berhasil.
+ - generic [ref=e243]:
+ - paragraph [ref=e244]: "4. Inovasi:"
+ - paragraph [ref=e245]: Kami terus berinovasi, termasuk melalui teknologi, agar layanan semakin mudah diakses.
+ - generic [ref=e246]:
+ - paragraph [ref=e247]: "5. Keadilan:"
+ - paragraph [ref=e248]: Kebijakan dan program disusun untuk memberi manfaat yang merata bagi seluruh warga.
+ - generic [ref=e249]:
+ - paragraph [ref=e250]: "6. Pemberdayaan:"
+ - paragraph [ref=e251]: Masyarakat didukung melalui pelatihan, pendampingan, dan pengembangan usaha lokal.
+ - generic [ref=e252]:
+ - paragraph [ref=e253]: "7. Ramah Lingkungan:"
+ - paragraph [ref=e254]: Seluruh kegiatan pembangunan memperhatikan keberlanjutan demi menjaga alam dan kesehatan warga.
+ - separator [ref=e255]
+ - generic [ref=e256]:
+ - heading "Visi Kami" [level=2] [ref=e257]
+ - paragraph [ref=e258]: Dengan visi ini, kami berkomitmen menjadikan desa sebagai tempat yang aman, sejahtera, dan nyaman bagi seluruh warga.
+ - paragraph [ref=e259]: Kami percaya kemajuan dimulai dari kerja sama antara pemerintah desa dan masyarakat, didukung tata kelola yang baik demi kepentingan bersama. Saran maupun keluhan dapat disampaikan melalui kontak di bawah ini.
+ - generic [ref=e260]:
+ - paragraph [ref=e261]: "\"Desa Kuat, Warga Sejahtera!\""
+ - button "Logo Desa" [ref=e262] [cursor=pointer]:
+ - generic [ref=e263]:
+ - img "Logo Desa"
+ - generic [ref=e265]:
+ - generic [ref=e267]:
+ - paragraph [ref=e268]: Tentang Darmasaba
+ - paragraph [ref=e269]: Darmasaba adalah desa budaya yang kaya akan tradisi dan nilai-nilai warisan Bali.
+ - generic [ref=e270]:
+ - link [ref=e271] [cursor=pointer]:
+ - /url: https://www.facebook.com/DarmasabaDesaku
+ - img [ref=e273]
+ - link [ref=e275] [cursor=pointer]:
+ - /url: https://www.instagram.com/ddarmasaba/
+ - img [ref=e277]
+ - link [ref=e280] [cursor=pointer]:
+ - /url: https://www.youtube.com/channel/UCtPw9WOQO7d2HIKzKgel4Xg
+ - img [ref=e282]
+ - link [ref=e285] [cursor=pointer]:
+ - /url: https://www.tiktok.com/@desa.darmasaba?is_from_webapp=1&sender_device=pc
+ - img [ref=e287]
+ - generic [ref=e290]:
+ - paragraph [ref=e291]: Layanan Desa
+ - link "Administrasi Kependudukan" [ref=e292] [cursor=pointer]:
+ - /url: /darmasaba/desa/layanan/
+ - link "Layanan Sosial" [ref=e293] [cursor=pointer]:
+ - /url: /darmasaba/ekonomi/program-kemiskinan
+ - link "Pengaduan Masyarakat" [ref=e294] [cursor=pointer]:
+ - /url: /darmasaba/keamanan/laporan-publik
+ - link "Informasi Publik" [ref=e295] [cursor=pointer]:
+ - /url: /darmasaba/ppid/daftar-informasi-publik-desa-darmasaba
+ - generic [ref=e297]:
+ - paragraph [ref=e298]: Tautan Penting
+ - link "Portal Badung" [ref=e299] [cursor=pointer]:
+ - /url: /darmasaba/desa/berita/semua
+ - link "E-Government" [ref=e300] [cursor=pointer]:
+ - /url: /darmasaba/inovasi/desa-digital-smart-village
+ - link "Transparansi" [ref=e301] [cursor=pointer]:
+ - /url: /darmasaba/ppid/daftar-informasi-publik-desa-darmasaba
+ - generic [ref=e303]:
+ - paragraph [ref=e304]: Berlangganan Info
+ - paragraph [ref=e305]: Dapatkan kabar terbaru tentang program dan kegiatan desa langsung ke email Anda.
+ - generic [ref=e306]:
+ - generic [ref=e308]:
+ - textbox "Masukkan email Anda" [ref=e309]
+ - img [ref=e311]
+ - button "Daftar" [ref=e314] [cursor=pointer]:
+ - generic [ref=e316]: Daftar
+ - separator [ref=e317]
+ - paragraph [ref=e318]: © 2025 Desa Darmasaba. Hak cipta dilindungi.
+ - region "Notifications Alt+T"
+ - button "Open Next.js Dev Tools" [ref=e324] [cursor=pointer]:
+ - img [ref=e325]
+ - alert [ref=e328]
+```
\ No newline at end of file
diff --git a/playwright-report/index.html b/playwright-report/index.html
new file mode 100644
index 00000000..412d60b5
--- /dev/null
+++ b/playwright-report/index.html
@@ -0,0 +1,85 @@
+
+
+
+
+
+
+
+
+ Playwright Test Report
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/playwright.config.ts b/playwright.config.ts
new file mode 100644
index 00000000..c5c67f4c
--- /dev/null
+++ b/playwright.config.ts
@@ -0,0 +1,25 @@
+import { defineConfig, devices } from '@playwright/test';
+
+export default defineConfig({
+ testDir: './__tests__/e2e',
+ fullyParallel: true,
+ forbidOnly: !!process.env.CI,
+ retries: process.env.CI ? 2 : 0,
+ workers: process.env.CI ? 1 : undefined,
+ reporter: 'html',
+ use: {
+ baseURL: 'http://localhost:3000',
+ trace: 'on-first-retry',
+ },
+ projects: [
+ {
+ name: 'chromium',
+ use: { ...devices['Desktop Chrome'] },
+ },
+ ],
+ webServer: {
+ command: 'bun run dev',
+ url: 'http://localhost:3000',
+ reuseExistingServer: !process.env.CI,
+ },
+});
diff --git a/postcss.config.cjs b/postcss.config.cjs
index 069b0528..9e9652a2 100644
--- a/postcss.config.cjs
+++ b/postcss.config.cjs
@@ -1,14 +1,15 @@
module.exports = {
- plugins: {
- 'postcss-preset-mantine': {},
- 'postcss-simple-vars': {
- variables: {
- 'mantine-breakpoint-xs': '36em',
- 'mantine-breakpoint-sm': '48em',
- 'mantine-breakpoint-md': '62em',
- 'mantine-breakpoint-lg': '75em',
- 'mantine-breakpoint-xl': '88em',
- },
+ plugins: {
+ 'postcss-preset-mantine': {},
+ 'postcss-simple-vars': {
+ variables: {
+ /* Mobile first */
+ 'mantine-breakpoint-xs': '30em', // 480px → mobile kecil–normal
+ 'mantine-breakpoint-sm': '48em', // 768px → tablet / mobile landscape
+ 'mantine-breakpoint-md': '64em', // 1024px → laptop & desktop kecil
+ 'mantine-breakpoint-lg': '80em', // 1280px → desktop standar
+ 'mantine-breakpoint-xl': '90em', // 1440px+ → desktop besar
},
},
- };
\ No newline at end of file
+ },
+};
diff --git a/prisma/_seeder_list/desa/berita/seed_berita.ts b/prisma/_seeder_list/desa/berita/seed_berita.ts
new file mode 100644
index 00000000..1d1fb9f0
--- /dev/null
+++ b/prisma/_seeder_list/desa/berita/seed_berita.ts
@@ -0,0 +1,71 @@
+import prisma from "@/lib/prisma";
+import kategoriBerita from "../../../data/desa/berita/kategori-berita.json";
+import beritaJson from "../../../data/desa/berita/berita.json";
+
+export async function seedBerita() {
+ // ================== SUBMENU BERITA ========================
+ console.log("🔄 Seeding Kategori Berita...");
+ for (const k of kategoriBerita) {
+ await prisma.kategoriBerita.upsert({
+ where: {
+ name: k.name, // ✅ cocok dengan @unique
+ },
+ update: {
+ name: k.name,
+ isActive: true,
+ },
+ create: {
+ id: k.id, // ✅ id tetap bisa disimpan
+ name: k.name,
+ isActive: true,
+ },
+ });
+ }
+
+ console.log("kategori berita success ...");
+
+ console.log("🔄 Seeding Berita...");
+
+ for (const b of beritaJson) {
+ let imageId: string | null = null;
+
+ if (b.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: b.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for berita "${b.judul}": ${b.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.berita.upsert({
+ where: { id: b.id },
+ update: {
+ judul: b.judul,
+ deskripsi: b.deskripsi,
+ content: b.content,
+ kategoriBeritaId: b.kategoriBeritaId,
+ imageId,
+ },
+ create: {
+ id: b.id,
+ judul: b.judul,
+ deskripsi: b.deskripsi,
+ content: b.content,
+ kategoriBeritaId: b.kategoriBeritaId,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Berita seeded: ${b.judul}`);
+ }
+
+ console.log("🎉 Berita seed selesai");
+}
+
diff --git a/prisma/_seeder_list/desa/gallery/foto/seed_foto.ts b/prisma/_seeder_list/desa/gallery/foto/seed_foto.ts
new file mode 100644
index 00000000..6abf33c2
--- /dev/null
+++ b/prisma/_seeder_list/desa/gallery/foto/seed_foto.ts
@@ -0,0 +1,40 @@
+import prisma from "@/lib/prisma";
+import foto from "../../../../data/desa/gallery/foto/foto.json";
+
+export async function seedFoto() {
+ console.log("🔄 Seeding Foto...");
+ for (const f of foto) {
+ let imagesId: string | null = null;
+
+ if (f.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: f.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for foto "${f.name}": ${f.imageName}`,
+ );
+ } else {
+ imagesId = image.id;
+ }
+ }
+
+ await prisma.galleryFoto.upsert({
+ where: { id: f.id },
+ update: {
+ name: f.name,
+ deskripsi: f.deskripsi,
+ imagesId,
+ },
+ create: {
+ id: f.id,
+ name: f.name,
+ deskripsi: f.deskripsi,
+ imagesId,
+ },
+ });
+ }
+ console.log("✅ Foto seeding completed");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/desa/gallery/video/seed_video.ts b/prisma/_seeder_list/desa/gallery/video/seed_video.ts
new file mode 100644
index 00000000..42d97860
--- /dev/null
+++ b/prisma/_seeder_list/desa/gallery/video/seed_video.ts
@@ -0,0 +1,25 @@
+import prisma from "@/lib/prisma";
+import galleryVideo from "../../../../data/desa/gallery/video/video.json";
+
+export async function seedVideo() {
+ console.log("🔄 Seeding Gallery Video...");
+ for (const v of galleryVideo) {
+ await prisma.galleryVideo.upsert({
+ where: {
+ id: v.id,
+ },
+ update: {
+ name: v.judul,
+ deskripsi: v.deskripsi,
+ linkVideo: v.linkVideo,
+ },
+ create: {
+ name: v.judul,
+ deskripsi: v.deskripsi,
+ linkVideo: v.linkVideo,
+ },
+ });
+ }
+
+ console.log("gallery video success ...");
+}
diff --git a/prisma/_seeder_list/desa/layanan/seed_layanan.ts b/prisma/_seeder_list/desa/layanan/seed_layanan.ts
new file mode 100644
index 00000000..a242458c
--- /dev/null
+++ b/prisma/_seeder_list/desa/layanan/seed_layanan.ts
@@ -0,0 +1,128 @@
+import prisma from "@/lib/prisma";
+import pelayananSuratKeterangan from "../../../data/desa/layanan/pelayananSuratKeterangan.json";
+import pelayananTelunjukSaktiDesa from "../../../data/desa/layanan/pelayananTelunjukSaktiDesa.json";
+import pelayananPerizinanBerusaha from "../../../data/desa/layanan/pelayananPerizinanBerusaha.json";
+import pelayananPendudukNonPermanen from "../../../data/desa/layanan/pelayananPendudukNonPermanen.json";
+
+export async function seedLayanan() {
+ console.log("🔄 Seeding Pelayanan Surat Keterangan...");
+
+ for (const p of pelayananSuratKeterangan) {
+ const existing = await prisma.pelayananSuratKeterangan.findUnique({
+ where: { id: p.id },
+ select: { imageId: true, image2Id: true }, // 📌 tambahkan image2Id
+ });
+
+ // 1️⃣ Handle imageId
+ let imageId = existing?.imageId ?? null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for pelayanan surat keterangan 1 "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ // 2️⃣ Handle image2Id
+ let image2Id = existing?.image2Id ?? null;
+
+ if (p.image2Name) {
+ const image2 = await prisma.fileStorage.findUnique({
+ where: { name: p.image2Name },
+ select: { id: true },
+ });
+
+ if (!image2) {
+ console.warn(
+ `⚠️ Image not found for pelayanan surat keterangan 2 "${p.name}": ${p.image2Name}`,
+ );
+ } else {
+ image2Id = image2.id;
+ }
+ }
+
+ // 3️⃣ Upsert dengan kedua image
+ await prisma.pelayananSuratKeterangan.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ image2Id, // 📌 tambahkan ini
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ image2Id, // 📌 tambahkan ini
+ },
+ });
+ }
+ console.log("✅ Pelayanan Surat Keterangan success...");
+
+ for (const p of pelayananTelunjukSaktiDesa) {
+ await prisma.pelayananTelunjukSaktiDesa.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ link: p.link,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ link: p.link,
+ },
+ });
+ }
+ console.log("pelayanan telunjuk sakti desa success ...");
+
+ for (const l of pelayananPerizinanBerusaha) {
+ await prisma.pelayananPerizinanBerusaha.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ name: l.name,
+ deskripsi: l.deskripsi,
+ link: l.link,
+ },
+ create: {
+ id: l.id,
+ name: l.name,
+ deskripsi: l.deskripsi,
+ link: l.link,
+ },
+ });
+ }
+
+ console.log("pelayanan perizinan berusaha success ...");
+
+ for (const l of pelayananPendudukNonPermanen) {
+ await prisma.pelayananPendudukNonPermanen.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ name: l.name,
+ deskripsi: l.deskripsi,
+ },
+ create: {
+ id: l.id,
+ name: l.name,
+ deskripsi: l.deskripsi,
+ },
+ });
+ }
+ console.log("pelayanan penduduk non permanen success ...");
+}
diff --git a/prisma/_seeder_list/desa/penghargaan/penghargaan.ts b/prisma/_seeder_list/desa/penghargaan/penghargaan.ts
new file mode 100644
index 00000000..b696ae75
--- /dev/null
+++ b/prisma/_seeder_list/desa/penghargaan/penghargaan.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import penghargaan from "../../../data/desa/penghargaan/penghargaan.json"
+
+export async function seedPenghargaan() {
+ console.log("🔄 Seeding Penghargaan...");
+ for (const m of penghargaan) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for penghargaan "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.penghargaan.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ juara: m.juara,
+ deskripsi: m.deskripsi,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ juara: m.juara,
+ deskripsi: m.deskripsi,
+ imageId,
+ },
+ });
+ }
+
+ console.log("penghargaan success ...");
+}
+
\ No newline at end of file
diff --git a/prisma/_seeder_list/desa/pengumuman/seed_pengumuman.ts b/prisma/_seeder_list/desa/pengumuman/seed_pengumuman.ts
new file mode 100644
index 00000000..184fc804
--- /dev/null
+++ b/prisma/_seeder_list/desa/pengumuman/seed_pengumuman.ts
@@ -0,0 +1,43 @@
+import prisma from "@/lib/prisma";
+import { safeSeedUnique } from "../../../safeseedUnique";
+import kategoriPengumuman from "../../../data/desa/pengumuman/kategori-pengumuman.json";
+import pengumuman from "../../../data/desa/pengumuman/pengumuman.json";
+
+export async function seedPengumuman() {
+ console.log("🔄 Seeding Kategori Pengumuman...");
+ for (const c of kategoriPengumuman) {
+ await safeSeedUnique(
+ "categoryPengumuman",
+ { name: c.name }, // ✅ where clause
+ {
+ id: c.id,
+ name: c.name,
+ },
+ );
+ }
+
+ console.log("kategori pengumuman success ...");
+
+ console.log("🔄 Seeding Pengumuman...");
+ for (const p of pengumuman) {
+ await prisma.pengumuman.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ content: p.content,
+ categoryPengumumanId: p.categoryPengumumanId,
+ },
+ create: {
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ content: p.content,
+ categoryPengumumanId: p.categoryPengumumanId,
+ },
+ });
+ }
+
+ console.log("pengumuman success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/desa/potensi/seed_potensi.ts b/prisma/_seeder_list/desa/potensi/seed_potensi.ts
new file mode 100644
index 00000000..7e1ad408
--- /dev/null
+++ b/prisma/_seeder_list/desa/potensi/seed_potensi.ts
@@ -0,0 +1,64 @@
+import prisma from "@/lib/prisma";
+import kategoriPotensi from "../../../data/desa/potensi/kategori-potensi.json";
+import potensiDesa from "../../../data/desa/potensi/potensi-desa.json";
+
+export async function seedPotensi() {
+ console.log("🔄Seeding Kategori Potensi Desa ...");
+ for (const c of kategoriPotensi) {
+ await prisma.kategoriPotensi.upsert({
+ where: {
+ id: c.id,
+ },
+ update: {
+ nama: c.nama,
+ },
+ create: {
+ id: c.id,
+ nama: c.nama,
+ },
+ });
+ }
+
+ console.log("kategori Potensi success ...");
+
+ console.log("🔄 Seeding Potensi Desa...");
+ for (const m of potensiDesa) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for potensi desa "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.potensiDesa.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ deskripsi: m.deskripsi,
+ content: m.content,
+ kategoriId: m.kategoriId,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ deskripsi: m.deskripsi,
+ content: m.content,
+ kategoriId: m.kategoriId,
+ imageId,
+ },
+ });
+ }
+
+ console.log("potensi desa success ...");
+}
diff --git a/prisma/_seeder_list/desa/profile-desa/seed_profile_desa.ts b/prisma/_seeder_list/desa/profile-desa/seed_profile_desa.ts
new file mode 100644
index 00000000..68af0dca
--- /dev/null
+++ b/prisma/_seeder_list/desa/profile-desa/seed_profile_desa.ts
@@ -0,0 +1,168 @@
+import prisma from "@/lib/prisma";
+import lambangDesa from "../../../data/desa/profile/lambang_desa.json";
+import maskotDesa from "../../../data/desa/profile/maskot_desa.json";
+import profilePerbekel from "../../../data/desa/profile/profil_perbekel.json";
+import profileDesaImage from "../../../data/desa/profile/profileDesaImage.json";
+import sejarahDesa from "../../../data/desa/profile/sejarah_desa.json";
+import visiMisiDesa from "../../../data/desa/profile/visi_misi_desa.json";
+
+export async function seedProfileDesa() {
+ // =========== SEJARAH DESA ===========
+ for (const l of sejarahDesa) {
+ await prisma.sejarahDesa.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ create: {
+ id: l.id,
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ });
+ }
+
+ console.log("sejarah desa success ...");
+
+ // =========== VISI MISI DESA ===========
+ for (const l of visiMisiDesa) {
+ await prisma.visiMisiDesa.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ visi: l.visi,
+ misi: l.misi,
+ },
+ create: {
+ id: l.id,
+ visi: l.visi,
+ misi: l.misi,
+ },
+ });
+ }
+
+ console.log("visi misi desa success ...");
+
+ // =========== MASKOT DESA ===========
+ for (const l of maskotDesa) {
+ await prisma.maskotDesa.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ create: {
+ id: l.id,
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ });
+ }
+
+ console.log("maskot desa success ...");
+
+ console.log("🔄 Seeding Profile Desa Image...");
+ for (const m of profileDesaImage) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for profile desa image "${m.label}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.profileDesaImage.upsert({
+ where: { id: m.id },
+ update: {
+ label: m.label,
+ maskotDesaId: m.maskotDesaId,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ label: m.label,
+ maskotDesaId: m.maskotDesaId,
+ imageId,
+ },
+ });
+ }
+
+ console.log("profile desa image success ...");
+
+ // =========== LAMBANG DESA ===========
+ for (const l of lambangDesa) {
+ await prisma.lambangDesa.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ create: {
+ id: l.id,
+ judul: l.judul,
+ deskripsi: l.deskripsi,
+ },
+ });
+ }
+
+ console.log("lambang desa success ...");
+
+ // =========== PROFILE PERBEKEL PROFILE DESA ===========
+ console.log("🔄 Seeding Profile Perbekel...");
+ for (const m of profilePerbekel) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for profile perbekel "${m.biodata}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.profilPerbekel.upsert({
+ where: { id: m.id },
+ update: {
+ biodata: m.biodata,
+ pengalaman: m.pengalaman,
+ pengalamanOrganisasi: m.pengalamanOrganisasi,
+ programUnggulan: m.programUnggulan,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ biodata: m.biodata,
+ pengalaman: m.pengalaman,
+ pengalamanOrganisasi: m.pengalamanOrganisasi,
+ programUnggulan: m.programUnggulan,
+ imageId,
+ },
+ });
+ }
+
+ console.log("profile perbekel desa success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/desa/profile-desa/seed_profile_perbekel.ts b/prisma/_seeder_list/desa/profile-desa/seed_profile_perbekel.ts
new file mode 100644
index 00000000..1507fe32
--- /dev/null
+++ b/prisma/_seeder_list/desa/profile-desa/seed_profile_perbekel.ts
@@ -0,0 +1,42 @@
+import prisma from "@/lib/prisma";
+import perbekelDariMasaKeMasa from "../../../data/desa/profile/profile-perbekel-lalu.json";
+
+export async function seedProfilePerbekel() {
+ console.log("🔄 Seeding Perbekel Dari Masa Ke Masa...");
+for (const p of perbekelDariMasaKeMasa) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for Perbekel Dari Masa Ke Masa "${p.nama}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.perbekelDariMasaKeMasa.upsert({
+ where: { id: p.id },
+ update: {
+ nama: p.nama,
+ periode: p.periode,
+ daerah: p.daerah,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ nama: p.nama,
+ periode: p.periode,
+ daerah: p.daerah,
+ imageId,
+ },
+ });
+}
+console.log("✅ Pejabat Desa seeding completed");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ekonomi/seed_demografi_pekerjaan.ts b/prisma/_seeder_list/ekonomi/seed_demografi_pekerjaan.ts
new file mode 100644
index 00000000..41d4dc07
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_demografi_pekerjaan.ts
@@ -0,0 +1,25 @@
+import prisma from "@/lib/prisma";
+import demografiPekerjaan from "../../data/ekonomi/demografi-pekerjaan/demografi-pekerjaan.json";
+
+export async function seedDemografiPekerjaan() {
+ console.log("🔄 Seeding Demografi Pekerjaan...");
+ for (const k of demografiPekerjaan) {
+ await prisma.dataDemografiPekerjaan.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ pekerjaan: k.pekerjaan,
+ lakiLaki: k.lakiLaki,
+ perempuan: k.perempuan,
+ },
+ create: {
+ id: k.id,
+ pekerjaan: k.pekerjaan,
+ lakiLaki: k.lakiLaki,
+ perempuan: k.perempuan,
+ },
+ });
+ }
+ console.log("✅ Demografi Pekerjaan seeded successfully");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_jumlah_penduduk_miskin.ts b/prisma/_seeder_list/ekonomi/seed_jumlah_penduduk_miskin.ts
new file mode 100644
index 00000000..48374742
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_jumlah_penduduk_miskin.ts
@@ -0,0 +1,23 @@
+import prisma from "@/lib/prisma";
+import jumlahPendudukMiskin from "../../data/ekonomi/jumlah-penduduk-miskin/jumlah-penduduk-miskin.json";
+
+export async function seedJumlahPendudukMiskin() {
+ console.log("🔄 Seeding Jumlah Penduduk Miskin...");
+ for (const k of jumlahPendudukMiskin) {
+ await prisma.grafikJumlahPendudukMiskin.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ year: k.year,
+ totalPoorPopulation: k.totalPoorPopulation,
+ },
+ create: {
+ id: k.id,
+ year: k.year,
+ totalPoorPopulation: k.totalPoorPopulation,
+ },
+ });
+ }
+ console.log("✅ Jumlah Penduduk Miskin seeded successfully");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_jumlah_pengangguran.ts b/prisma/_seeder_list/ekonomi/seed_jumlah_pengangguran.ts
new file mode 100644
index 00000000..b1f2cb03
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_jumlah_pengangguran.ts
@@ -0,0 +1,27 @@
+import prisma from "@/lib/prisma";
+import jumlahPengangguran from "../../data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json";
+
+export async function seedJumlahPengangguran() {
+ for (const d of jumlahPengangguran) {
+ await prisma.detailDataPengangguran.upsert({
+ where: {
+ month_year: { month: d.month, year: d.year },
+ },
+ update: {
+ totalUnemployment: d.totalUnemployment,
+ educatedUnemployment: d.educatedUnemployment,
+ uneducatedUnemployment: d.uneducatedUnemployment,
+ percentageChange: d.percentageChange,
+ },
+ create: {
+ month: d.month,
+ year: d.year,
+ totalUnemployment: d.totalUnemployment,
+ educatedUnemployment: d.educatedUnemployment,
+ uneducatedUnemployment: d.uneducatedUnemployment,
+ percentageChange: d.percentageChange,
+ },
+ });
+ }
+ console.log("📊 detailDataPengangguran success ...");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_lowongan_kerja_lokal.ts b/prisma/_seeder_list/ekonomi/seed_lowongan_kerja_lokal.ts
new file mode 100644
index 00000000..95494b24
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_lowongan_kerja_lokal.ts
@@ -0,0 +1,35 @@
+import prisma from "@/lib/prisma";
+import lowonganKerjaLokal from "../../data/ekonomi/lowongan-kerja-lokal/lowongan-kerja-lokal.json";
+
+export async function seedLowonganKerjaLokal() {
+ console.log("🔄 Seeding Lowongan Kerja Lokal...");
+ for (const k of lowonganKerjaLokal) {
+ await prisma.lowonganPekerjaan.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ posisi: k.posisi,
+ namaPerusahaan: k.namaPerusahaan,
+ lokasi: k.lokasi,
+ tipePekerjaan: k.tipePekerjaan,
+ gaji: k.gaji,
+ deskripsi: k.deskripsi,
+ kualifikasi: k.kualifikasi,
+ notelp: k.notelp,
+ },
+ create: {
+ id: k.id,
+ posisi: k.posisi,
+ namaPerusahaan: k.namaPerusahaan,
+ lokasi: k.lokasi,
+ tipePekerjaan: k.tipePekerjaan,
+ gaji: k.gaji,
+ deskripsi: k.deskripsi,
+ kualifikasi: k.kualifikasi,
+ notelp: k.notelp,
+ },
+ });
+ }
+ console.log("✅ Lowongan Kerja Lokal seeded successfully");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_pasar_desa.ts b/prisma/_seeder_list/ekonomi/seed_pasar_desa.ts
new file mode 100644
index 00000000..f8ad4105
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_pasar_desa.ts
@@ -0,0 +1,91 @@
+import prisma from "@/lib/prisma";
+import kategoriProduk from "../../data/ekonomi/pasar-desa/kategori-produk.json";
+import pasarDesa from "../../data/ekonomi/pasar-desa/pasar-desa.json";
+import kategoriToPasar from "../../data/ekonomi/pasar-desa/kategori-to-pasar.json";
+
+export async function seedPasarDesa() {
+ console.log("🔄 Seeding Kategori Produk...");
+ for (const k of kategoriProduk) {
+ await prisma.kategoriProduk.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ nama: k.nama,
+ },
+ create: {
+ id: k.id,
+ nama: k.nama,
+ },
+ });
+ }
+ console.log("✅ Kategori Produk seeded successfully");
+
+ console.log("🔄 Seeding Pasar Desa...");
+
+ for (const p of pasarDesa) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for pasar desa "${p.nama}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.pasarDesa.upsert({
+ where: { id: p.id },
+ update: {
+ nama: p.nama,
+ deskripsi: p.deskripsi,
+ harga: p.harga,
+ rating: p.rating,
+ alamatUsaha: p.alamatUsaha,
+ kontak: p.kontak,
+ imageId,
+ kategoriProdukId: p.kategoriProdukId,
+ },
+ create: {
+ id: p.id,
+ nama: p.nama,
+ deskripsi: p.deskripsi,
+ harga: p.harga,
+ rating: p.rating,
+ alamatUsaha: p.alamatUsaha,
+ kontak: p.kontak,
+ imageId,
+ kategoriProdukId: p.kategoriProdukId,
+ },
+ });
+
+ console.log(`✅ Pasar desa seeded: ${p.nama}`);
+ }
+
+ console.log("🎉 Pasar desa seed selesai");
+
+ console.log("🔄 Seeding Kategori To Pasar...");
+ for (const p of kategoriToPasar) {
+ await prisma.kategoriToPasar.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ kategoriId: p.kategoriId,
+ pasarDesaId: p.pasarDesaId,
+ },
+ create: {
+ id: p.id,
+ kategoriId: p.kategoriId,
+ pasarDesaId: p.pasarDesaId,
+ },
+ });
+ }
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_pendapatan_asli.ts b/prisma/_seeder_list/ekonomi/seed_pendapatan_asli.ts
new file mode 100644
index 00000000..93a30a5e
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_pendapatan_asli.ts
@@ -0,0 +1,81 @@
+import prisma from "@/lib/prisma";
+import apbdes from "../../data/ekonomi/pendapatan-asli-desa/apbDesa.json";
+import pendapatan from "../../data/ekonomi/pendapatan-asli-desa/pendapatanDesa.json";
+import belanja from "../../data/ekonomi/pendapatan-asli-desa/belanjaDesa.json";
+import pembiayaan from "../../data/ekonomi/pendapatan-asli-desa/pembiayaanDesa.json";
+
+export async function seedPendapatanAsli() {
+ console.log("🔄 Seeding Pendapatan Asli...");
+ for (const d of apbdes) {
+ await prisma.apbDesa.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ tahun: d.tahun,
+ },
+ create: {
+ id: d.id,
+ tahun: d.tahun,
+ },
+ });
+ }
+ console.log("✅ Pendapatan Asli seeded successfully");
+
+ console.log("🔄 Seeding Pendapatan...");
+ for (const d of pendapatan) {
+ await prisma.pendapatan.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ value: d.nilai
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ value: d.nilai
+ },
+ });
+ }
+ console.log("✅ Pendapatan seeded successfully");
+
+ console.log("🔄 Seeding Belanja...");
+ for (const d of belanja) {
+ await prisma.belanja.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ value: d.nilai
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ value: d.nilai
+ },
+ });
+ }
+ console.log("✅ Belanja seeded successfully");
+
+ console.log("🔄 Seeding Pembiayaan...");
+ for (const d of pembiayaan) {
+ await prisma.pembiayaan.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ value: d.nilai
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ value: d.nilai
+ },
+ });
+ }
+ console.log("✅ Pembiayaan seeded successfully");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_penduduk_usia_kerja_yang_menganggur.ts b/prisma/_seeder_list/ekonomi/seed_penduduk_usia_kerja_yang_menganggur.ts
new file mode 100644
index 00000000..ac64954f
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_penduduk_usia_kerja_yang_menganggur.ts
@@ -0,0 +1,50 @@
+import prisma from "@/lib/prisma";
+import grafikMenganggurBerdasarkanUsia from "../../data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-usia.json";
+import grafikMenganggurBerdasarkanPendidikan from "../../data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-pendidikan.json";
+
+export async function seedPendudukUsiaKerjaYangMenganggur() {
+ for (const p of grafikMenganggurBerdasarkanUsia) {
+ await prisma.grafikMenganggurBerdasarkanUsia.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ usia18_25: p.usia18_25,
+ usia26_35: p.usia26_35,
+ usia36_45: p.usia36_45,
+ usia46_keatas: p.usia46_keatas,
+ },
+ create: {
+ id: p.id,
+ usia18_25: p.usia18_25,
+ usia26_35: p.usia26_35,
+ usia36_45: p.usia36_45,
+ usia46_keatas: p.usia46_keatas,
+ },
+ });
+ }
+ console.log("📊 grafikMenganggurBerdasarkanUsia success ...");
+ for (const p of grafikMenganggurBerdasarkanPendidikan) {
+ await prisma.grafikMenganggurBerdasarkanPendidikan.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ SD: p.SD,
+ SMP: p.SMP,
+ SMA: p.SMA,
+ D3: p.D3,
+ S1: p.S1,
+ },
+ create: {
+ id: p.id,
+ SD: p.SD,
+ SMP: p.SMP,
+ SMA: p.SMA,
+ D3: p.D3,
+ S1: p.S1,
+ },
+ });
+ }
+ console.log("📊 grafikMenganggurBerdasarkanUsia success ...");
+}
diff --git a/prisma/_seeder_list/ekonomi/seed_program_kemiskinan.ts b/prisma/_seeder_list/ekonomi/seed_program_kemiskinan.ts
new file mode 100644
index 00000000..748b1f8b
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_program_kemiskinan.ts
@@ -0,0 +1,50 @@
+import prisma from "@/lib/prisma";
+import programKemiskinan from "../../data/ekonomi/program-kemiskinan/program-kemiskinan.json";
+import statistikKemiskinan from "../../data/ekonomi/program-kemiskinan/statistik-kemiskinan.json";
+
+export async function seedProgramKemiskinan() {
+ for (const s of statistikKemiskinan) {
+ await prisma.statistikKemiskinan.upsert({
+ where: { tahun: s.tahun }, // ✅ FIX
+ update: {
+ jumlah: s.jumlah,
+ },
+ create: {
+ id: s.id, // id boleh tetap
+ tahun: s.tahun,
+ jumlah: s.jumlah,
+ },
+ });
+ }
+
+ console.log("📊 Statistik Kemiskinan seeded successfully");
+
+ console.log("🔄 Seeding Program Kemiskinan...");
+ for (const k of programKemiskinan) {
+ await prisma.programKemiskinan.upsert({
+ where: { id: k.id },
+ update: {
+ nama: k.nama,
+ deskripsi: k.deskripsi,
+ icon: k.icon,
+ statistik: {
+ connect: {
+ tahun: k.tahun, // 👈 BUKAN ID
+ },
+ },
+ },
+ create: {
+ id: k.id,
+ nama: k.nama,
+ deskripsi: k.deskripsi,
+ icon: k.icon,
+ statistik: {
+ connect: {
+ tahun: k.tahun,
+ },
+ },
+ },
+ });
+ }
+ console.log("✅ Program Kemiskinan seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ekonomi/seed_sektor_unggulan_desa.ts b/prisma/_seeder_list/ekonomi/seed_sektor_unggulan_desa.ts
new file mode 100644
index 00000000..0bfd469e
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_sektor_unggulan_desa.ts
@@ -0,0 +1,25 @@
+import prisma from "@/lib/prisma";
+import sektorUnggulanDesa from "../../data/ekonomi/sektor-unggulan/sektor-unggulan.json";
+
+export async function seedSektorUnggulanDesa() {
+ console.log("🔄 Seeding Sektor Unggulan Desa...");
+ for (const k of sektorUnggulanDesa) {
+ await prisma.sektorUnggulanDesa.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ name: k.name,
+ description: k.description,
+ value: k.value,
+ },
+ create: {
+ id: k.id,
+ name: k.name,
+ description: k.description,
+ value: k.value,
+ },
+ });
+ }
+ console.log("✅ Sektor Unggulan Desa seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ekonomi/seed_struktur_bumdes.ts b/prisma/_seeder_list/ekonomi/seed_struktur_bumdes.ts
new file mode 100644
index 00000000..47f92ed7
--- /dev/null
+++ b/prisma/_seeder_list/ekonomi/seed_struktur_bumdes.ts
@@ -0,0 +1,58 @@
+import prisma from "@/lib/prisma";
+import posisiOrganisasiBumDes from "../../data/ekonomi/struktur-organisasi/posisi-organisasi-bumdes.json";
+import pegawai from "../../data/ekonomi/struktur-organisasi/pegawai-bumdes.json";
+
+export async function seedStrukturBumdes() {
+ const flattenedPosisi = posisiOrganisasiBumDes.flat();
+
+ // ✅ Urutkan berdasarkan hierarki
+ const sortedPosisi = flattenedPosisi.sort((a, b) => a.hierarki - b.hierarki);
+
+ for (const p of sortedPosisi) {
+ console.log(`Seeding: ${p.nama} (id: ${p.id}, parent: ${p.parentId})`);
+ if (p.parentId) {
+ const parentExists = flattenedPosisi.some((pos) => pos.id === p.parentId);
+ if (!parentExists) {
+ console.warn(
+ `⚠️ Parent tidak ditemukan: ${p.parentId} untuk ${p.nama}`,
+ );
+ continue;
+ }
+ }
+ await prisma.posisiOrganisasiBumDes.upsert({
+ where: { id: p.id },
+ update: p,
+ create: p,
+ });
+ }
+ console.log("posisi organisasi berhasil");
+ for (const p of pegawai) {
+ await prisma.pegawaiBumDes.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ namaLengkap: p.namaLengkap,
+ gelarAkademik: p.gelarAkademik,
+ tanggalMasuk: new Date(p.tanggalMasuk),
+ email: p.email,
+ telepon: p.telepon,
+ alamat: p.alamat,
+ posisiId: p.posisiId,
+ isActive: p.isActive,
+ },
+ create: {
+ id: p.id,
+ namaLengkap: p.namaLengkap,
+ gelarAkademik: p.gelarAkademik,
+ tanggalMasuk: new Date(p.tanggalMasuk),
+ email: p.email,
+ telepon: p.telepon,
+ alamat: p.alamat,
+ posisiId: p.posisiId,
+ isActive: p.isActive,
+ },
+ });
+ }
+ console.log("pegawai success ...");
+}
diff --git a/prisma/_seeder_list/inovasi/seed_ajukan.ts b/prisma/_seeder_list/inovasi/seed_ajukan.ts
new file mode 100644
index 00000000..ee690fc0
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_ajukan.ts
@@ -0,0 +1,31 @@
+import prisma from "@/lib/prisma";
+import ajukanIde from "../../data/inovasi/ajukan-ide/ajukan-ide.json";
+
+export async function seedAjukan() {
+ console.log("🔄 Seeding Ajukan Ide Inovatif...");
+ for (const d of ajukanIde) {
+ await prisma.ajukanIdeInovatif.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ alamat: d.alamat,
+ namaIde: d.namaIde,
+ deskripsi: d.deskripsi,
+ masalah: d.masalah,
+ benefit: d.benefit,
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ alamat: d.alamat,
+ namaIde: d.namaIde,
+ deskripsi: d.deskripsi,
+ masalah: d.masalah,
+ benefit: d.benefit,
+ },
+ });
+ }
+ console.log("✅ Ajukan Ide Inovatif seeded successfully");
+}
diff --git a/prisma/_seeder_list/inovasi/seed_desa_digital.ts b/prisma/_seeder_list/inovasi/seed_desa_digital.ts
new file mode 100644
index 00000000..0c0b2209
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_desa_digital.ts
@@ -0,0 +1,42 @@
+import prisma from "@/lib/prisma";
+import desaDigital from "../../data/inovasi/desa-digital/desa-digital.json";
+
+export async function seedDesaDigital() {
+ console.log("🔄 Seeding Desa Digital...");
+ for (const d of desaDigital) {
+ let imageId: string | null = null;
+
+ if (d.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: d.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for desa digital "${d.name}": ${d.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.desaDigital.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ deskripsi: d.deskripsi,
+ imageId: imageId,
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ deskripsi: d.deskripsi,
+ imageId: imageId,
+ },
+ });
+ }
+ console.log("✅ Desa Digital seeded successfully");
+}
diff --git a/prisma/_seeder_list/inovasi/seed_info_teknologi.ts b/prisma/_seeder_list/inovasi/seed_info_teknologi.ts
new file mode 100644
index 00000000..768cde5b
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_info_teknologi.ts
@@ -0,0 +1,42 @@
+import prisma from "@/lib/prisma";
+import infoTeknologi from "../../data/inovasi/info-teknologi/info-teknologi.json";
+
+export async function seedInfoTeknologi() {
+ console.log("🔄 Seeding Info Teknologi...");
+ for (const p of infoTeknologi) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for berita "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.infoTekno.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId: imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId: imageId,
+ },
+ });
+ }
+ console.log("✅ Info Teknologi seeded successfully");
+}
diff --git a/prisma/_seeder_list/inovasi/seed_kolaborasi_inovasi.ts b/prisma/_seeder_list/inovasi/seed_kolaborasi_inovasi.ts
new file mode 100644
index 00000000..7a725478
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_kolaborasi_inovasi.ts
@@ -0,0 +1,66 @@
+import prisma from "@/lib/prisma";
+import kolaborasiInovasi from "../../data/inovasi/kolaborasi-inovasi/kolaborasi-inovasi.json";
+import mitraKolaborasi from "../../data/inovasi/kolaborasi-inovasi/mitra-kolaborasi.json";
+
+export async function seedKolaborasiInovasi() {
+ console.log("🔄 Seeding Kolaborasi Inovasi...");
+ for (const p of kolaborasiInovasi) {
+ await prisma.kolaborasiInovasi.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ tahun: p.tahun,
+ slug: p.slug,
+ deskripsi: p.deskripsi,
+ kolaborator: p.kolaborator,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ tahun: p.tahun,
+ slug: p.slug,
+ deskripsi: p.deskripsi,
+ kolaborator: p.kolaborator,
+ },
+ });
+ }
+ console.log("✅ Kolaborasi Inovasi seeded successfully");
+
+ console.log("🔄 Seeding Mitra Kolaborasi...");
+ for (const p of mitraKolaborasi) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for mitra kolaborasi "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.mitraKolaborasi.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ imageId: imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ imageId: imageId,
+ },
+ });
+ }
+ console.log("✅ Mitra Kolaborasi seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/inovasi/seed_layanan_online_desa.ts b/prisma/_seeder_list/inovasi/seed_layanan_online_desa.ts
new file mode 100644
index 00000000..d233ca00
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_layanan_online_desa.ts
@@ -0,0 +1,113 @@
+import prisma from "@/lib/prisma";
+import jenisLayanan from "../../data/inovasi/layanan-online-desa/jenis-layanan.json";
+import administrasiOnline from "../../data/inovasi/layanan-online-desa/administrasi-online.json";
+import jenisPengaduan from "../../data/inovasi/layanan-online-desa/jenis-pengaduan.json";
+import pengaduanMasyarakat from "../../data/inovasi/layanan-online-desa/pengaduan-masyarakat.json";
+
+export async function seedLayananOnlineDesa() {
+ console.log("🔄 Seeding Jenis Layanan...");
+ for (const j of jenisLayanan) {
+ await prisma.jenisLayanan.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ nama: j.nama,
+ deskripsi: j.deskripsi,
+ },
+ create: {
+ id: j.id,
+ nama: j.nama,
+ deskripsi: j.deskripsi,
+ },
+ });
+ }
+ console.log("✅ Jenis Layanan seeded successfully");
+ console.log("🔄 Seeding Administrasi Online...");
+ for (const d of administrasiOnline) {
+ await prisma.administrasiOnline.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ alamat: d.alamat,
+ nomorTelepon: d.nomorTelepon,
+ jenisLayananId: d.jenisLayananId,
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ alamat: d.alamat,
+ nomorTelepon: d.nomorTelepon,
+ jenisLayananId: d.jenisLayananId,
+ },
+ });
+ }
+ console.log("✅ Administrasi Online seeded successfully");
+ console.log("🔄 Seeding Jenis Pengaduan Masyarakat...");
+ for (const d of jenisPengaduan) {
+ await prisma.jenisPengaduan.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ nama: d.nama,
+ },
+ create: {
+ id: d.id,
+ nama: d.nama,
+ },
+ });
+ }
+ console.log("✅ Jenis Pengaduan Masyarakat seeded successfully");
+ console.log("🔄 Seeding Pengaduan Masyarakat...");
+ for (const d of pengaduanMasyarakat) {
+ let imageId: string | null = null;
+
+ if (d.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: d.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for pengaduan masyarakat "${d.name}": ${d.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.pengaduanMasyarakat.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ name: d.name,
+ email: d.email,
+ nik: d.nik,
+ nomorTelepon: d.nomorTelepon,
+ judulPengaduan: d.judulPengaduan,
+ lokasiKejadian: d.lokasiKejadian,
+ imageId: imageId,
+ deskripsiPengaduan: d.deskripsiPengaduan,
+ jenisPengaduanId: d.jenisPengaduanId,
+ },
+ create: {
+ id: d.id,
+ name: d.name,
+ email: d.email,
+ nik: d.nik,
+ nomorTelepon: d.nomorTelepon,
+ judulPengaduan: d.judulPengaduan,
+ lokasiKejadian: d.lokasiKejadian,
+ imageId: imageId,
+ deskripsiPengaduan: d.deskripsiPengaduan,
+ jenisPengaduanId: d.jenisPengaduanId,
+ },
+ });
+ }
+ console.log("✅ Pengaduan Masyarakat seeded successfully");
+}
diff --git a/prisma/_seeder_list/inovasi/seed_program_kreatif_desa.ts b/prisma/_seeder_list/inovasi/seed_program_kreatif_desa.ts
new file mode 100644
index 00000000..e98584b2
--- /dev/null
+++ b/prisma/_seeder_list/inovasi/seed_program_kreatif_desa.ts
@@ -0,0 +1,27 @@
+import prisma from "@/lib/prisma";
+import programKreatif from "../../data/inovasi/program-kreatif-desa/program-kreatif-desa.json";
+
+export async function seedProgramKreatifDesa() {
+ console.log("🔄 Seeding Program Kreatif...");
+ for (const p of programKreatif) {
+ await prisma.programKreatif.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ slug: p.slug,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ slug: p.slug,
+ },
+ });
+ }
+ console.log("✅ Program Kreatif seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/keamanan/seed_keamanan_lingkungan.ts b/prisma/_seeder_list/keamanan/seed_keamanan_lingkungan.ts
new file mode 100644
index 00000000..018dcb19
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_keamanan_lingkungan.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import keamananLingkunganJson from "../../data/keamanan/keamanan-lingkungan/keamanan-lingkungan.json";
+
+export async function seedKeamananLingkungan() {
+console.log("🔄 Seeding Keamanan Lingkungan...");
+
+ for (const p of keamananLingkunganJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for keamanan lingkungan "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.keamananLingkungan.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Keamanan lingkungan seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Keamanan lingkungan seed selesai");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/keamanan/seed_kontak_darurat.ts b/prisma/_seeder_list/keamanan/seed_kontak_darurat.ts
new file mode 100644
index 00000000..d108c03a
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_kontak_darurat.ts
@@ -0,0 +1,87 @@
+import prisma from "@/lib/prisma";
+import kontakDaruratKeamanan from "../../data/keamanan/kontak-darurat-keamanan/kontak-darurat-keamanan.json";
+import kontakItem from "../../data/keamanan/kontak-darurat-keamanan/kontakItem.json";
+import kontakDaruratToItem from "../../data/keamanan/kontak-darurat-keamanan/kontakDaruratToItem.json";
+
+export async function seedKontakDaruratKeamanan() {
+ console.log("🔄 Seeding Kontak Item...");
+ for (const e of kontakItem) {
+ await prisma.kontakItem.upsert({
+ where: {
+ id: e.id,
+ },
+ update: {
+ nama: e.nama,
+ icon: e.icon,
+ nomorTelepon: e.nomorTelepon,
+ },
+ create: {
+ id: e.id, // ✅ WAJIB
+ nama: e.nama,
+ icon: e.icon,
+ nomorTelepon: e.nomorTelepon,
+ },
+ });
+ }
+ console.log("✅ Kontak Item seeded successfully");
+
+ console.log("🔄 Seeding Kontak Darurat Keamanan...");
+ for (const d of kontakDaruratKeamanan) {
+ await prisma.kontakDaruratKeamanan.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ nama: d.nama,
+ icon: d.icon,
+ kategoriId: d.kategoriId,
+ },
+ create: {
+ id: d.id,
+ nama: d.nama,
+ icon: d.icon,
+ kategoriId: d.kategoriId,
+ },
+ });
+ }
+
+ console.log("✅ Kontak Darurat Keamanan seeded successfully");
+
+ console.log("🔄 Seeding Kontak Darurat To Item...");
+ for (const f of kontakDaruratToItem) {
+ // ✅ Validasi foreign keys
+ const kontakDaruratExists = await prisma.kontakDaruratKeamanan.findUnique({
+ where: { id: f.kontakDaruratId },
+ });
+
+ const kontakItemExists = await prisma.kontakItem.findUnique({
+ where: { id: f.kontakItemId },
+ });
+
+ if (!kontakDaruratExists) {
+ console.warn(
+ `⚠️ KontakDarurat ${f.kontakDaruratId} not found, skipping...`,
+ );
+ continue;
+ }
+
+ if (!kontakItemExists) {
+ console.warn(`⚠️ KontakItem ${f.kontakItemId} not found, skipping...`);
+ continue;
+ }
+
+ await prisma.kontakDaruratToItem.upsert({
+ where: { id: f.id },
+ update: {
+ kontakDaruratId: f.kontakDaruratId,
+ kontakItemId: f.kontakItemId,
+ },
+ create: {
+ id: f.id,
+ kontakDaruratId: f.kontakDaruratId,
+ kontakItemId: f.kontakItemId,
+ },
+ });
+ }
+ console.log("✅ Kontak Darurat To Item seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/keamanan/seed_laporan_publik.ts b/prisma/_seeder_list/keamanan/seed_laporan_publik.ts
new file mode 100644
index 00000000..a5056bb6
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_laporan_publik.ts
@@ -0,0 +1,49 @@
+import prisma from "@/lib/prisma";
+import laporanPublik from "../../data/keamanan/laporan-publik/laporan-publik.json";
+import penangananLaporan from "../../data/keamanan/laporan-publik/penanganan-laporan.json";
+
+export async function seedLaporanPublik() {
+ console.log("🔄 Seeding Laporan Publik...");
+ for (const l of laporanPublik) {
+ await prisma.laporanPublik.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ judul: l.judul,
+ lokasi: l.lokasi,
+ tanggalWaktu: l.tanggalWaktu,
+ kronologi: l.kronologi,
+ },
+ create: {
+ id: l.id,
+ judul: l.judul,
+ lokasi: l.lokasi,
+ tanggalWaktu: l.tanggalWaktu,
+ kronologi: l.kronologi,
+ },
+ });
+ }
+
+ console.log("laporan publik success ...");
+
+ console.log("🔄 Seeding Penanganan Laporan...");
+ for (const l of penangananLaporan) {
+ await prisma.penangananLaporanPublik.upsert({
+ where: {
+ id: l.id,
+ },
+ update: {
+ deskripsi: l.deskripsi,
+ laporanId: l.laporanId,
+ },
+ create: {
+ id: l.id,
+ deskripsi: l.deskripsi,
+ laporanId: l.laporanId,
+ },
+ });
+ }
+
+ console.log("penanganan laporan success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/keamanan/seed_pencegahan_kriminalitas.ts b/prisma/_seeder_list/keamanan/seed_pencegahan_kriminalitas.ts
new file mode 100644
index 00000000..1a11b9a7
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_pencegahan_kriminalitas.ts
@@ -0,0 +1,28 @@
+import prisma from "@/lib/prisma";
+import pencegahanKriminalitas from "../../data/keamanan/pencegahan-kriminalitas/pencegahan-kriminalitas.json";
+
+export async function seedPencegahanKriminalitas() {
+ console.log("🔄 Seeding Pencegahan Kriminalitas...");
+ for (const d of pencegahanKriminalitas) {
+ await prisma.pencegahanKriminalitas.upsert({
+ where: {
+ id: d.id,
+ },
+ update: {
+ judul: d.judul,
+ deskripsi: d.deskripsi,
+ deskripsiSingkat: d.deskripsiSingkat,
+ linkVideo: d.linkVideo,
+ },
+ create: {
+ id: d.id,
+ judul: d.judul,
+ deskripsi: d.deskripsi,
+ deskripsiSingkat: d.deskripsiSingkat,
+ linkVideo: d.linkVideo,
+ },
+ });
+ }
+
+ console.log("✅ Pencegahan Kriminalitas seeded successfully");
+}
diff --git a/prisma/_seeder_list/keamanan/seed_polsek_terdekat.ts b/prisma/_seeder_list/keamanan/seed_polsek_terdekat.ts
new file mode 100644
index 00000000..1e8ff8de
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_polsek_terdekat.ts
@@ -0,0 +1,80 @@
+import prisma from "@/lib/prisma";
+import layananPolsek from "../../data/keamanan/polsek-terdekat/layanan-polsek.json";
+import polsekTerdekat from "../../data/keamanan/polsek-terdekat/polsek-terdekat.json";
+import layananToPolsek from "../../data/keamanan/polsek-terdekat/layanan-to-polsek.json";
+
+export async function seedPolsekTerdekat() {
+ console.log("🔄 Seeding Layanan Polsek...");
+ for (const k of layananPolsek) {
+ await prisma.layananPolsek.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ nama: k.nama,
+ },
+ create: {
+ id: k.id,
+ nama: k.nama,
+ },
+ });
+ }
+
+ console.log("layanan polsek success ...");
+
+ console.log("🔄 Seeding Polsek Terdekat...");
+ for (const k of polsekTerdekat) {
+ await prisma.polsekTerdekat.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ nama: k.nama,
+ jarakKeDesa: k.jarakKeDesa,
+ alamat: k.alamat,
+ nomorTelepon: k.nomorTelepon,
+ jamOperasional: k.jamOperasional,
+ embedMapUrl: k.embedMapUrl,
+ namaTempatMaps: k.namaTempatMaps,
+ alamatMaps: k.alamatMaps,
+ linkPetunjukArah: k.linkPetunjukArah,
+ layananPolsekId: k.layananPolsekId,
+ },
+ create: {
+ id: k.id,
+ nama: k.nama,
+ jarakKeDesa: k.jarakKeDesa,
+ alamat: k.alamat,
+ nomorTelepon: k.nomorTelepon,
+ jamOperasional: k.jamOperasional,
+ embedMapUrl: k.embedMapUrl,
+ namaTempatMaps: k.namaTempatMaps,
+ alamatMaps: k.alamatMaps,
+ linkPetunjukArah: k.linkPetunjukArah,
+ layananPolsekId: k.layananPolsekId,
+ },
+ });
+ }
+
+ console.log("polsek terdekat success ...");
+
+ console.log("🔄 Seeding Layanan To Polsek...");
+ for (const k of layananToPolsek) {
+ await prisma.layananToPolsek.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ layananId: k.layananId,
+ polsekTerdekatId: k.polsekTerdekatId,
+ },
+ create: {
+ id: k.id,
+ layananId: k.layananId,
+ polsekTerdekatId: k.polsekTerdekatId,
+ },
+ });
+ }
+
+ console.log("layanan to polsek success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/keamanan/seed_tips_keamanan.ts b/prisma/_seeder_list/keamanan/seed_tips_keamanan.ts
new file mode 100644
index 00000000..5b5a3abc
--- /dev/null
+++ b/prisma/_seeder_list/keamanan/seed_tips_keamanan.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import tipsKeamananJson from "../../data/keamanan/tips-keamanan/tips-keamanan.json";
+
+export async function seedTipsKeamanan() {
+ console.log("🔄 Seeding Tips Keamanan...");
+
+ for (const p of tipsKeamananJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for tips keamanan "${p.judul}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.menuTipsKeamanan.upsert({
+ where: { id: p.id },
+ update: {
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Tips Keamanan seeded: ${p.judul}`);
+ }
+
+ console.log("🎉 Tips Keamanan seed selesai");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/kesehatan/info-wabah-penyakit/seed_info_wabah_penyakit.ts b/prisma/_seeder_list/kesehatan/info-wabah-penyakit/seed_info_wabah_penyakit.ts
new file mode 100644
index 00000000..922bb848
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/info-wabah-penyakit/seed_info_wabah_penyakit.ts
@@ -0,0 +1,46 @@
+import prisma from "@/lib/prisma";
+import infoWabahPenyakitJson from "../../../data/kesehatan/infowabahpenyakit/infowabahpenyakit.json";
+
+export async function seedInfoWabahPenyakit() {
+console.log("🔄 Seeding Info Wabah Penyakit...");
+
+ for (const p of infoWabahPenyakitJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for info wabah penyakit "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.infoWabahPenyakit.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsiSingkat: p.deskripsiSingkat,
+ deskripsiLengkap: p.deskripsiLengkap,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsiSingkat: p.deskripsiSingkat,
+ deskripsiLengkap: p.deskripsiLengkap,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Info wabah penyakit seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Info wabah penyakit seed selesai");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/kesehatan/kontak-darurat/seed_kontak_darurat.ts b/prisma/_seeder_list/kesehatan/kontak-darurat/seed_kontak_darurat.ts
new file mode 100644
index 00000000..d42aec3a
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/kontak-darurat/seed_kontak_darurat.ts
@@ -0,0 +1,46 @@
+import kontakDaruratJson from "../../../data/kesehatan/kontak-darurat/kontak-darurat.json";
+import prisma from "@/lib/prisma";
+
+export async function seedKontakDarurat() {
+ console.log("🔄 Seeding Kontak Darurat...");
+
+ for (const p of kontakDaruratJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for kontak darurat "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.kontakDarurat.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ whatsapp: p.whatsapp,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ whatsapp: p.whatsapp,
+ },
+ });
+
+ console.log(`✅ Kontak darurat seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Kontak darurat seed selesai");
+}
diff --git a/prisma/_seeder_list/kesehatan/penanganan-darurat/seed_penanganan_darurat.ts b/prisma/_seeder_list/kesehatan/penanganan-darurat/seed_penanganan_darurat.ts
new file mode 100644
index 00000000..788b1dc7
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/penanganan-darurat/seed_penanganan_darurat.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import penangananDaruratJson from "../../../data/kesehatan/penanganan-darurat/penganan-darurat.json";
+
+export async function seedPenangananDarurat() {
+ console.log("🔄 Seeding Penanganan Darurat...");
+
+ for (const p of penangananDaruratJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for penanganan darurat "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.penangananDarurat.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Penanganan darurat seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Penanganan darurat seed selesai");
+}
diff --git a/prisma/_seeder_list/kesehatan/posyandu/seed_posyandu.ts b/prisma/_seeder_list/kesehatan/posyandu/seed_posyandu.ts
new file mode 100644
index 00000000..fce475ca
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/posyandu/seed_posyandu.ts
@@ -0,0 +1,48 @@
+import prisma from "@/lib/prisma";
+import posyanduJson from "../../../data/kesehatan/posyandu/posyandu.json";
+
+export async function seedPosyandu() {
+ console.log("🔄 Seeding Posyandu...");
+
+ for (const p of posyanduJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for posyandu "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.posyandu.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ nomor: p.nomor,
+ deskripsi: p.deskripsi,
+ jadwalPelayanan: p.jadwalPelayanan,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ nomor: p.nomor,
+ deskripsi: p.deskripsi,
+ jadwalPelayanan: p.jadwalPelayanan,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Posyandu seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Posyandu seed selesai");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/kesehatan/program-kesehatan/seed_program_kesehatan.ts b/prisma/_seeder_list/kesehatan/program-kesehatan/seed_program_kesehatan.ts
new file mode 100644
index 00000000..4c898e44
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/program-kesehatan/seed_program_kesehatan.ts
@@ -0,0 +1,42 @@
+import prisma from "@/lib/prisma";
+import programKesehatanJson from "../../../data/kesehatan/program-kesehatan/program-kesehatan.json";
+
+export async function seedProgramKesehatan() {
+ for (const p of programKesehatanJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for program kesehatan "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.programKesehatan.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsiSingkat: p.deskripsiSingkat,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsiSingkat: p.deskripsiSingkat,
+ deskripsi: p.deskripsi,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Program kesehatan seeded: ${p.name}`);
+ }
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/kesehatan/puskesmas/seed_puskesmas.ts b/prisma/_seeder_list/kesehatan/puskesmas/seed_puskesmas.ts
new file mode 100644
index 00000000..2c9a632b
--- /dev/null
+++ b/prisma/_seeder_list/kesehatan/puskesmas/seed_puskesmas.ts
@@ -0,0 +1,95 @@
+import prisma from "@/lib/prisma";
+import puskesmasJson from "../../../data/kesehatan/puskesmas/puskesmas.json";
+import kontakPuskesmasJson from "../../../data/kesehatan/puskesmas/kontak-puskesmas/kontak.json";
+import jamPuskesmasJson from "../../../data/kesehatan/puskesmas/jam-puskesmas/jam.json";
+
+export async function seedPuskesmas() {
+ console.log("🔄 Seeding Kontak Puskesmas...");
+ for (const k of kontakPuskesmasJson) {
+ await prisma.kontakPuskesmas.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ kontakPuskesmas: k.kontakPuskesmas,
+ email: k.email,
+ facebook: k.facebook,
+ kontakUGD: k.kontakUGD,
+ },
+ create: {
+ id: k.id,
+ kontakPuskesmas: k.kontakPuskesmas,
+ email: k.email,
+ facebook: k.facebook,
+ kontakUGD: k.kontakUGD,
+ },
+ });
+ }
+ console.log("kontak puskesmas success ...");
+
+ console.log("🔄 Seeding Jam Puskesmas...");
+ for (const k of jamPuskesmasJson) {
+ await prisma.jamOperasional.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ workDays: k.workDays,
+ weekDays: k.weekDays,
+ holiday: k.holiday,
+ },
+ create: {
+ id: k.id,
+ workDays: k.workDays,
+ weekDays: k.weekDays,
+ holiday: k.holiday,
+ },
+ });
+ }
+ console.log("jam puskesmas success ...");
+
+ console.log("🔄 Seeding Puskesmas...");
+
+ for (const p of puskesmasJson) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for puskesmas "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.puskesmas.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ alamat: p.alamat,
+ jamId: p.jamId,
+ kontakId: p.kontakId,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ alamat: p.alamat,
+ jamId: p.jamId,
+ kontakId: p.kontakId,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Puskesmas seeded: ${p.name}`);
+ }
+
+ console.log("🎉 Puskesmas seed selesai");
+}
+
diff --git a/prisma/_seeder_list/landing-page/desa-anti-korupsi/seed_desa_anti_korupsi.ts b/prisma/_seeder_list/landing-page/desa-anti-korupsi/seed_desa_anti_korupsi.ts
new file mode 100644
index 00000000..5a1927fa
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/desa-anti-korupsi/seed_desa_anti_korupsi.ts
@@ -0,0 +1,38 @@
+import prisma from "@/lib/prisma";
+import kategoriDesaAntiKorupsi from "../../../data/landing-page/desa-anti-korupsi/kategoriDesaAntiKorupsi.json"
+import desaAntiKorupsi from "../../../data/landing-page/desa-anti-korupsi/desaantiKorpusi.json"
+
+export async function seedDesaAntiKorupsi() {
+ for (const k of kategoriDesaAntiKorupsi) {
+ await prisma.kategoriDesaAntiKorupsi.upsert({
+ where: { id: k.id },
+ update: {
+ name: k.name,
+ },
+ create: {
+ id: k.id,
+ name: k.name,
+ },
+ });
+ }
+ console.log("kategori desa anti korupsi success ...");
+
+ // =========== DESA ANTI KORUPSI ===========
+ for (const p of desaAntiKorupsi) {
+ await prisma.desaAntiKorupsi.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ deskripsi: p.deskripsi,
+ kategoriId: p.kategoriId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ deskripsi: p.deskripsi,
+ kategoriId: p.kategoriId,
+ },
+ });
+ }
+ console.log("desa anti korupsi success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/landing-page/prestasi-desa/seed_prestasi_desa.ts b/prisma/_seeder_list/landing-page/prestasi-desa/seed_prestasi_desa.ts
new file mode 100644
index 00000000..18129dfb
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/prestasi-desa/seed_prestasi_desa.ts
@@ -0,0 +1,61 @@
+import prisma from "@/lib/prisma";
+import prestasiDesa from "../../../data/landing-page/prestasi-desa/prestasi-desa.json"
+import kategoriPrestasiDesa from "../../../data/landing-page/prestasi-desa/kategori-prestasi.json"
+
+export async function seedPrestasiDesa() {
+
+ console.log("🔄 Seeding Kategori Prestasi Desa...");
+for (const c of kategoriPrestasiDesa) {
+ await prisma.kategoriPrestasiDesa.upsert({
+ where: { id: c.id },
+ update: {
+ name: c.name,
+ },
+ create: {
+ id: c.id,
+ name: c.name,
+ },
+ });
+ }
+ console.log("kategori prestasi desa success ...");
+
+ console.log("🔄 Seeding Prestasi Desa...");
+ for (const m of prestasiDesa) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for prestasi desa "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.prestasiDesa.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ deskripsi: m.deskripsi,
+ kategoriId: m.kategoriId,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ deskripsi: m.deskripsi,
+ kategoriId: m.kategoriId,
+ imageId,
+ },
+ });
+ }
+
+ console.log("prestasi desa success ...");
+}
+
\ No newline at end of file
diff --git a/prisma/_seeder_list/landing-page/profil_landing_page/seed_media_sosial.ts b/prisma/_seeder_list/landing-page/profil_landing_page/seed_media_sosial.ts
new file mode 100644
index 00000000..7636202a
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/profil_landing_page/seed_media_sosial.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import mediaSosial from "../../../data/landing-page/profile/mediaSosial.json"
+
+export async function seedMediaSosial() {
+ console.log("🔄 Seeding Media Sosial...");
+ for (const m of mediaSosial) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for berita "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.mediaSosial.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ iconUrl: m.iconUrl,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ iconUrl: m.iconUrl,
+ imageId,
+ },
+ });
+ }
+
+ console.log("media sosial success ...");
+}
+
+
+
\ No newline at end of file
diff --git a/prisma/_seeder_list/landing-page/profil_landing_page/seed_profile_lp.ts b/prisma/_seeder_list/landing-page/profil_landing_page/seed_profile_lp.ts
new file mode 100644
index 00000000..a2f5f7a6
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/profil_landing_page/seed_profile_lp.ts
@@ -0,0 +1,40 @@
+import prisma from "@/lib/prisma";
+import profilePejabatDesa from "../../../data/landing-page/profile/profile.json";
+
+export async function seedProfileLP() {
+ console.log("🔄 Seeding Pejabat Desa...");
+ for (const p of profilePejabatDesa) {
+ let imageId: string | null = null;
+
+ if (p.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: p.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for profile "${p.name}": ${p.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.pejabatDesa.upsert({
+ where: { id: p.id },
+ update: {
+ name: p.name,
+ position: p.position,
+ imageId,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ position: p.position,
+ imageId,
+ },
+ });
+ }
+ console.log("✅ Pejabat Desa seeding completed");
+}
diff --git a/prisma/_seeder_list/landing-page/profil_landing_page/seed_program_inovasi.ts b/prisma/_seeder_list/landing-page/profil_landing_page/seed_program_inovasi.ts
new file mode 100644
index 00000000..207045b2
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/profil_landing_page/seed_program_inovasi.ts
@@ -0,0 +1,44 @@
+import prisma from "@/lib/prisma";
+import programInovasi from "../../../data/landing-page/profile/programInovasi.json";
+
+export async function seedProgramInovasi() {
+ console.log("🔄 Seeding Program Inovasi...");
+
+ for (const b of programInovasi) {
+ let imageId: string | null = null;
+
+ if (b.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: b.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for program inovasi "${b.name}": ${b.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.programInovasi.upsert({
+ where: { id: b.id },
+ update: {
+ name: b.name,
+ description: b.description,
+ link: b.link,
+ imageId,
+ },
+ create: {
+ id: b.id,
+ name: b.name,
+ description: b.description,
+ link: b.link,
+ imageId,
+ },
+ });
+
+ console.log(`✅ Program Inovasi seeded: ${b.name}`);
+ }
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/landing-page/sdgs/seed_sdgs.ts b/prisma/_seeder_list/landing-page/sdgs/seed_sdgs.ts
new file mode 100644
index 00000000..a8aac2a1
--- /dev/null
+++ b/prisma/_seeder_list/landing-page/sdgs/seed_sdgs.ts
@@ -0,0 +1,41 @@
+import prisma from "@/lib/prisma";
+import sdgsDesa from "../../../data/landing-page/sdgs-desa/sdgs-desa.json";
+
+export async function seedSDGSDesa() {
+ console.log("🔄 Seeding SDGS Desa...");
+ for (const m of sdgsDesa) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for sdgs desa "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.sdgsDesa.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ jumlah: m.jumlah,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ jumlah: m.jumlah,
+ imageId,
+ },
+ });
+ }
+
+ console.log("sdgs desa success ...");
+}
diff --git a/prisma/_seeder_list/lingkungan/seed_data_gotong_royong.ts b/prisma/_seeder_list/lingkungan/seed_data_gotong_royong.ts
new file mode 100644
index 00000000..c317dad6
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_data_gotong_royong.ts
@@ -0,0 +1,71 @@
+import prisma from "@/lib/prisma";
+import kategoriGotongRoyong from "../../data/lingkungan/gotong-royong/kategori-gotong-royong.json";
+import gotongRoyong from "../../data/lingkungan/gotong-royong/gotong-royong.json";
+
+export async function seedDataGotongRoyong() {
+ console.log("🔄 Seeding Kategori Gotong Royong...");
+
+ for (const k of kategoriGotongRoyong) {
+ await prisma.kategoriKegiatan.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ nama: k.nama,
+ },
+ create: {
+ id: k.id,
+ nama: k.nama,
+ },
+ });
+ }
+ console.log("✅ Kategori Gotong Royong seeded successfully");
+
+ console.log("🔄 Seeding Gotong Royong...");
+ for (const k of gotongRoyong) {
+ let imageId: string | null = null;
+
+ if (k.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: k.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for gotong royong "${k.judul}": ${k.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.kegiatanDesa.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ judul: k.judul,
+ deskripsiSingkat: k.deskripsiSingkat,
+ deskripsiLengkap: k.deskripsiLengkap,
+ tanggal: k.tanggal,
+ lokasi: k.lokasi,
+ partisipan: k.partisipan,
+ imageId: imageId,
+ kategoriKegiatanId: k.kategoriKegiatanId,
+ },
+ create: {
+ id: k.id,
+ judul: k.judul,
+ deskripsiSingkat: k.deskripsiSingkat,
+ deskripsiLengkap: k.deskripsiLengkap,
+ tanggal: k.tanggal,
+ lokasi: k.lokasi,
+ partisipan: k.partisipan,
+ imageId: imageId,
+ kategoriKegiatanId: k.kategoriKegiatanId,
+ },
+ });
+ }
+ console.log("✅ Gotong Royong seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/lingkungan/seed_data_lingkungan_desa.ts b/prisma/_seeder_list/lingkungan/seed_data_lingkungan_desa.ts
new file mode 100644
index 00000000..a182bf86
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_data_lingkungan_desa.ts
@@ -0,0 +1,27 @@
+import prisma from "@/lib/prisma";
+import dataLingkunganDesa from "../../data/lingkungan/data-lingkungan-desa/data-lingkungan-desa.json";
+
+export async function seedDataLingkunganDesa() {
+ console.log("🔄 Seeding Data Lingkungan Desa...");
+ for (const p of dataLingkunganDesa) {
+ await prisma.dataLingkunganDesa.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ jumlah: p.jumlah,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ jumlah: p.jumlah,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ },
+ });
+ }
+ console.log("✅ Data Lingkungan Desa seeded successfully");
+}
diff --git a/prisma/_seeder_list/lingkungan/seed_edukasi_lingkungan.ts b/prisma/_seeder_list/lingkungan/seed_edukasi_lingkungan.ts
new file mode 100644
index 00000000..5d11b061
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_edukasi_lingkungan.ts
@@ -0,0 +1,63 @@
+import prisma from "@/lib/prisma";
+import tujuanEdukasiLingkungan from "../../data/lingkungan/edukasi-lingkungan/tujuan-edukasi-lingkungan.json";
+import materiEdukasiLingkungan from "../../data/lingkungan/edukasi-lingkungan/materi-edukasi-yang-diberikan.json";
+import contohEdukasiLingkungan from "../../data/lingkungan/edukasi-lingkungan/contoh-kegiatan-di-desa-darmasaba.json";
+
+export async function seedEdukasiLingkungan() {
+ for (const e of tujuanEdukasiLingkungan) {
+ await prisma.tujuanEdukasiLingkungan.upsert({
+ where: {
+ id: e.id,
+ },
+ update: {
+ judul: e.judul,
+ deskripsi: e.deskripsi,
+ },
+ create: {
+ id: e.id,
+ judul: e.judul,
+ deskripsi: e.deskripsi,
+ },
+ });
+ }
+
+ console.log("tujuan edukasi lingkungan success ...");
+
+ for (const m of materiEdukasiLingkungan) {
+ await prisma.materiEdukasiLingkungan.upsert({
+ where: {
+ id: m.id,
+ },
+ update: {
+ judul: m.judul,
+ deskripsi: m.deskripsi,
+ },
+ create: {
+ id: m.id,
+ judul: m.judul,
+ deskripsi: m.deskripsi,
+ },
+ });
+ }
+
+ console.log("materi edukasi lingkungan success ...");
+
+ for (const c of contohEdukasiLingkungan) {
+ await prisma.contohEdukasiLingkungan.upsert({
+ where: {
+ id: c.id,
+ },
+ update: {
+ judul: c.judul,
+ deskripsi: c.deskripsi,
+ },
+ create: {
+ id: c.id,
+ judul: c.judul,
+ deskripsi: c.deskripsi,
+ },
+ });
+ }
+
+ console.log("contoh edukasi lingkungan success ...");
+}
diff --git a/prisma/_seeder_list/lingkungan/seed_konservasi_adat_bali.ts b/prisma/_seeder_list/lingkungan/seed_konservasi_adat_bali.ts
new file mode 100644
index 00000000..5278fbc5
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_konservasi_adat_bali.ts
@@ -0,0 +1,63 @@
+import prisma from "@/lib/prisma";
+import filosofiTriHita from "../../data/lingkungan/konservasi-adat-bali/filosofi-tri-hita.json";
+import bentukKonservasiBerdasarkanAdat from "../../data/lingkungan/konservasi-adat-bali/bentuk-konservasi.json";
+import nilaiKonservasiAdat from "../../data/lingkungan/konservasi-adat-bali/nilai-konservasi-adat.json";
+
+export async function seedKonservasiAdatBali() {
+ for (const f of filosofiTriHita) {
+ await prisma.filosofiTriHita.upsert({
+ where: {
+ id: f.id,
+ },
+ update: {
+ judul: f.judul,
+ deskripsi: f.deskripsi,
+ },
+ create: {
+ id: f.id,
+ judul: f.judul,
+ deskripsi: f.deskripsi,
+ },
+ });
+ }
+
+ console.log("filosofi tri hita success ...");
+
+ for (const b of bentukKonservasiBerdasarkanAdat) {
+ await prisma.bentukKonservasiBerdasarkanAdat.upsert({
+ where: {
+ id: b.id,
+ },
+ update: {
+ judul: b.judul,
+ deskripsi: b.deskripsi,
+ },
+ create: {
+ id: b.id,
+ judul: b.judul,
+ deskripsi: b.deskripsi,
+ },
+ });
+ }
+
+ console.log("bentuk konservasi berdasarkan adat success ...");
+
+ for (const n of nilaiKonservasiAdat) {
+ await prisma.nilaiKonservasiAdat.upsert({
+ where: {
+ id: n.id,
+ },
+ update: {
+ judul: n.judul,
+ deskripsi: n.deskripsi,
+ },
+ create: {
+ id: n.id,
+ judul: n.judul,
+ deskripsi: n.deskripsi,
+ },
+ });
+ }
+
+ console.log("nilai konservasi adat success ...");
+}
diff --git a/prisma/_seeder_list/lingkungan/seed_pengelolaan_sampah.ts b/prisma/_seeder_list/lingkungan/seed_pengelolaan_sampah.ts
new file mode 100644
index 00000000..5940c79b
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_pengelolaan_sampah.ts
@@ -0,0 +1,51 @@
+import prisma from "@/lib/prisma";
+import pengelolaanSampah from "../../data/lingkungan/pengelolaan-sampah/pengelolaan-sampah.json";
+import keteranganBankSampah from "../../data/lingkungan/pengelolaan-sampah/keterangan-bank-sampah.json";
+
+export async function seedPengelolaanSampah() {
+ console.log("🔄 Seeding Pengelolaan Sampah...");
+ for (const p of pengelolaanSampah) {
+ await prisma.pengelolaanSampah.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ icon: p.icon,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ icon: p.icon,
+ },
+ });
+ }
+ console.log("✅ Pengelolaan Sampah seeded successfully");
+
+ console.log("🔄 Seeding Keterangan Bank Sampah...");
+ for (const p of keteranganBankSampah) {
+ await prisma.keteranganBankSampahTerdekat.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ alamat: p.alamat,
+ namaTempatMaps: p.namaTempatMaps,
+ linkPetunjukArah: p.linkPetunjukArah,
+ lat: p.lat,
+ lng: p.lng,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ alamat: p.alamat,
+ namaTempatMaps: p.namaTempatMaps,
+ linkPetunjukArah: p.linkPetunjukArah,
+ lat: p.lat,
+ lng: p.lng,
+ },
+ });
+ }
+ console.log("✅ Keterangan Bank Sampah seeded successfully");
+}
diff --git a/prisma/_seeder_list/lingkungan/seed_program_penghijauan.ts b/prisma/_seeder_list/lingkungan/seed_program_penghijauan.ts
new file mode 100644
index 00000000..0dbd8bc5
--- /dev/null
+++ b/prisma/_seeder_list/lingkungan/seed_program_penghijauan.ts
@@ -0,0 +1,27 @@
+import prisma from "@/lib/prisma";
+import programPenghijauan from "../../data/lingkungan/program-penghijauan/program-penghijauan.json";
+
+export async function seedProgramPenghijauan() {
+ console.log("🔄 Seeding Program Penghijauan...");
+ for (const p of programPenghijauan) {
+ await prisma.programPenghijauan.upsert({
+ where: {
+ id: p.id,
+ },
+ update: {
+ name: p.name,
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ },
+ create: {
+ id: p.id,
+ name: p.name,
+ judul: p.judul,
+ deskripsi: p.deskripsi,
+ icon: p.icon,
+ },
+ });
+ }
+ console.log("✅ Program Penghijauan seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/pendidikan/seed_bimbingan_belajar.ts b/prisma/_seeder_list/pendidikan/seed_bimbingan_belajar.ts
new file mode 100644
index 00000000..e6e927c0
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_bimbingan_belajar.ts
@@ -0,0 +1,60 @@
+import prisma from "@/lib/prisma";
+import tujuanBimbinganBelajarDesa from "../../data/pendidikan/bimbingan-belajar-desa/tujuan-bimbingan-belajar-desa.json";
+import lokasiJadwalBimbinganBelajarDesa from "../../data/pendidikan/bimbingan-belajar-desa/lokasi-dan-jadwal.json";
+import fasilitasBimbinganBelajarDesa from "../../data/pendidikan/bimbingan-belajar-desa/fasilitas-yang-disediakan.json";
+
+export async function seedBimbinganBelajar() {
+for (const t of tujuanBimbinganBelajarDesa) {
+ await prisma.tujuanBimbinganBelajarDesa.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ tujuan bimbingan belajar desa seeded (editable later via UI)",
+ );
+
+ for (const t of lokasiJadwalBimbinganBelajarDesa) {
+ await prisma.lokasiJadwalBimbinganBelajarDesa.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ lokasi jadwal bimbingan belajar desa seeded (editable later via UI)",
+ );
+
+ for (const t of fasilitasBimbinganBelajarDesa) {
+ await prisma.fasilitasBimbinganBelajarDesa.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)",
+ );
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/pendidikan/seed_data_pendidikan.ts b/prisma/_seeder_list/pendidikan/seed_data_pendidikan.ts
new file mode 100644
index 00000000..632243f3
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_data_pendidikan.ts
@@ -0,0 +1,23 @@
+import prisma from "@/lib/prisma";
+import dataPendidikan from "../../data/pendidikan/data-pendidikan/data-pendidikan.json";
+
+export async function seedDataPendidikan() {
+ console.log("🔄 Seeding Data pendidikan...");
+ for (const k of dataPendidikan) {
+ await prisma.dataPendidikan.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ name: k.name,
+ jumlah: k.jumlah,
+ },
+ create: {
+ id: k.id,
+ name: k.name,
+ jumlah: k.jumlah,
+ },
+ });
+ }
+ console.log("✅ Data pendidikan seeded successfully");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/pendidikan/seed_data_perpustakaan.ts b/prisma/_seeder_list/pendidikan/seed_data_perpustakaan.ts
new file mode 100644
index 00000000..5ee80718
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_data_perpustakaan.ts
@@ -0,0 +1,71 @@
+import prisma from "@/lib/prisma";
+import dataPerpustakaan from "../../data/pendidikan/perpustakaan-digital/perpustakaan-digital.json";
+import kategoriBuku from "../../data/pendidikan/perpustakaan-digital/kategori-buku.json";
+
+export async function seedDataPerpustakaan() {
+ console.log("🔄 Seeding Kategori Buku...");
+ for (const k of kategoriBuku) {
+ await prisma.kategoriBuku.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ name: k.name,
+ },
+ create: {
+ id: k.id,
+ name: k.name,
+ },
+ });
+ }
+ console.log("✅ Kategori Buku seeded successfully");
+
+ console.log("🔄 Seeding Data perpustakaan...");
+ for (const k of dataPerpustakaan) {
+ let imageId: string | null = null;
+
+ if (k.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: k.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for perpustakaan "${k.judul}": ${k.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.dataPerpustakaan.upsert({
+ where: {
+ id: k.id,
+ },
+ update: {
+ judul: k.judul,
+ deskripsi: k.deskripsi,
+ kategoriId: k.kategoriId,
+ imageId: imageId
+ },
+ create: {
+ id: k.id,
+ judul: k.judul,
+ deskripsi: k.deskripsi,
+ kategoriId: k.kategoriId,
+ imageId: imageId
+ },
+ });
+ }
+ console.log("✅ Data perpustakaan seeded successfully");
+}
+if (import.meta.main) {
+ seedDataPerpustakaan()
+ .then(() => {
+ console.log("seed data perpustakaan success");
+ })
+ .catch((err) => {
+ console.log("gagal seed data perpustakaan", JSON.stringify(err));
+ });
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/pendidikan/seed_info_program_pendidikan.ts b/prisma/_seeder_list/pendidikan/seed_info_program_pendidikan.ts
new file mode 100644
index 00000000..cfa7798c
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_info_program_pendidikan.ts
@@ -0,0 +1,36 @@
+import prisma from "@/lib/prisma";
+import tujuanProgram from "../../data/pendidikan/program-pendidikan-anak/tujuan-program.json";
+import programUnggulan from "../../data/pendidikan/program-pendidikan-anak/program-unggulan.json";
+
+export async function seedInfoProgramPendidikan() {
+ for (const t of tujuanProgram) {
+ await prisma.tujuanProgram.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log("✅ tujuan program seeded (editable later via UI)");
+ for (const t of programUnggulan) {
+ await prisma.programUnggulan.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log("✅ program unggulan seeded (editable later via UI)");
+}
diff --git a/prisma/_seeder_list/pendidikan/seed_info_sekolah.ts b/prisma/_seeder_list/pendidikan/seed_info_sekolah.ts
new file mode 100644
index 00000000..39ba1154
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_info_sekolah.ts
@@ -0,0 +1,74 @@
+import prisma from "@/lib/prisma";
+import jenjangPendidikan from "../../data/pendidikan/info-sekolah/jenjang-pendidikan.json";
+import lembagaPendidikan from "../../data/pendidikan/info-sekolah/lembaga.json";
+import siswa from "../../data/pendidikan/info-sekolah/siswa.json";
+import pengajar from "../../data/pendidikan/info-sekolah/pengajar.json";
+
+export async function seedInfoSekolah() {
+ for (const j of jenjangPendidikan) {
+ await prisma.jenjangPendidikan.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ nama: j.nama,
+ },
+ create: {
+ id: j.id,
+ nama: j.nama,
+ },
+ });
+ }
+ console.log("✅ Jenjang Pendidikan seeded successfully");
+ for (const j of lembagaPendidikan) {
+ await prisma.lembaga.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ nama: j.nama,
+ jenjangId: j.jenjangId,
+ },
+ create: {
+ id: j.id,
+ nama: j.nama,
+ jenjangId: j.jenjangId,
+ },
+ });
+ }
+ console.log("✅ Lembaga Pendidikan seeded successfully");
+ for (const j of siswa) {
+ await prisma.siswa.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ nama: j.nama,
+ lembagaId: j.lembagaId,
+ },
+ create: {
+ id: j.id,
+ nama: j.nama,
+ lembagaId: j.lembagaId,
+ },
+ });
+ }
+ console.log("✅ siswa seeded successfully");
+ for (const j of pengajar) {
+ await prisma.pengajar.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ nama: j.nama,
+ lembagaId: j.lembagaId,
+ },
+ create: {
+ id: j.id,
+ nama: j.nama,
+ lembagaId: j.lembagaId,
+ },
+ });
+ }
+ console.log("✅ pengajar seeded successfully");
+}
diff --git a/prisma/_seeder_list/pendidikan/seed_pendidikan_non_formal.ts b/prisma/_seeder_list/pendidikan/seed_pendidikan_non_formal.ts
new file mode 100644
index 00000000..151c8dbc
--- /dev/null
+++ b/prisma/_seeder_list/pendidikan/seed_pendidikan_non_formal.ts
@@ -0,0 +1,60 @@
+import prisma from "@/lib/prisma";
+import tujuanProgram from "../../data/pendidikan/pendidikan-non-formal/tujuan-program2.json";
+import tempatKegiatan from "../../data/pendidikan/pendidikan-non-formal/tempat-kegiatan.json";
+import jenisProgramYangDiselenggarakan from "../../data/pendidikan/pendidikan-non-formal/jenis-program-yang-diselenggarakan.json";
+
+export async function seedPendidikanNonFormal() {
+ for (const t of tujuanProgram) {
+ await prisma.tujuanPendidikanNonFormal.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)",
+ );
+
+ for (const t of tempatKegiatan) {
+ await prisma.tempatKegiatan.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)",
+ );
+
+ for (const t of jenisProgramYangDiselenggarakan) {
+ await prisma.jenisProgramYangDiselenggarakan.upsert({
+ where: { id: t.id },
+ update: {
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ create: {
+ id: t.id,
+ judul: t.judul,
+ deskripsi: t.deskripsi,
+ },
+ });
+ }
+ console.log(
+ "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)",
+ );
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/daftar-informasi-publik-ppid/seed_daftar_informasi_publik_ppid.ts b/prisma/_seeder_list/ppid/daftar-informasi-publik-ppid/seed_daftar_informasi_publik_ppid.ts
new file mode 100644
index 00000000..3bf6b77b
--- /dev/null
+++ b/prisma/_seeder_list/ppid/daftar-informasi-publik-ppid/seed_daftar_informasi_publik_ppid.ts
@@ -0,0 +1,76 @@
+import prisma from "@/lib/prisma";
+import daftarInformasiPublik from "../../../data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json"
+import jenisInformasiDiminta from "../../../data/list-jenisInfromasi.json"
+import caraMemperolehInformasi from "../../../data/list-caraMemperolehInformasi.json"
+import caraMemperolehSalinanInformasi from "../../../data/list-caraMemperolehSalinanInformasi.json"
+
+export async function seedDaftarInformasiPublikPpid() {
+
+ for (const v of daftarInformasiPublik) {
+ // Convert string date to Date object
+ const tanggal = new Date(v.tanggal);
+
+ await prisma.daftarInformasiPublik.upsert({
+ where: {
+ id: v.id,
+ },
+ update: {
+ jenisInformasi: v.jenisInformasi,
+ deskripsi: v.deskripsi,
+ tanggal: tanggal,
+ },
+ create: {
+ id: v.id,
+ jenisInformasi: v.jenisInformasi,
+ deskripsi: v.deskripsi,
+ tanggal: tanggal,
+ },
+ });
+ }
+ console.log("daftar informasi publik PPID success ...");
+
+ for (const j of jenisInformasiDiminta) {
+ await prisma.jenisInformasiDiminta.upsert({
+ where: {
+ name: j.name,
+ },
+ update: {
+ name: j.name,
+ },
+ create: {
+ name: j.name,
+ },
+ });
+ }
+ console.log("jenis informasi diminta success ...");
+
+ for (const c of caraMemperolehInformasi) {
+ await prisma.caraMemperolehInformasi.upsert({
+ where: {
+ name: c.name,
+ },
+ update: {
+ name: c.name,
+ },
+ create: {
+ name: c.name,
+ },
+ });
+ }
+ console.log("cara memperoleh informasi success ...");
+
+ for (const c of caraMemperolehSalinanInformasi) {
+ await prisma.caraMemperolehSalinanInformasi.upsert({
+ where: {
+ name: c.name,
+ },
+ update: {
+ name: c.name,
+ },
+ create: {
+ name: c.name,
+ },
+ });
+ }
+ console.log("cara memperoleh salinan informasi success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/dasar-hukum-ppid/seed_dasar_hukum_ppid.ts b/prisma/_seeder_list/ppid/dasar-hukum-ppid/seed_dasar_hukum_ppid.ts
new file mode 100644
index 00000000..7d14359a
--- /dev/null
+++ b/prisma/_seeder_list/ppid/dasar-hukum-ppid/seed_dasar_hukum_ppid.ts
@@ -0,0 +1,22 @@
+import prisma from "@/lib/prisma";
+import dasarHukumPPID from "../../../data/ppid/dasar-hukum-ppid/dasarhukumPPID.json"
+
+export async function seedDasarHukumPpid() {
+ for (const v of dasarHukumPPID) {
+ await prisma.dasarHukumPPID.upsert({
+ where: {
+ id: v.id,
+ },
+ update: {
+ judul: v.judul,
+ content: v.content,
+ },
+ create: {
+ id: v.id,
+ judul: v.judul,
+ content: v.content,
+ },
+ });
+ }
+ console.log("dasar hukum PPID success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/ikm/seed_ikm.ts b/prisma/_seeder_list/ppid/ikm/seed_ikm.ts
new file mode 100644
index 00000000..905b4f4d
--- /dev/null
+++ b/prisma/_seeder_list/ppid/ikm/seed_ikm.ts
@@ -0,0 +1,54 @@
+import prisma from "@/lib/prisma";
+import jenisKelamin from "../../../data/ppid/ikm/jenis-kelamin/jenis-kelamin.json";
+import pilihanRatingResponden from "../../../data/ppid/ikm/pilihan-rating-responden/rating-responden.json";
+import umurResponden from "../../../data/ppid/ikm/umur-responden/umur-responden.json";
+
+export async function seedIkmPpid() {
+ for (const j of jenisKelamin) {
+ await prisma.jenisKelaminResponden.upsert({
+ where: {
+ id: j.id,
+ },
+ update: {
+ name: j.name,
+ },
+ create: {
+ id: j.id,
+ name: j.name,
+ },
+ });
+ }
+ console.log("jenis kelamin responden success ...");
+
+ for (const r of pilihanRatingResponden) {
+ await prisma.pilihanRatingResponden.upsert({
+ where: {
+ id: r.id,
+ },
+ update: {
+ name: r.name,
+ },
+ create: {
+ id: r.id,
+ name: r.name,
+ },
+ });
+ }
+ console.log("pilihan rating responden success ...");
+
+ for (const u of umurResponden) {
+ await prisma.umurResponden.upsert({
+ where: {
+ id: u.id,
+ },
+ update: {
+ name: u.name,
+ },
+ create: {
+ id: u.id,
+ name: u.name,
+ },
+ });
+ }
+ console.log("umur responden success ...");
+}
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/profil-ppid/seed_profil_ppd.ts b/prisma/_seeder_list/ppid/profil-ppid/seed_profil_ppd.ts
new file mode 100644
index 00000000..bf666e2d
--- /dev/null
+++ b/prisma/_seeder_list/ppid/profil-ppid/seed_profil_ppd.ts
@@ -0,0 +1,48 @@
+import prisma from "@/lib/prisma";
+import profilPpd from "../../../data/ppid/profile-ppid/profilePPid.json"
+
+export async function seedProfilPpd() {
+ console.log("🔄 Seeding Profil PPD...");
+ for (const m of profilPpd) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for berita "${m.name}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.profilePPID.upsert({
+ where: { id: m.id },
+ update: {
+ name: m.name,
+ biodata: m.biodata,
+ riwayat: m.riwayat,
+ pengalaman: m.pengalaman,
+ unggulan: m.unggulan,
+ imageId,
+ },
+ create: {
+ id: m.id,
+ name: m.name,
+ biodata: m.biodata,
+ riwayat: m.riwayat,
+ pengalaman: m.pengalaman,
+ unggulan: m.unggulan,
+ imageId,
+ },
+ });
+ }
+
+ console.log("profil ppd success ...");
+}
+
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/struktur-ppid/seed_struktur_ppid.ts b/prisma/_seeder_list/ppid/struktur-ppid/seed_struktur_ppid.ts
new file mode 100644
index 00000000..e2141e91
--- /dev/null
+++ b/prisma/_seeder_list/ppid/struktur-ppid/seed_struktur_ppid.ts
@@ -0,0 +1,82 @@
+import prisma from "@/lib/prisma";
+import pegawaiPpid from "../../../data/ppid/struktur-ppid/pegawai-PPID.json"
+import posisiOrganisasiPPID from "../../../data/ppid/struktur-ppid/posisi-organisasi-PPID.json"
+
+export async function seedPegawaiPpid() {
+
+const flattenedPosisi = posisiOrganisasiPPID.flat();
+
+ // ✅ Urutkan berdasarkan hierarki
+ const sortedPosisi = flattenedPosisi.sort((a, b) => a.hierarki - b.hierarki);
+
+ for (const p of sortedPosisi) {
+ console.log(`Seeding: ${p.nama} (id: ${p.id}, parent: ${p.parentId})`);
+
+ if (p.parentId) {
+ const parentExists = flattenedPosisi.some((pos) => pos.id === p.parentId);
+ if (!parentExists) {
+ console.warn(
+ `⚠️ Parent tidak ditemukan: ${p.parentId} untuk ${p.nama}`,
+ );
+ continue;
+ }
+ }
+
+ await prisma.posisiOrganisasiPPID.upsert({
+ where: { id: p.id },
+ update: p,
+ create: p,
+ });
+ }
+ console.log("posisi organisasi berhasil");
+
+ console.log("🔄 Seeding Struktur Ppid...");
+ for (const m of pegawaiPpid) {
+ let imageId: string | null = null;
+
+ if (m.imageName) {
+ const image = await prisma.fileStorage.findUnique({
+ where: { name: m.imageName },
+ select: { id: true },
+ });
+
+ if (!image) {
+ console.warn(
+ `⚠️ Image not found for pegawai ppid "${m.namaLengkap}": ${m.imageName}`,
+ );
+ } else {
+ imageId = image.id;
+ }
+ }
+
+ await prisma.pegawaiPPID.upsert({
+ where: { id: m.id },
+ update: {
+ namaLengkap: m.namaLengkap,
+ gelarAkademik: m.gelarAkademik,
+ tanggalMasuk: m.tanggalMasuk,
+ email: m.email,
+ telepon: m.telepon,
+ alamat: m.alamat,
+ imageId,
+ posisiId: m.posisiId,
+ isActive: m.isActive,
+ },
+ create: {
+ id: m.id,
+ namaLengkap: m.namaLengkap,
+ gelarAkademik: m.gelarAkademik,
+ tanggalMasuk: m.tanggalMasuk,
+ email: m.email,
+ telepon: m.telepon,
+ alamat: m.alamat,
+ imageId,
+ posisiId: m.posisiId,
+ isActive: m.isActive,
+ },
+ });
+ }
+
+ console.log("struktur ppid success ...");
+}
+
\ No newline at end of file
diff --git a/prisma/_seeder_list/ppid/visi-misi-ppid/seed_visi_misi_ppid.ts b/prisma/_seeder_list/ppid/visi-misi-ppid/seed_visi_misi_ppid.ts
new file mode 100644
index 00000000..e99087e3
--- /dev/null
+++ b/prisma/_seeder_list/ppid/visi-misi-ppid/seed_visi_misi_ppid.ts
@@ -0,0 +1,23 @@
+import prisma from "@/lib/prisma";
+import visiMisiPPID from "../../../data/ppid/visi-misi-ppid/visimisiPPID.json"
+
+export async function seedVisiMisiPpid() {
+ for (const v of visiMisiPPID) {
+ await prisma.visiMisiPPID.upsert({
+ where: {
+ id: v.id,
+ },
+ update: {
+ misi: v.misi,
+ visi: v.visi,
+ },
+ create: {
+ id: v.id,
+ misi: v.misi,
+ visi: v.visi,
+ },
+ });
+ }
+ console.log("visi misi PPID success ...");
+
+}
\ No newline at end of file
diff --git a/prisma/data/category-pengumuman.json b/prisma/data/category-pengumuman.json
deleted file mode 100644
index 5bacc742..00000000
--- a/prisma/data/category-pengumuman.json
+++ /dev/null
@@ -1,8 +0,0 @@
-[
- { "name": "Sosial & Kesehatan" },
- { "name": "Ekonomi & UMKM" },
- { "name": "Pendidikan & Kepemudaan" },
- { "name": "Lingkungan & Bencana" },
- { "name": "Adat & Budaya" },
- { "name": "Digitalisasi Desa" }
-]
diff --git a/prisma/data/desa/berita/berita.json b/prisma/data/desa/berita/berita.json
new file mode 100644
index 00000000..904361fc
--- /dev/null
+++ b/prisma/data/desa/berita/berita.json
@@ -0,0 +1,146 @@
+[
+ {
+ "id": "cmk6ae8rz00003b6r06x7hsqi",
+ "judul": "TP. Posyandu Bali Gelar Aksi Sosial ‘Membina dan Berbagi’ di Desa Darmasaba",
+ "deskripsi": "Kegiatan pembinaan dan bantuan kepada kader Posyandu Desa Darmasaba oleh TP Posyandu Provinsi Bali.
",
+ "content": "Sebanyak 50 kader posyandu mendapatkan pembinaan dan sembako sebagai dukungan terhadap peran Posyandu dalam pemberdayaan masyarakat. Kegiatan ini menunjukkan peran strategis posyandu dalam layanan publik desa.
",
+ "kategoriBeritaId": "cmk69tghy000vvnv8xeouenv5",
+ "imageName": "Wp41ccw3yma9W8i6zBr6E-mobile.webp"
+ },
+ {
+ "id": "cmk6af7vf00013b6rj2br4nv8",
+ "judul": "Desa Darmasaba Gelar Temu Sadar Hukum, Bahas Isu KDRT",
+ "deskripsi": "Temu Sadar Hukum untuk meningkatkan kesadaran hukum warga Desa Darmasaba.
",
+ "content": "Kegiatan ini membahas isu kekerasan dalam rumah tangga dengan narasumber dari Kanwil Kemenkum Bali, menjadi forum penting dalam membangun masyarakat yang melek hukum.
",
+ "kategoriBeritaId": "cmk69tghy000vvnv8xeouenv5",
+ "imageName": "IwedNmhjD_wGpY6PvYX7W-mobile.webp"
+ },
+ {
+ "id": "cmk6afo0g00033b6rjc2pae67",
+ "judul": "Bicara Darmasaba Bahas Berbagai Persoalan, Dorong Warga Sampaikan Aspirasi",
+ "deskripsi": "Ruang dialog terbuka di Desa Darmasaba untuk membahas persoalan sampah dan partisipasi publik.
",
+ "content": "Forum dialog ini membantu pemerintahan desa menyusun kebijakan tepat sasaran, terutama terkait permasalahan lingkungan dan aspirasi warga.
",
+ "kategoriBeritaId": "cmk69tghy000vvnv8xeouenv5",
+ "imageName": "EVkMxPdoWyL3y31L7d7x1-mobile.webp"
+ },
+ {
+ "id": "cmk6ag56y00053b6rj9481z6m",
+ "judul": "Bicara Darmasaba Bahas ‘Sampah Kita, Tanggung Jawab Siapa?’",
+ "deskripsi": "Diskusi terbuka antara pemerintah desa dan warga mengenai isu sampah.
",
+ "content": "Acara ini mendukung perumusan kebijakan desa yang sesuai kebutuhan warga dan meningkatkan partisipasi dalam pembangunan lingkungan.
",
+ "kategoriBeritaId": "cmk69tghx000tvnv8g2d206wv",
+ "imageName": "c_5xOKUbMiD8dTAbkAv9a-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxcsj",
+ "judul": "Penutupan KKN-PMM Periode II Universitas Warmadewa di Desa Darmasaba",
+ "deskripsi": "Penutupan program KKN-PMM yang berjalan dengan berbagai kegiatan pemberdayaan masyarakat.
",
+ "content": "Kegiatan KKN meliputi edukasi kesehatan, pengelolaan sampah, literasi keuangan, dan upaya ekonomi lokal sebagai bagian dari pembangunan desa berkelanjutan.
",
+ "kategoriBeritaId": "cmk69tghx000tvnv8g2d206wv",
+ "imageName": "I9CDBqdeDXRbbzbPWhy6h-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxasj",
+ "judul": "Desa Darmasaba Siap Kelola Sampah Mandiri, Anggarkan Rp1,5 Miliar",
+ "deskripsi": "Desa Darmasaba mengalokasikan anggaran untuk pengelolaan sampah berbasis komunitas.
",
+ "content": "Pengelolaan sampah mandiri melalui TPS3R, kader penyuluh, dan inovasi CINtA menjadi strategi desa dalam penanganan sampah sesuai kebijakan provinsi dan desa.
",
+ "kategoriBeritaId": "cmk69tghx000tvnv8g2d206wv",
+ "imageName": "Gq-gEDaGNb-FkKasVs7i4-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxbsj",
+ "judul": "Sekda Adi Arnawa Buka Darmasaba Village Festival II",
+ "deskripsi": "Pembukaan festival desa untuk mendorong UMKM dan ekonomi lokal.
",
+ "content": "Kegiatan ini menampilkan berbagai UMKM yang membantu meningkatkan pendapatan masyarakat setempat.
",
+ "kategoriBeritaId": "cmk69tghx000uvnv847ppcxqh",
+ "imageName": "TymQ5xDH7vEUOA9BY63hr-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxdsj",
+ "judul": "Membangun Desa Berkelanjutan Melalui Ekowisata dan Kuliner di Darmasaba",
+ "deskripsi": "Program inovatif untuk memperkuat ekonomi lokal melalui ekowisata dan kuliner.
",
+ "content": "Kegiatan mencakup pembangunan green house, edukasi pemasaran digital, literasi bahasa Inggris, dan pengembangan potensi kuliner desa.
",
+ "kategoriBeritaId": "cmk69tghx000uvnv847ppcxqh",
+ "imageName": "2nUvEBsMuigIJQWZIdcEJ-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxesj",
+ "judul": "Inovasi Desa Darmasaba Lanjutkan Perjuangan ke Tingkat Nasional",
+ "deskripsi": "Desa Darmasaba meraih penghargaan juara dalam evaluasi perkembangan desa untuk mendukung perekonomian dan pemerintahan lokal.
",
+ "content": "Prestasi desa ditandai dengan keberhasilan dalam lomba evaluasi perkembangan desa di tingkat provinsi dan kabupaten, yang berdampak positif pada ekonomi desa.
",
+ "kategoriBeritaId": "cmk69tghx000uvnv847ppcxqh",
+ "imageName": "Hokgum3-nI_NWTWJRnWi3-mobile.webp"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxfsj",
+ "judul": "Desa Darmasaba Kembali Ukir Prestasi Internasional BAJRA",
+ "deskripsi": "Prestasi desa dalam forum internasional mengenai penanggulangan rabies.
",
+ "content": "Partisipasi dalam konferensi Rabies in Borneo menunjukkan kolaborasi lintas sektor dan penggunaan data dalam pelayanan publik desa.
",
+ "kategoriBeritaId": "cmk69tght000svnv8ok5rid2v",
+ "imageName": "T1fcksUoZSUqNMbzvr9WI-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp1z",
+ "judul": "Cegah Penyebaran Rabies, Darmasaba Keluarkan Larangan Membuang Hewan",
+ "deskripsi": "Pemasangan spanduk larangan buang hewan sebagai langkah proteksi kesehatan masyarakat.
",
+ "content": "Pemerintah desa bersama Tim Bajra aktif dalam kampanye dan regulasi untuk mencegah rabies di tingkat desa.
",
+ "kategoriBeritaId": "cmk69tght000svnv8ok5rid2v",
+ "imageName": "-M_tICRVz6ZxOfvkuHQgU-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp2z",
+ "judul": "TP. Posyandu Bali dan Pemerintah Desa Kolaborasi Tingkatkan Pelayanan",
+ "deskripsi": "Kolaborasi pemerintahan desa dengan TP Posyandu untuk meningkatkan layanan masyarakat.
",
+ "content": "Kegiatan ini menunjukkan peran pemerintahan desa dalam mendukung layanan kesehatan dan pemberdayaan masyarakat.
",
+ "kategoriBeritaId": "cmk69tght000svnv8ok5rid2v",
+ "imageName": "O11hmN9oNwFKs_ACpH9uV-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp3z",
+ "judul": "Membangun Desa Berkelanjutan Melalui Inovasi Pengelolaan Sampah",
+ "deskripsi": "Inisiatif pengelolaan sampah desa sebagai bagian dari inovasi teknologi lokal.
",
+ "content": "Penerapan metode pengelolaan sampah dan biopori menunjukkan upaya Desa Darmasaba dalam menggunakan solusi teknologi sederhana untuk masalah lingkungan.
",
+ "kategoriBeritaId": "cmk69tghz000xvnv8kxzzt24h",
+ "imageName": "rrgHHUYHDuq3jW94HCRsq-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp4z",
+ "judul": "Inovasi BAJRA Integrasikan Pelaporan Cepat Berbasis Data",
+ "deskripsi": "Program BAJRA menerapkan mekanisme pelaporan cepat berbasis data untuk penanggulangan rabies.
",
+ "content": "Penggunaan teknologi informasi dalam pelaporan kasus rabies membantu respons cepat pemerintahan desa dan komunitas.
",
+ "kategoriBeritaId": "cmk69tghz000xvnv8kxzzt24h",
+ "imageName": "igz0V0MCoLYqAgLIBRZdG-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp5z",
+ "judul": "Digitalisasi Desa Darmasaba",
+ "deskripsi": "Digitalisasi Desa Darmasaba Bersama PT. Bali Interaktif Perkasa.
",
+ "content": "Digitalisasi Desa Darmasaba Bersama PT. Bali Interaktif Perkasa Dalam rangka mendukung transformasi digital dan inovasi desa, Desa Darmasaba bekerja sama dengan PT. Bali Interaktif Perkasa melaksanakan kegiatan Digitalisasi Desa. Program ini bertujuan untuk memperkuat kapasitas desa dalam pemanfaatan teknologi informasi dan komunikasi, sehingga pelayanan publik menjadi lebih efektif, transparan, dan cepat. Masyarakat juga diberikan pemahaman terkait pemanfaatan platform digital untuk kegiatan administrasi, komunikasi, dan pengembangan potensi desa. Kegiatan digitalisasi ini menjadi bagian dari komitmen Desa Darmasaba untuk mewujudkan desa cerdas (smart village) yang mampu bersaing dan beradaptasi di era digital, sekaligus meningkatkan inovasi dan pemberdayaan masyarakat. Dengan kolaborasi ini, Desa Darmasaba menegaskan tekadnya untuk terus berinovasi, menghadirkan kemudahan bagi masyarakat, dan memperkuat tata kelola desa berbasis teknologi modern. ? Digitalisasi hari ini, kemajuan desa esok! #DesaDarmasaba #DigitalisasiDesa #DesaCerdas #InovasiDesa #TransformasiDigital #PemdesDarmasaba #PerbekelDarmasaba #DesaDarmasaba #KitaDarmasaba #DarmasabaBisa @kostergubernurbali @giri.prasta @iwayanadiarnawa @gus.bota @puturasniathiadiarnawa @yunita_oktarini @surya.suamba @budhi.argawakba @pemkabbadung @ppidbadung @dinaspmddukcapilprovbali @surya_prabhawa @kecamatanabiansemal @dpmdbadungkab @pemprov_bali @prokompimbadung @seputar_darmasaba
",
+ "kategoriBeritaId": "cmk69tghz000xvnv8kxzzt24h",
+ "imageName": "DzVIfpiAP3OcCsZ_VJ02b-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp6z",
+ "judul": "Festival Desa Tingkatkan Kreativitas Digital UMKM",
+ "deskripsi": "Festival Darmasaba Village Festival II melibatkan promosi digital produk UMKM.
",
+ "content": "Promosi dan dokumentasi digital menjadi bagian dari strategi pemasaran UMKM dalam festival desa.
",
+ "kategoriBeritaId": "cmk69tghy000wvnv8umg2vloa",
+ "imageName": "xzM77A6bDW2silyp_8W7n-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp7z",
+ "judul": "Sekda Adi Arnawa dan Pementasan Seni Tradisional di Festival Darmasaba",
+ "deskripsi": "Pementasan seni tradisional menjadi bagian dari Darmasaba Village Festival II.
",
+ "content": "Kegiatan ini mengangkat warisan budaya lokal melalui pertunjukan dan lomba di festival desa.
",
+ "kategoriBeritaId": "cmk69tghy000wvnv8umg2vloa",
+ "imageName": "2wivBEDcVNxHGG8HUBsNH-mobile.webp"
+ },
+ {
+ "id": "cmk6aih8f00093b6rqw63yp8z",
+ "judul": "Dialog Publik Tingkatkan Partisipasi Budaya Lokal",
+ "deskripsi": "Forum dialog desa mengangkat tema partisipasi masyarakat dalam kegiatan budaya lokal.
",
+ "content": "Diskusi ini memperkuat peran budaya dalam pembangunan desa melalui keterlibatan warga dalam kegiatan adat dan sosial.
",
+ "kategoriBeritaId": "cmk69tghy000wvnv8umg2vloa",
+ "imageName": "T1fcksUoZSUqNMbzvr9WI-mobile.webp"
+ }
+]
diff --git a/prisma/data/desa/berita/kategori-berita.json b/prisma/data/desa/berita/kategori-berita.json
index 4d777965..e301a584 100644
--- a/prisma/data/desa/berita/kategori-berita.json
+++ b/prisma/data/desa/berita/kategori-berita.json
@@ -1,8 +1,8 @@
[
- { "name": "Pemerintahan" },
- { "name": "Pembangunan" },
- { "name": "Ekonomi" },
- { "name": "Sosial" },
- { "name": "Budaya" },
- { "name": "Teknologi" }
+ { "id": "cmk69tght000svnv8ok5rid2v", "name": "Pemerintahan" },
+ { "id": "cmk69tghx000tvnv8g2d206wv", "name": "Pembangunan" },
+ { "id": "cmk69tghx000uvnv847ppcxqh", "name": "Ekonomi" },
+ { "id": "cmk69tghy000vvnv8xeouenv5", "name": "Sosial" },
+ { "id": "cmk69tghy000wvnv8umg2vloa", "name": "Budaya" },
+ { "id": "cmk69tghz000xvnv8kxzzt24h", "name": "Teknologi" }
]
diff --git a/prisma/data/desa/gallery/foto/foto.json b/prisma/data/desa/gallery/foto/foto.json
new file mode 100644
index 00000000..f1141c30
--- /dev/null
+++ b/prisma/data/desa/gallery/foto/foto.json
@@ -0,0 +1,20 @@
+[
+ {
+ "id": "cml0aiiv1000004l754ldaf2v",
+ "name": "Kunjungan Ibu TP PKK Kabupaten Badung",
+ "deskripsi": "Dokumentasi kunjungan Ibu TP PKK Kabupaten Badung ke Desa Darmasaba pada awal tahun 2026.
",
+ "imageName": "foto1Gallery.webp"
+ },
+ {
+ "id": "cml0aiqd4000104l7f91ee3xu",
+ "name": "Darmasaba Village Festival II 2024",
+ "deskripsi": "Foto kegiatan Darmasaba Village Festival II Tahun 2024 yang diselenggarakan di Lapangan Umum Desa Darmasaba, menampilkan lomba, pementasan seni, dan UMKM lokal.
",
+ "imageName": "foto2Gallery.webp"
+ },
+ {
+ "id": "cml0aiyi7000204l7f2sy657c",
+ "name": "Lomba Mancing Air Deras Banjar Gulingan",
+ "deskripsi": "Galeri foto kegiatan lomba mancing air deras Komunitas Pemancing Gulingan di Desa Darmasaba, bagian dari pemberdayaan masyarakat.
",
+ "imageName": "foto3Gallery.webp"
+ }
+]
diff --git a/prisma/data/desa/gallery/video/video.json b/prisma/data/desa/gallery/video/video.json
new file mode 100644
index 00000000..e3ae007f
--- /dev/null
+++ b/prisma/data/desa/gallery/video/video.json
@@ -0,0 +1,20 @@
+[
+ {
+ "id": "cmk6kvn6b0000vn6qzg5z6qa6",
+ "judul": "TAHAP PENILAIAN VERIFIKASI LAPANGAN AJANG MANGUPURA AWARD TAHUN 2025",
+ "deskripsi": "TAHAP PENILAIAN VERIFIKASI LAPANGAN AJANG MANGUPURA AWARD Senin, 29 September 2025 – Pemerintah Desa Darmasaba mengikuti tahap penilaian verifikasi lapangan dalam rangkaian Ajang Mangupura Award Tahun 2025 pada kategori Pemerintah Desa. Ajang bergengsi ini merupakan bentuk apresiasi Pemerintah Kabupaten Badung kepada desa-desa yang berprestasi dalam tata kelola aset, keuangan, arsip, tata kelola sumber daya manusia, pelayanan publik dan persampahan, inovasi, sinergitas, dan akuntabilitas. Proses verifikasi lapangan diawali di Kantor Perbekel Darmasaba dengan pemeriksaan langsung kepada masing-masing pengampu indikator, kemudian dilanjutkan dengan kunjungan ke BUMDes Pudak Mesari serta TPS 3R Pudak Mesari sebagai bentuk evaluasi nyata terhadap kinerja dan program desa. Kegiatan ini menjadi langkah penting dalam menilai implementasi tata kelola pemerintahan desa yang transparan, inovatif, dan berkelanjutan. Melalui tahapan ini, Pemerintah Desa Darmasaba berharap dapat terus menghadirkan pelayanan terbaik bagi masyarakat, mengembangkan inovasi desa, serta memperkuat sinergi antara pemerintah, desa adat, dan masyarakat dalam mewujudkan pembangunan daerah yang maju, berdaya saing, dan berkelanjutan. #MangupuraAward2025 #Darmasaba #DesaBerprestasi #PemerintahDesa #Badung #Abiansemal #PemdesDarmasaba #PerbekelDarmasaba #DesaDarmasaba #KitaDarmasaba #DarmasabaBisa
",
+ "linkVideo": "https://www.youtube.com/watch?v=e2tSRnNkYDE"
+ },
+ {
+ "id": "cmk6kvn6b0000vn6qzg5z6qb7",
+ "judul": "Vaksinasi Rabies di Desa Darmasaba",
+ "deskripsi": "Vaksinasi Rabies di Desa Darmasaba Selasa, 7 Oktober 2025 Pemerintah Desa Darmasaba melalui Tim Bajra Desa Darmasaba, bekerja sama dengan Dinas Pertanian dan Pangan Kabupaten Badung Bidang Kesehatan Hewan, menyelenggarakan kegiatan Vaksinasi Rabies bagi hewan penular rabies (HPR) yang meliputi anjing, kucing, dan kera di wilayah Desa Darmasaba. Kegiatan ini dilaksanakan sebagai langkah preventif untuk menekan penyebaran virus rabies sekaligus memberikan perlindungan kesehatan bagi hewan peliharaan maupun masyarakat. Melalui program vaksinasi ini, diharapkan Desa Darmasaba dapat terbebas dari ancaman rabies dan semakin meningkatkan kesadaran masyarakat akan pentingnya menjaga kesehatan hewan peliharaan. Pemerintah Desa Darmasaba mengimbau seluruh warga untuk memastikan hewan kesayangan mendapatkan vaksinasi sesuai jadwal yang telah ditentukan. Informasi lengkap mengenai jadwal vaksinasi dapat dilihat pada pengumuman yang tertera. Selain itu, demi kelancaran proses vaksinasi, diharapkan pemilik hewan dapat mengikat atau mengandangkan hewan peliharaannya saat proses vaksinasi berlangsung. Dengan adanya kolaborasi antara pemerintah desa, dinas terkait, dan partisipasi aktif masyarakat, kegiatan ini diharapkan mampu memberikan manfaat nyata serta menciptakan lingkungan Desa Darmasaba yang lebih sehat, aman, dan terbebas dari rabies. #VaksinasiRabies #KesehatanHewan #BebasRabies #Badung #Abiansemal #TimBajraDarmasaba #PemdesDarmasaba #PerbekelDarmasaba #DesaDarmasaba #KitaDarmasaba #DarmasabaBisa
",
+ "linkVideo": "https://www.youtube.com/watch?v=bc99y94FBx8"
+ },
+ {
+ "id": "cmk6kvn6b0000vn6qzg5z6qc8",
+ "judul": "Musyawarah Perencanaan Pembangunan Desa Darmasaba Tahun 2026",
+ "deskripsi":"Musyawarah Perencanaan Pembangunan Desa Darmasaba Tahun 2026 Pemerintah Desa Darmasaba menyelenggarakan Musyawarah Perencanaan Pembangunan Desa (Musrenbangdes)dalam rangka penyusunan Rencana Kerja Pemerintah Desa (RKP Desa) Tahun 2026. Musrenbangdes ini merupakan forum resmi yang mempertemukan perwakilan dari DPMD Kab. Badung, Kecamatan Abiansemal, Pemerintah Desa, Lembaga Desa, Tokoh Masyarakat, serta perwakilan unsur lainnya untuk bersama-sama merumuskan arah pembangunan desa di tahun mendatang. Melalui kegiatan ini, seluruh peserta diberikan ruang untuk menyampaikan usulan, gagasan, serta masukan yang berkaitan dengan prioritas pembangunan, baik di bidang infrastruktur, pemberdayaan masyarakat, ekonomi, sosial, maupun pelestarian adat dan budaya. Proses musyawarah ini menjadi bagian penting dalam mewujudkan pembangunan desa yang partisipatif, transparan, dan berorientasi pada kebutuhan masyarakat. Dengan terselenggaranya Musrenbangdes ini, Pemerintah Desa Darmasaba berharap RKP Desa Tahun 2026 dapat disusun secara komprehensif, berkelanjutan, dan selaras dengan visi pembangunan daerah. Selain itu, kegiatan ini juga menegaskan komitmen Pemerintah Desa Darmasaba untuk terus menghadirkan pembangunan yang inklusif, berkeadilan, dan bermanfaat bagi seluruh lapisan masyarakat. #Musrenbangdes2026 #RKPDarmasaba2026 #PemdesDarmasaba #PerbekelDarmasaba #DesaDarmasaba #KitaDarmasaba #DarmasabaBisa
",
+ "linkVideo": "https://www.youtube.com/watch?v=7pirwEmyP-4"
+ }
+]
diff --git a/prisma/data/desa/layanan/pelayanaPendudukNonPermanen.json b/prisma/data/desa/layanan/pelayananPendudukNonPermanen.json
similarity index 100%
rename from prisma/data/desa/layanan/pelayanaPendudukNonPermanen.json
rename to prisma/data/desa/layanan/pelayananPendudukNonPermanen.json
diff --git a/prisma/data/desa/layanan/pelayananSuratKeterangan.json b/prisma/data/desa/layanan/pelayananSuratKeterangan.json
index 9f13159c..87ea9f17 100644
--- a/prisma/data/desa/layanan/pelayananSuratKeterangan.json
+++ b/prisma/data/desa/layanan/pelayananSuratKeterangan.json
@@ -1,57 +1,79 @@
[
- {
- "id" : "cmdxyb9zi0010vniiaeyi55ui",
- "name" : "Surat Keterangan Beda Biodata Diri",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Fotocopy dokumen bersangkutan yang terdapat perbedaan biodata diri misal : Sertifikat Tanah/Ijazah/Polis Asuransi dan lainnya.
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxycqz40014vniidftrixvf",
- "name" : "Surat Keterangan Yatim Piatu",
- "deskripsi" : "Persyaratan Dokumen :
Alur Pelayanan :
"
- },
- {
- "id" : "cmdwx3wph0003vnr74us2t7h7",
- "name" : "Surat Keterangan Domisili Organisasi",
- "deskripsi" : "Persyaratan Dokumen:
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy Surat Keterangan Terdaftar (SKT) organisasi atau Pengukuhan Kelompok
Jika Pengajuan baru pembuatan SKT maka melengkapi Susunan Pengurus lengkap dengan Kop Organisasi
Tanggal berdiri/Tahun berdiri/Sejak kapan berdirinya organisasi
Alur Pelayanan:
"
- },
- {
- "id" : "cmdxxv3i80004vniidg1mrucc",
- "name" : "Surat Keterangan Penghasilan",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP orang tua atau Fotocopy Kartu keluarga
Membuat Surat Pernyataan Penghasilan bermaterai (disertai jumlah penghasilan)
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxxwp070008vnii9jbdcto7",
- "name" : "Surat Keterangan Tidak Mampu",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP/KIA atau Kartu Keluarga
Fotocopy Kartu Indonesia Pintar/Kartu Perlindungan Sosial/Terdaftar dalam DTKS
Jika tidak memiliki Kartu tersebut diatas diwajibkan membuat Surat Pernyataan Tidak Mampu
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxxyfkl000cvnii1bxinnfi",
- "name" : "Surat Keterangan Kelahiran",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy Surat lahir dari dokter/bidan (jika ada)
Fotocopy Kartu Keluarga
Fotocopy KTP 2 orang saksi
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxy23pl000gvniihsg38aq4",
- "name" : "Surat Keterangan Usaha",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Foto Lokasi dan Kegiatan Usaha di cetak dalam selembar kertas (diparaf dan stempel oleh Kelian Banjar Dinas)
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxy4mgt000kvniib1nemjem",
- "name" : "Surat Keterangan Kematian",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Surat Kematian dari rumah sakit atau dokter (jika ada)
tanggal kematian
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxy61a1000ovniif4ytb9hs",
- "name" : "Surat Keterangan Tempat Usaha",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Foto Lokasi dan Kegiatan Usaha di cetak dalam selembar kertas (diparaf dan stempel oleh Kelian Banjar Dinas)
Surat Perjanjian Sewa/Kontrak atau Kwintansi Pembayaran Sewa 3 bulan terakhir bagi yang mengontrak tempat usaha, apabila tempat usaha milik sendiri lampiri dengan dokumen kepemilikan tempat usaha (dapat berupa fotocopy sppt atau Fotocopy Sertipikat Hak Milik)
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxy754q000svniiiz8oqyo0",
- "name" : "Surat Keterangan Belum Kawin",
- "deskripsi" : "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Khusus bagi yang berstatus duda atau janda melampirkan fotocopy akta cerai atau dokumen pendukung lainnya
Alur Pelayanan :
"
- },
- {
- "id" : "cmdxy8pi2000wvnii48fc1sxd",
- "name" : "Surat Keterangan Kelakuan Baik",
- "deskripsi" : "Persyaratan Dokumen :
Alur Pelayanan :
"
- }
-]
\ No newline at end of file
+ {
+ "id": "cmdxyb9zi0010vniiaeyi55ui",
+ "name": "Surat Keterangan Beda Biodata Diri",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Fotocopy dokumen bersangkutan yang terdapat perbedaan biodata diri misal : Sertifikat Tanah/Ijazah/Polis Asuransi dan lainnya.
Alur Pelayanan :
",
+ "imageName": "wFXYVHKHtU7posfhBKjZt-mobile.webp",
+ "image2Name": "kTiZ9uhRUEwu0WeASU00b-mobile.webp"
+ },
+ {
+ "id": "cmdxycqz40014vniidftrixvf",
+ "name": "Surat Keterangan Yatim Piatu",
+ "deskripsi": "Persyaratan Dokumen :
Alur Pelayanan :
",
+ "imageName": "ZkvjJ7Zx8uBy2-d8RWNEt-mobile.webp",
+ "image2Name": "hHfOWKz4USZK-z0nsD7Uz-mobile.webp"
+ },
+ {
+ "id": "cmdwx3wph0003vnr74us2t7h7",
+ "name": "Surat Keterangan Domisili Organisasi",
+ "deskripsi": "Persyaratan Dokumen:
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy Surat Keterangan Terdaftar (SKT) organisasi atau Pengukuhan Kelompok
Jika Pengajuan baru pembuatan SKT maka melengkapi Susunan Pengurus lengkap dengan Kop Organisasi
Tanggal berdiri/Tahun berdiri/Sejak kapan berdirinya organisasi
Alur Pelayanan:
",
+ "imageName": "03WZn5JMKffo62cKzShF4-mobile.webp",
+ "image2Name": "dnTcxeYQACzY5yzq-Rjoz-mobile.webp"
+ },
+ {
+ "id": "cmdxxv3i80004vniidg1mrucc",
+ "name": "Surat Keterangan Penghasilan",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP orang tua atau Fotocopy Kartu keluarga
Membuat Surat Pernyataan Penghasilan bermaterai (disertai jumlah penghasilan)
Alur Pelayanan :
",
+ "imageName": "7er4OinxhuKRZUSkKeSYR-mobile.webp",
+ "image2Name": "8ZtJeCnKwMAcEUBU0QvE7-mobile.webp"
+ },
+ {
+ "id": "cmdxxwp070008vnii9jbdcto7",
+ "name": "Surat Keterangan Tidak Mampu",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP/KIA atau Kartu Keluarga
Fotocopy Kartu Indonesia Pintar/Kartu Perlindungan Sosial/Terdaftar dalam DTKS
Jika tidak memiliki Kartu tersebut diatas diwajibkan membuat Surat Pernyataan Tidak Mampu
Alur Pelayanan :
",
+ "imageName": "Y7IuwpwjT3ZFYWpiXJZYw-mobile.webp",
+ "image2Name": "Fbi-6gnhokkwux9hvriS--mobile.webp"
+ },
+ {
+ "id": "cmdxxyfkl000cvnii1bxinnfi",
+ "name": "Surat Keterangan Kelahiran",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy Surat lahir dari dokter/bidan (jika ada)
Fotocopy Kartu Keluarga
Fotocopy KTP 2 orang saksi
Alur Pelayanan :
",
+ "imageName": "qtdJ39rIbjGTJJQZflrDL-mobile.webp",
+ "image2Name": "ldmfJBS2ZBhIda60yU2JW-mobile.webp"
+ },
+ {
+ "id": "cmdxy23pl000gvniihsg38aq4",
+ "name": "Surat Keterangan Usaha",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Foto Lokasi dan Kegiatan Usaha di cetak dalam selembar kertas (diparaf dan stempel oleh Kelian Banjar Dinas)
Alur Pelayanan :
",
+ "imageName": "CAtRFHiM11gsY4ExcvGgc-mobile.webp",
+ "image2Name": "MLWp-38kKygXKVekCh0q7-mobile.webp"
+ },
+ {
+ "id": "cmdxy4mgt000kvniib1nemjem",
+ "name": "Surat Keterangan Kematian",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Surat Kematian dari rumah sakit atau dokter (jika ada)
tanggal kematian
Alur Pelayanan :
",
+ "imageName": "C0zE5lneKa888VJDHzgh--mobile.webp",
+ "image2Name": "hhDK7OL0wQLik5dsh1a-L-mobile.webp"
+ },
+ {
+ "id": "cmdxy61a1000ovniif4ytb9hs",
+ "name": "Surat Keterangan Tempat Usaha",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Foto Lokasi dan Kegiatan Usaha di cetak dalam selembar kertas (diparaf dan stempel oleh Kelian Banjar Dinas)
Surat Perjanjian Sewa/Kontrak atau Kwintansi Pembayaran Sewa 3 bulan terakhir bagi yang mengontrak tempat usaha, apabila tempat usaha milik sendiri lampiri dengan dokumen kepemilikan tempat usaha (dapat berupa fotocopy sppt atau Fotocopy Sertipikat Hak Milik)
Alur Pelayanan :
",
+ "imageName": "sVIfCUGBplNU8h1x99l8z-mobile.webp",
+ "image2Name": "MM-1CZMai3eXwbVr2HAEv-mobile.webp"
+ },
+ {
+ "id": "cmdxy754q000svniiiz8oqyo0",
+ "name": "Surat Keterangan Belum Kawin",
+ "deskripsi": "Persyaratan Dokumen :
Pengantar Kelian Banjar Dinas di Wilayah Masing - masing
Fotocopy KTP atau Kartu Keluarga
Khusus bagi yang berstatus duda atau janda melampirkan fotocopy akta cerai atau dokumen pendukung lainnya
Alur Pelayanan :
",
+ "imageName": "oNH9VvRPlAmTI_Ndu3slN-mobile.webp",
+ "image2Name": "Wze2x68vn-ShZAqgkSCMC-mobile.webp"
+ },
+ {
+ "id": "cmdxy8pi2000wvnii48fc1sxd",
+ "name": "Surat Keterangan Kelakuan Baik",
+ "deskripsi": "Persyaratan Dokumen :
Alur Pelayanan :
",
+ "imageName": "tK9T6BObEQEdYU-5y1xnJ-mobile.webp",
+ "image2Name": "sX9yfcM05OOx2ELhcSNWl-mobile.webp"
+ }
+]
diff --git a/prisma/data/desa/penghargaan/penghargaan.json b/prisma/data/desa/penghargaan/penghargaan.json
new file mode 100644
index 00000000..6b073a23
--- /dev/null
+++ b/prisma/data/desa/penghargaan/penghargaan.json
@@ -0,0 +1,30 @@
+[
+ {
+ "id": "cmkal9dd70002vnexhvn4j2fv",
+ "name": "Perbekel Darmasaba Terima Penghargaan pada Indonesia Alternative Dispute Resolution Awards 2025",
+ "juara": "Penghargaan IADRA 2025",
+ "deskripsi": "Perbekel Darmasaba, Ida Bagus Surya Prabhawa Manuaba menerima penghargaan pada Indonesia Alternative Dispute Resolution Awards 2025 atas kontribusi dalam penyelesaian sengketa secara damai dan berkeadilan.
",
+ "imageName": "cte_gy0V8glhDcghMa3-B-mobile.webp"
+ },
+ {
+ "id": "cmkalfbux0005vnexj8zplube",
+ "name": "Penghargaan Bhawana Sewaka Nugraha kepada Perbekel Darmasaba",
+ "juara": "Penghargaan Bhawana Sewaka Nugraha",
+ "deskripsi": "Penghargaan Bhawana Sewaka Nugraha diberikan kepada Perbekel Desa Darmasaba atas dedikasi dan komitmen dalam menjaga kelestarian lingkungan serta menginspirasi masyarakat untuk menciptakan desa yang hijau, bersih, dan berkelanjutan.
",
+ "imageName": "hLoy_DjB_3Yie2vjaXDYz-mobile.webp"
+ },
+ {
+ "id": "cmkalg6hk0008vnexmsqwd3n7",
+ "name": "Desa Darmasaba Raih Juara 1 Lomba Desa Tingkat Provinsi Bali Tahun 2025",
+ "juara": "Juara 1 Lomba Desa Provinsi Bali 2025",
+ "deskripsi": "Desa Darmasaba meraih juara 1 dalam Lomba Desa Tingkat Provinsi Bali Tahun 2025 melalui sinergi elemen desa serta inovasi dalam pelayanan publik dan pemberdayaan masyarakat.
",
+ "imageName": "VxELyZ0ifdHOdRFu7aBSz-mobile.webp"
+ },
+ {
+ "id": "cmkalgucu000bvnexw2blt1v3",
+ "name": "Pemerintah Desa Darmasaba Raih Peringkat IV dalam Ajang Mangupura Award Tahun 2025",
+ "juara": "Peringkat IV Mangupura Award 2025",
+ "deskripsi": "Desa Darmasaba mendapatkan penghargaan Peringkat IV pada Mangupura Award 2025 dari Pemerintah Kabupaten Badung atas tata kelola pemerintahan, inovasi layanan, dan pemberdayaan masyarakat.
",
+ "imageName": "xgrBxDdo8g7KA-3zNbRGj-mobile.webp"
+ }
+]
diff --git a/prisma/data/desa/pengumuman/kategori-pengumuman.json b/prisma/data/desa/pengumuman/kategori-pengumuman.json
new file mode 100644
index 00000000..dd97e2b8
--- /dev/null
+++ b/prisma/data/desa/pengumuman/kategori-pengumuman.json
@@ -0,0 +1,8 @@
+[
+ { "id": "cmk69tghx111tvnv8g2d206wv", "name": "Digitalisasi Desa" },
+ { "id": "cmk69tghx111uvnv847ppcxqh", "name": "Adat & Budaya" },
+ { "id": "cmk69tght111svnv8ok5rid2v", "name": "Lingkungan & Bencana" },
+ { "id": "cmk69tghy111wvnv8umg2vloa", "name": "Pendidikan & Kepemudaan" },
+ { "id": "cmk69tghy111vvnv8xeouenv5", "name": "Ekonomi & UMKM" },
+ { "id": "cmk69tghz111xvnv8kxzzt24h", "name": "Sosial & Kesehatan" }
+]
diff --git a/prisma/data/desa/pengumuman/pengumuman.json b/prisma/data/desa/pengumuman/pengumuman.json
new file mode 100644
index 00000000..0ac49936
--- /dev/null
+++ b/prisma/data/desa/pengumuman/pengumuman.json
@@ -0,0 +1,37 @@
+[
+ {
+ "id": "cmk6ae8rz00003b6r06x7hsqz",
+ "judul": "Pengumuman Lelang Pengadaan Pick UpDump TPS3R Pudak Mesari",
+ "deskripsi": "Pengumuman lelang pengadaan alat Pick UpDump untuk TPS3R Pudak Mesari Desa Darmasaba.
",
+ "content": "Desa Darmasaba membuka lelang pengadaan Pick UpDump untuk mendukung kegiatan pengelolaan sampah di TPS3R Pudak Mesari.
",
+ "categoryPengumumanId": "cmk69tghy111vvnv8xeouenv5"
+ },
+ {
+ "id": "cmk6af7vf00013b6rj2br4nv7",
+ "judul": "Pengumuman Lelang Pembangunan Taman BR. Darmasaba",
+ "deskripsi": "Pengumuman lelang pembangunan taman di Banjar BR. Darmasaba.
",
+ "content": "Desa Darmasaba mengumumkan pelaksanaan lelang pembangunan taman sebagai bagian dari upaya memperindah ruang publik desa.
",
+ "categoryPengumumanId": "cmk69tghy111vvnv8xeouenv5"
+ },
+ {
+ "id": "cmk6afo0g00033b6rjc2pae69",
+ "judul": "Pengumuman Lelang Penataan Lapangan Desa Darmasaba Tahun Anggaran 2024",
+ "deskripsi": "Pengumuman lelang penataan lapangan Desa Darmasaba untuk tahun anggaran 2024.
",
+ "content": "Lelang penataan lapangan desa dilaksanakan untuk menunjang kegiatan desa dan fasilitas masyarakat di Darmasaba.
",
+ "categoryPengumumanId": "cmk69tghy111vvnv8xeouenv5"
+ },
+ {
+ "id": "cmk6ag56y00053b6rj9481z5l",
+ "judul": "Pengumuman Lelang Penataan Landscape Lapangan Desa",
+ "deskripsi": "Pengumuman lelang penataan landscape lapangan Desa Darmasaba.
",
+ "content": "Desa Darmasaba menyelenggarakan lelang untuk kegiatan penataan landscape lapangan desa dalam rangka peningkatan fasilitas umum.
",
+ "categoryPengumumanId": "cmk69tghx111tvnv8g2d206wv"
+ },
+ {
+ "id": "cmk6agzxx00073b6rr3vhxcti",
+ "judul": "Penutupan KKN-PMM Periode II Universitas Warmadewa di Desa Darmasaba",
+ "deskripsi": "Penutupan program KKN-PMM yang berjalan dengan berbagai kegiatan pemberdayaan masyarakat.
",
+ "content": "Kegiatan KKN meliputi edukasi kesehatan, pengelolaan sampah, literasi keuangan, dan upaya ekonomi lokal sebagai bagian dari pembangunan desa berkelanjutan.
",
+ "categoryPengumumanId": "cmk69tghx111tvnv8g2d206wv"
+ }
+]
diff --git a/prisma/data/desa/potensi/kategori-potensi.json b/prisma/data/desa/potensi/kategori-potensi.json
new file mode 100644
index 00000000..e8c2d753
--- /dev/null
+++ b/prisma/data/desa/potensi/kategori-potensi.json
@@ -0,0 +1,14 @@
+[
+ {
+ "id": "cmk3pmwq10008vn9bqdquv153",
+ "nama": "Wisata"
+ },
+ {
+ "id": "cmk3s1ks6000ivn9bcrv960ko",
+ "nama": "Ekonomi"
+ },
+ {
+ "id": "cmk3s1r0m000jvn9b8tlhogwn",
+ "nama": "Lingkungan"
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/desa/potensi/potensi-desa.json b/prisma/data/desa/potensi/potensi-desa.json
index 9c6c8b66..43b8dfbf 100644
--- a/prisma/data/desa/potensi/potensi-desa.json
+++ b/prisma/data/desa/potensi/potensi-desa.json
@@ -1,74 +1,90 @@
[
{
- "id": "cmdyamai40004vnw3sdjbvn48",
- "name": "TPS3R Pudak Mesari",
- "deskripsi": "TPS 3R Pudak Mesari Darmasaba layak mendapat penghargaan demikian apresiasi dari Delterra Sosial Indonesia nie Semeton Darmasaba!, Hal tersebut dikarenakan walaupun baru berdiri namun TPS 3R kebanggaan Desa Darmasaba tersebut sudah berjalan dengan sangat baik.",
- "content": "TPS3R Pudak Mesari adalah Tempat Pengolahan Sampah dengan konsep Reduce, Reuse, dan Recycle (TPS3R) yang berlokasi di Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali. Fasilitas ini berperan penting dalam pengelolaan sampah berbasis masyarakat, dengan tujuan mengurangi volume sampah yang masuk ke Tempat Pembuangan Akhir (TPA) dan meningkatkan kesadaran warga tentang pentingnya pengelolaan sampah yang berkelanjutan.
Potensi Desa melalui TPS3R Pudak Mesari:
Peningkatan Kesehatan Lingkungan:
Dengan pengelolaan sampah yang efektif, desa dapat menjaga kebersihan lingkungan, mengurangi risiko penyakit, dan menciptakan suasana yang lebih nyaman bagi warga.
Pemberdayaan Ekonomi Masyarakat:
TPS3R membuka peluang usaha bagi warga melalui pemilahan dan pengolahan sampah, seperti produksi kompos dari sampah organik dan kerajinan tangan dari sampah anorganik yang dapat meningkatkan pendapatan masyarakat.
Edukasi dan Kesadaran Lingkungan:
Fasilitas ini dapat menjadi pusat edukasi bagi masyarakat tentang pentingnya pengelolaan sampah, mendorong partisipasi aktif dalam menjaga kelestarian lingkungan.
Pengembangan Pariwisata Berkelanjutan:
Dengan lingkungan yang bersih dan asri, Desa Darmasaba memiliki potensi untuk menarik wisatawan yang tertarik pada ekowisata dan budaya lokal, sehingga meningkatkan perekonomian desa.
"
- },
- {
- "id": "cmdyb7h440003vngjapbc84f7",
- "name": "Bumdes Pudak Mesari",
- "deskripsi": "Bumdes Pudak Mesari sangat membantu warga desa Darmasaba dalam mengelola dan membangun sebuah desa yang lebih baik lagi",
- "content": "Badan Usaha Milik Desa (BUMDes) Pudak Mesari adalah lembaga ekonomi desa yang berperan penting dalam pengembangan potensi dan kesejahteraan masyarakat Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali. BUMDes ini berfungsi sebagai motor penggerak perekonomian desa melalui berbagai unit usaha yang dikelola secara profesional.
Potensi dan Peran BUMDes Pudak Mesari:
Pengembangan Usaha Mikro dan Kecil:
BUMDes Pudak Mesari menyediakan layanan bagi pelaku usaha mikro dan kecil di desa, seperti penyediaan konsumsi dan snack kotak untuk berbagai acara.
Pengelolaan Sampah Berbasis Masyarakat:
Melalui kolaborasi dengan komunitas pemuda peduli lingkungan, BUMDes Pudak Mesari aktif dalam pengelolaan sampah berbasis masyarakat.
Peningkatan Kapasitas dan Transparansi:
Untuk memastikan pengelolaan yang akuntabel, BUMDes Pudak Mesari rutin mengadakan rapat koordinasi dan pendampingan penyusunan laporan pertanggungjawaban.
Kolaborasi Internasional:
Desa Darmasaba, melalui BUMDes Pudak Mesari, menerima kunjungan dari tim Osaki Jepang untuk memperkuat pengelolaan sampah dan lingkungan.
Dengan berbagai inisiatif tersebut, BUMDes Pudak Mesari menunjukkan perannya sebagai pilar utama dalam pengembangan ekonomi dan kesejahteraan masyarakat Desa Darmasaba, sekaligus menjaga kelestarian lingkungan melalui program-program inovatif dan kolaboratif.
"
- },
- {
- "id": "cmdybb53i0007vngjet38spn8",
+ "id": "cmk3rbc7s000evn9bvb8p0xk1",
"name": "Taman Beji Cengana",
- "deskripsi": "Tirta Klebutan di Pura Taman Beji Cengana di Desa Adat Darmasaba, Badung, selain dipercaya nunas Taksu serta pembersihan diri. Tersemat juga asal usul cerita ditemukannya Tirta Klebutan yang tepat berada di pinggir Tukad Cengana tersebut.",
- "content": "Taman Beji Cengana, terletak di Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali, adalah situs suci yang memiliki nilai spiritual dan sejarah yang tinggi. Tempat ini dikenal sebagai lokasi untuk ritual pembersihan diri (melukat) dan peribadatan oleh umat Hindu Bali. Keberadaan mata air suci (Tirta Klebutan) di Taman Beji Cengana dipercaya memberikan berkah dan penyucian bagi mereka yang datang untuk berdoa dan melakukan ritual.
Potensi Desa melalui Taman Beji Cengana:
Pengembangan Pariwisata Spiritual:
Taman Beji Cengana memiliki potensi besar sebagai destinasi wisata spiritual. Wisatawan yang mencari pengalaman spiritual dan ketenangan batin dapat tertarik untuk mengunjungi tempat ini, mengikuti ritual melukat, dan merasakan suasana sakral yang ditawarkan.
Pelestarian Budaya dan Tradisi:
Dengan mempromosikan Taman Beji Cengana sebagai pusat kegiatan budaya dan ritual tradisional, desa dapat memastikan bahwa warisan budaya dan tradisi lokal tetap lestari.
Pendidikan dan Penelitian:
Taman Beji Cengana dapat dijadikan sebagai pusat pendidikan dan penelitian bagi akademisi, peneliti, dan pelajar yang tertarik mempelajari budaya, agama, dan sejarah Bali.
Pengembangan Ekonomi Kreatif:
Dengan meningkatnya jumlah pengunjung ke Taman Beji Cengana, peluang bagi pengembangan ekonomi kreatif juga terbuka lebar. Masyarakat lokal dapat mengembangkan produk kerajinan tangan, kuliner khas, dan suvenir yang mencerminkan budaya dan tradisi desa.
Konservasi Lingkungan:
Sebagai situs suci dengan mata air alami, Taman Beji Cengana memiliki peran penting dalam konservasi lingkungan. Upaya menjaga kebersihan dan kelestarian mata air serta lingkungan sekitarnya dapat menjadi contoh praktik konservasi yang baik.
Dengan memanfaatkan potensi yang dimiliki Taman Beji Cengana, Desa Darmasaba dapat mengembangkan sektor pariwisata, budaya, pendidikan, ekonomi, dan lingkungan secara berkelanjutan, yang pada gilirannya akan meningkatkan kesejahteraan masyarakat dan pelestarian warisan budaya.
"
+ "deskripsi": "Taman Beji Cengana adalah situs keagamaan dan budaya tradisional Bali
",
+ "content": "Taman Beji Cengana adalah situs keagamaan dan budaya tradisional Bali yang memadukan unsur spiritual, ritual pembersihan (melukat), serta nilai budaya lokal yang berpadu dengan suasana alam yang tenang dan harmonis.
",
+ "kategoriId": "cmk3pmwq10008vn9bqdquv153",
+ "imageName": "yySqpA7ougpXi6cJPskg9-mobile.webp"
},
{
- "id": "cmdybckcz000avngjfzpy60uk",
+ "id": "cmk3q0sez000bvn9b3iyyct5m",
"name": "Gumuh Sari Water Park",
- "deskripsi": "Gumuh Sari Rekreasi atau waterpark, tempat wisata yang asyik dan seru untuk kamu sekeluarga! Tempat liburan di Bali memang seakan nggak ada habisnya. Selalu ada aja destinasi wisata seru yang bisa jadi wishlist. Ada banyak banget tempat wisata yang kamu kunjungi di Bali, mulai dari wisata alam, wisata modern, sampai wisata air.",
- "content": "Gumuh Sari Waterpark, terletak di Jl. Tegal Gumuh No. 9, Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali, adalah destinasi rekreasi yang menawarkan berbagai fasilitas untuk pengunjung dari segala usia. Taman rekreasi ini tidak hanya menyediakan wahana air yang menyenangkan, tetapi juga fasilitas olahraga dan kuliner, menjadikannya tempat ideal untuk rekreasi keluarga dan komunitas.
Potensi Desa melalui Gumuh Sari Waterpark:
Pengembangan Pariwisata Lokal:
Dengan adanya destinasi seperti Gumuh Sari Waterpark, Desa Darmasaba dapat menarik lebih banyak wisatawan lokal maupun mancanegara. Kehadiran pengunjung ini berpotensi meningkatkan pendapatan desa dan membuka peluang usaha baru bagi masyarakat setempat.
Peningkatan Ekonomi Masyarakat:
Fasilitas seperti restoran dan pusat olahraga di dalam kompleks waterpark memberikan peluang bagi warga lokal untuk terlibat dalam sektor jasa dan perdagangan. Hal ini dapat menciptakan lapangan pekerjaan dan mendukung pertumbuhan ekonomi desa.
Pengembangan Fasilitas Olahraga dan Kesehatan:
Dengan adanya pusat futsal dan gym, Gumuh Sari Waterpark mendorong masyarakat untuk berpartisipasi dalam kegiatan olahraga, yang dapat meningkatkan kesehatan dan kesejahteraan warga.
Pemberdayaan Komunitas Melalui Event dan Acara:
Waterpark ini sering menjadi tuan rumah berbagai acara komunitas, seperti pesta busa dan bola, yang dapat mempererat hubungan antarwarga dan menciptakan lingkungan yang harmonis.
Peningkatan Infrastruktur dan Aksesibilitas:
Dengan meningkatnya jumlah pengunjung, infrastruktur desa seperti jalan, transportasi, dan layanan umum lainnya akan berkembang untuk memenuhi kebutuhan tersebut, yang pada gilirannya meningkatkan kualitas hidup masyarakat setempat.
Melalui pengelolaan dan pengembangan yang tepat, Gumuh Sari Waterpark dapat menjadi motor penggerak bagi kemajuan Desa Darmasaba, meningkatkan kesejahteraan masyarakat, dan menjadikan desa ini sebagai destinasi wisata yang dikenal luas.
"
+ "deskripsi": "Gumuh Sari merupakan tempat rekreasi nyaman di kawasan badung yang dapat dinikmati oleh semua umur dengan berbagai fasilitas hiburan, seperti Water Park, Pusat Kebugaran, Lapangan Futsal dan Restaurant.
",
+ "content": "Gumuh Sari merupakan tempat rekreasi nyaman di kawasan badung yang dapat dinikmati oleh semua umur dengan berbagai fasilitas hiburan, seperti Water Park, Pusat Kebugaran, Lapangan Futsal dan Restaurant.Buka setiap hari Senin-Jumat: pukul 10.00-18.00 WITA (Weekday) Sabtu-Minggu: pukul 09.00-18.00 WITA (Weekend)Wahana: - Kolam Renang Anak - Kolam Renang Dewasa - Kolam Renang Olympic - Kids Playground Indoor - Ayunan - ATVFasilitas: - Tempat ibadah - Area parkir - Gazebo - Restoran dan kantin - WiFi gratis - Kamar mandi dan ruang bilas - Ruang lokerKategori Tiket: Senin-Sabtu - Dewasa 35.000 - Anak-anak (3-12 tahun) 20.000 Minggu & Libur Nasional - Dewasa 40.000 - Anak-anak (3-12 tahun) 25.000 Hari Raya - Dewasa 45.000 - Anak-anak (3-12 tahun) 35.000Media Sosial: - Instagram: gumuhsari_rekreasihttps://www.instagram.com/gumuhsari_rekreasi/ - Tiktok: gumuhsari_rekreasihttps://www.tiktok.com/@gumuhsari_rekreasi
",
+ "kategoriId": "cmk3pmwq10008vn9bqdquv153",
+ "imageName": "uki9Zy0DqrEtSg9Lrgyyk-mobile.webp"
},
{
- "id": "cmdyjuij40002vns5qyyjmzf4",
+ "id": "cmk3rsch1000hvn9bhmyhsjqm",
+ "name": "TPS3R Pudak Mesari",
+ "deskripsi": "Sistem pengelolaan sampah terpadu di Desa Darmasaba yang dikenal dengan TPS3R Pudak Mesari.
",
+ "content": "TPS3R Pudak Mesari adalah fasilitas pengelolaan sampah terpadu yang mengoptimalkan pemilahan, pengolahan dan daur ulang sampah masyarakat sehingga mendukung lingkungan yang bersih dan berkelanjutan di Desa Darmasaba.
",
+ "kategoriId": "cmk3pmwq10008vn9bqdquv153",
+ "imageName": "F8SuL9DjCqLXlSwmgSJv_-mobile.webp"
+ },
+ {
+ "id": "cmk3s4kz7000mvn9b8thk98c4",
+ "name": "Bumdes Pudak Mesari",
+ "deskripsi": "Badan Usaha Milik Desa yang bergerak dalam berbagai usaha ekonomi masyarakat Darmasaba.
",
+ "content": "Bumdes Pudak Mesari merupakan badan usaha milik desa yang mengembangkan kegiatan usaha produktif untuk meningkatkan perekonomian masyarakat lokal, termasuk pengelolaan usaha jasa dan komoditas lokal yang dikelola secara partisipatif.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "daDa40Y8dxune0l_cxULD-mobile.webp"
+ },
+ {
+ "id": "cmk3sbgge000pvn9bd3r8i0on",
"name": "Pertanian",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, Bali, memiliki potensi pertanian yang besar sebagai bagian dari warisan agraris yang telah diwariskan secara turun-temurun. Dengan kondisi tanah yang subur serta sistem irigasi tradisional subak, pertanian di Darmasaba memainkan peran penting dalam ekonomi dan keberlanjutan lingkungan desa.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, Bali, memiliki potensi pertanian yang besar sebagai bagian dari warisan agraris yang telah diwariskan secara turun-temurun. Dengan kondisi tanah yang subur serta sistem irigasi tradisional subak, pertanian di Darmasaba memainkan peran penting dalam ekonomi dan keberlanjutan lingkungan desa.
Potensi Desa melalui Pertanian:
Potensi dan Komoditas Unggulan
Pertanian di Desa Darmasaba mengandalkan berbagai jenis tanaman yang memiliki nilai ekonomi tinggi, di antaranya:
- Padi : Sebagai salah satu desa yang masih mempertahankan sistem subak, Darmasaba menjadi bagian dari lumbung pangan di Bali.
- Sayur-mayur : Beberapa jenis sayuran seperti kangkung, bayam, cabai, dan tomat banyak dibudidayakan oleh petani lokal.
- Buah-buahan tropis : Termasuk pisang, mangga, dan kelapa, yang menjadi sumber pendapatan tambahan bagi petani.
- Tanaman obat dan rempah : Seperti jahe, kunyit, dan lengkuas, yang memiliki permintaan tinggi baik untuk kebutuhan rumah tangga maupun industri herbal.
Sistem Irigasi Tradisional Subak:
Sebagai bagian dari warisan budaya Bali, sistem irigasi subak masih diterapkan di Darmasaba. Sistem ini memungkinkan distribusi air yang adil di antara lahan pertanian dan membantu menjaga keberlanjutan produksi pangan desa.
Pengembangan Pertanian Organik:
Dengan meningkatnya kesadaran akan pentingnya produk sehat dan ramah lingkungan, beberapa petani di Darmasaba mulai beralih ke metode pertanian organik. Hal ini membuka peluang bagi desa untuk mengembangkan produk-produk pertanian yang memiliki nilai jual lebih tinggi.
Agrowisata sebagai Sumber Pendapatan Baru:
Potensi pertanian Darmasaba juga dapat dikembangkan menjadi agrowisata, di mana wisatawan dapat merasakan pengalaman langsung dalam bertani, mengikuti workshop bercocok tanam, serta menikmati hasil pertanian segar. Hal ini dapat menarik wisatawan lokal maupun mancanegara, meningkatkan perekonomian desa.
Pemberdayaan Petani dan UMKM Berbasis Pertanian:
Dengan adanya BUMDes Pudak Mesari dan dukungan dari pemerintah setempat, petani di Darmasaba dapat diberikan pelatihan dan akses pasar yang lebih luas. Produk pertanian dapat diolah menjadi berbagai produk turunan seperti keripik pisang, sambal khas desa, hingga minuman herbal yang dapat dipasarkan ke luar daerah.
Dengan berbagai potensi yang dimiliki, pertanian di Desa Darmasaba dapat terus berkembang melalui inovasi dan pemanfaatan teknologi pertanian modern. Dukungan dari masyarakat, pemerintah, dan lembaga terkait sangat penting untuk menjaga keberlanjutan sektor pertanian dan meningkatkan kesejahteraan petani di desa ini.
"
+ "deskripsi": "Sektor pertanian sebagai salah satu potensi utama Desa Darmasaba.
",
+ "content": "Pertanian di Darmasaba mencakup berbagai komoditas yang ditanam oleh masyarakat termasuk padi, tanaman hortikultura dan lain-lain, yang menjadi basis mata pencaharian dan kontribusi terhadap ekonomi desa.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "yms7mL_T6fy--Eze-mTwc-mobile.webp"
},
{
- "id": "cmdykerrq0002vn6fy4sv7uvm",
- "name": "Kawasan Kuliner",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, Bali, memiliki potensi besar dalam sektor kuliner. Sebagai desa yang strategis dan terus berkembang, Darmasaba mulai dikenal sebagai destinasi kuliner yang menawarkan beragam makanan khas Bali hingga makanan modern yang menarik minat wisatawan dan masyarakat lokal.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, Bali, memiliki potensi besar dalam sektor kuliner. Sebagai desa yang strategis dan terus berkembang, Darmasaba mulai dikenal sebagai destinasi kuliner yang menawarkan beragam makanan khas Bali hingga makanan modern yang menarik minat wisatawan dan masyarakat lokal.
Potensi Desa melalui Kawasan Kuliner:
Ragam Kuliner Khas Bali
Darmasaba memiliki banyak warung dan rumah makan yang menyajikan hidangan khas Bali yang otentik, seperti:
- Babi Guling : Salah satu kuliner favorit di Bali yang banyak ditemukan di Darmasaba.
- Ayam Betutu : Hidangan ayam berbumbu khas yang dimasak dengan teknik khas Bali.
- Lawar : Campuran daging dan sayuran berbumbu khas Bali.
- Sate Lilit : Sate khas Bali yang terbuat dari daging cincang yang dibalut pada batang serai.
Wisata Kuliner Modern & Cafe Kekinian:
Selain makanan tradisional, Darmasaba juga mulai berkembang dengan hadirnya cafe dan resto kekinian yang menyajikan menu modern seperti kopi spesial, burger, pizza, dan aneka dessert yang digemari anak muda. Keberadaan tempat-tempat ini menjadikan Darmasaba sebagai pilihan destinasi kuliner bagi wisatawan maupun warga sekitar.
Pasar Kuliner Malam:
Salah satu daya tarik Darmasaba adalah pusat kuliner malam yang menghadirkan aneka makanan kaki lima seperti nasi jinggo, tipat cantok, bakso, dan berbagai jajanan khas Bali. Suasana yang ramai dan harga yang terjangkau membuat pasar kuliner ini menjadi tempat favorit bagi masyarakat lokal.
Potensi Ekonomi & UMKM Kuliner:
Dengan berkembangnya sektor kuliner, banyak pelaku UMKM di Darmasaba mulai merintis usaha makanan, baik dalam bentuk warung makan, katering, hingga produksi makanan ringan seperti keripik, sambal, dan minuman tradisional. Potensi ini dapat terus dikembangkan dengan dukungan pemerintah desa dan promosi melalui media sosial.
Kawasan Kuliner Berbasis Pariwisata:
Untuk menarik lebih banyak pengunjung, Darmasaba berpotensi mengembangkan kawasan kuliner berbasis wisata yang menggabungkan pengalaman makan dengan konsep alam terbuka, pertunjukan seni, dan edukasi kuliner khas Bali. Hal ini dapat menjadi daya tarik tambahan bagi wisatawan yang ingin merasakan pengalaman kuliner yang lebih autentik.
Dengan kekayaan kuliner yang dimiliki, Desa Darmasaba berpotensi menjadi kawasan kuliner unggulan di Kabupaten Badung. Dukungan dari masyarakat, pemerintah desa, serta promosi yang lebih luas dapat menjadikan Darmasaba sebagai destinasi kuliner yang semakin dikenal dan berkembang.
"
+ "id": "cmk3smnj3000svn9b2ww5hesp",
+ "name": "Jogging Track Tegeh Aban, Karang Gadon dan Munduk Uma",
+ "deskripsi": "Jalur jogging dan area rekreasi alam yang dimanfaatkan masyarakat untuk aktivitas sehat dan pariwisata lokal.
",
+ "content": "Jogging Track Tegeh Aban, Karang Gadon dan Munduk Uma adalah area terbuka hijau yang dikembangkan sebagai fasilitas olahraga dan rekreasi, memberikan ruang beraktivitas fisik sambil menikmati suasana alam Desa Darmasaba.
",
+ "kategoriId": "cmk3pmwq10008vn9bqdquv153",
+ "imageName": "AFA1PZtHwKkwmWCC4M7lM-mobile.webp"
},
{
- "id": "cmdykhfhf0005vn6fnz25kify",
- "name": "IKM Berbasis Pengolahan Pangan",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam Industri Kecil dan Menengah (IKM) berbasis pengolahan pangan. Dengan sumber daya alam yang melimpah dan warisan kuliner khas Bali, Darmasaba dapat mengembangkan sektor ini untuk meningkatkan kesejahteraan masyarakat dan menciptakan lapangan kerja baru.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam Industri Kecil dan Menengah (IKM) berbasis pengolahan pangan. Dengan sumber daya alam yang melimpah dan warisan kuliner khas Bali, Darmasaba dapat mengembangkan sektor ini untuk meningkatkan kesejahteraan masyarakat dan menciptakan lapangan kerja baru.
Potensi dan Peran IKM Berbasis Pengolahan Pangan:
Produk Unggulan Pengolahan Pangan
Beberapa produk olahan pangan yang potensial dikembangkan di Darmasaba meliputi:
- Keripik dan Snack Tradisional : Seperti keripik pisang, keripik singkong, dan rengginang.
- Sambal Khas Bali : Seperti sambal matah dan sambal embe yang banyak diminati pasar lokal dan nasional.
- Minuman Herbal dan Jamu : Berbasis rempah seperti kunyit asam, beras kencur, dan wedang jahe.
- Olahan Makanan Berbasis Kelapa : Seperti virgin coconut oil (VCO), serundeng, dan gula aren.
- Kue Tradisional Bali : Seperti jaje laklak, jaje uli, dan klepon yang dapat dikemas secara modern.
Peluang Ekonomi dan Pemberdayaan UMKM:
IKM berbasis pengolahan pangan dapat membuka peluang bagi masyarakat, terutama ibu rumah tangga dan pemuda desa, untuk berwirausaha. Dengan dukungan modal dan pelatihan dari pemerintah desa atau BUMDes Pudak Mesari, usaha kecil ini dapat berkembang menjadi industri yang lebih besar.
Digitalisasi dan Pemasaran Online:
Darmasaba dapat mengembangkan kawasan sentra IKM sebagai pusat produksi, pelatihan, dan pemasaran produk olahan pangan. Dengan adanya fasilitas ini, para pelaku usaha dapat lebih mudah berkolaborasi, meningkatkan kualitas produk, serta mendapatkan akses ke permodalan dan distribusi yang lebih luas.
Pengembangan Kawasan Sentra IKM:
Dengan berkembangnya sektor kuliner, banyak pelaku UMKM di Darmasaba mulai merintis usaha makanan, baik dalam bentuk warung makan, katering, hingga produksi makanan ringan seperti keripik, sambal, dan minuman tradisional. Potensi ini dapat terus dikembangkan dengan dukungan pemerintah desa dan promosi melalui media sosial.
Sinergi dengan Pariwisata dan Agrowisata:
Dengan berkembangnya sektor wisata di Darmasaba, produk olahan pangan dapat dijadikan suvenir khas desa. Pengunjung dapat membeli oleh-oleh seperti sambal kemasan, jajanan khas, atau minuman herbal sebagai bagian dari pengalaman wisata mereka.
IKM berbasis pengolahan pangan memiliki potensi besar untuk menjadi sektor unggulan di Desa Darmasaba. Dengan inovasi, dukungan teknologi, serta pemasaran yang baik, produk-produk lokal dapat bersaing di pasar yang lebih luas, meningkatkan kesejahteraan masyarakat, dan menjadikan Darmasaba sebagai pusat industri pangan kreatif di Kabupaten Badung.
"
- },
- {
- "id": "cmdykjgmv0008vn6fwg0rr2nh",
- "name": "Genteng",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam industri genteng yang dikelola oleh Usaha Mikro, Kecil, dan Menengah (UMKM). Sebagai desa yang masih mempertahankan nilai-nilai tradisional dalam pembangunan, industri genteng di Darmasaba berperan penting dalam penyediaan bahan bangunan berkualitas bagi masyarakat lokal maupun luar daerah.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam industri genteng yang dikelola oleh Usaha Mikro, Kecil, dan Menengah (UMKM). Sebagai desa yang masih mempertahankan nilai-nilai tradisional dalam pembangunan, industri genteng di Darmasaba berperan penting dalam penyediaan bahan bangunan berkualitas bagi masyarakat lokal maupun luar daerah.
Potensi dan Peran UMKM Genteng:
Genteng Tradisional Berkualitas Tinggi
UMKM di Darmasaba memproduksi genteng dari bahan baku pilihan seperti tanah liat berkualitas, yang menghasilkan genteng dengan daya tahan tinggi, kuat, dan cocok untuk iklim tropis. Beberapa jenis genteng yang dihasilkan meliputi:
- Genteng Tanah Liat : Kuat, tahan lama, dan memiliki estetika khas tradisional.
- Genteng Beton : Cocok untuk bangunan modern dengan ketahanan lebih tinggi.
- Genteng Keramik : Memberikan tampilan elegan dan daya serap air yang lebih rendah.
Peluang Ekonomi dan Pemberdayaan Masyarakat:
Industri genteng di Darmasaba memberikan peluang kerja bagi masyarakat setempat, terutama dalam bidang produksi, distribusi, hingga pemasaran. UMKM genteng juga mendukung keberlanjutan ekonomi desa dengan meningkatkan pendapatan warga serta mengurangi angka pengangguran.
Inovasi dan Pengembangan Teknologi
Beberapa pengrajin genteng di Darmasaba telah mulai mengadopsi teknologi modern dalam proses produksi, seperti:
- Penggunaan cetakan dan oven pembakaran efisien untuk meningkatkan kualitas dan kapasitas produksi.
- Teknik pelapisan anti bocor dan anti lumut untuk membuat genteng lebih tahan lama.
- Desain genteng inovatif yang lebih ringan dan mudah dipasang.
Pemasaran dan Ekspansi Pasar
Dengan meningkatnya pembangunan perumahan dan proyek konstruksi di Bali, permintaan akan genteng berkualitas terus bertambah. UMKM genteng Darmasaba dapat memperluas pasarnya dengan:
- Menjalin kerja sama dengan kontraktor dan pengembang properti.
- Mempromosikan produk melalui media sosial dan marketplace online.
- Menyediakan layanan custom sesuai kebutuhan pelanggan.
Keberlanjutan dan Ramah Lingkungan:
Industri genteng di Darmasaba berpotensi dikembangkan secara lebih ramah lingkungan dengan menerapkan metode produksi yang mengurangi limbah dan emisi. Pemanfaatan energi alternatif serta daur ulang bahan limbah dapat membantu menciptakan industri yang lebih berkelanjutan.
UMKM genteng di Desa Darmasaba memiliki potensi besar untuk terus berkembang sebagai sektor industri unggulan. Dengan inovasi, pemasaran yang lebih luas, serta dukungan dari pemerintah dan masyarakat, industri ini dapat meningkatkan kesejahteraan warga dan memperkuat perekonomian desa.
"
- },
- {
- "id": "cmdyklax3000bvn6fdu53f3xq",
- "name": "Peternakan Ikan Lele",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam sektor peternakan lele. Dengan kondisi lingkungan yang mendukung serta meningkatnya permintaan ikan lele di pasaran, budidaya ikan lele dapat menjadi salah satu sektor ekonomi unggulan yang mampu meningkatkan kesejahteraan masyarakat desa.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam sektor peternakan lele. Dengan kondisi lingkungan yang mendukung serta meningkatnya permintaan ikan lele di pasaran, budidaya ikan lele dapat menjadi salah satu sektor ekonomi unggulan yang mampu meningkatkan kesejahteraan masyarakat desa.
Potensi dan Peran Peternakan Ikan Lele:
Kondisi Lingkungan yang Mendukung
Darmasaba memiliki sumber air yang cukup serta iklim yang cocok untuk budidaya ikan lele. Kolam-kolam budidaya dapat dibuat dengan berbagai sistem, seperti:
- Kolam Terpal : Mudah dibuat dan lebih efisien dalam perawatan.
- Kolam Beton : Lebih tahan lama dan cocok untuk produksi skala besar.
- Sistem Bioflok : Teknologi modern yang dapat meningkatkan kepadatan ikan dan mengurangi limbah.
Permintaan Pasar yang Tinggi:
Lele merupakan salah satu jenis ikan yang memiliki permintaan tinggi di Bali, baik untuk konsumsi rumah tangga, warung makan, hingga restoran. Produk olahan seperti lele goreng, pecel lele, dan abon lele semakin diminati, membuka peluang besar bagi peternak lele di Darmasaba.
Inovasi dan Pengembangan Teknologi
Beberapa pengrajin genteng di Darmasaba telah mulai mengadopsi teknologi modern dalam proses produksi, seperti:
- Penggunaan cetakan dan oven pembakaran efisien untuk meningkatkan kualitas dan kapasitas produksi.
- Teknik pelapisan anti bocor dan anti lumut untuk membuat genteng lebih tahan lama.
- Desain genteng inovatif yang lebih ringan dan mudah dipasang.
Pemasaran dan Ekspansi Pasar
Dengan meningkatnya pembangunan perumahan dan proyek konstruksi di Bali, permintaan akan genteng berkualitas terus bertambah. UMKM genteng Darmasaba dapat memperluas pasarnya dengan:
- Menjalin kerja sama dengan kontraktor dan pengembang properti.
- Mempromosikan produk melalui media sosial dan marketplace online.
- Menyediakan layanan custom sesuai kebutuhan pelanggan.
Keberlanjutan dan Ramah Lingkungan:
Industri genteng di Darmasaba berpotensi dikembangkan secara lebih ramah lingkungan dengan menerapkan metode produksi yang mengurangi limbah dan emisi. Pemanfaatan energi alternatif serta daur ulang bahan limbah dapat membantu menciptakan industri yang lebih berkelanjutan.
UMKM genteng di Desa Darmasaba memiliki potensi besar untuk terus berkembang sebagai sektor industri unggulan. Dengan inovasi, pemasaran yang lebih luas, serta dukungan dari pemerintah dan masyarakat, industri ini dapat meningkatkan kesejahteraan warga dan memperkuat perekonomian desa.
"
- },
- {
- "id": "cmdykpkwf000gvn6ftas2cjje",
- "name": "Jogging Track Tegeh Aban, Karang Gadon dan Munduk Uma Desa",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi wisata olahraga dan rekreasi melalui Jogging Track Tegeh Aban, Karang Gadon, dan Munduk Uma Desa. Jalur jogging ini tidak hanya menjadi fasilitas olahraga bagi warga, tetapi juga berpotensi dikembangkan sebagai destinasi wisata sehat berbasis alam yang menarik bagi wisatawan lokal maupun luar daerah.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi wisata olahraga dan rekreasi melalui Jogging Track Tegeh Aban, Karang Gadon, dan Munduk Uma Desa. Jalur jogging ini tidak hanya menjadi fasilitas olahraga bagi warga, tetapi juga berpotensi dikembangkan sebagai destinasi wisata sehat berbasis alam yang menarik bagi wisatawan lokal maupun luar daerah.
Potensi dan Peran Jogging Track Tegeh Aban, Karang Gadon dan Munduk Uma Desa:
Keindahan Alam dan Udara Segar:
Jogging track yang membentang di Tegeh Aban, Karang Gadon, dan Munduk Uma Desa menawarkan pemandangan alam yang asri dengan udara segar khas pedesaan. Jalur ini melewati area persawahan hijau, perkebunan, serta hutan kecil yang memberikan pengalaman jogging yang lebih menyenangkan dan menenangkan.
Fasilitas Olahraga dan Rekreasi
Selain untuk jogging, jalur ini juga cocok digunakan untuk:
- Bersepeda santai : Jalur yang nyaman untuk pecinta sepeda.
- Trekking ringan : Cocok bagi wisatawan yang ingin menikmati suasana pedesaan.
- Meditasi dan Yoga : Area yang tenang dan alami, ideal untuk relaksasi.
Destinasi Wisata Sehat dan Edukasi:
Jogging track ini berpotensi dikembangkan sebagai wisata sehat berbasis alam, di mana pengunjung bisa menikmati udara segar sambil berolahraga. Selain itu, jalur ini dapat dijadikan sebagai rute edukasi lingkungan, mengenalkan keanekaragaman hayati, pertanian, serta kehidupan masyarakat desa.
Potensi Ekonomi bagi Masyarakat
Dengan meningkatnya jumlah pengunjung, masyarakat sekitar dapat memanfaatkan peluang usaha seperti:
- Warung sehat dan kuliner lokal : Menyediakan makanan dan minuman sehat bagi para pengunjung.
- Jasa penyewaan sepeda : Menarik bagi wisatawan yang ingin berkeliling lebih jauh.
- Pemandu wisata lokal : Memberikan pengalaman lebih bagi wisatawan yang ingin mengenal sejarah dan budaya desa.
Pengembangan Berkelanjutan:
Agar semakin menarik, jogging track ini bisa dilengkapi dengan fasilitas tambahan seperti tempat istirahat, spot foto alami, papan informasi tentang flora dan fauna, serta area taman bunga untuk mempercantik jalur jogging.
Jogging Track Tegeh Aban, Karang Gadon, dan Munduk Uma Desa memiliki potensi besar sebagai destinasi wisata sehat dan olahraga berbasis alam. Dengan pengelolaan yang baik serta dukungan dari pemerintah desa dan masyarakat, jalur ini bisa menjadi ikon baru Desa Darmasaba yang menarik bagi wisatawan serta meningkatkan perekonomian warga setempat.
"
- },
- {
- "id": "cmdykr76v000jvn6fqngibbmq",
+ "id": "cmk3t8hgs000vvn9bna6pnw8r",
"name": "Dam Tanah Putih",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi wisata olahraga dan rekreasi melalui Jogging Track Tegeh Aban, Karang Gadon, dan Munduk Uma Desa. Jalur jogging ini tidak hanya menjadi fasilitas olahraga bagi warga, tetapi juga berpotensi dikembangkan sebagai destinasi wisata sehat berbasis alam yang menarik bagi wisatawan lokal maupun luar daerah.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki Dam Tanah Putih sebagai salah satu potensi desa yang bernilai strategis. Selain berfungsi sebagai infrastruktur pengairan, dam ini juga memiliki potensi untuk dikembangkan sebagai destinasi wisata alam, edukasi, dan rekreasi bagi masyarakat lokal maupun wisatawan.
Potensi dan Peran Dam Tanah Putih:
Fungsi Utama Sebagai Sumber Pengairan
Dam Tanah Putih memiliki peran penting dalam sistem irigasi yang menopang sektor pertanian di Darmasaba. Air dari dam ini digunakan untuk:
- Mengairi sawah dan ladang : Menjamin ketersediaan air bagi petani sepanjang tahun.
- Menjaga keseimbangan ekosistem : Menjadi habitat bagi ikan air tawar dan berbagai biota air.
- Menampung air hujan : Membantu mengurangi risiko banjir dan kekeringan.
Potensi Wisata Alam dan Rekreasi
Dengan pemandangan alam yang asri dan suasana yang sejuk, Dam Tanah Putih memiliki potensi besar untuk dikembangkan sebagai tempat wisata alam. Beberapa kegiatan yang bisa dikembangkan di area ini antara lain:
- Trekking dan jogging di sekitar dam : Menikmati udara segar dan pemandangan indah.
- Berkemah dan piknik : Cocok untuk keluarga dan komunitas yang ingin menikmati alam.
- Wisata air : Seperti pemancingan atau wisata perahu kecil yang dapat menarik wisatawan.
- Spot fotografi alam : Keindahan dam dan sekitarnya menjadi latar yang menarik bagi para fotografer.
Potensi Ekonomi dan UMKM Lokal
Dengan pengembangan dam sebagai destinasi wisata, masyarakat sekitar dapat memanfaatkan peluang usaha seperti:
- Warung makan dan jajanan tradisional : Menyediakan makanan khas Bali bagi wisatawan.
- Jasa penyewaan alat rekreasi : Seperti pancing atau perahu kecil.
- Produk kerajinan tangan dan suvenir : Oleh-oleh khas Darmasaba yang menarik bagi pengunjung.
Pengembangan Konservasi dan Edukasi Lingkungan
Dam Tanah Putih juga bisa menjadi tempat edukasi lingkungan dengan konsep konservasi, di mana pengunjung bisa belajar tentang:
- Pengelolaan sumber daya air yang berkelanjutan.
- Keanekaragaman hayati di sekitar dam.
- Pentingnya ekosistem perairan bagi pertanian dan kehidupan masyarakat.
Dengan berbagai fungsi dan keindahannya, Dam Tanah Putih memiliki potensi besar untuk dikembangkan sebagai destinasi wisata alam, rekreasi, serta edukasi lingkungan. Dengan pengelolaan yang baik dan dukungan dari masyarakat serta pemerintah desa, dam ini dapat menjadi aset penting bagi Darmasaba, baik dari sisi ekonomi maupun kelestarian lingkungan.
"
+ "deskripsi": "Bendungan lokal yang berfungsi sebagai sumber irigasi pertanian.
",
+ "content": "Dam Tanah Putih merupakan bendungan yang mendukung pengairan sawah dan lahan pertanian di Desa Darmasaba, yang membantu peningkatan produktivitas pertanian masyarakat setempat.
",
+ "kategoriId": "cmk3pmwq10008vn9bqdquv153",
+ "imageName": "84pnYeV2_9f4xplh9RBXW-mobile.webp"
},
{
- "id": "cmdyku9qh000mvn6ft76322sv",
+ "id": "cmk3tlrkb000yvn9bo1vse4c5",
"name": "UMKM",
- "deskripsi": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam sektor Usaha Mikro, Kecil, dan Menengah (UMKM). Keberadaan UMKM di desa ini tidak hanya menjadi motor penggerak ekonomi lokal, tetapi juga mendukung pelestarian budaya dan kearifan lokal melalui berbagai produk unggulan.",
- "content": "Desa Darmasaba, yang terletak di Kecamatan Abiansemal, Kabupaten Badung, memiliki potensi besar dalam sektor Usaha Mikro, Kecil, dan Menengah (UMKM). Keberadaan UMKM di desa ini tidak hanya menjadi motor penggerak ekonomi lokal, tetapi juga mendukung pelestarian budaya dan kearifan lokal melalui berbagai produk unggulan.
Potensi dan Peran UMKM:
Kerajinan Tangan dan Produk Lokal
Darmasaba memiliki banyak pengrajin yang menghasilkan produk unik dengan nilai seni tinggi, seperti:
- Genteng dan bahan bangunan tradisional : Genteng khas Darmasaba yang berkualitas tinggi.
- Kerajinan anyaman dan ukiran : Produk berbasis rotan dan kayu yang banyak diminati pasar lokal dan internasional.
- Pakaian adat dan kain tradisional : Mendukung pelestarian budaya Bali.
Industri Kuliner Khas Darmasaba
Kuliner khas desa ini memiliki potensi besar untuk dikembangkan sebagai bisnis UMKM, seperti:
- Babi Guling : Salah satu kuliner favorit yang banyak diminati wisatawan.
- Jajanan tradisional Bali : Seperti laklak, jaja uli, dan klepon yang masih dibuat dengan cara tradisional.
- Olahan ikan lele : Seperti abon lele, lele asap, dan pecel lele yang memiliki pasar luas.
UMKM Berbasis Pengolahan Pangan
Beberapa UMKM di Darmasaba mengolah hasil pertanian dan peternakan menjadi produk bernilai tambah, seperti:
- Keripik singkong dan pisang : Camilan sehat berbasis bahan lokal.
- Olahan kelapa : Seperti minyak kelapa murni dan gula aren.
- Produk herbal dan jamu : Menggunakan bahan-bahan alami dari tanaman lokal.
Dukungan dan Pengembangan UMKM
Agar UMKM di Darmasaba semakin berkembang, perlu adanya:
- Pelatihan dan pendampingan usaha : Untuk meningkatkan kualitas produk dan manajemen usaha.
- Pemasaran digital : Menggunakan media sosial dan e-commerce untuk menjangkau pasar lebih luas.
- Kerja sama dengan BUMDes Pudak Mesari : Untuk membantu akses modal dan pengelolaan bisnis yang lebih profesional.
UMKM di Desa Darmasaba memiliki potensi besar dalam berbagai sektor, mulai dari kerajinan tangan, kuliner, hingga wisata berbasis masyarakat. Dengan inovasi, pemasaran yang lebih luas, dan dukungan dari pemerintah desa serta masyarakat, UMKM Darmasaba dapat berkembang pesat dan menjadi tulang punggung perekonomian desa.
"
+ "deskripsi": "Usaha mikro, kecil, dan menengah masyarakat Desa Darmasaba.
",
+ "content": "UMKM di Darmasaba mencakup berbagai usaha kecil produktif seperti kuliner, kerajinan tangan, dan jasa lainnya, yang menjadi pilar kegiatan ekonomi dan pemberdayaan masyarakat lokal.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "5VSeupjyjtam02PNJW3ZG-mobile.webp"
+ },
+ {
+ "id": "cmk3tzpku0011vn9beq2kzyen",
+ "name": "Kawasan Kuliner",
+ "deskripsi": "Wilayah yang menyediakan berbagai pilihan kuliner khas lokal Desa Darmasaba.
",
+ "content": "Kawasan kuliner di Darmasaba menjadi tempat berkumpul dan menikmati makanan khas lokal, mendukung kegiatan ekonomi UMKM kuliner desa serta daya tarik wisata kuliner.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "gjwx1dcHKMJEICbLF6g88-mobile.webp"
+ },
+ {
+ "id": "cmk3u7gzh0014vn9brvlz3fjp",
+ "name": "IKM berbasis Pengolahan Pangan",
+ "deskripsi": "Industri kecil menengah yang fokus pada pengolahan produk pangan lokal.
",
+ "content": "IKM berbasis pengolahan pangan di desa mencakup produksi makanan olahan yang dihasilkan oleh masyarakat setempat, memperkaya nilai tambah hasil pertanian lokal dan menciptakan peluang bisnis.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "j3q1NLvwPFrxAmgMXBmq2-mobile.webp"
+ },
+ {
+ "id": "cmk3udr960017vn9bq33ce6mw",
+ "name": "Peternakan Ikan Lele",
+ "deskripsi": "Usaha budidaya ikan lele sebagai potensi perikanan desa.
",
+ "content": "Peternakan ikan lele menjadi salah satu bentuk usaha budidaya perikanan di Darmasaba, memberikan sumber pendapatan tambahan bagi petani ikan dan diversifikasi komoditas desa.
",
+ "kategoriId": "cmk3s1ks6000ivn9bcrv960ko",
+ "imageName": "eZCq49lXnx6zKslb6hLxz-mobile.webp"
}
]
diff --git a/prisma/data/desa/profile/profil_perbekel.json b/prisma/data/desa/profile/profil_perbekel.json
index efaf0021..2ce61afe 100644
--- a/prisma/data/desa/profile/profil_perbekel.json
+++ b/prisma/data/desa/profile/profil_perbekel.json
@@ -1,9 +1,10 @@
[
- {
- "id": "edit",
- "biodata": "I.B Surya Prabhawa Manuaba, S.H., M.H., adalah Perbekel Darmasaba periode 2021-2027, seorang advokat, pendiri Mantra Legal Consultants & Advocates, serta aktif di bidang musik dan akademis. Dia menempuh pendidikan hukum di Universitas Udayana dan Universitas Mahasaraswati Denpasar serta memiliki pengalaman luas di berbagai organisasi dan kepemimpinan.
",
- "pengalaman": "2021 - 2027: Perbekel Desa Darmasaba 2015 - Sekarang: Founder & Managing Director Mantra Legal Consultants & Advocates 2020 - Sekarang: Founder Ugawa Record Music Studio 2010 - 2016: Dosen Fakultas Hukum Universitas Mahasaraswati Denpasar ",
- "pengalamanOrganisasi": " 1996 – 1997: Ketua OSIS SMP Negeri 1 Abiansemal 1999 – 2000: Ketua OSIS SMA Negeri 1 Mengwi 2008 – 2009: Ketua BEM Universitas Mahasaraswati Denpasar 2008 – 2010: Ketua Sekaa Taruna Sila Dharma, Banjar Tengah, Desa Adat Tegal, Darmasaba 2020 – Sekarang: Pengurus Young Lawyer Committee Peradi Denpasar 2021 – Sekarang: Dewan Kehormatan Himpunan Pengusaha Muda Indonesia (HIPMI) Badung 2023 – 2028: Komite Tetap Advokasi – Bidang Hukum dan Regulasi Kamar Dagang dan Industri Badung ",
- "programUnggulan": "Pemberdayaan Ekonomi dan UMKM Pelatihan dan pendampingan UMKM lokal Program bantuan modal usaha bagi pelaku usaha kecil Digitalisasi UMKM untuk meningkatkan pemasaran produk lokal "
- }
-]
\ No newline at end of file
+ {
+ "id": "edit",
+ "biodata": "I.B Surya Prabhawa Manuaba, S.H., M.H., adalah Perbekel Darmasaba periode 2021-2027, seorang advokat, pendiri Mantra Legal Consultants & Advocates, serta aktif di bidang musik dan akademis. Dia menempuh pendidikan hukum di Universitas Udayana dan Universitas Mahasaraswati Denpasar serta memiliki pengalaman luas di berbagai organisasi dan kepemimpinan.
",
+ "pengalaman": "2021 - 2027: Perbekel Desa Darmasaba 2015 - Sekarang: Founder & Managing Director Mantra Legal Consultants & Advocates 2020 - Sekarang: Founder Ugawa Record Music Studio 2010 - 2016: Dosen Fakultas Hukum Universitas Mahasaraswati Denpasar ",
+ "pengalamanOrganisasi": " 1996 – 1997: Ketua OSIS SMP Negeri 1 Abiansemal 1999 – 2000: Ketua OSIS SMA Negeri 1 Mengwi 2008 – 2009: Ketua BEM Universitas Mahasaraswati Denpasar 2008 – 2010: Ketua Sekaa Taruna Sila Dharma, Banjar Tengah, Desa Adat Tegal, Darmasaba 2020 – Sekarang: Pengurus Young Lawyer Committee Peradi Denpasar 2021 – Sekarang: Dewan Kehormatan Himpunan Pengusaha Muda Indonesia (HIPMI) Badung 2023 – 2028: Komite Tetap Advokasi – Bidang Hukum dan Regulasi Kamar Dagang dan Industri Badung ",
+ "programUnggulan": "Pemberdayaan Ekonomi dan UMKM Pelatihan dan pendampingan UMKM lokal Program bantuan modal usaha bagi pelaku usaha kecil Digitalisasi UMKM untuk meningkatkan pemasaran produk lokal ",
+ "imageName": "perbekel-profile-desa.webp"
+ }
+]
diff --git a/prisma/data/desa/profile/profile-perbekel-lalu.json b/prisma/data/desa/profile/profile-perbekel-lalu.json
new file mode 100644
index 00000000..22f49658
--- /dev/null
+++ b/prisma/data/desa/profile/profile-perbekel-lalu.json
@@ -0,0 +1,65 @@
+[
+ {
+ "id": "cmk4yswiy0003vnwq97wem6t2",
+ "nama": "Si Gede Kania",
+ "periode": "Tahun 1943 - 1946",
+ "imageName": "120DDmtAkWEOlu_z56EVj-mobile.webp",
+ "daerah": "Perbekel Tegal"
+ },
+ {
+ "id": "cmk4yvuxg0006vnwqa0nk7jcx",
+ "nama": "Si Gede Gandem",
+ "periode": "Tahun 1946 - 1950",
+ "imageName": "bJEAL2K_DcR9AhabklcdP-mobile.webp",
+ "daerah": "Perbekel Tegal"
+ },
+ {
+ "id": "cmk4ywml70009vnwqnnrp0cpq",
+ "nama": "I Wayan Sama",
+ "periode": "Tahun 1950 - 1953",
+ "imageName": "szxGnXEFqa0k0nvZ-ALVG-mobile.webp",
+ "daerah": "Perbekel Tegal"
+ },
+ {
+ "id": "cmk4yzix3000cvnwqn2pa57gs",
+ "nama": "I Wayan Nambreg",
+ "periode": "Tahun 1950 - 1960",
+ "imageName": "ectWSAf6wfzIPzKMf3wzo-mobile.webp",
+ "daerah": "Perbekel Darmasaba"
+ },
+ {
+ "id": "cmk4z0sgd000fvnwqlgjasqx9",
+ "nama": "Ida Bagus Putu Oka",
+ "periode": "Tahun 1953 - 1974",
+ "imageName": "VwjN6P8HyaKpNngtLCt6z-mobile.webp",
+ "daerah": "Perbekel Tegal / Darmasaba"
+ },
+ {
+ "id": "cmk4z1w8m000ivnwqe68w5vjg",
+ "nama": "I Nyoman Patra",
+ "periode": "Tahun 1974 - 1991",
+ "imageName": "f3tdB74WjhVVJXmAE7Sev-mobile.webp",
+ "daerah": "Perbekel Darmasaba"
+ },
+ {
+ "id": "cmk4z2y3u000lvnwq7fbr3paz",
+ "nama": "I Made Rudja",
+ "periode": "Tahun 1991 - 2007",
+ "imageName": "X3FbO9ns3We5GM1uAwds1-mobile.webp",
+ "daerah": "Perbekel Darmasaba"
+ },
+ {
+ "id": "cmk4z6esm000ovnwqacek3jlx",
+ "nama": "I Wayan Kaler, S.H., M.H.",
+ "periode": "Tahun 2007 - 2013",
+ "imageName": "AO-mB9JXm0ahmNnSAEDJ3-mobile.webp",
+ "daerah": "Perbekel Darmasaba"
+ },
+ {
+ "id": "cmk4z7e77000rvnwqo2wkmbwl",
+ "nama": "I Made Taram, S.H.",
+ "periode": "Tahun 2013 - 2019",
+ "imageName": "ropKaJjSbK_XAyxd8mEYt-mobile.webp",
+ "daerah": "Perbekel Darmasaba"
+ }
+]
diff --git a/prisma/data/desa/profile/profileDesaImage.json b/prisma/data/desa/profile/profileDesaImage.json
new file mode 100644
index 00000000..678198a6
--- /dev/null
+++ b/prisma/data/desa/profile/profileDesaImage.json
@@ -0,0 +1,26 @@
+[
+ {
+ "id": "cmkz5lumu0008vn5gso2vrejt",
+ "label": "Bunga Pudak",
+ "imageName": "fpovTIXnnsX5duqLeoSx6-mobile.webp",
+ "maskotDesaId": "edit"
+ },
+ {
+ "id": "cmkz5lumu0009vn5gcn3ah0ku",
+ "label": "Pohon Pudak",
+ "imageName": "I01xTmkCZo3u8-y0w07TG-mobile.webp",
+ "maskotDesaId": "edit"
+ },
+ {
+ "id": "cmkz5lumu000avn5grvxbmaux",
+ "label": "Tari Sekar Pudak",
+ "imageName": "MlTQffVnQ-itl-vgO0BGM-mobile.webp",
+ "maskotDesaId": "edit"
+ },
+ {
+ "id": "cmkz5lumu000bvn5gwujxhp2r",
+ "label": "Klimaks Tari Pudak",
+ "imageName": "PdIqEFHq5Q12PEi0NRUJ3-mobile.webp",
+ "maskotDesaId": "edit"
+ }
+]
diff --git a/prisma/data/ekonomi/demografi-pekerjaan/demografi-pekerjaan.json b/prisma/data/ekonomi/demografi-pekerjaan/demografi-pekerjaan.json
new file mode 100644
index 00000000..344868c1
--- /dev/null
+++ b/prisma/data/ekonomi/demografi-pekerjaan/demografi-pekerjaan.json
@@ -0,0 +1,56 @@
+[
+ {
+ "id": "cmkf3kv0b0004vnys6hh7ugj2",
+ "pekerjaan": "Petani/Pekebun",
+ "lakiLaki": 180,
+ "perempuan": 120
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh8vhk3",
+ "pekerjaan": "Perajin Industri",
+ "lakiLaki": 95,
+ "perempuan": 140
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh9wil4",
+ "pekerjaan": "Pedagang/UMKM",
+ "lakiLaki": 130,
+ "perempuan": 170
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh0xjm5",
+ "pekerjaan": "Karyawan Swasta",
+ "lakiLaki": 260,
+ "perempuan": 310
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh1ykn6",
+ "pekerjaan": "PNS/TNI/Polri",
+ "lakiLaki": 85,
+ "perempuan": 75
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh2zlo7",
+ "pekerjaan": "Buruh Harian Lepas",
+ "lakiLaki": 140,
+ "perempuan": 90
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh3amp8",
+ "pekerjaan": "Wiraswasta",
+ "lakiLaki": 165,
+ "perempuan": 110
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh4bnq9",
+ "pekerjaan": "Pelajar/Mahasiswa",
+ "lakiLaki": 220,
+ "perempuan": 240
+ },
+ {
+ "id": "cmkf3kv0b0004vnys6hh5cor0",
+ "pekerjaan": "Belum/Tidak Bekerja",
+ "lakiLaki": 70,
+ "perempuan": 95
+ }
+]
diff --git a/prisma/data/ekonomi/jumlah-penduduk-miskin/jumlah-penduduk-miskin.json b/prisma/data/ekonomi/jumlah-penduduk-miskin/jumlah-penduduk-miskin.json
new file mode 100644
index 00000000..3bb2b1e5
--- /dev/null
+++ b/prisma/data/ekonomi/jumlah-penduduk-miskin/jumlah-penduduk-miskin.json
@@ -0,0 +1,27 @@
+[
+ {
+ "id": "795b3e69-2bdb-4d2b-8331-9433fc604723",
+ "year": 2021,
+ "totalPoorPopulation": 480
+ },
+ {
+ "id": "0e61ed28-1789-4329-86bc-67bdb715d4fa",
+ "year": 2022,
+ "totalPoorPopulation": 450
+ },
+ {
+ "id": "0f80e31e-56d7-42aa-aede-e3104f1bca4f",
+ "year": 2023,
+ "totalPoorPopulation": 430
+ },
+ {
+ "id": "3668ecd4-4c4c-4a87-a39a-72661fce5a2c",
+ "year": 2024,
+ "totalPoorPopulation": 410
+ },
+ {
+ "id": "854a375e-4662-4e4c-abd8-59115b4910a9",
+ "year": 2025,
+ "totalPoorPopulation": 390
+ }
+]
diff --git a/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-pendidikan.json b/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-pendidikan.json
new file mode 100644
index 00000000..8ca5ab9d
--- /dev/null
+++ b/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-pendidikan.json
@@ -0,0 +1,10 @@
+[
+ {
+ "id": "cmkko983k000104l2ff4i8awk",
+ "SD": "35",
+ "SMP": "28",
+ "SMA": "20",
+ "D3": "8",
+ "S1": "15"
+ }
+]
diff --git a/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-usia.json b/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-usia.json
new file mode 100644
index 00000000..3f4c4873
--- /dev/null
+++ b/prisma/data/ekonomi/jumlah-penduduk-usia-kerja-yang-menganggur/pengangguran-berdasarkan-usia.json
@@ -0,0 +1,9 @@
+[
+ {
+ "id": "cmkko8wam000004l2h99rekid",
+ "usia18_25": "25",
+ "usia26_35": "18",
+ "usia36_45": "12",
+ "usia46_keatas": "10"
+ }
+]
diff --git a/prisma/data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json b/prisma/data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json
index f64b6977..3f5da844 100644
--- a/prisma/data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json
+++ b/prisma/data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json
@@ -94,6 +94,24 @@
"educatedUnemployment": 78,
"uneducatedUnemployment": 54,
"percentageChange": -2.22
+ },
+ {
+ "id": "13f11e4f-c96e-4155-bf30-3f45076ee34c",
+ "month": "Jan",
+ "year": 2026,
+ "totalUnemployment": 350,
+ "educatedUnemployment": 200,
+ "uneducatedUnemployment": 150,
+ "percentageChange": 0.0
+ },
+ {
+ "id": "477b3b0c-af18-4816-a00b-dcac530d57cd",
+ "month": "Feb",
+ "year": 2026,
+ "totalUnemployment": 270,
+ "educatedUnemployment": 150,
+ "uneducatedUnemployment": 120,
+ "percentageChange": -22.9
}
]
\ No newline at end of file
diff --git a/prisma/data/ekonomi/lowongan-kerja-lokal/lowongan-kerja-lokal.json b/prisma/data/ekonomi/lowongan-kerja-lokal/lowongan-kerja-lokal.json
new file mode 100644
index 00000000..4747aaa9
--- /dev/null
+++ b/prisma/data/ekonomi/lowongan-kerja-lokal/lowongan-kerja-lokal.json
@@ -0,0 +1,57 @@
+[
+ {
+ "id": "3b3e817b-c136-4488-ac79-9a7d408939a2",
+ "posisi": "Lowongan TPS3R Pudak Mesari",
+ "namaPerusahaan": "TPS3R Pudak Mesari Desa Darmasaba",
+ "lokasi": "Desa Darmasaba, Abiansemal, Kabupaten Badung, Bali",
+ "tipePekerjaan": "Freelance",
+ "gaji": "1.500.000",
+ "deskripsi": "Menjalankan tugas di TPS3R Pudak Mesari.",
+ "kualifikasi": "Usia 18-30 tahun, SMA/SMK minimal",
+ "notelp": "089647037426"
+ },
+ {
+ "id": "3b3e817b-c136-4488-bd80-9a7d408939a2",
+ "posisi": "Marketing Executive",
+ "namaPerusahaan": "PT Mitra Krida Mandiri (Dealer Honda MKM Darmasaba)",
+ "lokasi": "Jalan Raya Darmasaba No.169, Abiansemal, Badung, Bali",
+ "tipePekerjaan": "Full Time",
+ "gaji": "2.500.000",
+ "deskripsi": "Menjalankan tugas pemasaran dan penjualan produk Honda di area Darmasaba.",
+ "kualifikasi": "Usia 18-30 tahun, SMA/SMK minimal, memiliki sepeda motor Honda dan smartphone.",
+ "notelp": "081296001047"
+ },
+ {
+ "id": "3b3e817b-c136-4488-ce91-9a7d408939a2",
+ "posisi": "Kasir",
+ "namaPerusahaan": "GOGO DARMASABA",
+ "lokasi": "Jl. Raya Darmasaba, Darmasaba, Kec. Abiansemal, Kabupaten Badung, Bali",
+ "tipePekerjaan": "Full Time",
+ "gaji": "2.500.000",
+ "deskripsi": "Melakukan pelayanan kasir dan administrasi pelanggan di restoran/food service.",
+ "kualifikasi": "Wanita, 18-30 tahun, SMA/SMK minimal pengalaman 1-3 tahun sebagai kasir atau Customer Service.",
+ "notelp": "089647037426"
+ },
+ {
+ "id": "3b3e817b-c136-4488-df02-9a7d408939a2",
+ "posisi": "Kasir / Teknisi Handphone",
+ "namaPerusahaan": "Jaya Cell Darmasaba",
+ "lokasi": "Jl. Raya Darmasaba, Darmasaba, Abiansemal, Badung, Bali",
+ "tipePekerjaan": "Full Time",
+ "gaji": "2.000.000",
+ "deskripsi": "Melakukan pelayanan kasir serta teknisi ponsel termasuk troubleshooting dan perbaikan.",
+ "kualifikasi": "Tidak disebutkan pengalaman khusus, memiliki KTP dan keinginan kuat untuk bekerja.",
+ "notelp": "089647037426"
+ },
+ {
+ "id": "3b3e817b-c136-4488-eg13-9a7d408939a2",
+ "posisi": "Guru Les (Pengajar Anak)",
+ "namaPerusahaan": "Bimba AIUEO Darmasaba",
+ "lokasi": "Darmasaba, Kabupaten Badung, Bali",
+ "tipePekerjaan": "Full Time",
+ "gaji": "2.000.000",
+ "deskripsi": "Mengajar calistung dan perkembangan dasar anak usia 3-6 tahun.",
+ "kualifikasi": "Minimal SMA/SMK, komunikasi baik, berinteraksi dengan anak-anak.",
+ "notelp": "089647037426"
+ }
+]
diff --git a/prisma/data/ekonomi/pasar-desa/kategori-produk.json b/prisma/data/ekonomi/pasar-desa/kategori-produk.json
index c8c9b59e..7093047e 100644
--- a/prisma/data/ekonomi/pasar-desa/kategori-produk.json
+++ b/prisma/data/ekonomi/pasar-desa/kategori-produk.json
@@ -6,5 +6,17 @@
{
"id": "5c06chf7-123f-6hfe-0663-5e9h76e55060",
"nama": "Minuman"
+ },
+ {
+ "id": "5c06chf7-123f-7igd-0663-5e9h76e55060",
+ "nama": "Sembako"
+ },
+ {
+ "id": "5c06chf7-123f-8jhe-0663-5e9h76e55060",
+ "nama": "Sayur Mayur"
+ },
+ {
+ "id": "5c06chf7-123f-9kif-0663-5e9h76e55060",
+ "nama": "Protein Hewani"
}
]
diff --git a/prisma/data/ekonomi/pasar-desa/kategori-to-pasar.json b/prisma/data/ekonomi/pasar-desa/kategori-to-pasar.json
new file mode 100644
index 00000000..0ac0fae2
--- /dev/null
+++ b/prisma/data/ekonomi/pasar-desa/kategori-to-pasar.json
@@ -0,0 +1,42 @@
+[
+ {
+ "id": "f6b52033-5016-45d9-b0fd-b9d4b6c4729b",
+ "kategoriId": "5c06chf7-123f-9kif-0663-5e9h76e55060",
+ "pasarDesaId": "d62660a2-ac6b-428a-acf6-58cc837ef789"
+ },
+ {
+ "id": "d2ef373c-043c-44b5-adde-6a25a54199d3",
+ "kategoriId": "5c06chf7-123f-7igd-0663-5e9h76e55060",
+ "pasarDesaId": "d62660a2-ac6b-428a-acf6-58cc837ef789"
+ },
+ {
+ "id": "ad427752-fea0-4ef3-a312-5961eefd5ee3",
+ "kategoriId": "4b95bge6-012e-5ged-9552-4d8g65d44959",
+ "pasarDesaId": "d62660a2-ac6b-428a-acf6-58cc837ef789"
+ },
+ {
+ "id": "bd00ab59-7ac8-4d40-94de-a86bb0eb4557",
+ "kategoriId": "5c06chf7-123f-8jhe-0663-5e9h76e55060",
+ "pasarDesaId": "24c6b992-49da-4c6e-aebb-72cf89f75438"
+ },
+ {
+ "id": "b7d311a2-a23a-499d-a339-823c5e30849a",
+ "kategoriId": "5c06chf7-123f-7igd-0663-5e9h76e55060",
+ "pasarDesaId": "24c6b992-49da-4c6e-aebb-72cf89f75438"
+ },
+ {
+ "id": "50ccc6c9-92c1-4d86-9585-85d48d35f640",
+ "kategoriId": "4b95bge6-012e-5ged-9552-4d8g65d44959",
+ "pasarDesaId": "24c6b992-49da-4c6e-aebb-72cf89f75438"
+ },
+ {
+ "id": "3b27f795-1d1d-4655-90f9-b779a009094e",
+ "kategoriId": "4b95bge6-012e-5ged-9552-4d8g65d44959",
+ "pasarDesaId": "6dea2257-b710-4cd2-8d94-9b6737e658d8"
+ },
+ {
+ "id": "d45873c5-5948-40f9-a88d-aa0861132bae",
+ "kategoriId": "4b95bge6-012e-5ged-9552-4d8g65d44959",
+ "pasarDesaId": "1b7a17ea-83f7-4e73-a94d-96e2b4a623f2"
+ }
+]
diff --git a/prisma/data/ekonomi/pasar-desa/pasar-desa.json b/prisma/data/ekonomi/pasar-desa/pasar-desa.json
new file mode 100644
index 00000000..5ab82ca5
--- /dev/null
+++ b/prisma/data/ekonomi/pasar-desa/pasar-desa.json
@@ -0,0 +1,46 @@
+[
+ {
+ "id": "1b7a17ea-83f7-4e73-a94d-96e2b4a623f2",
+ "nama": "Warung Pasar Darmasaba",
+ "harga": 30000,
+ "imageName": "YdCBnK-bWxlyHjwsk4Qie-mobile.webp",
+ "rating": 4.3,
+ "alamatUsaha": "Br. Baler Pasar, Desa Darmasaba, Kec. Abiansemal",
+ "kontak": "081234567890",
+ "deskripsi": "Warung tradisional yang menjual kebutuhan pokok harian seperti sembako, jajanan pasar, dan minuman.",
+ "kategoriProdukId": "5c06chf7-123f-7igd-0663-5e9h76e55060"
+ },
+ {
+ "id": "6dea2257-b710-4cd2-8d94-9b6737e658d8",
+ "nama": "Jajanan Pasar Bu Made",
+ "imageName": "TWdNTZZbTOhFTNJGGPDyG-mobile.webp",
+ "harga": 5000,
+ "rating": 4.6,
+ "alamatUsaha": "Jl. Raya Darmasaba, dekat Banjar Baler Pasar",
+ "kontak": "082145678901",
+ "deskripsi": "Menjual berbagai jajanan pasar tradisional Bali seperti laklak, klepon, dan pisang rai.",
+ "kategoriProdukId": "4b95bge6-012e-5ged-9552-4d8g65d44959"
+ },
+ {
+ "id": "24c6b992-49da-4c6e-aebb-72cf89f75438",
+ "nama": "Sayur Segar Pak Wayan",
+ "imageName": "mtQsaKtQnhxIYVIooCkiQ-mobile.webp",
+ "harga": 20000,
+ "rating": 4.4,
+ "alamatUsaha": "Area Pasar Desa Darmasaba",
+ "kontak": "087865432109",
+ "deskripsi": "Lapak sayur segar yang menyediakan sayuran lokal hasil petani sekitar Desa Darmasaba.",
+ "kategoriProdukId": "5c06chf7-123f-8jhe-0663-5e9h76e55060"
+ },
+ {
+ "id": "d62660a2-ac6b-428a-acf6-58cc837ef789",
+ "nama": "Ayam & Daging Segar Darmasaba",
+ "imageName": "Ez-SkRyf_F-1gksz_amNg-mobile.webp",
+ "harga": 80000,
+ "rating": 4.2,
+ "alamatUsaha": "Br. Baler Pasar, Desa Darmasaba",
+ "kontak": "081998877665",
+ "deskripsi": "Menjual ayam potong dan daging segar untuk kebutuhan rumah tangga dan upacara adat.",
+ "kategoriProdukId": "5c06chf7-123f-9kif-0663-5e9h76e55060"
+ }
+]
diff --git a/prisma/data/ekonomi/pendapatan-asli-desa/apbDesa.json b/prisma/data/ekonomi/pendapatan-asli-desa/apbDesa.json
new file mode 100644
index 00000000..8bad2769
--- /dev/null
+++ b/prisma/data/ekonomi/pendapatan-asli-desa/apbDesa.json
@@ -0,0 +1,6 @@
+[
+ {
+ "id": "6b52d644-09b7-4af5-a78b-3dad854973e7",
+ "tahun": 2025
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/ekonomi/pendapatan-asli-desa/belanjaDesa.json b/prisma/data/ekonomi/pendapatan-asli-desa/belanjaDesa.json
new file mode 100644
index 00000000..4293bdff
--- /dev/null
+++ b/prisma/data/ekonomi/pendapatan-asli-desa/belanjaDesa.json
@@ -0,0 +1,7 @@
+[
+ {
+ "id": "82b8c96f-7817-482f-ba5e-85ad78c7bd57",
+ "name": "Belanja Desa (Realisasi Semester I)",
+ "nilai": 43871400
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/ekonomi/pendapatan-asli-desa/pembiayaanDesa.json b/prisma/data/ekonomi/pendapatan-asli-desa/pembiayaanDesa.json
new file mode 100644
index 00000000..a11284f4
--- /dev/null
+++ b/prisma/data/ekonomi/pendapatan-asli-desa/pembiayaanDesa.json
@@ -0,0 +1,7 @@
+[
+ {
+ "id": "cb593f99-9f15-4294-9c17-a93c8c149a25",
+ "name": "Pembiayaan Neto (SILPA 2024)",
+ "nilai": 69289666
+ }
+]
diff --git a/prisma/data/ekonomi/pendapatan-asli-desa/pendapatanDesa.json b/prisma/data/ekonomi/pendapatan-asli-desa/pendapatanDesa.json
new file mode 100644
index 00000000..82b0c1dd
--- /dev/null
+++ b/prisma/data/ekonomi/pendapatan-asli-desa/pendapatanDesa.json
@@ -0,0 +1,7 @@
+[
+ {
+ "id": "24576ca1-3dee-4608-9fb3-51f16efce901",
+ "name": "Pendapatan Asli Desa (PADes)",
+ "nilai": 10610011
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/ekonomi/program-kemiskinan/program-kemiskinan.json b/prisma/data/ekonomi/program-kemiskinan/program-kemiskinan.json
new file mode 100644
index 00000000..dfb09288
--- /dev/null
+++ b/prisma/data/ekonomi/program-kemiskinan/program-kemiskinan.json
@@ -0,0 +1,23 @@
+[
+ {
+ "id": "c2760e40-f770-11f0-89ff-719f813f71b3",
+ "nama": "BLT-DD (Bantuan Langsung Tunai Dana Desa)",
+ "icon": "bantuan",
+ "deskripsi": "Program pemberian Bantuan Langsung Tunai yang dibiayai dari Dana Desa untuk meringankan beban ekonomi keluarga miskin/prasejahtera di Desa Darmasaba.
",
+ "tahun": 2023
+ },
+ {
+ "id": "c7f0f2e0-f770-11f0-89ff-719f813f71b3",
+ "nama": "Penguatan Ketahanan Pangan",
+ "icon": "air",
+ "deskripsi": "Kegiatan pemberdayaan masyarakat dalam ketahanan pangan untuk mendukung ketersediaan pangan keluarga kurang mampu dan meningkatkan kemampuan produksi pangan lokal.
",
+ "tahun": 2024
+ },
+ {
+ "id": "ccda45e0-f770-11f0-89ff-719f813f71b3",
+ "nama": "Peningkatan IKM berbasis E-commerce",
+ "icon": "ekonomi",
+ "deskripsi": "Program peningkatan keterampilan usaha mikro kecil (IKM) termasuk pelatihan branding, pengemasan, dan promosi digital untuk memperkuat ekonomi rumah tangga melalui pemasaran online.
",
+ "tahun": 2025
+ }
+]
diff --git a/prisma/data/ekonomi/program-kemiskinan/statistik-kemiskinan.json b/prisma/data/ekonomi/program-kemiskinan/statistik-kemiskinan.json
new file mode 100644
index 00000000..475f015d
--- /dev/null
+++ b/prisma/data/ekonomi/program-kemiskinan/statistik-kemiskinan.json
@@ -0,0 +1,17 @@
+[
+ {
+ "id": "85d8a150-f770-11f0-89ff-719f813f71b3",
+ "tahun": 2023,
+ "jumlah": 20
+ },
+ {
+ "id": "993b8d20-f770-11f0-89ff-719f813f71b3",
+ "tahun": 2024,
+ "jumlah": 30
+ },
+ {
+ "id": "9eb3b2a0-f770-11f0-89ff-719f813f71b3",
+ "tahun": 2025,
+ "jumlah": 20
+ }
+]
diff --git a/prisma/data/ekonomi/sektor-unggulan/sektor-unggulan.json b/prisma/data/ekonomi/sektor-unggulan/sektor-unggulan.json
new file mode 100644
index 00000000..fca71eb1
--- /dev/null
+++ b/prisma/data/ekonomi/sektor-unggulan/sektor-unggulan.json
@@ -0,0 +1,44 @@
+[
+ {
+ "id": "053999e8-e5c4-4a50-b587-0e0ce15aba1a",
+ "name": "Pertanian",
+ "description": "Sektor pertanian meliputi kegiatan bercocok tanam padi, palawija, dan tanaman lain di subak yang menjadi basis mata pencaharian warga",
+ "value": 90
+ },
+ {
+ "id": "8e0d2f2d-512d-4c05-8880-b6e7d144a34d",
+ "name": "UMKM Kecil",
+ "description": "Usaha Mikro Kecil Menengah termasuk IKM berbasis pengolahan pangan dan kuliner yang tumbuh di desa sebagai penggerak ekonomi lokal",
+ "value": 45
+ },
+ {
+ "id": "0378b10a-f0e3-421c-9272-225d931179ce",
+ "name": "Peternakan",
+ "description": "Peternakan ikan lele dan mata pencaharian lain yang mendukung ketahanan pangan dan ekonomi masyarakat desa",
+ "value": 30
+ },
+ {
+ "id": "4fa28680-8014-4c46-9dd0-1aa910630fd3",
+ "name": "BUMDes",
+ "description": "BUMDes Pudak Mesari sebagai lembaga usaha desa yang mengembangkan potensi lokal dan layanan ekonomi",
+ "value": 20
+ },
+ {
+ "id": "669464b2-dd7e-44be-b609-97a9b844df8b",
+ "name": "Kawasan Kuliner",
+ "description": "Potensi kawasan kuliner desa yang menjadi daya tarik ekonomi dan pariwisata kecil di daerah Darmasaba",
+ "value": 15
+ },
+ {
+ "id": "ef65e122-84ce-4483-93e9-c1a8bcee9b79",
+ "name": "Pariwisata",
+ "description": "Potensi wisata lokal seperti Jogging Track, taman dan water park yang memberikan nilai tambah ekonomi masyarakat",
+ "value": 35
+ },
+ {
+ "id": "08443c84-8ca9-4690-b900-e5e3e753cc97",
+ "name": "Kerajinan Genteng",
+ "description": "Kerajinan genteng press di Desa Adat Tegal yang merupakan usaha kerajinan lokal dengan kontribusi ekonomi",
+ "value": 25
+ }
+]
diff --git a/prisma/data/fetchWithRetry.ts b/prisma/data/fetchWithRetry.ts
new file mode 100644
index 00000000..e8a55d73
--- /dev/null
+++ b/prisma/data/fetchWithRetry.ts
@@ -0,0 +1,33 @@
+export default async function fetchWithRetry(
+ url: string,
+ retries = 3,
+ timeoutMs = 20000
+) {
+ if (!url || url.trim() === "") {
+ throw new Error("fetchWithRetry called with empty URL");
+ }
+ for (let attempt = 1; attempt <= retries; attempt++) {
+ const controller = new AbortController();
+ const timeout = setTimeout(() => controller.abort(), timeoutMs);
+
+ try {
+ const res = await fetch(url, { signal: controller.signal });
+
+ if (!res.ok) {
+ throw new Error(`HTTP ${res.status} ${res.statusText}`);
+ }
+
+ return res;
+ } catch (err) {
+ console.warn(`⚠️ Download attempt ${attempt} failed`);
+
+ if (attempt === retries) {
+ throw err;
+ }
+ } finally {
+ clearTimeout(timeout);
+ }
+ }
+
+ throw new Error("Unreachable");
+}
diff --git a/prisma/data/file-storage.json b/prisma/data/file-storage.json
index 53c27a0b..2f5267dd 100644
--- a/prisma/data/file-storage.json
+++ b/prisma/data/file-storage.json
@@ -1,137 +1,406 @@
[
{
- "id": "cmff0rr4z0002vn0twp333m2",
- "name": "S6RIjFaPvdQm3oq4rM4X9-desktop.webp",
- "realName": "bares.png",
- "path": "uploads/images",
+ "name": "42RCCpBZla4ZWxXcwx7kG-desktop.webp",
+ "path": "image/42RCCpBZla4ZWxXcwx7kG-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d05a9e22-feac-4955-b1a2-75faad37f0ac/42RCCpBZla4ZWxXcwx7kG-desktop.webp",
"mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/S6RIjFaPvdQm3oq4rM4X9-desktop.webp",
+ "link": "/api/fileStorage/findUnique/Gc79mlIlGuoRQuTqskFj--desktop.webp",
"category": "image"
+
},
{
- "id": "cmff0tnf00003vn0t3kgzi0u0",
- "name": "_pVNEmThU5ICGa8gv3gh_-desktop.webp",
- "realName": "bicara-darma.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/_pVNEmThU5ICGa8gv3gh_-desktop.webp",
- "category": "image"
+ "name": "42RCCpBZla4ZWxXcwx7kG-mobile.webp",
+ "path": "image/42RCCpBZla4ZWxXcwx7kG-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c91212b5-7408-4ec9-b7c0-6c0fde2a9fc5/42RCCpBZla4ZWxXcwx7kG-mobile.webp"
},
{
- "id": "cmff0uykf0004vn0trmmxpgfh",
- "name": "bv6rdKvjxkkjUSGLQ0lvB-desktop.webp",
- "realName": "daves.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/bv6rdKvjxkkjUSGLQ0lvB-desktop.webp",
- "category": "image"
+ "name": "6DQbAvn0St-xHdPGW3vpY-desktop.webp",
+ "path": "image/6DQbAvn0St-xHdPGW3vpY-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/68cc521e-05e5-4258-af8d-c87fb76c927e/6DQbAvn0St-xHdPGW3vpY-desktop.webp"
},
{
- "id": "cmff0z34f0005vn0tjtvq519p",
- "name": "Z4hWaV04CvoE20MjccQsV-desktop.webp",
- "realName": "mangan.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/Z4hWaV04CvoE20MjccQsV-desktop.webp",
- "category": "image"
+ "name": "6DQbAvn0St-xHdPGW3vpY-mobile.webp",
+ "path": "image/6DQbAvn0St-xHdPGW3vpY-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/15f7ddd6-448c-4e4a-9b24-0bdc4bce5ce1/6DQbAvn0St-xHdPGW3vpY-mobile.webp"
},
{
- "id": "cmff38cyq000bvn0t9f01cz3f",
- "name": "LvLAtOqWojx4sn6NjJWB9-desktop.webp",
- "realName": "gelah-melah.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/LvLAtOqWojx4sn6NjJWB9-desktop.webp",
- "category": "image"
+ "name": "buku1 (1).jpeg",
+ "path": "image/buku1 (1).jpeg",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d72b02a6-91f5-4d64-9729-521b306328d3/buku1%20%281%29.jpeg"
},
{
- "id": "cmff0zqvd0007vn0tv6o5hjcq",
- "name": "gR2mcvAQVgJ2-rM5coYJj-desktop.webp",
- "realName": "inovasi-desa-darmasaba.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/gR2mcvAQVgJ2-rM5coYJj-desktop.webp",
- "category": "image"
+ "name": "buku1.jpeg",
+ "path": "image/buku1.jpeg",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/912bd7e5-be8a-4c19-8ab8-df6114a38864/buku1.jpeg"
},
{
- "id": "cmff1013m0008vn0th7t0d64d",
- "name": "JpL-9F8-IGztMn8E2ce02-desktop.webp",
- "realName": "pdkt.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/JpL-9F8-IGztMn8E2ce02-desktop.webp",
- "category": "image"
+ "name": "buku6.jpg",
+ "path": "image/buku6.jpg",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/ea56bc7c-094b-464e-859a-6c65bf65361d/buku6.jpg"
},
{
- "id": "cmff10cwq0009vn0tse8dzu3j",
- "name": "bxAk4AsGbJTC705_IVdes-desktop.webp",
- "realName": "sajjiana-dharma-raksaka.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/bxAk4AsGbJTC705_IVdes-desktop.webp",
- "category": "image"
+ "name": "c7xWNyoYp8Cak28NG5NoG-desktop.webp",
+ "path": "image/c7xWNyoYp8Cak28NG5NoG-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d84fb066-4352-44f0-b7bc-5d8c5b8ffb61/c7xWNyoYp8Cak28NG5NoG-desktop.webp"
},
{
- "id": "cmff2w5ly000avn0telhct71k",
- "name": "Vbj_osnMJUkGEQGDTLwV--desktop.webp",
- "realName": "perbekel.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/Vbj_osnMJUkGEQGDTLwV--desktop.webp",
- "category": "image"
+ "name": "c7xWNyoYp8Cak28NG5NoG-mobile.webp",
+ "path": "image/c7xWNyoYp8Cak28NG5NoG-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/9fb1c742-9a80-4788-9c73-9d06108e0051/c7xWNyoYp8Cak28NG5NoG-mobile.webp"
},
{
- "id": "cmff3joae0000vn6h8sgs0ilg",
- "name": "7hox9spUxj56hY_EBYLnj-desktop.webp",
- "realName": "youtube.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/7hox9spUxj56hY_EBYLnj-desktop.webp",
- "category": "image"
+ "name": "cg78Sb_QzZFlli9s2FPVc-mobile.webp",
+ "path": "image/cg78Sb_QzZFlli9s2FPVc-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c5a2cd18-806d-4dcb-b5c5-b0c6bcef7f35/cg78Sb_QzZFlli9s2FPVc-mobile.webp"
},
{
- "id": "cmff3ll130001vn6hkhls3f5y",
- "name": "ChihV7_1eS-AGtSg9UwMv-desktop.webp",
- "realName": "gmail.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/ChihV7_1eS-AGtSg9UwMv-desktop.webp",
- "category": "image"
+ "name": "d3v1AgLoSJhf5xvmmO3oP-mobile.webp",
+ "path": "image/d3v1AgLoSJhf5xvmmO3oP-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/ae72c51f-9df7-4b07-9fd5-e945c775d9ab/d3v1AgLoSJhf5xvmmO3oP-mobile.webp"
},
{
- "id": "cmff3mtat0002vn6hs8vyyhdd",
- "name": "z8v9ZREwOJHKGIRYauROt-desktop.webp",
- "realName": "facebook.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/z8v9ZREwOJHKGIRYauROt-desktop.webp",
- "category": "image"
+ "name": "d6hJgycQawWN3VEcHaqtR-desktop.webp",
+ "path": "image/d6hJgycQawWN3VEcHaqtR-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/4e5b9387-67b5-4c00-8f80-95ec6c54ff4a/d6hJgycQawWN3VEcHaqtR-desktop.webp"
},
{
- "id": "cmff3nv180003vn6h5jvedidq",
- "name": "BLjMxTKoCNE31uOURR3IU-desktop.webp",
- "realName": "telephone-call.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/BLjMxTKoCNE31uOURR3IU-desktop.webp",
- "category": "image"
+ "name": "d6hJgycQawWN3VEcHaqtR-mobile.webp",
+ "path": "image/d6hJgycQawWN3VEcHaqtR-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/011f65c9-4273-43e8-b6d7-ce7a5319ae83/d6hJgycQawWN3VEcHaqtR-mobile.webp"
},
{
- "id": "cmff3oouh0004vn6hd94brzv9",
- "name": "hkJYAeTNWK_vYaYS20w3I-desktop.webp",
- "realName": "instagram.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/hkJYAeTNWK_vYaYS20w3I-desktop.webp",
- "category": "image"
+ "name": "DyX82oztXbHfu6HEvbrpt-desktop.webp",
+ "path": "image/DyX82oztXbHfu6HEvbrpt-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/b715dce5-1606-44cf-976f-57d8142e218e/DyX82oztXbHfu6HEvbrpt-desktop.webp"
},
{
- "id": "cmff3q12g0005vn6h5ojov2qa",
- "name": "6XEoZ9SFu59COpil03Gya-desktop.webp",
- "realName": "tiktok.png",
- "path": "uploads/images",
- "mimeType": "image/webp",
- "link": "/api/fileStorage/findUnique/6XEoZ9SFu59COpil03Gya-desktop.webp",
- "category": "image"
+ "name": "DyX82oztXbHfu6HEvbrpt-mobile.webp",
+ "path": "image/DyX82oztXbHfu6HEvbrpt-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d0ff5925-ec84-4100-920c-93e2eb479f13/DyX82oztXbHfu6HEvbrpt-mobile.webp"
+ },
+ {
+ "name": "EcQIGOF6LW1dIKE53vmba-desktop.webp",
+ "path": "image/EcQIGOF6LW1dIKE53vmba-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/7a97403e-9d6c-4a00-9c54-d5add0bd5915/EcQIGOF6LW1dIKE53vmba-desktop.webp"
+ },
+ {
+ "name": "EcQIGOF6LW1dIKE53vmba-mobile.webp",
+ "path": "image/EcQIGOF6LW1dIKE53vmba-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/7e85b8e8-27f5-46c7-b4fc-590d66eef2ce/EcQIGOF6LW1dIKE53vmba-mobile.webp"
+ },
+ {
+ "name": "Ez-SkRyf_F-1gksz_amNg-desktop.webp",
+ "path": "image/Ez-SkRyf_F-1gksz_amNg-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c16de343-1297-4248-b104-83b3e3605f32/Ez-SkRyf_F-1gksz_amNg-desktop.webp"
+ },
+ {
+ "name": "Ez-SkRyf_F-1gksz_amNg-mobile.webp",
+ "path": "image/Ez-SkRyf_F-1gksz_amNg-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/ece148ca-8aa1-43ef-a8de-ea0bde0a315a/Ez-SkRyf_F-1gksz_amNg-mobile.webp"
+ },
+ {
+ "name": "g4ICsRrmOaIqS_yqlQLZK-desktop.webp",
+ "path": "image/g4ICsRrmOaIqS_yqlQLZK-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/31280b01-6cdb-4d96-9c96-0eecec6a238c/g4ICsRrmOaIqS_yqlQLZK-desktop.webp"
+ },
+ {
+ "name": "g4ICsRrmOaIqS_yqlQLZK-mobile.webp",
+ "path": "image/g4ICsRrmOaIqS_yqlQLZK-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/af7f3e11-1f63-4307-88e9-64382a949279/g4ICsRrmOaIqS_yqlQLZK-mobile.webp"
+ },
+ {
+ "name": "Gc79mlIlGuoRQuTqskFj--desktop.webp",
+ "path": "image/Gc79mlIlGuoRQuTqskFj--desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/05214122-51bb-4fba-9b7c-f69e072d8a0d/Gc79mlIlGuoRQuTqskFj--desktop.webp"
+ },
+ {
+ "name": "Gc79mlIlGuoRQuTqskFj--mobile.webp",
+ "path": "image/Gc79mlIlGuoRQuTqskFj--mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d49ab5d2-7087-4088-8e59-51f116f21e27/Gc79mlIlGuoRQuTqskFj--mobile.webp"
+ },
+ {
+ "name": "Gi8EX3pBmT719AfzXirDS-desktop.webp",
+ "path": "image/Gi8EX3pBmT719AfzXirDS-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/4b456bad-228a-4211-a1bb-431dc081ecc7/Gi8EX3pBmT719AfzXirDS-desktop.webp"
+ },
+ {
+ "name": "Gi8EX3pBmT719AfzXirDS-mobile.webp",
+ "path": "image/Gi8EX3pBmT719AfzXirDS-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/68308100-d468-4e53-9cc6-3aa12035c8ab/Gi8EX3pBmT719AfzXirDS-mobile.webp"
+ },
+ {
+ "name": "gyNi4s8TnK2UrViU-gN2C-desktop.webp",
+ "path": "image/gyNi4s8TnK2UrViU-gN2C-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/539debe4-f3fc-4574-a256-ac8c8dbf5a00/gyNi4s8TnK2UrViU-gN2C-desktop.webp"
+ },
+ {
+ "name": "gyNi4s8TnK2UrViU-gN2C-mobile.webp",
+ "path": "image/gyNi4s8TnK2UrViU-gN2C-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/4971a22a-c0dd-4492-bff5-a8aa3e93f27f/gyNi4s8TnK2UrViU-gN2C-mobile.webp"
+ },
+ {
+ "name": "h_Gd0SoeIJVTi_5TWUO-P-desktop.webp",
+ "path": "image/h_Gd0SoeIJVTi_5TWUO-P-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/8409886e-c01b-421a-ac44-6d3a4bfd0985/h_Gd0SoeIJVTi_5TWUO-P-desktop.webp"
+ },
+ {
+ "name": "h_Gd0SoeIJVTi_5TWUO-P-mobile.webp",
+ "path": "image/h_Gd0SoeIJVTi_5TWUO-P-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/f4a5280d-bd2d-4c5a-8040-183f9e5d951b/h_Gd0SoeIJVTi_5TWUO-P-mobile.webp"
+ },
+ {
+ "name": "hLeF0GRFZqDUngZnDMAAk-desktop.webp",
+ "path": "image/hLeF0GRFZqDUngZnDMAAk-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/6b33097e-8d9d-4864-964b-6dc49b62b4ae/hLeF0GRFZqDUngZnDMAAk-desktop.webp"
+ },
+ {
+ "name": "hLeF0GRFZqDUngZnDMAAk-mobile.webp",
+ "path": "image/hLeF0GRFZqDUngZnDMAAk-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/7dff816b-65bb-429c-b87e-3ff892d547dc/hLeF0GRFZqDUngZnDMAAk-mobile.webp"
+ },
+ {
+ "name": "hsHiD59dZQxr8G2SAfUYp-mobile.webp",
+ "path": "image/hsHiD59dZQxr8G2SAfUYp-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/55103cd4-6f57-491d-bd41-03c0068974ef/hsHiD59dZQxr8G2SAfUYp-mobile.webp"
+ },
+ {
+ "name": "hyyTFi8EApjzFEZ9EvJgB-desktop.webp",
+ "path": "image/hyyTFi8EApjzFEZ9EvJgB-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/0ec4a67e-695e-4524-bf22-f823b80c7e6b/hyyTFi8EApjzFEZ9EvJgB-desktop.webp"
+ },
+ {
+ "name": "hyyTFi8EApjzFEZ9EvJgB-mobile.webp",
+ "path": "image/hyyTFi8EApjzFEZ9EvJgB-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c982541e-f8b2-455a-a3cd-f856cd954bed/hyyTFi8EApjzFEZ9EvJgB-mobile.webp"
+ },
+ {
+ "name": "isTT2LmPbeOWD5wAdqleX-mobile.webp",
+ "path": "image/isTT2LmPbeOWD5wAdqleX-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/ef6f8237-0803-422a-83af-4daca61c7065/isTT2LmPbeOWD5wAdqleX-mobile.webp"
+ },
+ {
+ "name": "JhJigMo269K1TFGzSB1OS-desktop.webp",
+ "path": "image/JhJigMo269K1TFGzSB1OS-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/668d2f32-3277-4842-8a01-c3cb5ca0852b/JhJigMo269K1TFGzSB1OS-desktop.webp"
+ },
+ {
+ "name": "JhJigMo269K1TFGzSB1OS-mobile.webp",
+ "path": "image/JhJigMo269K1TFGzSB1OS-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/df8c52a3-189a-48a0-b7e7-98efe2479414/JhJigMo269K1TFGzSB1OS-mobile.webp"
+ },
+ {
+ "name": "jYxEXspWH5g6eTTVqK72c-desktop.webp",
+ "path": "image/jYxEXspWH5g6eTTVqK72c-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/a13572b4-2e22-4d76-8826-64a8e6ae4e13/jYxEXspWH5g6eTTVqK72c-desktop.webp"
+ },
+ {
+ "name": "jYxEXspWH5g6eTTVqK72c-mobile.webp",
+ "path": "image/jYxEXspWH5g6eTTVqK72c-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/55f171fa-585f-4a94-bfeb-d2bad2f7ee39/jYxEXspWH5g6eTTVqK72c-mobile.webp"
+ },
+ {
+ "name": "K0wY911212dinYA3AFB_f-desktop.webp",
+ "path": "image/K0wY911212dinYA3AFB_f-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/31ea22bf-9ce1-4dc6-b901-caaec86c35c4/K0wY911212dinYA3AFB_f-desktop.webp"
+ },
+ {
+ "name": "K0wY911212dinYA3AFB_f-mobile.webp",
+ "path": "image/K0wY911212dinYA3AFB_f-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/ea061aa6-7e5e-447c-bec8-be1d927cc578/K0wY911212dinYA3AFB_f-mobile.webp"
+ },
+ {
+ "name": "l4qsUEw2JiclGAkkrXp9g-desktop.webp",
+ "path": "image/l4qsUEw2JiclGAkkrXp9g-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/05088eb1-44bc-44d6-9e67-08dd6bca00ae/l4qsUEw2JiclGAkkrXp9g-desktop.webp"
+ },
+ {
+ "name": "l4qsUEw2JiclGAkkrXp9g-mobile.webp",
+ "path": "image/l4qsUEw2JiclGAkkrXp9g-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/cf411a77-b7fc-4ac7-a98d-f9f89bc27e85/l4qsUEw2JiclGAkkrXp9g-mobile.webp"
+ },
+ {
+ "name": "M9QlgVKIEfCdY3g4F_tRZ-desktop.webp",
+ "path": "image/M9QlgVKIEfCdY3g4F_tRZ-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/dfa58064-c17b-453a-b0a1-613109757844/M9QlgVKIEfCdY3g4F_tRZ-desktop.webp"
+ },
+ {
+ "name": "M9QlgVKIEfCdY3g4F_tRZ-mobile.webp",
+ "path": "image/M9QlgVKIEfCdY3g4F_tRZ-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/4be366ca-1a39-46e9-adf2-d1d28fd83961/M9QlgVKIEfCdY3g4F_tRZ-mobile.webp"
+ },
+ {
+ "name": "mtQsaKtQnhxIYVIooCkiQ-desktop.webp",
+ "path": "image/mtQsaKtQnhxIYVIooCkiQ-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/5015a484-ac9a-485f-9fb0-a29e3824a6ce/mtQsaKtQnhxIYVIooCkiQ-desktop.webp"
+ },
+ {
+ "name": "mtQsaKtQnhxIYVIooCkiQ-mobile.webp",
+ "path": "image/mtQsaKtQnhxIYVIooCkiQ-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/10ddd132-5b85-4a8f-b129-56a10540fc8c/mtQsaKtQnhxIYVIooCkiQ-mobile.webp"
+ },
+ {
+ "name": "NBPAqjPXn7GQmYTDBI5hu-desktop.webp",
+ "path": "image/NBPAqjPXn7GQmYTDBI5hu-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/fd8cfea3-5cf2-489e-9f21-e7985335dc98/NBPAqjPXn7GQmYTDBI5hu-desktop.webp"
+ },
+ {
+ "name": "NBPAqjPXn7GQmYTDBI5hu-mobile.webp",
+ "path": "image/NBPAqjPXn7GQmYTDBI5hu-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/f5b02049-95db-4e1b-8bfd-b1a4c431ee49/NBPAqjPXn7GQmYTDBI5hu-mobile.webp"
+ },
+ {
+ "name": "NyPGo-1AtfNm5wkAq7Om6-mobile.webp",
+ "path": "image/NyPGo-1AtfNm5wkAq7Om6-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/b7361eb8-37e8-4baa-a6d6-01b6131dd788/NyPGo-1AtfNm5wkAq7Om6-mobile.webp"
+ },
+ {
+ "name": "OsMY3AYPyGC_CoN1xUjOn-desktop.webp",
+ "path": "image/OsMY3AYPyGC_CoN1xUjOn-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/509a3603-5d11-4857-baa5-d439da43825a/OsMY3AYPyGC_CoN1xUjOn-desktop.webp"
+ },
+ {
+ "name": "OsMY3AYPyGC_CoN1xUjOn-mobile.webp",
+ "path": "image/OsMY3AYPyGC_CoN1xUjOn-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/cbe9798a-4cf0-45f7-b041-70269490128b/OsMY3AYPyGC_CoN1xUjOn-mobile.webp"
+ },
+ {
+ "name": "pps1ZgzJxDb4VZxEvtZeu-desktop.webp",
+ "path": "image/pps1ZgzJxDb4VZxEvtZeu-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/500fa1f3-d412-4f48-bc5f-465b91149c6e/pps1ZgzJxDb4VZxEvtZeu-desktop.webp"
+ },
+ {
+ "name": "pps1ZgzJxDb4VZxEvtZeu-mobile.webp",
+ "path": "image/pps1ZgzJxDb4VZxEvtZeu-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/19c2b1a6-8c9b-4be9-981c-72d7a3ab6e38/pps1ZgzJxDb4VZxEvtZeu-mobile.webp"
+ },
+ {
+ "name": "r_gBF0FuFpFPfSENHc4XI-desktop.webp",
+ "path": "image/r_gBF0FuFpFPfSENHc4XI-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/2e4ab451-df52-4f04-8af0-db2e789e58bb/r_gBF0FuFpFPfSENHc4XI-desktop.webp"
+ },
+ {
+ "name": "r_gBF0FuFpFPfSENHc4XI-mobile.webp",
+ "path": "image/r_gBF0FuFpFPfSENHc4XI-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/1e2e8884-9da3-4610-9c57-8925740d4128/r_gBF0FuFpFPfSENHc4XI-mobile.webp"
+ },
+ {
+ "name": "SQqSobKRg3ShvgPw_H41h-desktop.webp",
+ "path": "image/SQqSobKRg3ShvgPw_H41h-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/4360cb84-f82a-4a68-afd3-65decd912f30/SQqSobKRg3ShvgPw_H41h-desktop.webp"
+ },
+ {
+ "name": "SQqSobKRg3ShvgPw_H41h-mobile.webp",
+ "path": "image/SQqSobKRg3ShvgPw_H41h-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c808dff8-2e8a-4358-af03-4a838c9a6d6c/SQqSobKRg3ShvgPw_H41h-mobile.webp"
+ },
+ {
+ "name": "TDQReg1lQ73s39crXW0ra-desktop.webp",
+ "path": "image/TDQReg1lQ73s39crXW0ra-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/04d15638-31ed-440f-9fa0-bb30d71bbc59/TDQReg1lQ73s39crXW0ra-desktop.webp"
+ },
+ {
+ "name": "TDQReg1lQ73s39crXW0ra-mobile.webp",
+ "path": "image/TDQReg1lQ73s39crXW0ra-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/da174313-329c-4156-88e2-fc929325dfff/TDQReg1lQ73s39crXW0ra-mobile.webp"
+ },
+ {
+ "name": "TTur8BttDlAS9UgZVe3M8-desktop.webp",
+ "path": "image/TTur8BttDlAS9UgZVe3M8-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/32fe30ab-0d7b-4ada-8d5d-993baf23545c/TTur8BttDlAS9UgZVe3M8-desktop.webp"
+ },
+ {
+ "name": "TTur8BttDlAS9UgZVe3M8-mobile.webp",
+ "path": "image/TTur8BttDlAS9UgZVe3M8-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/85c5ecb0-2fb5-4431-af11-0c851e52de4e/TTur8BttDlAS9UgZVe3M8-mobile.webp"
+ },
+ {
+ "name": "TWdNTZZbTOhFTNJGGPDyG-desktop.webp",
+ "path": "image/TWdNTZZbTOhFTNJGGPDyG-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d606d946-fdf9-41b9-b0bd-1421b2ec6843/TWdNTZZbTOhFTNJGGPDyG-desktop.webp"
+ },
+ {
+ "name": "TWdNTZZbTOhFTNJGGPDyG-mobile.webp",
+ "path": "image/TWdNTZZbTOhFTNJGGPDyG-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/46859e6a-ebfc-4124-adfe-320953256fe5/TWdNTZZbTOhFTNJGGPDyG-mobile.webp"
+ },
+ {
+ "name": "TXknK9CSRSxwvM2hPW6BO-desktop.webp",
+ "path": "image/TXknK9CSRSxwvM2hPW6BO-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/af38b038-0d38-4222-be29-09cb81054ce7/TXknK9CSRSxwvM2hPW6BO-desktop.webp"
+ },
+ {
+ "name": "TXknK9CSRSxwvM2hPW6BO-mobile.webp",
+ "path": "image/TXknK9CSRSxwvM2hPW6BO-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/eab1c384-aafa-438f-ac0f-003ddd51c9a5/TXknK9CSRSxwvM2hPW6BO-mobile.webp"
+ },
+ {
+ "name": "U7rePDZq5E59z-Eo9tLBe-desktop.webp",
+ "path": "image/U7rePDZq5E59z-Eo9tLBe-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/a2858ed9-8bb6-47cd-9e1b-831928a0389f/U7rePDZq5E59z-Eo9tLBe-desktop.webp"
+ },
+ {
+ "name": "U7rePDZq5E59z-Eo9tLBe-mobile.webp",
+ "path": "image/U7rePDZq5E59z-Eo9tLBe-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/1f4a189d-d312-46ef-a68c-c0c7261860d0/U7rePDZq5E59z-Eo9tLBe-mobile.webp"
+ },
+ {
+ "name": "uDxAalFV0qRv_RrW9flM8-mobile.webp",
+ "path": "image/uDxAalFV0qRv_RrW9flM8-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/5f006c3c-47f5-4e48-8bdd-7fbe005bf810/uDxAalFV0qRv_RrW9flM8-mobile.webp"
+ },
+ {
+ "name": "uE2QwpbcXyBWxVYqCWQQT-desktop.webp",
+ "path": "image/uE2QwpbcXyBWxVYqCWQQT-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/edf73617-214b-44df-960a-dd68f0bad97a/uE2QwpbcXyBWxVYqCWQQT-desktop.webp"
+ },
+ {
+ "name": "uE2QwpbcXyBWxVYqCWQQT-mobile.webp",
+ "path": "image/uE2QwpbcXyBWxVYqCWQQT-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/d7f5d738-b18f-44c8-92bb-8e526f47d9ee/uE2QwpbcXyBWxVYqCWQQT-mobile.webp"
+ },
+ {
+ "name": "v7Ac2xQvTiJy-HYh1AxF4-desktop.webp",
+ "path": "image/v7Ac2xQvTiJy-HYh1AxF4-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/3e1b8dd9-2bf5-4daf-9dd9-dd687e9b2f2c/v7Ac2xQvTiJy-HYh1AxF4-desktop.webp"
+ },
+ {
+ "name": "v7Ac2xQvTiJy-HYh1AxF4-mobile.webp",
+ "path": "image/v7Ac2xQvTiJy-HYh1AxF4-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/44c78026-f802-46f3-8ec2-271f0f001f7a/v7Ac2xQvTiJy-HYh1AxF4-mobile.webp"
+ },
+ {
+ "name": "wh79hF4HTZMEFtYc-OfZg-mobile.webp",
+ "path": "image/wh79hF4HTZMEFtYc-OfZg-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/b4b4378e-76ad-4b15-84cf-00178553b3d4/wh79hF4HTZMEFtYc-OfZg-mobile.webp"
+ },
+ {
+ "name": "x0_-siY2V8IehBzo4_uph-desktop.webp",
+ "path": "image/x0_-siY2V8IehBzo4_uph-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/7f82bcea-e7c6-4cde-9701-8d3afe49c0f8/x0_-siY2V8IehBzo4_uph-desktop.webp"
+ },
+ {
+ "name": "x0_-siY2V8IehBzo4_uph-mobile.webp",
+ "path": "image/x0_-siY2V8IehBzo4_uph-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/c4b23ebc-3915-4102-a25c-4e2f6da0d097/x0_-siY2V8IehBzo4_uph-mobile.webp"
+ },
+ {
+ "name": "y78xZ2axTOjz87gRKjVAf-desktop.webp",
+ "path": "image/y78xZ2axTOjz87gRKjVAf-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/180d0c29-c93e-4bbe-b399-7e6a34fbeb49/y78xZ2axTOjz87gRKjVAf-desktop.webp"
+ },
+ {
+ "name": "y78xZ2axTOjz87gRKjVAf-mobile.webp",
+ "path": "image/y78xZ2axTOjz87gRKjVAf-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/71b5091a-ebcd-4d4c-9aae-1b7e548051fc/y78xZ2axTOjz87gRKjVAf-mobile.webp"
+ },
+ {
+ "name": "YdCBnK-bWxlyHjwsk4Qie-desktop.webp",
+ "path": "image/YdCBnK-bWxlyHjwsk4Qie-desktop.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/bf6f2e11-b328-4da1-bda7-81af2336d03f/YdCBnK-bWxlyHjwsk4Qie-desktop.webp"
+ },
+ {
+ "name": "YdCBnK-bWxlyHjwsk4Qie-mobile.webp",
+ "path": "image/YdCBnK-bWxlyHjwsk4Qie-mobile.webp",
+ "downloadUrl": "https://cld-dkr-makuro-seafile.wibudev.com/seafhttp/files/9d7e5232-3b30-4ead-9482-1fed3a86245a/YdCBnK-bWxlyHjwsk4Qie-mobile.webp"
}
]
diff --git a/prisma/data/inovasi/ajukan-ide/ajukan-ide.json b/prisma/data/inovasi/ajukan-ide/ajukan-ide.json
new file mode 100644
index 00000000..e2418571
--- /dev/null
+++ b/prisma/data/inovasi/ajukan-ide/ajukan-ide.json
@@ -0,0 +1,38 @@
+[
+ {
+ "id": "cmkm710n3000xvnsw1ni0vlam",
+ "name": "I Made Rudi",
+ "alamat": "Darmasaba, Abiansemal, Badung, Bali",
+ "namaIde": "Program Bersama Jaga Rabies (BAJRA)",
+ "deskripsi": "Inovasi berbasis komunitas yang mengintegrasikan pelaporan cepat masyarakat, edukasi berkelanjutan, serta koordinasi lintas sektor untuk penanggulangan rabies secara efektif di Desa Darmasaba.",
+ "masalah": "Tingginya kasus rabies dan rendahnya pelaporan serta koordinasi penanganan antar warga sehingga perlunya pendekatan komunitas yang terstruktur.",
+ "benefit": "Meningkatkan kesadaran vaksinasi, respons cepat terhadap kasus gigitan, serta memperkuat sistem kesiapsiagaan kesehatan hewan dan manusia di komunitas."
+ },
+ {
+ "id": "cmkm710n3000xvnsw1ni1wmbn",
+ "name": "I Made Rudja",
+ "alamat": "Darmasaba, Abiansemal, Badung, Bali",
+ "namaIde": "Program LESTARI – Ekowisata dan Edukasi Lingkungan",
+ "deskripsi": "Inovasi pengembangan ekowisata berbasis persawahan yang menggabungkan edukasi lingkungan dan potensi desa untuk menarik wisatawan serta meningkatkan kesejahteraan masyarakat.",
+ "masalah": "Kurangnya sarana edukasi lingkungan dan kurang optimalnya pemanfaatan potensi lahan persawahan sebagai sumber pendapatan masyarakat.",
+ "benefit": "Menambah peluang ekonomi desa, meningkatkan kesadaran lingkungan, dan memperkuat daya tarik wisata lokal Desa Darmasaba."
+ },
+ {
+ "id": "cmkm710n3000xvnsw1ni2xnco",
+ "name": "I Wayan Sumi",
+ "alamat": "Darmasaba, Abiansemal, Badung, Bali",
+ "namaIde": "Darmasaba Digital Project",
+ "deskripsi": "Ide inovatif untuk mengembangkan satu platform digital terpadu bagi layanan pemerintahan desa serta administrasi kerja desa yang lebih cepat dan transparan.",
+ "masalah": "Proses administrasi dan pelayanan publik desa yang masih belum terintegrasi secara digital sehingga kurang efisien.",
+ "benefit": "Meningkatkan efisiensi kerja perangkat desa, transparansi data, dan kecepatan layanan kepada masyarakat."
+ },
+ {
+ "id": "cmkm710n3000xvnsw1ni3yodp",
+ "name": "I Ketut Surya",
+ "alamat": "Darmasaba, Abiansemal, Badung, Bali",
+ "namaIde": "Program CINTA – Cara Indah Tangani Sampah",
+ "deskripsi": "Inovasi pengelolaan sampah melalui mekanisme pilah, kompos, dan bursa sampah (BARES) untuk mengurangi volume sampah di desa dan meningkatkan nilai ekonomi dari sampah yang ternyata bernilai jual.",
+ "masalah": "Volume sampah meningkat yang berdampak pada lingkungan, kurangnya sistem pengelolaan sampah berbasis komunitas.",
+ "benefit": "Lingkungan desa menjadi bersih, sampah bernilai ekonomi dan peningkatan keterlibatan masyarakat dalam pengelolaan sampah."
+ }
+]
diff --git a/prisma/data/inovasi/desa-digital/desa-digital.json b/prisma/data/inovasi/desa-digital/desa-digital.json
new file mode 100644
index 00000000..8ec4f5ee
--- /dev/null
+++ b/prisma/data/inovasi/desa-digital/desa-digital.json
@@ -0,0 +1,32 @@
+[
+ {
+ "id": "cmkkshcox000504l88lp54coc",
+ "name": "Darmasaba Digital App",
+ "deskripsi": "Aplikasi digital desa yang dikembangkan oleh Pemerintah Desa Darmasaba pada tahun 2024 untuk mempermudah pelayanan publik dan informasi pemerintahan berbasis digital.
",
+ "imageName": "r_gBF0FuFpFPfSENHc4XI-mobile.webp"
+ },
+ {
+ "id": "cmkkshln8000604l8c9b5b4il",
+ "name": "D’DAMART (Darmasaba Digital Market)",
+ "deskripsi": "Sistem pasar UMKM digital berbasis website yang dikembangkan untuk meningkatkan akses pasar dan pemasaran produk UMKM Desa Darmasaba melalui platform digital.
",
+ "imageName": "uE2QwpbcXyBWxVYqCWQQT-mobile.webp"
+ },
+ {
+ "id": "cmkm1a1g80007vnsw8ejmj816",
+ "name": "Media Aspirasi dan Pengaduan Warga",
+ "deskripsi": "Media aspirasi dan pengaduan warga disediakan sebagai wadah partisipasi masyarakat dalam menyampaikan saran, masukan, maupun keluhan secara transparan dan terstruktur. Fitur ini memperkuat komunikasi dua arah antara pemerintah desa dan masyarakat, sehingga setiap aspirasi dapat ditindaklanjuti secara lebih cepat dan akuntabel.
",
+ "imageName": "c7xWNyoYp8Cak28NG5NoG-mobile.webp"
+ },
+ {
+ "id": "cmkm0w0s50003vnswmwpnqsi5",
+ "name": "Website Desa Resmi",
+ "deskripsi": "Website Desa Darmasaba berfungsi sebagai sarana utama penyampaian informasi resmi kepada masyarakat. Melalui website ini, pemerintah desa menghadirkan keterbukaan informasi publik, mempermudah akses warga terhadap berita, pengumuman, serta agenda kegiatan desa, sekaligus menjadi pusat data dan referensi terkait profil dan struktur pemerintahan desa.
",
+ "imageName": "kN09yF3sahmy-d5EaeGqA-mobile.webp"
+ },
+ {
+ "id": "cmkm1c8wx000avnswksc56orq",
+ "name": "Publikasi Kegiatan Desa Secara Digital",
+ "deskripsi": "Publikasi kegiatan desa secara digital bertujuan untuk mendokumentasikan dan menyebarluaskan berbagai aktivitas serta program kerja pemerintah desa. Melalui artikel dan dokumentasi foto, masyarakat dapat mengetahui perkembangan kegiatan desa secara terbuka, sekaligus meningkatkan kepercayaan publik terhadap pelaksanaan program desa.
",
+ "imageName": "h_Gd0SoeIJVTi_5TWUO-P-mobile.webp"
+ }
+]
diff --git a/prisma/data/inovasi/info-teknologi/info-teknologi.json b/prisma/data/inovasi/info-teknologi/info-teknologi.json
new file mode 100644
index 00000000..aac5d02f
--- /dev/null
+++ b/prisma/data/inovasi/info-teknologi/info-teknologi.json
@@ -0,0 +1,26 @@
+[
+ {
+ "id": "cmkm2xlqr000mvnswdaymiho6",
+ "name": "Darmasaba Digital App",
+ "deskripsi": "Aplikasi layanan desa berbasis teknologi untuk transparansi informasi dan layanan publik di Desa Darmasaba yang membantu warga mendapatkan informasi administratif, berita desa, dan pelayanan digital lainnya secara cepat dan mudah.
",
+ "imageName": "xVrwJgdwtcoABPU6DB__Y-mobile.webp"
+ },
+ {
+ "id": "cmkm3b1fw000pvnswpr7hgzhp",
+ "name": "Program Digitalisasi Desa",
+ "deskripsi": "Program kerja sama Desa Darmasaba bersama PT. Bali Interaktif Perkasa untuk memperkuat kapasitas pemanfaatan teknologi informasi dan komunikasi dalam administrasi desa, pelayanan publik, serta pemberdayaan digital masyarakat.
",
+ "imageName": "JjUDrfqxuEMYSAza-s7A8-mobile.webp"
+ },
+ {
+ "id": "cmkm3fwmq000tvnswejmhm7yc",
+ "name": "Pengembangan Sistem Informasi Desa",
+ "deskripsi": "Inisiatif pengembangan Sistem Informasi Desa yang mendukung pengelolaan data desa secara digital, termasuk data publik, laporan, dan statistik warga, sebagai bagian dari peningkatan kapabilitas teknologi informasi desa.
",
+ "imageName": "42RCCpBZla4ZWxXcwx7kG-mobile.webp"
+ },
+ {
+ "id": "cmkm3hjp6000wvnswkuylnf53",
+ "name": "Pelayanan Kependudukan Berbasis Digital",
+ "deskripsi": "Program untuk menyediakan layanan kependudukan secara digital, termasuk integrasi sistem administrasi kependudukan desa dengan sistem nasional, guna mempercepat layanan e-KTP, kartu keluarga, dan berkas kependudukan lainnya.
",
+ "imageName": "TrbkwnYM5rKZeHlISHCX4-mobile.webp"
+ }
+]
diff --git a/prisma/data/inovasi/kolaborasi-inovasi/kolaborasi-inovasi.json b/prisma/data/inovasi/kolaborasi-inovasi/kolaborasi-inovasi.json
new file mode 100644
index 00000000..bb7ddccc
--- /dev/null
+++ b/prisma/data/inovasi/kolaborasi-inovasi/kolaborasi-inovasi.json
@@ -0,0 +1,26 @@
+[
+ {
+ "id": "cmklzhfjb0000vnswml6z7xnj",
+ "name": "Bersama Jaga Rabies (BAJRA)",
+ "tahun": 2025,
+ "slug": "Program kolaborasi penanggulangan rabies",
+ "deskripsi": "Inovasi BAJRA (Bersama Jaga Rabies) merupakan program kolaboratif Desa Darmasaba bersama berbagai pihak, termasuk Sahabat Anti Rabies Indonesia dan Dinas Pertanian dan Pangan Kabupaten Badung, yang fokus pada edukasi publik, pelaporan cepat, serta strategis vaksinasi anjing guna upaya eliminasi rabies. Inovasi ini diangkat hingga tingkat internasional di Konferensi Rabies in Borneo 2025.
",
+ "kolaborator": "Sahabat Anti Rabies Indonesia, Bidang Kesehatan Hewan Dinas Pertanian dan Pangan Kabupaten Badung"
+ },
+ {
+ "id": "cmklzhfjb0000vnswml6z8yok",
+ "name": "Inovasi Eco-Enzyme dan Optimalisasi Tata Kelola BUMDes",
+ "tahun": 2025,
+ "slug": "Kolaborasi inovasi ekonomi & lingkungan",
+ "deskripsi": "Program kerja sama Desa Darmasaba dengan Universitas Warmadewa dan BUMDes setempat yang berfokus pada pelatihan produksi eco-enzyme ramah lingkungan dan pengembangan tata kelola BUMDes modern untuk meningkatkan kesejahteraan masyarakat berbasis usaha hijau dan ekonomi kreatif.
",
+ "kolaborator": "Universitas Warmadewa, Pengurus BUMDes Darmasaba"
+ },
+ {
+ "id": "cmklzhfjb0000vnswml6z9zpl",
+ "name": "Ekowisata dan Kuliner Berkelanjutan",
+ "tahun": 2025,
+ "slug": "Kolaborasi pengembangan ekowisata dan kuliner berbasis budaya",
+ "deskripsi": "Kegiatan yang melibatkan desa adat, BUMDes, UMKM, akademisi, dan pemerintah desa untuk mengembangkan ekowisata dan kuliner lokal berbasis keberlanjutan lingkungan serta potensi budaya setempat, sebagai wujud sinergi inovatif di Darmasaba.
",
+ "kolaborator": "Desa Adat Darmasaba, BUMDes, UMKM lokal, Akademisi"
+ }
+]
diff --git a/prisma/data/inovasi/kolaborasi-inovasi/mitra-kolaborasi.json b/prisma/data/inovasi/kolaborasi-inovasi/mitra-kolaborasi.json
new file mode 100644
index 00000000..91f49e24
--- /dev/null
+++ b/prisma/data/inovasi/kolaborasi-inovasi/mitra-kolaborasi.json
@@ -0,0 +1,17 @@
+[
+ {
+ "id": "cmkm1ziyi000dvnsweg8lp3f7",
+ "name": "TP Posyandu Bali",
+ "imageName": "qJFWokQLCaO60j0XJU_33-mobile.webp"
+ },
+ {
+ "id": "cmkm1ziyi000dvnsweg8lq4g8",
+ "name": "BRI Peduli",
+ "imageName": "nzLJoEAfl7HkpUcYa8Y1E-mobile.webp"
+ },
+ {
+ "id": "cmkm1ziyi000dvnsweg8lr5h9",
+ "name": "Universitas Warmadewa (KKN-PMM)",
+ "imageName": "JFd5C2FoaZcgDQUmvp-AO-mobile.webp"
+ }
+]
diff --git a/prisma/data/inovasi/layanan-online-desa/administrasi-online.json b/prisma/data/inovasi/layanan-online-desa/administrasi-online.json
new file mode 100644
index 00000000..e5a72af6
--- /dev/null
+++ b/prisma/data/inovasi/layanan-online-desa/administrasi-online.json
@@ -0,0 +1,44 @@
+[
+ {
+ "id": "cmkkt91ma000004lb4dpq7ll1",
+ "name": "Surat Keterangan Domisili Organisasi",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9876543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq7ll1"
+ },
+ {
+ "id": "cmkkt91ma000004lb4dpq8mm2",
+ "name": "Surat Keterangan Penghasilan",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9776543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq8mm2"
+ },
+ {
+ "id": "cmkkt91ma000004lb4dpq9nn3",
+ "name": "Surat Keterangan Tidak Mampu",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9676543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq9nn3"
+ },
+ {
+ "id": "cmkkt91ma000004lb4dpq0oo4",
+ "name": "Surat Keterangan Kelahiran",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9576543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq0oo4"
+ },
+ {
+ "id": "cmkkt91ma000004lb4dpq1pp5",
+ "name": "Surat Keterangan Usaha",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9476543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq1pp5"
+ },
+ {
+ "id": "cmkkt91ma000004lb4dpq2qq6",
+ "name": "Perizinan Berusaha OSS",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "nomorTelepon": "0361-9376543",
+ "jenisLayananId": "dnllt91ma000004lb4dpq2qq6"
+ }
+]
diff --git a/prisma/data/inovasi/layanan-online-desa/jenis-layanan.json b/prisma/data/inovasi/layanan-online-desa/jenis-layanan.json
new file mode 100644
index 00000000..acf3ce17
--- /dev/null
+++ b/prisma/data/inovasi/layanan-online-desa/jenis-layanan.json
@@ -0,0 +1,32 @@
+[
+ {
+ "id": "dnllt91ma000004lb4dpq7ll1",
+ "nama": "Surat Keterangan Domisili Organisasi",
+ "deskripsi": "Administrasi Online Surat Keterangan Domisili Organisasi"
+ },
+ {
+ "id": "dnllt91ma000004lb4dpq8mm2",
+ "nama": "Surat Keterangan Penghasilan",
+ "deskripsi": "Administrasi Online Surat Keterangan Penghasilan"
+ },
+ {
+ "id": "dnllt91ma000004lb4dpq9nn3",
+ "nama": "Surat Keterangan Tidak Mampu",
+ "deskripsi": "Administrasi Online Surat Keterangan Tidak Mampu"
+ },
+ {
+ "id": "dnllt91ma000004lb4dpq0oo4",
+ "nama": "Surat Keterangan Kelahiran",
+ "deskripsi": "Administrasi Online Surat Keterangan Kelahiran"
+ },
+ {
+ "id": "dnllt91ma000004lb4dpq1pp5",
+ "nama": "Surat Keterangan Usaha",
+ "deskripsi": "Administrasi Online Surat Keterangan Usaha"
+ },
+ {
+ "id": "dnllt91ma000004lb4dpq2qq6",
+ "nama": "Perizinan Berusaha OSS",
+ "deskripsi": "Administrasi Online Perizinan Usaha Online Single Submission (OSS)"
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/inovasi/layanan-online-desa/jenis-pengaduan.json b/prisma/data/inovasi/layanan-online-desa/jenis-pengaduan.json
new file mode 100644
index 00000000..31292ae9
--- /dev/null
+++ b/prisma/data/inovasi/layanan-online-desa/jenis-pengaduan.json
@@ -0,0 +1,26 @@
+[
+ {
+ "id": "eommt91ma000004lb4dpq7ll1",
+ "nama": "Lingkungan"
+ },
+ {
+ "id": "eommt91ma000004lb4dpq8mm2",
+ "nama": "Ketertiban Umum"
+ },
+ {
+ "id": "eommt91ma000004lb4dpq9nn3",
+ "nama": "Kesehatan Masyarakat"
+ },
+ {
+ "id": "eommt91ma000004lb4dpq0oo4",
+ "nama": "Pendidikan"
+ },
+ {
+ "id": "eommt91ma000004lb4dpq1pp5",
+ "nama": "Pembangunan Infrastuktur"
+ },
+ {
+ "id": "eommt91ma000004lb4dpq2qq6",
+ "nama": "Perizinan Berusaha"
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/inovasi/layanan-online-desa/pengaduan-masyarakat.json b/prisma/data/inovasi/layanan-online-desa/pengaduan-masyarakat.json
new file mode 100644
index 00000000..7c915a85
--- /dev/null
+++ b/prisma/data/inovasi/layanan-online-desa/pengaduan-masyarakat.json
@@ -0,0 +1,38 @@
+[
+ {
+ "id": "cmkkrxmub0004vni41cwyhid4",
+ "name": "Nyoman Putra",
+ "email": "nyoman.putra@example.com",
+ "nomorTelepon": "081234567890",
+ "nik": "3175011234567890",
+ "judulPengaduan": "Permintaan Pemasangan Spanduk Larangan Bagi Hewan",
+ "deskripsiPengaduan": "Permintaan Pemasangan Spanduk Larangan Bagi Hewan
",
+ "lokasiKejadian": "Banjar Darmasaba Tengah",
+ "jenisPengaduanId": "eommt91ma000004lb4dpq7ll1",
+ "imageName": "gyNi4s8TnK2UrViU-gN2C-mobile.webp"
+ },
+ {
+ "id": "cmkkrxmub0004vni41cwyhid5",
+ "name": "I Made Sari",
+ "email": "imade.sari@example.com",
+ "nomorTelepon": "087654321098",
+ "nik": "3175010987654321",
+ "judulPengaduan": "Laporan Anjing Liar Sering Menyerang Warga",
+ "deskripsiPengaduan": "Laporan Anjing Liar Sering Menyerang Warga
",
+ "lokasiKejadian": "Jl. Raya Darmasaba",
+ "jenisPengaduanId": "eommt91ma000004lb4dpq8mm2",
+ "imageName": "SQqSobKRg3ShvgPw_H41h-mobile.webp"
+ },
+ {
+ "id": "cmkkrxmub0004vni41cwyhid6",
+ "name": "Ketut Widi",
+ "email": "ketut.widi@example.com",
+ "nomorTelepon": "085612347890",
+ "nik": "3175011122334455",
+ "judulPengaduan": "Pengelolaan Sampah Rumah Tangga Belum Efektif",
+ "deskripsiPengaduan": "Pengelolaan Sampah Rumah Tangga Belum Efektif
",
+ "lokasiKejadian": "Banjar Bucu",
+ "jenisPengaduanId": "eommt91ma000004lb4dpq7ll1",
+ "imageName": "y78xZ2axTOjz87gRKjVAf-mobile.webp"
+ }
+]
diff --git a/prisma/data/inovasi/program-kreatif-desa/program-kreatif-desa.json b/prisma/data/inovasi/program-kreatif-desa/program-kreatif-desa.json
new file mode 100644
index 00000000..45632974
--- /dev/null
+++ b/prisma/data/inovasi/program-kreatif-desa/program-kreatif-desa.json
@@ -0,0 +1,44 @@
+[
+ {
+ "id": "cmkkyux9x000104jo4gwi9frv",
+ "name": "Pelatihan Pembuatan Eco-Enzyme",
+ "slug": "Pelatihan pembuatan cairan eco-enzyme dari limbah organik rumah tangga",
+ "deskripsi": "Program ini mengajarkan masyarakat Desa Darmasaba cara membuat eco-enzyme dari limbah organik yang dapat digunakan sebagai pembersih alami, pupuk organik, dan produk ramah lingkungan untuk meningkatkan nilai tambah ekonomi lokal.
",
+ "icon": "lingkunganSehat"
+ },
+ {
+ "id": "cmkkyux9x000104jo4gwi0gsw",
+ "name": "Green House dan Biopori Berkelanjutan",
+ "slug": "Perancangan rumah kaca dan lubang biopori di TPS3R",
+ "deskripsi": "Program perancangan dan pembangunan green house di kawasan TPS3R Pudak Mesari serta pembuatan lubang biopori untuk pengelolaan sampah organik dan konservasi air yang ramah lingkungan sebagai bagian dari pengembangan ekonomi hijau desa.
",
+ "icon": "lingkunganSehat"
+ },
+ {
+ "id": "cmkkyux9x000104jo4gwi1htx",
+ "name": "Ekowisata dan Eco-Kuliner Desa",
+ "slug": "Pengembangan paket ekowisata dan kuliner berbasis potensi lokal",
+ "deskripsi": "Kolaborasi antara desa, BUMDes, dan akademisi untuk mengembangkan potensi ekowisata berkelanjutan serta produk kuliner khas desa dengan pendekatan pemasaran digital untuk memperkuat ekonomi kreatif dan pariwisata lokal.
",
+ "icon": "ekowisata"
+ },
+ {
+ "id": "cmkkyux9x000104jo4gwi2iuy",
+ "name": "Sosialisasi Kewirausahaan",
+ "slug": "Sosialisasi kewirausahaan bagi masyarakat desa",
+ "deskripsi": "Program ini memberikan pengetahuan dan motivasi kepada warga Darmasaba tentang kewirausahaan, pemasaran digital, pencatatan usaha, dan strategi pengembangan UMKM agar produk lokal lebih produktif dan bernilai ekonomi.
",
+ "icon": "ekonomi"
+ },
+ {
+ "id": "cmkkyux9x000104jo4gwi3jvz",
+ "name": "E-Book Kuliner Lokal",
+ "slug": "Publikasi e-book tentang kuliner khas Desa Darmasaba",
+ "deskripsi": "Produksi dan distribusi e-book yang mengangkat resep dan cerita kuliner khas Desa Darmasaba sebagai media edukatif dan promosi budaya pangan lokal yang kreatif serta bisa diperluas untuk pemasaran UMKM desa.
",
+ "icon": "book"
+ },
+ {
+ "id": "cmkkyux9x000104jo4gwi4kwa",
+ "name": "Tes Kompetensi",
+ "slug": "Program pemetaan minat dan keterampilan masyarakat untuk mendukung pengembangan potensi dan kesiapan kerja.",
+ "deskripsi": "Tes Kompetensi merupakan program untuk membantu masyarakat, khususnya generasi muda, dalam mengenali minat, bakat, dan keterampilan yang dimiliki. Hasil tes dapat menjadi dasar dalam perencanaan pelatihan, pengembangan potensi diri, serta kesiapan menghadapi dunia kerja dan wirausaha.
",
+ "icon": "layananPublik"
+ }
+]
diff --git a/prisma/data/keamanan/keamanan-lingkungan/keamanan-lingkungan.json b/prisma/data/keamanan/keamanan-lingkungan/keamanan-lingkungan.json
new file mode 100644
index 00000000..472d4c03
--- /dev/null
+++ b/prisma/data/keamanan/keamanan-lingkungan/keamanan-lingkungan.json
@@ -0,0 +1,20 @@
+[
+ {
+ "id": "cmkc2tcs00002vnt9c0ssj05n",
+ "name": "Sosialisasi dan Pembinaan Keamanan Lingkungan Desa Darmasaba",
+ "deskripsi": "Pemerintah Desa Darmasaba melaksanakan Sosialisasi dan Pembinaan tentang keamanan dan ketertiban lingkungan kepada warga Perumahan Darmasaba Permai di Wantilan Perum Darmasaba Permai, Desa Darmasaba. Kegiatan ini melibatkan Perbekel Darmasaba, Bhabinkamtibmas, Babinsa, anggota BPD, LPM Desa, KBD dan KBA untuk mengajak warga berperan aktif dalam menjaga keamanan lingkungan, serta mendukung pemasangan lampu penerangan jalan guna mencegah kriminalitas dan kecelakaan di wilayah lingkungan.
",
+ "imageName": "K0wY911212dinYA3AFB_f-mobile.webp"
+ },
+ {
+ "id": "cmkc2xmdh0005vnt9ri6f4nk8",
+ "name": "Sinergi Aparat dan Masyarakat untuk Keamanan Lingkungan",
+ "deskripsi": "Desa Darmasaba bersama aparat seperti Polres Badung dan elemen masyarakat berkomitmen menjalin sinergi untuk menciptakan keamanan dan ketertiban lingkungan yang kondusif, memperkuat kepedulian serta tindakan nyata dalam menjaga situasi kamtibmas desa.
",
+ "imageName": "x0_-siY2V8IehBzo4_uph-mobile.webp"
+ },
+ {
+ "id": "cmkc36qbl0008vnt9odvekex6",
+ "name": "Peran Sistem Keamanan Lingkungan (Siskamling) dan Pecalang di Bali",
+ "deskripsi": "Sistem keamanan lingkungan (Siskamling) di Bali termasuk di Desa Darmasaba melibatkan kolaborasi antara pemerintah desa, satlinmas, dan pecalang sebagai pranata adat Bali. Sinergi ini penting untuk menjaga ketertiban masyarakat serta harmoni sosial berdasarkan kearifan lokal seperti Tri Hita Karana, meskipun perlu pembinaan dan koordinasi terus menerus dari desa dan aparat terkait.
",
+ "imageName": "TXknK9CSRSxwvM2hPW6BO-mobile.webp"
+ }
+]
diff --git a/prisma/data/keamanan/kontak-darurat-keamanan/kontak-darurat-keamanan.json b/prisma/data/keamanan/kontak-darurat-keamanan/kontak-darurat-keamanan.json
new file mode 100644
index 00000000..989d7ab9
--- /dev/null
+++ b/prisma/data/keamanan/kontak-darurat-keamanan/kontak-darurat-keamanan.json
@@ -0,0 +1,20 @@
+[
+ {
+ "id": "keamanan-polisi",
+ "nama": "Kepolisian",
+ "icon": "keamanan",
+ "kategoriId": "item-polisi"
+ },
+ {
+ "id": "keamanan-damkar",
+ "nama": "Pemadam Kebakaran",
+ "icon": "pemadam",
+ "kategoriId": "item-damkar"
+ },
+ {
+ "id": "keamanan-sar",
+ "nama": "SAR & Evakuasi",
+ "icon": "sar",
+ "kategoriId": "item-sar"
+ }
+]
diff --git a/prisma/data/keamanan/kontak-darurat-keamanan/kontakDaruratToItem.json b/prisma/data/keamanan/kontak-darurat-keamanan/kontakDaruratToItem.json
new file mode 100644
index 00000000..b3e18d3b
--- /dev/null
+++ b/prisma/data/keamanan/kontak-darurat-keamanan/kontakDaruratToItem.json
@@ -0,0 +1,17 @@
+[
+ {
+ "id": "map-polisi-1",
+ "kontakDaruratId": "keamanan-polisi",
+ "kontakItemId": "item-polsek-darmasaba"
+ },
+ {
+ "id": "map-polisi-2",
+ "kontakDaruratId": "keamanan-polisi",
+ "kontakItemId": "item-polres-badung"
+ },
+ {
+ "id": "map-damkar-1",
+ "kontakDaruratId": "keamanan-damkar",
+ "kontakItemId": "item-damkar-badung"
+ }
+]
diff --git a/prisma/data/keamanan/kontak-darurat-keamanan/kontakItem.json b/prisma/data/keamanan/kontak-darurat-keamanan/kontakItem.json
new file mode 100644
index 00000000..1c00152d
--- /dev/null
+++ b/prisma/data/keamanan/kontak-darurat-keamanan/kontakItem.json
@@ -0,0 +1,45 @@
+[
+ {
+ "id": "item-polisi",
+ "nama": "Polisi",
+ "nomorTelepon": "110",
+ "icon": "keamanan"
+ },
+ {
+ "id": "item-damkar",
+ "nama": "Pemadam Kebakaran",
+ "nomorTelepon": "113",
+ "icon": "pemadam"
+ },
+ {
+ "id": "item-sar",
+ "nama": "BASARNAS",
+ "nomorTelepon": "115",
+ "icon": "sar"
+ },
+
+ {
+ "id": "item-polsek-darmasaba",
+ "nama": "Polsek Darmasaba",
+ "nomorTelepon": "0361123456",
+ "icon": "bangunan"
+ },
+ {
+ "id": "item-polres-badung",
+ "nama": "Polres Badung",
+ "nomorTelepon": "0361123999",
+ "icon": "bangunan"
+ },
+ {
+ "id": "item-damkar-badung",
+ "nama": "Damkar Kabupaten Badung",
+ "nomorTelepon": "0361900113",
+ "icon": "pemadam"
+ },
+ {
+ "id": "item-bpbd-badung",
+ "nama": "BPBD Badung",
+ "nomorTelepon": "0361900113",
+ "icon": "sar"
+ }
+]
diff --git a/prisma/data/keamanan/laporan-publik/laporan-publik.json b/prisma/data/keamanan/laporan-publik/laporan-publik.json
new file mode 100644
index 00000000..2cb9f817
--- /dev/null
+++ b/prisma/data/keamanan/laporan-publik/laporan-publik.json
@@ -0,0 +1,16 @@
+[
+ {
+ "id": "cmkdt14my0000vn4lrvfv6jxr",
+ "judul": "LAPORAN REALISASI APBDES SEMESTER I TAHUN ANGGARAN 2025",
+ "lokasi": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung",
+ "tanggalWaktu": "2025-08-04T08:58:55.080Z",
+ "kronologi": "Pemerintah Desa Darmasaba menyampaikan realisasi pendapatan dan belanja desa hingga semester I tahun 2025 yang mencakup berbagai sumber pendapatan dan rincian belanja desa.
"
+ },
+ {
+ "id": "cmkdt14my0000vn4lrvfv7kys",
+ "judul": "Aksi Penanganan Maraknya Pembuangan Sampah Liar di Wilayah Desa Darmasaba",
+ "lokasi": "Desa Darmasaba, Kabupaten Badung",
+ "tanggalWaktu": "2025-11-24T08:58:55.080Z",
+ "kronologi": "Tim mendatangi rumah pihak yang diduga melakukan pembuangan sampah liar, melakukan pendataan dan penelusuran, serta koordinasi lintas wilayah untuk memastikan penanganan yang tepat.
"
+ }
+]
diff --git a/prisma/data/keamanan/laporan-publik/penanganan-laporan.json b/prisma/data/keamanan/laporan-publik/penanganan-laporan.json
new file mode 100644
index 00000000..170c9527
--- /dev/null
+++ b/prisma/data/keamanan/laporan-publik/penanganan-laporan.json
@@ -0,0 +1,12 @@
+[
+ {
+ "id": "cmkdt41lx0001vn4lrlcqz735",
+ "laporanId": "cmkdt14my0000vn4lrvfv6jxr",
+ "deskripsi": "Laporan ini disampaikan sebagai bentuk komitmen transparansi pengelolaan keuangan desa dan dapat dimonitor oleh masyarakat.
"
+ },
+ {
+ "id": "cmkdt41lx0002vn4lrlcqz846",
+ "laporanId": "cmkdt14my0000vn4lrvfv7kys",
+ "deskripsi": "Pemerintah Desa bersama Penyidik Lingkungan Hidup melakukan investigasi lapangan terhadap laporan masyarakat mengenai aksi pembuangan sampah liar, serta melakukan koordinasi untuk penindakan sesuai ketentuan.
"
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/keamanan/pencegahan-kriminalitas/pencegahan-kriminalitas.json b/prisma/data/keamanan/pencegahan-kriminalitas/pencegahan-kriminalitas.json
new file mode 100644
index 00000000..8854bbd1
--- /dev/null
+++ b/prisma/data/keamanan/pencegahan-kriminalitas/pencegahan-kriminalitas.json
@@ -0,0 +1,23 @@
+[
+ {
+ "id": "cmh48mn850003qq091pvs7rf1",
+ "judul": "Maling Motor di Darmasaba, Residivis Begal Didor",
+ "deskripsi": "Maling Motor di Darmasaba, Residivis Begal Didor David Andiansyah kembali terancam dibui lantaran melakukan aksi pencurian.
",
+ "deskripsiSingkat": "Maling Motor di Darmasaba, Residivis Begal Didor David Andiansyah kembali terancam dibui lantaran melakukan aksi pencurian. Mantan narapidana kasus begal ini diciduk polisi usai mencuri sepeda motor di garase rumah, kawasan Banjar Cabe, Desa Darmasaba, Abiansemal, Badung, Kamis (24/6). Kasi Humas Polres Badung Iptu Ketut Sudana menerangkan, peristiwa tersebut bermula ketika korban Ni Made Desniari berkunjung ke rumah tetangganya Putu Juliati. Saat itu korban memarkirkan sepeda motor Honda Beat miliknya di garase (TKP), tetapi kunci motornya masih nyantol di motor. Ketika korban kembali sekitar pukul 13.00, didapati motornya telah raib. Atas kejadian ini wanita asli Banjar Cabe melapor ke polisi.
",
+ "linkVideo": "https://www.youtube.com/embed/2rxK5A-KoeY"
+ },
+ {
+ "id": "cmh59no850003qq091pvs7rf1",
+ "judul": "Integrasi Digital & Akuntabilitas Desa (Pencegahan Penyalahgunaan Wewenang)",
+ "deskripsi": "Video ini membahas bagaimana integrasi digital seperti Simpades, Siskeudes, dan program Jaga Desa memperkuat akuntabilitas pemerintahan desa Darmasaba. Program tersebut mendukung transparansi dan pencegahan tindak pidana administrasi maupun penyalahgunaan wewenang karena tata kelola data desa yang baik membantu mencegah korupsi atau kecurangan dalam pengelolaan dana dan layanan desa.
",
+ "deskripsiSingkat": "Integrasi digital meningkatkan transparansi dan mencegah penyalahgunaan wewenang di pemerintahan desa.
",
+ "linkVideo": "https://www.youtube.com/embed/l7NIqjA2b_k"
+ },
+ {
+ "id": "cmh60op850003qq091pvs7rf1",
+ "judul": "PROFIL DESA DARMASABA",
+ "deskripsiSingkat": "PROFIL DESA DARMASABA Desa Darmasaba adalah permata di ujung selatan Kecamatan Abiansemal, Kabupaten Badung",
+ "deskripsi": "PROFIL DESA DARMASABA Desa Darmasaba adalah permata di ujung selatan Kecamatan Abiansemal, Kabupaten Badung, yang dikenal dengan kekayaan budaya, inovasi desa, dan potensi ekonomi yang terus berkembang. Berbatasan langsung dengan Kota Denpasar, Desa Darmasaba memiliki luas wilayah 567 hektar dengan jumlah penduduk 10.141 jiwa (per akhir Desember 2024). Video ini menampilkan potensi dan keunggulan Desa Darmasaba: 🏞️ Keindahan alam dan kawasan persawahan yang dikelola empat subak aktif. 🍽️ Keberhasilan UMKM dan kuliner lokal, termasuk produk unggulan ACK yang merambah pasar nasional. 🌱 Inovasi ketahanan pangan, pengolahan sampah ramah lingkungan (TPS3R Pudak Mesari, CINTA Darmasaba), hingga Graha Sari Boga dengan program makan bergizi gratis. 🎭 Pelestarian seni, budaya, dan tradisi, termasuk maskot Sekar Pudak dan tradisi Ngerebeg. 🏆 Prestasi desa tingkat nasional dan internasional. 📱 Transformasi digital dengan aplikasi Darmasaba Digital App, talkshow Bicara Darmasaba, hingga perpustakaan digital Pustaka Ananta Loka. Dengan motto \\\"Menggali Warisan, Merangkai Inovasi\\\", Desa Darmasaba menghadirkan 13 inovasi unggulan di bidang pemerintahan, kewilayahan, UMKM, dan kemasyarakatan. Desa ini membuktikan bahwa kolaborasi masyarakat dan pemerintah mampu mewujudkan desa yang mandiri, berdaya saing, dan berkelanjutan. 🎥 Produksi: Tim Media Kreatif Desa Darmasaba 🤝 Dukungan: Seluruh Lembaga & Elemen Masyarakat Desa Darmasaba
",
+ "linkVideo": "https://www.youtube.com/watch?v=9eCnhJvdv6A"
+ }
+]
diff --git a/prisma/data/keamanan/polsek-terdekat/layanan-polsek.json b/prisma/data/keamanan/polsek-terdekat/layanan-polsek.json
new file mode 100644
index 00000000..0d64f8c7
--- /dev/null
+++ b/prisma/data/keamanan/polsek-terdekat/layanan-polsek.json
@@ -0,0 +1,14 @@
+[
+ {
+ "id": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478",
+ "nama": "Layanan Administrasi & Surat-Menyurat"
+ },
+ {
+ "id": "b5af284c-6aa1-4442-935f-869d78eb7ecf",
+ "nama": "Penanganan Laporan & Pengaduan"
+ },
+ {
+ "id": "56b37d7f-d717-4e33-b05d-ea22b5f7af91",
+ "nama": "Perlindungan & Pengayoman Masyarakat"
+ }
+]
diff --git a/prisma/data/keamanan/polsek-terdekat/layanan-to-polsek.json b/prisma/data/keamanan/polsek-terdekat/layanan-to-polsek.json
new file mode 100644
index 00000000..299631cf
--- /dev/null
+++ b/prisma/data/keamanan/polsek-terdekat/layanan-to-polsek.json
@@ -0,0 +1,62 @@
+[
+ {
+ "id": "a50d52f2-e70f-4f29-9133-e294c40d14d3",
+ "layananId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478",
+ "polsekTerdekatId": "b92ee048-fb7f-44e9-aa5b-822e6cd0085d"
+ },
+ {
+ "id": "012454f8-f5d7-41c0-9dce-2754c0356523",
+ "layananId": "b5af284c-6aa1-4442-935f-869d78eb7ecf",
+ "polsekTerdekatId": "b92ee048-fb7f-44e9-aa5b-822e6cd0085d"
+ },
+ {
+ "id": "50736038-4ba6-47f8-8399-35b73b389f12",
+ "layananId": "56b37d7f-d717-4e33-b05d-ea22b5f7af91",
+ "polsekTerdekatId": "b92ee048-fb7f-44e9-aa5b-822e6cd0085d"
+ },
+ {
+ "id": "e2dc3487-1f62-4f63-9a12-49ac30da3372",
+ "layananId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478",
+ "polsekTerdekatId": "de769c40-10d4-4fbc-a5ef-12f2ce53a0a2"
+ },
+ {
+ "id": "47fe1f9c-4072-4203-90f9-3294d1369ac5",
+ "layananId": "b5af284c-6aa1-4442-935f-869d78eb7ecf",
+ "polsekTerdekatId": "de769c40-10d4-4fbc-a5ef-12f2ce53a0a2"
+ },
+ {
+ "id": "2cc1ba81-6b62-42ff-af21-09f8165a2dd0",
+ "layananId": "56b37d7f-d717-4e33-b05d-ea22b5f7af91",
+ "polsekTerdekatId": "de769c40-10d4-4fbc-a5ef-12f2ce53a0a2"
+ },
+ {
+ "id": "3ca2ce42-2e7d-4750-87b7-f1f52ed141de",
+ "layananId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478",
+ "polsekTerdekatId": "9a3fff54-8854-4929-b9b5-b5b2751011ea"
+ },
+ {
+ "id": "90472bca-cf3d-47ca-92e5-db43c4c7e579",
+ "layananId": "b5af284c-6aa1-4442-935f-869d78eb7ecf",
+ "polsekTerdekatId": "9a3fff54-8854-4929-b9b5-b5b2751011ea"
+ },
+ {
+ "id": "41cfc1fe-a193-446d-b574-64b1124c6f55",
+ "layananId": "56b37d7f-d717-4e33-b05d-ea22b5f7af91",
+ "polsekTerdekatId": "9a3fff54-8854-4929-b9b5-b5b2751011ea"
+ },
+ {
+ "id": "f35443e1-6aca-416d-8c55-00e3f4a0f5f9",
+ "layananId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478",
+ "polsekTerdekatId": "c2d272e1-737d-44f5-bd85-ae268cb06cbf"
+ },
+ {
+ "id": "e09797f6-82e5-4b77-946e-319eee431c8f",
+ "layananId": "b5af284c-6aa1-4442-935f-869d78eb7ecf",
+ "polsekTerdekatId": "c2d272e1-737d-44f5-bd85-ae268cb06cbf"
+ },
+ {
+ "id": "bd9b1f27-cd1b-4e23-b162-3a757745f78a",
+ "layananId": "56b37d7f-d717-4e33-b05d-ea22b5f7af91",
+ "polsekTerdekatId": "c2d272e1-737d-44f5-bd85-ae268cb06cbf"
+ }
+]
diff --git a/prisma/data/keamanan/polsek-terdekat/polsek-terdekat.json b/prisma/data/keamanan/polsek-terdekat/polsek-terdekat.json
new file mode 100644
index 00000000..97a41792
--- /dev/null
+++ b/prisma/data/keamanan/polsek-terdekat/polsek-terdekat.json
@@ -0,0 +1,54 @@
+[
+ {
+ "id": "b92ee048-fb7f-44e9-aa5b-822e6cd0085d",
+ "nama": "Kantor Polisi Abian Semal",
+ "jarakKeDesa": "9,6 Km",
+ "alamat": "ABIAN SEMAL POLICE STAT, JL. Pasar, Blahkiuh, Kec. Abiansemal, Kabupaten Badung, Bali 80352",
+ "nomorTelepon": "0361813972",
+ "jamOperasional": "Buka 24 Jam",
+ "embedMapUrl": "https://www.google.com/maps/embed?pb=!1m18!1m12!1m3!1d63127.118683990586!2d115.16592643905703!3d-8.553143643198624!2m3!1f0!2f0!3f0!3m2!1i1024!2i768!4f13.1!3m3!1m2!1s0x2dd23ceb4f6e5363%3A0xa353662f070f47d8!2sAbian%20Semal%20Police%20Station!5e0!3m2!1sid!2sid!4v1768376981008!5m2!1sid!2sid",
+ "namaTempatMaps": "Abian Semal Police Station",
+ "alamatMaps": "ABIAN SEMAL POLICE STAT, JL. Pasar, Blahkiuh, Kec. Abiansemal, Kabupaten Badung, Bali 80352",
+ "linkPetunjukArah": "https://maps.app.goo.gl/GhHVNQqffNrXSMFX7",
+ "layananPolsekId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478"
+ },
+ {
+ "id": "de769c40-10d4-4fbc-a5ef-12f2ce53a0a2",
+ "nama": "Polres Badung",
+ "jarakKeDesa": "5,8 Km",
+ "alamat": "Jl. Kebo Iwa No.1, Mengwitani, Kec. Mengwi, Kabupaten Badung, Bali 80351",
+ "nomorTelepon": "0361829949",
+ "jamOperasional": "Buka 24 Jam",
+ "embedMapUrl": "https://www.google.com/maps/embed?pb=!1m18!1m12!1m3!1d15780.907469872707!2d115.17829950197888!3d-8.574172113520685!2m3!1f0!2f0!3f0!3m2!1i1024!2i768!4f13.1!3m3!1m2!1s0x2dd238cade1488c3%3A0x918ba5cac3ef00b7!2sPolres%20Badung!5e0!3m2!1sid!2sid!4v1768377100998!5m2!1sid!2sid",
+ "namaTempatMaps": "Polres Badung",
+ "alamatMaps": "Jl. Kebo Iwa No.1, Mengwitani, Kec. Mengwi, Kabupaten Badung, Bali 80351",
+ "linkPetunjukArah": "https://maps.app.goo.gl/7yQQod4PFhpqqe7Z8",
+ "layananPolsekId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478"
+ },
+ {
+ "id": "9a3fff54-8854-4929-b9b5-b5b2751011ea",
+ "nama": "Polsek Mengwi",
+ "jarakKeDesa": "9,7 Km",
+ "alamat": "Jl. I Gusti Ngurah Rai No.110, Werdi Bhuwana, Kec. Mengwi, Kabupaten Badung, Bali 80351",
+ "nomorTelepon": "0361411270",
+ "jamOperasional": "Buka 24 Jam",
+ "embedMapUrl": "https://www.google.com/maps/embed?pb=!1m14!1m8!1m3!1d63126.100310757916!2d115.1716545!3d-8.5592871!3m2!1i1024!2i768!4f13.1!3m3!1m2!1s0x2dd23b9ddc3980af%3A0xa1f4f195483537b!2sPolsek%20Mengwi!5e0!3m2!1sid!2sid!4v1768377317955!5m2!1sid!2sid",
+ "namaTempatMaps": "Polsek Mengwi",
+ "alamatMaps": "Jl. I Gusti Ngurah Rai No.110, Werdi Bhuwana, Kec. Mengwi, Kabupaten Badung, Bali 80351",
+ "linkPetunjukArah": "https://maps.app.goo.gl/cJD44NSUdpA7Ly2m6",
+ "layananPolsekId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478"
+ },
+ {
+ "id": "c2d272e1-737d-44f5-bd85-ae268cb06cbf",
+ "nama": "Pos Polisi Ahmad Yani",
+ "jarakKeDesa": "7 Km",
+ "alamat": "Jl. Ahmad Yani Utara No.5, Peguyangan, Kec. Denpasar Utara, Kota Denpasar, Bali 80239",
+ "nomorTelepon": "-",
+ "jamOperasional": "-",
+ "embedMapUrl": "https://www.google.com/maps/embed?pb=!1m18!1m12!1m3!1d31558.650325984465!2d115.18791122296605!3d-8.612190901728288!2m3!1f0!2f0!3f0!3m2!1i1024!2i768!4f13.1!3m3!1m2!1s0x2dd23f635f1186b1%3A0xcfdde508d897fb04!2sPos%20Polisi%20Simpang%20Ahmad%20Yani!5e0!3m2!1sid!2sid!4v1768377470154!5m2!1sid!2sid",
+ "namaTempatMaps": "Pos Polisi Simpang Ahmad Yani",
+ "alamatMaps": "Jl. Ahmad Yani Utara No.5, Peguyangan, Kec. Denpasar Utara, Kota Denpasar, Bali 80239",
+ "linkPetunjukArah": "https://maps.app.goo.gl/D8HGs4mSAQqJm9KRA",
+ "layananPolsekId": "7ded635b-bf4a-4c1b-b6fa-2f13f4dfc478"
+ }
+]
diff --git a/prisma/data/keamanan/tips-keamanan/tips-keamanan.json b/prisma/data/keamanan/tips-keamanan/tips-keamanan.json
new file mode 100644
index 00000000..fcf6d4d5
--- /dev/null
+++ b/prisma/data/keamanan/tips-keamanan/tips-keamanan.json
@@ -0,0 +1,14 @@
+[
+ {
+ "id": "cmkp70zau0002vnu9o1jtpi1i",
+ "judul": "Keamanan Rumah",
+ "deskripsi": "
Pastikan pintu dan jendela selalu terkunci saat meninggalkan rumah
Pasang lampu penerangan di halaman dan area sekitar rumah untuk mencegah tindak kejahatan.
Jangan mudah memberikan akses masuk ke orang yang tidak dikenal.
",
+ "imageName": "dSe0xyvNLkP2t2f6iq-Hk-mobile.webp"
+ },
+ {
+ "id": "cmkp71pzo0005vnu9p3n9646d",
+ "judul": "Keamanan Lingkungan Tanggungjawab Bersama",
+ "deskripsi": "Pemerintah Desa Darmasaba melaksanakan sosialisasi dan pembinaan tentang keamanan dan ketertiban lingkungan kepada warga Perumahan Darmasaba Permai. Warga diajak berperan aktif dalam menjaga keamanan lingkungan serta mendukung penyediaan lampu penerangan jalan untuk mencegah tindak kriminal dan kecelakaan. Bhabinkamtibmas dan Babinsa turut memberikan materi keamanan dan ketertiban kepada warga, menekankan pentingnya partisipasi masyarakat dalam menjaga keamanan desa.
",
+ "imageName": "vwZsaxcoFWDlxG1PW7FC0-mobile.webp"
+ }
+]
\ No newline at end of file
diff --git a/prisma/data/kesehatan/infowabahpenyakit/infowabahpenyakit.json b/prisma/data/kesehatan/infowabahpenyakit/infowabahpenyakit.json
new file mode 100644
index 00000000..62a96e1a
--- /dev/null
+++ b/prisma/data/kesehatan/infowabahpenyakit/infowabahpenyakit.json
@@ -0,0 +1,37 @@
+[
+ {
+ "id": "cmkax3ptc000tvn6ytq1lpb2z",
+ "name": "Diare dan Kolera",
+ "deskripsiSingkat": "Apa itu Diare dan Kolera penyebab, gejala dan cara penanganannya?
Yuk Kenali gelaja dan cara penanganan Diare dan Kolera yang efektif untuk melindungi keluarga anda.
",
+ "deskripsiLengkap": "Apa itu Diare dan Kolera penyebab, gejala dan cara penanganannya?
Yuk Kenali gelaja dan cara penanganan Diare dan Kolera yang efektif untuk melindungi keluarga anda.
Penyebab: Bakteri Vibrio cholerae (Kolera) atau Escherichia coli (diare) akibat makanan/minuman yang terkontaminasi.
Gejala: Buang air besar cair terus-menerus, dehidrasi, dan lemas. Pencegahan: Menjaga kebersihan makanan dan air, serta mencuci tangan dengan sabun.
",
+ "imageName": "5giLSHSnWEFoZoMEcjhL7-mobile.webp"
+ },
+ {
+ "id": "cmkax5urc000wvn6yxfw0970w",
+ "name": "TBC (Tuberkulosis)",
+ "deskripsiSingkat": "Apa itu TBC penyebab, gejala dan cara penanganannya?
Yuk Kenali gelaja dan cara penanganan TBC yang efektif untuk melindungi keluarga anda.
",
+ "deskripsiLengkap": "Apa itu TBC penyebab, gejala dan cara penanganannya?
Yuk Kenali gelaja dan cara penanganan TBC yang efektif untuk melindungi keluarga anda.
Penyebab: Bakteri Mycobacterium tuberculosis yang menyebar melalui udara.
Gejala: Batuk lebih dari 2 minggu, berkeringat di malam hari, dan berat badan turun.
Pencegahan: Vaksin BCG, pola hidup sehat, dan pengobatan bagi penderita agar tidak menular.
",
+ "imageName": "3faPo-1wjhVDVU6S7S8sS-mobile.webp"
+ },
+ {
+ "id": "cmkax72s7000zvn6yz3nmvrry",
+ "name": "Demam Berdarah Dengue (DBD)",
+ "deskripsiSingkat": "Yuk Kenali gelaja dan cara penanganan DBD yang efektif untuk melindungi keluarga anda selama musim hujan.
",
+ "deskripsiLengkap": "Apa itu DBD penyebab, gejala dan cara penanganannya?
Yuk Kenali gelaja dan cara penanganan DBD yang efektif untuk melindungi keluarga anda selama musim hujan.
Penyebab: Virus dengue yang ditularkan oleh nyamuk Aedes aegypti.
Gejala: Demam tinggi, nyeri sendi, ruam kulit, dan pendarahan ringan.
Pencegahan: Menguras tempat air, menutup wadah air, fogging, dan menggunakan lotion anti-nyamuk.
",
+ "imageName": "DyX82oztXbHfu6HEvbrpt-mobile.webp"
+ },
+ {
+ "id": "cmkbyny4f0002vn67kmjmjrpl",
+ "name": "Fogging sebagai Pencegah DBD di Br. Umahanyar Desa Darmasaba",
+ "deskripsiSingkat": "Pemerintah Desa Darmasaba melaksanakan fogging di wilayah Br. Umahanyar sebagai upaya pencegahan DBD di Desa Darmasaba.
",
+ "deskripsiLengkap": "Pemerintah Desa Darmasaba melaksanakan fogging (pengasapan) di wilayah Br. Umahanyar Desa Darmasaba Kecamatan Abiansemal Kabupaten Badung dari tanggal 12 sampai dengan 13 April 2023.
Fogging ini merupakan salah satu metode yang dilakukan oleh Pemdes Darmasaba dalam pencegahan penyakit Demam Berdarah Dengue (DBD) dengan menargetkan nyamuk Aedes aegypti sebagai vektor penyebabnya.
",
+ "imageName": "pps1ZgzJxDb4VZxEvtZeu-mobile.webp"
+ },
+ {
+ "id": "cmkbyr3rx0005vn674uhycsxc",
+ "name": "Gerakan Serentak Penyemprotan Pencegahan PMK di Desa Darmasaba",
+ "deskripsiSingkat": "Penyemprotan serentak dilakukan di Desa Darmasaba untuk mencegah Penyakit Mulut dan Kaki (PMK) pada hewan ternak.
",
+ "deskripsiLengkap": "Setelah dilakukan vaksinasi Penyakit Mulut dan Kaki (PMK) pada hewan ternak yaitu sapi di wilayah Desa Darmasaba, Pemerintah Desa Darmasaba melaksanakan gerakan serentak penyemprotan pencegahan PMK pada hari Rabu (20/7/2022) di seputaran wilayah Desa Darmasaba.
Upaya ini dilakukan sebagai bentuk pencegahan terhadap penyebaran PMK dan menjaga kesehatan hewan ternak di desa.
",
+ "imageName": "JhJigMo269K1TFGzSB1OS-mobile.webp"
+ }
+]
diff --git a/prisma/data/kesehatan/kontak-darurat/kontak-darurat.json b/prisma/data/kesehatan/kontak-darurat/kontak-darurat.json
new file mode 100644
index 00000000..91d74dcf
--- /dev/null
+++ b/prisma/data/kesehatan/kontak-darurat/kontak-darurat.json
@@ -0,0 +1,30 @@
+[
+ {
+ "id": "cmkax1vks000qvn6yyxuvfsi8",
+ "name": "Puskesmas Pembantu Darmasaba",
+ "deskripsi": "Puskesmas Pembantu Darmasaba merupakan fasilitas kesehatan tingkat pertama yang berada di Desa Darmasaba, melayani berbagai layanan kesehatan masyarakat termasuk pemeriksaan umum dan imunisasi.
",
+ "imageName": "g4ICsRrmOaIqS_yqlQLZK-mobile.webp",
+ "whatsapp": "089647037430"
+ },
+ {
+ "id": "cmkawzrvg000nvn6ywyx529em",
+ "name": "UPTD Puskesmas Abiansemal III (melayani Darmasaba)",
+ "deskripsi": "Puskesmas Abiansemal III adalah fasilitas kesehatan utama di kecamatan Abiansemal yang melayani wilayah Desa Darmasaba dan sekitarnya. Puskesmas ini memiliki layanan 24 jam serta pelayanan darurat kesehatan dasar.
",
+ "imageName": "1NkzPzQailqE5yNOiUjB9-mobile.webp",
+ "whatsapp": "03618463263"
+ },
+ {
+ "id": "cmkawy5in000kvn6yza82pkkg",
+ "name": "UPTD Puskesmas Abiansemal I",
+ "deskripsi": "Puskesmas Abiansemal I melayani masyarakat di wilayah kecamatan Abiansemal, termasuk pelayanan kesehatan darurat dan program kesehatan masyarakat.
",
+ "imageName": "NBPAqjPXn7GQmYTDBI5hu-mobile.webp",
+ "whatsapp": "087858367111"
+ },
+ {
+ "id": "cmkb6ehu20003vn14ca4xr057",
+ "name": "Kantor Desa Darmasaba (Kontak Informasi Kesehatan)",
+ "deskripsi": "Kantor Pemerintahan Desa Darmasaba dapat menjadi saluran kontak awal untuk rujukan layanan kesehatan darurat atau informasi lebih lanjut mengenai fasilitas kesehatan di wilayah desa.
",
+ "imageName": "EcQIGOF6LW1dIKE53vmba-mobile.webp",
+ "whatsapp": "081239580000"
+ }
+]
diff --git a/prisma/data/kesehatan/penanganan-darurat/penganan-darurat.json b/prisma/data/kesehatan/penanganan-darurat/penganan-darurat.json
new file mode 100644
index 00000000..d2c9cf38
--- /dev/null
+++ b/prisma/data/kesehatan/penanganan-darurat/penganan-darurat.json
@@ -0,0 +1,26 @@
+[
+ {
+ "id": "cmkawso7y000evn6ygob15cqb",
+ "name": "Rembug Stunting di Desa Darmasaba",
+ "deskripsi": "Pemerintah Desa Darmasaba melaksanakan kegiatan rembug stunting dengan melibatkan bidan desa, kader posyandu, dan tokoh masyarakat. Tujuan kegiatan ini adalah untuk memperkuat upaya pencegahan kekerdilan (stunting) melalui koordinasi layanan kesehatan, edukasi gizi, serta percepatan penanganan gizi buruk di lingkungan desa sebagai bagian dari respons terhadap kondisi kesehatan yang mendesak.
",
+ "imageName": "Gi8EX3pBmT719AfzXirDS-mobile.webp"
+ },
+ {
+ "id": "cmkawq3ef000bvn6y387vub0y",
+ "name": "Posko Kesehatan Darurat dan Bencana",
+ "deskripsi": "Posko Kesehatan Darurat dan Bencana Desa Darmasaba dibentuk sebagai pusat koordinasi dan pertolongan bagi warga yang terdampak situasi darurat seperti banjir, tanah longsor, atau wabah penyakit. Posko ini dilengkapi dengan tenaga medis, obat-obatan dasar, serta dukungan logistik untuk memastikan penanganan cepat dan tepat sasaran. Kegiatan ini juga melibatkan kader kesehatan desa dan karang taruna sebagai relawan lapangan.
",
+ "imageName": "v7Ac2xQvTiJy-HYh1AxF4-mobile.webp"
+ },
+ {
+ "id": "cmkawso7y000evn6ygob14bpa",
+ "name": "Layanan Ambulans Desa Darmasaba",
+ "deskripsi": "Layanan Ambulans Desa Darmasaba disiapkan untuk membantu masyarakat yang membutuhkan transportasi medis darurat ke fasilitas kesehatan terdekat. Layanan ini beroperasi 24 jam dan dapat dihubungi melalui nomor darurat desa. Tim ambulans terdiri dari relawan terlatih dan tenaga medis yang siap memberikan pertolongan pertama di lokasi kejadian sebelum dirujuk ke rumah sakit atau puskesmas.
",
+ "imageName": "jYxEXspWH5g6eTTVqK72c-mobile.webp"
+ },
+ {
+ "id": "cmkawu7te000hvn6yh3pdnv4w",
+ "name": "Penanganan Darurat Sosial & Kesehatan Desa Darmasaba",
+ "deskripsi": "Program Penanganan Darurat Sosial & Kesehatan Desa Darmasaba bertujuan memberikan respon cepat terhadap situasi darurat seperti warga sakit mendadak, kecelakaan, bencana alam, maupun kondisi sosial yang membutuhkan bantuan segera. Tim Siaga Desa Darmasaba berkoordinasi dengan Puskesmas Abiansemal dan BPBD untuk memastikan penanganan yang cepat, tepat, dan manusiawi. Program ini juga mencakup layanan ambulans desa, posko kesehatan darurat, serta bantuan logistik bagi warga terdampak.
",
+ "imageName": "3tNQ9J8I3Ewq5H8CWuqvp-mobile.webp"
+ }
+]
diff --git a/prisma/data/kesehatan/posyandu/posyandu.json b/prisma/data/kesehatan/posyandu/posyandu.json
new file mode 100644
index 00000000..8c3aac84
--- /dev/null
+++ b/prisma/data/kesehatan/posyandu/posyandu.json
@@ -0,0 +1,10 @@
+[
+ {
+ "id": "cmkanjnmx0006vntz1cn7owpb",
+ "name": "Posyandu Pudak Amara",
+ "nomor": "(0361) 8463263",
+ "deskripsi": "Posyandu Pudak Amara merupakan salah satu posyandu aktif di Desa Darmasaba dan pernah berkompetisi dalam lomba kader dan posyandu berprestasi tingkat Provinsi Bali tahun 2025.
Kegiatan ini melibatkan kader posyandu serta didampingi pihak desa dan puskesmas setempat untuk meningkatkan pelayanan kesehatan ibu dan anak.
",
+ "jadwalPelayanan": "Setiap bulan pada satu hari tertentu (mis. minggu ke-2): 08:00 – 12:00 WITA (posyandu balita & ibu hamil)
",
+ "imageName": "TDQReg1lQ73s39crXW0ra-mobile.webp"
+ }
+]
diff --git a/prisma/data/kesehatan/program-kesehatan/program-kesehatan.json b/prisma/data/kesehatan/program-kesehatan/program-kesehatan.json
new file mode 100644
index 00000000..770b44d1
--- /dev/null
+++ b/prisma/data/kesehatan/program-kesehatan/program-kesehatan.json
@@ -0,0 +1,51 @@
+[
+ {
+ "id": "cmkawkji50002vn6yzyrlqhh1",
+ "name": "Gerakan Kulkul PKK dan Posyandu Desa Darmasaba",
+ "deskripsiSingkat": "Kegiatan bersama PKK dan Posyandu untuk meningkatkan pelayanan kesehatan masyarakat.
",
+ "deskripsi": "Pada hari Minggu, 11 Januari 2025, Pemerintah Desa Darmasaba melalui TP PKK dan TP Posyandu melaksanakan kegiatan Gerakan Kulkul PKK dan Posyandu yang berlangsung serentak di seluruh wilayah Desa Darmasaba untuk memperkuat pelayanan kesehatan dasar dan peningkatan partisipasi masyarakat dalam program Posyandu.
",
+ "imageName": "hLeF0GRFZqDUngZnDMAAk-mobile.webp"
+ },
+ {
+ "id": "cmkawmlg40005vn6yja2xiev0",
+ "name": "Pendampingan Kunjungan Rumah oleh Puskesmas Abiansemal 3",
+ "deskripsiSingkat": "Pendataan kesehatan penyandang disabilitas lewat kunjungan rumah di Desa Darmasaba.
",
+ "deskripsi": "Pemerintah Desa Darmasaba bersama Kelian Banjar Dinas dan kader kesehatan mendampingi kegiatan kunjungan rumah yang dilaksanakan oleh Puskesmas Abiansemal 3 pada 21 Juli 2025, difokuskan pada pendataan dan pemantauan kondisi kesehatan penyandang disabilitas di Banjar Bersih, Desa Darmasaba.
",
+ "imageName": "hyyTFi8EApjzFEZ9EvJgB-mobile.webp"
+ },
+ {
+ "id": "cmkawnr9k0008vn6ymwv0foiv",
+ "name": "Kegiatan Aksi Sosial Tim Penggerak Posyandu Provinsi Bali di Desa Darmasaba",
+ "deskripsiSingkat": "Aksi sosial TP Posyandu Bali untuk memperkuat pelayanan posyandu di desa.
",
+ "deskripsi": "Pada 10 Desember 2025, Desa Darmasaba menjadi lokasi pelaksanaan Aksi Sosial Tim Penggerak Posyandu Provinsi Bali yang bertujuan memperkuat pelayanan Posyandu serta meningkatkan kesejahteraan masyarakat, khususnya keluarga dan balita.
",
+ "imageName": "l4qsUEw2JiclGAkkrXp9g-mobile.webp"
+ },
+ {
+ "id": "cmkawnr9k0008vn6ymwv0dpjw",
+ "name": "Inovasi BAJRA dalam Penanggulangan Rabies",
+ "deskripsiSingkat": "Program BAJRA untuk penanggulangan rabies di Desa Darmasaba.
",
+ "deskripsi": "Desa Darmasaba mengembangkan inovasi BAJRA (Bersama Jaga Rabies), sebuah program berbasis komunitas untuk penanggulangan rabies yang mengintegrasikan pelaporan cepat masyarakat, edukasi berkelanjutan dan koordinasi lintas sektor antara kesehatan hewan, manusia, dan pemerintahan desa.
",
+ "imageName": "Gc79mlIlGuoRQuTqskFj--mobile.webp"
+ },
+ {
+ "id": "cmkawnr9k0008vn6ymwv0eqkx",
+ "name": "Posyandu Pudak Amara Berkompetisi",
+ "deskripsiSingkat": "Partisipasi Posyandu Pudak Amara dalam lomba prestasi Posyandu tingkat provinsi.
",
+ "deskripsi": "Kader Posyandu Pudak Amara Br. Cabe mendapat pendampingan dari Perbekel Darmasaba, Dinas Kesehatan Kab. Badung, Puskesmas Abiansemal III, dan Pustu Desa Darmasaba dalam ajang lomba kader dan Posyandu berprestasi tingkat Provinsi Bali tahun 2025.
",
+ "imageName": "OsMY3AYPyGC_CoN1xUjOn-mobile.webp"
+ },
+ {
+ "id": "cmkawnr9k0008vn6ymwv1frly",
+ "name": "Outbound Kader Posyandu Darmasaba",
+ "deskripsiSingkat": "Program pembinaan dan pengembangan kapasitas kader Posyandu.
",
+ "deskripsi": "Pemdes Darmasaba melaksanakan kegiatan Outbound Posyandu untuk meningkatkan kapasitas dan wawasan Kader Posyandu se-Desa Darmasaba sebagai bagian dari upaya peningkatan kualitas pelayanan kesehatan dasar di masyarakat.
",
+ "imageName": "M9QlgVKIEfCdY3g4F_tRZ-mobile.webp"
+ },
+ {
+ "id": "cmkdu8ki10004vn4lpbxm2zqo",
+ "name": "PEMBANGUNAN JAMBAN BAGI MASYARAKAT",
+ "deskripsiSingkat": "Program pengadaan jamban bagi Masyarakat ini diharapkan menjadi stimulus agar masyarakat peduli terhadap lingkungan sehat sehingga Badung Open Defection Free atau terbebas dari buang air besar di tempat terbuka dapat terwujud.
",
+ "deskripsi": "Desa Darmasaba sebagai desa yang berkomitmen selalu selaras dengan pembangunan Pemerintah Kabupaten Badung pada tahun anggaran 2023 ini turut ambil bagian dalam menyukseskan program Bupati Badung I Nyoman Giri Prasta, S.Sos dalam bidang kesehatan sanitasi masyarakat. Program pengadaan jamban bagi Masyarakat ini diharapkan menjadi stimulus agar masyarakat peduli terhadap lingkungan sehat sehingga Badung Open Defection Free atau terbebas dari buang air besar di tempat terbuka dapat terwujud.
Pemberian bantuan jamban ini dilaksanakan di 11 banjar dengan menyasar 22 keluarga yang memang belum memiliki jamban yang sumber dananya sepenuhnya dari APBDes Darmasaba T. A. 2023. Pembangunan Jamban bagi Masyarakat ini juga menjadi bukti komitmen Pemerintah Desa Darmasaba dalam melaksanakan salah satu visi mewujudkan masyarakat yang sejahtera dan berbudaya untuk menjaga lingkungan yang bersih dan sehat.
",
+ "imageName": "6DQbAvn0St-xHdPGW3vpY-mobile.webp"
+ }
+]
diff --git a/prisma/data/kesehatan/puskesmas/jam-puskesmas/jam.json b/prisma/data/kesehatan/puskesmas/jam-puskesmas/jam.json
new file mode 100644
index 00000000..b59aa3e6
--- /dev/null
+++ b/prisma/data/kesehatan/puskesmas/jam-puskesmas/jam.json
@@ -0,0 +1,14 @@
+[
+ {
+ "id": "cmkao2zwx0008vntzmvqdsdzo",
+ "workDays": "09:00",
+ "weekDays": "17:00",
+ "holiday": "08:00 - 16:00"
+ },
+ {
+ "id": "cmkao2zwx0008vntzmvqdseal",
+ "workDays": "08:00",
+ "weekDays": "12:00",
+ "holiday": "–"
+ }
+]
diff --git a/prisma/data/kesehatan/puskesmas/kontak-puskesmas/kontak.json b/prisma/data/kesehatan/puskesmas/kontak-puskesmas/kontak.json
new file mode 100644
index 00000000..df7263ae
--- /dev/null
+++ b/prisma/data/kesehatan/puskesmas/kontak-puskesmas/kontak.json
@@ -0,0 +1,16 @@
+[
+ {
+ "id": "cmkao2zxc0009vntz00kev051",
+ "kontakPuskesmas": "(0361) 8463263",
+ "email": "puskesmas@gmail.com",
+ "facebook": "puskesmas@gmail.com",
+ "kontakUGD": "(0361) 8463263"
+ },
+ {
+ "id": "cmkao2zxc0009vntz00kev162",
+ "kontakPuskesmas": "–",
+ "email": "–",
+ "facebook": "–",
+ "kontakUGD": "–"
+ }
+]
diff --git a/prisma/data/kesehatan/puskesmas/puskesmas.json b/prisma/data/kesehatan/puskesmas/puskesmas.json
new file mode 100644
index 00000000..df818aac
--- /dev/null
+++ b/prisma/data/kesehatan/puskesmas/puskesmas.json
@@ -0,0 +1,18 @@
+[
+ {
+ "id": "cmkao2zxk000bvntzbavkbg6p",
+ "name": "Puskesmas Abiansemal III",
+ "alamat": "Jl. Ratna, Sibang Kaja, Abiansemal, Badung, Bali 80352",
+ "jamId": "cmkao2zwx0008vntzmvqdsdzo",
+ "imageName": "d6hJgycQawWN3VEcHaqtR-mobile.webp",
+ "kontakId": "cmkao2zxc0009vntz00kev051"
+ },
+ {
+ "id": "cmkao2zxk000bvntzbavkbh7q",
+ "name": "Puskesmas Pembantu Darmasaba",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "jamId": "cmkao2zwx0008vntzmvqdseal",
+ "imageName": "cg78Sb_QzZFlli9s2FPVc-mobile.webp",
+ "kontakId": "cmkao2zxc0009vntz00kev162"
+ }
+]
diff --git a/prisma/data/landing-page/prestasi-desa/prestasi-desa.json b/prisma/data/landing-page/prestasi-desa/prestasi-desa.json
index f9d9bff7..645964d7 100644
--- a/prisma/data/landing-page/prestasi-desa/prestasi-desa.json
+++ b/prisma/data/landing-page/prestasi-desa/prestasi-desa.json
@@ -1,20 +1,30 @@
[
{
- "id": "cmdwrrxkh0005vnd3p5rxkiev",
- "name": "Tim Bola Voli Putri Dharma Temaja meraih juara 3 dalam Turnamen Bola Voli Mangupura Cup 2024 kategori Putri Se-Bali",
- "deskripsi": "Tim Bola Voli Putri Dharma Temaja meraih juara 3 dalam Turnamen Bola Voli Mangupura Cup 2024 kategori Putri Se-Bali
",
- "kategoriId": "cmdwrolsl0000vnd3e24q5440"
+ "id": "cmdws0sgq000bvnd32o7m94im",
+ "name": "Desa Darmasaba Raih Juara 1 Lomba Desa Tingkat Provinsi Bali Tahun 2025",
+ "deskripsi": "Pemerintah Desa Darmasaba meraih prestasi gemilang dengan menyabet Juara 1 Lomba Desa Tingkat Provinsi Bali Tahun 2025 dalam Peringatan Hari Jadi Pemerintah Provinsi Bali ke-67. Inovasi pelayanan publik dan sinergi masyarakat menjadi kunci keberhasilan.
",
+ "kategoriId": "cmdwrp0pr0002vnd35w6nkjh0",
+ "imageName": "_CIAQk1Yem-_6LFjrkmVg-mobile.webp"
},
{
"id": "cmdwrzs740008vnd329ysez5x",
- "name": "Prestasi Juara 3 dalam Lomba Keluarga Sadar Hukum Kabupaten Badung Tahun 2024",
- "deskripsi": "Prestasi Juara 3 dalam Lomba Keluarga Sadar Hukum Kabupaten Badung Tahun 2024
",
- "kategoriId": "cmdwrot900001vnd30b5kj96g"
+ "name": "Desa Darmasaba Raih Juara 1 Lomba Desa Tingkat Kabupaten Badung Tahun 2025",
+ "deskripsi": "Desa Darmasaba berhasil meraih Juara 1 Lomba Desa Tingkat Kabupaten Badung Tahun 2025 yang diserahkan langsung oleh Bupati Badung dalam rangka memperingati HUT RI ke-80. Prestasi ini menjadi simbol kerja keras dan inovasi desa.
",
+ "kategoriId": "cmdwrp0pr0002vnd35w6nkjh0",
+ "imageName": "pxpQFGBkTRpFOwJFTwjbD-mobile.webp"
},
{
- "id": "cmdws0sgq000bvnd32o7m94im",
- "name": "Peringkat 5 Dalam Ajang Bergengsi Mangupura Award",
- "deskripsi": "Peringkat 5 Dalam Ajang Bergengsi Mangupura Award
",
- "kategoriId": "cmdwrp0pr0002vnd35w6nkjh0"
+ "id": "cmdwrrxkh0005vnd3p5rxkiev",
+ "name": "TPS3R Pudak Mesari Desa Darmasaba Raih Juara 2 dalam Kompetisi TPS3R Kabupaten Badung",
+ "deskripsi": "TPS3R Pudak Mesari Desa Darmasaba meraih Juara 2 dalam Kompetisi TPS3R tingkat Kabupaten Badung pada acara HUT Mangupura ke-16 sebagai wujud komitmen pengelolaan sampah berbasis masyarakat.
",
+ "kategoriId": "cmdwrp0pr0002vnd35w6nkjh0",
+ "imageName": "3rshaPxCgk5unArWbKPpk-mobile.webp"
+ },
+ {
+ "id": "cmk3nxfkd0005vn9bcw8d10sh",
+ "name": "Keluarga Sadar Hukum Desa Darmasaba Raih Prestasi Gemilang (Juara 3)",
+ "deskripsi": "KADARKUM Desa Darmasaba berhasil meraih Juara 3 dalam Lomba Keluarga Sadar Hukum Kabupaten Badung tahun 2024, bukti semangat kolaborasi masyarakat dengan perangkat desa dalam pendidikan hukum.
",
+ "kategoriId": "cmdwrp0pr0002vnd35w6nkjh0",
+ "imageName": "D4za5fHh_y92Bj9qZkp5j-mobile.webp"
}
]
\ No newline at end of file
diff --git a/prisma/data/landing-page/profile/mediaSosial.json b/prisma/data/landing-page/profile/mediaSosial.json
index 9af092a0..799990bb 100644
--- a/prisma/data/landing-page/profile/mediaSosial.json
+++ b/prisma/data/landing-page/profile/mediaSosial.json
@@ -3,24 +3,24 @@
"id": "cmds9023u0008vnbe3oxmhwyf",
"name": "Desa Darmasaba",
"iconUrl": "https://www.youtube.com/channel/UCtPw9WOQO7d2HIKzKgel4Xg",
- "imageId": "cmff3joae0000vn6h8sgs0ilg"
+ "imageName": "q1G995W7cLkC_qquLTlKN-mobile.webp"
},
{
"id": "cmds90oul000bvnbe2bqkptoi",
"name": "Pemerintah Desa Darmasaba",
"iconUrl": "https://www.facebook.com/DarmasabaDesaku",
- "imageId": "cmff3mtat0002vn6hs8vyyhdd"
+ "imageName": "I6mlQ4nRmPX26gm79C_rM-mobile.webp"
},
{
"id": "cmds91i4e000evnbe8gtf1gub",
"name": "ddarmasaba",
"iconUrl": "https://www.instagram.com/ddarmasaba/",
- "imageId": "cmff3oouh0004vn6hd94brzv9"
+ "imageName": "WArLC_yvU33MjoqEnQeQ1-mobile.webp"
},
{
"id": "cmds92de5000hvnbemlu6sq5x",
"name": "desa.darmasaba",
"iconUrl": "https://www.tiktok.com/@desa.darmasaba?is_from_webapp=1&sender_device=pc",
- "imageId": "cmff3q12g0005vn6h5ojov2qa"
+ "imageName": "D3RPbNiaNSCjacLjeR_qO-mobile.webp"
}
]
diff --git a/prisma/data/landing-page/profile/profile.json b/prisma/data/landing-page/profile/profile.json
index a83a8f62..31cab010 100644
--- a/prisma/data/landing-page/profile/profile.json
+++ b/prisma/data/landing-page/profile/profile.json
@@ -3,6 +3,6 @@
"id": "edit",
"name": "I.B Surya Prabhawa Manuaba, S.H., M.H.",
"position": "Perbekel Darmasaba periode 2021-2027",
- "imageId": "cmff2w5ly000avn0telhct71k"
+ "imageName": "Eqlrr1W-pK8ShMGqgPGL3-mobile.webp"
}
]
diff --git a/prisma/data/landing-page/profile/programInovasi.json b/prisma/data/landing-page/profile/programInovasi.json
index f6603f6d..aadd4e79 100644
--- a/prisma/data/landing-page/profile/programInovasi.json
+++ b/prisma/data/landing-page/profile/programInovasi.json
@@ -1,51 +1,58 @@
[
{
- "id": "cmdr755pf0005vn5rp8tyuubw",
- "name": "Dmangan",
- "description": "Darmasaba Aman Pangan",
- "link": "https://darmasaba.desa.id/berita/61452-kader-d-mangan-berhasil-meraih-prestasi-dalam-ajang-lomba-banjar-bali-quis-bbq-tahun-2024",
- "imageId" : "cmff0z34f0005vn0tjtvq519p"
+ "id": "cmdr77vbw000bvn5rvpmoq31s",
+ "name": "Bares",
+ "description": "BARES (Darmasaba Recycling Waste Stock/Exchange) adalah Bursa Sampah yang merupakan sebuah sistem inovatif untuk mengelola dan menukar sampah daur ulang menjadi nilai ekonomis melalui TPS3R Pudak Mesari, dengan tujuan memberdayakan masyarakat dan menjaga kebersihan lingkungan, menjadikannya bagian dari upaya lebih besar seperti program desa pintar (smart village) dan pengelolaan sampah terintegrasi.
",
+ "link": "https://darmasaba.desa.id/berita/56722-bares#:~:text=Sampah%20jadi%20uang%20nie%20Semeton,masing%20banjar%20se%2DDesa%20Darmasaba.",
+ "imageName" : "thpgPSJkBxUIRajZt3AVo-mobile.webp"
},
{
"id": "cmdr76nqk0008vn5rdddvcxnr",
"name": "Bicara Darmasaba",
- "description": "Bicara Darmasaba",
- "link": "https://darmasaba.desa.id/berita/42506-bicara-darmasaba",
- "imageId" : "cmff0tnf00003vn0t3kgzi0u0"
+ "description": "Bicara Darmasaba adalah program ruang dialog terbuka yang digagas Pemerintah Desa Darmasaba sebagai forum strategis untuk menggali ide, menyampaikan aspirasi masyarakat, serta merangkai aksi nyata dalam menyelesaikan persoalan desa, seperti pengelolaan sampah. Kegiatan ini diselenggarakan secara berkala dengan pendekatan partisipatif dan ditayangkan secara live streaming agar transparan dan melibatkan banyak pihak, sehingga mendukung tata kelola desa yang responsif terhadap kebutuhan warga.
",
+ "link": "https://badungkab.go.id/kab/berita/67159-bicara-darmasaba-bahas-berbagai-persoalan-warga-didorong-berani-speak-up-sampaikan-aspirasi-dan-cari-solusi-nyata",
+ "imageName" : "ubna9N6r7RgVWN5plO5mq-mobile.webp"
},
{
- "id": "cmdr77vbw000bvn5rvpmoq31s",
- "name": "Bares",
- "description": "Darmasaba Recycling Stock/Exchange",
- "link": "http://darmasaba.desa.id/berita/56722-bares",
- "imageId" : "cmff0rr4z0002vn0twp333m2"
+ "id": "cmdr755pf0005vn5rp8tyuubw",
+ "name": "Dmangan",
+ "description": "Dmangan (Darmasaba Aman Pangan) adalah program inovasi Desa Darmasaba yang fokus pada ketahanan pangan masyarakat desa melalui pengembangan produksi hasil pangan lokal, peningkatan kemandirian pangan, serta pembinaan dan pemberdayaan warga dalam praktik pertanian dan pangan berkelanjutan.
",
+ "link": "https://darmasaba.desa.id/berita/51787-desa-darmasaba-juara-1-lomba-desa-pangan-aman-provinsi-bali-tahun-2023#:~:text=Salah%20satu%20program%20inovasi%20Pemdes%20Darmasaba%20yaitu,Darmasaba%20dan%20seluruh%20stakeholder%20terkait%20yang%20telah",
+ "imageName" : "y4yaE4XdUP1TSUGhWPW9h-mobile.webp"
},
{
- "id": "cmdr7bxtp000evn5rmy85wihx",
- "name": "Sajjana Dharma Raksaka",
- "description": "Sajjana Dharma Raksaka",
- "link": "https://ppid.badungkab.go.id/storage/dokumen/5RS9dldGkrgzMQq6bKdZsqsVRHI8gffWv4PGfb3r.pdf",
- "imageId" : "cmff10cwq0009vn0tse8dzu3j"
- },
- {
- "id": "cmdr7dlnk000hvn5r9lur3z35",
- "name": "PDKT",
- "description": "Perangkat Desa Kuat Teknologi",
- "link": "https://darmasaba.desa.id/berita/53752-p-d-k-t",
- "imageId" : "cmff1013m0008vn0th7t0d64d"
+ "id": "cmk228ust0009vnev5p8i377o",
+ "name": "Davest",
+ "description": "DAVEST (Darmasaba Investment) merupakan program inovasi Desa Darmasaba yang bertujuan mempromosikan potensi investasi desa secara terintegrasi melalui media digital dan pendampingan langsung. Program ini menjadi sarana penghubung antara pemerintah desa, pelaku usaha, dan investor dalam rangka mendorong pertumbuhan ekonomi desa yang berkelanjutan.
DAVEST menyajikan informasi potensi unggulan desa seperti sektor UMKM, pariwisata, ekonomi kreatif, serta peluang investasi berbasis sumber daya lokal dengan prinsip transparansi dan kemudahan akses informasi.
Di tahun 2024 ini Davest (Darmasaba Village Festival) akan diadakan lagi, dengan berbagai kegiatan pemerdayaan, edukasi dan hiburan yang tentunya lebih waahhhh dari dua tahun lalu. Untuk memantapkan hal tersebut, Pemdes Darmasaba melakukan rapat koordinasi (rakor) Davest 2024 yang dipimpin langsung oleh Perbekel Darmasaba I. B. Surya Prabhawa Manuaba, S.H.,M.H. pada hari Senin (22/1/2024) bertempat di Ruang Shanti Gosana Kantor Perbekel Darmasaba.
Tujuan Program Meningkatkan daya tarik investasi di Desa Darmasaba
Mempromosikan potensi unggulan desa secara profesional
Mendorong pertumbuhan ekonomi dan penciptaan lapangan kerja
Mendukung visi Desa Darmasaba sebagai desa inovatif dan berdaya saing
Sasaran Program Calon investor lokal dan regional
Pelaku UMKM dan kelompok usaha desa
Masyarakat Desa Darmasaba
Bentuk Inovasi Ruang Lingkup Kegiatan Penyusunan profil potensi investasi desa
Digitalisasi informasi investasi desa
Promosi peluang investasi melalui media online
Fasilitasi komunikasi antara investor dan desa
Pendampingan awal investasi berbasis desa
",
+ "link": "https://darmasaba.desa.id/berita/55862-rakor-davest-2024",
+ "imageName" : "Z4i2RRnnlHq2iWj94ldyo-mobile.webp"
},
{
"id": "cmdr7ftob000mvn5rfhgdtg8v",
"name": "GM",
- "description": "Galah Melah",
- "link": "https://darmasaba.desa.id/berita/52880-galah-melah",
- "imageId" : "cmff38cyq000bvn0t9f01cz3f"
+ "description": "Galah Melah (Gerak dan Langkah Memilah Sampah) adalah program inovasi pengelolaan sampah yang dikembangkan oleh Pemerintah Desa Darmasaba melalui TPS3R Pudak Mesari, dengan fokus pada praktik pemisahan sampah dari sumbernya (di rumah, banjar, maupun fasilitas umum).
",
+ "link": "https://mcinews.id/2025/09/12/serius-tangani-sampah-desa-darmasaba-terapkan-inovasi-dan-osaki-composting-system-di-tps-3r-pudak-mesari-dari-2023",
+ "imageName" : "Vr7CoaYDpk2dIkHx9PxRj-mobile.webp"
},
{
"id": "cmdr7glue000pvn5r6onzslju",
"name": "Inovasi Desa Darmasaba",
- "description": "Inovasi Desa Darmasaba",
- "link": "https://darmasaba.desa.id/produk-lokal-desa",
- "imageId" : "cmff0zqvd0007vn0tv6o5hjcq"
+ "description": "Inovasi Desa Darmasaba adalah kumpulan program inovatif yang dikembangkan oleh Pemerintah Desa Darmasaba untuk memperkuat penyelenggaraan pemerintahan desa, pemberdayaan masyarakat, pengelolaan lingkungan, serta peningkatan kualitas sosial-ekonomi dan budaya desa.
",
+ "link": "https://mcinews.id/2025/09/11/inovasi-desa-darmasaba-lanjutkan-perjuangan-ke-tingkat-nasional-dan-diakui-negara-tetangga",
+ "imageName" : "ceoB_sg-HOzljN8j_2nZA-mobile.webp"
+ },
+ {
+ "id": "cmdr7dlnk000hvn5r9lur3z35",
+ "name": "PDKT",
+ "description": "PDKT (Perangkat Desa Kuat Teknologi) merupakan program inovasi Desa Darmasaba yang dirancang untuk menguatkan kapasitas dan kompetensi perangkat desa dalam memanfaatkan teknologi informasi dan komunikasi guna mendukung tata kelola pemerintahan desa yang modern, transparan, cepat, dan responsif.
",
+ "link": "https://darmasaba.desa.id/berita/67168-desa-darmasaba-masuk-5-besar-nasional-dalam-ajang-pemerintah-desa-dan-kelurahan-award-2025",
+ "imageName" : "vOy5YVUXfHXfiFOHylIN7-mobile.webp"
+ },
+ {
+ "id": "cmdr7bxtp000evn5rmy85wihx",
+ "name": "Sajjana Dharma Raksaka",
+ "description": "Sajjana Dharma Raksaka adalah program inovasi Desa Darmasaba yang bertujuan untuk memperkuat akses perlindungan hukum dan pendampingan hukum non-litigasi bagi masyarakat desa. Melalui inovasi ini, desa menyediakan layanan pendampingan, mediasi, dan bantuan dalam menyelesaikan persoalan hukum secara bijaksana, humanis, dan berperspektif keadilan sosial
",
+ "link": "https://ppid.badungkab.go.id/storage/dokumen/5RS9dldGkrgzMQq6bKdZsqsVRHI8gffWv4PGfb3r.pdf",
+ "imageName" : "gE_qcqIbY0mqI6FV9V4CL-mobile.webp"
}
]
diff --git a/prisma/data/landing-page/sdgs-desa/sdgs-desa.json b/prisma/data/landing-page/sdgs-desa/sdgs-desa.json
index 9f1fc623..d8969097 100644
--- a/prisma/data/landing-page/sdgs-desa/sdgs-desa.json
+++ b/prisma/data/landing-page/sdgs-desa/sdgs-desa.json
@@ -3,109 +3,109 @@
"id": "cmdsjzdl30002vneknuvo4irv",
"name": "Desa Tanpa Kemiskinan",
"jumlah": "52.62",
- "imageId": ""
+ "imageName": "kxpXwgyx74-HkAGXML7Hv-mobile.webp"
},
{
"id": "cmdskargd0005vnek0mu2ofk9",
"name": "Desa Tanpa Kelaparan",
"jumlah": "35.75",
- "imageId": ""
+ "imageName": "D0DEORcSkSWXEpY29mh_x-mobile.webp"
},
{
"id": "cmdskbvl0008vnek5dmieatb",
"name": "Desa Sehat Dan Sejahtera",
"jumlah": "77.37",
- "imageId": ""
+ "imageName": "jrUol97rV5whAUtpljbyF-mobile.webp"
},
{
"id": "cmdskcx91000bvneko7tuaoqa",
"name": "Pendidikan Desa Berkualitas",
"jumlah": "34.11",
- "imageId": ""
+ "imageName": "Kh8RTS4ac1bG4ZXy3L2-y-mobile.webp"
},
{
"id": "cmdskjare000evnek1hglu0x8",
"name": "Keterlibatan Perempuan Desa",
"jumlah": "45.70",
- "imageId": ""
+ "imageName": "8bAk-cuzJ73W1dgj4lVqM-mobile.webp"
},
{
"id": "cmdskqcpc0002vnvnqjkqgm92",
"name": "Desa Layak Air Bersih Dan Sanitasi",
"jumlah": "48.54",
- "imageId": ""
+ "imageName": "k4zOPRqxi6PS0W_rHgiVp-mobile.webp"
},
{
"id": "cmdsktl3x0005vnvne15seefw",
"name": "Desa Berenergi Bersih Dan Terbarukan",
"jumlah": "99.64",
- "imageId": ""
+ "imageName": "txtX9hDmrHrRIkmpyQ6rp-mobile.webp"
},
{
"id": "cmdskuncw0008vnvcsdqoeog",
"name": "Pertumbuhan Ekonomi Desa Merata",
"jumlah": "40.92",
- "imageId": ""
+ "imageName": "Yx7ruGLJ_0QtD5RCvYhEC-mobile.webp"
},
{
"id": "cmdskw83j000bvvn9szqrea6",
"name": "Infrastruktur Dan Inovasi Desa Sesuai Kebutuhan",
"jumlah": "35.37",
- "imageId": ""
+ "imageName": "j1kZLRY4HLFBxdRt24XQw-mobile.webp"
},
{
"id": "cmdskwrq7000envnvy0c5nbgf",
"name": "Desa Tanpa Kesenjangan",
"jumlah": "35.47",
- "imageId": ""
+ "imageName": "Iqx5__9VtAQiScvvGb21S-mobile.webp"
},
{
"id": "cmdskxivx000hnvnvsx520gv1",
"name": "Kawasan Pemukiman Desa Aman Dan Nyaman",
"jumlah": "40.35",
- "imageId": ""
+ "imageName": "Eelib_Lr7GYv0nPj3XHVo-mobile.webp"
},
{
"id": "cmdskzg4c000kvnnkiv61gkt",
"name": "Konsumsi Dan Produksi Desa Sadar Lingkungan",
"jumlah": "16.67",
- "imageId": ""
+ "imageName": "6MzAfXZyjsSTOkckqYJug-mobile.webp"
},
{
"id": "cmdsl07lk000nvnnvnrepsdy5m",
"name": "Desa Tanggap Perubahan Iklim",
"jumlah": "0.00",
- "imageId": ""
+ "imageName": "f05zcftt1hdvkhCeN7wxk-mobile.webp"
},
{
"id": "cmdsl10rq000qvnvnlch9c1yv",
"name": "Desa Peduli Lingkungan Laut",
"jumlah": "50.00",
- "imageId": ""
+ "imageName": "3tuC1Bq5WvH7QEBHQVDaM-mobile.webp"
},
{
"id": "cmdsl1mc2000tvnvn357n8usi",
"name": "Desa Peduli Lingkungan Darat",
"jumlah": "0.00",
- "imageId": ""
+ "imageName": "oTKQeqAydsV5fXelI0S7Y-mobile.webp"
},
{
"id": "cmdsl2bx3000wvnvntshi4gnj",
"name": "Desa Damai Berkeadilan",
"jumlah": "78.65",
- "imageId": ""
+ "imageName": "16sdgs.webp"
},
{
"id": "cmdsl2yz3000zvnvnmf60ok7q",
"name": "Kemitraan Untuk Pembangunan Desa",
"jumlah": "20.00",
- "imageId": ""
+ "imageName": "L8jv3Wvj0GwsQgS7JSJQO-mobile.webp"
},
{
"id": "cmdsl492h0012vnvnmckm3n2x",
"name": "Kelembagaan Desa Dinamis Dan Budaya Desa Adaptif",
"jumlah": "47.22",
- "imageId": ""
+ "imageName": "18sdgs.webp"
}
]
diff --git a/prisma/data/lingkungan/data-lingkungan-desa/data-lingkungan-desa.json b/prisma/data/lingkungan/data-lingkungan-desa/data-lingkungan-desa.json
new file mode 100644
index 00000000..a96fcc1f
--- /dev/null
+++ b/prisma/data/lingkungan/data-lingkungan-desa/data-lingkungan-desa.json
@@ -0,0 +1,44 @@
+[
+ {
+ "id": "bnlneq3zg0000vn8issps1eg8",
+ "name": "Pelaksanaan Kebersihan Lingkungan Desa",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Kegiatan pelaksanaan kebersihan lingkungan desa Darmasaba untuk meningkatkan kualitas lingkungan dan kesehatan masyarakat.",
+ "icon": "lingkunganSehat"
+ },
+ {
+ "id": "bnlneq3zg0000vn8issps2fh9",
+ "name": "Penataan Taman Telajakan",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Program penataan taman telajakan sebagai upaya memperindah lingkungan desa serta ruang terbuka publik di desa Darmasaba.",
+ "icon": "lingkunganSehat"
+ },
+ {
+ "id": "bnlneq3zg0000vn8issps3gi0",
+ "name": "Pelatihan/Sosialisasi Peduli Lingkungan",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Pelatihan dan sosialisasi penyuluhan tentang kepedulian lingkungan hidup dan kehutanan bagi masyarakat.",
+ "icon": "pelatihan"
+ },
+ {
+ "id": "bnlneq3zg0000vn8issps4hj2",
+ "name": "Pemanfaatan Pekarangan Rumah",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Program pemanfaatan lingkungan pekarangan rumah menjadi produktif untuk mendukung ketahanan pangan keluarga.",
+ "icon": "sumberOksigen"
+ },
+ {
+ "id": "bnlneq3zg0000vn8issps5ik3",
+ "name": "Pengelolaan Daerah Aliran Sungai",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Kegiatan pengelolaan dan pemeliharaan daerah aliran sungai untuk menjaga kualitas sumber air dan mencegah erosi.",
+ "icon": "air"
+ },
+ {
+ "id": "bnlneq3zg0000vn8issps6jl4",
+ "name": "Pengelolaan Sampah Desa (Inovasi CINtA)",
+ "jumlah": "1 paket kegiatan",
+ "deskripsi": "Program inovatif cara indah tangani sampah (CINtA) di desa Darmasaba termasuk TPS3R dan pemilahan sampah berbasis rumah tangga.",
+ "icon": "sampah"
+ }
+]
diff --git a/prisma/data/lingkungan/gotong-royong/gotong-royong.json b/prisma/data/lingkungan/gotong-royong/gotong-royong.json
new file mode 100644
index 00000000..b03662dc
--- /dev/null
+++ b/prisma/data/lingkungan/gotong-royong/gotong-royong.json
@@ -0,0 +1,35 @@
+[
+ {
+ "id": "ee38b9ee-88ca-4a11-8fa0-7d407ea7e774",
+ "judul": "Gotong Royong PKK Mareresik Pura Desa dan Pura Dalem",
+ "deskripsiSingkat": "TP PKK Desa Darmasaba melaksanakan gotong royong mareresik di Pura Desa dan Pura Dalem Desa Adat Tegal.",
+ "deskripsiLengkap": "Pada tanggal 28 Januari 2024, TP PKK Desa Darmasaba bersama Ketua Kelompok PKK Banjar se-Desa Darmasaba melakukan kegiatan gotong royong membersihkan area Pura Desa dan Pura Dalem untuk mendukung kebersihan dan kesehatan lingkungan desa.",
+ "tanggal": "2024-01-28T00:00:00.000Z",
+ "lokasi": "Pura Desa dan Pura Dalem, Desa Adat Tegal, Desa Darmasaba, Badung",
+ "partisipan": 30,
+ "imageName": "YgOX5qAP3O1PHG5XmQXkr-mobile.webp",
+ "kategoriKegiatanId": "cmknan39v000004l8eiql149r"
+ },
+ {
+ "id": "dead9da0-e2e2-494c-9f88-721f2c3aa62a",
+ "judul": "Mareresik (Gotong Royong)",
+ "deskripsiSingkat": "Kegiatan mareresik (gotong royong) yang dilaksanakan di lingkungan desa Darmasaba.",
+ "deskripsiLengkap": "Mareresik (gotong royong) dilakukan pada tanggal 17 November 2023 oleh warga Desa Darmasaba sebagai bagian dari tradisi membersihkan dan menjaga lingkungan desa secara kolektif.",
+ "tanggal": "2023-11-17T00:00:00.000Z",
+ "lokasi": "Desa Darmasaba, Badung",
+ "partisipan": 25,
+ "imageName": "qxqSDHe-akIRi1EkQFUbG-mobile.webp",
+ "kategoriKegiatanId": "cmknan39v000004l8eiql149r"
+ },
+ {
+ "id": "5186e23b-28a6-4f64-89b8-43a35e5048a5",
+ "judul": "Gotong Royong dan Rapat Rutin TP PKK Desa Darmasaba",
+ "deskripsiSingkat": "Gotong royong bersama TP PKK Desa Darmasaba di Pura Dalem Kangin Desa Adat Tegal sekaligus rapat rutin.",
+ "deskripsiLengkap": "Pada tanggal 26 Mei 2022, TP PKK Desa Darmasaba bersama anggota se-Desa Darmasaba melaksanakan gotong royong mereresik Pura Dalem Kangin Desa Adat Tegal dan dilanjutkan dengan rapat rutin TP PKK sebagai wujud semangat kerja bersama dan kebersamaan sosial.",
+ "tanggal": "2022-05-26T00:00:00.000Z",
+ "lokasi": "Pura Dalem Kangin, Desa Adat Tegal, Desa Darmasaba, Badung",
+ "partisipan": 28,
+ "imageName": "iHTVkQZ1VdkMOXLt5qdAd-mobile.webp",
+ "kategoriKegiatanId": "cmknan39v000004l8eiql149r"
+ }
+]
diff --git a/prisma/data/lingkungan/gotong-royong/kategori-gotong-royong.json b/prisma/data/lingkungan/gotong-royong/kategori-gotong-royong.json
index 874e2e32..39605ca1 100644
--- a/prisma/data/lingkungan/gotong-royong/kategori-gotong-royong.json
+++ b/prisma/data/lingkungan/gotong-royong/kategori-gotong-royong.json
@@ -1,6 +1,6 @@
[
- { "nama": "Kebersihan" },
- { "nama": "Infrastruktur" },
- { "nama": "Sosial" },
- { "nama": "Lingkungan" }
- ]
\ No newline at end of file
+ { "id": "cmknan39v000004l8eiql816o", "nama": "Kebersihan" },
+ { "id": "cmknan39v000004l8eiql927p", "nama": "Infrastruktur" },
+ { "id": "cmknan39v000004l8eiql038q", "nama": "Sosial" },
+ { "id": "cmknan39v000004l8eiql149r", "nama": "Lingkungan" }
+]
diff --git a/prisma/data/lingkungan/pengelolaan-sampah/keterangan-bank-sampah.json b/prisma/data/lingkungan/pengelolaan-sampah/keterangan-bank-sampah.json
new file mode 100644
index 00000000..3275fce1
--- /dev/null
+++ b/prisma/data/lingkungan/pengelolaan-sampah/keterangan-bank-sampah.json
@@ -0,0 +1,20 @@
+[
+ {
+ "id": "cmkm8z5v70000vnrmu60qyd7b",
+ "name": "TPS 3R Pudak Mesari",
+ "alamat": "Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali 80352",
+ "namaTempatMaps": "TPS 3R Pudak Mesari",
+ "linkPetunjukArah": "https://www.google.com/maps/dir/?api=1&destination=-8.5680,115.2040",
+ "lat": -8.5680,
+ "lng": 115.2040
+ },
+ {
+ "id": "dummy0000000000000000000001",
+ "name": "Bank Sampah Darmasaba Mandiri",
+ "alamat": "Banjar Darmasaba Tengah, Desa Darmasaba, Kecamatan Abiansemal, Kabupaten Badung, Bali",
+ "namaTempatMaps": "Bank Sampah Darmasaba Mandiri",
+ "linkPetunjukArah": "https://www.google.com/maps/dir/?api=1&destination=-8.5670,115.2000",
+ "lat": -8.5670,
+ "lng": 115.2000
+ }
+]
diff --git a/prisma/data/lingkungan/pengelolaan-sampah/pengelolaan-sampah.json b/prisma/data/lingkungan/pengelolaan-sampah/pengelolaan-sampah.json
new file mode 100644
index 00000000..7e3feab0
--- /dev/null
+++ b/prisma/data/lingkungan/pengelolaan-sampah/pengelolaan-sampah.json
@@ -0,0 +1,27 @@
+[
+ {
+ "id": "cmkkshcox000504l88lp54coc",
+ "name": "Pilah sampah sesuai jenisnya",
+ "icon": "trash"
+ },
+ {
+ "id": "cmkkshcox000504l99mq65dpd",
+ "name": "Bawa sampah ke Bank Sampah",
+ "icon": "truck"
+ },
+ {
+ "id": "cmkkshcox000504l00nr76eqe",
+ "name": "Timbang sampah di Bank Sampah",
+ "icon": "scale"
+ },
+ {
+ "id": "cmkkshcox000504l11os87frf",
+ "name": "Catat hasil timbangan di buku tabungan",
+ "icon": "clipboard"
+ },
+ {
+ "id": "cmkkshcox000504l22pt98gsg",
+ "name": "Sampah didaur ulang oleh petugas Bank Sampah",
+ "icon": "sampah"
+ }
+]
diff --git a/prisma/data/lingkungan/program-penghijauan/program-penghijauan.json b/prisma/data/lingkungan/program-penghijauan/program-penghijauan.json
new file mode 100644
index 00000000..9fdc655c
--- /dev/null
+++ b/prisma/data/lingkungan/program-penghijauan/program-penghijauan.json
@@ -0,0 +1,23 @@
+[
+ {
+ "id": "cmkmdp3zg0000vn8issps1eg8",
+ "name": "Penghijauan Desa Darmasaba",
+ "judul": "Program Penghijauan Lingkungan",
+ "deskripsi": "Program penghijauan desa Darmasaba bertujuan menanam pohon di area publik dan pinggiran jalan desa sebagai upaya meningkatkan kualitas lingkungan, menyerap karbon, dan memperindah desa.
",
+ "icon": "pohon"
+ },
+ {
+ "id": "cmkmdp3zg0000vn8isspt2fh9",
+ "name": "Komunitas Tanam Pohon",
+ "judul": "Gerakan Komunitas Tanam Pohon",
+ "deskripsi": "Pelibatan warga desa dalam kegiatan rutin penanaman pohon di area perbukitan dan ruang terbuka hijau untuk meningkatkan kesadaran lingkungan dan mengurangi erosi.
",
+ "icon": "pohon"
+ },
+ {
+ "id": "cmkmdp3zg0000vn8isspu3gi0",
+ "name": "Taman Hijau Bersama",
+ "judul": "Pembangunan Taman Hijau Desa",
+ "deskripsi": "Pembangunan taman hijau di beberapa titik strategis desa, melibatkan banjar adat dan kelompok pemuda untuk menciptakan ruang publik yang nyaman dan ramah lingkungan.
",
+ "icon": "pohon"
+ }
+]
diff --git a/prisma/data/pendidikan/data-pendidikan/data-pendidikan.json b/prisma/data/pendidikan/data-pendidikan/data-pendidikan.json
new file mode 100644
index 00000000..4e5b64ed
--- /dev/null
+++ b/prisma/data/pendidikan/data-pendidikan/data-pendidikan.json
@@ -0,0 +1,22 @@
+[
+ {
+ "id": "cmkqmqbv30000vn84kf0ogf61",
+ "name": "TK",
+ "jumlah": "120"
+ },
+ {
+ "id": "cmkqmr20h0001vn84jxtuukfk",
+ "name": "SD",
+ "jumlah": "874"
+ },
+ {
+ "id": "cmkqmshcx0002vn84ufnz9mue",
+ "name": "SMP",
+ "jumlah": "50"
+ },
+ {
+ "id": "cmkqmsuoc0003vn84glk1d0rc",
+ "name": "SMA",
+ "jumlah": "862"
+ }
+]
diff --git a/prisma/data/pendidikan/info-sekolah/lembaga.json b/prisma/data/pendidikan/info-sekolah/lembaga.json
new file mode 100644
index 00000000..7bae725d
--- /dev/null
+++ b/prisma/data/pendidikan/info-sekolah/lembaga.json
@@ -0,0 +1,97 @@
+[
+ {
+ "id": "cmghqwjs4000404l8c6vwd200",
+ "nama": "TK Widya Kumara",
+ "jenjangId": "cmghqwjs4000404l8c5uvc301"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd202",
+ "nama": "TK Widya Sari",
+ "jenjangId": "cmghqwjs4000404l8c5uvc301"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd203",
+ "nama": "TK Kuntala Dewi I",
+ "jenjangId": "cmghqwjs4000404l8c5uvc301"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd204",
+ "nama": "TK Widya Kumarayasa",
+ "jenjangId": "cmghqwjs4000404l8c5uvc301"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd205",
+ "nama": "TK Dewi Ganadwati",
+ "jenjangId": "cmghqwjs4000404l8c5uvc301"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd400",
+ "nama": "SD No. 1 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd300",
+ "nama": "SD No. 2 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd401",
+ "nama": "SD No. 3 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd402",
+ "nama": "SD No. 4 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd403",
+ "nama": "SD No. 5 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd404",
+ "nama": "SD No. 6 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd405",
+ "nama": "SD No. 7 Darmasaba",
+ "jenjangId": "cmghqwjs4000404l8c5uvc302"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd406",
+ "nama": "SMP Negeri 1 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc303"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd407",
+ "nama": "SMP Negeri 2 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc303"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd408",
+ "nama": "SMP Negeri 3 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc303"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd409",
+ "nama": "SMP Negeri 4 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc303"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd410",
+ "nama": "SMP Negeri 5 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc303"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd411",
+ "nama": "SMA Negeri 1 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc304"
+ },
+ {
+ "id": "cmghqwjs4000404l8c6vwd412",
+ "nama": "SMA Negeri 2 Abiansemal",
+ "jenjangId": "cmghqwjs4000404l8c5uvc304"
+ }
+]
diff --git a/prisma/data/pendidikan/info-sekolah/pengajar.json b/prisma/data/pendidikan/info-sekolah/pengajar.json
new file mode 100644
index 00000000..2a82cdfc
--- /dev/null
+++ b/prisma/data/pendidikan/info-sekolah/pengajar.json
@@ -0,0 +1,967 @@
+[
+ {
+ "id": "dnkowzpeu000004l7exka3arm",
+ "nama": "I Kadek Ariyasa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3brm",
+ "nama": "Ni Luh Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3crm",
+ "nama": "I Made Dharma Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3drm",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3erm",
+ "nama": "I Komang Aditya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3frm",
+ "nama": "Ni Made Intan Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3grm",
+ "nama": "I Putu Bayu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3hrm",
+ "nama": "Ni Ketut Sri Wahyuni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3irm",
+ "nama": "I Nyoman Yoga Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "dnkowzpeu000004l7exka3jrm",
+ "nama": "Ni Komang Ratna Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+
+ {
+ "id": "dnkox31m7000004lagpwk9z5r",
+ "nama": "I Made Satria",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z1r",
+ "nama": "Ni Luh Putri Ayu",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z2r",
+ "nama": "I Kadek Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z3r",
+ "nama": "Ni Made Rani Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z4r",
+ "nama": "I Komang Yuda Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z5r",
+ "nama": "Ni Kadek Ayu Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z6r",
+ "nama": "I Putu Ardi Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z7r",
+ "nama": "Ni Ketut Melati",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z8r",
+ "nama": "I Nyoman Agung",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "dnkox31m7000004lagpwk9z9r",
+ "nama": "Ni Komang Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+
+ {
+ "id": "dnkox64cu000104la7mge67yy",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67zz",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67aa",
+ "nama": "I Komang Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67bb",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67cc",
+ "nama": "I Putu Gede Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67dd",
+ "nama": "Ni Made Cahya Utami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67ee",
+ "nama": "I Nyoman Aditya Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67ff",
+ "nama": "Ni Komang Ratna Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67gg",
+ "nama": "I Kadek Bima Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "dnkox64cu000104la7mge67hh",
+ "nama": "Ni Putu Ayu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+
+ {
+ "id": "dnkox9520000204la1ans1pag",
+ "nama": "I Made Yoga Wirawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pbg",
+ "nama": "Ni Luh Desi Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pcg",
+ "nama": "I Komang Putra Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pdg",
+ "nama": "Ni Kadek Intan Paramitha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1peg",
+ "nama": "I Putu Adi Wicaksana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pfg",
+ "nama": "Ni Made Ayu Kirana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1phg",
+ "nama": "I Nyoman Bayu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pih",
+ "nama": "Ni Komang Sri Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pjh",
+ "nama": "I Kadek Rama Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pjk",
+ "nama": "Ni Putu Diah Anggraini",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+
+ {
+ "id": "dnkox9520000204la1ans1pzg",
+ "nama": "I Made Krisna Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzh",
+ "nama": "Ni Luh Ayu Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzi",
+ "nama": "I Komang Yuda Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzj",
+ "nama": "Ni Kadek Putri Anjani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzk",
+ "nama": "I Putu Surya Adi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzl",
+ "nama": "Ni Made Ayu Cahyani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzm",
+ "nama": "I Nyoman Artha Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzn",
+ "nama": "Ni Komang Sinta Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzo",
+ "nama": "I Kadek Gede Pranaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "dnkox9520000204la1ans1pzp",
+ "nama": "Ni Putu Ratih Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+
+ {
+ "id": "dnkoxzl8u000704la0jla6y5a",
+ "nama": "I Made Arya Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5b",
+ "nama": "I Nyoman Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5c",
+ "nama": "I Kadek Surya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5d",
+ "nama": "I Komang Aditya Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5e",
+ "nama": "Ni Luh Putri Ayu",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5f",
+ "nama": "Ni Made Intan Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5g",
+ "nama": "I Gede Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5h",
+ "nama": "Ni Kadek Sri Wahyuni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5i",
+ "nama": "I Putu Andika Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "dnkoxzl8u000704la0jla6y5j",
+ "nama": "Ni Komang Diah Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq1",
+ "nama": "I Made Bima Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq2",
+ "nama": "I Nyoman Rizky Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq3",
+ "nama": "I Kadek Wahyu Nugraha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq4",
+ "nama": "I Komang Fajar Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq5",
+ "nama": "Ni Luh Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq6",
+ "nama": "Ni Made Putu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq7",
+ "nama": "I Gede Arjuna Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq8",
+ "nama": "Ni Kadek Ayu Puspita",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq9",
+ "nama": "I Putu Danu Kresna",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "dnkoy1mvk000804ladmmq5qq0",
+ "nama": "Ni Komang Ratna Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+
+ {
+ "id": "dnkoy2m6j000904la2fbthtda",
+ "nama": "I Made Yoga Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdb",
+ "nama": "I Nyoman Bayu Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdc",
+ "nama": "I Kadek Rama Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdd",
+ "nama": "I Komang Agus Prabawa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtde",
+ "nama": "Ni Luh Ayu Citra Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdf",
+ "nama": "Ni Made Sari Indrayani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdg",
+ "nama": "I Gede Mahesa Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdh",
+ "nama": "Ni Kadek Purnami Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdi",
+ "nama": "I Putu Arta Gunawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "dnkoy2m6j000904la2fbthtdj",
+ "nama": "Ni Komang Indah Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpya",
+ "nama": "I Made Arya Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyb",
+ "nama": "I Nyoman Dika Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyc",
+ "nama": "I Kadek Putra Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyd",
+ "nama": "I Komang Agus Setiawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpye",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyf",
+ "nama": "Ni Kadek Ayu Puspita",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyg",
+ "nama": "I Putu Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyh",
+ "nama": "Ni Komang Dwi Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyi",
+ "nama": "I Made Bayu Kurniawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "dnkoy5cs5000a04la9jlcbpyj",
+ "nama": "Ni Putu Ayu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ua",
+ "nama": "I Made Dewa Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ub",
+ "nama": "I Nyoman Surya Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60uc",
+ "nama": "I Kadek Bima Santosa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ud",
+ "nama": "I Komang Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ue",
+ "nama": "Ni Luh Made Pertiwi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60uf",
+ "nama": "Ni Kadek Citra Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ug",
+ "nama": "I Putu Rangga Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60uh",
+ "nama": "Ni Komang Rina Apriyani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60ui",
+ "nama": "I Made Aditya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "dnkoy6dtp000b04la1zlo60uj",
+ "nama": "Ni Putu Ayu Cahyaningrum",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+
+ {
+ "id": "dnkoy76ux000c04lags2adcaa",
+ "nama": "I Made Krisna Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcab",
+ "nama": "I Nyoman Gede Putrawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcac",
+ "nama": "I Kadek Yoga Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcad",
+ "nama": "I Komang Danu Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcae",
+ "nama": "Ni Luh Ayu Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcaf",
+ "nama": "Ni Kadek Intan Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcag",
+ "nama": "I Putu Wira Gunawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcah",
+ "nama": "Ni Komang Ayu Ratnasari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcai",
+ "nama": "I Made Fajar Nugraha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy76ux000c04lags2adcaj",
+ "nama": "Ni Putu Desi Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn1",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn2",
+ "nama": "I Nyoman Surya Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn3",
+ "nama": "I Komang Aditya Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn4",
+ "nama": "I Putu Gede Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn5",
+ "nama": "Ni Luh Putri Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn6",
+ "nama": "Ni Kadek Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn7",
+ "nama": "I Made Yoga Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn8",
+ "nama": "I Nyoman Wira Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn9",
+ "nama": "Ni Komang Diah Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "dnkoy85nz000d04lahr5hepn0",
+ "nama": "I Putu Bayu Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+
+ {
+ "id": "dnkoy91wc000e04la966390na",
+ "nama": "I Made Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nb",
+ "nama": "I Nyoman Agus Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nc",
+ "nama": "I Komang Yuda Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nd",
+ "nama": "I Putu Gede Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390ne",
+ "nama": "Ni Luh Desi Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nf",
+ "nama": "Ni Kadek Ayu Prameswari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390ng",
+ "nama": "I Made Rizky Ananta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nh",
+ "nama": "I Nyoman Dika Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390ni",
+ "nama": "Ni Komang Puspita Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "dnkoy91wc000e04la966390nj",
+ "nama": "I Putu Andika Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi1",
+ "nama": "I Made Wahyu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi2",
+ "nama": "I Nyoman Kevin Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi3",
+ "nama": "I Komang Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi4",
+ "nama": "I Putu Gede Satya Wibawa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi5",
+ "nama": "Ni Luh Ayu Citra Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi6",
+ "nama": "Ni Kadek Putri Anggraini",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi7",
+ "nama": "I Made Fajar Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi8",
+ "nama": "I Nyoman Rama Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi9",
+ "nama": "Ni Komang Sinta Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoy9zzy000f04la7rze3fi0",
+ "nama": "I Putu Dimas Pradipta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wya",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyb",
+ "nama": "I Nyoman Adi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyc",
+ "nama": "I Wayan Putra Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyd",
+ "nama": "Ni Kadek Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyf",
+ "nama": "I Komang Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyg",
+ "nama": "Ni Luh Putu Anggreni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyh",
+ "nama": "I Made Dewa Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyi",
+ "nama": "Ni Putu Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyj",
+ "nama": "I Nyoman Agus Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "dnkoyapqt000g04laaob10wyk",
+ "nama": "Ni Komang Ratih Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+
+ {
+ "id": "dnkoybh2p000h04lahbds4wc1",
+ "nama": "I Wayan Surya Dharma",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc2",
+ "nama": "Ni Kadek Ayu Prameswari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc3",
+ "nama": "I Made Gede Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc4",
+ "nama": "I Nyoman Krisna Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc5",
+ "nama": "Ni Luh Desi Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc6",
+ "nama": "I Komang Bima Santosa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc7",
+ "nama": "Ni Putu Cahya Utami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc8",
+ "nama": "I Wayan Dimas Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc9",
+ "nama": "Ni Kadek Purnama Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "dnkoybh2p000h04lahbds4wc0",
+ "nama": "I Made Yoga Kencana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1a",
+ "nama": "I Nyoman Gede Sapta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1b",
+ "nama": "Ni Luh Ayu Pertiwi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1c",
+ "nama": "I Wayan Danu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1d",
+ "nama": "Ni Kadek Melati Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1e",
+ "nama": "I Made Raka Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1f",
+ "nama": "Ni Komang Sinta Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1g",
+ "nama": "I Nyoman Arta Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1h",
+ "nama": "Ni Putu Indah Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1i",
+ "nama": "I Wayan Bagus Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoyc6ij000i04la3r8i7f1j",
+ "nama": "Ni Kadek Ayu Wulandari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9ba",
+ "nama": "I Made Arya Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bb",
+ "nama": "I Komang Adi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bc",
+ "nama": "I Nyoman Dewa Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bd",
+ "nama": "I Ketut Agus Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9be",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bf",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bg",
+ "nama": "I Made Surya Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bh",
+ "nama": "I Komang Yoga Pradipta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bi",
+ "nama": "Ni Putu Citra Wulandari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "dnkoycyfl000j04la757zg9bj",
+ "nama": "I Nyoman Bayu Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfa",
+ "nama": "I Made Gede Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfb",
+ "nama": "I Komang Krisna Yudha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfc",
+ "nama": "I Nyoman Putra Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfd",
+ "nama": "I Ketut Wira Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfe",
+ "nama": "Ni Luh Kadek Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbff",
+ "nama": "Ni Made Ayu Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfg",
+ "nama": "I Putu Yoga Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfh",
+ "nama": "I Komang Dimas Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfi",
+ "nama": "Ni Kadek Sinta Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "dnkoydnnl000k04lae6jvhbfj",
+ "nama": "I Nyoman Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ }
+]
diff --git a/prisma/data/pendidikan/info-sekolah/siswa.json b/prisma/data/pendidikan/info-sekolah/siswa.json
new file mode 100644
index 00000000..9068ddf3
--- /dev/null
+++ b/prisma/data/pendidikan/info-sekolah/siswa.json
@@ -0,0 +1,967 @@
+[
+ {
+ "id": "cmkowzpeu000004l7exka3arm",
+ "nama": "I Kadek Ariyasa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3brm",
+ "nama": "Ni Luh Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3crm",
+ "nama": "I Made Dharma Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3drm",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3erm",
+ "nama": "I Komang Aditya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3frm",
+ "nama": "Ni Made Intan Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3grm",
+ "nama": "I Putu Bayu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3hrm",
+ "nama": "Ni Ketut Sri Wahyuni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3irm",
+ "nama": "I Nyoman Yoga Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+ {
+ "id": "cmkowzpeu000004l7exka3jrm",
+ "nama": "Ni Komang Ratna Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd200"
+ },
+
+ {
+ "id": "cmkox31m7000004lagpwk9z5r",
+ "nama": "I Made Satria",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z1r",
+ "nama": "Ni Luh Putri Ayu",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z2r",
+ "nama": "I Kadek Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z3r",
+ "nama": "Ni Made Rani Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z4r",
+ "nama": "I Komang Yuda Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z5r",
+ "nama": "Ni Kadek Ayu Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z6r",
+ "nama": "I Putu Ardi Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z7r",
+ "nama": "Ni Ketut Melati",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z8r",
+ "nama": "I Nyoman Agung",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+ {
+ "id": "cmkox31m7000004lagpwk9z9r",
+ "nama": "Ni Komang Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd202"
+ },
+
+ {
+ "id": "cmkox64cu000104la7mge67yy",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67zz",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67aa",
+ "nama": "I Komang Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67bb",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67cc",
+ "nama": "I Putu Gede Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67dd",
+ "nama": "Ni Made Cahya Utami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67ee",
+ "nama": "I Nyoman Aditya Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67ff",
+ "nama": "Ni Komang Ratna Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67gg",
+ "nama": "I Kadek Bima Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+ {
+ "id": "cmkox64cu000104la7mge67hh",
+ "nama": "Ni Putu Ayu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd203"
+ },
+
+ {
+ "id": "cmkox9520000204la1ans1pag",
+ "nama": "I Made Yoga Wirawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pbg",
+ "nama": "Ni Luh Desi Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pcg",
+ "nama": "I Komang Putra Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pdg",
+ "nama": "Ni Kadek Intan Paramitha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1peg",
+ "nama": "I Putu Adi Wicaksana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pfg",
+ "nama": "Ni Made Ayu Kirana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1phg",
+ "nama": "I Nyoman Bayu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pih",
+ "nama": "Ni Komang Sri Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pjh",
+ "nama": "I Kadek Rama Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pjk",
+ "nama": "Ni Putu Diah Anggraini",
+ "lembagaId": "cmghqwjs4000404l8c6vwd204"
+ },
+
+ {
+ "id": "cmkox9520000204la1ans1pzg",
+ "nama": "I Made Krisna Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzh",
+ "nama": "Ni Luh Ayu Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzi",
+ "nama": "I Komang Yuda Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzj",
+ "nama": "Ni Kadek Putri Anjani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzk",
+ "nama": "I Putu Surya Adi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzl",
+ "nama": "Ni Made Ayu Cahyani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzm",
+ "nama": "I Nyoman Artha Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzn",
+ "nama": "Ni Komang Sinta Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzo",
+ "nama": "I Kadek Gede Pranaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+ {
+ "id": "cmkox9520000204la1ans1pzp",
+ "nama": "Ni Putu Ratih Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd205"
+ },
+
+ {
+ "id": "cmkoxzl8u000704la0jla6y5a",
+ "nama": "I Made Arya Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5b",
+ "nama": "I Nyoman Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5c",
+ "nama": "I Kadek Surya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5d",
+ "nama": "I Komang Aditya Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5e",
+ "nama": "Ni Luh Putri Ayu",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5f",
+ "nama": "Ni Made Intan Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5g",
+ "nama": "I Gede Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5h",
+ "nama": "Ni Kadek Sri Wahyuni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5i",
+ "nama": "I Putu Andika Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+ {
+ "id": "cmkoxzl8u000704la0jla6y5j",
+ "nama": "Ni Komang Diah Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd400"
+ },
+
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq1",
+ "nama": "I Made Bima Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq2",
+ "nama": "I Nyoman Rizky Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq3",
+ "nama": "I Kadek Wahyu Nugraha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq4",
+ "nama": "I Komang Fajar Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq5",
+ "nama": "Ni Luh Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq6",
+ "nama": "Ni Made Putu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq7",
+ "nama": "I Gede Arjuna Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq8",
+ "nama": "Ni Kadek Ayu Puspita",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq9",
+ "nama": "I Putu Danu Kresna",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+ {
+ "id": "cmkoy1mvk000804ladmmq5qq0",
+ "nama": "Ni Komang Ratna Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd300"
+ },
+
+ {
+ "id": "cmkoy2m6j000904la2fbthtda",
+ "nama": "I Made Yoga Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdb",
+ "nama": "I Nyoman Bayu Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdc",
+ "nama": "I Kadek Rama Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdd",
+ "nama": "I Komang Agus Prabawa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtde",
+ "nama": "Ni Luh Ayu Citra Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdf",
+ "nama": "Ni Made Sari Indrayani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdg",
+ "nama": "I Gede Mahesa Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdh",
+ "nama": "Ni Kadek Purnami Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdi",
+ "nama": "I Putu Arta Gunawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+ {
+ "id": "cmkoy2m6j000904la2fbthtdj",
+ "nama": "Ni Komang Indah Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd401"
+ },
+
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpya",
+ "nama": "I Made Arya Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyb",
+ "nama": "I Nyoman Dika Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyc",
+ "nama": "I Kadek Putra Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyd",
+ "nama": "I Komang Agus Setiawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpye",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyf",
+ "nama": "Ni Kadek Ayu Puspita",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyg",
+ "nama": "I Putu Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyh",
+ "nama": "Ni Komang Dwi Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyi",
+ "nama": "I Made Bayu Kurniawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+ {
+ "id": "cmkoy5cs5000a04la9jlcbpyj",
+ "nama": "Ni Putu Ayu Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd402"
+ },
+
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ua",
+ "nama": "I Made Dewa Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ub",
+ "nama": "I Nyoman Surya Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60uc",
+ "nama": "I Kadek Bima Santosa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ud",
+ "nama": "I Komang Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ue",
+ "nama": "Ni Luh Made Pertiwi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60uf",
+ "nama": "Ni Kadek Citra Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ug",
+ "nama": "I Putu Rangga Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60uh",
+ "nama": "Ni Komang Rina Apriyani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60ui",
+ "nama": "I Made Aditya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+ {
+ "id": "cmkoy6dtp000b04la1zlo60uj",
+ "nama": "Ni Putu Ayu Cahyaningrum",
+ "lembagaId": "cmghqwjs4000404l8c6vwd403"
+ },
+
+ {
+ "id": "cmkoy76ux000c04lags2adcaa",
+ "nama": "I Made Krisna Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcab",
+ "nama": "I Nyoman Gede Putrawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcac",
+ "nama": "I Kadek Yoga Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcad",
+ "nama": "I Komang Danu Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcae",
+ "nama": "Ni Luh Ayu Sinta Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcaf",
+ "nama": "Ni Kadek Intan Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcag",
+ "nama": "I Putu Wira Gunawan",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcah",
+ "nama": "Ni Komang Ayu Ratnasari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcai",
+ "nama": "I Made Fajar Nugraha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy76ux000c04lags2adcaj",
+ "nama": "Ni Putu Desi Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd404"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn1",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn2",
+ "nama": "I Nyoman Surya Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn3",
+ "nama": "I Komang Aditya Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn4",
+ "nama": "I Putu Gede Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn5",
+ "nama": "Ni Luh Putri Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn6",
+ "nama": "Ni Kadek Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn7",
+ "nama": "I Made Yoga Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn8",
+ "nama": "I Nyoman Wira Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn9",
+ "nama": "Ni Komang Diah Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+ {
+ "id": "cmkoy85nz000d04lahr5hepn0",
+ "nama": "I Putu Bayu Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd405"
+ },
+
+ {
+ "id": "cmkoy91wc000e04la966390na",
+ "nama": "I Made Dwi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nb",
+ "nama": "I Nyoman Agus Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nc",
+ "nama": "I Komang Yuda Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nd",
+ "nama": "I Putu Gede Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390ne",
+ "nama": "Ni Luh Desi Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nf",
+ "nama": "Ni Kadek Ayu Prameswari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390ng",
+ "nama": "I Made Rizky Ananta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nh",
+ "nama": "I Nyoman Dika Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390ni",
+ "nama": "Ni Komang Puspita Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+ {
+ "id": "cmkoy91wc000e04la966390nj",
+ "nama": "I Putu Andika Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd406"
+ },
+
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi1",
+ "nama": "I Made Wahyu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi2",
+ "nama": "I Nyoman Kevin Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi3",
+ "nama": "I Komang Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi4",
+ "nama": "I Putu Gede Satya Wibawa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi5",
+ "nama": "Ni Luh Ayu Citra Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi6",
+ "nama": "Ni Kadek Putri Anggraini",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi7",
+ "nama": "I Made Fajar Pranata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi8",
+ "nama": "I Nyoman Rama Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi9",
+ "nama": "Ni Komang Sinta Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoy9zzy000f04la7rze3fi0",
+ "nama": "I Putu Dimas Pradipta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd407"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wya",
+ "nama": "I Made Arya Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyb",
+ "nama": "I Nyoman Adi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyc",
+ "nama": "I Wayan Putra Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyd",
+ "nama": "Ni Kadek Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyf",
+ "nama": "I Komang Yoga Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyg",
+ "nama": "Ni Luh Putu Anggreni",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyh",
+ "nama": "I Made Dewa Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyi",
+ "nama": "Ni Putu Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyj",
+ "nama": "I Nyoman Agus Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+ {
+ "id": "cmkoyapqt000g04laaob10wyk",
+ "nama": "Ni Komang Ratih Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd408"
+ },
+
+ {
+ "id": "cmkoybh2p000h04lahbds4wc1",
+ "nama": "I Wayan Surya Dharma",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc2",
+ "nama": "Ni Kadek Ayu Prameswari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc3",
+ "nama": "I Made Gede Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc4",
+ "nama": "I Nyoman Krisna Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc5",
+ "nama": "Ni Luh Desi Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc6",
+ "nama": "I Komang Bima Santosa",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc7",
+ "nama": "Ni Putu Cahya Utami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc8",
+ "nama": "I Wayan Dimas Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc9",
+ "nama": "Ni Kadek Purnama Sari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+ {
+ "id": "cmkoybh2p000h04lahbds4wc0",
+ "nama": "I Made Yoga Kencana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd409"
+ },
+
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1a",
+ "nama": "I Nyoman Gede Sapta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1b",
+ "nama": "Ni Luh Ayu Pertiwi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1c",
+ "nama": "I Wayan Danu Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1d",
+ "nama": "Ni Kadek Melati Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1e",
+ "nama": "I Made Raka Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1f",
+ "nama": "Ni Komang Sinta Laksmi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1g",
+ "nama": "I Nyoman Arta Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1h",
+ "nama": "Ni Putu Indah Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1i",
+ "nama": "I Wayan Bagus Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoyc6ij000i04la3r8i7f1j",
+ "nama": "Ni Kadek Ayu Wulandari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd410"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9ba",
+ "nama": "I Made Arya Putra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bb",
+ "nama": "I Komang Adi Saputra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bc",
+ "nama": "I Nyoman Dewa Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bd",
+ "nama": "I Ketut Agus Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9be",
+ "nama": "Ni Luh Putu Sari Dewi",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bf",
+ "nama": "Ni Kadek Ayu Lestari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bg",
+ "nama": "I Made Surya Mahendra",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bh",
+ "nama": "I Komang Yoga Pradipta",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bi",
+ "nama": "Ni Putu Citra Wulandari",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+ {
+ "id": "cmkoycyfl000j04la757zg9bj",
+ "nama": "I Nyoman Bayu Pramana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd411"
+ },
+
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfa",
+ "nama": "I Made Gede Pratama",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfb",
+ "nama": "I Komang Krisna Yudha",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfc",
+ "nama": "I Nyoman Putra Santika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfd",
+ "nama": "I Ketut Wira Adnyana",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfe",
+ "nama": "Ni Luh Kadek Maharani",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbff",
+ "nama": "Ni Made Ayu Purnami",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfg",
+ "nama": "I Putu Yoga Mahardika",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfh",
+ "nama": "I Komang Dimas Prasetya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfi",
+ "nama": "Ni Kadek Sinta Permata",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ },
+ {
+ "id": "cmkoydnnl000k04lae6jvhbfj",
+ "nama": "I Nyoman Arta Wijaya",
+ "lembagaId": "cmghqwjs4000404l8c6vwd412"
+ }
+]
diff --git a/prisma/data/pendidikan/perpustakaan-digital/kategori-buku.json b/prisma/data/pendidikan/perpustakaan-digital/kategori-buku.json
new file mode 100644
index 00000000..71e58601
--- /dev/null
+++ b/prisma/data/pendidikan/perpustakaan-digital/kategori-buku.json
@@ -0,0 +1,47 @@
+[
+ {
+ "id": "cmkqb11mc000104jibq76bdzu",
+ "name": "Fiksi"
+ },
+ {
+ "id": "cmkqb11mc000104jibq87bdzu",
+ "name": "Non Fiksi"
+ },
+ {
+ "id": "cmkqb11mc000104jibq97bdzu",
+ "name": "Pendidikan"
+ },
+ {
+ "id": "cmkqb11mc000104jibqa7bdzu",
+ "name": "Ilmiah"
+ },
+ {
+ "id": "cmkqb11mc000104jibqb7bdzu",
+ "name": "Drama"
+ },
+ {
+ "id": "cmkqb11mc000104jibqc7bdzu",
+ "name": "Sejarah"
+ },
+ {
+ "id": "cmkqb11mc000104jibqd7bdzu",
+ "name": "Teknologi"
+ },
+ {
+ "id": "cmkqb11mc000104jibqe7bdzu",
+ "name": "Agama"
+ },
+ {
+ "id": "cmkqb11mc000104jibqf7bdzu",
+ "name": "Pengembangan Diri"
+ },
+ {
+ "id": "cmkqb11mc000104jibqg7bdzu",
+ "name": "Kesehatan"
+ },
+ {
+ "id": "cmkqb11mc000104jibqh7bdzu",
+ "name": "Anak Dan Remaja"
+ }
+
+]
\ No newline at end of file
diff --git a/prisma/data/pendidikan/perpustakaan-digital/perpustakaan-digital.json b/prisma/data/pendidikan/perpustakaan-digital/perpustakaan-digital.json
new file mode 100644
index 00000000..abfe70cd
--- /dev/null
+++ b/prisma/data/pendidikan/perpustakaan-digital/perpustakaan-digital.json
@@ -0,0 +1,198 @@
+[
+ {
+ "id": "cmkqhbi6f0002vneao4my49k9",
+ "judul": "Laskar Pelangi",
+ "deskripsi": "Novel inspiratif tentang perjuangan anak-anak di Belitung dalam meraih pendidikan dan mimpi mereka
",
+ "kategoriId": "cmkqb11mc000104jibq76bdzu",
+ "imageName": "RnAdv7O0QAFrxkFLAXJSa-mobile.webp"
+ },
+ {
+ "id": "cmkqhedff0005vneas3rtbumi",
+ "judul": "Bumi Manusia",
+ "deskripsi": "Kisah kehidupan Minke di masa kolonial yang menggambarkan perjuangan, pendidikan, dan identitas bangsa
",
+ "kategoriId": "cmkqb11mc000104jibqc7bdzu",
+ "imageName": "71eZShq4FYAFLxpLfZB0W-mobile.webp"
+ },
+ {
+ "id": "cmkqhg1g70008vneajbpz8phh",
+ "judul": "Atomic Habits",
+ "deskripsi": "Panduan membangun kebiasaan kecil yang konsisten untuk menghasilkan perubahan besar dalam hidup
",
+ "kategoriId": "cmkqb11mc000104jibqf7bdzu",
+ "imageName": "Uxq3GXPqh7HN9fHmRkr3r-mobile.webp"
+ },
+ {
+ "id": "cmkqhl6sr000bvneampx0svus",
+ "judul": "Clean Code",
+ "deskripsi": "Buku wajib programmer tentang cara menulis kode yang bersih, mudah dibaca, dan mudah dirawat
",
+ "kategoriId": "cmkqb11mc000104jibqd7bdzu",
+ "imageName": "W5Fc0uRADNkIY3nZicvQA-mobile.webp"
+ },
+ {
+ "id": "cmkqhoaa1000evnearppgpyxo",
+ "judul": "Sejarah Indonesia Modern",
+ "deskripsi": "Membahas perjalanan sejarah Indonesia dari masa kolonial hingga era modern
",
+ "kategoriId": "cmkqb11mc000104jibqc7bdzu",
+ "imageName": "mp77Op-MwtPQZnH3so4JY-mobile.webp"
+ },
+ {
+ "id": "cmkqhr9oc000hvnea677ad3kb",
+ "judul": "Ensiklopedia Anak Pintar",
+ "deskripsi": "Buku referensi bergambar yang membantu anak mengenal ilmu pengetahuan secara menyenangkan
",
+ "kategoriId": "cmkqb11mc000104jibqh7bdzu",
+ "imageName": "V09ZxN1wOwbSFLQiDK0VQ-mobile.webp"
+ },
+ {
+ "id": "cmkqi5ksf000kvnea9c04n2hy",
+ "judul": "Filosofi Teras",
+ "deskripsi": "Pengenalan filsafat Stoikisme untuk menghadapi kehidupan modern dengan lebih tenang
",
+ "kategoriId": "cmkqb11mc000104jibq87bdzu",
+ "imageName": "Wqp4AyVkGjqRMED9Q5XAs-mobile.webp"
+ },
+ {
+ "id": "cmkqi97hq000nvneaparjbcrm",
+ "judul": "Pemrograman JavaScript Dasar",
+ "deskripsi": "Panduan dasar belajar JavaScript untuk pemula dalam dunia pengembangan web
",
+ "kategoriId": "cmkqb11mc000104jibqd7bdzu",
+ "imageName": "NH4aLc7cVuutdQBCofTC0-mobile.webp"
+ },
+ {
+ "id": "cmkqibjt9000qvnea13ox7fmv",
+ "judul": "Pendidikan Karakter",
+ "deskripsi": "Buku yang membahas pentingnya pendidikan karakter dalam membentuk generasi bangsa
",
+ "kategoriId": "cmkqb11mc000104jibqf7bdzu",
+ "imageName": "MLrsPrD6oiHsrNP4Lc8J7-mobile.webp"
+ },
+ {
+ "id": "cmkqidnar000tvneaohk5v8k6",
+ "judul": "Psikologi Kepribadian",
+ "deskripsi": "Mengenal teori-teori kepribadian manusia dalam perspektif psikologi
",
+ "kategoriId": "cmkqb11mc000104jibq87bdzu",
+ "imageName": "iaIeNdhuxqltqKP7aZncQ-mobile.webp"
+ },
+ {
+ "id": "cmkqifdiu000wvnea7xd0yi4f",
+ "judul": "Ayat-Ayat Cinta",
+ "deskripsi": "Novel religi yang mengangkat kisah cinta, iman, dan perjuangan hidup
",
+ "kategoriId": "cmkqb11mc000104jibqe7bdzu",
+ "imageName": "WUDssJ59pTKE_3IuTiZ2s-mobile.webp"
+ },
+ {
+ "id": "cmkqik7vi000zvneae7d5cq9i",
+ "judul": "Negeri 5 Menara",
+ "deskripsi": "Cerita persahabatan dan perjuangan santri dalam mengejar mimpi hingga ke mancanegara
",
+ "kategoriId": "cmkqb11mc000104jibq76bdzu",
+ "imageName": "RJH_-4_R_nlP7GVEQeD1M-mobile.webp"
+ },
+ {
+ "id": "cmkqinno30012vneac1sgsvis",
+ "judul": "Belajar UI/UX Design",
+ "deskripsi": "Panduan praktis memahami desain antarmuka dan pengalaman pengguna
",
+ "kategoriId": "cmkqb11mc000104jibqd7bdzu",
+ "imageName": "9MA-Jx_36uoho2Tg40_G9-mobile.webp"
+ },
+ {
+ "id": "cmkqiqegd0015vneawv5u5tpm",
+ "judul": "Manajemen Waktu Efektif",
+ "deskripsi": "Teknik mengatur waktu agar lebih produktif dan fokus pada hal penting
",
+ "kategoriId": "cmkqb11mc000104jibqf7bdzu",
+ "imageName": "dkb7ZWFl28TREVcvH8sWd-mobile.webp"
+ },
+ {
+ "id": "cmkqiurc60018vneavyd3pj9q",
+ "judul": "Dongeng Nusantara",
+ "deskripsi": "Kumpulan dongeng tradisional Indonesia yang sarat pesan moral
",
+ "kategoriId": "cmkqb11mc000104jibq76bdzu",
+ "imageName": "nVj3one6CLuWRd04QnsWo-mobile.webp"
+ },
+ {
+ "id": "cmkqix2kb001bvnea5v81cw7p",
+ "judul": "Ekonomi Makro",
+ "deskripsi": "Pembahasan konsep ekonomi makro secara sistematis dan mudah dipahami
",
+ "kategoriId": "cmkqb11mc000104jibq87bdzu",
+ "imageName": "AnB7JO4_6tlPTX3ypOVLi-mobile.webp"
+ },
+ {
+ "id": "cmkqiyts2001evneahnk45ry5",
+ "judul": "Seni Berpikir Kritis",
+ "deskripsi": "Buku yang membantu pembaca menghindari kesalahan berpikir dalam pengambilan keputusan
",
+ "kategoriId": "cmkqb11mc000104jibq87bdzu",
+ "imageName": "sAyoMERxL6JgFfiO22KPb-mobile.webp"
+ },
+ {
+ "id": "cmkqj0nq0001hvnea06r8m3kj",
+ "judul": "Seni Berpikir Kritis",
+ "deskripsi": "Buku yang membantu pembaca menghindari kesalahan berpikir dalam pengambilan keputusan
",
+ "kategoriId": "cmkqb11mc000104jibq87bdzu",
+ "imageName": "WeA-JP2Ks_32fv1k529vj-mobile.webp"
+ },
+ {
+ "id": "cmkqj37w4001kvnea04n9w2bx",
+ "judul": "Panduan Shalat Lengkap",
+ "deskripsi": "Panduan praktis dan lengkap tentang tata cara shalat sesuai tuntunan
",
+ "kategoriId": "cmkqb11mc000104jibqe7bdzu",
+ "imageName": "pxlHu2kDmIprQqC2PuXaL-mobile.webp"
+ },
+ {
+ "id": "cmkqj5qp6001nvnea4xhvluz3",
+ "judul": "Cerita Sains untuk Anak",
+ "deskripsi": "Cerita edukatif yang mengenalkan sains kepada anak dengan bahasa sederhana
",
+ "kategoriId": "cmkqb11mc000104jibqh7bdzu",
+ "imageName": "G0iELZb2DhQDCCP5OdzJR-desktop.webp"
+ },
+ {
+ "id": "cml7fq776000104jscnj58sgm",
+ "judul": "Pedagogy of the Oppressed",
+ "deskripsi": "Klasik pemikiran pendidikan kritis; menggali hubungan guru-murid dan peran pendidikan dalam pembebasan sosial
",
+ "kategoriId": "cmkqb11mc000104jibq97bdzu",
+ "imageName": "pendidikan-1.webp"
+ },
+ {
+ "id": "cml7fqurm000204js5p60hkym",
+ "judul": "The Courage to Teach",
+ "deskripsi": "Tentang refleksi diri seorang pendidik; cocok untuk pengajar yang ingin lebih dari sekedar “metode mengajar”
",
+ "kategoriId": "cmkqb11mc000104jibq97bdzu",
+ "imageName": "pendidikan-2.webp"
+ },
+ {
+ "id": "cml7fqurm000204js5p60hkzn",
+ "judul": "A Brief History of Time",
+ "deskripsi": "Penjelasan kosmologi yang terkenal dunia; sains kompleks dibahas dengan bahasa yang bisa dinikmati pembaca umum
",
+ "kategoriId": "cmkqb11mc000104jibqa7bdzu",
+ "imageName": "ilmiah-1.webp"
+ },
+ {
+ "id": "cml7fqurm000204js5p60hkao",
+ "judul": "The Selfish Gene",
+ "deskripsi": "Membawa perspektif baru tentang evolusi melalui “gen” sebagai unit seleksi
",
+ "kategoriId": "cmkqb11mc000104jibqa7bdzu",
+ "imageName": "ilmiah-2.webp"
+ },
+ {
+ "id": "cml7fx09c000304jshams3xbg",
+ "judul": "A Little Life",
+ "deskripsi": "Novel yang menggambarkan hidup seorang remaja yang mengalami kehidupan yang sangat sulit
",
+ "kategoriId": "cmkqb11mc000104jibqb7bdzu",
+ "imageName": "drama-1.webp"
+ },
+ {
+ "id": "cml7fx09c000304jshams3xch",
+ "judul": "Death of a Salesman",
+ "deskripsi": "Drama teater klasik Amerika tentang harapan, keluarga, dan realitas hidup.
",
+ "kategoriId": "cmkqb11mc000104jibqb7bdzu",
+ "imageName": "drama-2.webp"
+ },
+ {
+ "id": "cml7fx09c000304jshams3xdi",
+ "judul": "How Not to Die",
+ "deskripsi": "Panduan berbasis penelitian tentang pola makan untuk mencegah dan menangani penyakit.
",
+ "kategoriId": "cmkqb11mc000104jibqg7bdzu",
+ "imageName": "kesehatan-1.webp"
+ },
+ {
+ "id": "cml7fx09c000304jshams3xej",
+ "judul": "The Body Keeps the Score",
+ "deskripsi": "Fokus pada trauma, otak & tubuh; penting untuk memahami kesehatan mental secara mendalam.
",
+ "kategoriId": "cmkqb11mc000104jibqg7bdzu",
+ "imageName": "kesehatan-2.webp"
+ }
+]
diff --git a/prisma/data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json b/prisma/data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json
index 731ccdb9..795e82cb 100644
--- a/prisma/data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json
+++ b/prisma/data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json
@@ -1,14 +1,14 @@
[
{
"id": "cmeppcwzk0000vn5exmudcipd",
- "jenisInformasi": "Potensi Desa",
- "deskripsi": "“Potensi desa adalah segenap sumber daya alam dan sumber daya manusia yang dimiliki desa sebagai modal dasar yang perlu dikelola dan dikembangkan bagi kelangsungan dan perkembangan desa. Adapun potensi yang dimiliki Desa Darmasaba yaitu:
TPS3R Pudak Mesari
Bumdes Pudak Mesari
Pertanian
Jogging Track Tegeh Aban, Karang Gadon dan Munduk Uma Desa
Taman Beji Cengana
Dam Tanah Putih
Gumuh Sari Water Park
UMKM
Kawasan Kuliner
IKM berbasis Pengolahan Pangan
Genteng
Peternakan Ikan Lele
Pemotongan Daging”
",
+ "jenisInformasi": "Penyelenggaraan Informasi Publik Desa",
+ "deskripsi": "Kegiatan pengadaan/pembuatan aplikasi dan pelatihan untuk kemudahan akses informasi publik desa Darmasaba melalui sarana digital sebagai bagian dari transparansi layanan masyarakat.
",
"tanggal": "2021-05-25"
},
{
"id": "cmeppieay0001vn5e8qe658ub",
- "jenisInformasi": "Layanan Surat Keterangan Desa",
- "deskripsi": "“Desa Darmasaba menyediakan berbagai jenis layanan surat keterangan untuk kebutuhan administratif, antara lain:
Surat Keterangan Domisili Organisasi
Surat Keterangan Penghasilan
Surat Keterangan Tidak Mampu
Surat Keterangan Kelahiran
Surat Keterangan Usaha
Surat Keterangan Tempat Usaha
Surat Keterangan Belum Kawin
Surat Keterangan Kelakuan Baik (Pengantar SKCK)
Surat Keterangan Kematian
Surat Keterangan Perbedaan Biodata Diri
Surat Keterangan Yatim/Piatu/Yatim Piatu Untuk surat keterangan lainnya, masyarakat dapat berkonsultasi langsung ke kantor Perbekel Darmasaba.”(Sumber: Laman Layanan Desa Darmasaba)
",
+ "jenisInformasi": "Kemudahan Akses Internet dan Informasi Desa",
+ "deskripsi": "Penyediaan jaringan/ instalasi komunikasi dan informasi desa untuk mempermudah akses internet dan informasi kepada masyarakat desa Darmasaba, bagian dari penyelenggaraan informasi desa.
",
"tanggal": "2025-02-21"
}
]
\ No newline at end of file
diff --git a/prisma/data/ppid/profile-ppid/profilePPid.json b/prisma/data/ppid/profile-ppid/profilePPid.json
index 0c6828f7..26fed553 100644
--- a/prisma/data/ppid/profile-ppid/profilePPid.json
+++ b/prisma/data/ppid/profile-ppid/profilePPid.json
@@ -5,6 +5,7 @@
"biodata": "I.B Surya Prabhawa Manuaba, S.H., M.H., adalah Perbekel Darmasaba periode 2021-2027, seorang advokat, pendiri Mantra Legal Consultants & Advocates, serta aktif di bidang musik dan akademis. Dia menempuh pendidikan hukum di Universitas Udayana dan Universitas Mahasaraswati Denpasar, serta memiliki pengalaman luas di berbagai organisasi dan kepemimpinan.
",
"riwayat": " 2021 - 2027: Perbekel Desa Darmasaba 2015 - Sekarang: Founder & Managing Director Mantra Legal Consultants & Advocates 2020 - Sekarang: Founder Ugawa Record Music Studio 2010 - 2016: Dosen Fakultas Hukum Universitas Mahasaraswati Denpasar ",
"pengalaman": " 1996 – 1997: Ketua OSIS SMP Negeri 1 Abiansemal 1999 – 2000: Ketua OSIS SMA Negeri 1 Mengwi 2008 – 2009: Ketua BEM Universitas Mahasaraswati Denpasar 2008 – 2010: Ketua Sekaa Taruna Sila Dharma, Banjar Tengah, Desa Adat Tegal, Darmasaba 2020 – Sekarang: Pengurus Young Lawyer Committee Peradi Denpasar 2021 – Sekarang: Dewan Kehormatan Himpunan Pengusaha Muda Indonesia (HIPMI) Badung 2023 – 2028: Komite Tetap Advokasi – Bidang Hukum dan Regulasi Kamar Dagang dan Industri Badung ",
- "unggulan": "Pemberdayaan Ekonomi dan UMKM Pelatihan dan pendampingan UMKM lokal Program bantuan modal usaha bagi pelaku usaha kecil Digitalisasi UMKM untuk meningkatkan pemasaran produk lokal "
+ "unggulan": "Pemberdayaan Ekonomi dan UMKM Pelatihan dan pendampingan UMKM lokal Program bantuan modal usaha bagi pelaku usaha kecil Digitalisasi UMKM untuk meningkatkan pemasaran produk lokal ",
+ "imageName" : "t9x6bSW8WnbdPS9cx9PLC-mobile.webp"
}
]
diff --git a/prisma/data/ppid/struktur-ppid/pegawai-PPID.json b/prisma/data/ppid/struktur-ppid/pegawai-PPID.json
index 713cb799..8602e313 100644
--- a/prisma/data/ppid/struktur-ppid/pegawai-PPID.json
+++ b/prisma/data/ppid/struktur-ppid/pegawai-PPID.json
@@ -8,7 +8,8 @@
"telepon": "081234567891",
"alamat": "Jl. Raya Desa No. 1",
"posisiId": "kepala_desa",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewxfvw000004ibee5013f4",
@@ -19,7 +20,8 @@
"telepon": "081234567892",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "sekretaris_desa",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewxvqw000104ibgm5l8fzs",
@@ -30,7 +32,8 @@
"telepon": "081234567892",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kaur_keuangan",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewy1g9000204ib2n7hbx0i",
@@ -41,7 +44,8 @@
"telepon": "081234567892",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kadus_banjar_dinas_menesa",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewybah000304ibgqhn1gm2",
@@ -52,7 +56,8 @@
"telepon": "081234567893",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kadus_banjar_dinas_darmasaba",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewygqz000404ib20sv8nvg",
@@ -63,7 +68,8 @@
"telepon": "081234567893",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kadus_banjar_dinas_bucu",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewyos1000504ibcu8o2gyk",
@@ -74,7 +80,8 @@
"telepon": "081234567893",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kadus_banjar_dinas_gulingan",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
},
{
"id": "cmgewyxk7000604ib8djs3i6c",
@@ -85,7 +92,8 @@
"telepon": "081234567893",
"alamat": "Jl. Raya Desa No. 2",
"posisiId": "kadus_banjar_dinas_taman",
- "isActive": true
+ "isActive": true,
+ "imageName": ""
}
]
\ No newline at end of file
diff --git a/prisma/data/resolveImageId.ts b/prisma/data/resolveImageId.ts
new file mode 100644
index 00000000..58227739
--- /dev/null
+++ b/prisma/data/resolveImageId.ts
@@ -0,0 +1,11 @@
+import safeImageId from "./safeImageId";
+
+export default async function resolveImageIdForSeed(
+ existingImageId: string | null | undefined,
+ seedImageId: string | null | undefined
+) {
+ if (existingImageId) return existingImageId;
+
+ // ✅ Skip validasi saat seed
+ return await safeImageId(seedImageId, true);
+}
\ No newline at end of file
diff --git a/prisma/data/safeImageId.ts b/prisma/data/safeImageId.ts
new file mode 100644
index 00000000..ab01a8e6
--- /dev/null
+++ b/prisma/data/safeImageId.ts
@@ -0,0 +1,24 @@
+import prisma from "@/lib/prisma";
+
+export default async function safeImageId(
+ imageId?: string | null,
+ skipValidation = false // ✅ tambah param
+) {
+ if (!imageId) return null;
+
+ if (skipValidation) {
+ console.log(`⚠️ Skipping validation for ${imageId} (seed mode)`);
+ return imageId; // langsung return tanpa cek DB
+ }
+
+ const exists = await prisma.fileStorage.findUnique({
+ where: { id: imageId },
+ });
+
+ if (!exists) {
+ console.warn(`⚠️ imageId ${imageId} not found in FileStorage`);
+ return null;
+ }
+
+ return imageId;
+}
\ No newline at end of file
diff --git a/prisma/data/user/roles.json b/prisma/data/user/roles.json
index b79f3928..4d2a1046 100644
--- a/prisma/data/user/roles.json
+++ b/prisma/data/user/roles.json
@@ -1,23 +1,32 @@
[
- {
- "id": "role-1",
- "name": "ADMIN DESA",
- "description": "Administrator Desa",
- "permissions": ["manage_users", "manage_content", "view_reports"],
- "isActive": true
- },
- {
- "id": "role-2",
- "name": "ADMIN KESEHATAN",
- "description": "Administrator Bidang Kesehatan",
- "permissions": ["manage_health_data", "view_reports"],
- "isActive": true
- },
- {
- "id": "role-3",
- "name": "ADMIN SEKOLAH",
- "description": "Administrator Sekolah",
- "permissions": ["manage_school_data", "view_reports"],
- "isActive": true
- }
- ]
\ No newline at end of file
+ {
+ "id": "0",
+ "name": "DEVELOPER",
+ "description": "Developer",
+ "isActive": true
+ },
+ {
+ "id": "1",
+ "name": "SUPER ADMIN",
+ "description": "Administrator",
+ "isActive": true
+ },
+ {
+ "id": "2",
+ "name": "ADMIN DESA",
+ "description": "Administrator Desa",
+ "isActive": true
+ },
+ {
+ "id": "3",
+ "name": "ADMIN KESEHATAN",
+ "description": "Administrator Bidang Kesehatan",
+ "isActive": true
+ },
+ {
+ "id": "4",
+ "name": "ADMIN PENDIDIKAN",
+ "description": "Administrator Bidang Pendidikan",
+ "isActive": true
+ }
+]
diff --git a/prisma/data/user/users.json b/prisma/data/user/users.json
index eea2a98a..733aeba4 100644
--- a/prisma/data/user/users.json
+++ b/prisma/data/user/users.json
@@ -1,23 +1,10 @@
[
{
- "id": "user-1",
- "nama": "Admin Desa",
- "nomor": "089647037426",
- "roleId": "role-1",
- "isActive": true
- },
- {
- "id": "user-2",
- "nama": "Admin Kesehatan",
- "nomor": "082339004198",
- "roleId": "role-2",
- "isActive": true
- },
- {
- "id": "user-3",
- "nama": "Admin Sekolah",
- "nomor": "085237157222",
- "roleId": "role-3",
- "isActive": true
+ "id": "cmie1o0zh0002vn132vtzg7hh",
+ "username": "SuperAdmin-Nico",
+ "nomor": "6289647037426",
+ "roleId": 0,
+ "isActive": true,
+ "sessionInvalid": false
}
]
diff --git a/prisma/lib/create_file_share_folder.ts b/prisma/lib/create_file_share_folder.ts
new file mode 100644
index 00000000..025d0490
--- /dev/null
+++ b/prisma/lib/create_file_share_folder.ts
@@ -0,0 +1,246 @@
+// import { getValidAuthToken } from "../../src/lib/seafile-auth-service";
+
+// type CdnItem = {
+// name: string;
+// path: string;
+// cdnUrl: string;
+// };
+
+// type DirItem = {
+// type: "file" | "dir";
+// name: string;
+// };
+
+// const BASE_URL = process.env.SEAFILE_BASE_URL!;
+// const REPO_ID = process.env.SEAFILE_REPO_ID!;
+
+// // folder yang dishare (RELATIVE, tanpa slash depan)
+// const DIR_TARGET = "asset-web";
+
+// // 🔑 TOKEN DIRECTORY SHARE (/d/{token})
+// const PUBLIC_SHARE_TOKEN = process.env.SEAFILE_PUBLIC_SHARE_TOKEN!;
+
+// /**
+// * Ambil list file dari repo (butuh token sekali)
+// */
+// async function getDirItems(): Promise {
+// const token = await getValidAuthToken();
+
+// // Validasi bahwa semua variabel lingkungan telah diatur
+// if (!BASE_URL) {
+// throw new Error('SEAFILE_BASE_URL environment variable is not set');
+// }
+
+// if (!REPO_ID) {
+// throw new Error('SEAFILE_REPO_ID environment variable is not set');
+// }
+
+// // Bangun URL dan pastikan valid
+// const url = `${BASE_URL}/api2/repos/${REPO_ID}/dir/?p=/${DIR_TARGET}`;
+
+// try {
+// new URL(url); // Ini akan melempar error jika URL tidak valid
+// } catch (error) {
+// throw new Error(`Invalid URL constructed: ${url}. Error: ${error}`);
+// }
+
+// const res = await fetch(url, {
+// headers: {
+// Authorization: `Token ${token}`,
+// },
+// });
+
+// if (!res.ok) {
+// const text = await res.text();
+// throw new Error(`Failed get dir items: ${text}`);
+// }
+
+// return res.json();
+// }
+
+// /**
+// * Build PUBLIC CDN URL
+// */
+// function buildPublicCdnUrl(fileName: string) {
+// return `${BASE_URL}/d/${PUBLIC_SHARE_TOKEN}/files/?p=${encodeURIComponent(
+// fileName,
+// )}&raw=1`;
+// }
+
+// /**
+// * Ambil semua PUBLIC CDN URL
+// */
+// export async function getAllPublicCdnUrls(): Promise {
+// const items = await getDirItems();
+
+// return items
+// .filter((item) => item.type === "file")
+// .map((file) => {
+// const path = `${DIR_TARGET}/${file.name}`;
+// return {
+// name: file.name,
+// path,
+// cdnUrl: buildPublicCdnUrl(file.name),
+// };
+// });
+// }
+
+// /**
+// * Run langsung (optional)
+// */
+// if (import.meta.main) {
+// const data = await getAllPublicCdnUrls();
+// console.log(data);
+// }
+
+
+
+// import { getValidAuthToken } from "../../src/lib/seafile-auth-service";
+
+// type CdnItem = {
+// name: string;
+// path: string;
+// cdnUrl: string;
+// };
+
+// type DirItem = {
+// type: "file" | "dir";
+// name: string;
+// };
+
+// // ✅ PAKAI ENV YANG BENAR
+// const BASE_URL = process.env.SEAFILE_URL!;
+// const REPO_ID = process.env.SEAFILE_REPO_ID!;
+// const PUBLIC_SHARE_TOKEN = process.env.SEAFILE_PUBLIC_SHARE_TOKEN!;
+
+// // folder yang dishare (RELATIVE, TANPA slash depan)
+// const DIR_TARGET = "asset-web";
+
+// /**
+// * Ambil list file dari repo (token dipakai SEKALI)
+// */
+// async function getDirItems(): Promise {
+// if (!BASE_URL || !REPO_ID) {
+// throw new Error("SEAFILE env not configured correctly");
+// }
+
+// const token = await getValidAuthToken();
+
+// const url = `${BASE_URL}/api2/repos/${REPO_ID}/dir/?p=/${DIR_TARGET}`;
+
+// const res = await fetch(url, {
+// headers: {
+// Authorization: `Token ${token}`,
+// },
+// });
+
+// if (!res.ok) {
+// const text = await res.text();
+// throw new Error(`Failed get dir items: ${text}`);
+// }
+
+// return res.json();
+// }
+
+// /**
+// * Build PUBLIC CDN URL (DIRECTORY SHARE)
+// */
+// function buildPublicCdnUrl(fileName: string) {
+// const fullPath = `/${DIR_TARGET}/${fileName}`;
+// return `${BASE_URL}/d/${PUBLIC_SHARE_TOKEN}/files/?p=${encodeURIComponent(
+// fullPath,
+// )}&raw=1`;
+// }
+
+// /**
+// * Ambil semua PUBLIC CDN URL
+// */
+// export async function getAllPublicCdnUrls(): Promise {
+// const items = await getDirItems();
+
+// return items
+// .filter((item) => item.type === "file")
+// .map((file) => ({
+// name: file.name,
+// path: `${DIR_TARGET}/${file.name}`,
+// cdnUrl: buildPublicCdnUrl(file.name),
+// }));
+// }
+
+// /**
+// * Run langsung
+// */
+// if (import.meta.main) {
+// const data = await getAllPublicCdnUrls();
+// console.log(data);
+// }
+
+
+import { getValidAuthToken } from "../../src/lib/seafile-auth-service";
+type CdnItem = {
+ name: string;
+ path: string;
+ cdnUrl: string;
+};
+type DirItem = {
+ type: "file" | "dir";
+ name: string;
+};
+const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com";
+const REPO_ID = process.env.SEAFILE_REPO_ID!;
+// folder yang dishare (RELATIVE, tanpa slash depan)
+const DIR_TARGET = "asset-web";
+// 🔑 TOKEN DIRECTORY SHARE (/d/{token})
+const PUBLIC_SHARE_TOKEN = "3a9a9ecb5e244f4da8ae";
+/**
+ * Ambil list file dari repo (butuh token sekali)
+ */
+async function getDirItems(): Promise {
+ const token = await getValidAuthToken();
+ const res = await fetch(
+ `${BASE_URL}/api2/repos/${REPO_ID}/dir/?p=/${DIR_TARGET}`,
+ {
+ headers: {
+ Authorization: `Token ${token}`,
+ },
+ },
+ );
+ if (!res.ok) {
+ const text = await res.text();
+ throw new Error(`Failed get dir items: ${text}`);
+ }
+ return res.json();
+}
+/**
+ * Build PUBLIC CDN URL
+ */
+function buildPublicCdnUrl(fileName: string) {
+ return `${BASE_URL}/d/${PUBLIC_SHARE_TOKEN}/files/?p=${encodeURIComponent(
+ fileName,
+ )}&raw=1`;
+}
+/**
+ * Ambil semua PUBLIC CDN URL
+ */
+export async function getAllPublicCdnUrls(): Promise {
+ const items = await getDirItems();
+ return items
+ .filter((item) => item.type === "file")
+ .map((file) => {
+ // const path = `${DIR_TARGET}/${file.name}`;
+ const path = `/${file.name}`;
+ return {
+ name: file.name,
+ path,
+ cdnUrl: buildPublicCdnUrl(file.name),
+ };
+ });
+}
+/**
+ * Run langsung (optional)
+ */
+if (import.meta.main) {
+ const data = await getAllPublicCdnUrls();
+ console.log(data);
+}
+
diff --git a/prisma/lib/get_images.ts b/prisma/lib/get_images.ts
new file mode 100644
index 00000000..dc5421ae
--- /dev/null
+++ b/prisma/lib/get_images.ts
@@ -0,0 +1,81 @@
+import { getValidAuthToken } from "../../src/lib/seafile-auth-service";
+
+type DirItem = {
+ type: "file" | "dir";
+ name: string;
+ path: string;
+ size?: number;
+};
+
+const REPO_ID = process.env.SEAFILE_REPO_ID!;
+
+// ⛔ PENTING: RELATIVE PATH (tanpa slash depan)
+const DIR_TARGET = "asset-web";
+
+const BASE_URL = process.env.SEAFILE_URL;
+
+async function getDirItems(): Promise {
+ const token = await getValidAuthToken();
+ const headers = {
+ Authorization: `Token ${token}`,
+ };
+
+ const res = await fetch(`${BASE_URL}/repos/${REPO_ID}/dir/?p=${DIR_TARGET}`, {
+ headers,
+ });
+
+ if (!res.ok) {
+ throw new Error(`Failed get dir items: ${res.statusText}`);
+ }
+
+ return res.json();
+}
+
+async function getDownloadUrl(filePath: string): Promise {
+ const token = await getValidAuthToken();
+ const headers = {
+ Authorization: `Token ${token}`,
+ };
+
+ const res = await fetch(
+ `${BASE_URL}/repos/${REPO_ID}/file/?p=${encodeURIComponent(filePath)}&reuse=1`,
+ { headers },
+ );
+
+ if (!res.ok) {
+ const text = await res.text();
+ console.error("Seafile error:", {
+ status: res.status,
+ body: text,
+ url: res.url,
+ });
+ throw new Error(`Failed get file url`);
+ }
+
+ const data = await res.json();
+
+ return data;
+}
+
+/**
+ * Ambil semua download URL dari target dir
+ */
+export async function getAllDownloadUrls() {
+ const items = await getDirItems();
+
+ const files = items.filter((item) => item.type === "file");
+
+ const results = await Promise.all(
+ files.map(async (file) => {
+ const filePath = `${DIR_TARGET}/${file.name}`;
+ const url = await getDownloadUrl(filePath);
+ return {
+ name: file.name,
+ path: filePath,
+ downloadUrl: url,
+ };
+ }),
+ );
+
+ return results;
+}
diff --git a/prisma/lib/get_shared_images.ts b/prisma/lib/get_shared_images.ts
new file mode 100644
index 00000000..c76b9064
--- /dev/null
+++ b/prisma/lib/get_shared_images.ts
@@ -0,0 +1,71 @@
+//ini code awal cari image by folder di seafile
+
+
+type CdnItem = {
+ name: string;
+ path: string;
+ cdnUrl: string;
+};
+
+const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com";
+const SHARE_ID = "3325e9db2c504ebf9584";
+
+// https://cld-dkr-makuro-seafile.wibudev.com/d/3a9a9ecb5e244f4da8ae/
+// https://cld-dkr-makuro-seafile.wibudev.com/d/3a9a9ecb5e244f4da8ae/files/?p=-M_tICRVz6ZxOfvkuHQgU-mobile.webp&raw=1
+
+
+/**
+ * Build CDN URL langsung (tanpa API, tanpa token)
+ */
+export function buildCdnUrl(filePath: string) {
+ // filePath contoh: "banner/home.jpg"
+ return `${BASE_URL}/f/${SHARE_ID}/${filePath}?raw=1`;
+}
+
+/**
+ * Ambil daftar file dari PUBLIC SHARE (optional)
+ * Tidak pakai token
+ */
+async function getPublicDirItems(path = "/"): Promise {
+ const res = await fetch(
+ `${BASE_URL}/api/v2.1/share-links/${SHARE_ID}/dir/?p=${encodeURIComponent(
+ path,
+ )}`,
+ );
+
+ if (!res.ok) {
+ const text = await res.text();
+ throw new Error(`Failed get public dir items: ${text}`);
+ }
+
+ return res.json();
+}
+
+/**
+ * Ambil semua CDN URL dari folder public share
+ */
+export async function getAllCdnUrls(
+ dirPath = "/",
+): Promise {
+ const items = await getPublicDirItems(dirPath);
+
+ return items
+ .filter((item: any) => item.type === "file")
+ .map((file: any) => {
+ const filePath =
+ dirPath === "/"
+ ? file.name
+ : `${dirPath.replace(/\/$/, "")}/${file.name}`;
+
+ return {
+ name: file.name,
+ path: filePath,
+ cdnUrl: buildCdnUrl(filePath),
+ };
+ });
+}
+
+if(import.meta.main) {
+ const allCdnUrls = await getAllCdnUrls();
+ console.log(allCdnUrls);
+}
diff --git a/prisma/lib/get_sharef.ts b/prisma/lib/get_sharef.ts
new file mode 100644
index 00000000..13c2dc45
--- /dev/null
+++ b/prisma/lib/get_sharef.ts
@@ -0,0 +1,33 @@
+const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com";
+const ADMIN_TOKEN = process.env.SEAFILE_TOKEN!;
+const REPO_ID = process.env.SEAFILE_REPO_ID!;
+
+export async function createFileShareForFolder() {
+ const res = await fetch(`${BASE_URL}/api/v2.1/share-links/`, {
+ method: "POST",
+ headers: {
+ Authorization: `Token ${ADMIN_TOKEN}`,
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({
+ repo_id: REPO_ID,
+ path: "/asset-web", // FOLDER
+ permission: "r",
+ }),
+ });
+
+ if (!res.ok) {
+ const text = await res.text();
+ throw new Error(text);
+ }
+
+ const data = await res.json();
+ console.log("FILE SHARE LINK:", data);
+
+ // data.link -> https://domain/f/XXXX/
+ // data.token / data.id (tergantung versi)
+}
+
+if (import.meta.main) {
+ await createFileShareForFolder();
+}
diff --git a/prisma/migrations/20251119062255_add_unique_username/migration.sql b/prisma/migrations/20251119062255_add_unique_username/migration.sql
new file mode 100644
index 00000000..d8d9e087
--- /dev/null
+++ b/prisma/migrations/20251119062255_add_unique_username/migration.sql
@@ -0,0 +1,1127 @@
+/*
+ Warnings:
+
+ - You are about to drop the column `kelahiranKasar` on the `DataKematian_Kelahiran` table. All the data in the column will be lost.
+ - You are about to drop the column `kematianBayi` on the `DataKematian_Kelahiran` table. All the data in the column will be lost.
+ - You are about to drop the column `kematianKasar` on the `DataKematian_Kelahiran` table. All the data in the column will be lost.
+ - You are about to drop the column `tahun` on the `DataKematian_Kelahiran` table. All the data in the column will be lost.
+ - You are about to drop the column `jumlah` on the `GrafikKepuasan` table. All the data in the column will be lost.
+ - You are about to drop the column `label` on the `GrafikKepuasan` table. All the data in the column will be lost.
+ - You are about to drop the column `imageId` on the `KolaborasiInovasi` table. All the data in the column will be lost.
+ - You are about to drop the column `imageId` on the `KontakDaruratKeamanan` table. All the data in the column will be lost.
+ - You are about to drop the column `imageId` on the `KontakItem` table. All the data in the column will be lost.
+ - You are about to drop the column `kategoriId` on the `KontakItem` table. All the data in the column will be lost.
+ - You are about to drop the column `programKeamananId` on the `PencegahanKriminalitas` table. All the data in the column will be lost.
+ - You are about to drop the column `tipsKeamananId` on the `PencegahanKriminalitas` table. All the data in the column will be lost.
+ - You are about to drop the column `videoKeamananId` on the `PencegahanKriminalitas` table. All the data in the column will be lost.
+ - You are about to drop the column `kategori` on the `PotensiDesa` table. All the data in the column will be lost.
+ - You are about to drop the column `ikonUrl` on the `ProgramKemiskinan` table. All the data in the column will be lost.
+ - You are about to drop the `ProgramKeamanan` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `TipsKeamanan` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `VideoKeamanan` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `hubungan_organisasi` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `pegawai` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `posisi_organisasi` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `struktur_organisasi` table. If the table is not empty, all the data it contains will be lost.
+ - Added the required column `kelahiranId` to the `DataKematian_Kelahiran` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `kematianId` to the `DataKematian_Kelahiran` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `category` to the `FileStorage` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `alamat` to the `GrafikKepuasan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `jenisKelamin` to the `GrafikKepuasan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `nama` to the `GrafikKepuasan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `penyakit` to the `GrafikKepuasan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `tanggal` to the `GrafikKepuasan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `whatsapp` to the `KontakDarurat` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `icon` to the `KontakDaruratKeamanan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `kategoriId` to the `KontakDaruratKeamanan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `icon` to the `KontakItem` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `notelp` to the `LowonganPekerjaan` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `name` to the `MediaSosial` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `kontak` to the `PasarDesa` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `deskripsi` to the `PencegahanKriminalitas` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `deskripsiSingkat` to the `PencegahanKriminalitas` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `judul` to the `PencegahanKriminalitas` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `linkVideo` to the `PencegahanKriminalitas` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `jadwalPelayanan` to the `Posyandu` table without a default value. This is not possible if the table is not empty.
+ - Added the required column `icon` to the `ProgramKemiskinan` table without a default value. This is not possible if the table is not empty.
+
+*/
+-- CreateEnum
+CREATE TYPE "JenisKelamin" AS ENUM ('LAKI_LAKI', 'PEREMPUAN');
+
+-- CreateEnum
+CREATE TYPE "Agama" AS ENUM ('ISLAM', 'KRISTEN_PROTESTAN', 'KRISTEN_KATOLIK', 'HINDU', 'BUDDHA', 'KONGHUCU', 'LAINNYA');
+
+-- CreateEnum
+CREATE TYPE "StatusPernikahan" AS ENUM ('BELUM_MENIKAH', 'MENIKAH', 'JANDA_DUDA');
+
+-- CreateEnum
+CREATE TYPE "UkuranBaju" AS ENUM ('S', 'M', 'L', 'XL', 'XXL', 'LAINNYA');
+
+-- CreateEnum
+CREATE TYPE "StatusPeminjaman" AS ENUM ('Dipinjam', 'Dikembalikan', 'Terlambat', 'Dibatalkan');
+
+-- DropForeignKey
+ALTER TABLE "JadwalKegiatan" DROP CONSTRAINT "JadwalKegiatan_pendaftaranJadwalKegiatanId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KolaborasiInovasi" DROP CONSTRAINT "KolaborasiInovasi_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KontakDaruratKeamanan" DROP CONSTRAINT "KontakDaruratKeamanan_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KontakItem" DROP CONSTRAINT "KontakItem_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KontakItem" DROP CONSTRAINT "KontakItem_kategoriId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "MediaSosial" DROP CONSTRAINT "MediaSosial_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PelayananSuratKeterangan" DROP CONSTRAINT "PelayananSuratKeterangan_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PencegahanKriminalitas" DROP CONSTRAINT "PencegahanKriminalitas_programKeamananId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PencegahanKriminalitas" DROP CONSTRAINT "PencegahanKriminalitas_tipsKeamananId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PencegahanKriminalitas" DROP CONSTRAINT "PencegahanKriminalitas_videoKeamananId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "Penghargaan" DROP CONSTRAINT "Penghargaan_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PotensiDesa" DROP CONSTRAINT "PotensiDesa_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "hubungan_organisasi" DROP CONSTRAINT "hubungan_organisasi_atasanId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "hubungan_organisasi" DROP CONSTRAINT "hubungan_organisasi_bawahanId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "pegawai" DROP CONSTRAINT "pegawai_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "pegawai" DROP CONSTRAINT "pegawai_posisiId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "struktur_organisasi" DROP CONSTRAINT "struktur_organisasi_hubunganOrganisasiId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "struktur_organisasi" DROP CONSTRAINT "struktur_organisasi_pegawaiId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "struktur_organisasi" DROP CONSTRAINT "struktur_organisasi_posisiOrganisasiId_fkey";
+
+-- AlterTable
+ALTER TABLE "ArtikelKesehatan" ADD COLUMN "imageId" TEXT;
+
+-- AlterTable
+ALTER TABLE "DataKematian_Kelahiran" DROP COLUMN "kelahiranKasar",
+DROP COLUMN "kematianBayi",
+DROP COLUMN "kematianKasar",
+DROP COLUMN "tahun",
+ADD COLUMN "kelahiranId" TEXT NOT NULL,
+ADD COLUMN "kematianId" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "FileStorage" ADD COLUMN "category" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "GrafikKepuasan" DROP COLUMN "jumlah",
+DROP COLUMN "label",
+ADD COLUMN "alamat" TEXT NOT NULL,
+ADD COLUMN "jenisKelamin" TEXT NOT NULL,
+ADD COLUMN "nama" TEXT NOT NULL,
+ADD COLUMN "penyakit" TEXT NOT NULL,
+ADD COLUMN "tanggal" TIMESTAMP(3) NOT NULL;
+
+-- AlterTable
+ALTER TABLE "JadwalKegiatan" ALTER COLUMN "pendaftaranJadwalKegiatanId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "KolaborasiInovasi" DROP COLUMN "imageId";
+
+-- AlterTable
+ALTER TABLE "KontakDarurat" ADD COLUMN "whatsapp" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "KontakDaruratKeamanan" DROP COLUMN "imageId",
+ADD COLUMN "deletedAt" TIMESTAMP(3),
+ADD COLUMN "icon" TEXT NOT NULL,
+ADD COLUMN "isActive" BOOLEAN NOT NULL DEFAULT true,
+ADD COLUMN "kategoriId" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "KontakItem" DROP COLUMN "imageId",
+DROP COLUMN "kategoriId",
+ADD COLUMN "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ADD COLUMN "icon" TEXT NOT NULL,
+ADD COLUMN "isActive" BOOLEAN NOT NULL DEFAULT true;
+
+-- AlterTable
+ALTER TABLE "LaporanPublik" ADD COLUMN "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ADD COLUMN "isActive" BOOLEAN NOT NULL DEFAULT true,
+ALTER COLUMN "status" SET DEFAULT 'Proses';
+
+-- AlterTable
+ALTER TABLE "LowonganPekerjaan" ADD COLUMN "notelp" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "MediaSosial" ADD COLUMN "name" TEXT NOT NULL,
+ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PasarDesa" ADD COLUMN "kontak" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PelayananSuratKeterangan" ADD COLUMN "image2Id" TEXT,
+ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PencegahanKriminalitas" DROP COLUMN "programKeamananId",
+DROP COLUMN "tipsKeamananId",
+DROP COLUMN "videoKeamananId",
+ADD COLUMN "deskripsi" TEXT NOT NULL,
+ADD COLUMN "deskripsiSingkat" TEXT NOT NULL,
+ADD COLUMN "judul" TEXT NOT NULL,
+ADD COLUMN "linkVideo" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "Penghargaan" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "Posyandu" ADD COLUMN "jadwalPelayanan" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PotensiDesa" DROP COLUMN "kategori",
+ADD COLUMN "kategoriId" TEXT,
+ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "ProgramKemiskinan" DROP COLUMN "ikonUrl",
+ADD COLUMN "icon" TEXT NOT NULL;
+
+-- AlterTable
+ALTER TABLE "StrukturPPID" ADD COLUMN "pegawaiPPIDId" TEXT,
+ADD COLUMN "posisiOrganisasiPPIDId" TEXT;
+
+-- DropTable
+DROP TABLE "ProgramKeamanan";
+
+-- DropTable
+DROP TABLE "TipsKeamanan";
+
+-- DropTable
+DROP TABLE "VideoKeamanan";
+
+-- DropTable
+DROP TABLE "hubungan_organisasi";
+
+-- DropTable
+DROP TABLE "pegawai";
+
+-- DropTable
+DROP TABLE "posisi_organisasi";
+
+-- DropTable
+DROP TABLE "struktur_organisasi";
+
+-- CreateTable
+CREATE TABLE "DesaAntiKorupsi" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "kategoriId" TEXT NOT NULL,
+ "fileId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "DesaAntiKorupsi_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KategoriDesaAntiKorupsi" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "KategoriDesaAntiKorupsi_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "SdgsDesa" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "jumlah" TEXT NOT NULL,
+ "imageId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "SdgsDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "APBDes" (
+ "id" TEXT NOT NULL,
+ "tahun" INTEGER,
+ "name" TEXT,
+ "deskripsi" TEXT,
+ "jumlah" TEXT,
+ "imageId" TEXT,
+ "fileId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "APBDes_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "APBDesItem" (
+ "id" TEXT NOT NULL,
+ "kode" TEXT NOT NULL,
+ "uraian" TEXT NOT NULL,
+ "anggaran" DOUBLE PRECISION NOT NULL,
+ "realisasi" DOUBLE PRECISION NOT NULL,
+ "selisih" DOUBLE PRECISION NOT NULL,
+ "persentase" DOUBLE PRECISION NOT NULL,
+ "tipe" TEXT,
+ "level" INTEGER NOT NULL,
+ "parentId" TEXT,
+ "apbdesId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "APBDesItem_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PrestasiDesa" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "kategoriId" TEXT NOT NULL,
+ "imageId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "PrestasiDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KategoriPrestasiDesa" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "KategoriPrestasiDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Responden" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "tanggal" DATE NOT NULL,
+ "jenisKelaminId" TEXT NOT NULL,
+ "ratingId" TEXT NOT NULL,
+ "kelompokUmurId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Responden_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "JenisKelaminResponden" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "JenisKelaminResponden_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PilihanRatingResponden" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "PilihanRatingResponden_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "UmurResponden" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "UmurResponden_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PosisiOrganisasiPPID" (
+ "id" TEXT NOT NULL,
+ "nama" VARCHAR(100) NOT NULL,
+ "deskripsi" TEXT,
+ "hierarki" INTEGER NOT NULL,
+ "parentId" TEXT,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "PosisiOrganisasiPPID_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PegawaiPPID" (
+ "id" TEXT NOT NULL,
+ "namaLengkap" VARCHAR(255) NOT NULL,
+ "gelarAkademik" VARCHAR(100),
+ "imageId" TEXT,
+ "tanggalMasuk" DATE,
+ "email" VARCHAR(255),
+ "telepon" VARCHAR(20),
+ "alamat" TEXT,
+ "posisiId" VARCHAR(50) NOT NULL,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "PegawaiPPID_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "StrukturOrganisasiPPID" (
+ "id" TEXT NOT NULL,
+ "posisiOrganisasiId" VARCHAR(50) NOT NULL,
+ "pegawaiId" TEXT NOT NULL,
+ "hubunganOrganisasiId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "StrukturOrganisasiPPID_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PerbekelDariMasaKeMasa" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "periode" TEXT NOT NULL,
+ "imageId" TEXT,
+ "daerah" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "PerbekelDariMasaKeMasa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KategoriPotensi" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "KategoriPotensi_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "AjukanPermohonan" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "nik" TEXT NOT NULL,
+ "alamat" TEXT NOT NULL,
+ "nomorKk" TEXT NOT NULL,
+ "kategoriId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "AjukanPermohonan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Kelahiran" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "tanggal" TIMESTAMP(3) NOT NULL,
+ "jenisKelamin" TEXT NOT NULL,
+ "alamat" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Kelahiran_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Kematian" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "tanggal" TIMESTAMP(3) NOT NULL,
+ "jenisKelamin" TEXT NOT NULL,
+ "alamat" TEXT NOT NULL,
+ "penyebab" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Kematian_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KontakDaruratToItem" (
+ "id" TEXT NOT NULL,
+ "kontakDaruratId" TEXT NOT NULL,
+ "kontakItemId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+
+ CONSTRAINT "KontakDaruratToItem_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "StrukturBumDes" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "imageId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "posisiOrganisasiBumDesId" TEXT,
+ "pegawaiBumDesId" TEXT,
+
+ CONSTRAINT "StrukturBumDes_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PosisiOrganisasiBumDes" (
+ "id" TEXT NOT NULL,
+ "nama" VARCHAR(100) NOT NULL,
+ "deskripsi" TEXT,
+ "hierarki" INTEGER NOT NULL,
+ "parentId" TEXT,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "PosisiOrganisasiBumDes_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PegawaiBumDes" (
+ "id" TEXT NOT NULL,
+ "namaLengkap" VARCHAR(255) NOT NULL,
+ "gelarAkademik" VARCHAR(100),
+ "imageId" TEXT,
+ "tanggalMasuk" DATE,
+ "email" VARCHAR(255),
+ "telepon" VARCHAR(20),
+ "alamat" TEXT,
+ "posisiId" VARCHAR(50) NOT NULL,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "PegawaiBumDes_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "StrukturOrganisasiBumDes" (
+ "id" TEXT NOT NULL,
+ "posisiOrganisasiId" VARCHAR(50) NOT NULL,
+ "pegawaiId" TEXT NOT NULL,
+ "hubunganOrganisasiId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "StrukturOrganisasiBumDes_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "MitraKolaborasi" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "imageId" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "MitraKolaborasi_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "JenjangPendidikan" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "JenjangPendidikan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Lembaga" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "jenjangId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Lembaga_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Siswa" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "lembagaId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Siswa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "Pengajar" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "lembagaId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "Pengajar_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KeunggulanProgram" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "KeunggulanProgram_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "BeasiswaPendaftar" (
+ "id" TEXT NOT NULL,
+ "namaLengkap" TEXT NOT NULL,
+ "nis" TEXT,
+ "kelas" TEXT,
+ "jenisKelamin" "JenisKelamin" NOT NULL,
+ "alamatDomisili" TEXT,
+ "tempatLahir" TEXT NOT NULL,
+ "tanggalLahir" TIMESTAMP(3) NOT NULL,
+ "namaOrtu" TEXT,
+ "nik" TEXT NOT NULL,
+ "pekerjaanOrtu" TEXT,
+ "penghasilan" TEXT,
+ "noHp" TEXT NOT NULL,
+ "kewarganegaraan" TEXT,
+ "agama" "Agama",
+ "alamatKTP" TEXT,
+ "email" TEXT,
+ "statusPernikahan" "StatusPernikahan",
+ "ukuranBaju" "UkuranBaju",
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "BeasiswaPendaftar_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "TujuanProgram" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "TujuanProgram_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "ProgramUnggulan" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "ProgramUnggulan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "TujuanBimbinganBelajarDesa" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "TujuanBimbinganBelajarDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "LokasiJadwalBimbinganBelajarDesa" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "LokasiJadwalBimbinganBelajarDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "FasilitasBimbinganBelajarDesa" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "FasilitasBimbinganBelajarDesa_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "TujuanPendidikanNonFormal" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "TujuanPendidikanNonFormal_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "TempatKegiatan" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "TempatKegiatan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "JenisProgramYangDiselenggarakan" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "JenisProgramYangDiselenggarakan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "DataPerpustakaan" (
+ "id" TEXT NOT NULL,
+ "judul" TEXT NOT NULL,
+ "deskripsi" TEXT NOT NULL,
+ "kategoriId" TEXT NOT NULL,
+ "imageId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "DataPerpustakaan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KategoriBuku" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "KategoriBuku_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "PeminjamanBuku" (
+ "id" TEXT NOT NULL,
+ "nama" TEXT NOT NULL,
+ "noTelp" TEXT NOT NULL,
+ "alamat" TEXT NOT NULL,
+ "bukuId" TEXT NOT NULL,
+ "tanggalPinjam" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "batasKembali" TIMESTAMP(3) NOT NULL,
+ "tanggalKembali" TIMESTAMP(3),
+ "status" "StatusPeminjaman" NOT NULL DEFAULT 'Dipinjam',
+ "catatan" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "PeminjamanBuku_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "User" (
+ "id" TEXT NOT NULL,
+ "username" TEXT NOT NULL,
+ "nomor" TEXT NOT NULL,
+ "roleId" TEXT NOT NULL DEFAULT '1',
+ "instansi" TEXT,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "lastLogin" TIMESTAMP(3),
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+
+ CONSTRAINT "User_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "roles" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "description" TEXT,
+ "permissions" JSONB NOT NULL,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3),
+
+ CONSTRAINT "roles_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "KodeOtp" (
+ "id" TEXT NOT NULL,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "nomor" TEXT NOT NULL,
+ "otp" INTEGER NOT NULL,
+
+ CONSTRAINT "KodeOtp_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "permissions" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "description" TEXT,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "permissions_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "UserSession" (
+ "id" TEXT NOT NULL,
+ "token" TEXT NOT NULL,
+ "expires" TIMESTAMP(3),
+ "active" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "userId" TEXT NOT NULL,
+
+ CONSTRAINT "UserSession_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "DataPendidikan" (
+ "id" TEXT NOT NULL,
+ "name" TEXT NOT NULL,
+ "jumlah" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "DataPendidikan_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateIndex
+CREATE UNIQUE INDEX "DesaAntiKorupsi_name_key" ON "DesaAntiKorupsi"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "KategoriDesaAntiKorupsi_name_key" ON "KategoriDesaAntiKorupsi"("name");
+
+-- CreateIndex
+CREATE INDEX "APBDesItem_kode_idx" ON "APBDesItem"("kode");
+
+-- CreateIndex
+CREATE INDEX "APBDesItem_level_idx" ON "APBDesItem"("level");
+
+-- CreateIndex
+CREATE INDEX "APBDesItem_apbdesId_idx" ON "APBDesItem"("apbdesId");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "KategoriPrestasiDesa_name_key" ON "KategoriPrestasiDesa"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "Responden_name_key" ON "Responden"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "JenisKelaminResponden_name_key" ON "JenisKelaminResponden"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "PilihanRatingResponden_name_key" ON "PilihanRatingResponden"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "UmurResponden_name_key" ON "UmurResponden"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "PegawaiPPID_email_key" ON "PegawaiPPID"("email");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "PegawaiBumDes_email_key" ON "PegawaiBumDes"("email");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "BeasiswaPendaftar_nik_key" ON "BeasiswaPendaftar"("nik");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "BeasiswaPendaftar_email_key" ON "BeasiswaPendaftar"("email");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "User_username_key" ON "User"("username");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "User_nomor_key" ON "User"("nomor");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "roles_name_key" ON "roles"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "permissions_name_key" ON "permissions"("name");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "UserSession_userId_key" ON "UserSession"("userId");
+
+-- AddForeignKey
+ALTER TABLE "MediaSosial" ADD CONSTRAINT "MediaSosial_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DesaAntiKorupsi" ADD CONSTRAINT "DesaAntiKorupsi_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriDesaAntiKorupsi"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DesaAntiKorupsi" ADD CONSTRAINT "DesaAntiKorupsi_fileId_fkey" FOREIGN KEY ("fileId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "SdgsDesa" ADD CONSTRAINT "SdgsDesa_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "APBDes" ADD CONSTRAINT "APBDes_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "APBDes" ADD CONSTRAINT "APBDes_fileId_fkey" FOREIGN KEY ("fileId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "APBDesItem" ADD CONSTRAINT "APBDesItem_parentId_fkey" FOREIGN KEY ("parentId") REFERENCES "APBDesItem"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "APBDesItem" ADD CONSTRAINT "APBDesItem_apbdesId_fkey" FOREIGN KEY ("apbdesId") REFERENCES "APBDes"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PrestasiDesa" ADD CONSTRAINT "PrestasiDesa_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriPrestasiDesa"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PrestasiDesa" ADD CONSTRAINT "PrestasiDesa_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Responden" ADD CONSTRAINT "Responden_jenisKelaminId_fkey" FOREIGN KEY ("jenisKelaminId") REFERENCES "JenisKelaminResponden"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Responden" ADD CONSTRAINT "Responden_ratingId_fkey" FOREIGN KEY ("ratingId") REFERENCES "PilihanRatingResponden"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Responden" ADD CONSTRAINT "Responden_kelompokUmurId_fkey" FOREIGN KEY ("kelompokUmurId") REFERENCES "UmurResponden"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturPPID" ADD CONSTRAINT "StrukturPPID_posisiOrganisasiPPIDId_fkey" FOREIGN KEY ("posisiOrganisasiPPIDId") REFERENCES "PosisiOrganisasiPPID"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturPPID" ADD CONSTRAINT "StrukturPPID_pegawaiPPIDId_fkey" FOREIGN KEY ("pegawaiPPIDId") REFERENCES "PegawaiPPID"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PosisiOrganisasiPPID" ADD CONSTRAINT "PosisiOrganisasiPPID_parentId_fkey" FOREIGN KEY ("parentId") REFERENCES "PosisiOrganisasiPPID"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PegawaiPPID" ADD CONSTRAINT "PegawaiPPID_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PegawaiPPID" ADD CONSTRAINT "PegawaiPPID_posisiId_fkey" FOREIGN KEY ("posisiId") REFERENCES "PosisiOrganisasiPPID"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturOrganisasiPPID" ADD CONSTRAINT "StrukturOrganisasiPPID_posisiOrganisasiId_fkey" FOREIGN KEY ("posisiOrganisasiId") REFERENCES "PosisiOrganisasiPPID"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturOrganisasiPPID" ADD CONSTRAINT "StrukturOrganisasiPPID_pegawaiId_fkey" FOREIGN KEY ("pegawaiId") REFERENCES "PegawaiPPID"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PerbekelDariMasaKeMasa" ADD CONSTRAINT "PerbekelDariMasaKeMasa_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PotensiDesa" ADD CONSTRAINT "PotensiDesa_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriPotensi"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PotensiDesa" ADD CONSTRAINT "PotensiDesa_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PelayananSuratKeterangan" ADD CONSTRAINT "PelayananSuratKeterangan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PelayananSuratKeterangan" ADD CONSTRAINT "PelayananSuratKeterangan_image2Id_fkey" FOREIGN KEY ("image2Id") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "AjukanPermohonan" ADD CONSTRAINT "AjukanPermohonan_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "PelayananSuratKeterangan"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Penghargaan" ADD CONSTRAINT "Penghargaan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "JadwalKegiatan" ADD CONSTRAINT "JadwalKegiatan_pendaftaranJadwalKegiatanId_fkey" FOREIGN KEY ("pendaftaranJadwalKegiatanId") REFERENCES "PendaftaranJadwalKegiatan"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DataKematian_Kelahiran" ADD CONSTRAINT "DataKematian_Kelahiran_kematianId_fkey" FOREIGN KEY ("kematianId") REFERENCES "Kematian"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DataKematian_Kelahiran" ADD CONSTRAINT "DataKematian_Kelahiran_kelahiranId_fkey" FOREIGN KEY ("kelahiranId") REFERENCES "Kelahiran"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "ArtikelKesehatan" ADD CONSTRAINT "ArtikelKesehatan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "KontakDaruratKeamanan" ADD CONSTRAINT "KontakDaruratKeamanan_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KontakItem"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "KontakDaruratToItem" ADD CONSTRAINT "KontakDaruratToItem_kontakDaruratId_fkey" FOREIGN KEY ("kontakDaruratId") REFERENCES "KontakDaruratKeamanan"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "KontakDaruratToItem" ADD CONSTRAINT "KontakDaruratToItem_kontakItemId_fkey" FOREIGN KEY ("kontakItemId") REFERENCES "KontakItem"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturBumDes" ADD CONSTRAINT "StrukturBumDes_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturBumDes" ADD CONSTRAINT "StrukturBumDes_posisiOrganisasiBumDesId_fkey" FOREIGN KEY ("posisiOrganisasiBumDesId") REFERENCES "PosisiOrganisasiBumDes"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturBumDes" ADD CONSTRAINT "StrukturBumDes_pegawaiBumDesId_fkey" FOREIGN KEY ("pegawaiBumDesId") REFERENCES "PegawaiBumDes"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PosisiOrganisasiBumDes" ADD CONSTRAINT "PosisiOrganisasiBumDes_parentId_fkey" FOREIGN KEY ("parentId") REFERENCES "PosisiOrganisasiBumDes"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PegawaiBumDes" ADD CONSTRAINT "PegawaiBumDes_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PegawaiBumDes" ADD CONSTRAINT "PegawaiBumDes_posisiId_fkey" FOREIGN KEY ("posisiId") REFERENCES "PosisiOrganisasiBumDes"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturOrganisasiBumDes" ADD CONSTRAINT "StrukturOrganisasiBumDes_posisiOrganisasiId_fkey" FOREIGN KEY ("posisiOrganisasiId") REFERENCES "PosisiOrganisasiBumDes"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "StrukturOrganisasiBumDes" ADD CONSTRAINT "StrukturOrganisasiBumDes_pegawaiId_fkey" FOREIGN KEY ("pegawaiId") REFERENCES "PegawaiBumDes"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "MitraKolaborasi" ADD CONSTRAINT "MitraKolaborasi_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Lembaga" ADD CONSTRAINT "Lembaga_jenjangId_fkey" FOREIGN KEY ("jenjangId") REFERENCES "JenjangPendidikan"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Siswa" ADD CONSTRAINT "Siswa_lembagaId_fkey" FOREIGN KEY ("lembagaId") REFERENCES "Lembaga"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Pengajar" ADD CONSTRAINT "Pengajar_lembagaId_fkey" FOREIGN KEY ("lembagaId") REFERENCES "Lembaga"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DataPerpustakaan" ADD CONSTRAINT "DataPerpustakaan_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriBuku"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DataPerpustakaan" ADD CONSTRAINT "DataPerpustakaan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PeminjamanBuku" ADD CONSTRAINT "PeminjamanBuku_bukuId_fkey" FOREIGN KEY ("bukuId") REFERENCES "DataPerpustakaan"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "User" ADD CONSTRAINT "User_roleId_fkey" FOREIGN KEY ("roleId") REFERENCES "roles"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "UserSession" ADD CONSTRAINT "UserSession_userId_fkey" FOREIGN KEY ("userId") REFERENCES "User"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
diff --git a/prisma/migrations/20260106072549_nico_6_jan2025/migration.sql b/prisma/migrations/20260106072549_nico_6_jan2025/migration.sql
new file mode 100644
index 00000000..6952c280
--- /dev/null
+++ b/prisma/migrations/20260106072549_nico_6_jan2025/migration.sql
@@ -0,0 +1,142 @@
+/*
+ Warnings:
+
+ - You are about to drop the column `dokterdanTenagaMedisId` on the `FasilitasKesehatan` table. All the data in the column will be lost.
+ - You are about to drop the column `tarifDanLayananId` on the `FasilitasKesehatan` table. All the data in the column will be lost.
+ - You are about to drop the `User` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `UserSession` table. If the table is not empty, all the data it contains will be lost.
+ - You are about to drop the `permissions` table. If the table is not empty, all the data it contains will be lost.
+
+*/
+-- DropForeignKey
+ALTER TABLE "FasilitasKesehatan" DROP CONSTRAINT "FasilitasKesehatan_dokterdanTenagaMedisId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "FasilitasKesehatan" DROP CONSTRAINT "FasilitasKesehatan_tarifDanLayananId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "User" DROP CONSTRAINT "User_roleId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "UserSession" DROP CONSTRAINT "UserSession_userId_fkey";
+
+-- AlterTable
+ALTER TABLE "DokterdanTenagaMedis" ADD COLUMN "jadwalLibur" TEXT,
+ADD COLUMN "jamBukaLibur" TEXT,
+ADD COLUMN "jamBukaOperasional" TEXT,
+ADD COLUMN "jamTutupLibur" TEXT,
+ADD COLUMN "jamTutupOperasional" TEXT;
+
+-- AlterTable
+ALTER TABLE "FasilitasKesehatan" DROP COLUMN "dokterdanTenagaMedisId",
+DROP COLUMN "tarifDanLayananId";
+
+-- AlterTable
+ALTER TABLE "MediaSosial" ADD COLUMN "icon" TEXT;
+
+-- AlterTable
+ALTER TABLE "roles" ALTER COLUMN "permissions" DROP NOT NULL;
+
+-- DropTable
+DROP TABLE "User";
+
+-- DropTable
+DROP TABLE "UserSession";
+
+-- DropTable
+DROP TABLE "permissions";
+
+-- CreateTable
+CREATE TABLE "users" (
+ "id" TEXT NOT NULL,
+ "username" TEXT NOT NULL,
+ "nomor" TEXT NOT NULL,
+ "roleId" TEXT NOT NULL DEFAULT '2',
+ "isActive" BOOLEAN NOT NULL DEFAULT false,
+ "sessionInvalid" BOOLEAN NOT NULL DEFAULT false,
+ "lastLogin" TIMESTAMP(3),
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "permissions" JSONB,
+
+ CONSTRAINT "users_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "user_sessions" (
+ "id" TEXT NOT NULL,
+ "token" TEXT NOT NULL,
+ "expiresAt" TIMESTAMP(3) NOT NULL,
+ "active" BOOLEAN NOT NULL DEFAULT true,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "userId" TEXT NOT NULL,
+
+ CONSTRAINT "user_sessions_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "UserMenuAccess" (
+ "id" TEXT NOT NULL,
+ "userId" TEXT NOT NULL,
+ "menuId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+
+ CONSTRAINT "UserMenuAccess_pkey" PRIMARY KEY ("id")
+);
+
+-- CreateTable
+CREATE TABLE "_Tarif" (
+ "A" TEXT NOT NULL,
+ "B" TEXT NOT NULL,
+
+ CONSTRAINT "_Tarif_AB_pkey" PRIMARY KEY ("A","B")
+);
+
+-- CreateTable
+CREATE TABLE "_Dokter" (
+ "A" TEXT NOT NULL,
+ "B" TEXT NOT NULL,
+
+ CONSTRAINT "_Dokter_AB_pkey" PRIMARY KEY ("A","B")
+);
+
+-- CreateIndex
+CREATE UNIQUE INDEX "users_nomor_key" ON "users"("nomor");
+
+-- CreateIndex
+CREATE INDEX "user_sessions_userId_idx" ON "user_sessions"("userId");
+
+-- CreateIndex
+CREATE INDEX "user_sessions_token_idx" ON "user_sessions"("token");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "UserMenuAccess_userId_menuId_key" ON "UserMenuAccess"("userId", "menuId");
+
+-- CreateIndex
+CREATE INDEX "_Tarif_B_index" ON "_Tarif"("B");
+
+-- CreateIndex
+CREATE INDEX "_Dokter_B_index" ON "_Dokter"("B");
+
+-- AddForeignKey
+ALTER TABLE "users" ADD CONSTRAINT "users_roleId_fkey" FOREIGN KEY ("roleId") REFERENCES "roles"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "user_sessions" ADD CONSTRAINT "user_sessions_userId_fkey" FOREIGN KEY ("userId") REFERENCES "users"("id") ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "UserMenuAccess" ADD CONSTRAINT "UserMenuAccess_userId_fkey" FOREIGN KEY ("userId") REFERENCES "users"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "_Tarif" ADD CONSTRAINT "_Tarif_A_fkey" FOREIGN KEY ("A") REFERENCES "FasilitasKesehatan"("id") ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "_Tarif" ADD CONSTRAINT "_Tarif_B_fkey" FOREIGN KEY ("B") REFERENCES "TarifDanLayanan"("id") ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "_Dokter" ADD CONSTRAINT "_Dokter_A_fkey" FOREIGN KEY ("A") REFERENCES "DokterdanTenagaMedis"("id") ON DELETE CASCADE ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "_Dokter" ADD CONSTRAINT "_Dokter_B_fkey" FOREIGN KEY ("B") REFERENCES "FasilitasKesehatan"("id") ON DELETE CASCADE ON UPDATE CASCADE;
diff --git a/prisma/migrations/20260122074939_21_jan_26/migration.sql b/prisma/migrations/20260122074939_21_jan_26/migration.sql
new file mode 100644
index 00000000..3363be3d
--- /dev/null
+++ b/prisma/migrations/20260122074939_21_jan_26/migration.sql
@@ -0,0 +1,84 @@
+-- DropForeignKey
+ALTER TABLE "Berita" DROP CONSTRAINT "Berita_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "InfoWabahPenyakit" DROP CONSTRAINT "InfoWabahPenyakit_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KontakDarurat" DROP CONSTRAINT "KontakDarurat_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PenangananDarurat" DROP CONSTRAINT "PenangananDarurat_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "Posyandu" DROP CONSTRAINT "Posyandu_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "ProgramKesehatan" DROP CONSTRAINT "ProgramKesehatan_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "Puskesmas" DROP CONSTRAINT "Puskesmas_imageId_fkey";
+
+-- AlterTable
+ALTER TABLE "Berita" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "InfoWabahPenyakit" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "KontakDarurat" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PasarDesa" ADD COLUMN "deskripsi" TEXT;
+
+-- AlterTable
+ALTER TABLE "PenangananDarurat" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "Posyandu" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "ProgramKesehatan" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "Puskesmas" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- CreateTable
+CREATE TABLE "LayananToPolsek" (
+ "id" TEXT NOT NULL,
+ "layananId" TEXT NOT NULL,
+ "polsekTerdekatId" TEXT NOT NULL,
+ "createdAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "updatedAt" TIMESTAMP(3) NOT NULL,
+ "deletedAt" TIMESTAMP(3) NOT NULL DEFAULT CURRENT_TIMESTAMP,
+ "isActive" BOOLEAN NOT NULL DEFAULT true,
+
+ CONSTRAINT "LayananToPolsek_pkey" PRIMARY KEY ("id")
+);
+
+-- AddForeignKey
+ALTER TABLE "Berita" ADD CONSTRAINT "Berita_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Posyandu" ADD CONSTRAINT "Posyandu_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "Puskesmas" ADD CONSTRAINT "Puskesmas_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "ProgramKesehatan" ADD CONSTRAINT "ProgramKesehatan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PenangananDarurat" ADD CONSTRAINT "PenangananDarurat_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "KontakDarurat" ADD CONSTRAINT "KontakDarurat_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "InfoWabahPenyakit" ADD CONSTRAINT "InfoWabahPenyakit_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "LayananToPolsek" ADD CONSTRAINT "LayananToPolsek_layananId_fkey" FOREIGN KEY ("layananId") REFERENCES "LayananPolsek"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "LayananToPolsek" ADD CONSTRAINT "LayananToPolsek_polsekTerdekatId_fkey" FOREIGN KEY ("polsekTerdekatId") REFERENCES "PolsekTerdekat"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
diff --git a/prisma/migrations/20260225082505_deploy/migration.sql b/prisma/migrations/20260225082505_deploy/migration.sql
new file mode 100644
index 00000000..ac4024a7
--- /dev/null
+++ b/prisma/migrations/20260225082505_deploy/migration.sql
@@ -0,0 +1,170 @@
+/*
+ Warnings:
+
+ - You are about to alter the column `nama` on the `KategoriPotensi` table. The data in that column could be lost. The data in that column will be cast from `Text` to `VarChar(100)`.
+ - You are about to alter the column `name` on the `PotensiDesa` table. The data in that column could be lost. The data in that column will be cast from `Text` to `VarChar(255)`.
+ - You are about to alter the column `kategoriId` on the `PotensiDesa` table. The data in that column could be lost. The data in that column will be cast from `Text` to `VarChar(36)`.
+ - A unique constraint covering the columns `[nama]` on the table `KategoriPotensi` will be added. If there are existing duplicate values, this will fail.
+ - A unique constraint covering the columns `[name]` on the table `PotensiDesa` will be added. If there are existing duplicate values, this will fail.
+ - Made the column `kategoriId` on table `PotensiDesa` required. This step will fail if there are existing NULL values in that column.
+
+*/
+-- DropForeignKey
+ALTER TABLE "DataPerpustakaan" DROP CONSTRAINT "DataPerpustakaan_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "DesaDigital" DROP CONSTRAINT "DesaDigital_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "InfoTekno" DROP CONSTRAINT "InfoTekno_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "KegiatanDesa" DROP CONSTRAINT "KegiatanDesa_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PengaduanMasyarakat" DROP CONSTRAINT "PengaduanMasyarakat_imageId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "PotensiDesa" DROP CONSTRAINT "PotensiDesa_kategoriId_fkey";
+
+-- DropForeignKey
+ALTER TABLE "ProfileDesaImage" DROP CONSTRAINT "ProfileDesaImage_imageId_fkey";
+
+-- AlterTable
+ALTER TABLE "CaraMemperolehInformasi" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "CaraMemperolehSalinanInformasi" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "DaftarInformasiPublik" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "DasarHukumPPID" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "DataPerpustakaan" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "DesaDigital" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "FormulirPermohonanKeberatan" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "InfoTekno" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "JenisInformasiDiminta" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "JenisKelaminResponden" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "KategoriPotensi" ALTER COLUMN "nama" SET DATA TYPE VARCHAR(100),
+ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "KategoriPrestasiDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "KegiatanDesa" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "LambangDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "MaskotDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "PegawaiPPID" ADD COLUMN "deletedAt" TIMESTAMP(3);
+
+-- AlterTable
+ALTER TABLE "PengaduanMasyarakat" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "PermohonanInformasiPublik" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "PilihanRatingResponden" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "PosisiOrganisasiPPID" ADD COLUMN "deletedAt" TIMESTAMP(3);
+
+-- AlterTable
+ALTER TABLE "PotensiDesa" ALTER COLUMN "name" SET DATA TYPE VARCHAR(255),
+ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT,
+ALTER COLUMN "kategoriId" SET NOT NULL,
+ALTER COLUMN "kategoriId" SET DATA TYPE VARCHAR(36);
+
+-- AlterTable
+ALTER TABLE "PrestasiDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "ProfileDesaImage" ALTER COLUMN "imageId" DROP NOT NULL;
+
+-- AlterTable
+ALTER TABLE "ProfilePPID" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "Responden" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "SejarahDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "UmurResponden" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "VisiMisiDesa" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- AlterTable
+ALTER TABLE "VisiMisiPPID" ALTER COLUMN "deletedAt" DROP NOT NULL,
+ALTER COLUMN "deletedAt" DROP DEFAULT;
+
+-- CreateIndex
+CREATE UNIQUE INDEX "KategoriPotensi_nama_key" ON "KategoriPotensi"("nama");
+
+-- CreateIndex
+CREATE UNIQUE INDEX "PotensiDesa_name_key" ON "PotensiDesa"("name");
+
+-- AddForeignKey
+ALTER TABLE "ProfileDesaImage" ADD CONSTRAINT "ProfileDesaImage_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PotensiDesa" ADD CONSTRAINT "PotensiDesa_kategoriId_fkey" FOREIGN KEY ("kategoriId") REFERENCES "KategoriPotensi"("id") ON DELETE RESTRICT ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DesaDigital" ADD CONSTRAINT "DesaDigital_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "InfoTekno" ADD CONSTRAINT "InfoTekno_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "PengaduanMasyarakat" ADD CONSTRAINT "PengaduanMasyarakat_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "KegiatanDesa" ADD CONSTRAINT "KegiatanDesa_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
+
+-- AddForeignKey
+ALTER TABLE "DataPerpustakaan" ADD CONSTRAINT "DataPerpustakaan_imageId_fkey" FOREIGN KEY ("imageId") REFERENCES "FileStorage"("id") ON DELETE SET NULL ON UPDATE CASCADE;
diff --git a/prisma/migrations/migration_lock.toml b/prisma/migrations/migration_lock.toml
index 648c57fd..044d57cd 100644
--- a/prisma/migrations/migration_lock.toml
+++ b/prisma/migrations/migration_lock.toml
@@ -1,3 +1,3 @@
# Please do not edit this file manually
# It should be added in your version-control system (e.g., Git)
-provider = "postgresql"
\ No newline at end of file
+provider = "postgresql"
diff --git a/prisma/newseed-asset.ts b/prisma/newseed-asset.ts
new file mode 100644
index 00000000..fabdbafc
--- /dev/null
+++ b/prisma/newseed-asset.ts
@@ -0,0 +1,82 @@
+type DirItem = {
+ type: "file" | "dir";
+ name: string;
+ path: string;
+ size?: number;
+};
+
+type FileDownload = {
+ name: string;
+ path: string;
+ link: string;
+};
+
+const TOKEN = process.env.SEAFILE_TOKEN!;
+const REPO_ID = process.env.SEAFILE_REPO_ID!;
+
+// ⛔ PENTING: RELATIVE PATH (tanpa slash depan)
+const DIR_TARGET = "asset-web";
+
+const BASE_URL = "https://cld-dkr-makuro-seafile.wibudev.com/api2";
+
+const headers = {
+ Authorization: `Token ${TOKEN}`,
+};
+
+/**
+ * Ambil list item di directory
+ */
+async function getDirItems(): Promise {
+ const res = await fetch(
+ `${BASE_URL}/repos/${REPO_ID}/dir/?p=${encodeURIComponent(DIR_TARGET)}`,
+ { headers },
+ );
+
+ if (!res.ok) {
+ const body = await res.text();
+ throw new Error(`Failed get dir items: ${body}`);
+ }
+
+ return res.json();
+}
+
+
+/**
+ * Ambil download URL file
+ */
+async function getDownloadUrl(filePath: string): Promise {
+ const res = await fetch(
+ `${BASE_URL}/repos/${REPO_ID}/file/?p=${encodeURIComponent(filePath)}`,
+ { headers },
+ );
+
+ if (!res.ok) {
+ const body = await res.text();
+ throw new Error(`Failed get file url: ${body}`);
+ }
+
+ const data = await res.json();
+ return data.url;
+}
+
+/**
+ * Ambil semua download URL dari folder image
+ */
+export async function getImageDownloadList(): Promise {
+ const items = await getDirItems();
+
+ const files = items.filter((item) => item.type === "file");
+
+ return Promise.all(
+ files.map(async (file) => {
+ const filePath = `${DIR_TARGET}/${file.name}`; // → /image/xxx.webp
+
+ return {
+ name: file.name,
+ path: filePath,
+ link: await getDownloadUrl(filePath),
+ };
+ }),
+ );
+}
+
diff --git a/prisma/resolveImageByName.ts b/prisma/resolveImageByName.ts
new file mode 100644
index 00000000..e105eef3
--- /dev/null
+++ b/prisma/resolveImageByName.ts
@@ -0,0 +1,78 @@
+// import prisma from "@/lib/prisma";
+
+// // Ganti nama fungsi dan logikanya
+// export default async function resolveImageById(
+// imageId?: string | null
+// ): Promise {
+// if (!imageId) return null;
+
+// const image = await prisma.fileStorage.findFirst({
+// where: {
+// id: imageId, // ← cari berdasarkan ID
+// category: "image",
+// isActive: true,
+// deletedAt: null,
+// },
+// select: { id: true },
+// });
+
+// if (!image) {
+// console.warn(`⚠️ Image with ID ${imageId} not found`);
+// return null;
+// }
+
+// return image.id;
+// }
+
+import prisma from "@/lib/prisma";
+
+/**
+ * Resolve image ID by checking multiple possible names
+ * @param imageId - The ID from JSON (could be filename or actual ID)
+ * @returns The actual database ID or null
+ */
+export default async function resolveImageById(imageId: string | null): Promise {
+ if (!imageId) return null;
+
+ try {
+ // 1. Coba cari berdasarkan ID langsung
+ const byId = await prisma.fileStorage.findUnique({
+ where: { id: imageId },
+ select: { id: true },
+ });
+ if (byId) return byId.id;
+
+ // 2. Coba cari berdasarkan name (exact match)
+ const byName = await prisma.fileStorage.findUnique({
+ where: { name: imageId },
+ select: { id: true },
+ });
+ if (byName) return byName.id;
+
+ // 3. Coba cari berdasarkan realName
+ const byRealName = await prisma.fileStorage.findFirst({
+ where: { realName: imageId },
+ select: { id: true },
+ });
+ if (byRealName) return byRealName.id;
+
+ // 4. Coba dengan menambahkan ekstensi .webp
+ const withWebp = `${imageId.replace(/\.(jpg|jpeg|png)$/i, '')}.webp`;
+ const byWebp = await prisma.fileStorage.findFirst({
+ where: {
+ OR: [
+ { name: withWebp },
+ { name: { contains: imageId.split('.')[0] } },
+ ],
+ },
+ select: { id: true },
+ });
+ if (byWebp) return byWebp.id;
+
+ console.warn(`⚠️ Image not found for: ${imageId}`);
+ return null;
+ } catch (error) {
+ console.error(`❌ Error resolving image ${imageId}:`, error);
+ return null;
+ }
+}
\ No newline at end of file
diff --git a/prisma/safeSeedMany.ts b/prisma/safeSeedMany.ts
new file mode 100644
index 00000000..d909bb2c
--- /dev/null
+++ b/prisma/safeSeedMany.ts
@@ -0,0 +1,45 @@
+/* eslint-disable @typescript-eslint/no-unused-vars */
+/* eslint-disable @typescript-eslint/no-explicit-any */
+import { PrismaClient } from "@prisma/client";
+import { safeSeedUnique } from "./safeseedUnique";
+import cliProgress from 'cli-progress';
+
+type SafeSeedOptions = {
+ skipUpdate?: boolean;
+ silent?: boolean; // Opsional: untuk suppress log
+};
+
+/**
+ * Batch upsert with progress logging
+ */
+export async function safeSeedMany(
+ model: T,
+ items: Array<{ where: Record; data: Record }>,
+ options: SafeSeedOptions = {}
+) {
+ const bar = new cliProgress.SingleBar({}, cliProgress.Presets.shades_classic);
+ bar.start(items.length, 0);
+
+ let success = 0;
+ let failed = 0;
+ let skipped = 0;
+
+ for (const [index, item] of items.entries()) {
+ try {
+ const result = await safeSeedUnique(model, item.where, item.data, {
+ ...options,
+ silent: true,
+ });
+ if (result) success++;
+ else skipped++;
+ } catch (err) {
+ failed++;
+ }
+ bar.update(index + 1);
+ }
+
+ bar.stop();
+ console.log(`✅ ${String(model)}: ${success} seeded, ${skipped} skipped, ${failed} failed`);
+
+ return { success, skipped, failed };
+}
\ No newline at end of file
diff --git a/prisma/safeseedUnique.ts b/prisma/safeseedUnique.ts
index 92d16071..e0ea0d52 100644
--- a/prisma/safeseedUnique.ts
+++ b/prisma/safeseedUnique.ts
@@ -1,30 +1,134 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
-// helpers/safeSeedUnique.ts
+import prisma from "@/lib/prisma";
import { PrismaClient } from "@prisma/client";
-const prisma = new PrismaClient();
+type SafeSeedOptions = {
+ skipUpdate?: boolean;
+ silent?: boolean; // Opsional: untuk suppress log
+};
/**
- * Helper generic buat seed dengan upsert aman
+ * Safely upsert data with error handling
+ * @param model - Prisma model name
+ * @param where - Unique identifier(s)
+ * @param data - Full data object (will be used for create)
+ * @param options - Additional options
*/
export async function safeSeedUnique(
model: T,
where: Record,
- data: Record
+ data: Record,
+ options: SafeSeedOptions = {}
) {
- const m = prisma[model];
-
- if (!m) throw new Error(`Model ${String(model)} tidak ditemukan di PrismaClient`);
+ const m = prisma[model] as any;
+
+ if (!m) {
+ throw new Error(`❌ Model ${String(model)} tidak ditemukan di Prisma Client`);
+ }
try {
- // @ts-expect-error upsert dynamic
- await m.upsert({
+ const result = await m.upsert({
where,
- update: data,
- create: { ...where, ...data },
+ update: options.skipUpdate ? {} : data,
+ create: data,
});
- console.log(`✅ Seeded ${String(model)} -> ${JSON.stringify(where)}`);
- } catch (err) {
- console.error(`❌ Gagal seed ${String(model)} -> ${JSON.stringify(where)}`, err);
+
+ if (!options.silent) {
+ console.log(`✅ Seeded ${String(model)}:`, where);
+ }
+
+ return result;
+ } catch (err: any) {
+ // Handle specific Prisma errors
+ if (err.code === "P2002") {
+ console.warn(`⚠️ Duplicate ${String(model)} (skipped):`, where);
+ return null;
+ }
+
+ if (err.code === "P2003") {
+ console.error(`❌ Foreign key constraint failed for ${String(model)}:`, where);
+ console.error(" Missing relation:", err.meta?.field_name);
+ throw err;
+ }
+
+ if (err.code === "P2025") {
+ console.error(`❌ Record not found for ${String(model)}:`, where);
+ throw err;
+ }
+
+ // Log unexpected errors with full details
+ console.error(`❌ Failed to seed ${String(model)}:`, where);
+ console.error(" Error:", err.message);
+ console.error(" Code:", err.code);
+
+ throw err;
}
}
+
+
+
+//ini yang bener pertama
+
+// /* eslint-disable @typescript-eslint/no-explicit-any */
+// import prisma from "@/lib/prisma";
+// import { PrismaClient } from "@prisma/client";
+
+// type SafeSeedOptions = {
+// skipUpdate?: boolean;
+// };
+
+// // prisma/safeseedUnique.ts
+// export async function safeSeedUnique(
+// model: T,
+// where: Record,
+// data: Record,
+// options: SafeSeedOptions = {}
+// ) {
+// const m = prisma[model] as any;
+// if (!m) throw new Error(`Model ${String(model)} tidak ditemukan`);
+
+// try {
+// // Pastikan `where` berisi field yang benar-benar unique (misal: `id`)
+// const result = await m.upsert({
+// where,
+// update: options.skipUpdate ? {} : data,
+// create: data, // ✅ Jangan duplikasi `where` ke `create`
+// });
+// console.log(`✅ Seed ${String(model)}:`, where);
+// return result;
+// } catch (err) {
+// console.error(`❌ Gagal seed ${String(model)}:`, where, err);
+// throw err; // ✅ Rethrow agar seeding berhenti jika kritis
+// }
+// }
+
+// /* eslint-disable @typescript-eslint/no-explicit-any */
+// import { PrismaClient } from "@prisma/client";
+
+// const prisma = new PrismaClient();
+
+// type SafeSeedOptions = {
+// skipUpdate?: boolean;
+// };
+
+// export async function safeSeedUnique(
+// model: T,
+// where: Record,
+// data: Record,
+// options: SafeSeedOptions = {}
+// ) {
+// const m = prisma[model] as any;
+// if (!m) throw new Error(`Model ${String(model)} tidak ditemukan`);
+
+// try {
+// await m.upsert({
+// where,
+// update: options.skipUpdate ? {} : data,
+// create: { ...where, ...data },
+// });
+
+// console.log(`✅ Seed ${String(model)}:`, where);
+// } catch (err) {
+// console.error(`❌ Gagal seed ${String(model)}:`, where, err);
+// }
+// }
diff --git a/prisma/schema.prisma b/prisma/schema.prisma
index a52a0cb1..5a9168cd 100644
--- a/prisma/schema.prisma
+++ b/prisma/schema.prisma
@@ -136,6 +136,7 @@ model MediaSosial {
name String
image FileStorage? @relation(fields: [imageId], references: [id])
imageId String?
+ icon String?
iconUrl String? @db.VarChar(255)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@ -184,18 +185,46 @@ model SdgsDesa {
//========================================= APBDes ========================================= //
model APBDes {
id String @id @default(cuid())
- name String
- jumlah String
+ tahun Int?
+ name String? // misalnya: "APBDes Tahun 2025"
+ deskripsi String?
+ jumlah String? // total keseluruhan (opsional, bisa juga dihitung dari items)
+ items APBDesItem[]
image FileStorage? @relation("APBDesImage", fields: [imageId], references: [id])
imageId String?
file FileStorage? @relation("APBDesFile", fields: [fileId], references: [id])
fileId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime? // opsional, tidak perlu default now()
isActive Boolean @default(true)
}
+model APBDesItem {
+ id String @id @default(cuid())
+ kode String // contoh: "4", "4.1", "4.1.2"
+ uraian String // nama item, contoh: "Pendapatan Asli Desa", "Hasil Usaha"
+ anggaran Float // dalam satuan Rupiah (bisa DECIMAL di DB, tapi Float umum di TS/JS)
+ realisasi Float
+ selisih Float // realisasi - anggaran
+ persentase Float
+ tipe String? // (realisasi / anggaran) * 100
+ level Int // 1 = kelompok utama, 2 = sub-kelompok, 3 = detail
+ parentId String? // untuk relasi hierarki
+ parent APBDesItem? @relation("APBDesItemParent", fields: [parentId], references: [id])
+ children APBDesItem[] @relation("APBDesItemParent")
+ apbdesId String
+ apbdes APBDes @relation(fields: [apbdesId], references: [id])
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime?
+ isActive Boolean @default(true)
+
+ @@index([kode])
+ @@index([level])
+ @@index([apbdesId])
+}
+
//========================================= PRESTASI DESA ========================================= //
model PrestasiDesa {
id String @id @default(cuid())
@@ -207,7 +236,7 @@ model PrestasiDesa {
imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -216,7 +245,7 @@ model KategoriPrestasiDesa {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
PrestasiDesa PrestasiDesa[]
}
@@ -234,7 +263,7 @@ model Responden {
kelompokUmurId String
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -243,7 +272,7 @@ model JenisKelaminResponden {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
Responden Responden[]
}
@@ -253,7 +282,7 @@ model PilihanRatingResponden {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
Responden Responden[]
}
@@ -263,7 +292,7 @@ model UmurResponden {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
Responden Responden[]
}
@@ -297,6 +326,7 @@ model PosisiOrganisasiPPID {
isActive Boolean @default(true)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
+ deletedAt DateTime?
parent PosisiOrganisasiPPID? @relation("Parent", fields: [parentId], references: [id])
children PosisiOrganisasiPPID[] @relation("Parent")
StrukturOrganisasiPPID StrukturOrganisasiPPID[]
@@ -316,6 +346,7 @@ model PegawaiPPID {
isActive Boolean @default(true)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
+ deletedAt DateTime?
posisi PosisiOrganisasiPPID @relation(fields: [posisiId], references: [id])
strukturOrganisasi StrukturPPID[] // Relasi balik
StrukturOrganisasiPPID StrukturOrganisasiPPID[]
@@ -341,7 +372,7 @@ model VisiMisiPPID {
misi String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -352,7 +383,7 @@ model DasarHukumPPID {
content String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -369,7 +400,7 @@ model ProfilePPID {
imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -381,7 +412,7 @@ model DaftarInformasiPublik {
tanggal DateTime @db.Date
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -402,7 +433,7 @@ model PermohonanInformasiPublik {
caraMemperolehSalinanInformasiId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -411,7 +442,7 @@ model JenisInformasiDiminta {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
PermohonanInformasiPublik PermohonanInformasiPublik[]
}
@@ -421,7 +452,7 @@ model CaraMemperolehInformasi {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
PermohonanInformasiPublik PermohonanInformasiPublik[]
}
@@ -431,7 +462,7 @@ model CaraMemperolehSalinanInformasi {
name String @unique
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
PermohonanInformasiPublik PermohonanInformasiPublik[]
}
@@ -445,7 +476,7 @@ model FormulirPermohonanKeberatan {
alasan String
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -502,7 +533,7 @@ model SejarahDesa {
deskripsi String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -512,7 +543,7 @@ model VisiMisiDesa {
misi String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -522,7 +553,7 @@ model LambangDesa {
deskripsi String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
@@ -533,15 +564,15 @@ model MaskotDesa {
images ProfileDesaImage[]
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
model ProfileDesaImage {
id String @id @default(cuid())
label String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
MaskotDesa MaskotDesa @relation(fields: [maskotDesaId], references: [id])
maskotDesaId String
}
@@ -578,8 +609,8 @@ model Berita {
id String @id @default(cuid())
judul String
deskripsi String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
content String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@ -602,25 +633,25 @@ model KategoriBerita {
// ========================================= POTENSI DESA ========================================= //
model PotensiDesa {
id String @id @default(cuid())
- name String
- deskripsi String
+ name String @unique @db.VarChar(255)
+ deskripsi String @db.Text
kategori KategoriPotensi? @relation(fields: [kategoriId], references: [id])
- kategoriId String?
+ kategoriId String @db.VarChar(36)
image FileStorage? @relation(fields: [imageId], references: [id])
imageId String?
content String @db.Text
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
}
model KategoriPotensi {
id String @id @default(cuid())
- nama String
+ nama String @unique @db.VarChar(100)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
+ deletedAt DateTime?
isActive Boolean @default(true)
PotensiDesa PotensiDesa[]
}
@@ -754,24 +785,22 @@ model Penghargaan {
// ========================================= FASILITAS KESEHATAN ========================================= //
model FasilitasKesehatan {
- id String @id @default(cuid())
- name String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
- informasiumum InformasiUmum @relation(fields: [informasiUmumId], references: [id])
- informasiUmumId String
- layananunggulan LayananUnggulan @relation(fields: [layananUnggulanId], references: [id])
- layananUnggulanId String
- dokterdantenagamedis DokterdanTenagaMedis @relation(fields: [dokterdanTenagaMedisId], references: [id])
- dokterdanTenagaMedisId String
- fasilitaspendukung FasilitasPendukung @relation(fields: [fasilitasPendukungId], references: [id])
- fasilitasPendukungId String
- prosedurpendaftaran ProsedurPendaftaran @relation(fields: [prosedurPendaftaranId], references: [id])
- prosedurPendaftaranId String
- tarifdanlayanan TarifDanLayanan @relation(fields: [tarifDanLayananId], references: [id])
- tarifDanLayananId String
+ id String @id @default(cuid())
+ name String
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
+ informasiumum InformasiUmum @relation(fields: [informasiUmumId], references: [id])
+ informasiUmumId String
+ layananunggulan LayananUnggulan @relation(fields: [layananUnggulanId], references: [id])
+ layananUnggulanId String
+ dokterdantenagamedis DokterdanTenagaMedis[] @relation("Dokter")
+ fasilitaspendukung FasilitasPendukung @relation(fields: [fasilitasPendukungId], references: [id])
+ fasilitasPendukungId String
+ prosedurpendaftaran ProsedurPendaftaran @relation(fields: [prosedurPendaftaranId], references: [id])
+ prosedurPendaftaranId String
+ tarifdanlayanan TarifDanLayanan[] @relation("Tarif")
}
model InformasiUmum {
@@ -797,15 +826,20 @@ model LayananUnggulan {
}
model DokterdanTenagaMedis {
- id String @id @default(cuid())
- name String
- specialist String
- jadwal String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
- FasilitasKesehatan FasilitasKesehatan[]
+ id String @id @default(cuid())
+ name String
+ specialist String
+ jadwal String
+ jadwalLibur String?
+ jamBukaOperasional String?
+ jamTutupOperasional String?
+ jamBukaLibur String?
+ jamTutupLibur String?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
+ FasilitasKesehatan FasilitasKesehatan[] @relation("Dokter")
}
model FasilitasPendukung {
@@ -836,7 +870,7 @@ model TarifDanLayanan {
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
isActive Boolean @default(true)
- FasilitasKesehatan FasilitasKesehatan[]
+ FasilitasKesehatan FasilitasKesehatan[] @relation("Tarif")
}
// ========================================= JADWAL KEGIATAN ========================================= //
@@ -1081,17 +1115,17 @@ model DoctorSign {
// ========================================= POSYANDU ========================================= //
model Posyandu {
- id String @id @default(cuid())
+ id String @id @default(cuid())
name String
nomor String
deskripsi String
jadwalPelayanan String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= PUSKESMAS ========================================= //
@@ -1101,8 +1135,8 @@ model Puskesmas {
alamat String
jam JamOperasional @relation(fields: [jamId], references: [id])
jamId String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
kontak KontakPuskesmas @relation(fields: [kontakId], references: [id])
kontakId String
createdAt DateTime @default(now())
@@ -1138,57 +1172,57 @@ model KontakPuskesmas {
// ========================================= PROGRAM KESSEHATAN ========================================= //
model ProgramKesehatan {
- id String @id @default(cuid())
+ id String @id @default(cuid())
name String
deskripsiSingkat String
deskripsi String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= PENANGANAN DARURAT ========================================= //
model PenangananDarurat {
- id String @id @default(cuid())
+ id String @id @default(cuid())
name String
deskripsi String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= KONTAK DARURAT ========================================= //
model KontakDarurat {
- id String @id @default(cuid())
+ id String @id @default(cuid())
name String
deskripsi String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
whatsapp String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= INFO WABAH PENYAKIT ========================================= //
model InfoWabahPenyakit {
- id String @id @default(cuid())
+ id String @id @default(cuid())
name String
deskripsiSingkat String
deskripsiLengkap String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= MENU KEAMANAN ========================================= //
@@ -1207,7 +1241,7 @@ model KeamananLingkungan {
// ========================================= POLSEK TERDEKAT ========================================= //
model PolsekTerdekat {
- id String @id @default(uuid())
+ id String @id @default(uuid())
nama String
jarakKeDesa String
alamat String
@@ -1217,22 +1251,36 @@ model PolsekTerdekat {
namaTempatMaps String
alamatMaps String
linkPetunjukArah String
- layananPolsek LayananPolsek @relation(fields: [layananPolsekId], references: [id])
+ layananPolsek LayananPolsek @relation(fields: [layananPolsekId], references: [id])
layananPolsekId String
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
+ LayananToPolsek LayananToPolsek[]
}
model LayananPolsek {
- id String @id @default(uuid())
- nama String // contoh: "Pelayanan SKCK", "Laporan Kriminal"
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime?
- isActive Boolean @default(true)
- PolsekTerdekat PolsekTerdekat[]
+ id String @id @default(uuid())
+ nama String // contoh: "Pelayanan SKCK", "Laporan Kriminal"
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime?
+ isActive Boolean @default(true)
+ PolsekTerdekat PolsekTerdekat[]
+ LayananToPolsek LayananToPolsek[]
+}
+
+model LayananToPolsek {
+ id String @id @default(uuid())
+ layanan LayananPolsek @relation(fields: [layananId], references: [id])
+ layananId String
+ polsekTerdekat PolsekTerdekat @relation(fields: [polsekTerdekatId], references: [id])
+ polsekTerdekatId String
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
}
// ========================================= KONTAK DARURAT ========================================= //
@@ -1345,6 +1393,7 @@ model PasarDesa {
rating Float
alamatUsaha String
kontak String
+ deskripsi String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
@@ -1612,8 +1661,8 @@ model DesaDigital {
id String @id @default(cuid())
name String
deskripsi String @db.Text
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
@@ -1663,8 +1712,8 @@ model InfoTekno {
id String @id @default(cuid())
name String
deskripsi String @db.Text
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
@@ -1719,8 +1768,8 @@ model PengaduanMasyarakat {
nik String
judulPengaduan String
lokasiKejadian String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
deskripsiPengaduan String @db.Text
jenisPengaduan JenisPengaduan @relation(fields: [jenisPengaduanId], references: [id])
jenisPengaduanId String
@@ -1801,8 +1850,8 @@ model KegiatanDesa {
tanggal DateTime
lokasi String
partisipan Int
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
@@ -1887,7 +1936,7 @@ model NilaiKonservasiAdat {
// ========================================= INFO SEKOLAH & PAUD ========================================= //
model JenjangPendidikan {
id String @id @default(cuid())
- nama String // TK/PAUD, SD, SMP, SMA/SMK
+ nama String // TK/PAUD, SD, SMP, SMA/SMK
lembagas Lembaga[] // Relasi ke lembaga
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@ -1942,23 +1991,28 @@ model KeunggulanProgram {
}
model BeasiswaPendaftar {
- id String @id @default(cuid())
+ id String @id @default(cuid())
namaLengkap String
- nik String @unique
+ nis String?
+ kelas String?
+ jenisKelamin JenisKelamin
+ alamatDomisili String?
tempatLahir String
tanggalLahir DateTime
- jenisKelamin JenisKelamin
- kewarganegaraan String
- agama Agama
- alamatKTP String
- alamatDomisili String?
+ namaOrtu String?
+ nik String @unique
+ pekerjaanOrtu String?
+ penghasilan String?
noHp String
- email String @unique
- statusPernikahan StatusPernikahan
+ kewarganegaraan String?
+ agama Agama?
+ alamatKTP String?
+ email String? @unique
+ statusPernikahan StatusPernikahan?
ukuranBaju UkuranBaju?
- isActive Boolean @default(true)
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
+ isActive Boolean @default(true)
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
}
enum JenisKelamin {
@@ -2081,8 +2135,8 @@ model DataPerpustakaan {
deskripsi String @db.Text
kategori KategoriBuku @relation(fields: [kategoriId], references: [id])
kategoriId String
- image FileStorage @relation(fields: [imageId], references: [id])
- imageId String
+ image FileStorage? @relation(fields: [imageId], references: [id])
+ imageId String?
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
deletedAt DateTime @default(now())
@@ -2127,28 +2181,43 @@ enum StatusPeminjaman {
Dibatalkan
}
+
+// ========================================= DATA PENDIDIKAN ========================================= //
+model DataPendidikan {
+ id String @id @default(cuid())
+ name String
+ jumlah String
+ createdAt DateTime @default(now())
+ updatedAt DateTime @updatedAt
+ deletedAt DateTime @default(now())
+ isActive Boolean @default(true)
+}
+
// ========================================= USER ========================================= //
model User {
- id String @id @default(cuid())
- username String
- nomor String @unique
- role Role @relation(fields: [roleId], references: [id])
- roleId String @default("1")
- instansi String?
- UserSession UserSession? // Nama instansi (Puskesmas, Sekolah, dll)
- isActive Boolean @default(true)
- lastLogin DateTime?
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
- deletedAt DateTime?
+ id String @id @default(cuid())
+ username String
+ nomor String @unique
+ roleId String @default("2")
+ isActive Boolean @default(false)
+ sessionInvalid Boolean @default(false)
+ lastLogin DateTime?
+ createdAt DateTime @default(now())
+ updatedAt DateTime @default(now()) @updatedAt
+ permissions Json?
+ sessions UserSession[] // ✅ Relasi one-to-many
+ role Role @relation(fields: [roleId], references: [id])
+ menuAccesses UserMenuAccess[]
+
+ @@map("users")
}
model Role {
id String @id @default(cuid())
name String @unique // ADMIN_DESA, ADMIN_KESEHATAN, ADMIN_SEKOLAH
description String?
- permissions Json // Menyimpan permission dalam format JSON
+ permissions Json?
isActive Boolean @default(true)
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
@@ -2167,35 +2236,30 @@ model KodeOtp {
otp Int
}
-// Tabel untuk menyimpan permission
-model Permission {
- id String @id @default(cuid())
- name String @unique
- description String?
- createdAt DateTime @default(now())
- updatedAt DateTime @updatedAt
-
- @@map("permissions")
-}
-
model UserSession {
- id String @id @default(cuid())
- token String
- expires DateTime?
- active Boolean @default(true)
- createdAt DateTime @default(now())
- updatedAt DateTime @default(now()) @updatedAt
- User User @relation(fields: [userId], references: [id])
- userId String @unique
+ id String @id @default(cuid())
+ token String @db.Text // ✅ JWT bisa panjang
+ expiresAt DateTime // ✅ Ubah jadi expiresAt (konsisten)
+ active Boolean @default(true)
+ createdAt DateTime @default(now())
+ updatedAt DateTime @default(now()) @updatedAt
+
+ user User @relation(fields: [userId], references: [id], onDelete: Cascade)
+ userId String // ✅ HAPUS @unique - user bisa punya multiple sessions
+
+ @@index([userId]) // ✅ Index untuk query cepat
+ @@index([token]) // ✅ Index untuk verify cepat
+ @@map("user_sessions")
}
-// ========================================= DATA PENDIDIKAN ========================================= //
-model DataPendidikan {
+model UserMenuAccess {
id String @id @default(cuid())
- name String
- jumlah String
+ userId String
+ menuId String // ID menu (misal: "Landing Page", "Kesehatan")
createdAt DateTime @default(now())
updatedAt DateTime @updatedAt
- deletedAt DateTime @default(now())
- isActive Boolean @default(true)
+
+ user User @relation(fields: [userId], references: [id])
+
+ @@unique([userId, menuId]) // Satu user tidak bisa punya akses menu yang sama dua kali
}
diff --git a/prisma/seed.ts b/prisma/seed.ts
index 52de4324..3c645b5b 100644
--- a/prisma/seed.ts
+++ b/prisma/seed.ts
@@ -1,1189 +1,349 @@
+/* eslint-disable @typescript-eslint/no-explicit-any */
/* eslint-disable @typescript-eslint/no-unused-vars */
import prisma from "@/lib/prisma";
-import profilePejabatDesa from "./data/landing-page/profile/profile.json";
-import programInovasi from "./data/landing-page/profile/programInovasi.json";
-import mediaSosial from "./data/landing-page/profile/mediaSosial.json";
-import desaAntiKorupsi from "./data/landing-page/desa-anti-korupsi/desaantiKorpusi.json";
-import kategoriDesaAntiKorupsi from "./data/landing-page/desa-anti-korupsi/kategoriDesaAntiKorupsi.json";
-import sdgsDesa from "./data/landing-page/sdgs-desa/sdgs-desa.json";
-import apbdes from "./data/landing-page/apbdes/apbdes.json";
-import kategoriPrestasiDesa from "./data/landing-page/prestasi-desa/kategori-prestasi.json";
-import prestasiDesa from "./data/landing-page/prestasi-desa/prestasi-desa.json";
-import penghargaan from "./data/landing-page/penghargaan/penghargaan.json";
-import profilePPID from "./data/ppid/profile-ppid/profilePPid.json";
-import pegawaiPPID from "./data/ppid/struktur-ppid/pegawai-PPID.json";
-import posisiOrganisasiPPID from "./data/ppid/struktur-ppid/posisi-organisasi-PPID.json";
-import visiMisiPPID from "./data/ppid/visi-misi-ppid/visimisiPPID.json";
-import dasarHukumPPID from "./data/ppid/dasar-hukum-ppid/dasarhukumPPID.json";
-import jenisKelamin from "./data/ppid/ikm/jenis-kelamin/jenis-kelamin.json";
-import daftarInformasiPublik from "./data/ppid/daftar-informasi-publik-desa-darmasaba/daftarInformasi.json";
-import pilihanRatingResponden from "./data/ppid/ikm/pilihan-rating-responden/rating-responden.json";
-import umurResponden from "./data/ppid/ikm/umur-responden/umur-responden.json";
-import categoryPengumuman from "./data/category-pengumuman.json";
-import pelayananPerizinanBerusaha from "./data/desa/layanan/pelayananPerizinanBerusaha.json";
-import pelayananSuratKeterangan from "./data/desa/layanan/pelayananSuratKeterangan.json";
-import pelayananTelunjukSaktiDesa from "./data/desa/layanan/pelayananTelunjukSaktiDesa.json";
-import pelayananPendudukNonPermanen from "./data/desa/layanan/pelayanaPendudukNonPermanen.json";
-import lambangDesa from "./data/desa/profile/lambang_desa.json";
-import maskotDesa from "./data/desa/profile/maskot_desa.json";
-import profilPerbekel from "./data/desa/profile/profil_perbekel.json";
-import sejarahDesa from "./data/desa/profile/sejarah_desa.json";
-import visiMisiDesa from "./data/desa/profile/visi_misi_desa.json";
-import detailDataPengangguran from "./data/ekonomi/jumlah-pengangguran/detail-data-pengangguran.json";
-import kategoriProduk from "./data/ekonomi/pasar-desa/kategori-produk.json";
-import pegawai from "./data/ekonomi/struktur-organisasi/pegawai-bumdes.json";
-import posisiOrganisasi from "./data/ekonomi/struktur-organisasi/posisi-organisasi-bumdes.json";
-import kategoriBerita from "./data/desa/berita/kategori-berita.json";
-import contohEdukasiLingkungan from "./data/lingkungan/edukasi-lingkungan/contoh-kegiatan-di-desa-darmasaba.json";
-import materiEdukasiLingkungan from "./data/lingkungan/edukasi-lingkungan/materi-edukasi-yang-diberikan.json";
-import tujuanEdukasiLingkungan from "./data/lingkungan/edukasi-lingkungan/tujuan-edukasi-lingkungan.json";
-import bentukKonservasiBerdasarkanAdat from "./data/lingkungan/konservasi-adat-bali/bentuk-konservasi.json";
-import kategoriKegiatanData from "./data/lingkungan/gotong-royong/kategori-gotong-royong.json";
-import filosofiTriHita from "./data/lingkungan/konservasi-adat-bali/filosofi-tri-hita.json";
-import nilaiKonservasiAdat from "./data/lingkungan/konservasi-adat-bali/nilai-konservasi-adat.json";
-import caraMemperolehInformasi from "./data/list-caraMemperolehInformasi.json";
-import caraMemperolehSalinanInformasi from "./data/list-caraMemperolehSalinanInformasi.json";
-import jenisInformasiDiminta from "./data/list-jenisInfromasi.json";
-import potensi from "./data/list-potensi.json";
-import fasilitasBimbinganBelajarDesa from "./data/pendidikan/bimbingan-belajar-desa/fasilitas-yang-disediakan.json";
-import lokasiJadwalBimbinganBelajarDesa from "./data/pendidikan/bimbingan-belajar-desa/lokasi-dan-jadwal.json";
-import tujuanBimbinganBelajarDesa from "./data/pendidikan/bimbingan-belajar-desa/tujuan-bimbingan-belajar-desa.json";
-import jenisProgramYangDiselenggarakan from "./data/pendidikan/pendidikan-non-formal/jenis-program-yang-diselenggarakan.json";
-import tempatKegiatan from "./data/pendidikan/pendidikan-non-formal/tempat-kegiatan.json";
-import tujuanProgram2 from "./data/pendidikan/pendidikan-non-formal/tujuan-program2.json";
-import programUnggulan from "./data/pendidikan/program-pendidikan-anak/program-unggulan.json";
-import tujuanProgram from "./data/pendidikan/program-pendidikan-anak/tujuan-program.json";
+import { seedVideo } from "./_seeder_list/desa/gallery/video/seed_video";
+import { seedLayanan } from "./_seeder_list/desa/layanan/seed_layanan";
+import { seedPenghargaan } from "./_seeder_list/desa/penghargaan/penghargaan";
+import { seedPengumuman } from "./_seeder_list/desa/pengumuman/seed_pengumuman";
+import { seedPotensi } from "./_seeder_list/desa/potensi/seed_potensi";
+import { seedProfileDesa } from "./_seeder_list/desa/profile-desa/seed_profile_desa";
+import { seedProfilePerbekel } from "./_seeder_list/desa/profile-desa/seed_profile_perbekel";
+import { seedDesaAntiKorupsi } from "./_seeder_list/landing-page/desa-anti-korupsi/seed_desa_anti_korupsi";
+import { seedPrestasiDesa } from "./_seeder_list/landing-page/prestasi-desa/seed_prestasi_desa";
+import { seedMediaSosial } from "./_seeder_list/landing-page/profil_landing_page/seed_media_sosial";
+import { seedProfileLP } from "./_seeder_list/landing-page/profil_landing_page/seed_profile_lp";
+import { seedProgramInovasi } from "./_seeder_list/landing-page/profil_landing_page/seed_program_inovasi";
+import { seedSDGSDesa } from "./_seeder_list/landing-page/sdgs/seed_sdgs";
+import { seedDaftarInformasiPublikPpid } from "./_seeder_list/ppid/daftar-informasi-publik-ppid/seed_daftar_informasi_publik_ppid";
+import { seedDasarHukumPpid } from "./_seeder_list/ppid/dasar-hukum-ppid/seed_dasar_hukum_ppid";
+import { seedIkmPpid } from "./_seeder_list/ppid/ikm/seed_ikm";
+import { seedPegawaiPpid } from "./_seeder_list/ppid/struktur-ppid/seed_struktur_ppid";
+import { seedVisiMisiPpid } from "./_seeder_list/ppid/visi-misi-ppid/seed_visi_misi_ppid";
+import dataPendidikan from "./data/pendidikan/data-pendidikan/data-pendidikan.json";
import roles from "./data/user/roles.json";
import users from "./data/user/users.json";
-import fileStorage from "./data/file-storage.json";
-import jenjangPendidikan from "./data/pendidikan/info-sekolah/jenjang-pendidikan.json";
-import seedAssets from "./seed_assets";
import { safeSeedUnique } from "./safeseedUnique";
+import seedAssets from "./seed_assets";
+import { seedBerita } from "./_seeder_list/desa/berita/seed_berita";
+import { seedFoto } from "./_seeder_list/desa/gallery/foto/seed_foto";
+import { seedPosyandu } from "./_seeder_list/kesehatan/posyandu/seed_posyandu";
+import { seedPuskesmas } from "./_seeder_list/kesehatan/puskesmas/seed_puskesmas";
+import { seedProgramKesehatan } from "./_seeder_list/kesehatan/program-kesehatan/seed_program_kesehatan";
+import { seedPenangananDarurat } from "./_seeder_list/kesehatan/penanganan-darurat/seed_penanganan_darurat";
+import { seedKontakDarurat } from "./_seeder_list/kesehatan/kontak-darurat/seed_kontak_darurat";
+import { seedInfoWabahPenyakit } from "./_seeder_list/kesehatan/info-wabah-penyakit/seed_info_wabah_penyakit";
+import { seedKeamananLingkungan } from "./_seeder_list/keamanan/seed_keamanan_lingkungan";
+import { seedPolsekTerdekat } from "./_seeder_list/keamanan/seed_polsek_terdekat";
+import { seedKontakDaruratKeamanan } from "./_seeder_list/keamanan/seed_kontak_darurat";
+import { seedPencegahanKriminalitas } from "./_seeder_list/keamanan/seed_pencegahan_kriminalitas";
+import { seedLaporanPublik } from "./_seeder_list/keamanan/seed_laporan_publik";
+import { seedPasarDesa } from "./_seeder_list/ekonomi/seed_pasar_desa";
+import { seedLowonganKerjaLokal } from "./_seeder_list/ekonomi/seed_lowongan_kerja_lokal";
+import { seedDemografiPekerjaan } from "./_seeder_list/ekonomi/seed_demografi_pekerjaan";
+import { seedSektorUnggulanDesa } from "./_seeder_list/ekonomi/seed_sektor_unggulan_desa";
+import { seedProgramKemiskinan } from "./_seeder_list/ekonomi/seed_program_kemiskinan";
+import { seedJumlahPendudukMiskin } from "./_seeder_list/ekonomi/seed_jumlah_penduduk_miskin";
+import { seedPendudukUsiaKerjaYangMenganggur } from "./_seeder_list/ekonomi/seed_penduduk_usia_kerja_yang_menganggur";
+import { seedJumlahPengangguran } from "./_seeder_list/ekonomi/seed_jumlah_pengangguran";
+import { seedPendapatanAsli } from "./_seeder_list/ekonomi/seed_pendapatan_asli";
+import { seedStrukturBumdes } from "./_seeder_list/ekonomi/seed_struktur_bumdes";
+import { seedAjukan } from "./_seeder_list/inovasi/seed_ajukan";
+import { seedDesaDigital } from "./_seeder_list/inovasi/seed_desa_digital";
+import { seedLayananOnlineDesa } from "./_seeder_list/inovasi/seed_layanan_online_desa";
+import { seedProgramKreatifDesa } from "./_seeder_list/inovasi/seed_program_kreatif_desa";
+import { seedKolaborasiInovasi } from "./_seeder_list/inovasi/seed_kolaborasi_inovasi";
+import { seedInfoTeknologi } from "./_seeder_list/inovasi/seed_info_teknologi";
+import { seedPengelolaanSampah } from "./_seeder_list/lingkungan/seed_pengelolaan_sampah";
+import { seedProgramPenghijauan } from "./_seeder_list/lingkungan/seed_program_penghijauan";
+import { seedDataLingkunganDesa } from "./_seeder_list/lingkungan/seed_data_lingkungan_desa";
+import { seedDataGotongRoyong } from "./_seeder_list/lingkungan/seed_data_gotong_royong";
+import { seedEdukasiLingkungan } from "./_seeder_list/lingkungan/seed_edukasi_lingkungan";
+import { seedKonservasiAdatBali } from "./_seeder_list/lingkungan/seed_konservasi_adat_bali";
+import { seedInfoSekolah } from "./_seeder_list/pendidikan/seed_info_sekolah";
+import { seedInfoProgramPendidikan } from "./_seeder_list/pendidikan/seed_info_program_pendidikan";
+import { seedBimbinganBelajar } from "./_seeder_list/pendidikan/seed_bimbingan_belajar";
+import { seedDataPendidikan } from "./_seeder_list/pendidikan/seed_data_pendidikan";
+import { seedPendidikanNonFormal } from "./_seeder_list/pendidikan/seed_pendidikan_non_formal";
+import { seedDataPerpustakaan } from "./_seeder_list/pendidikan/seed_data_perpustakaan";
+import { seedProfilPpd } from "./_seeder_list/ppid/profil-ppid/seed_profil_ppd";
(async () => {
- // =========== USER & ROLE ===========
- // In your seed.ts
- // =========== ROLES ===========
- console.log("🔄 Seeding roles...");
- for (const r of roles) {
- await safeSeedUnique("role", { id: r.id }, {
- name: r.name,
- description: r.description,
- permissions: r.permissions,
- isActive: r.isActive,
- });
- }
-
- console.log("✅ Roles seeded");
-
- // =========== USERS ===========
- console.log("🔄 Seeding users...");
- for (const u of users) {
- // First verify the role exists
- const roleExists = await prisma.role.findUnique({
- where: { id: u.roleId },
- });
-
- if (!roleExists) {
- console.error(`❌ Role with id ${u.roleId} not found for user ${u.nama}`);
- continue;
- }
-
- await safeSeedUnique("user", { id: u.id }, {
- username: u.nama,
- nomor: u.nomor,
- roleId: u.roleId,
- isActive: u.isActive,
- });
- }
- console.log("✅ Users seeded");
-
- // =========== FILE STORAGE ===========
- console.log("🔄 Seeding file storage...");
- for (const f of fileStorage) {
- await prisma.fileStorage.upsert({
- where: { id: f.id },
- update: {
- name: f.name,
- realName: f.realName,
- path: f.path,
- mimeType: f.mimeType,
- link: f.link,
- category: f.category,
- },
- create: {
- id: f.id,
- name: f.name,
- realName: f.realName,
- path: f.path,
- mimeType: f.mimeType,
- link: f.link,
- category: f.category,
- },
- });
- }
- console.log("✅ File storage seeded");
- // =========== LANDING PAGE ===========
- // =========== SUBMENU PROFILE ===========
- // =========== PROFILE PEJABAT DESA ===========
- for (const p of profilePejabatDesa) {
- await prisma.pejabatDesa.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- position: p.position,
- imageId: p.imageId,
- },
- create: {
- id: p.id,
- name: p.name,
- position: p.position,
- imageId: p.imageId,
- },
- });
- }
- console.log(
- "✅ profilePejabatDesa seeded without imageId (editable later via UI)"
- );
-
- // =========== PROGRAM INOVASI ===========
- for (const p of programInovasi) {
- let imageId: string | null = null;
-
- if (p.imageId) {
- const imageExists = await prisma.fileStorage.findUnique({
- where: { id: p.imageId },
- });
-
- if (imageExists) {
- imageId = p.imageId;
- } else {
- console.warn(
- `⚠️ imageId ${p.imageId} tidak ditemukan untuk ProgramInovasi ${p.name}`
- );
- }
- }
- await prisma.programInovasi.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- description: p.description,
- link: p.link,
- imageId: p.imageId,
- },
- create: {
- id: p.id,
- name: p.name,
- description: p.description,
- link: p.link,
- imageId: p.imageId,
- },
- });
- }
- console.log("program inovasi success ...");
-
- // =========== MEDIA SOSIAL ===========
- for (const p of mediaSosial) {
- await prisma.mediaSosial.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- iconUrl: p.iconUrl,
- imageId: p.imageId,
- },
- create: {
- id: p.id,
- name: p.name,
- iconUrl: p.iconUrl,
- imageId: p.imageId,
- },
- });
- }
- console.log("media sosial success ...");
-
- // =========== SUBMENU DESA ANTI KORUPSI ===========
- // =========== KATEGORI DESA ANTI KORUPSI ===========
- for (const k of kategoriDesaAntiKorupsi) {
- await prisma.kategoriDesaAntiKorupsi.upsert({
- where: { id: k.id },
- update: {
- name: k.name,
- },
- create: {
- id: k.id,
- name: k.name,
- },
- });
- }
- console.log("kategori desa anti korupsi success ...");
-
- // =========== DESA ANTI KORUPSI ===========
- for (const p of desaAntiKorupsi) {
- await prisma.desaAntiKorupsi.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- deskripsi: p.deskripsi,
- kategoriId: p.kategoriId,
- },
- create: {
- id: p.id,
- name: p.name,
- deskripsi: p.deskripsi,
- kategoriId: p.kategoriId,
- },
- });
- }
- console.log("desa anti korupsi success ...");
-
- // =========== KATEGORI DESA ANTI KORUPSI ===========
- for (const p of kategoriDesaAntiKorupsi) {
- await prisma.kategoriDesaAntiKorupsi.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- },
- create: {
- id: p.id,
- name: p.name,
- },
- });
- }
- console.log("desa anti korupsi success ...");
-
- // =========== KATEGORI PRESTASI DESA===========
- for (const c of kategoriPrestasiDesa) {
- await prisma.kategoriPrestasiDesa.upsert({
- where: { id: c.id },
- update: {
- name: c.name,
- },
- create: {
- id: c.id,
- name: c.name,
- },
- });
- }
- console.log("kategori prestasi desa success ...");
-
- // =========== PRESTASI DESA===========
- for (const p of prestasiDesa) {
- await prisma.prestasiDesa.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- deskripsi: p.deskripsi,
- kategoriId: p.kategoriId,
- },
- create: {
- id: p.id,
- name: p.name,
- deskripsi: p.deskripsi,
- kategoriId: p.kategoriId,
- },
- });
- }
- console.log("prestasi desa success ...");
-
- // =========== PENGHARGAAN ===========
- for (const p of penghargaan) {
- await prisma.penghargaan.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- juara: p.juara,
- deskripsi: p.deskripsi,
- },
- create: {
- id: p.id,
- name: p.name,
- juara: p.juara,
- deskripsi: p.deskripsi,
- },
- });
- }
- console.log("penghargaan success ...");
-
- // =========== LAYANAN DESA ===========
- for (const p of pelayananSuratKeterangan) {
- await prisma.pelayananSuratKeterangan.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- deskripsi: p.deskripsi,
- },
- create: {
- id: p.id,
- name: p.name,
- deskripsi: p.deskripsi,
- },
- });
- }
- console.log("pelayanan surat keterangan success ...");
-
- for (const p of pelayananTelunjukSaktiDesa) {
- await prisma.pelayananTelunjukSaktiDesa.upsert({
- where: { id: p.id },
- update: {
- name: p.name,
- deskripsi: p.deskripsi,
- link: p.link,
- },
- create: {
- id: p.id,
- name: p.name,
- deskripsi: p.deskripsi,
- link: p.link,
- },
- });
- }
- console.log("pelayanan surat keterangan success ...");
-
- // =========== SDGSDesa ===========
- for (const l of sdgsDesa) {
- await prisma.sdgsDesa.upsert({
- where: { id: l.id },
- update: {
- name: l.name,
- jumlah: l.jumlah,
- },
- create: {
- id: l.id,
- name: l.name,
- jumlah: l.jumlah,
- },
- });
- }
-
- console.log("sdgs desa success ...");
-
- // =========== APBDes ===========
- for (const l of apbdes) {
- await prisma.aPBDes.upsert({
- where: {
- id: l.id,
- },
- update: {
- name: l.name,
- jumlah: l.jumlah,
- },
- create: {
- name: l.name,
- jumlah: l.jumlah,
- },
- });
- }
-
- console.log("sdgs desa success ...");
-
- // =========== MENU DESA ===========
- // =========== SUBMENU PROFILE ===========
- // =========== SEJARAH DESA ===========
- for (const l of sejarahDesa) {
- await prisma.sejarahDesa.upsert({
- where: {
- id: l.id,
- },
- update: {
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- create: {
- id: l.id,
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- });
- }
-
- console.log("sejarah desa success ...");
-
- // =========== MASKOT DESA ===========
- for (const l of maskotDesa) {
- await prisma.maskotDesa.upsert({
- where: {
- id: l.id,
- },
- update: {
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- create: {
- id: l.id,
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- });
- }
-
- console.log("maskot desa success ...");
-
- // =========== LAMBANG DESA ===========
- for (const l of lambangDesa) {
- await prisma.lambangDesa.upsert({
- where: {
- id: l.id,
- },
- update: {
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- create: {
- id: l.id,
- judul: l.judul,
- deskripsi: l.deskripsi,
- },
- });
- }
-
- console.log("lambang desa success ...");
-
- // =========== PROFIL PERBEKEL ===========
- for (const c of profilPerbekel) {
- await prisma.profilPerbekel.upsert({
- where: { id: c.id },
- update: {
- biodata: c.biodata,
- pengalaman: c.pengalaman,
- pengalamanOrganisasi: c.pengalamanOrganisasi,
- programUnggulan: c.programUnggulan,
- // imageId tidak di-update
- },
- create: {
- id: c.id,
- biodata: c.biodata,
- pengalaman: c.pengalaman,
- pengalamanOrganisasi: c.pengalamanOrganisasi,
- programUnggulan: c.programUnggulan,
- // imageId tidak di-create
- },
- });
- }
- console.log(
- "✅ profilePerbekel seeded without imageId (editable later via UI)"
- );
-
- // =========== VISI MISI DESA ===========
- for (const l of visiMisiDesa) {
- await prisma.visiMisiDesa.upsert({
- where: {
- id: l.id,
- },
- update: {
- visi: l.visi,
- misi: l.misi,
- },
- create: {
- id: l.id,
- visi: l.visi,
- misi: l.misi,
- },
- });
- }
-
- console.log("visi misi desa success ...");
-
- // =========== MENU PPID ===========
- // =========== SUBMENU PROFILE PPID ===========
- for (const c of profilePPID) {
- await prisma.profilePPID.upsert({
- where: { id: c.id },
- update: {
- name: c.name,
- biodata: c.biodata,
- riwayat: c.riwayat,
- pengalaman: c.pengalaman,
- unggulan: c.unggulan,
- // imageId tidak di-update
- },
- create: {
- id: c.id,
- name: c.name,
- biodata: c.biodata,
- riwayat: c.riwayat,
- pengalaman: c.pengalaman,
- unggulan: c.unggulan,
- // imageId tidak di-create
- },
- });
- }
- console.log("✅ profilePPID seeded without imageId (editable later via UI)");
-
- // =========== SUBMENU STRUKTUR PPID ===========
- // =========== POSISI ORGANISASI PPID ===========
-
- const flattenedPosisi = posisiOrganisasiPPID.flat();
-
- // ✅ Urutkan berdasarkan hierarki
- const sortedPosisi = flattenedPosisi.sort((a, b) => a.hierarki - b.hierarki);
-
- for (const p of sortedPosisi) {
- console.log(`Seeding: ${p.nama} (id: ${p.id}, parent: ${p.parentId})`);
-
- if (p.parentId) {
- const parentExists = flattenedPosisi.some((pos) => pos.id === p.parentId);
- if (!parentExists) {
- console.warn(
- `⚠️ Parent tidak ditemukan: ${p.parentId} untuk ${p.nama}`
- );
- continue;
- }
- }
-
- await prisma.posisiOrganisasiPPID.upsert({
- where: { id: p.id },
- update: p,
- create: p,
- });
- }
- console.log("posisi organisasi berhasil");
-
- // =========== PEGAWAI PPID ===========
- const flattenedPegawai = pegawaiPPID.flat();
- for (const p of flattenedPegawai) {
- await prisma.pegawaiPPID.upsert({
- where: { id: p.id },
- update: p,
- create: p,
- });
- }
- console.log("pegawai berhasil");
-
- // =========== SUBMENU VISI MISI PPID ===========
-
- for (const v of visiMisiPPID) {
- await prisma.visiMisiPPID.upsert({
- where: {
- id: v.id,
- },
- update: {
- misi: v.misi,
- visi: v.visi,
- },
- create: {
- id: v.id,
- misi: v.misi,
- visi: v.visi,
- },
- });
- }
- console.log("visi misi PPID success ...");
-
- // =========== SUBMENU DASAR HUKUM PPID ===========
- for (const v of dasarHukumPPID) {
- await prisma.dasarHukumPPID.upsert({
- where: {
- id: v.id,
- },
- update: {
- judul: v.judul,
- content: v.content,
- },
- create: {
- id: v.id,
- judul: v.judul,
- content: v.content,
- },
- });
- }
- console.log("dasar hukum PPID success ...");
-
- // =========== SUBMENU DAFTAR INFORMASI PUBLIK PPID ===========
- for (const v of daftarInformasiPublik) {
- // Convert string date to Date object
- const tanggal = new Date(v.tanggal);
-
- await prisma.daftarInformasiPublik.upsert({
- where: {
- id: v.id,
- },
- update: {
- jenisInformasi: v.jenisInformasi,
- deskripsi: v.deskripsi,
- tanggal: tanggal,
- },
- create: {
- id: v.id,
- jenisInformasi: v.jenisInformasi,
- deskripsi: v.deskripsi,
- tanggal: tanggal,
- },
- });
- }
- console.log("daftar informasi publik PPID success ...");
-
- for (const l of pelayananPerizinanBerusaha) {
- await prisma.pelayananPerizinanBerusaha.upsert({
- where: {
- id: l.id,
- },
- update: {
- name: l.name,
- deskripsi: l.deskripsi,
- link: l.link,
- },
- create: {
- id: l.id,
- name: l.name,
- deskripsi: l.deskripsi,
- link: l.link,
- },
- });
- }
-
- console.log("pelayanan perizinan berusaha success ...");
-
- for (const l of pelayananPendudukNonPermanen) {
- await prisma.pelayananPendudukNonPermanen.upsert({
- where: {
- id: l.id,
- },
- update: {
- name: l.name,
- deskripsi: l.deskripsi,
- },
- create: {
- id: l.id,
- name: l.name,
- deskripsi: l.deskripsi,
- },
- });
- }
- console.log("pelayanan penduduk non permanen success ...");
-
- for (const p of potensi) {
- await prisma.potensi.upsert({
- where: {
- name: p.name,
- },
- update: {
- name: p.name,
- },
- create: {
- name: p.name,
- },
- });
- }
-
- console.log("potensi success ...");
-
- for (const k of kategoriBerita) {
- await prisma.kategoriBerita.upsert({
- where: {
- name: k.name,
- },
- update: {
- name: k.name,
- },
- create: {
- name: k.name,
- },
- });
- }
-
- console.log("kategori berita success ...");
-
- for (const c of categoryPengumuman) {
- await prisma.categoryPengumuman.upsert({
- where: {
- name: c.name,
- },
- update: {
- name: c.name,
- },
- create: {
- name: c.name,
- },
- });
- }
-
- console.log("category pengumuman success ...");
-
- for (const j of jenisInformasiDiminta) {
- await prisma.jenisInformasiDiminta.upsert({
- where: {
- name: j.name,
- },
- update: {
- name: j.name,
- },
- create: {
- name: j.name,
- },
- });
- }
- console.log("jenis informasi diminta success ...");
-
- for (const c of caraMemperolehInformasi) {
- await prisma.caraMemperolehInformasi.upsert({
- where: {
- name: c.name,
- },
- update: {
- name: c.name,
- },
- create: {
- name: c.name,
- },
- });
- }
- console.log("cara memperoleh informasi success ...");
-
- for (const c of caraMemperolehSalinanInformasi) {
- await prisma.caraMemperolehSalinanInformasi.upsert({
- where: {
- name: c.name,
- },
- update: {
- name: c.name,
- },
- create: {
- name: c.name,
- },
- });
- }
- console.log("cara memperoleh salinan informasi success ...");
-
- for (const j of jenisKelamin) {
- await prisma.jenisKelaminResponden.upsert({
- where: {
- id: j.id,
- },
- update: {
- name: j.name,
- },
- create: {
- id: j.id,
- name: j.name,
- },
- });
- }
- console.log("jenis kelamin responden success ...");
-
- for (const r of pilihanRatingResponden) {
- await prisma.pilihanRatingResponden.upsert({
- where: {
- id: r.id,
- },
- update: {
- name: r.name,
- },
- create: {
- id: r.id,
- name: r.name,
- },
- });
- }
- console.log("pilihan rating responden success ...");
-
- for (const u of umurResponden) {
- await prisma.umurResponden.upsert({
- where: {
- id: u.id,
- },
- update: {
- name: u.name,
- },
- create: {
- id: u.id,
- name: u.name,
- },
- });
- }
- console.log("umur responden success ...");
-
- for (const k of kategoriProduk) {
- await prisma.kategoriProduk.upsert({
- where: {
- id: k.id,
- },
- update: {
- nama: k.nama,
- },
- create: {
- id: k.id,
- nama: k.nama,
- },
- });
- }
- console.log("kategori produk success ...");
-
- const flattenedPosisiBumdes = posisiOrganisasi.flat();
-
- // ✅ Urutkan berdasarkan hierarki
- const sortedPosisiBumdes = flattenedPosisiBumdes.sort((a, b) => a.hierarki - b.hierarki);
-
- for (const p of sortedPosisiBumdes) {
- console.log(`Seeding: ${p.nama} (id: ${p.id}, parent: ${p.parentId})`);
-
- if (p.parentId) {
- const parentExists = flattenedPosisi.some((pos) => pos.id === p.parentId);
- if (!parentExists) {
- console.warn(
- `⚠️ Parent tidak ditemukan: ${p.parentId} untuk ${p.nama}`
- );
- continue;
- }
- }
-
- await prisma.posisiOrganisasiBumDes.upsert({
- where: { id: p.id },
- update: p,
- create: p,
- });
- }
- console.log("posisi organisasi berhasil");
-
- for (const p of pegawai) {
- await prisma.pegawaiBumDes.upsert({
- where: {
- id: p.id,
- },
- update: {
- namaLengkap: p.namaLengkap,
- gelarAkademik: p.gelarAkademik,
- tanggalMasuk: new Date(p.tanggalMasuk),
- email: p.email,
- telepon: p.telepon,
- alamat: p.alamat,
- posisiId: p.posisiId,
- isActive: p.isActive,
- },
- create: {
- id: p.id,
- namaLengkap: p.namaLengkap,
- gelarAkademik: p.gelarAkademik,
- tanggalMasuk: new Date(p.tanggalMasuk),
- email: p.email,
- telepon: p.telepon,
- alamat: p.alamat,
- posisiId: p.posisiId,
- isActive: p.isActive,
- },
- });
- }
- console.log("pegawai success ...");
-
- for (const d of detailDataPengangguran) {
- await prisma.detailDataPengangguran.upsert({
- where: {
- month_year: { month: d.month, year: d.year },
- },
- update: {
- totalUnemployment: d.totalUnemployment,
- educatedUnemployment: d.educatedUnemployment,
- uneducatedUnemployment: d.uneducatedUnemployment,
- percentageChange: d.percentageChange,
- },
- create: {
- month: d.month,
- year: d.year,
- totalUnemployment: d.totalUnemployment,
- educatedUnemployment: d.educatedUnemployment,
- uneducatedUnemployment: d.uneducatedUnemployment,
- percentageChange: d.percentageChange,
- },
- });
- }
- console.log("📊 detailDataPengangguran success ...");
-
- // =========== KATEGORI GOTONG ROYONG ===========
- // Add IDs to the kategoriKegiatan data
- const kategoriKegiatan = kategoriKegiatanData.map((k, index) => ({
- ...k,
- id: `kategori-${index + 1}`
- }));
-
- for (const k of kategoriKegiatan) {
- await prisma.kategoriKegiatan.upsert({
- where: {
- id: k.id,
- },
- update: {
- nama: k.nama,
- },
- create: {
- id: k.id,
- nama: k.nama,
- },
- });
- }
-
- console.log("kategori kegiatan success ...");
-
- for (const e of tujuanEdukasiLingkungan) {
- await prisma.tujuanEdukasiLingkungan.upsert({
- where: {
- id: e.id,
- },
- update: {
- judul: e.judul,
- deskripsi: e.deskripsi,
- },
- create: {
- id: e.id,
- judul: e.judul,
- deskripsi: e.deskripsi,
- },
- });
- }
-
- console.log("tujuan edukasi lingkungan success ...");
-
- for (const m of materiEdukasiLingkungan) {
- await prisma.materiEdukasiLingkungan.upsert({
- where: {
- id: m.id,
- },
- update: {
- judul: m.judul,
- deskripsi: m.deskripsi,
- },
- create: {
- id: m.id,
- judul: m.judul,
- deskripsi: m.deskripsi,
- },
- });
- }
-
- console.log("materi edukasi lingkungan success ...");
-
- for (const c of contohEdukasiLingkungan) {
- await prisma.contohEdukasiLingkungan.upsert({
- where: {
- id: c.id,
- },
- update: {
- judul: c.judul,
- deskripsi: c.deskripsi,
- },
- create: {
- id: c.id,
- judul: c.judul,
- deskripsi: c.deskripsi,
- },
- });
- }
-
- console.log("contoh edukasi lingkungan success ...");
-
- for (const f of filosofiTriHita) {
- await prisma.filosofiTriHita.upsert({
- where: {
- id: f.id,
- },
- update: {
- judul: f.judul,
- deskripsi: f.deskripsi,
- },
- create: {
- id: f.id,
- judul: f.judul,
- deskripsi: f.deskripsi,
- },
- });
- }
-
- console.log("filosofi tri hita success ...");
-
- for (const b of bentukKonservasiBerdasarkanAdat) {
- await prisma.bentukKonservasiBerdasarkanAdat.upsert({
- where: {
- id: b.id,
- },
- update: {
- judul: b.judul,
- deskripsi: b.deskripsi,
- },
- create: {
- id: b.id,
- judul: b.judul,
- deskripsi: b.deskripsi,
- },
- });
- }
-
- console.log("bentuk konservasi berdasarkan adat success ...");
-
- for (const n of nilaiKonservasiAdat) {
- await prisma.nilaiKonservasiAdat.upsert({
- where: {
- id: n.id,
- },
- update: {
- judul: n.judul,
- deskripsi: n.deskripsi,
- },
- create: {
- id: n.id,
- judul: n.judul,
- deskripsi: n.deskripsi,
- },
- });
- }
-
- console.log("nilai konservasi adat success ...");
-
- for (const t of tujuanProgram) {
- await prisma.tujuanProgram.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log("✅ tujuan program seeded (editable later via UI)");
-
- for (const t of programUnggulan) {
- await prisma.programUnggulan.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log("✅ program unggulan seeded (editable later via UI)");
-
- for (const t of tujuanBimbinganBelajarDesa) {
- await prisma.tujuanBimbinganBelajarDesa.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ tujuan bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const t of lokasiJadwalBimbinganBelajarDesa) {
- await prisma.lokasiJadwalBimbinganBelajarDesa.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ lokasi jadwal bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const t of fasilitasBimbinganBelajarDesa) {
- await prisma.fasilitasBimbinganBelajarDesa.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const t of tujuanProgram2) {
- await prisma.tujuanPendidikanNonFormal.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const t of tempatKegiatan) {
- await prisma.tempatKegiatan.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const t of jenisProgramYangDiselenggarakan) {
- await prisma.jenisProgramYangDiselenggarakan.upsert({
- where: { id: t.id },
- update: {
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- create: {
- id: t.id,
- judul: t.judul,
- deskripsi: t.deskripsi,
- },
- });
- }
- console.log(
- "✅ fasilitas bimbingan belajar desa seeded (editable later via UI)"
- );
-
- for (const j of jenjangPendidikan) {
- await prisma.jenjangPendidikan.upsert({
- where: {
- id: j.id || undefined,
- },
- update: {
- nama: j.nama,
- },
- create: {
- nama: j.nama,
- },
- });
- }
-
- console.log("✅ Jenjang Pendidikan seeded successfully");
-
- // seed assets
+ // Always run seedAssets to handle new images without duplication
+ console.log("📂 Checking for new assets to seed...");
await seedAssets();
+ // // =========== FILE STORAGE ===========
+
+ console.log("🔄 Seeding roles...");
+
+ for (const r of roles) {
+ try {
+ // ✅ Destructure to remove permissions if exists
+ const { permissions, ...roleData } = r as any;
+
+ await safeSeedUnique(
+ "role",
+ { name: roleData.name },
+ {
+ id: roleData.id,
+ name: roleData.name,
+ description: roleData.description,
+ permissions: roleData.permissions || {}, // ✅ Include permissions
+ isActive: roleData.isActive,
+ },
+ );
+ console.log(`✅ Seeded role -> ${roleData.name}`);
+ } catch (error: any) {
+ if (error.code === "P2002") {
+ console.warn(`⚠️ Role already exists (skipping): ${r.name}`);
+ } else {
+ console.error(`❌ Failed to seed role ${r.name}:`, error.message);
+ }
+ }
+ }
+ console.log("✅ Roles seeding completed");
+
+ // =========== USER ===========
+ console.log("🔄 Seeding users...");
+ for (const u of users) {
+ try {
+ // Verify role exists first
+ const roleExists = await prisma.role.findUnique({
+ where: { id: u.roleId.toString() },
+ select: { id: true }, // Only select id to minimize query
+ });
+
+ if (!roleExists) {
+ console.error(
+ `❌ Role with id ${u.roleId} not found for user ${u.username}`,
+ );
+ continue;
+ }
+
+ await safeSeedUnique(
+ "user",
+ { nomor: u.nomor },
+ {
+ id: u.id,
+ username: u.username,
+ nomor: u.nomor,
+ roleId: u.roleId.toString(),
+ isActive: u.isActive,
+ sessionInvalid: false,
+ },
+ );
+ console.log(`✅ Seeded user -> ${u.username}`);
+ } catch (error: any) {
+ if (error.code === "P2003") {
+ console.error(
+ `❌ Foreign key constraint failed for user ${u.username}: Role ${u.roleId} does not exist`,
+ );
+ } else {
+ console.error(`❌ Failed to seed user ${u.username}:`, error.message);
+ }
+ }
+ }
+ console.log("✅ Users seeding completed");
+ // =========== LANDING PAGE ===========
+ // =========== SUBMENU PROFILE ===========
+ await seedProgramInovasi();
+ await seedProfileLP();
+ await seedMediaSosial();
+
+ // // =========== SUBMENU DESA ANTI KORUPSI ===========
+ await seedDesaAntiKorupsi();
+
+ // // =========== SDGSDesa ===========
+ await seedSDGSDesa();
+
+ // // =========== APBDes ===========
+ // for (const l of apbdes) {
+ // await prisma.aPBDes.upsert({
+ // where: {
+ // id: l.id,
+ // },
+ // update: {
+ // name: l.name,
+ // jumlah: l.jumlah,
+ // },
+ // create: {
+ // name: l.name,
+ // jumlah: l.jumlah,
+ // },
+ // });
+ // }
+
+ // console.log("apbdes success ...");
+
+ // // =========== PRESTASI DESA===========
+ await seedPrestasiDesa();
+
+ // // =========== MENU PPID ===========
+
+ // // =========== SUBMENU PROFIL PPID ===========
+ await seedProfilPpd();
+
+ // // =========== SUBMENU STRUKTUR PPID ===========
+ await seedPegawaiPpid();
+
+ // // =========== SUBMENU VISI MISI PPID ===========
+ await seedVisiMisiPpid();
+
+ // // =========== SUBMENU DASAR HUKUM PPID ===========
+ await seedDasarHukumPpid();
+
+ // // =========== SUBMENU DAFTAR INFORMASI PUBLIK PPID ===========
+ await seedDaftarInformasiPublikPpid();
+
+ // // =========== SUBMENU INDEKS KEPUASAN MASYARAKAT ===========
+ await seedIkmPpid();
+
+ // // =========== MENU DESA ===========
+ // // =========== SUBMENU PROFILE ===========
+ await seedProfileDesa();
+ await seedProfilePerbekel();
+
+ // // =========== SUBMENU POTENSI DESA ===========
+ await seedPotensi();
+
+ // // =========== SUBMENU BERITA ===========
+ await seedBerita();
+
+ // // ================== SUBMENU PENGUMUMAN ========================
+ await seedPengumuman();
+
+ // // ================== SUBMENU GALLERY ========================
+ await seedVideo();
+ await seedFoto();
+
+ // // =========== SUBMENU LAYANAN ===========
+ await seedLayanan();
+
+ // // =========== PENGHARGAAN ===========
+ await seedPenghargaan();
+
+ // // ====================== MENU KESEHATAN ========================
+ // // ==================== SUBMENU POSYANDU =========================
+ await seedPosyandu();
+
+ // // ==================== SUBMENU PUSKESMAS =========================
+ await seedPuskesmas();
+
+ // // ==================== SUBMENU PROGRAM KESEHATAN =========================
+ await seedProgramKesehatan();
+
+ // // ==================== SUBMENU PENANGANAN DARURAT =========================
+ await seedPenangananDarurat();
+
+ // // ==================== SUBMENU KONTAK DARURAT =========================
+ await seedKontakDarurat();
+
+ // // ==================== SUBMENU INFO WABAH PENYAKIT =========================
+ await seedInfoWabahPenyakit();
+
+ // // ====================== MENU KEAMANAN ========================
+ // // ==================== SUBMENU KEAMANAN LINGKUNGAN ============
+ await seedKeamananLingkungan();
+
+ // // ==================== SUBMENU POLSEK TERDEKAT ================
+ await seedPolsekTerdekat();
+
+ // // ==================== SUBMENU KONTAK DARURAT =================
+ await seedKontakDaruratKeamanan();
+
+ // // ==================== SUBMENU PENCEGAHAN KRIMINALITAS ========
+ await seedPencegahanKriminalitas();
+ // // ==================== SUBMENU LAPORAN PUBLIK =================
+ await seedLaporanPublik();
+
+ // // ==================== SUBMENU TIPS KEAMANAN ==================
+ await seedKeamananLingkungan();
+
+ // // ====================== MENU EKONOMI ========================
+ // // ==================== SUBMENU PASAR DESA ====================
+ await seedPasarDesa();
+
+ // // ==================== SUBMENU LOWONGAN KERJA LOKAL ==========
+ await seedLowonganKerjaLokal();
+
+ // // ==================== SUBMENU STRUKTUR ORGANISASI DAN SK PENGURUS BUMDES ==========
+ await seedStrukturBumdes();
+
+ // // ==================== SUBMENU PENDAPATAN ASLI DESA ==========
+ await seedPendapatanAsli();
+
+ // // ==================== SUBMENU JUMLAH PENGANGGURAN ==========
+ await seedJumlahPengangguran();
+
+ // // ==================== SUBMENU PENDUDUK USIA KERJA ==========
+ await seedPendudukUsiaKerjaYangMenganggur();
+
+ // // ==================== SUBMENU PENDUDUK MISKIN =============
+ await seedJumlahPendudukMiskin();
+
+ // // ==================== SUBMENU PROGRAM KEMISKINAN =============
+ await seedProgramKemiskinan();
+
+ // // ==================== SUBMENU SEKTOR UNGGULAN DESA =============
+ await seedSektorUnggulanDesa();
+
+ // // ==================== SUBMENU DEMOGRAFI PEKERJAAN =============
+ await seedDemografiPekerjaan();
+
+ // // ====================== MENU INOVASI ==========================
+ // // ====================== SUBMENU AJUKAN IDE INOVATIF ===========
+ await seedAjukan();
+ // // ==================== SUBMENU DESA DIGITAL ====================
+ await seedDesaDigital();
+
+ // // ==================== SUBMENU LAYANAN ONLINE DESA ==========
+ await seedLayananOnlineDesa();
+
+ // // ==================== SUBMENU PROGRAM KREATIF ==========
+ await seedProgramKreatifDesa();
+ // // ==================== SUBMENU KOLABORASI INOVASI ==========
+ await seedKolaborasiInovasi();
+
+ // // ==================== SUBMENU INFO TEKNOLOGI TEPAT GUNA ==========
+ await seedInfoTeknologi();
+
+ // // ====================== MENU LINGKUNGAN ==========================
+ // // ==================== SUBMENU PENGELOLAAN SAMPAH ==========
+ await seedPengelolaanSampah();
+
+ // // ==================== SUBMENU PROGRAM PENGHIJAUAN ==========
+ await seedProgramPenghijauan();
+
+ // // ==================== SUBMENU DATA LINGKUNGAN DESA ==========
+ await seedDataLingkunganDesa();
+
+ // // =========== SUBMENU GOTONG ROYONG ===========
+ await seedDataGotongRoyong();
+
+ // // =========== SUBMENU EDUKASI LINGKUNGAN ===========
+ await seedEdukasiLingkungan();
+
+ // // =========== SUBMENU KONSERVASI ADAT BALI ===========
+ await seedKonservasiAdatBali();
+
+ // // ====================== MENU PENDIDIKAN ==========================
+ // // =========== SUBMENU INFO SEKOLAH =====================
+ await seedInfoSekolah();
+
+ // // =========== SUBMENU PROGRAM PENDIDIKAN ANAK =====================
+
+ await seedInfoProgramPendidikan();
+
+ // // =========== SUBMENU BIMBINGAN BELAJAR DESA =====================
+ await seedBimbinganBelajar();
+
+ // // =========== SUBMENU PENDIDIKAN NON FORMAL =====================
+ await seedPendidikanNonFormal();
+ // // =========== SUBMENU PERPUSTAKAAN DIGITAL =====================
+ await seedDataPerpustakaan();
+
+ // =========== SUBMENU DATA PENDIDIKAN =====================
+ await seedDataPendidikan();
})()
.then(() => prisma.$disconnect())
.catch((e) => {
diff --git a/prisma/seed_assets.ts b/prisma/seed_assets.ts
index f92c0d36..11f1ede2 100644
--- a/prisma/seed_assets.ts
+++ b/prisma/seed_assets.ts
@@ -1,118 +1,49 @@
-// prisma/seedAssets.ts
-import fs from "fs/promises";
-import path from "path";
-import sharp from "sharp";
-import fetch from "node-fetch";
-import AdmZip from "adm-zip";
+/* eslint-disable @typescript-eslint/no-explicit-any */
import prisma from "@/lib/prisma";
-const UPLOADS_DIR =
- process.env.WIBU_UPLOAD_DIR || path.join(process.cwd(), "uploads");
-
-// --- Helper: deteksi kategori file ---
-function detectCategory(filename: string): "image" | "document" | "other" {
- const ext = path.extname(filename).toLowerCase();
- if ([".jpg", ".jpeg", ".png", ".webp"].includes(ext)) return "image";
- if ([".pdf", ".doc", ".docx"].includes(ext)) return "document";
- return "other";
-}
-
-// --- Helper: recursive walk dir ---
-async function walkDir(dir: string, fileList: string[] = []): Promise {
- const entries = await fs.readdir(dir, { withFileTypes: true });
-
- for (const entry of entries) {
- const fullPath = path.join(dir, entry.name);
-
- if (entry.isDirectory()) {
- if (entry.name === "__MACOSX") continue; // skip folder sampah
- await walkDir(fullPath, fileList);
- } else {
- if (entry.name.startsWith(".") || entry.name === ".DS_Store") continue; // skip file sampah
- fileList.push(fullPath);
- }
- }
-
- return fileList;
-}
+import { getAllPublicCdnUrls } from "./lib/create_file_share_folder";
export default async function seedAssets() {
- console.log("🚀 Seeding assets...");
+ const images = await getAllPublicCdnUrls();
- // 1. Download zip
- const url =
- "https://cld-dkr-makuro-seafile.wibudev.com/f/ffd5a548a04f47939474/?dl=1";
- const res = await fetch(url);
- if (!res.ok) throw new Error(`Gagal download assets: ${res.statusText}`);
- const buffer = Buffer.from(await res.arrayBuffer());
+ for (const img of images) {
+ try {
+ // Check if the image already exists by name
+ const existingImage = await prisma.fileStorage.findUnique({
+ where: { name: img.name },
+ });
- // 2. Extract zip ke folder tmp
- const extractDir = path.join(process.cwd(), "tmp_assets");
- await fs.rm(extractDir, { recursive: true, force: true });
- await fs.mkdir(extractDir, { recursive: true });
-
- const zip = new AdmZip(buffer);
- zip.extractAllTo(extractDir, true);
-
- // 3. Cari semua file valid (recursive)
- const files = await walkDir(extractDir);
-
- // 4. Loop tiap file & simpan
- for (const filePath of files) {
- const entryName = path.basename(filePath);
- const category = detectCategory(entryName);
-
- let finalName = entryName;
- let mimeType = "application/octet-stream";
- let targetPath = "";
-
- if (category === "image") {
- const fileBaseName = path.parse(entryName).name;
- finalName = `${fileBaseName}.webp`;
- targetPath = path.join(UPLOADS_DIR, "images", finalName);
- await fs.mkdir(path.dirname(targetPath), { recursive: true });
- await sharp(filePath).webp({ quality: 80 }).toFile(targetPath);
- mimeType = "image/webp";
- } else if (category === "document") {
- targetPath = path.join(UPLOADS_DIR, "documents", entryName);
- await fs.mkdir(path.dirname(targetPath), { recursive: true });
- await fs.copyFile(filePath, targetPath);
- mimeType = "application/pdf";
- } else {
- targetPath = path.join(UPLOADS_DIR, "other", entryName);
- await fs.mkdir(path.dirname(targetPath), { recursive: true });
- await fs.copyFile(filePath, targetPath);
+ if (!existingImage) {
+ // Only create if it doesn't exist
+ await prisma.fileStorage.create({
+ data: {
+ name: img.name,
+ category: "image",
+ mimeType: "image/webp",
+ link: img.cdnUrl,
+ path: "images",
+ realName: img.name,
+ isActive: true,
+ },
+ });
+ console.log(`✅ Created new image: ${img.name}`);
+ } else {
+ console.log(`ℹ️ Image already exists, skipping: ${img.name}`);
+ }
+ } catch (err) {
+ console.log(`❌ Failed to seed asset ${img.name}:`, JSON.stringify(err));
}
-
- // 5. Simpan ke DB
- await prisma.fileStorage.create({
- data: {
- name: finalName,
- realName: entryName,
- path: targetPath,
- mimeType,
- link: `/uploads/${category}/${finalName}`,
- category,
- },
- });
-
- console.log(`📂 saved: ${category}/${finalName}`);
}
- // 6. Cleanup
- await fs.rm(extractDir, { recursive: true, force: true });
-
- console.log("✅ Selesai seed assets!");
+ console.log("🎉 Image seeding completed");
}
-// --- Auto run kalau dipanggil langsung ---
if (import.meta.main) {
seedAssets()
- .catch((err) => {
- console.error("❌ Error seeding assets:", err);
- process.exit(1);
+ .then(() => {
+ console.log("seed assets success");
})
- .finally(async () => {
- await prisma.$disconnect();
+ .catch((err) => {
+ console.log("gagal seed assets", JSON.stringify(err));
});
}
diff --git a/public/mangupuraaward.jpeg b/public/mangupuraaward.jpeg
new file mode 100644
index 00000000..22606461
Binary files /dev/null and b/public/mangupuraaward.jpeg differ
diff --git a/public/manifest.json b/public/manifest.json
new file mode 100644
index 00000000..11c993b7
--- /dev/null
+++ b/public/manifest.json
@@ -0,0 +1,21 @@
+{
+ "name": "Desa Darmasaba",
+ "short_name": "Darmasaba",
+ "description": "Website resmi Desa Darmasaba, Kabupaten Badung, Bali",
+ "start_url": "/",
+ "display": "standalone",
+ "background_color": "#ffffff",
+ "theme_color": "#1e40af",
+ "icons": [
+ {
+ "src": "/darmasaba-icon.png",
+ "sizes": "192x192",
+ "type": "image/png"
+ },
+ {
+ "src": "/darmasaba-icon.png",
+ "sizes": "512x512",
+ "type": "image/png"
+ }
+ ]
+}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/_com/createEditor.tsx b/src/app/admin/(dashboard)/_com/createEditor.tsx
index 7878e59a..86cf608c 100644
--- a/src/app/admin/(dashboard)/_com/createEditor.tsx
+++ b/src/app/admin/(dashboard)/_com/createEditor.tsx
@@ -7,6 +7,7 @@ import Underline from '@tiptap/extension-underline';
import TextAlign from '@tiptap/extension-text-align';
import Superscript from '@tiptap/extension-superscript';
import SubScript from '@tiptap/extension-subscript';
+import { useEffect } from 'react';
type CreateEditorProps = {
value: string;
@@ -32,6 +33,13 @@ export default function CreateEditor({ value, onChange }: CreateEditorProps) {
},
});
+ // 👇 Tambahkan efek untuk sinkronisasi value dari luar (resetForm)
+ useEffect(() => {
+ if (editor && value !== editor.getHTML()) {
+ editor.commands.setContent(value || '');
+ }
+ }, [value, editor]);
+
return (
diff --git a/src/app/admin/(dashboard)/_com/editEditor.tsx b/src/app/admin/(dashboard)/_com/editEditor.tsx
index 66317cc2..004a0dc0 100644
--- a/src/app/admin/(dashboard)/_com/editEditor.tsx
+++ b/src/app/admin/(dashboard)/_com/editEditor.tsx
@@ -47,6 +47,7 @@ export default function EditEditor({ value, onChange }: EditEditorProps) {
editor.off('update', updateHandler);
};
}, [editor, onChange]);
+
return (
diff --git a/src/app/admin/(dashboard)/_com/header.tsx b/src/app/admin/(dashboard)/_com/header.tsx
index 39735f4d..0b7c345b 100644
--- a/src/app/admin/(dashboard)/_com/header.tsx
+++ b/src/app/admin/(dashboard)/_com/header.tsx
@@ -1,7 +1,11 @@
+'use client';
+
import React from 'react';
-import { Grid, GridCol, Paper, TextInput, Title } from '@mantine/core';
+import { Grid, GridCol, Paper, TextInput } from '@mantine/core';
import { IconSearch } from '@tabler/icons-react';
-import colors from '@/con/colors';
+import { useDarkMode } from '@/state/darkModeStore';
+import { themeTokens } from '@/utils/themeTokens';
+import { UnifiedTitle } from '@/components/admin/UnifiedTypography';
type HeaderSearchProps = {
title: string;
@@ -18,13 +22,16 @@ const HeaderSearch = ({
value,
onChange,
}: HeaderSearchProps) => {
+ const { isDark } = useDarkMode();
+ const tokens = themeTokens(isDark);
+
return (
- {title}
+ {title}
-
+
diff --git a/src/app/admin/(dashboard)/_com/iconMap.tsx b/src/app/admin/(dashboard)/_com/iconMap.tsx
index de88a3dc..f6da2248 100644
--- a/src/app/admin/(dashboard)/_com/iconMap.tsx
+++ b/src/app/admin/(dashboard)/_com/iconMap.tsx
@@ -27,8 +27,26 @@ import {
IconFiretruck,
IconBuilding,
IconAlertTriangle,
+
+ // ===== Tambahan =====
+ IconLifebuoy,
+ IconRun,
+ IconShield,
+ IconPhoneCall,
+ IconFirstAidKit,
+ IconStethoscope,
+ IconBuildingCommunity,
+ IconFileText,
+ IconInfoCircle,
+ IconMessageReport,
+ IconUsers,
+ IconQuestionMark,
+ IconBook,
} from '@tabler/icons-react'
+/* =======================
+ Icon Keys (DB Safe)
+======================= */
export type IconKey =
| 'ekowisata'
| 'kompetisi'
@@ -50,14 +68,34 @@ export type IconKey =
| 'pelatihan'
| 'subsidi'
| 'layananKesehatan'
+
+ // ===== Keamanan & Darurat =====
| 'polisi'
| 'ambulans'
| 'pemadam'
- | 'rumahSakit'
- | 'bangunan'
| 'darurat'
+ | 'sar'
+ | 'evakuasi'
+ | 'keamanan'
+ | 'teleponDarurat'
+ // ===== Kesehatan =====
+ | 'rumahSakit'
+ | 'puskesmas'
+ | 'klinik'
+ // ===== Pemerintahan =====
+ | 'bangunan'
+ | 'kantorDesa'
+ | 'administrasi'
+ | 'informasi'
+ | 'pengaduan'
+ | 'layananPublik'
+ | 'book'
+
+/* =======================
+ Icon Map
+======================= */
const iconMap: Record> = {
ekowisata: IconLeaf,
kompetisi: IconTrophy,
@@ -79,22 +117,46 @@ const iconMap: Record> = {
pelatihan: IconSchool,
subsidi: IconShoppingCart,
layananKesehatan: IconHospital,
+
+ // ===== Keamanan & Darurat =====
polisi: IconShieldFilled,
ambulans: IconAmbulance,
pemadam: IconFiretruck,
+ darurat: IconAlertTriangle,
+ sar: IconLifebuoy,
+ evakuasi: IconRun,
+ keamanan: IconShield,
+ teleponDarurat: IconPhoneCall,
+
+ // ===== Kesehatan =====
rumahSakit: IconHospital,
+ puskesmas: IconFirstAidKit,
+ klinik: IconStethoscope,
+
+ // ===== Pemerintahan =====
bangunan: IconBuilding,
- darurat: IconAlertTriangle
+ kantorDesa: IconBuildingCommunity,
+ administrasi: IconFileText,
+ informasi: IconInfoCircle,
+ pengaduan: IconMessageReport,
+ layananPublik: IconUsers,
+ book: IconBook
}
+/* =======================
+ Icon Mapper Component
+======================= */
type Props = {
name: IconKey
size?: number
color?: string
}
-export const IconMapper: React.FC = ({ name, size = 24, color }) => {
- const IconComponent = iconMap[name]
- if (!IconComponent) return null
+export const IconMapper: React.FC = ({
+ name,
+ size = 24,
+ color,
+}) => {
+ const IconComponent = iconMap[name] ?? IconQuestionMark
return
}
diff --git a/src/app/admin/(dashboard)/_com/judulList.tsx b/src/app/admin/(dashboard)/_com/judulList.tsx
index 4eaa5731..7f376d11 100644
--- a/src/app/admin/(dashboard)/_com/judulList.tsx
+++ b/src/app/admin/(dashboard)/_com/judulList.tsx
@@ -1,12 +1,16 @@
'use client'
-import colors from '@/con/colors';
-import { Grid, GridCol, Button, Text } from '@mantine/core';
+import { Grid, GridCol, Button } from '@mantine/core';
import { IconCircleDashedPlus } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import React from 'react';
+import { useDarkMode } from '@/state/darkModeStore';
+import { themeTokens } from '@/utils/themeTokens';
+import { UnifiedText } from '@/components/admin/UnifiedTypography';
const JudulList = ({ title = "", href = "#" }) => {
+ const { isDark } = useDarkMode();
+ const tokens = themeTokens(isDark);
const router = useRouter();
const handleNavigate = () => {
@@ -16,10 +20,18 @@ const JudulList = ({ title = "", href = "#" }) => {
return (
- {title}
+ {title}
-
+
diff --git a/src/app/admin/(dashboard)/_com/judulListTab.tsx b/src/app/admin/(dashboard)/_com/judulListTab.tsx
index 21037671..dda1fe69 100644
--- a/src/app/admin/(dashboard)/_com/judulListTab.tsx
+++ b/src/app/admin/(dashboard)/_com/judulListTab.tsx
@@ -1,9 +1,11 @@
'use client'
-import colors from '@/con/colors';
-import { Grid, GridCol, Button, Text, Paper, TextInput } from '@mantine/core';
+import { Grid, GridCol, Button, Paper, TextInput } from '@mantine/core';
import { IconCircleDashedPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import React from 'react';
+import { useDarkMode } from '@/state/darkModeStore';
+import { themeTokens } from '@/utils/themeTokens';
+import { UnifiedText } from '@/components/admin/UnifiedTypography';
type JudulListTabProps = {
title: string;
@@ -14,17 +16,16 @@ type JudulListTabProps = {
onChange?: (e: React.ChangeEvent) => void;
}
-
-
-
const JudulListTab = ({
title = "",
href = "#",
placeholder = "pencarian",
searchIcon = ,
value,
- onChange
+ onChange
}: JudulListTabProps) => {
+ const { isDark } = useDarkMode();
+ const tokens = themeTokens(isDark);
const router = useRouter();
const handleNavigate = () => {
@@ -34,10 +35,17 @@ const JudulListTab = ({
return (
- {title}
+
+ {title}
+
-
+
-
+
diff --git a/src/app/admin/(dashboard)/_com/modalNonaktif.tsx b/src/app/admin/(dashboard)/_com/modalNonaktif.tsx
new file mode 100644
index 00000000..9b3c3826
--- /dev/null
+++ b/src/app/admin/(dashboard)/_com/modalNonaktif.tsx
@@ -0,0 +1,36 @@
+// components/modal/ModalKonfirmasiHapus.tsx
+import colors from "@/con/colors"
+import { Modal, Text, Button, Flex } from "@mantine/core"
+
+interface ModalKonfirmasiNonAktifProps {
+ opened: boolean
+ loading?: boolean
+ onClose: () => void
+ onConfirm: () => void
+ text: string
+}
+
+export function ModalKonfirmasiNonAktif({
+ opened,
+ loading = false,
+ onClose,
+ onConfirm,
+ text,
+}: ModalKonfirmasiNonAktifProps) {
+ return (
+ Konfirmasi Non Aktif}
+ centered
+ >
+ {text}
+
+ Batal
+
+ Yakin Non Aktif
+
+
+
+ )
+}
diff --git a/src/app/admin/(dashboard)/_com/selectIcon.tsx b/src/app/admin/(dashboard)/_com/selectIcon.tsx
index 88e17af8..421ba3ca 100644
--- a/src/app/admin/(dashboard)/_com/selectIcon.tsx
+++ b/src/app/admin/(dashboard)/_com/selectIcon.tsx
@@ -5,28 +5,40 @@ import { Box, rem, Select } from '@mantine/core';
import {
IconAlertTriangle,
IconAmbulance,
+ IconBook,
IconBuilding,
+ IconBuildingCommunity,
IconCash,
IconChartLine,
IconChristmasTreeFilled,
IconClipboardTextFilled,
IconDroplet,
+ IconFileText,
IconFiretruck,
+ IconFirstAidKit,
IconHome,
IconHomeEco,
IconHospital,
+ IconInfoCircle,
IconLeaf,
+ IconLifebuoy,
+ IconMessageReport,
+ IconPhoneCall,
IconRecycle,
+ IconRun,
IconScale,
IconSchool,
+ IconShield,
IconShieldFilled,
IconShoppingCart,
+ IconStethoscope,
IconTent,
IconTrashFilled,
IconTree,
IconTrendingUp,
IconTrophy,
IconTruckFilled,
+ IconUsers,
} from '@tabler/icons-react';
import { useEffect, useState } from 'react';
@@ -51,15 +63,33 @@ const iconMap = {
pelatihan: { label: 'Pelatihan', icon: IconSchool },
subsidi: { label: 'Subsidi', icon: IconShoppingCart },
layananKesehatan: { label: 'Layanan Kesehatan', icon: IconHospital },
+
+ // ===== Keamanan & Darurat =====
polisi: { label: 'Polisi', icon: IconShieldFilled },
ambulans: { label: 'Ambulans', icon: IconAmbulance },
- pemadam: { label: 'Pemadam', icon: IconFiretruck },
- rumahSakit: { label: 'Rumah Sakit', icon: IconHospital },
- bangunan: { label: 'Bangunan', icon: IconBuilding },
+ pemadam: { label: 'Pemadam Kebakaran', icon: IconFiretruck },
darurat: { label: 'Darurat', icon: IconAlertTriangle },
+ sar: { label: 'SAR / Basarnas', icon: IconLifebuoy },
+ evakuasi: { label: 'Evakuasi', icon: IconRun },
+ keamanan: { label: 'Keamanan', icon: IconShield },
+ teleponDarurat: { label: 'Telepon Darurat', icon: IconPhoneCall },
+ // ===== Kesehatan =====
+ rumahSakit: { label: 'Rumah Sakit', icon: IconHospital },
+ puskesmas: { label: 'Puskesmas', icon: IconFirstAidKit },
+ klinik: { label: 'Klinik', icon: IconStethoscope },
+
+ // ===== Pemerintahan & Fasilitas =====
+ bangunan: { label: 'Bangunan', icon: IconBuilding },
+ kantorDesa: { label: 'Kantor Desa', icon: IconBuildingCommunity },
+ administrasi: { label: 'Administrasi', icon: IconFileText },
+ informasi: { label: 'Informasi', icon: IconInfoCircle },
+ pengaduan: { label: 'Pengaduan', icon: IconMessageReport },
+ layananPublik: { label: 'Layanan Publik', icon: IconUsers },
+ book: { label: 'Buku', icon: IconBook }
};
+
type IconKey = keyof typeof iconMap;
const iconList = Object.entries(iconMap).map(([value, data]) => ({
diff --git a/src/app/admin/(dashboard)/_com/selectIconEdit.tsx b/src/app/admin/(dashboard)/_com/selectIconEdit.tsx
index 093a41df..2f87177e 100644
--- a/src/app/admin/(dashboard)/_com/selectIconEdit.tsx
+++ b/src/app/admin/(dashboard)/_com/selectIconEdit.tsx
@@ -1,6 +1,6 @@
-'use client'
+'use client';
-import { Box, rem, Select } from '@mantine/core';
+import { Box, Group, rem, Select, SelectProps } from '@mantine/core';
import {
IconAmbulance,
IconCash,
@@ -25,7 +25,19 @@ import {
IconTrophy,
IconTruckFilled,
IconBuilding,
- IconAlertTriangle
+ IconAlertTriangle,
+ IconBuildingCommunity,
+ IconFileText,
+ IconFirstAidKit,
+ IconInfoCircle,
+ IconLifebuoy,
+ IconMessageReport,
+ IconPhoneCall,
+ IconRun,
+ IconShield,
+ IconStethoscope,
+ IconUsers,
+ IconBook,
} from '@tabler/icons-react';
const iconMap = {
@@ -38,26 +50,44 @@ const iconMap = {
scale: { label: 'Scale', icon: IconScale },
clipboard: { label: 'Clipboard', icon: IconClipboardTextFilled },
trash: { label: 'Trash', icon: IconTrashFilled },
- lingkunganSehat: {label: 'Lingkungan Sehat', icon: IconHomeEco},
- sumberOksigen: {label: 'Sumber Oksigen', icon: IconChristmasTreeFilled},
- ekonomiBerkelanjutan: {label: 'Ekonomi Berkelanjutan', icon: IconTrendingUp},
- mencegahBencana: {label: 'Mencegah Bencana', icon: IconShieldFilled},
- rumah: {label: 'Rumah', icon: IconHome},
- pohon: {label: 'Pohon', icon: IconTree},
- air: {label: 'Air', icon: IconDroplet},
- bantuan: {label: 'Bantuan', icon: IconCash},
- pelatihan: {label: 'Pelatihan', icon: IconSchool},
- subsidi: {label: 'Subsidi', icon: IconShoppingCart},
- layananKesehatan: {label: 'Layanan Kesehatan', icon: IconHospital},
- polisi: {label: 'Polisi', icon: IconShieldFilled},
- ambulans: {label: 'Ambulans', icon: IconAmbulance},
- pemadam: {label: 'Pemadam', icon: IconFiretruck},
- rumahSakit: {label: 'Rumah Sakit', icon: IconHospital},
- bangunan: {label: 'Bangunan', icon: IconBuilding},
- darurat: {label: 'Darurat', icon: IconAlertTriangle},
+ lingkunganSehat: { label: 'Lingkungan Sehat', icon: IconHomeEco },
+ sumberOksigen: { label: 'Sumber Oksigen', icon: IconChristmasTreeFilled },
+ ekonomiBerkelanjutan: { label: 'Ekonomi Berkelanjutan', icon: IconTrendingUp },
+ mencegahBencana: { label: 'Mencegah Bencana', icon: IconShieldFilled },
+ rumah: { label: 'Rumah', icon: IconHome },
+ pohon: { label: 'Pohon', icon: IconTree },
+ air: { label: 'Air', icon: IconDroplet },
+ bantuan: { label: 'Bantuan', icon: IconCash },
+ pelatihan: { label: 'Pelatihan', icon: IconSchool },
+ subsidi: { label: 'Subsidi', icon: IconShoppingCart },
+ layananKesehatan: { label: 'Layanan Kesehatan', icon: IconHospital },
+
+ // ===== Keamanan & Darurat =====
+ polisi: { label: 'Polisi', icon: IconShieldFilled },
+ ambulans: { label: 'Ambulans', icon: IconAmbulance },
+ pemadam: { label: 'Pemadam Kebakaran', icon: IconFiretruck },
+ darurat: { label: 'Darurat', icon: IconAlertTriangle },
+ sar: { label: 'SAR / Basarnas', icon: IconLifebuoy },
+ evakuasi: { label: 'Evakuasi', icon: IconRun },
+ keamanan: { label: 'Keamanan', icon: IconShield },
+ teleponDarurat: { label: 'Telepon Darurat', icon: IconPhoneCall },
+
+ // ===== Kesehatan =====
+ rumahSakit: { label: 'Rumah Sakit', icon: IconHospital },
+ puskesmas: { label: 'Puskesmas', icon: IconFirstAidKit },
+ klinik: { label: 'Klinik', icon: IconStethoscope },
+
+ // ===== Pemerintahan & Fasilitas =====
+ bangunan: { label: 'Bangunan', icon: IconBuilding },
+ kantorDesa: { label: 'Kantor Desa', icon: IconBuildingCommunity },
+ administrasi: { label: 'Administrasi', icon: IconFileText },
+ informasi: { label: 'Informasi', icon: IconInfoCircle },
+ pengaduan: { label: 'Pengaduan', icon: IconMessageReport },
+ layananPublik: { label: 'Layanan Publik', icon: IconUsers },
+ book: { label: 'Buku', icon: IconBook }
};
-type IconKey = keyof typeof iconMap;
+export type IconKey = keyof typeof iconMap;
const iconList = Object.entries(iconMap).map(([value, data]) => ({
value,
@@ -67,44 +97,52 @@ const iconList = Object.entries(iconMap).map(([value, data]) => ({
export default function SelectIconProgramEdit({
onChange,
value,
+ ...props
}: {
- onChange: (value: IconKey) => void;
- value: IconKey;
-}) {
- const IconComponent = iconMap[value]?.icon || null;
-
+ onChange: (value: IconKey | '') => void;
+ value: IconKey | '';
+} & Omit) {
return (
{
- if (value) onChange(value as IconKey);
+ value={value || ''}
+ onChange={(val: string | null) => {
+ if (val) {
+ onChange(val as IconKey);
+ } else {
+ onChange('');
+ }
}}
data={iconList}
+ renderOption={({ option }) => {
+ const Icon = iconMap[option.value as IconKey]?.icon;
+ return (
+
+ {Icon && }
+ {option.label}
+
+ );
+ }}
leftSection={
- IconComponent && (
-
-
+ value && iconMap[value as IconKey] ? (
+
+ {(() => {
+ const Icon = iconMap[value as IconKey].icon;
+ return ;
+ })()}
- )
+ ) : null
}
- withCheckIcon={false}
- searchable={false}
- rightSectionWidth={0}
+ searchable
styles={{
input: {
- textAlign: 'left',
- fontSize: rem(16),
paddingLeft: 40,
- },
- section: {
- left: 10,
- right: 'auto',
+ fontSize: rem(16),
},
}}
+ {...props}
/>
);
-}
-
+}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/_com/selectSocialMedia.tsx b/src/app/admin/(dashboard)/_com/selectSocialMedia.tsx
new file mode 100644
index 00000000..e92222ed
--- /dev/null
+++ b/src/app/admin/(dashboard)/_com/selectSocialMedia.tsx
@@ -0,0 +1,76 @@
+'use client';
+
+import { Box, Image, Select, rem } from '@mantine/core';
+
+const sosmedMap = {
+ facebook: { label: 'Facebook', src: '/assets/images/sosmed/facebook.png' },
+ instagram: { label: 'Instagram', src: '/assets/images/sosmed/instagram.png' },
+ tiktok: { label: 'Tiktok', src: '/assets/images/sosmed/tiktok.png' },
+ youtube: { label: 'YouTube', src: '/assets/images/sosmed/youtube.png' },
+ whatsapp: { label: 'WhatsApp', src: '/assets/images/sosmed/whatsapp.png' },
+ gmail: { label: 'Gmail', src: '/assets/images/sosmed/gmail.png' },
+ telegram: { label: 'Telegram', src: '/assets/images/sosmed/telegram.png' },
+ x: { label: 'X (Twitter)', src: '/assets/images/sosmed/x-twitter.png' },
+ telephone: { label: 'Telephone', src: '/assets/images/sosmed/telephone-call.png' },
+ custom: { label: 'Custom Icon', src: null },
+};
+
+type SosmedKey = keyof typeof sosmedMap;
+
+const sosmedList = Object.entries(sosmedMap).map(([value, item]) => ({
+ value,
+ label: item.label,
+}));
+
+export default function SelectSosialMedia({
+ value,
+ onChange,
+}: {
+ value: SosmedKey;
+ onChange: (value: SosmedKey) => void;
+}) {
+ const selected = value;
+ const selectedImage = sosmedMap[selected]?.src;
+
+ return (
+
+ val && onChange(val as SosmedKey)}
+ styles={{
+ input: {
+ textAlign: 'left',
+ fontSize: rem(16),
+ paddingLeft: 36,
+ },
+ section: {
+ left: 10,
+ right: 'auto',
+ },
+ }}
+ />
+
+ {/* 🔥 PREVIEW DIPISAH DI LUAR SELECT */}
+ {selectedImage && (
+
+
+
+ )}
+
+ );
+}
diff --git a/src/app/admin/(dashboard)/_com/selectSocialMediaEdit.tsx b/src/app/admin/(dashboard)/_com/selectSocialMediaEdit.tsx
new file mode 100644
index 00000000..185037fd
--- /dev/null
+++ b/src/app/admin/(dashboard)/_com/selectSocialMediaEdit.tsx
@@ -0,0 +1,56 @@
+'use client';
+
+import { Box, Select } from '@mantine/core';
+import { useEffect, useState } from 'react';
+
+export const sosmedMap = {
+ facebook: { label: 'Facebook', src: '/assets/images/sosmed/facebook.png' },
+ instagram: { label: 'Instagram', src: '/assets/images/sosmed/instagram.png' },
+ tiktok: { label: 'Tiktok', src: '/assets/images/sosmed/tiktok.png' },
+ youtube: { label: 'YouTube', src: '/assets/images/sosmed/youtube.png' },
+ whatsapp: { label: 'WhatsApp', src: '/assets/images/sosmed/whatsapp.png' },
+ gmail: { label: 'Gmail', src: '/assets/images/sosmed/gmail.png' },
+ telegram: { label: 'Telegram', src: '/assets/images/sosmed/telegram.png' },
+ x: { label: 'X (Twitter)', src: '/assets/images/sosmed/x-twitter.png' },
+ telephone: { label: 'Telephone', src: '/assets/images/sosmed/telephone-call.png' },
+ custom: { label: 'Custom Icon', src: null },
+};
+
+type SosmedKey = keyof typeof sosmedMap;
+
+const sosmedList = Object.entries(sosmedMap).map(([value, item]) => ({
+ value,
+ label: item.label,
+}));
+
+export default function SelectSocialMediaEdit({
+ value,
+ onChange,
+}: {
+ value: string;
+ onChange: (val: SosmedKey) => void;
+}) {
+ const [selected, setSelected] = useState('facebook');
+
+ useEffect(() => {
+ if (value && sosmedMap[value as SosmedKey]) {
+ setSelected(value as SosmedKey);
+ }
+ }, [value]);
+
+ return (
+
+ {
+ if (!val) return;
+ setSelected(val as SosmedKey);
+ onChange(val as SosmedKey);
+ }}
+ />
+
+ );
+}
diff --git a/src/app/admin/(dashboard)/_state/desa/penghargaan.ts b/src/app/admin/(dashboard)/_state/desa/penghargaan.ts
index 68be0ba7..20921427 100644
--- a/src/app/admin/(dashboard)/_state/desa/penghargaan.ts
+++ b/src/app/admin/(dashboard)/_state/desa/penghargaan.ts
@@ -39,7 +39,7 @@ const penghargaanState = proxy({
);
if (res.status === 200) {
penghargaanState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/desa/pengumuman.ts b/src/app/admin/(dashboard)/_state/desa/pengumuman.ts
index 09320003..702b006d 100644
--- a/src/app/admin/(dashboard)/_state/desa/pengumuman.ts
+++ b/src/app/admin/(dashboard)/_state/desa/pengumuman.ts
@@ -68,7 +68,7 @@ const category = proxy({
const res = await ApiFetch.api.desa.kategoripengumuman[
"findMany"
].get({
- query: { page, limit },
+ query: { page, limit, search },
});
if (res.status === 200 && res.data?.success) {
@@ -287,7 +287,7 @@ const pengumuman = proxy({
);
if (res.status === 200) {
pengumuman.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/desa/potensi.ts b/src/app/admin/(dashboard)/_state/desa/potensi.ts
index 0c158b38..3eb55c54 100644
--- a/src/app/admin/(dashboard)/_state/desa/potensi.ts
+++ b/src/app/admin/(dashboard)/_state/desa/potensi.ts
@@ -65,7 +65,7 @@ const potensiDesa = proxy({
const res = await ApiFetch.api.desa.potensi[
"find-many"
].get({
- query: { page, limit },
+ query: { page, limit, search },
});
if (res.status === 200 && res.data?.success) {
diff --git a/src/app/admin/(dashboard)/_state/desa/profile.ts b/src/app/admin/(dashboard)/_state/desa/profile.ts
index b8bb16d8..26ee5240 100644
--- a/src/app/admin/(dashboard)/_state/desa/profile.ts
+++ b/src/app/admin/(dashboard)/_state/desa/profile.ts
@@ -552,7 +552,7 @@ const maskotDesa = proxy({
deskripsi: profileData.deskripsi || "",
images: (profileData.images || []).map((img) => ({
label: img.label,
- imageId: img.image.id,
+ imageId: img?.image?.id || "",
})),
};
},
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/PADesa.ts b/src/app/admin/(dashboard)/_state/ekonomi/PADesa.ts
index fd3dd897..6b09c7e5 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/PADesa.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/PADesa.ts
@@ -101,6 +101,38 @@ const ApbDesa = proxy({
}
},
},
+ findFirst: {
+ data: null as Prisma.ApbDesaGetPayload<{
+ include: { pendapatan: true; belanja: true; pembiayaan: true };
+ }> | null,
+ loading: false,
+ async load(params?: Record) {
+ try {
+ this.loading = true;
+
+ // ✅ request ke endpoint find-first
+ const res = await ApiFetch.api.ekonomi.pendapatanaslidesa.apbdesa[
+ "find-first"
+ ].get({ query: params || {} });
+
+ if (res.status === 200 && res.data?.success) {
+ this.data = res.data.data ?? null;
+ } else {
+ this.data = null;
+ toast.error(res.data?.message || "Gagal memuat data pertama APB Desa");
+ }
+ } catch (error) {
+ console.error("Error findFirst APB Desa:", error);
+ toast.error("Gagal memuat data APB Desa pertama");
+ this.data = null;
+ } finally {
+ this.loading = false;
+ }
+ },
+ reset() {
+ this.data = null;
+ },
+ },
update: {
id: "",
form: { ...ApbDesaDefaultForm },
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/demografi-pekerjaan.ts b/src/app/admin/(dashboard)/_state/ekonomi/demografi-pekerjaan.ts
index bf29247b..3f4da69b 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/demografi-pekerjaan.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/demografi-pekerjaan.ts
@@ -49,7 +49,7 @@ const demografiPekerjaan = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
demografiPekerjaan.create.form = { ...defaultForm };
demografiPekerjaan.findMany.load();
return id;
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/jumlah-penduduk-miskin.ts b/src/app/admin/(dashboard)/_state/ekonomi/jumlah-penduduk-miskin.ts
index 2eb11a03..5acad685 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/jumlah-penduduk-miskin.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/jumlah-penduduk-miskin.ts
@@ -47,7 +47,7 @@ const jumlahPendudukMiskin = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
jumlahPendudukMiskin.create.form = {
year: 0,
totalPoorPopulation: 0,
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/jumlah-pengangguran.ts b/src/app/admin/(dashboard)/_state/ekonomi/jumlah-pengangguran.ts
index c45ac24a..4fe9939a 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/jumlah-pengangguran.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/jumlah-pengangguran.ts
@@ -89,7 +89,7 @@ const jumlahPengangguran = proxy({
if (res.status === 200) {
const id = res.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
jumlahPengangguran.create.form = { ...jumlahPengangguranForm };
jumlahPengangguran.findMany.load();
return id;
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/lowongan-kerja.ts b/src/app/admin/(dashboard)/_state/ekonomi/lowongan-kerja.ts
index 227c6b8c..1cf5941b 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/lowongan-kerja.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/lowongan-kerja.ts
@@ -47,7 +47,7 @@ const lowonganKerjaState = proxy({
);
if (res.status === 200) {
lowonganKerjaState.create.loading = false;
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/pasar-desa/pasar-desa.ts b/src/app/admin/(dashboard)/_state/ekonomi/pasar-desa/pasar-desa.ts
index 21d8e7b4..4d64fda8 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/pasar-desa/pasar-desa.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/pasar-desa/pasar-desa.ts
@@ -13,6 +13,7 @@ const templatePasarDesaForm = z.object({
rating: z.number().min(1, "Rating minimal 1"),
kategoriId: z.array(z.string()).min(1, "Minimal pilih satu kategori"),
kontak: z.string().min(1, "Kontak wajib diisi"),
+ deskripsi: z.string().min(1, "Deskripsi wajib diisi"),
});
const defaultPasarDesaForm = {
@@ -23,6 +24,7 @@ const defaultPasarDesaForm = {
rating: 0,
kategoriId: [] as string[],
kontak: "",
+ deskripsi: ""
};
const pasarDesa = proxy({
@@ -191,6 +193,7 @@ const pasarDesa = proxy({
rating: data.rating,
kategoriId: data.kategoriId,
kontak: data.kontak,
+ deskripsi: data.deskripsi
};
return data;
} else {
@@ -229,6 +232,7 @@ const pasarDesa = proxy({
rating: this.form.rating,
kategoriId: this.form.kategoriId,
kontak: this.form.kontak,
+ deskripsi: this.form.deskripsi
}),
});
if (!response.ok) {
@@ -312,15 +316,15 @@ const kategoriProduk = proxy({
page: 1,
totalPages: 1,
loading: false,
- search2: "",
- load: async (page = 1, limit = 10, search2 = "") => {
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
kategoriProduk.findMany.loading = true; // ✅ Akses langsung via nama path
kategoriProduk.findMany.page = page;
- kategoriProduk.findMany.search2 = search2;
+ kategoriProduk.findMany.search = search;
try {
const query: any = { page, limit };
- if (search2) query.search2 = search2;
+ if (search) query.search = search;
const res = await ApiFetch.api.ekonomi.kategoriproduk["find-many"].get({ query });
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts b/src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts
index 1332bbea..8de0655e 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/program-kemiskinan.ts
@@ -45,7 +45,7 @@ const programKemiskinanState = proxy({
);
if (res.status === 200) {
programKemiskinanState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/sektor-unggulan-desa.ts b/src/app/admin/(dashboard)/_state/ekonomi/sektor-unggulan-desa.ts
index 341398f0..0c78ec15 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/sektor-unggulan-desa.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/sektor-unggulan-desa.ts
@@ -46,7 +46,7 @@ const grafikSektorUnggulan = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
grafikSektorUnggulan.create.form = {
name: "",
description: "",
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/struktur-organisasi/struktur-organisasi.ts b/src/app/admin/(dashboard)/_state/ekonomi/struktur-organisasi/struktur-organisasi.ts
index cef5c3fa..e347f63c 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/struktur-organisasi/struktur-organisasi.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/struktur-organisasi/struktur-organisasi.ts
@@ -194,7 +194,7 @@ const posisiOrganisasi = proxy({
try {
this.loading = true;
- const res = await ApiFetch.api.ekonomi['struktur-organisasi']['posisi-organisasi']['create'].post(this.form);
+ const res = await ApiFetch.api.ekonomi["struktur-organisasi"]["posisi-organisasi"]["create"].post(this.form);
if (res.status === 200) {
toast.success("Berhasil menambahkan posisi organisasi");
posisiOrganisasi.findMany.load();
diff --git a/src/app/admin/(dashboard)/_state/ekonomi/usia-kerja-nganggur.ts b/src/app/admin/(dashboard)/_state/ekonomi/usia-kerja-nganggur.ts
index 52cd2f92..7625d59f 100644
--- a/src/app/admin/(dashboard)/_state/ekonomi/usia-kerja-nganggur.ts
+++ b/src/app/admin/(dashboard)/_state/ekonomi/usia-kerja-nganggur.ts
@@ -51,7 +51,7 @@ const grafikBerdasarkanUsiaKerjaNganggur = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
grafikBerdasarkanUsiaKerjaNganggur.create.form = {
usia18_25: "",
usia26_35: "",
@@ -255,7 +255,7 @@ const grafikBerdasarkanPendidikan = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
grafikBerdasarkanPendidikan.create.form = {
SD: "",
SMP: "",
diff --git a/src/app/admin/(dashboard)/_state/inovasi/desa-digital.ts b/src/app/admin/(dashboard)/_state/inovasi/desa-digital.ts
index 83d4d5cd..7afacdc2 100644
--- a/src/app/admin/(dashboard)/_state/inovasi/desa-digital.ts
+++ b/src/app/admin/(dashboard)/_state/inovasi/desa-digital.ts
@@ -37,7 +37,7 @@ const desaDigitalState = proxy({
);
if (res.status === 200) {
desaDigitalState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/inovasi/info-tekno.ts b/src/app/admin/(dashboard)/_state/inovasi/info-tekno.ts
index 710bbcd8..7facc93b 100644
--- a/src/app/admin/(dashboard)/_state/inovasi/info-tekno.ts
+++ b/src/app/admin/(dashboard)/_state/inovasi/info-tekno.ts
@@ -37,7 +37,7 @@ const infoTeknoState = proxy({
);
if (res.status === 200) {
infoTeknoState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/inovasi/program-kreatif.ts b/src/app/admin/(dashboard)/_state/inovasi/program-kreatif.ts
index 345910cc..125d6b71 100644
--- a/src/app/admin/(dashboard)/_state/inovasi/program-kreatif.ts
+++ b/src/app/admin/(dashboard)/_state/inovasi/program-kreatif.ts
@@ -6,9 +6,9 @@ import { proxy } from "valtio";
import { z } from "zod";
const templateForm = z.object({
- name: z.string().min(1, "Nama minimal 1 karakter"),
- deskripsi: z.string().min(1, "Deskripsi minimal 1 karakter"),
- slug: z.string().min(1, "Deskripsi singkat minimal 1 karakter"),
+ name: z.string().min(5, "Nama minimal 5 karakter"),
+ deskripsi: z.string().min(5, "Deskripsi minimal 5 karakter"),
+ slug: z.string().min(5, "Deskripsi singkat minimal 5 karakter"),
icon: z.string().min(1, "Icon minimal 1 karakter"),
});
@@ -29,26 +29,33 @@ const programKreatifState = proxy({
const err = `[${cek.error.issues
.map((v) => `${v.path.join(".")}`)
.join("\n")}] required`;
- return toast.error(err);
+ toast.error(err);
+ return false; // ⬅️ ini penting
}
-
+
try {
programKreatifState.create.loading = true;
const res = await ApiFetch.api.inovasi.programkreatif["create"].post(
programKreatifState.create.form
);
+
if (res.status === 200) {
programKreatifState.findMany.load();
- return toast.success("success create");
+ toast.success("Sukses menambahkan");
+ return true;
}
- console.log(res);
- return toast.error("failed create");
+
+ toast.error("failed create");
+ return false;
} catch (error) {
- console.log((error as Error).message);
+ console.error((error as Error).message);
+ toast.error("Terjadi kesalahan saat create");
+ return false;
} finally {
programKreatifState.create.loading = false;
}
- },
+ }
+
},
findMany: {
data: null as any[] | null,
diff --git a/src/app/admin/(dashboard)/_state/keamanan/keamanan-lingkungan.ts b/src/app/admin/(dashboard)/_state/keamanan/keamanan-lingkungan.ts
index 3fd458f4..1a19352e 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/keamanan-lingkungan.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/keamanan-lingkungan.ts
@@ -37,7 +37,7 @@ const keamananLingkunganState = proxy({
].post(keamananLingkunganState.create.form);
if (res.status === 200) {
keamananLingkunganState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/keamanan/kontak-darurat-keamanan.ts b/src/app/admin/(dashboard)/_state/keamanan/kontak-darurat-keamanan.ts
index af0d2401..c3b70625 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/kontak-darurat-keamanan.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/kontak-darurat-keamanan.ts
@@ -38,7 +38,7 @@ const kontakDaruratKeamananState = proxy({
].post(kontakDaruratKeamananState.create.form);
if (res.status === 200) {
kontakDaruratKeamananState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
@@ -294,7 +294,7 @@ const kontakDaruratItem = proxy({
);
if (res.status === 200) {
kontakDaruratItem.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/keamanan/laporan-publik.ts b/src/app/admin/(dashboard)/_state/keamanan/laporan-publik.ts
index 6db65c2b..5e61039b 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/laporan-publik.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/laporan-publik.ts
@@ -88,7 +88,7 @@ const laporanPublikState = proxy({
if (res.status === 200) {
laporanPublikState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
diff --git a/src/app/admin/(dashboard)/_state/keamanan/pencegahan-kriminalitas.ts b/src/app/admin/(dashboard)/_state/keamanan/pencegahan-kriminalitas.ts
index 8530d809..644bf0dd 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/pencegahan-kriminalitas.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/pencegahan-kriminalitas.ts
@@ -40,7 +40,7 @@ const pencegahanKriminalitasState = proxy({
].post(pencegahanKriminalitasState.create.form);
if (res.status === 200) {
pencegahanKriminalitasState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/keamanan/polsek-terdekat.ts b/src/app/admin/(dashboard)/_state/keamanan/polsek-terdekat.ts
index ad0c696f..03415edb 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/polsek-terdekat.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/polsek-terdekat.ts
@@ -15,7 +15,7 @@ const templateForm = z.object({
namaTempatMaps: z.string().min(1, "Nama Tempat Maps minimal 1 karakter"),
alamatMaps: z.string().min(1, "Alamat Maps minimal 1 karakter"),
linkPetunjukArah: z.string().min(1, "Link Petunjuk Arah minimal 1 karakter"),
- layananPolsekId: z.string().min(1, "Layanan Polsek Id minimal 1 karakter"),
+ layananPolsekId: z.array(z.string()).min(1, "Pilih minimal 1 layanan polsek"),
});
const defaultForm = {
@@ -28,7 +28,7 @@ const defaultForm = {
namaTempatMaps: "",
alamatMaps: "",
linkPetunjukArah: "",
- layananPolsekId: "",
+ layananPolsekId: [] as string[],
};
const polsekTerdekatState = proxy({
@@ -66,6 +66,11 @@ const polsekTerdekatState = proxy({
| Prisma.PolsekTerdekatGetPayload<{
include: {
layananPolsek: true;
+ LayananToPolsek: {
+ include: {
+ layanan: true;
+ }
+ }
};
}>[]
| null,
@@ -104,7 +109,14 @@ const polsekTerdekatState = proxy({
},
findUnique: {
data: null as Prisma.PolsekTerdekatGetPayload<{
- include: { layananPolsek: true };
+ include: {
+ layananPolsek: true;
+ LayananToPolsek: {
+ include: {
+ layanan: true;
+ }
+ }
+ };
}> | null,
async load(id: string) {
try {
@@ -117,7 +129,7 @@ const polsekTerdekatState = proxy({
polsekTerdekatState.findUnique.data = null;
}
} catch (error) {
- console.error("Error fetching data:", error);
+ console.error("Gagal fetch detail polsek terdekat:", error);
polsekTerdekatState.findUnique.data = null;
}
},
@@ -273,10 +285,13 @@ const polsekTerdekatState = proxy({
};
}> | null,
loading: false,
- load: async () => { // Changed to arrow function
+ load: async () => {
+ // Changed to arrow function
polsekTerdekatState.findFirst.loading = true;
try {
- const res = await ApiFetch.api.keamanan.polsekterdekat["find-first"].get();
+ const res = await ApiFetch.api.keamanan.polsekterdekat[
+ "find-first"
+ ].get();
if (res.status === 200 && res.data?.success) {
polsekTerdekatState.findFirst.data = res.data.data || null;
} else {
@@ -287,8 +302,284 @@ const polsekTerdekatState = proxy({
} finally {
polsekTerdekatState.findFirst.loading = false;
}
- }
- }
+ },
+ },
});
-export default polsekTerdekatState;
+const templateFormLayananPolsek = z.object({
+ nama: z.string().min(1, "Nama harus diisi"),
+});
+
+const defaultFormLayananPolsek = {
+ nama: "",
+};
+
+const layananPolsek = proxy({
+ create: {
+ form: { ...defaultFormLayananPolsek },
+ loading: false,
+ async create() {
+ const cek = templateFormLayananPolsek.safeParse(
+ layananPolsek.create.form
+ );
+ if (!cek.success) {
+ const err = `[${cek.error.issues
+ .map((v) => `${v.path.join(".")}`)
+ .join("\n")}] required`;
+ return toast.error(err);
+ }
+
+ try {
+ layananPolsek.create.loading = true;
+ const res = await ApiFetch.api.keamanan["layananpolsek"][
+ "create"
+ ].post(layananPolsek.create.form);
+ if (res.status === 200) {
+ layananPolsek.findManyAll.load();
+ return toast.success("Data Kategori Berita Berhasil Dibuat");
+ }
+ console.log(res);
+ return toast.error("failed create");
+ } catch (error) {
+ console.log(error);
+ return toast.error("failed create");
+ } finally {
+ layananPolsek.create.loading = false;
+ }
+ },
+ },
+ findMany: {
+ data: null as
+ | Prisma.LayananPolsekGetPayload<{
+ omit: {
+ isActive: true;
+ };
+ }>[]
+ | null,
+ page: 1,
+ totalPages: 1,
+ loading: false,
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ layananPolsek.findMany.loading = true; // ✅ Akses langsung via nama path
+ layananPolsek.findMany.page = page;
+ layananPolsek.findMany.search = search;
+
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.keamanan["layananpolsek"]["findMany"].get({ query });
+
+ if (res.status === 200 && res.data?.success) {
+ layananPolsek.findMany.data = res.data.data ?? [];
+ layananPolsek.findMany.totalPages = res.data.totalPages ?? 1;
+ } else {
+ layananPolsek.findMany.data = [];
+ layananPolsek.findMany.totalPages = 1;
+ }
+ } catch (err) {
+ console.error("Gagal fetch layanan polsek paginated:", err);
+ layananPolsek.findMany.data = [];
+ layananPolsek.findMany.totalPages = 1;
+ } finally {
+ layananPolsek.findMany.loading = false;
+ }
+ },
+ },
+ findManyAll: {
+ data: [] as Prisma.LayananPolsekGetPayload<{ omit: { isActive: true } }>[],
+ loading: false,
+
+ async load() {
+ this.loading = true;
+ try {
+ const res = await ApiFetch.api.keamanan["layananpolsek"][
+ "findManyAll"
+ ].get();
+
+ if (res.status === 200 && res.data?.success) {
+ this.data = (res.data.data ?? []).map((item: any) => ({
+ id: String(item.id),
+ nama: String(item.nama || ""),
+ createdAt: item.createdAt ? new Date(item.createdAt) : new Date(),
+ updatedAt: item.updatedAt ? new Date(item.updatedAt) : new Date(),
+ deletedAt: item.deletedAt ? new Date(item.deletedAt) : null,
+ }));
+ } else {
+ this.data = [];
+ }
+ } catch (error) {
+ console.error("Gagal fetch layanan polsek:", error);
+ this.data = [];
+ } finally {
+ this.loading = false;
+ }
+ },
+ },
+ findUnique: {
+ data: null as Prisma.LayananPolsekGetPayload<{
+ omit: {
+ isActive: true;
+ };
+ }> | null,
+ loading: false,
+ async load(id: string) {
+ try {
+ const res = await fetch(`/api/keamanan/layananpolsek/${id}`);
+ if (res.ok) {
+ const data = await res.json();
+ layananPolsek.findUnique.data = data.data ?? null;
+ } else {
+ console.error("Failed to fetch data", res.status, res.statusText);
+ layananPolsek.findUnique.data = null;
+ }
+ } catch (error) {
+ console.error("Error fetching data:", error);
+ layananPolsek.findUnique.data = null;
+ }
+ },
+ },
+ delete: {
+ loading: false,
+ async delete(id: string) {
+ if (!id) return toast.warn("ID tidak valid");
+
+ try {
+ layananPolsek.delete.loading = true;
+
+ const response = await fetch(`/api/keamanan/layananpolsek/del/${id}`, {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ });
+
+ const result = await response.json();
+
+ if (response.ok && result?.success) {
+ toast.success(
+ result.message || "Data layanan polsek berhasil dihapus"
+ );
+ await layananPolsek.findManyAll.load(); // refresh list
+ } else {
+ toast.error(result?.message || "Gagal menghapus Data layanan polsek");
+ }
+ } catch (error) {
+ console.error("Gagal delete:", error);
+ toast.error("Terjadi kesalahan saat menghapus Data layanan polsek");
+ } finally {
+ layananPolsek.delete.loading = false;
+ }
+ },
+ },
+ update: {
+ id: "",
+ form: { ...defaultFormLayananPolsek },
+ loading: false,
+ async load(id: string) {
+ if (!id) {
+ toast.warn("ID tidak valid");
+ return null;
+ }
+
+ try {
+ const response = await fetch(`/api/keamanan/layananpolsek/${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,
+ };
+ return data; // Return the loaded data
+ } else {
+ throw new Error(result?.message || "Gagal memuat data");
+ }
+ } catch (error) {
+ console.error("Error loading layanan polsek:", error);
+ toast.error(
+ error instanceof Error ? error.message : "Gagal memuat data"
+ );
+ return null;
+ }
+ },
+ async update() {
+ const cek = templateFormLayananPolsek.safeParse(
+ layananPolsek.update.form
+ );
+ if (!cek.success) {
+ const err = `[${cek.error.issues
+ .map((v) => `${v.path.join(".")}`)
+ .join("\n")}] required`;
+ toast.error(err);
+ return false;
+ }
+
+ try {
+ layananPolsek.update.loading = true;
+
+ const response = await fetch(
+ `/api/keamanan/layananpolsek/${this.id}`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify({
+ nama: this.form.nama,
+ }),
+ }
+ );
+
+ 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 data layanan polsek");
+ await layananPolsek.findManyAll.load(); // refresh list
+ return true;
+ } else {
+ throw new Error(result.message || "Gagal update data layanan polsek");
+ }
+ } catch (error) {
+ console.error("Error updating data layanan polsek:", error);
+ toast.error(
+ error instanceof Error
+ ? error.message
+ : "Terjadi kesalahan saat update data layanan polsek"
+ );
+ return false;
+ } finally {
+ layananPolsek.update.loading = false;
+ }
+ },
+ reset() {
+ layananPolsek.update.id = "";
+ layananPolsek.update.form = { ...defaultFormLayananPolsek };
+ },
+ },
+});
+
+const statePolsekTerdekat = proxy({
+ polsekTerdekatState,
+ layananPolsek,
+});
+
+export default statePolsekTerdekat;
diff --git a/src/app/admin/(dashboard)/_state/keamanan/tips-keamanan.ts b/src/app/admin/(dashboard)/_state/keamanan/tips-keamanan.ts
index ad1ee158..4bd4609c 100644
--- a/src/app/admin/(dashboard)/_state/keamanan/tips-keamanan.ts
+++ b/src/app/admin/(dashboard)/_state/keamanan/tips-keamanan.ts
@@ -37,7 +37,7 @@ const tipsKeamananState = proxy({
);
if (res.status === 200) {
tipsKeamananState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/fasilitasKesehatan.ts b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/fasilitasKesehatan.ts
index 71c04389..3aa3224e 100644
--- a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/fasilitasKesehatan.ts
+++ b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/fasilitasKesehatan.ts
@@ -9,29 +9,30 @@ import { z } from "zod";
// Validasi form
const templateForm = z.object({
name: z.string().min(1, "Nama harus diisi"),
+
informasiUmum: z.object({
- fasilitas: z.string().min(1, "Fasilitas harus diisi"),
- alamat: z.string().min(1, "Alamat harus diisi"),
- jamOperasional: z.string().min(1, "Jam operasional harus diisi"),
+ fasilitas: z.string().min(1),
+ alamat: z.string().min(1),
+ jamOperasional: z.string().min(1),
}),
+
layananUnggulan: z.object({
- content: z.string().min(1, "Layanan unggulan harus diisi"),
- }),
- dokterdanTenagaMedis: z.object({
- name: z.string().min(1, "Nama dokter harus diisi"),
- specialist: z.string().min(1, "Spesialis harus diisi"),
- jadwal: z.string().min(1, "Jadwal harus diisi"),
+ content: z.string().min(1),
}),
+
+ // NOW ARRAY OF STRING (ID)
+ dokterdanTenagaMedis: z.array(z.string()).min(1, "Minimal pilih 1 dokter"),
+
fasilitasPendukung: z.object({
- content: z.string().min(1, "Fasilitas pendukung harus diisi"),
+ content: z.string().min(1),
}),
+
prosedurPendaftaran: z.object({
- content: z.string().min(1, "Prosedur pendaftaran harus diisi"),
- }),
- tarifDanLayanan: z.object({
- layanan: z.string().min(1, "Layanan harus diisi"),
- tarif: z.string().min(1, "Tarif harus diisi"),
+ content: z.string().min(1),
}),
+
+ // NOW ARRAY OF STRING (ID)
+ tarifDanLayanan: z.array(z.string()).min(1, "Minimal pilih 1 tarif"),
});
// Default form kosong
@@ -45,21 +46,34 @@ const defaultForm = {
layananUnggulan: {
content: "",
},
- dokterdanTenagaMedis: {
- name: "",
- specialist: "",
- jadwal: "",
- },
+
+ dokterdanTenagaMedis: [] as string[], // ← array kosong
+ tarifDanLayanan: [] as string[], // ← array kosong
+
fasilitasPendukung: {
content: "",
},
prosedurPendaftaran: {
content: "",
},
- tarifDanLayanan: {
- layanan: "",
- tarif: "",
- },
+};
+
+type DokterItem = {
+ id: string;
+ name: string;
+ specialist: string;
+ jadwal: string;
+ jadwalLibur: string;
+ jamBukaOperasional: string;
+ jamTutupOperasional: string;
+ jamBukaLibur: string;
+ jamTutupLibur: string;
+};
+
+type TarifItem = {
+ id: string;
+ layanan: string;
+ tarif: string;
};
const fasilitasKesehatan = proxy({
@@ -186,33 +200,26 @@ const fasilitasKesehatan = proxy({
const result = await res.json();
const data = result.data;
-
- fasilitasKesehatan.edit.id = data.id;
- fasilitasKesehatan.edit.form = {
+ this.id = data.id;
+ this.form = {
name: data.name,
informasiUmum: {
fasilitas: data.informasiumum.fasilitas,
alamat: data.informasiumum.alamat,
jamOperasional: data.informasiumum.jamOperasional,
},
- layananUnggulan: {
- content: data.layananunggulan.content,
- },
- dokterdanTenagaMedis: {
- name: data.dokterdantenagamedis.name,
- specialist: data.dokterdantenagamedis.specialist,
- jadwal: data.dokterdantenagamedis.jadwal,
- },
fasilitasPendukung: {
content: data.fasilitaspendukung.content,
},
prosedurPendaftaran: {
content: data.prosedurpendaftaran.content,
},
- tarifDanLayanan: {
- layanan: data.tarifdanlayanan.layanan,
- tarif: data.tarifdanlayanan.tarif,
+ // map relasi -> array of IDs
+ layananUnggulan: {
+ content: data.layananunggulan.content,
},
+ dokterdanTenagaMedis: data.dokterdantenagamedis?.map((v: DokterItem) => v.id) ?? [],
+ tarifDanLayanan: data.tarifdanlayanan?.map((v: TarifItem) => v.id) ?? [],
};
},
async submit() {
@@ -238,22 +245,15 @@ const fasilitasKesehatan = proxy({
layananUnggulan: {
content: fasilitasKesehatan.edit.form.layananUnggulan.content,
},
- dokterdanTenagaMedis: {
- name: fasilitasKesehatan.edit.form.dokterdanTenagaMedis.name,
- specialist:
- fasilitasKesehatan.edit.form.dokterdanTenagaMedis.specialist,
- jadwal: fasilitasKesehatan.edit.form.dokterdanTenagaMedis.jadwal,
- },
+ dokterdanTenagaMedis:
+ fasilitasKesehatan.edit.form.dokterdanTenagaMedis,
fasilitasPendukung: {
content: fasilitasKesehatan.edit.form.fasilitasPendukung.content,
},
prosedurPendaftaran: {
content: fasilitasKesehatan.edit.form.prosedurPendaftaran.content,
},
- tarifDanLayanan: {
- layanan: fasilitasKesehatan.edit.form.tarifDanLayanan.layanan,
- tarif: fasilitasKesehatan.edit.form.tarifDanLayanan.tarif,
- },
+ tarifDanLayanan: fasilitasKesehatan.edit.form.tarifDanLayanan,
};
const res = await fetch(
@@ -320,12 +320,26 @@ const templateDokterForm = z.object({
name: z.string().min(1, "Nama tidak boleh kosong"),
specialist: z.string().min(1, "Spesialis tidak boleh kosong"),
jadwal: z.string().min(1, "Jadwal tidak boleh kosong"),
+ jadwalLibur: z.string().min(1, "Jadwal libur tidak boleh kosong"),
+ jamBukaOperasional: z
+ .string()
+ .min(1, "Jam buka operasional tidak boleh kosong"),
+ jamTutupOperasional: z
+ .string()
+ .min(1, "Jam tutup operasional tidak boleh kosong"),
+ jamBukaLibur: z.string().min(1, "Jam buka libur tidak boleh kosong"),
+ jamTutupLibur: z.string().min(1, "Jam tutup libur tidak boleh kosong"),
});
const defaultDokterForm = {
name: "",
specialist: "",
jadwal: "",
+ jadwalLibur: "",
+ jamBukaOperasional: "",
+ jamTutupOperasional: "",
+ jamBukaLibur: "",
+ jamTutupLibur: "",
};
const dokter = proxy({
@@ -351,7 +365,7 @@ const dokter = proxy({
if (res.status === 200) {
const id = res.data?.data;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
dokter.create.create.form = { ...defaultDokterForm };
dokter.findMany.load();
return id;
@@ -463,6 +477,11 @@ const dokter = proxy({
name: data.name,
specialist: data.specialist,
jadwal: data.jadwal,
+ jadwalLibur: data.jadwalLibur,
+ jamBukaOperasional: data.jamBukaOperasional,
+ jamTutupOperasional: data.jamTutupOperasional,
+ jamBukaLibur: data.jamBukaLibur,
+ jamTutupLibur: data.jamTutupLibur,
};
return data; // Return the loaded data
} else {
@@ -487,6 +506,11 @@ const dokter = proxy({
name: this.form.name,
specialist: this.form.specialist,
jadwal: this.form.jadwal,
+ jadwalLibur: this.form.jadwalLibur,
+ jamBukaOperasional: this.form.jamBukaOperasional,
+ jamTutupOperasional: this.form.jamTutupOperasional,
+ jamBukaLibur: this.form.jamBukaLibur,
+ jamTutupLibur: this.form.jamTutupLibur,
};
const cek = templateDokterForm.safeParse(formData);
@@ -567,9 +591,255 @@ const dokter = proxy({
},
});
+const templateTarifForm = z.object({
+ tarif: z.string().min(1, "Tarif tidak boleh kosong"),
+ layanan: z.string().min(1, "Layanan tidak boleh kosong"),
+});
+
+const defaultTarifForm = {
+ tarif: "",
+ layanan: "",
+};
+
+const tarif = proxy({
+ create: {
+ form: defaultTarifForm,
+ loading: false,
+ async create() {
+ const cek = templateTarifForm.safeParse(tarif.create.form);
+ if (!cek.success) {
+ const err = `[${cek.error.issues
+ .map((v) => `${v.path.join(".")}`)
+ .join("\n")}] required`;
+ toast.error(err);
+ return null;
+ }
+ try {
+ tarif.create.loading = true;
+ const res = await ApiFetch.api.kesehatan.tarifdanlayanan["create"].post(
+ tarif.create.form
+ );
+
+ if (res.status === 200) {
+ const id = res.data?.data;
+ if (id) {
+ toast.success("Sukses menambahkan");
+ tarif.create.form = { ...defaultTarifForm };
+ tarif.findMany.load();
+ return id;
+ }
+ }
+ toast.error("failed create");
+ return null;
+ } catch (error) {
+ console.log((error as Error).message);
+ return null;
+ } finally {
+ tarif.create.loading = false;
+ }
+ },
+ },
+ findMany: {
+ data: null as
+ | Prisma.TarifDanLayananGetPayload<{
+ omit: {
+ isActive: true;
+ };
+ }>[]
+ | null,
+ page: 1,
+ totalPages: 1,
+ loading: false,
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ tarif.findMany.loading = true; // ✅ Akses langsung via nama path
+ tarif.findMany.page = page;
+ tarif.findMany.search = search;
+
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.kesehatan.tarifdanlayanan[
+ "findMany"
+ ].get({ query });
+
+ if (res.status === 200 && res.data?.success) {
+ tarif.findMany.data = res.data.data ?? [];
+ tarif.findMany.totalPages = res.data.totalPages ?? 1;
+ } else {
+ tarif.findMany.data = [];
+ tarif.findMany.totalPages = 1;
+ }
+ } catch (err) {
+ console.error("Gagal fetch tarif dan layanan paginated:", err);
+ tarif.findMany.data = [];
+ tarif.findMany.totalPages = 1;
+ } finally {
+ tarif.findMany.loading = false;
+ }
+ },
+ },
+ findUnique: {
+ data: null as Prisma.TarifDanLayananGetPayload<{
+ omit: { isActive: true };
+ }> | null,
+ async load(id: string) {
+ try {
+ const res = await fetch(`/api/kesehatan/tarifdanlayanan/${id}`);
+ if (res.ok) {
+ const data = await res.json();
+ tarif.findUnique.data = data.data ?? null;
+ } else {
+ console.error(
+ "Failed to fetch tarif dan layanan",
+ res.statusText
+ );
+ tarif.findUnique.data = null;
+ }
+ } catch (error) {
+ console.error("Error fetching tarif dan layanan", error);
+ tarif.findUnique.data = null;
+ }
+ },
+ },
+ update: {
+ id: "",
+ form: { ...defaultTarifForm },
+ loading: false,
+ async load(id: string) {
+ if (!id) {
+ toast.warn("ID tidak valid");
+ return null;
+ }
+
+ try {
+ const response = await fetch(`/api/kesehatan/tarifdanlayanan/${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 = {
+ tarif: data.tarif,
+ layanan: data.layanan
+ };
+ return data; // Return the loaded data
+ } else {
+ throw new Error(result?.message || "Gagal memuat data");
+ }
+ } catch (error) {
+ console.error("Error loading tarif dan layanan:", error);
+ toast.error(
+ error instanceof Error ? error.message : "Gagal memuat data"
+ );
+ return null;
+ }
+ },
+ async submit() {
+ const id = this.id;
+ if (!id) {
+ toast.warn("ID tidak valid");
+ return null;
+ }
+
+ const formData = {
+ tarif: this.form.tarif,
+ layanan: this.form.layanan
+ };
+
+ const cek = templateTarifForm.safeParse(formData);
+ if (!cek.success) {
+ const err = `[${cek.error.issues
+ .map((v: any) => `${v.path.join(".")}`)
+ .join("\n")}] required`;
+ toast.error(err);
+ return null;
+ }
+
+ try {
+ this.loading = true;
+ const res = await fetch(`/api/kesehatan/tarifdanlayanan/${id}`, {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify(formData),
+ });
+
+ const result = await res.json();
+
+ if (!res.ok || !result?.success) {
+ throw new Error(result?.message || "Gagal update data");
+ }
+
+ toast.success("Berhasil update data!");
+ await tarif.findMany.load();
+ return result.data;
+ } catch (error) {
+ console.error("Update error:", error);
+ toast.error("Gagal update data tarif dan layanan");
+ throw error;
+ } finally {
+ this.loading = false;
+ }
+ },
+ },
+ delete: {
+ loading: false,
+ async byId(id: string) {
+ if (!id) {
+ return toast.warn("ID tidak valid");
+ }
+ try {
+ tarif.delete.loading = true;
+
+ const response = await fetch(
+ `/api/kesehatan/tarifdanlayanan/del/${id}`,
+ {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ }
+ );
+
+ const result = await response.json();
+
+ if (response.ok && result?.success) {
+ toast.success(
+ result.message || "tarif dan layanan berhasil dihapus"
+ );
+ await tarif.findMany.load(); // refresh list
+ } else {
+ toast.error(
+ result?.message || "Gagal menghapus tarif dan layanan"
+ );
+ }
+ } catch (error) {
+ console.error("Gagal delete:", error);
+ toast.error("Terjadi kesalahan saat menghapus tarif dan layanan");
+ } finally {
+ tarif.delete.loading = false;
+ }
+ },
+ },
+});
+
const fasilitasKesehatanState = proxy({
fasilitasKesehatan,
dokter,
+ tarif
});
export default fasilitasKesehatanState;
diff --git a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/grafikKepuasan.ts b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/grafikKepuasan.ts
index 19f7e80a..15b7b4b8 100644
--- a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/grafikKepuasan.ts
+++ b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/grafikKepuasan.ts
@@ -43,7 +43,7 @@ const grafikkepuasan = proxy({
if (res.status === 200) {
const id = res.data?.data;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
grafikkepuasan.create.form = { ...defaultForm };
grafikkepuasan.findMany.load();
return id;
diff --git a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/persentaseKelahiran.ts b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/persentaseKelahiran.ts
index 4da79df8..e00ab588 100644
--- a/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/persentaseKelahiran.ts
+++ b/src/app/admin/(dashboard)/_state/kesehatan/data_kesehatan_warga/persentaseKelahiran.ts
@@ -50,7 +50,7 @@ const persentasekelahiran = proxy({
if (res.status === 200) {
const id = res.data?.data;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
persentasekelahiran.create.form = { ...defaultForm };
persentasekelahiran.findMany.load();
return id;
diff --git a/src/app/admin/(dashboard)/_state/landing-page/apbdes.ts b/src/app/admin/(dashboard)/_state/landing-page/apbdes.ts
index c780b1a9..9b5cb438 100644
--- a/src/app/admin/(dashboard)/_state/landing-page/apbdes.ts
+++ b/src/app/admin/(dashboard)/_state/landing-page/apbdes.ts
@@ -5,58 +5,115 @@ import { toast } from "react-toastify";
import { proxy } from "valtio";
import { z } from "zod";
-const templateapbDesaForm = z.object({
- name: z.string().min(1, "Judul minimal 1 karakter"),
- jumlah: z.string().min(1, "Deskripsi minimal 1 karakter"),
- imageId: z.string().min(1, "File minimal 1"),
- fileId: z.string().min(1, "File minimal 1"),
+// --- Zod Schema ---
+const ApbdesItemSchema = z.object({
+ kode: z.string().min(1, "Kode wajib diisi"),
+ uraian: z.string().min(1, "Uraian wajib diisi"),
+ anggaran: z.number().min(0),
+ realisasi: z.number().min(0),
+ selisih: z.number(),
+ persentase: z.number(),
+ level: z.number().int().min(1).max(3),
+ tipe: z.enum(['pendapatan', 'belanja', 'pembiayaan']).nullable().optional(),
});
-const defaultapbdesForm = {
- name: "",
- jumlah: "",
+const ApbdesFormSchema = z.object({
+ tahun: z.number().int().min(2000, "Tahun tidak valid"),
+ imageId: z.string().min(1, "Gambar wajib diunggah"),
+ fileId: z.string().min(1, "File wajib diunggah"),
+ items: z.array(ApbdesItemSchema).min(1, "Minimal ada 1 item"),
+});
+
+// --- Default Form ---
+const defaultApbdesForm = {
+ tahun: new Date().getFullYear(),
imageId: "",
fileId: "",
+ items: [] as z.infer[],
};
+// --- Helper: hitung selisih & persentase otomatis (opsional di frontend) ---
+// --- Helper: hitung selisih & persentase otomatis (opsional di frontend) ---
+function normalizeItem(item: Partial>): z.infer {
+ const anggaran = item.anggaran ?? 0;
+ const realisasi = item.realisasi ?? 0;
+
+
+ // ✅ Formula yang benar
+ const selisih = realisasi - anggaran; // positif = sisa anggaran, negatif = over budget
+ const persentase = anggaran > 0 ? (realisasi / anggaran) * 100 : 0; // persentase realisasi terhadap anggaran
+
+ return {
+ kode: item.kode || "",
+ uraian: item.uraian || "",
+ anggaran,
+ realisasi,
+ selisih,
+ persentase,
+ level: item.level || 1,
+ tipe: item.tipe, // biarkan null jika memang null
+ };
+}
+
+// --- State Utama ---
const apbdes = proxy({
create: {
- form: { ...defaultapbdesForm },
+ form: { ...defaultApbdesForm },
loading: false,
- async create() {
- const cek = templateapbDesaForm.safeParse(apbdes.create.form);
- if (!cek.success) {
- const err = `[${cek.error.issues
- .map((v) => `${v.path.join(".")}`)
- .join("\n")}] required`;
- return toast.error(err);
- }
- try {
- apbdes.create.loading = true;
- const res = await ApiFetch.api.landingpage.apbdes["create"].post({
- ...apbdes.create.form,
- });
- if (res.status === 200) {
+ addItem(item: Partial>) {
+ const normalized = normalizeItem(item);
+ this.form.items.push(normalized);
+ },
+
+ removeItem(index: number) {
+ this.form.items.splice(index, 1);
+ },
+
+ updateItem(index: number, updates: Partial>) {
+ const current = this.form.items[index];
+ if (current) {
+ const updated = normalizeItem({ ...current, ...updates });
+ this.form.items[index] = updated;
+ }
+ },
+
+ reset() {
+ this.form = { ...defaultApbdesForm };
+ },
+
+ async create() {
+ const parsed = ApbdesFormSchema.safeParse(this.form);
+ if (!parsed.success) {
+ const errors = parsed.error.issues.map((issue) => `${issue.path.join(".")} - ${issue.message}`);
+ toast.error(`Validasi gagal:\n${errors.join("\n")}`);
+ return;
+ }
+
+ try {
+ this.loading = true;
+ const res = await ApiFetch.api.landingpage.apbdes["create"].post(parsed.data);
+
+ if (res.data?.success) {
+ toast.success("APBDes berhasil dibuat");
apbdes.findMany.load();
- return toast.success("Data berhasil ditambahkan");
+ this.reset();
+ } else {
+ toast.error(res.data?.message || "Gagal membuat APBDes");
}
- return toast.error("Gagal menambahkan data");
- } catch (error) {
- console.log(error);
- toast.error("Gagal menambahkan data");
+ } catch (error: any) {
+ console.error("Create APBDes error:", error);
+ toast.error(error?.message || "Terjadi kesalahan saat membuat APBDes");
} finally {
- apbdes.create.loading = false;
+ this.loading = false;
}
},
},
+
findMany: {
data: null as
| Prisma.APBDesGetPayload<{
- include: {
- image: true;
- file: true;
- };
+ include: { image: true; file: true; items: true };
}>[]
| null,
page: 1,
@@ -64,194 +121,202 @@ const apbdes = proxy({
total: 0,
loading: false,
search: "",
- load: async (page = 1, limit = 10, search = "") => { // Change to arrow function
- apbdes.findMany.loading = true; // Use the full path to access the property
+
+ load: async (page = 1, limit = 10, search = "") => {
+ apbdes.findMany.loading = true;
apbdes.findMany.page = page;
apbdes.findMany.search = search;
+
try {
- const query: any = { page, limit };
+ const query: Record = { page: String(page), limit: String(limit) };
if (search) query.search = search;
-
- const res = await ApiFetch.api.landingpage.apbdes[
- "findMany"
- ].get({
- query
- });
-
- if (res.status === 200 && res.data?.success) {
+
+ const res = await ApiFetch.api.landingpage.apbdes["findMany"].get({ query });
+
+ if (res.data?.success) {
apbdes.findMany.data = res.data.data || [];
- apbdes.findMany.total = res.data.total || 0;
- apbdes.findMany.totalPages = res.data.totalPages || 1;
+ apbdes.findMany.total = res.data.meta?.total || 0;
+ apbdes.findMany.totalPages = res.data.meta?.totalPages || 1;
} else {
- console.error("Failed to load pegawai:", res.data?.message);
apbdes.findMany.data = [];
apbdes.findMany.total = 0;
apbdes.findMany.totalPages = 1;
+ toast.error(res.data?.message || "Gagal memuat data");
}
} catch (error) {
- console.error("Error loading pegawai:", error);
+ console.error("FindMany error:", error);
apbdes.findMany.data = [];
apbdes.findMany.total = 0;
apbdes.findMany.totalPages = 1;
+ toast.error("Gagal memuat daftar APBDes");
} finally {
apbdes.findMany.loading = false;
}
},
},
+
findUnique: {
- data: null as Prisma.APBDesGetPayload<{
- include: {
- image: true;
- file: true;
- };
- }> | null,
+ data: null as
+ | Prisma.APBDesGetPayload<{
+ include: { image: true; file: true; items: true };
+ }>
+ | null,
+ loading: false,
+ error: null as string | null,
+
async load(id: string) {
+ if (!id || id.trim() === '') {
+ this.data = null;
+ this.error = "ID tidak valid";
+ return;
+ }
+
+ this.loading = true;
+ this.error = null;
+
try {
- const res = await fetch(`/api/landingpage/apbdes/${id}`);
- if (res.ok) {
- const data = await res.json();
- apbdes.findUnique.data = data.data ?? null;
+ // Pastikan URL-nya benar
+ const url = `/api/landingpage/apbdes/${id}`;
+ console.log("🌐 Fetching:", url);
+
+ // Gunakan fetch biasa atau ApiFetch dengan cara yang benar
+ const response = await fetch(url);
+ const res = await response.json();
+
+ console.log("📦 Response:", res);
+
+ if (res.success && res.data) {
+ this.data = res.data;
} else {
- console.error("Failed to fetch data", res.status, res.statusText);
- apbdes.findUnique.data = null;
+ this.data = null;
+ this.error = res.message || "Gagal memuat detail APBDes";
+ toast.error(this.error);
}
} catch (error) {
- console.error("Error fetching data:", error);
- apbdes.findUnique.data = null;
+ console.error("❌ FindUnique error:", error);
+ this.data = null;
+ this.error = "Gagal memuat detail APBDes";
+ toast.error(this.error);
+ } finally {
+ this.loading = false;
}
- },
+ }
},
+
delete: {
loading: false,
async byId(id: string) {
if (!id) return toast.warn("ID tidak valid");
try {
- apbdes.delete.loading = true;
+ this.loading = true;
+ const res = await (ApiFetch.api.landingpage.apbdes as any)["del"][id].delete();
- const response = await fetch(`/api/landingpage/apbdes/del/${id}`, {
- method: "DELETE",
- headers: {
- "Content-Type": "application/json",
- },
- });
-
- const result = await response.json();
-
- if (response.ok && result?.success) {
- toast.success(result.message || "apbdes berhasil dihapus");
- await apbdes.findMany.load(); // refresh list
+ if (res.data?.success) {
+ toast.success("APBDes berhasil dihapus");
+ apbdes.findMany.load();
} else {
- toast.error(result?.message || "Gagal menghapus apbdes");
+ toast.error(res.data?.message || "Gagal menghapus APBDes");
}
- } catch (error) {
- console.error("Gagal delete:", error);
- toast.error("Terjadi kesalahan saat menghapus apbdes");
+ } catch (error: any) {
+ console.error("Delete error:", error);
+ toast.error(error?.message || "Terjadi kesalahan saat menghapus");
} finally {
- apbdes.delete.loading = false;
+ this.loading = false;
}
},
},
+
edit: {
id: "",
- form: { ...defaultapbdesForm },
+ form: { ...defaultApbdesForm },
loading: false,
async load(id: string) {
- if (!id) {
- toast.warn("ID tidak valid");
- return null;
- }
+ if (!id) return toast.warn("ID tidak valid");
try {
- apbdes.edit.loading = true;
+ this.loading = true;
+ const res = await (ApiFetch.api.landingpage.apbdes as any)[id].get();
- const response = await fetch(`/api/landingpage/apbdes/${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;
+ if (res.data?.success) {
+ const data = res.data.data;
this.id = data.id;
this.form = {
- name: data.name,
- jumlah: data.jumlah,
- imageId: data.imageId,
- fileId: data.fileId,
+ tahun: data.tahun || new Date().getFullYear(),
+ imageId: data.imageId || "",
+ fileId: data.fileId || "",
+ items: (data.items || []).map((item: any) => ({
+ kode: item.kode,
+ uraian: item.uraian,
+ anggaran: item.anggaran,
+ realisasi: item.realisasi,
+ selisih: item.selisih,
+ persentase: item.persentase,
+ level: item.level,
+ tipe: item.tipe || 'pendapatan',
+ })),
};
return data;
} else {
- throw new Error(result?.message || "Gagal memuat data");
+ throw new Error(res.data?.message || "Gagal memuat data");
}
- } catch (error) {
- console.error("Error loading apbdes:", error);
- toast.error(
- error instanceof Error ? error.message : "Gagal memuat data"
- );
- return null;
+ } catch (error: any) {
+ console.error("Edit load error:", error);
+ toast.error(error.message || "Gagal memuat data untuk diedit");
} finally {
- apbdes.edit.loading = false;
+ this.loading = false;
}
},
async update() {
- const cek = templateapbDesaForm.safeParse(apbdes.edit.form);
- if (!cek.success) {
- const err = `[${cek.error.issues
- .map((v) => `${v.path.join(".")}`)
- .join("\n")}] required`;
- return toast.error(err);
+ const parsed = ApbdesFormSchema.safeParse(this.form);
+ if (!parsed.success) {
+ const errors = parsed.error.issues.map((issue) => `${issue.path.join(".")} - ${issue.message}`);
+ toast.error(`Validasi gagal:\n${errors.join("\n")}`);
+ return false;
}
try {
- apbdes.edit.loading = true;
- const response = await fetch(`/api/landingpage/apbdes/${this.id}`, {
- method: "PUT",
- headers: {
- "Content-Type": "application/json",
- },
- body: JSON.stringify({
- name: this.form.name,
- jumlah: this.form.jumlah,
- imageId: this.form.imageId,
- fileId: this.form.fileId,
- }),
- });
- 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 apbdes");
- await apbdes.findMany.load(); // refresh list
+ this.loading = true;
+ // Include the ID in the request body
+ const requestData = {
+ ...parsed.data,
+ id: this.id, // Add the ID to the request body
+ };
+
+ const res = await (ApiFetch.api.landingpage.apbdes as any)[this.id].put(requestData);
+
+ if (res.data?.success) {
+ toast.success("APBDes berhasil diperbarui");
+ apbdes.findMany.load();
return true;
} else {
- throw new Error(result.message || "Gagal mengupdate apbdes");
+ throw new Error(res.data?.message || "Gagal memperbarui APBDes");
}
- } catch (error) {
- console.error("Error updating apbdes:", error);
- toast.error(
- error instanceof Error ? error.message : "Gagal mengupdate apbdes"
- );
+ } catch (error: any) {
+ console.error("Update error:", error);
+ toast.error(error.message || "Gagal memperbarui APBDes");
return false;
} finally {
- apbdes.edit.loading = false;
+ this.loading = false;
}
},
+
+ addItem(item: Partial>) {
+ const normalized = normalizeItem(item);
+ this.form.items.push(normalized);
+ },
+
+ removeItem(index: number) {
+ this.form.items.splice(index, 1);
+ },
+
reset() {
- apbdes.edit.id = "";
- apbdes.edit.form = { ...defaultapbdesForm };
+ this.id = "";
+ this.form = { ...defaultApbdesForm };
},
},
});
-export default apbdes;
+export default apbdes;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/_state/landing-page/indeks-kepuasan.ts b/src/app/admin/(dashboard)/_state/landing-page/indeks-kepuasan.ts
index 48d707b7..9caeb019 100644
--- a/src/app/admin/(dashboard)/_state/landing-page/indeks-kepuasan.ts
+++ b/src/app/admin/(dashboard)/_state/landing-page/indeks-kepuasan.ts
@@ -60,13 +60,18 @@ const responden = proxy({
totalPages: 1,
total: 0,
loading: false,
- load: async (page = 1, limit = 10) => {
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
// Change to arrow function
responden.findMany.loading = true; // Use the full path to access the property
responden.findMany.page = page;
+ responden.findMany.search = search;
try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
const res = await ApiFetch.api.landingpage.responden["findMany"].get({
- query: { page, limit },
+ query,
});
if (res.status === 200 && res.data?.success) {
diff --git a/src/app/admin/(dashboard)/_state/landing-page/profile.ts b/src/app/admin/(dashboard)/_state/landing-page/profile.ts
index be6b2e7f..152f16d7 100644
--- a/src/app/admin/(dashboard)/_state/landing-page/profile.ts
+++ b/src/app/admin/(dashboard)/_state/landing-page/profile.ts
@@ -27,7 +27,7 @@ const programInovasi = proxy({
name: "",
description: "",
imageId: "",
- link: ""
+ link: "",
} as ProgramInovasiForm,
loading: false,
async create() {
@@ -53,12 +53,17 @@ const programInovasi = proxy({
].post(formData);
if (res.status === 200) {
programInovasi.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
+ }
+ if (process.env.NODE_ENV === 'development') {
+ console.log(res);
}
- console.log(res);
return toast.error("failed create");
} catch (error) {
- console.log((error as Error).message);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Create error:", error);
+ }
+ toast.error("Gagal menambahkan data");
} finally {
programInovasi.create.loading = false;
}
@@ -71,32 +76,37 @@ const programInovasi = proxy({
total: 0,
loading: false,
search: "",
- load: async (page = 1, limit = 10, search = "") => { // Change to arrow function
- programInovasi.findMany.loading = true; // Use the full path to access the property
+ load: async (page = 1, limit = 10, search = "") => {
+ // Change to arrow function
+ programInovasi.findMany.loading = true; // Use the full path to access the property
programInovasi.findMany.page = page;
programInovasi.findMany.search = search;
try {
const query: any = { page, limit };
if (search) query.search = search;
-
+
const res = await ApiFetch.api.landingpage.programinovasi[
"findMany"
].get({
- query
+ query,
});
-
+
if (res.status === 200 && res.data?.success) {
programInovasi.findMany.data = res.data.data || [];
programInovasi.findMany.total = res.data.total || 0;
programInovasi.findMany.totalPages = res.data.totalPages || 1;
} else {
- console.error("Failed to load pegawai:", res.data?.message);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Failed to load pegawai:", res.data?.message);
+ }
programInovasi.findMany.data = [];
programInovasi.findMany.total = 0;
programInovasi.findMany.totalPages = 1;
}
} catch (error) {
- console.error("Error loading pegawai:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error loading pegawai:", error);
+ }
programInovasi.findMany.data = [];
programInovasi.findMany.total = 0;
programInovasi.findMany.totalPages = 1;
@@ -111,19 +121,25 @@ const programInovasi = proxy({
image: true;
};
}> | null,
+ loading: false,
async load(id: string) {
try {
- const res = await fetch(`/api/landingpage/programinovasi/${id}`);
- if (res.ok) {
- const data = await res.json();
- programInovasi.findUnique.data = data.data ?? null;
+ programInovasi.findUnique.loading = true;
+ const res = await (ApiFetch.api.landingpage.programinovasi as any)[id].get();
+ if (res.data?.success) {
+ programInovasi.findUnique.data = res.data.data ?? null;
+ return res.data.data;
} else {
- console.error("Failed to fetch program inovasi:", res.statusText);
+ toast.error(res.data?.message || "Gagal memuat data program inovasi");
programInovasi.findUnique.data = null;
+ return null;
}
} catch (error) {
console.error("Error fetching program inovasi:", error);
programInovasi.findUnique.data = null;
+ return null;
+ } finally {
+ programInovasi.findUnique.loading = false;
}
},
},
@@ -134,27 +150,18 @@ const programInovasi = proxy({
try {
programInovasi.delete.loading = true;
+ const res = await (ApiFetch.api.landingpage.programinovasi as any)["del"][id].delete();
- const response = await fetch(
- `/api/landingpage/programinovasi/del/${id}`,
- {
- method: "DELETE",
- headers: {
- "Content-Type": "application/json",
- },
- }
- );
-
- const result = await response.json();
-
- if (response.ok && result?.success) {
- toast.success(result.message || "Program inovasi berhasil dihapus");
- await programInovasi.findMany.load(); // refresh list
+ if (res.data?.success) {
+ toast.success(res.data.message || "Program inovasi berhasil dihapus");
+ await programInovasi.findMany.load();
} else {
- toast.error(result?.message || "Gagal menghapus program inovasi");
+ toast.error(res.data?.message || "Gagal menghapus program inovasi");
}
} catch (error) {
- console.error("Gagal delete:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Gagal delete:", error);
+ }
toast.error("Terjadi kesalahan saat menghapus program inovasi");
} finally {
programInovasi.delete.loading = false;
@@ -173,20 +180,11 @@ const programInovasi = proxy({
}
try {
- const response = await fetch(`/api/landingpage/programinovasi/${id}`, {
- method: "GET",
- headers: {
- "Content-Type": "application/json",
- },
- });
- if (!response.ok) {
- throw new Error(`HTTP error! status: ${response.status}`);
- }
+ programInovasi.update.loading = true;
+ const res = await (ApiFetch.api.landingpage.programinovasi as any)[id].get();
- const result = await response.json();
-
- if (result?.success) {
- const data = result.data;
+ if (res.data?.success) {
+ const data = res.data.data;
this.id = data.id;
this.form = {
name: data.name,
@@ -196,13 +194,15 @@ const programInovasi = proxy({
};
return data;
} else {
- throw new Error(
- result?.message || "Gagal mengambil data program inovasi"
- );
+ toast.error(res.data?.message || "Gagal mengambil data program inovasi");
+ return null;
}
} catch (error) {
- console.error((error as Error).message);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error loading program inovasi:", error);
+ }
toast.error("Terjadi kesalahan saat mengambil data program inovasi");
+ return null;
} finally {
programInovasi.update.loading = false;
}
@@ -220,41 +220,25 @@ const programInovasi = proxy({
try {
programInovasi.update.loading = true;
+ const res = await (ApiFetch.api.landingpage.programinovasi as any)[this.id].put({
+ name: this.form.name,
+ description: this.form.description,
+ imageId: this.form.imageId,
+ link: this.form.link,
+ });
- const response = await fetch(
- `/api/landingpage/programinovasi/${this.id}`,
- {
- method: "PUT",
- headers: {
- "Content-Type": "application/json",
- },
- body: JSON.stringify({
- name: this.form.name,
- description: this.form.description,
- imageId: this.form.imageId,
- link: this.form.link,
- }),
- }
- );
-
- 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) {
+ if (res.data?.success) {
toast.success("Berhasil update program inovasi");
- await programInovasi.findMany.load(); // refresh list
+ await programInovasi.findMany.load();
return true;
} else {
- throw new Error(result.message || "Gagal update program inovasi");
+ toast.error(res.data?.message || "Gagal update program inovasi");
+ return false;
}
} catch (error) {
- console.error("Error updating program inovasi:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error updating program inovasi:", error);
+ }
toast.error(
error instanceof Error
? error.message
@@ -389,7 +373,10 @@ const pejabatDesa = proxy({
try {
// Ensure ID is properly encoded in the URL
- const url = new URL(`/api/landingpage/pejabatdesa/${encodeURIComponent(this.id)}`, window.location.origin);
+ const url = new URL(
+ `/api/landingpage/pejabatdesa/${encodeURIComponent(this.id)}`,
+ window.location.origin
+ );
const response = await fetch(url.toString(), {
method: "PUT",
headers: {
@@ -438,16 +425,19 @@ const pejabatDesa = proxy({
const templateMediaSosial = z.object({
name: z.string().min(3, "Nama minimal 3 karakter"),
- imageId: z.string().min(1, "Gambar wajib dipilih"),
- iconUrl: z.string().min(3, "Icon URL minimal 3 karakter"),
+ imageId: z.string().nullable().optional(),
+ iconUrl: z.string().optional(), // ✅ Optional - tidak selalu required
+ icon: z.string().nullable().optional(),
});
type MediaSosialForm = {
name: string;
- imageId: string;
+ imageId: string | null; // boleh null
iconUrl: string;
+ icon: string | null; // boleh null
};
+
const mediaSosial = proxy({
create: {
form: {} as MediaSosialForm,
@@ -455,9 +445,10 @@ const mediaSosial = proxy({
async create() {
// Ensure all required fields are non-null
const formData = {
- name: mediaSosial.create.form.name || "",
- imageId: mediaSosial.create.form.imageId || "",
- iconUrl: mediaSosial.create.form.iconUrl || "",
+ name: mediaSosial.create.form.name ?? "",
+ imageId: mediaSosial.create.form.imageId ?? null, // FIXED
+ iconUrl: mediaSosial.create.form.iconUrl ?? "",
+ icon: mediaSosial.create.form.icon ?? null, // FIXED
};
const cek = templateMediaSosial.safeParse(formData);
@@ -474,12 +465,17 @@ const mediaSosial = proxy({
);
if (res.status === 200) {
mediaSosial.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
+ }
+ if (process.env.NODE_ENV === 'development') {
+ console.log(res);
}
- console.log(res);
return toast.error("failed create");
} catch (error) {
- console.log((error as Error).message);
+ if (process.env.NODE_ENV === 'development') {
+ console.log((error as Error).message);
+ }
+ toast.error("Gagal menambahkan data");
} finally {
mediaSosial.create.loading = false;
}
@@ -492,32 +488,35 @@ const mediaSosial = proxy({
total: 0,
loading: false,
search: "",
- load: async (page = 1, limit = 10, search = "") => { // Change to arrow function
- mediaSosial.findMany.loading = true; // Use the full path to access the property
+ load: async (page = 1, limit = 10, search = "") => {
+ // Change to arrow function
+ mediaSosial.findMany.loading = true; // Use the full path to access the property
mediaSosial.findMany.page = page;
mediaSosial.findMany.search = search;
- try {
+ try {
const query: any = { page, limit };
if (search) query.search = search;
-
- const res = await ApiFetch.api.landingpage.mediasosial[
- "findMany"
- ].get({
+
+ const res = await ApiFetch.api.landingpage.mediasosial["findMany"].get({
query,
});
-
+
if (res.status === 200 && res.data?.success) {
mediaSosial.findMany.data = res.data.data || [];
mediaSosial.findMany.total = res.data.total || 0;
mediaSosial.findMany.totalPages = res.data.totalPages || 1;
} else {
- console.error("Failed to load media sosial:", res.data?.message);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Failed to load media sosial:", res.data?.message);
+ }
mediaSosial.findMany.data = [];
mediaSosial.findMany.total = 0;
mediaSosial.findMany.totalPages = 1;
}
} catch (error) {
- console.error("Error loading media sosial:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error loading media sosial:", error);
+ }
mediaSosial.findMany.data = [];
mediaSosial.findMany.total = 0;
mediaSosial.findMany.totalPages = 1;
@@ -532,25 +531,32 @@ const mediaSosial = proxy({
image: true;
};
}> | null,
+ loading: false,
async load(id: string) {
if (!id) {
toast.warn("ID tidak valid");
return null;
}
-
- mediaSosial.update.loading = true;
+
+ mediaSosial.findUnique.loading = true;
try {
- const res = await fetch(`/api/landingpage/mediasosial/${id}`);
- if (res.ok) {
- const data = await res.json();
- mediaSosial.findUnique.data = data.data ?? null;
+ const res = await (ApiFetch.api.landingpage.mediasosial as any)[id].get();
+ if (res.data?.success) {
+ mediaSosial.findUnique.data = res.data.data ?? null;
+ return res.data.data;
} else {
- console.error("Failed to fetch media sosial:", res.statusText);
+ toast.error(res.data?.message || "Gagal memuat data media sosial");
mediaSosial.findUnique.data = null;
+ return null;
}
} catch (error) {
- console.error("Error fetching media sosial:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error fetching media sosial:", error);
+ }
mediaSosial.findUnique.data = null;
+ return null;
+ } finally {
+ mediaSosial.findUnique.loading = false;
}
},
},
@@ -561,24 +567,18 @@ const mediaSosial = proxy({
try {
mediaSosial.delete.loading = true;
+ const res = await (ApiFetch.api.landingpage.mediasosial as any)["del"][id].delete();
- const response = await fetch(`/api/landingpage/mediasosial/del/${id}`, {
- method: "DELETE",
- headers: {
- "Content-Type": "application/json",
- },
- });
-
- const result = await response.json();
-
- if (response.ok && result?.success) {
- toast.success(result.message || "Media Sosial berhasil dihapus");
- await mediaSosial.findMany.load(); // refresh list
+ if (res.data?.success) {
+ toast.success(res.data.message || "Media Sosial berhasil dihapus");
+ await mediaSosial.findMany.load();
} else {
- toast.error(result?.message || "Gagal menghapus media sosial");
+ toast.error(res.data?.message || "Gagal menghapus media sosial");
}
} catch (error) {
- console.error("Gagal delete:", error);
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Gagal delete:", error);
+ }
toast.error("Terjadi kesalahan saat menghapus media sosial");
} finally {
mediaSosial.delete.loading = false;
@@ -586,106 +586,87 @@ const mediaSosial = proxy({
},
},
update: {
- id: "",
- form: {} as MediaSosialForm,
- loading: false,
-
- async load(id: string) {
- if (!id) {
- toast.warn("ID tidak valid");
+ id: "",
+ form: {} as MediaSosialForm,
+ loading: false,
+
+ async load(id: string) {
+ if (!id) {
+ toast.warn("ID tidak valid");
+ return null;
+ }
+
+ mediaSosial.update.loading = true;
+ try {
+ const res = await (ApiFetch.api.landingpage.mediasosial as any)[id].get();
+
+ if (res.data?.success) {
+ const data = res.data.data;
+ this.id = data.id;
+ this.form = {
+ name: data.name || "",
+ imageId: data.imageId || null,
+ iconUrl: data.iconUrl || "",
+ icon: data.icon || null,
+ };
+ return data;
+ } else {
+ toast.error(res.data?.message || "Gagal mengambil data media sosial");
return null;
}
-
- mediaSosial.update.loading = true; // ✅ Tambahkan ini di awal
-
- try {
- const response = await fetch(`/api/landingpage/mediasosial/${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 = {
- name: data.name || "",
- imageId: data.imageId || "",
- iconUrl: data.iconUrl || "",
- };
- return data;
- } else {
- throw new Error(result?.message || "Gagal mengambil data media sosial");
- }
- } catch (error) {
- console.error((error as Error).message);
- toast.error("Terjadi kesalahan saat mengambil data media sosial");
- } finally {
- mediaSosial.update.loading = false; // ✅ Supaya berhenti loading walau error
+ } catch (error) {
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error loading media sosial:", error);
}
- },
-
- async update() {
- const cek = templateMediaSosial.safeParse(mediaSosial.update.form);
- if (!cek.success) {
- const err = `[${cek.error.issues
- .map((v) => `${v.path.join(".")}`)
- .join("\n")}] required`;
- toast.error(err);
- return false;
- }
-
- try {
- mediaSosial.update.loading = true;
-
- const response = await fetch(`/api/landingpage/mediasosial/${this.id}`, {
- method: "PUT",
- headers: {
- "Content-Type": "application/json",
- },
- body: JSON.stringify({
- name: this.form.name,
- imageId: this.form.imageId,
- iconUrl: this.form.iconUrl,
- }),
- });
-
- 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 media sosial");
- await mediaSosial.findMany.load(); // refresh list
- return true;
- } else {
- throw new Error(result.message || "Gagal update media sosial");
- }
- } catch (error) {
- console.error("Error updating media sosial:", error);
- toast.error(
- error instanceof Error
- ? error.message
- : "Terjadi kesalahan saat update media sosial"
- );
- return false;
- } finally {
- mediaSosial.update.loading = false;
- }
- },
+ toast.error("Terjadi kesalahan saat mengambil data media sosial");
+ return null;
+ } finally {
+ mediaSosial.update.loading = false;
+ }
},
+
+ async update() {
+ const cek = templateMediaSosial.safeParse(mediaSosial.update.form);
+ if (!cek.success) {
+ const err = `[${cek.error.issues
+ .map((v) => `${v.path.join(".")}`)
+ .join("\n")}] required`;
+ toast.error(err);
+ return false;
+ }
+
+ try {
+ mediaSosial.update.loading = true;
+ const res = await (ApiFetch.api.landingpage.mediasosial as any)[this.id].put({
+ name: this.form.name,
+ imageId: this.form.imageId,
+ iconUrl: this.form.iconUrl,
+ icon: this.form.icon,
+ });
+
+ if (res.data?.success) {
+ toast.success("Berhasil update media sosial");
+ await mediaSosial.findMany.load();
+ return true;
+ } else {
+ toast.error(res.data?.message || "Gagal update media sosial");
+ return false;
+ }
+ } catch (error) {
+ if (process.env.NODE_ENV === 'development') {
+ console.error("Error updating media sosial:", error);
+ }
+ toast.error(
+ error instanceof Error
+ ? error.message
+ : "Terjadi kesalahan saat update media sosial"
+ );
+ return false;
+ } finally {
+ mediaSosial.update.loading = false;
+ }
+ },
+ },
});
const profileLandingPageState = proxy({
diff --git a/src/app/admin/(dashboard)/_state/landing-page/sdgs-desa.ts b/src/app/admin/(dashboard)/_state/landing-page/sdgs-desa.ts
index 51686ee5..832d6457 100644
--- a/src/app/admin/(dashboard)/_state/landing-page/sdgs-desa.ts
+++ b/src/app/admin/(dashboard)/_state/landing-page/sdgs-desa.ts
@@ -93,6 +93,34 @@ const sdgsDesa = proxy({
}
},
},
+ findManyAll: {
+ data: null as any[] | null,
+ loading: false,
+ load: async () => { // Change to arrow function
+ sdgsDesa.findManyAll.loading = true; // Use the full path to access the property
+ try {
+ const query: any = {};
+
+ const res = await ApiFetch.api.landingpage.sdgsdesa[
+ "findManyAll"
+ ].get({
+ query,
+ });
+
+ if (res.status === 200 && res.data?.success) {
+ sdgsDesa.findManyAll.data = res.data.data || [];
+ } else {
+ console.error("Failed to load media sosial:", res.data?.message);
+ sdgsDesa.findManyAll.data = [];
+ }
+ } catch (error) {
+ console.error("Error loading media sosial:", error);
+ sdgsDesa.findManyAll.data = [];
+ } finally {
+ sdgsDesa.findManyAll.loading = false;
+ }
+ },
+ },
findUnique: {
data: null as Prisma.SdgsDesaGetPayload<{
include: {
diff --git a/src/app/admin/(dashboard)/_state/lingkungan/data-lingkungan-desa.ts b/src/app/admin/(dashboard)/_state/lingkungan/data-lingkungan-desa.ts
index f7092245..b1bf9143 100644
--- a/src/app/admin/(dashboard)/_state/lingkungan/data-lingkungan-desa.ts
+++ b/src/app/admin/(dashboard)/_state/lingkungan/data-lingkungan-desa.ts
@@ -39,7 +39,7 @@ const dataLingkunganDesaState = proxy({
);
if (res.status === 200) {
dataLingkunganDesaState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/lingkungan/pengelolaan-sampah.ts b/src/app/admin/(dashboard)/_state/lingkungan/pengelolaan-sampah.ts
index 584bad91..80750349 100644
--- a/src/app/admin/(dashboard)/_state/lingkungan/pengelolaan-sampah.ts
+++ b/src/app/admin/(dashboard)/_state/lingkungan/pengelolaan-sampah.ts
@@ -35,7 +35,7 @@ const pengelolaanSampah = proxy({
].post(pengelolaanSampah.create.form);
if (res.status === 200) {
pengelolaanSampah.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/lingkungan/program-penghijauan.ts b/src/app/admin/(dashboard)/_state/lingkungan/program-penghijauan.ts
index 84d3083e..982bdadb 100644
--- a/src/app/admin/(dashboard)/_state/lingkungan/program-penghijauan.ts
+++ b/src/app/admin/(dashboard)/_state/lingkungan/program-penghijauan.ts
@@ -39,7 +39,7 @@ const programPenghijauanState = proxy({
);
if (res.status === 200) {
programPenghijauanState.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
console.log(res);
return toast.error("failed create");
diff --git a/src/app/admin/(dashboard)/_state/pendidikan/beasiswa-desa.ts b/src/app/admin/(dashboard)/_state/pendidikan/beasiswa-desa.ts
index 7337d64c..73d1dc94 100644
--- a/src/app/admin/(dashboard)/_state/pendidikan/beasiswa-desa.ts
+++ b/src/app/admin/(dashboard)/_state/pendidikan/beasiswa-desa.ts
@@ -9,34 +9,32 @@ import { z } from "zod";
const templateBeasiswaPendaftar = z.object({
namaLengkap: z.string().min(1, "Nama harus diisi"),
- nik: z.string().min(1, "NIK harus diisi"),
+ nis: z.string().min(1, "NIS harus diisi"),
+ kelas: z.string().min(1, "Kelas harus diisi"),
+ jenisKelamin: z.string().min(1, "Jenis kelamin harus diisi"),
+ alamatDomisili: z.string().min(1, "Alamat domisili harus diisi"),
tempatLahir: z.string().min(1, "Tempat lahir harus diisi"),
tanggalLahir: z.string().min(1, "Tanggal lahir harus diisi"),
- jenisKelamin: z.string().min(1, "Jenis kelamin harus diisi"),
- kewarganegaraan: z.string().min(1, "Kewarganegaraan harus diisi"),
- agama: z.string().min(1, "Agama harus diisi"),
- alamatKTP: z.string().min(1, "Alamat KTP harus diisi"),
- alamatDomisili: z.string().min(1, "Alamat domisili harus diisi"),
+ namaOrtu: z.string().min(1, "Nama ortu harus diisi"),
+ nik: z.string().min(1, "NIK harus diisi"),
+ pekerjaanOrtu: z.string().min(1, "Pekerjaan ortu harus diisi"),
+ penghasilan: z.string().min(1, "Penghasilan ortu harus diisi"),
noHp: z.string().min(1, "No HP harus diisi"),
- email: z.string().min(1, "Email harus diisi"),
- statusPernikahan: z.string().min(1, "Status pernikahan harus diisi"),
- ukuranBaju: z.string().min(1, "Ukuran baju harus diisi"),
});
const defaultBeasiswaPendaftar = {
namaLengkap: "",
- nik: "",
+ nis: "",
+ kelas: "",
+ jenisKelamin: "",
+ alamatDomisili: "",
tempatLahir: "",
tanggalLahir: "",
- jenisKelamin: "",
- kewarganegaraan: "",
- agama: "",
- alamatKTP: "",
- alamatDomisili: "",
+ namaOrtu: "",
+ nik: "",
+ pekerjaanOrtu: "",
+ penghasilan: "",
noHp: "",
- email: "",
- statusPernikahan: "",
- ukuranBaju: "",
};
const beasiswaPendaftar = proxy({
@@ -200,18 +198,17 @@ const beasiswaPendaftar = proxy({
this.id = data.id;
this.form = {
namaLengkap: data.namaLengkap,
- nik: data.nik,
+ nis: data.nis,
+ kelas: data.kelas,
+ jenisKelamin: data.jenisKelamin,
+ alamatDomisili: data.alamatDomisili,
tempatLahir: data.tempatLahir,
tanggalLahir: data.tanggalLahir,
- jenisKelamin: data.jenisKelamin,
- kewarganegaraan: data.kewarganegaraan,
- agama: data.agama,
- alamatKTP: data.alamatKTP,
- alamatDomisili: data.alamatDomisili,
+ namaOrtu: data.namaOrtu,
+ nik: data.nik,
+ pekerjaanOrtu: data.pekerjaanOrtu,
+ penghasilan: data.penghasilan,
noHp: data.noHp,
- email: data.email,
- statusPernikahan: data.statusPernikahan,
- ukuranBaju: data.ukuranBaju,
};
return data; // Return the loaded data
} else {
@@ -249,17 +246,17 @@ const beasiswaPendaftar = proxy({
},
body: JSON.stringify({
namaLengkap: this.form.namaLengkap,
- nik: this.form.nik,
- tanggalLahir: this.form.tanggalLahir,
+ nis: this.form.nis,
+ kelas: this.form.kelas,
jenisKelamin: this.form.jenisKelamin,
- kewarganegaraan: this.form.kewarganegaraan,
- agama: this.form.agama,
- alamatKTP: this.form.alamatKTP,
alamatDomisili: this.form.alamatDomisili,
+ tempatLahir: this.form.tempatLahir,
+ tanggalLahir: this.form.tanggalLahir,
+ namaOrtu: this.form.namaOrtu,
+ nik: this.form.nik,
+ pekerjaanOrtu: this.form.pekerjaanOrtu,
+ penghasilan: this.form.penghasilan,
noHp: this.form.noHp,
- email: this.form.email,
- statusPernikahan: this.form.statusPernikahan,
- ukuranBaju: this.form.ukuranBaju,
}),
}
);
diff --git a/src/app/admin/(dashboard)/_state/pendidikan/data-pendidikan.ts b/src/app/admin/(dashboard)/_state/pendidikan/data-pendidikan.ts
index 08189a83..5e6d4cb1 100644
--- a/src/app/admin/(dashboard)/_state/pendidikan/data-pendidikan.ts
+++ b/src/app/admin/(dashboard)/_state/pendidikan/data-pendidikan.ts
@@ -1,3 +1,4 @@
+/* eslint-disable @typescript-eslint/no-explicit-any */
import ApiFetch from "@/lib/api-fetch";
import { Prisma } from "@prisma/client";
import { toast } from "react-toastify";
@@ -42,7 +43,7 @@ const dataPendidikan = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
dataPendidikan.create.form = {
name: "",
jumlah: "",
@@ -65,13 +66,46 @@ const dataPendidikan = proxy({
select: { id: true; name: true; jumlah: true };
}>[]
| null,
+ page: 1,
+ totalPages: 1,
+ total: 0,
loading: false,
- async load() {
- const res = await ApiFetch.api.pendidikan.datapendidikan[
- "findMany"
- ].get();
- if (res.status === 200) {
- dataPendidikan.findMany.data = res.data?.data ?? [];
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ // Change to arrow function
+ dataPendidikan.findMany.loading = true; // Use the full path to access the property
+ dataPendidikan.findMany.page = page;
+ dataPendidikan.findMany.search = search;
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.pendidikan.datapendidikan[
+ "findMany"
+ ].get({
+ query,
+ });
+
+ if (res.status === 200 && res.data?.success) {
+ dataPendidikan.findMany.data = res.data.data || [];
+ dataPendidikan.findMany.total = res.data.total || 0;
+ dataPendidikan.findMany.totalPages = res.data.totalPages || 1;
+ } else {
+ console.error(
+ "Failed to load data pendidikan:",
+ res.data?.message
+ );
+ dataPendidikan.findMany.data = [];
+ dataPendidikan.findMany.total = 0;
+ dataPendidikan.findMany.totalPages = 1;
+ }
+ } catch (error) {
+ console.error("Error loading data pendidikan:", error);
+ dataPendidikan.findMany.data = [];
+ dataPendidikan.findMany.total = 0;
+ dataPendidikan.findMany.totalPages = 1;
+ } finally {
+ dataPendidikan.findMany.loading = false;
}
},
},
diff --git a/src/app/admin/(dashboard)/_state/pendidikan/perpustakaan-digital.ts b/src/app/admin/(dashboard)/_state/pendidikan/perpustakaan-digital.ts
index 69234af2..101b33b9 100644
--- a/src/app/admin/(dashboard)/_state/pendidikan/perpustakaan-digital.ts
+++ b/src/app/admin/(dashboard)/_state/pendidikan/perpustakaan-digital.ts
@@ -400,6 +400,42 @@ const kategoriBuku = proxy({
}
},
},
+ findManyAll: {
+ data: [] as Prisma.KategoriBukuGetPayload<{
+ omit: {
+ isActive: true;
+ };
+ }>[],
+ loading: false,
+ search: "",
+ load: async (search = "") => {
+ // Change to arrow function
+ kategoriBuku.findManyAll.loading = true; // Use the full path to access the property
+ kategoriBuku.findManyAll.search = search;
+ try {
+ const query: any = { search };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.pendidikan.perpustakaandigital.kategoribuku[
+ "findManyAll"
+ ].get({
+ query,
+ });
+
+ if (res.status === 200 && res.data?.success) {
+ kategoriBuku.findManyAll.data = res.data.data || [];
+ } else {
+ console.error("Failed to load pegawai:", res.data?.message);
+ kategoriBuku.findManyAll.data = [];
+ }
+ } catch (error) {
+ console.error("Error loading pegawai:", error);
+ kategoriBuku.findManyAll.data = [];
+ } finally {
+ kategoriBuku.findManyAll.loading = false;
+ }
+ },
+ },
findUnique: {
data: null as Prisma.KategoriBukuGetPayload<{
omit: {
diff --git a/src/app/admin/(dashboard)/_state/ppid/daftar_informasi_publik/daftarInformasiPublik.ts b/src/app/admin/(dashboard)/_state/ppid/daftar_informasi_publik/daftarInformasiPublik.ts
index 61f0f23b..07e3b5be 100644
--- a/src/app/admin/(dashboard)/_state/ppid/daftar_informasi_publik/daftarInformasiPublik.ts
+++ b/src/app/admin/(dashboard)/_state/ppid/daftar_informasi_publik/daftarInformasiPublik.ts
@@ -38,7 +38,7 @@ const daftarInformasiPublik = proxy({
].post(daftarInformasiPublik.create.form);
if (res.status === 200) {
daftarInformasiPublik.findMany.load();
- return toast.success("success create");
+ return toast.success("Sukses menambahkan");
}
return toast.error("failed create");
} catch (error) {
diff --git a/src/app/admin/(dashboard)/_state/ppid/indeks_kepuasan_masyarakat/grafikBerdasarkanUmur.ts b/src/app/admin/(dashboard)/_state/ppid/indeks_kepuasan_masyarakat/grafikBerdasarkanUmur.ts
index 4eb184ad..ad74dc6b 100644
--- a/src/app/admin/(dashboard)/_state/ppid/indeks_kepuasan_masyarakat/grafikBerdasarkanUmur.ts
+++ b/src/app/admin/(dashboard)/_state/ppid/indeks_kepuasan_masyarakat/grafikBerdasarkanUmur.ts
@@ -41,7 +41,7 @@ const grafikBerdasarkanUmur = proxy({
if (res.status === 200) {
const id = res.data?.data?.id;
if (id) {
- toast.success("Success create");
+ toast.success("Sukses menambahkan");
grafikBerdasarkanUmur.create.form = {
remaja: "",
dewasa: "",
diff --git a/src/app/admin/(dashboard)/_state/ppid/permohonan_informasi_publik/permohonanInformasiPublik.ts b/src/app/admin/(dashboard)/_state/ppid/permohonan_informasi_publik/permohonanInformasiPublik.ts
index 56b734d9..4a4cd5dc 100644
--- a/src/app/admin/(dashboard)/_state/ppid/permohonan_informasi_publik/permohonanInformasiPublik.ts
+++ b/src/app/admin/(dashboard)/_state/ppid/permohonan_informasi_publik/permohonanInformasiPublik.ts
@@ -1,3 +1,4 @@
+/* eslint-disable @typescript-eslint/no-explicit-any */
import ApiFetch from "@/lib/api-fetch";
import { Prisma } from "@prisma/client";
import { toast } from "react-toastify";
@@ -6,145 +7,207 @@ import { z } from "zod";
const templateForm = z.object({
name: z.string().min(3, "Nama minimal 3 karakter"),
- nik: z.string().min(3, "NIK minimal 3 karakter"),
- notelp: z.string().min(3, "Nomor Telepon minimal 3 karakter"),
+ nik: z
+ .string()
+ .min(3, "NIK minimal 3 karakter")
+ .max(16, "NIK maksimal 16 angka"),
+ notelp: z
+ .string()
+ .min(3, "Nomor Telepon minimal 3 karakter")
+ .max(15, "Nomor Telepon maksimal 15 angka"),
alamat: z.string().min(3, "Alamat minimal 3 karakter"),
email: z.string().min(3, "Email minimal 3 karakter"),
jenisInformasiDimintaId: z.string().nonempty(),
caraMemperolehInformasiId: z.string().nonempty(),
caraMemperolehSalinanInformasiId: z.string().nonempty(),
-})
+});
const jenisInformasiDiminta = proxy({
- findMany: {
- data: null as
- | null
- | Prisma.JenisInformasiDimintaGetPayload<{ omit: { isActive: true } }>[],
- async load(){
- const res = await ApiFetch.api.ppid.permohonaninformasipublik.jenisInformasi["find-many"].get();
- if (res.status === 200) {
- jenisInformasiDiminta.findMany.data = res.data?.data ?? [];
- }
- }
- }
-})
+ findMany: {
+ data: null as
+ | null
+ | Prisma.JenisInformasiDimintaGetPayload<{ omit: { isActive: true } }>[],
+ async load() {
+ const res =
+ await ApiFetch.api.ppid.permohonaninformasipublik.jenisInformasi[
+ "find-many"
+ ].get();
+ if (res.status === 200) {
+ jenisInformasiDiminta.findMany.data = res.data?.data ?? [];
+ }
+ },
+ },
+});
const caraMemperolehInformasi = proxy({
- findMany: {
- data: null as
- | null
- | Prisma.CaraMemperolehInformasiGetPayload<{ omit: { isActive: true } }>[],
- async load() {
- const res = await ApiFetch.api.ppid.permohonaninformasipublik.memperolehInformasi["find-many"].get();
- if (res.status === 200) {
- caraMemperolehInformasi.findMany.data = res.data?.data ?? [];
- }
- }
- }
-})
+ findMany: {
+ data: null as
+ | null
+ | Prisma.CaraMemperolehInformasiGetPayload<{
+ omit: { isActive: true };
+ }>[],
+ async load() {
+ const res =
+ await ApiFetch.api.ppid.permohonaninformasipublik.memperolehInformasi[
+ "find-many"
+ ].get();
+ if (res.status === 200) {
+ caraMemperolehInformasi.findMany.data = res.data?.data ?? [];
+ }
+ },
+ },
+});
const caraMemperolehSalinanInformasi = proxy({
- findMany: {
- data: null as
- | null
- | Prisma.CaraMemperolehSalinanInformasiGetPayload<{ omit: { isActive: true } }>[],
- async load() {
- const res = await ApiFetch.api.ppid.permohonaninformasipublik.salinanInformasi["find-many"].get();
- if (res.status === 200) {
- caraMemperolehSalinanInformasi.findMany.data = res.data?.data ?? [];
- }
- }
- }
-})
-console.log(caraMemperolehSalinanInformasi)
+ findMany: {
+ data: null as
+ | null
+ | Prisma.CaraMemperolehSalinanInformasiGetPayload<{
+ omit: { isActive: true };
+ }>[],
+ async load() {
+ const res =
+ await ApiFetch.api.ppid.permohonaninformasipublik.salinanInformasi[
+ "find-many"
+ ].get();
+ if (res.status === 200) {
+ caraMemperolehSalinanInformasi.findMany.data = res.data?.data ?? [];
+ }
+ },
+ },
+});
+console.log(caraMemperolehSalinanInformasi);
-type PermohonanInformasiPublikForm = Prisma.PermohonanInformasiPublikGetPayload<{
+type PermohonanInformasiPublikForm =
+ Prisma.PermohonanInformasiPublikGetPayload<{
select: {
- name: true;
- nik: true;
- notelp: true;
- alamat: true;
- email: true;
- jenisInformasiDimintaId: true;
- caraMemperolehInformasiId: true;
- caraMemperolehSalinanInformasiId: true;
+ name: true;
+ nik: true;
+ notelp: true;
+ alamat: true;
+ email: true;
+ jenisInformasiDimintaId: true;
+ caraMemperolehInformasiId: true;
+ caraMemperolehSalinanInformasiId: true;
};
-}>;
+ }>;
const statepermohonanInformasiPublik = proxy({
- create: {
- form: {} as PermohonanInformasiPublikForm,
- loading: false,
- async create(){
- const cek = templateForm.safeParse(statepermohonanInformasiPublik.create.form);
- if(!cek.success) {
- const err = `[${cek.error.issues
- .map((v) => `${v.path.join(".")}`)
- .join("\n")}] required`;
- return toast.error(err);
- }
- try {
- statepermohonanInformasiPublik.create.loading = true;
- const res = await ApiFetch.api.ppid.permohonaninformasipublik["create"].post(statepermohonanInformasiPublik.create.form);
- if (res.status === 200) {
- statepermohonanInformasiPublik.findMany.load();
- return toast.success("success create");
- }
- return toast.error("failed create");
- } catch (error) {
- console.log((error as Error).message);
- } finally {
- statepermohonanInformasiPublik.create.loading = false;
- }
+ create: {
+ form: {} as PermohonanInformasiPublikForm,
+ loading: false,
+ async create() {
+ const cek = templateForm.safeParse(
+ statepermohonanInformasiPublik.create.form
+ );
+
+ if (!cek.success) {
+ toast.error(cek.error.issues.map((i) => i.message).join("\n"));
+ return false; // ⬅️ tambahkan return false
+ }
+
+ try {
+ statepermohonanInformasiPublik.create.loading = true;
+ const res = await ApiFetch.api.ppid.permohonaninformasipublik[
+ "create"
+ ].post(statepermohonanInformasiPublik.create.form);
+
+ if (res.data?.success === false) {
+ toast.error(res.data?.message);
+ return false; // ⬅️ gagal
}
+
+ toast.success("Sukses menambahkan");
+ return true; // ⬅️ sukses
+ } catch {
+ toast.error("Terjadi kesalahan server");
+ return false;
+ } finally {
+ statepermohonanInformasiPublik.create.loading = false;
+ }
},
- findMany: {
- data: null as
- | Prisma.PermohonanInformasiPublikGetPayload<{ include: {
- caraMemperolehSalinanInformasi: true,
- jenisInformasiDiminta: true,
- caraMemperolehInformasi: true,
- } }>[]
- | null,
- async load() {
- const res = await ApiFetch.api.ppid.permohonaninformasipublik["find-many"].get();
- if (res.status === 200) {
- statepermohonanInformasiPublik.findMany.data = res.data?.data ?? [];
- }
- }
- },
- findUnique: {
- data: null as Prisma.PermohonanInformasiPublikGetPayload<{
+ },
+ findMany: {
+ data: null as
+ | Prisma.PermohonanInformasiPublikGetPayload<{
include: {
- jenisInformasiDiminta: true,
- caraMemperolehInformasi: true,
- caraMemperolehSalinanInformasi: true,
+ caraMemperolehSalinanInformasi: true;
+ jenisInformasiDiminta: true;
+ caraMemperolehInformasi: true;
};
- }> | null,
- async load(id: string) {
- try {
- const res = await fetch(`/api/ppid/permohonaninformasipublik/${id}`);
- if (res.ok) {
- const data = await res.json();
- statepermohonanInformasiPublik.findUnique.data = data.data ?? null;
- } else {
- console.error("Failed to fetch program inovasi:", res.statusText);
- statepermohonanInformasiPublik.findUnique.data = null;
- }
- } catch (error) {
- console.error("Error fetching program inovasi:", error);
- statepermohonanInformasiPublik.findUnique.data = null;
- }
- },
- },
-
-})
+ }>[]
+ | null,
+ page: 1,
+ totalPages: 1,
+ total: 0,
+ loading: false,
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ // Change to arrow function
+ statepermohonanInformasiPublik.findMany.loading = true; // Use the full path to access the property
+ statepermohonanInformasiPublik.findMany.page = page;
+ statepermohonanInformasiPublik.findMany.search = search;
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.ppid.permohonaninformasipublik[
+ "find-many"
+ ].get({
+ query,
+ });
+
+ if (res.status === 200 && res.data?.success) {
+ statepermohonanInformasiPublik.findMany.data = res.data.data || [];
+ statepermohonanInformasiPublik.findMany.total = res.data.total || 0;
+ statepermohonanInformasiPublik.findMany.totalPages = res.data.totalPages || 1;
+ } else {
+ console.error("Failed to load permohonan keberatan informasi:", res.data?.message);
+ statepermohonanInformasiPublik.findMany.data = [];
+ statepermohonanInformasiPublik.findMany.total = 0;
+ statepermohonanInformasiPublik.findMany.totalPages = 1;
+ }
+ } catch (error) {
+ console.error("Error loading permohonan keberatan informasi:", error);
+ statepermohonanInformasiPublik.findMany.data = [];
+ statepermohonanInformasiPublik.findMany.total = 0;
+ statepermohonanInformasiPublik.findMany.totalPages = 1;
+ } finally {
+ statepermohonanInformasiPublik.findMany.loading = false;
+ }
+ },
+ },
+ findUnique: {
+ data: null as Prisma.PermohonanInformasiPublikGetPayload<{
+ include: {
+ jenisInformasiDiminta: true;
+ caraMemperolehInformasi: true;
+ caraMemperolehSalinanInformasi: true;
+ };
+ }> | null,
+ async load(id: string) {
+ try {
+ const res = await fetch(`/api/ppid/permohonaninformasipublik/${id}`);
+ if (res.ok) {
+ const data = await res.json();
+ statepermohonanInformasiPublik.findUnique.data = data.data ?? null;
+ } else {
+ console.error("Failed to fetch program inovasi:", res.statusText);
+ statepermohonanInformasiPublik.findUnique.data = null;
+ }
+ } catch (error) {
+ console.error("Error fetching program inovasi:", error);
+ statepermohonanInformasiPublik.findUnique.data = null;
+ }
+ },
+ },
+});
const statepermohonanInformasiPublikForm = proxy({
- statepermohonanInformasiPublik,
- jenisInformasiDiminta,
- caraMemperolehInformasi,
- caraMemperolehSalinanInformasi,
-})
+ statepermohonanInformasiPublik,
+ jenisInformasiDiminta,
+ caraMemperolehInformasi,
+ caraMemperolehSalinanInformasi,
+});
export default statepermohonanInformasiPublikForm;
diff --git a/src/app/admin/(dashboard)/_state/ppid/permohonan_keberatan_informasi_publik/permohonanKeberatanInformasi.ts b/src/app/admin/(dashboard)/_state/ppid/permohonan_keberatan_informasi_publik/permohonanKeberatanInformasi.ts
index 92373c3e..4ba89cb7 100644
--- a/src/app/admin/(dashboard)/_state/ppid/permohonan_keberatan_informasi_publik/permohonanKeberatanInformasi.ts
+++ b/src/app/admin/(dashboard)/_state/ppid/permohonan_keberatan_informasi_publik/permohonanKeberatanInformasi.ts
@@ -1,3 +1,4 @@
+/* eslint-disable @typescript-eslint/no-explicit-any */
import ApiFetch from "@/lib/api-fetch";
import { Prisma } from "@prisma/client";
import { toast } from "react-toastify";
@@ -5,82 +6,130 @@ import { proxy } from "valtio";
import { z } from "zod";
const templateForm = z.object({
- name: z.string().min(3, "Nama minimal 3 karakter"),
- email: z.string().min(3, "Email minimal 3 karakter"),
- notelp: z.string().min(3, "Nomor Telepon minimal 3 karakter"),
- alasan: z.string().min(3, "Alasan minimal 3 karakter"),
-})
+ name: z.string().min(3, "Nama minimal 3 karakter"),
+ email: z.string().min(3, "Email minimal 3 karakter"),
+ notelp: z
+ .string()
+ .min(3, "Nomor Telepon minimal 3 karakter")
+ .max(15, "Nomor Telepon maksimal 15 angka"),
+ alasan: z.string().min(3, "Alasan minimal 3 karakter"),
+});
-type PermohonanKeberatanInformasiForm = Prisma.FormulirPermohonanKeberatanGetPayload<{
+type PermohonanKeberatanInformasiForm =
+ Prisma.FormulirPermohonanKeberatanGetPayload<{
select: {
- name: true;
- email: true;
- notelp: true;
- alasan: true;
+ name: true;
+ email: true;
+ notelp: true;
+ alasan: true;
};
-}>;
+ }>;
const permohonanKeberatanInformasi = proxy({
- create: {
- form: {} as PermohonanKeberatanInformasiForm,
- loading: false,
- async create(){
- const cek = templateForm.safeParse(permohonanKeberatanInformasi.create.form);
- if(!cek.success) {
- const err = `[${cek.error.issues
- .map((v) => `${v.path.join(".")}`)
- .join("\n")}] required`;
- return toast.error(err);
- }
- try {
- permohonanKeberatanInformasi.create.loading = true;
- const res = await ApiFetch.api.ppid.permohonankeberataninformasipublik["create"].post(permohonanKeberatanInformasi.create.form);
- if (res.status === 200) {
- permohonanKeberatanInformasi.findMany.load();
- return toast.success("success create");
- }
- return toast.error("failed create");
- } catch (error) {
- console.log((error as Error).message);
- } finally {
- permohonanKeberatanInformasi.create.loading = false;
- }
- },
- },
- findMany: {
- data: null as
- | Prisma.FormulirPermohonanKeberatanGetPayload<{omit: {isActive: true}}>[]
- | null,
- async load() {
- const res = await ApiFetch.api.ppid.permohonankeberataninformasipublik["find-many"].get();
- if (res.status === 200) {
- permohonanKeberatanInformasi.findMany.data = res.data?.data ?? [];
- }
- }
- },
- findUnique: {
- data: null as Prisma.FormulirPermohonanKeberatanGetPayload<{
- omit: {
- isActive: true;
- };
- }> | null,
- async load(id: string) {
- try {
- const res = await fetch(`/api/ppid/permohonankeberataninformasipublik/${id}`);
- if (res.ok) {
- const data = await res.json();
- permohonanKeberatanInformasi.findUnique.data = data.data ?? null;
- } else {
- console.error("Failed to fetch permohonan keberatan informasi:", res.statusText);
- permohonanKeberatanInformasi.findUnique.data = null;
- }
- } catch (error) {
- console.error("Error fetching permohonan keberatan informasi:", error);
- permohonanKeberatanInformasi.findUnique.data = null;
- }
- },
+ create: {
+ form: {} as PermohonanKeberatanInformasiForm,
+ loading: false,
+ async create() {
+ const cek = templateForm.safeParse(
+ permohonanKeberatanInformasi.create.form
+ );
+ if (!cek.success) {
+ toast.error(cek.error.issues.map((i) => i.message).join("\n"));
+ return false; // ⬅️ tambahkan return false
}
+ try {
+ permohonanKeberatanInformasi.create.loading = true;
+ const res = await ApiFetch.api.ppid.permohonankeberataninformasipublik[
+ "create"
+ ].post(permohonanKeberatanInformasi.create.form);
+ if (res.data?.success === false) {
+ toast.error(res.data?.message);
+ return false; // ⬅️ gagal
+ }
+
+ toast.success("Sukses menambahkan");
+ return true; // ⬅️ sukses
+ } catch {
+ toast.error("Terjadi kesalahan server");
+ return false;
+ } finally {
+ permohonanKeberatanInformasi.create.loading = false;
+ }
+ },
+ },
+ findMany: {
+ data: null as
+ | null
+ | Prisma.FormulirPermohonanKeberatanGetPayload<{
+ omit: { isActive: true };
+ }>[],
+ page: 1,
+ totalPages: 1,
+ total: 0,
+ loading: false,
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ // Change to arrow function
+ permohonanKeberatanInformasi.findMany.loading = true; // Use the full path to access the property
+ permohonanKeberatanInformasi.findMany.page = page;
+ permohonanKeberatanInformasi.findMany.search = search;
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.ppid.permohonankeberataninformasipublik[
+ "find-many"
+ ].get({
+ query,
+ });
+
+ if (res.status === 200 && res.data?.success) {
+ permohonanKeberatanInformasi.findMany.data = res.data.data || [];
+ permohonanKeberatanInformasi.findMany.total = res.data.total || 0;
+ permohonanKeberatanInformasi.findMany.totalPages = res.data.totalPages || 1;
+ } else {
+ console.error("Failed to load permohonan keberatan informasi:", res.data?.message);
+ permohonanKeberatanInformasi.findMany.data = [];
+ permohonanKeberatanInformasi.findMany.total = 0;
+ permohonanKeberatanInformasi.findMany.totalPages = 1;
+ }
+ } catch (error) {
+ console.error("Error loading permohonan keberatan informasi:", error);
+ permohonanKeberatanInformasi.findMany.data = [];
+ permohonanKeberatanInformasi.findMany.total = 0;
+ permohonanKeberatanInformasi.findMany.totalPages = 1;
+ } finally {
+ permohonanKeberatanInformasi.findMany.loading = false;
+ }
+ },
+ },
+ findUnique: {
+ data: null as Prisma.FormulirPermohonanKeberatanGetPayload<{
+ omit: {
+ isActive: true;
+ };
+ }> | null,
+ async load(id: string) {
+ try {
+ const res = await fetch(
+ `/api/ppid/permohonankeberataninformasipublik/${id}`
+ );
+ if (res.ok) {
+ const data = await res.json();
+ permohonanKeberatanInformasi.findUnique.data = data.data ?? null;
+ } else {
+ console.error(
+ "Failed to fetch permohonan keberatan informasi:",
+ res.statusText
+ );
+ permohonanKeberatanInformasi.findUnique.data = null;
+ }
+ } catch (error) {
+ console.error("Error fetching permohonan keberatan informasi:", error);
+ permohonanKeberatanInformasi.findUnique.data = null;
+ }
+ },
+ },
});
export default permohonanKeberatanInformasi;
-
diff --git a/src/app/admin/(dashboard)/_state/ppid/profile_ppid/profile_PPID.ts b/src/app/admin/(dashboard)/_state/ppid/profile_ppid/profile_PPID.ts
index 0a6304b8..fe01b462 100644
--- a/src/app/admin/(dashboard)/_state/ppid/profile_ppid/profile_PPID.ts
+++ b/src/app/admin/(dashboard)/_state/ppid/profile_ppid/profile_PPID.ts
@@ -3,9 +3,6 @@ import { toast } from "react-toastify";
import { proxy } from "valtio";
import { z } from "zod";
-/**
- * Schema validasi form ProfilePPID menggunakan Zod.
- */
const templateForm = z.object({
name: z.string().min(3, "Nama minimal 3 karakter"),
biodata: z.string().min(3, "Biodata minimal 3 karakter"),
@@ -33,25 +30,16 @@ type ProfilePPIDForm = Prisma.ProfilePPIDGetPayload<{
pengalaman: true;
unggulan: true;
imageId: true;
- image?: {
- select: {
- link: true;
- };
- };
+ image?: { select: { link: true } };
};
}>;
-/**
- * Improved State Management - Consolidated and more robust
- */
const stateProfilePPID = proxy({
- // Consolidated data management
profile: {
data: null as ProfilePPIDForm | null,
loading: false,
error: null as string | null,
- // Single method to load profile data
async load(id: string) {
if (!id) {
toast.warn("ID tidak valid");
@@ -62,52 +50,54 @@ const stateProfilePPID = proxy({
this.error = null;
try {
- const response = await fetch(`/api/ppid/profileppid/${id}`);
-
- if (!response.ok) {
- throw new Error(`HTTP error! status: ${response.status}`);
- }
+ const res = await fetch(`/api/ppid/profileppid/${id}`);
+ if (!res.ok) throw new Error(`HTTP ${res.status}`);
- const result = await response.json();
-
+ const result = await res.json();
if (result.success) {
this.data = result.data;
return result.data;
} else {
- throw new Error(result.message || "Gagal mengambil data profile");
+ // Jika pesan adalah "Data tidak ditemukan" atau "Belum ada data profil PPID yang aktif",
+ // tetap simpan sebagai error tapi tidak perlu menampilkan toast error karena ini bukan error sebenarnya
+ if (result.message === "Data tidak ditemukan" || result.message === "Belum ada data profil PPID yang aktif") {
+ this.error = result.message;
+ return null;
+ } else {
+ throw new Error(result.message || "Gagal memuat data profile");
+ }
+ }
+ } catch (err) {
+ const msg = (err as Error).message;
+ this.error = msg;
+ console.error("Load profile error:", msg);
+ // Hanya tampilkan toast error jika bukan karena data tidak ditemukan
+ if (msg !== "Data tidak ditemukan" && msg !== "Belum ada data profil PPID yang aktif") {
+ toast.error("Gagal memuat data profile");
}
- } catch (error) {
- const errorMessage = (error as Error).message;
- this.error = errorMessage;
- console.error("Load profile error:", errorMessage);
- toast.error("Terjadi kesalahan saat mengambil data profile");
return null;
} finally {
this.loading = false;
}
},
- // Reset profile data
reset() {
this.data = null;
this.error = null;
this.loading = false;
- }
+ },
},
- // Edit form management
editForm: {
id: "",
form: { ...defaultForm },
+ originalForm: { ...defaultForm }, // ✅ Tambah field originalForm
loading: false,
error: null as string | null,
- isReadOnly: false, // Flag untuk data yang tidak bisa diedit
- // Initialize form with profile data
initialize(profileData: ProfilePPIDForm) {
this.id = profileData.id;
- this.isReadOnly = false; // Semua data bisa diedit
- this.form = {
+ const data = {
name: profileData.name || "",
biodata: profileData.biodata || "",
riwayat: profileData.riwayat || "",
@@ -115,23 +105,20 @@ const stateProfilePPID = proxy({
unggulan: profileData.unggulan || "",
imageId: profileData.imageId || "",
};
+ this.form = { ...data };
+ this.originalForm = { ...data }; // ✅ Simpan versi original
},
- // Update form field
updateField(field: keyof typeof defaultForm, value: string) {
this.form[field] = value;
},
- // Submit form
async submit() {
- // Validate form
- const validation = templateForm.safeParse(this.form);
-
- if (!validation.success) {
- const errors = validation.error.issues
- .map((issue) => `${issue.path.join(".")}: ${issue.message}`)
- .join(", ");
- toast.error(`Form tidak valid: ${errors}`);
+ const check = templateForm.safeParse(this.form);
+ if (!check.success) {
+ toast.error(
+ check.error.issues.map((i) => `${i.path.join(".")}: ${i.message}`).join(", ")
+ );
return false;
}
@@ -139,63 +126,54 @@ const stateProfilePPID = proxy({
this.error = null;
try {
- const response = await fetch(`/api/ppid/profileppid/${this.id}`, {
+ const res = await fetch(`/api/ppid/profileppid/${this.id}`, {
method: "PUT",
- headers: {
- "Content-Type": "application/json",
- },
+ headers: { "Content-Type": "application/json" },
body: JSON.stringify(this.form),
});
-
- 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 (!res.ok) throw new Error(`HTTP ${res.status}`);
+ const result = await res.json();
if (result.success) {
toast.success("Berhasil update profile");
- // Refresh profile data
- await stateProfilePPID.profile.load(this.id);
+ this.originalForm = { ...this.form }; // ✅ Update original setelah sukses
return true;
- } else {
- throw new Error(result.message || "Gagal update profile");
- }
- } catch (error) {
- const errorMessage = (error as Error).message;
- this.error = errorMessage;
- console.error("Update profile error:", errorMessage);
- toast.error("Terjadi kesalahan saat update profile");
+ } else throw new Error(result.message || "Gagal update profile");
+ } catch (err) {
+ const msg = (err as Error).message;
+ this.error = msg;
+ toast.error(msg);
return false;
} finally {
this.loading = false;
}
},
- // Reset form
+ // ✅ Tambahan reset ke original data
+ resetToOriginal() {
+ this.form = { ...this.originalForm };
+ toast.info("Data dikembalikan ke kondisi awal");
+ },
+
reset() {
this.id = "";
this.form = { ...defaultForm };
+ this.originalForm = { ...defaultForm };
this.error = null;
this.loading = false;
- this.isReadOnly = false;
- }
+ },
},
- // Helper methods
async loadForEdit(id: string) {
- const profileData = await this.profile.load(id);
- if (profileData) {
- this.editForm.initialize(profileData);
- }
- return profileData;
+ const data = await this.profile.load(id);
+ if (data) this.editForm.initialize(data);
+ return data;
},
reset() {
this.profile.reset();
this.editForm.reset();
- }
+ },
});
-export default stateProfilePPID;
\ No newline at end of file
+export default stateProfilePPID;
diff --git a/src/app/admin/(dashboard)/_state/user/user-state.ts b/src/app/admin/(dashboard)/_state/user/user-state.ts
index 93594956..c049d703 100644
--- a/src/app/admin/(dashboard)/_state/user/user-state.ts
+++ b/src/app/admin/(dashboard)/_state/user/user-state.ts
@@ -90,42 +90,96 @@ const userState = proxy({
}
},
},
- updateActive: {
+ deleteUser: {
loading: false,
- async submit(id: string, isActive: boolean) {
- this.loading = true;
+
+ async delete(id: string) {
+ if (!id) return toast.warn("ID tidak valid");
+
try {
- const res = await fetch(`/api/user/updt`, {
- method: "PUT",
- headers: { "Content-Type": "application/json" },
- body: JSON.stringify({ id, isActive }),
+ userState.deleteUser.loading = true;
+
+ const response = await fetch(`/api/user/delUser/${id}`, {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
});
-
- const data = await res.json();
- if (res.status === 200 && data.success) {
- toast.success(data.message);
- userState.findMany.load(userState.findMany.page, 10, userState.findMany.search);
+
+ const result = await response.json();
+
+ if (response.ok && result?.success) {
+ toast.success(result.message || "User berhasil dihapus permanen");
+ await userState.findMany.load(); // refresh list user setelah delete
} else {
- toast.error(data.message || "Gagal update status user");
+ toast.error(result?.message || "Gagal menghapus user");
}
- } catch (e) {
- console.error(e);
- toast.error("Gagal update status user");
+ } catch (error) {
+ console.error("Gagal delete user:", error);
+ toast.error("Terjadi kesalahan saat menghapus user");
} finally {
- this.loading = false;
+ userState.deleteUser.loading = false;
}
},
+ },
+ // Di file userState.ts atau dimana state user berada
+
+update: {
+ loading: false,
+
+ async submit(payload: { id: string; isActive?: boolean; roleId?: string }) {
+ this.loading = true;
+ try {
+ const res = await fetch(`/api/user/updt`, {
+ method: "PUT",
+ headers: { "Content-Type": "application/json" },
+ body: JSON.stringify(payload),
+ });
+
+ const data = await res.json();
+
+ if (res.status === 200 && data.success) {
+ // ✅ Tampilkan pesan yang berbeda jika role berubah
+ if (data.roleChanged) {
+ toast.success(
+ `${data.message}\n\nUser akan logout otomatis dalam beberapa detik.`,
+ {
+ autoClose: 5000,
+ }
+ );
+ } else {
+ toast.success(data.message);
+ }
+
+ // Refresh list
+ await userState.findMany.load(
+ userState.findMany.page,
+ 10,
+ userState.findMany.search
+ );
+
+ return true; // ✅ Return success untuk handling di component
+ } else {
+ toast.error(data.message || "Gagal update user");
+ return false;
+ }
+ } catch (e) {
+ console.error("❌ Error update user:", e);
+ toast.error("Gagal update user");
+ return false;
+ } finally {
+ this.loading = false;
+ }
},
+},
});
const templateRole = z.object({
name: z.string().min(1, "Nama harus diisi"),
- permissions: z.array(z.string()).min(1, "Permission harus diisi"),
});
const defaultRole = {
name: "",
- permissions: [] as string[],
};
const roleState = proxy({
@@ -166,11 +220,34 @@ const roleState = proxy({
isActive: true;
};
}>[],
+ page: 1,
+ totalPages: 1,
loading: false,
- async load() {
- const res = await ApiFetch.api.role["findMany"].get();
- if (res.status === 200) {
- roleState.findMany.data = res.data?.data ?? [];
+ search: "",
+ load: async (page = 1, limit = 10, search = "") => {
+ roleState.findMany.loading = true; // ✅ Akses langsung via nama path
+ roleState.findMany.page = page;
+ roleState.findMany.search = search;
+
+ try {
+ const query: any = { page, limit };
+ if (search) query.search = search;
+
+ const res = await ApiFetch.api.role["findMany"].get({ query });
+
+ if (res.status === 200 && res.data?.success) {
+ roleState.findMany.data = res.data.data ?? [];
+ roleState.findMany.totalPages = res.data.totalPages ?? 1;
+ } else {
+ roleState.findMany.data = [];
+ roleState.findMany.totalPages = 1;
+ }
+ } catch (err) {
+ console.error("Gagal fetch role paginated:", err);
+ roleState.findMany.data = [];
+ roleState.findMany.totalPages = 1;
+ } finally {
+ roleState.findMany.loading = false;
}
},
},
@@ -237,7 +314,7 @@ const roleState = proxy({
toast.warn("ID tidak valid");
return null;
}
-
+
try {
const response = await fetch(`/api/role/${id}`, {
method: "GET",
@@ -245,31 +322,25 @@ const roleState = proxy({
"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 = {
+
+ // langsung set melalui root state, bukan this
+ roleState.update.id = data.id;
+ roleState.update.form = {
name: data.name,
- permissions: data.permissions,
};
- return data; // Return the loaded data
- } else {
- throw new Error(result?.message || "Gagal memuat data");
+
+ return data;
}
} catch (error) {
console.error("Error loading role:", error);
- toast.error(
- error instanceof Error ? error.message : "Gagal memuat data"
- );
- return null;
+ toast.error("Gagal memuat data");
}
- },
+ },
async update() {
const cek = templateRole.safeParse(roleState.update.form);
if (!cek.success) {
@@ -290,7 +361,6 @@ const roleState = proxy({
},
body: JSON.stringify({
name: this.form.name,
- permissions: this.form.permissions,
}),
});
diff --git a/src/app/admin/(dashboard)/auth/login-admin/page.tsx b/src/app/admin/(dashboard)/auth/login-admin/page.tsx
index ab591207..3457cf45 100644
--- a/src/app/admin/(dashboard)/auth/login-admin/page.tsx
+++ b/src/app/admin/(dashboard)/auth/login-admin/page.tsx
@@ -1,104 +1,103 @@
-'use client'
-import { apiFetchLogin } from '@/app/admin/auth/_lib/api_fetch_auth';
+'use client';
+import { apiFetchLogin } from '@/app/api/auth/_lib/api_fetch_auth';
import colors from '@/con/colors';
-import { Box, Button, Center, Flex, Image, Paper, Stack, Text, Title } from '@mantine/core';
-import Link from 'next/link';
+import { Box, Button, Center, Image, Paper, Stack, Title } from '@mantine/core';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
-import { PhoneInput } from "react-international-phone";
-import "react-international-phone/style.css";
+import { PhoneInput } from 'react-international-phone';
+import 'react-international-phone/style.css';
import { toast } from 'react-toastify';
-
-
function Login() {
- const router = useRouter()
- const [phone, setPhone] = useState("")
- const [isError, setError] = useState(false)
- const [loading, setLoading] = useState(false)
+ const router = useRouter();
+ const [phone, setPhone] = useState('');
+ const [loading, setLoading] = useState(false);
+ // Login.tsx
async function onLogin() {
- const nomor = phone.substring(1);
- if (nomor.length <= 4) return setError(true)
+ const cleanPhone = phone.replace(/\D/g, '');
+ console.log(cleanPhone);
+ if (cleanPhone.length < 10) {
+ toast.error('Nomor telepon tidak valid');
+ return;
+ }
try {
setLoading(true);
- const response = await apiFetchLogin({ nomor: nomor })
- if (response && response.success) {
- localStorage.setItem("hipmi_auth_code_id", response.kodeId);
- toast.success(response.message);
- router.push("/validasi", { scroll: false });
+ const response = await apiFetchLogin({ nomor: cleanPhone });
+
+ console.log(response);
+
+ if (!response.success) {
+ toast.error(response.message || 'Gagal memproses login');
+ return;
+ }
+
+ // Simpan nomor untuk register
+ localStorage.setItem('auth_nomor', cleanPhone);
+ if (response.isRegistered) {
+ // ✅ User lama: simpan kodeId
+ localStorage.setItem('auth_kodeId', response.kodeId);
+
+ // ✅ Cookie sudah di-set oleh API, langsung redirect
+ router.push('/validasi'); // Clean URL
} else {
- setLoading(false);
- toast.error(response?.message);
+ // ❌ User baru: langsung ke registrasi (tanpa kodeId)
+ router.push('/registrasi');
}
} catch (error) {
- setLoading(false)
- console.log("Error Login", error)
- toast.error("Terjadi kesalahan saat login")
+ console.error('Error Login:', error);
+ toast.error('Terjadi kesalahan saat login');
+ } finally {
+ setLoading(false);
}
}
return (
-
+
-
-
-
+
+
+
-
+
Login
-
+
-
- {/*
- Masuk Untuk Akses Admin
- setUsername(e.target.value)}
- required
- />
- */}
+
{
- setPhone(val);
- }}
+ value={phone}
+ onChange={(val) => setPhone(val)}
/>
- {isError ? (
- toast.error("Masukan nomor telepon anda")
- ) : (
- ""
- )}
-
+
Masuk
+ loading={loading}
+ >
+ Masuk
-
- Belum punya akun?
-
- Registrasi
-
-
@@ -108,4 +107,4 @@ function Login() {
);
}
-export default Login;
+export default Login;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/auth/registrasi-admin/page.tsx b/src/app/admin/(dashboard)/auth/registrasi-admin/page.tsx
index 62d2554b..b932413b 100644
--- a/src/app/admin/(dashboard)/auth/registrasi-admin/page.tsx
+++ b/src/app/admin/(dashboard)/auth/registrasi-admin/page.tsx
@@ -1,113 +1,153 @@
-/* eslint-disable @typescript-eslint/no-unused-expressions */
-'use client'
-import { apiFetchRegister } from '@/app/admin/auth/_lib/api_fetch_auth';
+// app/registrasi/page.tsx
+'use client';
+
+import { apiFetchRegister } from '@/app/api/auth/_lib/api_fetch_auth';
import BackButton from '@/app/darmasaba/(pages)/desa/layanan/_com/BackButto';
import colors from '@/con/colors';
-import { Box, Button, Center, Checkbox, Image, Paper, Stack, Text, TextInput, Title } from '@mantine/core';
+import {
+ Box, Button, Center, Checkbox, Image, Paper, Stack, Text, TextInput, Title,
+} from '@mantine/core';
import { useRouter } from 'next/navigation';
-import { useState } from 'react';
-import { PhoneInput } from "react-international-phone";
-import "react-international-phone/style.css";
+import { useEffect, useState } from 'react';
+import { PhoneInput } from 'react-international-phone';
+import 'react-international-phone/style.css';
import { toast } from 'react-toastify';
-function Registrasi() {
- const [phone, setPhone] = useState("")
- const router = useRouter()
- const [value, setValue] = useState("")
- const [isValue, setIsValue] = useState(false);
+export default function Registrasi() {
+ const router = useRouter();
+ const [username, setUsername] = useState('');
const [loading, setLoading] = useState(false);
+ const [phone, setPhone] = useState(''); // ✅ tambahkan state untuk phone
+ const [agree, setAgree] = useState(false)
- async function onRegistarsi() {
- if (value.length < 5) {
- toast.error("Username minimal 5 karakter!");
+ // Ambil data dari localStorage (dari login)
+ useEffect(() => {
+ const storedNomor = localStorage.getItem('auth_nomor');
+ if (!storedNomor) {
+ toast.error('Akses tidak valid');
+ router.push('/login');
return;
}
-
- if (value.includes(" ")) {
- toast.error("Username tidak boleh ada spasi!");
+ setPhone(storedNomor);
+ }, [router]);
+
+ const handleRegister = async () => {
+ if (!username || username.trim().length < 5) {
+ toast.error('Username minimal 5 karakter!');
return;
}
-
- if (!phone) {
- toast.error("Nomor telepon wajib diisi!");
+ if (username.includes(' ')) {
+ toast.error('Username tidak boleh ada spasi!');
return;
}
-
+
+ const cleanPhone = phone.replace(/\D/g, '');
+ if (cleanPhone.length < 10) {
+ toast.error('Nomor tidak valid!');
+ return;
+ }
+
+ if (!agree) {
+ toast.error("Anda harus menyetujui syarat dan ketentuan!");
+ return;
+ }
+
try {
setLoading(true);
- const respone = await apiFetchRegister({ nomor: phone, username: value });
+ // ✅ Hanya kirim username & nomor → dapat kodeId
+ const response = await apiFetchRegister({ username, nomor: cleanPhone });
- if (respone.success) {
- router.push("/login", { scroll: false });
- toast.success(respone.message);
+ if (response.success) {
+ // Simpan sementara
+ localStorage.setItem('auth_kodeId', response.kodeId);
+ localStorage.setItem('auth_username', username); // simpan username
- } else {
- setLoading(false);
- toast.error(respone.message);
+ toast.success('Kode verifikasi dikirim!');
+ router.push('/validasi'); // ✅ ke halaman validasi
}
} catch (error) {
+ console.error('Error Registrasi:', error);
+ toast.error('Gagal mengirim OTP');
+ } finally {
setLoading(false);
- console.log("Error Registrasi", error);
}
- }
+ };
+
return (
-
+
-
-
-
-
+
+
+
+
Registrasi
-
+
-
-
+ setUsername(e.currentTarget.value)}
error={
- value.length > 0 && value.length < 5
- ? "Minimal 5 karakter !"
- : value.includes(" ")
- ? "Tidak boleh ada spasi"
- : isValue
- ? "Masukan username anda"
- : ""
+ username.length > 0 && username.length < 5
+ ? 'Minimal 5 karakter!'
+ : username.includes(' ')
+ ? 'Tidak boleh ada spasi'
+ : ''
}
- onChange={(val) => {
- val.currentTarget.value.length > 0 ? setIsValue(false) : "";
- setValue(val.currentTarget.value);
- }}
required
-
/>
-
- Nomor Telepon
+
+
+ Nomor Telepon
{
- setPhone(val);
- }}
+ value={phone}
+ disabled
/>
-
+
+
setAgree(e.currentTarget.checked)}
+ label={
+
+ Saya menyetujui{" "}
+
+ syarat dan ketentuan
+
+
+ }
/>
-
- Daftar
+
+
+
+
+ Kirim Kode Verifikasi
+
@@ -116,6 +156,4 @@ function Registrasi() {
);
-}
-
-export default Registrasi;
+}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/auth/validasi-admin/page.tsx b/src/app/admin/(dashboard)/auth/validasi-admin/page.tsx
index 862edb33..e6fe0a4b 100644
--- a/src/app/admin/(dashboard)/auth/validasi-admin/page.tsx
+++ b/src/app/admin/(dashboard)/auth/validasi-admin/page.tsx
@@ -1,31 +1,306 @@
-'use client'
-import colors from '@/con/colors';
-import { Box, Button, Paper, PinInput, Stack, Text, Title } from '@mantine/core';
-import { useRouter } from 'next/navigation';
+'use client';
+
+import colors from '@/con/colors';
+import {
+ Box,
+ Button,
+ Center,
+ Loader,
+ Paper,
+ PinInput,
+ Stack,
+ Text,
+ Title,
+} from '@mantine/core';
+import { useRouter } from 'next/navigation';
+import { useEffect, useState } from 'react';
+import { toast } from 'react-toastify';
+import { authStore } from '@/store/authStore';
+
+export default function Validasi() {
+ const router = useRouter();
+
+ const [nomor, setNomor] = useState(null);
+ const [otp, setOtp] = useState('');
+ const [loading, setLoading] = useState(false);
+ const [isLoading, setIsLoading] = useState(true);
+ const [kodeId, setKodeId] = useState(null);
+ const [isRegistrationFlow, setIsRegistrationFlow] = useState(false);
+
+ // ✅ Deteksi flow dari cookie via API
+ useEffect(() => {
+ const checkFlow = async () => {
+ try {
+ const res = await fetch('/api/auth/get-flow', {
+ credentials: 'include'
+ });
+ const data = await res.json();
+
+ if (data.success) {
+ setIsRegistrationFlow(data.flow === 'register');
+ console.log('🔍 Flow detected from cookie:', data.flow);
+ }
+ } catch (error) {
+ console.error('❌ Error getting flow:', error);
+ setIsRegistrationFlow(false);
+ }
+ };
+
+ checkFlow();
+ }, []);
+
+ useEffect(() => {
+ const storedKodeId = localStorage.getItem('auth_kodeId');
+ if (!storedKodeId) {
+ toast.error('Akses tidak valid');
+ router.replace('/login');
+ return;
+ }
+
+ setKodeId(storedKodeId);
+ const loadOtpData = async () => {
+ try {
+ const res = await fetch(`/api/auth/otp-data?kodeId=${encodeURIComponent(storedKodeId)}`);
+ const result = await res.json();
+
+ if (res.ok && result.data?.nomor) {
+ setNomor(result.data.nomor);
+ } else {
+ throw new Error('Data OTP tidak valid');
+ }
+ } catch (error) {
+ console.error('Gagal memuat data OTP:', error);
+ toast.error('Kode verifikasi tidak valid');
+ router.replace('/login');
+ } finally {
+ setIsLoading(false);
+ }
+ };
+ loadOtpData();
+ }, [router]);
+
+ const handleVerify = async () => {
+ if (!kodeId || !nomor || otp.length < 4) return;
+
+ setLoading(true);
+ try {
+ if (isRegistrationFlow) {
+ await handleRegistrationVerification();
+ } else {
+ await handleLoginVerification();
+ }
+ } catch (error) {
+ console.error('Error saat verifikasi:', error);
+ toast.error('Terjadi kesalahan sistem');
+ } finally {
+ setLoading(false);
+ }
+ };
+
+ const handleRegistrationVerification = async () => {
+ const username = localStorage.getItem('auth_username');
+ if (!username) {
+ toast.error('Data registrasi tidak ditemukan.');
+ return;
+ }
+
+ const cleanNomor = nomor?.replace(/\D/g, '') ?? '';
+ if (cleanNomor.length < 10 || username.trim().length < 5) {
+ toast.error('Data tidak valid');
+ return;
+ }
+
+ // ✅ Verify OTP
+ const verifyRes = await fetch('/api/auth/verify-otp-register', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ nomor: cleanNomor, otp, kodeId }),
+ credentials: 'include'
+ });
+
+ const verifyData = await verifyRes.json();
+ if (!verifyRes.ok) {
+ toast.error(verifyData.message || 'Verifikasi OTP gagal');
+ return;
+ }
+
+ // ✅ Finalize registration
+ const finalizeRes = await fetch('/api/auth/finalize-registration', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ nomor: cleanNomor, username, kodeId }),
+ credentials: 'include'
+ });
+
+ const data = await finalizeRes.json();
+
+ // ✅ Check JSON response (bukan redirect)
+ if (data.success) {
+ toast.success('Registrasi berhasil! Menunggu persetujuan admin.');
+ await cleanupStorage();
+
+ // ✅ Client-side redirect
+ setTimeout(() => {
+ window.location.href = '/waiting-room';
+ }, 1000);
+ } else {
+ toast.error(data.message || 'Registrasi gagal');
+ }
+ };
+
+ const handleLoginVerification = async () => {
+ const loginRes = await fetch('/api/auth/verify-otp-login', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ nomor, otp, kodeId }),
+ credentials: 'include'
+ });
+
+ const loginData = await loginRes.json();
+
+ if (!loginRes.ok) {
+ toast.error(loginData.message || 'Verifikasi gagal');
+ return;
+ }
+
+ const { id, name, roleId, isActive } = loginData.user;
+
+ authStore.setUser({
+ id,
+ name: name || 'User',
+ roleId: Number(roleId),
+ });
+
+ // ✅ Cleanup setelah login sukses
+ await cleanupStorage();
+
+ if (!isActive) {
+ window.location.href = '/waiting-room';
+ return;
+ }
+
+ const redirectPath = getRedirectPath(Number(roleId));
+ router.replace(redirectPath);
+ };
+
+ const getRedirectPath = (roleId: number): string => {
+ switch (roleId) {
+ case 0:
+ case 1:
+ case 2:
+ return '/admin/landing-page/profil/program-inovasi';
+ case 3:
+ return '/admin/kesehatan/posyandu';
+ case 4:
+ return '/admin/pendidikan/info-sekolah/jenjang-pendidikan';
+ default:
+ return '/admin';
+ }
+ };
+
+ // ✅ CLEANUP FUNCTION - Hapus localStorage + Cookie
+ const cleanupStorage = async () => {
+ // Clear localStorage
+ localStorage.removeItem('auth_kodeId');
+ localStorage.removeItem('auth_nomor');
+ localStorage.removeItem('auth_username');
+
+ // Clear cookie
+ try {
+ await fetch('/api/auth/clear-flow', {
+ method: 'POST',
+ credentials: 'include'
+ });
+ } catch (error) {
+ console.error('Error clearing flow cookie:', error);
+ }
+ };
+
+ const handleResend = async () => {
+ if (!nomor) return;
+ try {
+ const res = await fetch('/api/auth/resend', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ nomor }),
+ });
+ const data = await res.json();
+ if (data.success) {
+ localStorage.setItem('auth_kodeId', data.kodeId);
+ toast.success('OTP baru dikirim');
+ } else {
+ toast.error(data.message || 'Gagal mengirim ulang OTP');
+ }
+ } catch {
+ toast.error('Gagal menghubungi server');
+ }
+ };
+
+ if (isLoading) {
+ return (
+
+
+
+ );
+ }
+
+ if (!nomor) return null;
-function Validasi() {
- const router = useRouter()
return (
-
+
-
-
-
+
+
+
-
- Kode Verifikasi
+
+ {isRegistrationFlow ? 'Verifikasi Registrasi' : 'Verifikasi Login'}
+
+ Kami telah mengirim kode ke nomor {nomor}
+
-
-
- Masukkan Kode Verifikasi
-
+
+
+
+ Masukkan Kode Verifikasi
+
+
+
+
-
- router.push("/admin/landing-page/profile/program-inovasi")}>
- Page
+
+
+ Verifikasi
+
+
+
+ Tidak menerima kode?{' '}
+
+ Kirim Ulang
-
+
@@ -33,6 +308,4 @@ function Validasi() {
);
-}
-
-export default Validasi;
+}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/_com/layoutTabLayanan.tsx b/src/app/admin/(dashboard)/desa/_com/layoutTabLayanan.tsx
index 4ec40e47..c0cf829b 100644
--- a/src/app/admin/(dashboard)/desa/_com/layoutTabLayanan.tsx
+++ b/src/app/admin/(dashboard)/desa/_com/layoutTabLayanan.tsx
@@ -1,10 +1,10 @@
/* eslint-disable react-hooks/exhaustive-deps */
'use client'
import colors from '@/con/colors';
-import { ScrollArea, Stack, Tabs, TabsList, TabsPanel, TabsTab, Title, Tooltip } from '@mantine/core';
+import { Box, ScrollArea, Stack, Tabs, TabsList, TabsPanel, TabsTab, Title } from '@mantine/core';
+import { IconBuildingStore, IconFileText, IconSparkles, IconUsers, IconUsersPlus } from '@tabler/icons-react';
import { usePathname, useRouter } from 'next/navigation';
import React, { useEffect, useState } from 'react';
-import { IconFileText, IconBuildingStore, IconSparkles, IconUsers, IconUsersPlus } from '@tabler/icons-react';
function LayoutTabsLayanan({ children }: { children: React.ReactNode }) {
const router = useRouter()
@@ -14,36 +14,31 @@ function LayoutTabsLayanan({ children }: { children: React.ReactNode }) {
label: "Pelayanan Surat Keterangan",
value: "pelayanansuratketerangan",
href: "/admin/desa/layanan/pelayanan_surat_keterangan",
- icon: ,
- tooltip: "Layanan terkait surat keterangan resmi desa"
+ icon:
},
{
label: "Pelayanan Perizinan Berusaha",
value: "pelayananperizinanusaha",
href: "/admin/desa/layanan/pelayanan_perizinan_berusaha",
- icon: ,
- tooltip: "Layanan untuk izin usaha masyarakat"
+ icon:
},
{
label: "Pelayanan Telunjuk Sakti Desa",
value: "pelayanantelunjuksaktidesa",
href: "/admin/desa/layanan/pelayanan_telunjuk_sakti_desa",
- icon: ,
- tooltip: "Layanan inovasi khusus desa"
+ icon:
},
{
label: "Pelayanan Penduduk Non-Permanent",
value: "pelayanannonpermanent",
href: "/admin/desa/layanan/pelayanan_penduduk_non_permanent",
- icon: ,
- tooltip: "Pendataan penduduk non-permanent"
+ icon:
},
{
label: "Ajukan Permohonan",
value: "ajukanpermohonan",
href: "/admin/desa/layanan/ajukan_permohonan",
- icon: ,
- tooltip: "Ajukan permohonan"
+ icon:
}
];
@@ -77,42 +72,76 @@ function LayoutTabsLayanan({ children }: { children: React.ReactNode }) {
keepMounted={false}
>
{/* ✅ Scroll horizontal wrapper */}
-
-
- {tabs.map((tab, i) => (
-
+
+
+
+ {tabs.map((tab, i) => (
{tab.label}
-
- ))}
-
-
+ ))}
+
+
+
+
+
+
+
+
+ {tabs.map((tab, i) => (
+
+ {tab.label}
+
+ ))}
+
+
+
{tabs.map((tab, i) => (
,
- tooltip: "Lihat dan kelola semua berita desa"
+ icon:
},
{
label: "Kategori Berita",
value: "kategori_berita",
href: "/admin/desa/berita/kategori-berita",
- icon: ,
- tooltip: "Kelola kategori berita desa"
+ icon:
},
];
@@ -71,46 +69,39 @@ function LayoutTabsBerita({ children }: { children: React.ReactNode }) {
}}
>
{tabs.map((tab, i) => (
-
-
- {tab.label}
-
-
+ {tab.label}
+
))}
-
-
+
+
- {tabs.map((tab, i) => (
-
- {/* Konten dummy, bisa diganti sesuai routing */}
- <>{children}>
-
- ))}
-
-
+ {tabs.map((tab, i) => (
+
+ {/* Konten dummy, bisa diganti sesuai routing */}
+ <>{children}>
+
+ ))}
+
+
);
}
diff --git a/src/app/admin/(dashboard)/desa/berita/kategori-berita/[id]/page.tsx b/src/app/admin/(dashboard)/desa/berita/kategori-berita/[id]/page.tsx
index 4ffdd64a..dfb687c4 100644
--- a/src/app/admin/(dashboard)/desa/berita/kategori-berita/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/kategori-berita/[id]/page.tsx
@@ -11,7 +11,7 @@ import {
Stack,
TextInput,
Title,
- Tooltip,
+ Loader
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -23,11 +23,23 @@ function EditKategoriBerita() {
const editState = useProxy(stateDashboardBerita.kategoriBerita);
const router = useRouter();
const params = useParams();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ const [originalData, setOriginalData] = useState({
+ name: '',
+ });
const [formData, setFormData] = useState({
name: '',
});
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== ''
+ );
+ };
+
useEffect(() => {
const loadKategori = async () => {
const id = params?.id as string;
@@ -39,6 +51,9 @@ function EditKategoriBerita() {
setFormData({
name: data.name || '',
});
+ setOriginalData({
+ name: data.name || '',
+ });
}
} catch (error) {
console.error('Error loading kategori Berita:', error);
@@ -56,8 +71,21 @@ function EditKategoriBerita() {
}));
};
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ });
+ toast.info('Form dikembalikan ke data awal');
+ };
+
const handleSubmit = async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Nama kategori berita wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
// update global state hanya saat submit
editState.update.form = {
...editState.update.form,
@@ -70,14 +98,15 @@ function EditKategoriBerita() {
} catch (error) {
console.error('Error updating kategori Berita:', error);
toast.error('Terjadi kesalahan saat memperbarui kategori Berita');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Back Button + Title */}
-
router.back()}
@@ -86,7 +115,6 @@ function EditKategoriBerita() {
>
-
Edit Kategori Berita
@@ -95,7 +123,7 @@ function EditKategoriBerita() {
{/* Form Wrapper */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/berita/kategori-berita/create/page.tsx b/src/app/admin/(dashboard)/desa/berita/kategori-berita/create/page.tsx
index db9e2b6a..c047ccfa 100644
--- a/src/app/admin/(dashboard)/desa/berita/kategori-berita/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/kategori-berita/create/page.tsx
@@ -9,15 +9,25 @@ import {
Stack,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { toast } from 'react-toastify';
import { useProxy } from 'valtio/utils';
function CreateKategoriBerita() {
const createState = useProxy(stateDashboardBerita.kategoriBerita);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ createState.create.form.name?.trim() !== ''
+ );
+ };
const resetForm = () => {
createState.create.form = {
@@ -26,16 +36,28 @@ function CreateKategoriBerita() {
};
const handleSubmit = async () => {
- await createState.create.create();
- resetForm();
- router.push('/admin/desa/berita/kategori-berita');
+ if (!createState.create.form.name?.trim()) {
+ toast.error('Nama kategori berita wajib diisi');
+ return;
+ }
+
+ setIsSubmitting(true);
+ try {
+ await createState.create.create();
+ resetForm();
+ router.push('/admin/desa/berita/kategori-berita');
+ } catch (error) {
+ console.error('Error creating kategori berita:', error);
+ toast.error('Gagal menambahkan kategori berita');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header dengan back button */}
-
router.back()}
@@ -44,7 +66,6 @@ function CreateKategoriBerita() {
>
-
Tambah Kategori Berita
@@ -63,23 +84,37 @@ function CreateKategoriBerita() {
(createState.create.form.name = e.target.value)}
required
/>
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/berita/kategori-berita/page.tsx b/src/app/admin/(dashboard)/desa/berita/kategori-berita/page.tsx
index 94dc8326..5f6d7a78 100644
--- a/src/app/admin/(dashboard)/desa/berita/kategori-berita/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/kategori-berita/page.tsx
@@ -17,8 +17,7 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
import { IconEdit, IconPlus, IconSearch, IconTrash } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
@@ -27,6 +26,7 @@ import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
import stateDashboardBerita from '../../../_state/desa/berita';
+import { useDebouncedValue } from '@mantine/hooks';
function KategoriBerita() {
const [search, setSearch] = useState('');
@@ -49,6 +49,7 @@ function ListKategoriBerita({ search }: { search: string }) {
const router = useRouter();
const [modalHapus, setModalHapus] = useState(false);
const [selectedId, setSelectedId] = useState(null);
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const {
data,
@@ -59,8 +60,8 @@ function ListKategoriBerita({ search }: { search: string }) {
} = listDataState.findMany;
useEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const handleDelete = () => {
if (selectedId) {
@@ -82,83 +83,86 @@ function ListKategoriBerita({ search }: { search: string }) {
}
return (
-
-
-
- Daftar Kategori Berita
-
- }
- color="blue"
- variant="light"
- onClick={() =>
- router.push('/admin/desa/berita/kategori-berita/create')
- }
- >
- Tambah Baru
-
-
+
+
+
+
+ Daftar Kategori Berita
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/desa/berita/kategori-berita/create')
+ }
+ >
+ Tambah Baru
+
-
-
+ {/* Desktop Table */}
+
+
- No
- Nama
- Edit
- Hapus
+
+ Kategori
+
+
+ Edit
+
+
+ Hapus
+
{filteredData.length > 0 ? (
- filteredData.map((item, index) => (
+ filteredData.map((item) => (
- {index + 1}
-
-
-
+
{item.name}
-
-
-
- router.push(
- `/admin/desa/berita/kategori-berita/${item.id}`
- )
- }
- >
-
-
-
+
+
+ router.push(
+ `/admin/desa/berita/kategori-berita/${item.id}`
+ )
+ }
+ size="compact-sm"
+ >
+
+
-
-
- {
- setSelectedId(item.id);
- setModalHapus(true);
- }}
- >
-
-
-
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ size="compact-sm"
+ >
+
+
))
) : (
-
-
-
+ {/* ✅ Match column count (3 columns) */}
+
+
Tidak ada data kategori berita yang cocok
@@ -168,22 +172,68 @@ function ListKategoriBerita({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+ Kategori
+
+ {item.name}
+
+
+
+
+ router.push(
+ `/admin/desa/berita/kategori-berita/${item.id}`
+ )
+ }
+ >
+
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ >
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data kategori berita yang cocok
+
+
+ )}
+
-
- {
- load(newPage, 10, search);
- window.scrollTo({ top: 0, behavior: 'smooth' });
- }}
- total={totalPages}
- mt="md"
- mb="md"
- color="blue"
- radius="md"
- />
-
+
+ {
+ load(newPage, 10, search);
+ window.scrollTo({ top: 0, behavior: 'smooth' });
+ }}
+ total={totalPages}
+ color="blue"
+ radius="md"
+ />
+
{/* Modal Konfirmasi Hapus */}
= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
+
return (
{children}
diff --git a/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/edit/page.tsx
index c34049cc..414e2749 100644
--- a/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/edit/page.tsx
@@ -6,6 +6,7 @@ import stateDashboardBerita from "@/app/admin/(dashboard)/_state/desa/berita";
import colors from "@/con/colors";
import ApiFetch from "@/lib/api-fetch";
import {
+ ActionIcon,
Box,
Button,
Group,
@@ -16,7 +17,7 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader
} from "@mantine/core";
import { Dropzone } from "@mantine/dropzone";
import {
@@ -45,6 +46,35 @@ function EditBerita() {
imageId: "",
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.judul?.trim() !== '' &&
+ formData.kategoriBeritaId !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ (file !== null || originalData.imageId !== '') && // Either a new file is selected or an existing image exists
+ !isHtmlEmpty(formData.content)
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ judul: "",
+ deskripsi: "",
+ kategoriBeritaId: "",
+ content: "",
+ imageId: "",
+ imageUrl: ""
+ });
+
// Load kategori + berita
useEffect(() => {
beritaState.kategoriBerita.findMany.load();
@@ -64,6 +94,15 @@ function EditBerita() {
imageId: data.imageId || "",
});
+ setOriginalData({
+ judul: data.judul || "",
+ deskripsi: data.deskripsi || "",
+ kategoriBeritaId: data.kategoriBeritaId || "",
+ content: data.content || "",
+ imageId: data.imageId || "",
+ imageUrl: data.image?.link || ""
+ });
+
if (data?.image?.link) {
setPreviewImage(data.image.link);
}
@@ -82,7 +121,33 @@ function EditBerita() {
};
const handleSubmit = async () => {
+ if (!formData.judul?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!formData.kategoriBeritaId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
+ if (!file && !originalData.imageId) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.content)) {
+ toast.error('Konten wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
// Update global state hanya sekali di sini
beritaState.berita.edit.form = {
...beritaState.berita.edit.form,
@@ -109,23 +174,36 @@ function EditBerita() {
} catch (error) {
console.error("Error updating berita:", error);
toast.error("Terjadi kesalahan saat memperbarui berita");
+ } finally {
+ setIsSubmitting(false);
}
};
+ const handleResetForm = () => {
+ setFormData({
+ judul: originalData.judul,
+ deskripsi: originalData.deskripsi,
+ kategoriBeritaId: originalData.kategoriBeritaId,
+ content: originalData.content,
+ imageId: originalData.imageId,
+ });
+ setPreviewImage(originalData.imageUrl || null);
+ setFile(null);
+ toast.info("Form dikembalikan ke data awal");
+ };
+
return (
-
+
{/* Header */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Berita
@@ -219,14 +297,14 @@ function EditBerita() {
Seret gambar atau klik untuk memilih file
- Maksimal 5MB, format gambar wajib
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
{previewImage && (
-
+
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -257,17 +353,32 @@ function EditBerita() {
{/* Action */}
+ {/* Tombol Batal */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/page.tsx b/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/page.tsx
index 2030f188..3f64306e 100644
--- a/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/list-berita/[id]/page.tsx
@@ -1,14 +1,14 @@
'use client'
-import { useProxy } from 'valtio/utils';
-import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text, Tooltip } from '@mantine/core';
+import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text } from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
-import colors from '@/con/colors';
import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
import stateDashboardBerita from '@/app/admin/(dashboard)/_state/desa/berita';
+import colors from '@/con/colors';
function DetailBerita() {
const beritaState = useProxy(stateDashboardBerita);
@@ -41,7 +41,7 @@ function DetailBerita() {
const data = beritaState.berita.findUnique.data;
return (
-
+
{/* Tombol Back */}
-
{
@@ -124,9 +123,7 @@ function DetailBerita() {
>
-
-
router.push(`/admin/desa/berita/list-berita/${data.id}/edit`)}
@@ -136,7 +133,6 @@ function DetailBerita() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/berita/list-berita/create/page.tsx b/src/app/admin/(dashboard)/desa/berita/list-berita/create/page.tsx
index ab176266..0ae98722 100644
--- a/src/app/admin/(dashboard)/desa/berita/list-berita/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/list-berita/create/page.tsx
@@ -14,7 +14,8 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader,
+ ActionIcon
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { useShallowEffect } from '@mantine/hooks';
@@ -29,6 +30,25 @@ export default function CreateBerita() {
const [previewImage, setPreviewImage] = useState(null);
const [file, setFile] = useState(null);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ beritaState.berita.create.form.judul?.trim() !== '' &&
+ beritaState.berita.create.form.kategoriBeritaId !== '' &&
+ !isHtmlEmpty(beritaState.berita.create.form.deskripsi) &&
+ file !== null &&
+ !isHtmlEmpty(beritaState.berita.create.form.content)
+ );
+ };
useShallowEffect(() => {
beritaState.kategoriBerita.findMany.load();
@@ -47,42 +67,73 @@ export default function CreateBerita() {
};
const handleSubmit = async () => {
+ if (!beritaState.berita.create.form.judul?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!beritaState.berita.create.form.kategoriBeritaId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(beritaState.berita.create.form.deskripsi)) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
if (!file) {
- return toast.warn('Silakan pilih file gambar terlebih dahulu');
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(beritaState.berita.create.form.content)) {
+ toast.error('Konten wajib diisi');
+ return;
}
- const res = await ApiFetch.api.fileStorage.create.post({
- file,
- name: file.name,
- });
+ try {
+ setIsSubmitting(true);
+ if (!file) {
+ return toast.warn('Silakan pilih file gambar terlebih dahulu');
+ }
- const uploaded = res.data?.data;
- if (!uploaded?.id) {
- return toast.error('Gagal mengunggah gambar, silakan coba lagi');
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
+
+ const uploaded = res.data?.data;
+ if (!uploaded?.id) {
+ return toast.error('Gagal mengunggah gambar, silakan coba lagi');
+ }
+
+ beritaState.berita.create.form.imageId = uploaded.id;
+
+ await beritaState.berita.create.create();
+
+ resetForm();
+ router.push('/admin/desa/berita/list-berita');
+ } catch (error) {
+ console.error('Error creating berita:', error);
+ toast.error('Terjadi kesalahan saat membuat berita');
+ } finally {
+ setIsSubmitting(false);
}
-
- beritaState.berita.create.form.imageId = uploaded.id;
-
- await beritaState.berita.create.create();
-
- resetForm();
- router.push('/admin/desa/berita/list-berita');
};
return (
-
+
{/* Header dengan tombol kembali */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Tambah Berita
@@ -100,7 +151,7 @@ export default function CreateBerita() {
(beritaState.berita.create.form.judul = e.target.value)}
required
/>
@@ -112,7 +163,7 @@ export default function CreateBerita() {
label: item.name,
value: item.id,
}))}
- defaultValue={beritaState.berita.create.form.kategoriBeritaId || null}
+ value={beritaState.berita.create.form.kategoriBeritaId || null}
onChange={(val: string | null) => {
if (val) {
const selected = beritaState.kategoriBerita.findMany.data?.find(
@@ -157,7 +208,7 @@ export default function CreateBerita() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -178,7 +229,7 @@ export default function CreateBerita() {
{previewImage && (
-
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -207,17 +278,31 @@ export default function CreateBerita() {
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/berita/list-berita/page.tsx b/src/app/admin/(dashboard)/desa/berita/list-berita/page.tsx
index 47b628d8..c36447f6 100644
--- a/src/app/admin/(dashboard)/desa/berita/list-berita/page.tsx
+++ b/src/app/admin/(dashboard)/desa/berita/list-berita/page.tsx
@@ -16,10 +16,9 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
import { IconCircleDashedPlus, IconDeviceImacCog, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
@@ -46,16 +45,17 @@ function Berita() {
function ListBerita({ search }: { search: string }) {
const beritaState = useProxy(stateDashboardBerita);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const { data, page, totalPages, loading, load } = beritaState.berita.findMany;
useShallowEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
if (loading || !data) {
return (
-
+
);
@@ -64,66 +64,68 @@ function ListBerita({ search }: { search: string }) {
const filteredData = data || [];
return (
-
+
Daftar Berita
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/berita/list-berita/create')}
- >
- Tambah Baru
-
-
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/berita/list-berita/create')}
+ >
+ Tambah Baru
+
-
-
+ {/* Desktop Table */}
+
+
- Judul
- Kategori
- Aksi
+ Judul
+ Kategori
+ Aksi
{filteredData.length > 0 ? (
filteredData.map((item) => (
-
-
-
- {item.judul}
-
-
+
+
+ {item.judul}
+
-
-
+
+
{item.kategoriBerita?.name || '-'}
-
+
router.push(`/admin/desa/berita/list-berita/${item.id}`)
}
+ fz="sm"
+ px="sm"
+ h={36}
>
-
- Detail
+
+ Detail
))
) : (
-
-
-
+
+
+
Tidak ada data berita yang cocok
@@ -133,13 +135,59 @@ function ListBerita({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Judul
+
+
+ {item.judul}
+
+
+
+ Kategori
+
+
+ {item.kategoriBerita?.name || '-'}
+
+
+
+ router.push(`/admin/desa/berita/list-berita/${item.id}`)
+ }
+ fz="sm"
+ h={36}
+ >
+
+ Detail
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data berita yang cocok
+
+
+ )}
+
{
- load(newPage, 10);
+ load(newPage, 10, debouncedSearch); // ✅ Include search parameter
window.scrollTo({ top: 0, behavior: 'smooth' });
}}
total={totalPages}
@@ -153,4 +201,4 @@ function ListBerita({ search }: { search: string }) {
);
}
-export default Berita;
+export default Berita;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/gallery/foto/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/gallery/foto/[id]/edit/page.tsx
new file mode 100644
index 00000000..741e0b99
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/gallery/foto/[id]/edit/page.tsx
@@ -0,0 +1,337 @@
+/* eslint-disable react-hooks/exhaustive-deps */
+"use client";
+
+import EditEditor from "@/app/admin/(dashboard)/_com/editEditor";
+import stateGallery from "@/app/admin/(dashboard)/_state/desa/gallery";
+import colors from "@/con/colors";
+import ApiFetch from "@/lib/api-fetch";
+import {
+ ActionIcon,
+ Box,
+ Button,
+ Group,
+ Image,
+ Loader,
+ Paper,
+ Stack,
+ Text,
+ TextInput,
+ Title
+} from "@mantine/core";
+import { Dropzone } from "@mantine/dropzone";
+import {
+ IconArrowBack,
+ IconPhoto,
+ IconUpload,
+ IconX,
+} from "@tabler/icons-react";
+import { useParams, useRouter } from "next/navigation";
+import { useEffect, useState } from "react";
+import { toast } from "react-toastify";
+import { useProxy } from "valtio/utils";
+
+function EditFoto() {
+ const FotoState = useProxy(stateGallery.foto);
+ const router = useRouter();
+ const params = useParams();
+
+ const [previewImage, setPreviewImage] = useState(null);
+ const [file, setFile] = useState(null);
+ const [formData, setFormData] = useState({
+ name: "",
+ deskripsi: "",
+ imagesId: "",
+ });
+
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ (file !== null || originalData.imagesId !== '') // Either a new file is selected or an existing image exists
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ name: "",
+ deskripsi: "",
+ imagesId: "",
+ imageUrl: "",
+ });
+
+ // Load kategori + Foto
+ useEffect(() => {
+ FotoState.findMany.load();
+
+ const loadFoto = async () => {
+ const id = params?.id as string;
+ if (!id) return;
+
+ try {
+ const data = await FotoState.update.load(id);
+ if (data) {
+ setFormData({
+ name: data.name || "",
+ deskripsi: data.deskripsi || "",
+ imagesId: data.imagesId || "",
+ });
+
+ setOriginalData({
+ name: data.name || "",
+ deskripsi: data.deskripsi || "",
+ imagesId: data.imagesId || "",
+ imageUrl: data.imageGalleryFoto?.link || ""
+ });
+
+ if (data?.imageGalleryFoto?.link) {
+ setPreviewImage(data.imageGalleryFoto.link);
+ }
+ }
+ } catch (error) {
+ console.error("Error loading Foto:", error);
+ toast.error("Gagal memuat data Foto");
+ }
+ };
+
+ loadFoto();
+ }, [params?.id]);
+
+ const handleChange = (field: string, value: string) => {
+ setFormData((prev) => ({ ...prev, [field]: value }));
+ };
+
+ const handleSubmit = async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ if (!file && !originalData.imagesId) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+ // Update global state hanya sekali di sini
+ FotoState.update.form = {
+ ...FotoState.update.form,
+ ...formData,
+ };
+
+ if (file) {
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
+ const uploaded = res.data?.data;
+
+ if (!uploaded?.id) {
+ return toast.error("Gagal upload gambar");
+ }
+
+ FotoState.update.form.imagesId = uploaded.id;
+ }
+
+ await FotoState.update.update();
+ toast.success("Foto berhasil diperbarui!");
+ router.push("/admin/desa/gallery/foto");
+ } catch (error) {
+ console.error("Error updating foto:", error);
+ toast.error("Terjadi kesalahan saat memperbarui foto");
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ deskripsi: originalData.deskripsi,
+ imagesId: originalData.imagesId,
+ });
+ setPreviewImage(originalData.imageUrl || null);
+ setFile(null);
+ toast.info("Form dikembalikan ke data awal");
+ };
+
+ return (
+
+ {/* Header */}
+
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
+
+ Edit Foto
+
+
+
+ {/* Form */}
+
+
+ handleChange("name", e.target.value)}
+ required
+ />
+
+ {/* Upload Gambar */}
+
+
+ Gambar Foto
+
+ {
+ const selectedFile = files[0];
+ if (selectedFile) {
+ setFile(selectedFile);
+ setPreviewImage(URL.createObjectURL(selectedFile));
+ }
+ }}
+ onReject={() =>
+ toast.error("File tidak valid, gunakan format gambar")
+ }
+ maxSize={5 * 1024 ** 2}
+ accept={{ "image/*": [] }}
+ radius="md"
+ p="xl"
+ >
+
+
+
+
+
+
+
+
+
+
+
+
+ Seret gambar atau klik untuk memilih file
+
+
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
+
+
+
+
+
+ {previewImage && (
+
+
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
+
+ )}
+
+
+ {/* Deskripsi */}
+
+
+ Deskripsi Foto
+
+
+ setFormData((prev) => ({ ...prev, deskripsi: htmlContent }))
+ }
+ />
+
+
+ {/* Action */}
+
+ {/* Tombol Batal */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
+
+
+
+
+ );
+}
+
+export default EditFoto;
diff --git a/src/app/admin/(dashboard)/desa/gallery/foto/[id]/page.tsx b/src/app/admin/(dashboard)/desa/gallery/foto/[id]/page.tsx
new file mode 100644
index 00000000..da51e858
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/gallery/foto/[id]/page.tsx
@@ -0,0 +1,175 @@
+'use client';
+
+import { Box, Button, Group, Paper, Skeleton, Stack, Text, Alert } from '@mantine/core';
+import Image from 'next/image';
+import { useShallowEffect } from '@mantine/hooks';
+import { IconArrowBack, IconEdit, IconTrash, IconPhoto } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
+
+import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
+import colors from '@/con/colors';
+import stateGallery from '@/app/admin/(dashboard)/_state/desa/gallery';
+
+function DetailFoto() {
+ const FotoState = useProxy(stateGallery.foto);
+ const [modalHapus, setModalHapus] = useState(false);
+ const [selectedId, setSelectedId] = useState(null);
+ const [imageError, setImageError] = useState(false);
+ const params = useParams();
+ const router = useRouter();
+
+ useShallowEffect(() => {
+ FotoState.findUnique.load(params?.id as string);
+ }, []);
+
+ const handleHapus = () => {
+ if (selectedId) {
+ FotoState.delete.byId(selectedId);
+ setModalHapus(false);
+ setSelectedId(null);
+ router.push("/admin/desa/gallery/foto");
+ }
+ };
+
+ if (!FotoState.findUnique.data) {
+ return (
+
+
+
+ );
+ }
+
+ const data = FotoState.findUnique.data;
+ const imageUrl = data.imageGalleryFoto?.link;
+
+ return (
+
+ router.back()}
+ leftSection={ }
+ mb={15}
+ >
+ Kembali
+
+
+
+
+
+ Detail Foto
+
+
+
+
+
+ Judul Foto
+ {data.name || '-'}
+
+
+
+ Deskripsi
+
+
+
+
+ Gambar
+ {imageUrl ? (
+
+ setImageError(true)}
+ />
+
+ ) : imageError ? (
+ }
+ title="Gagal memuat gambar"
+ radius="md"
+ >
+ Gambar tidak dapat ditampilkan.
+
+ ) : (
+ Tidak ada gambar
+ )}
+
+
+ {/* Action Buttons */}
+
+ {
+ setSelectedId(data.id);
+ setModalHapus(true);
+ }}
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
+
+ router.push(`/admin/desa/gallery/foto/${data.id}/edit`)}
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
+
+
+
+
+
+
+ setModalHapus(false)}
+ onConfirm={handleHapus}
+ text="Apakah Anda yakin ingin menghapus foto ini?"
+ />
+
+ );
+}
+
+export default DetailFoto;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/gallery/foto/create/page.tsx b/src/app/admin/(dashboard)/desa/gallery/foto/create/page.tsx
new file mode 100644
index 00000000..cf076af2
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/gallery/foto/create/page.tsx
@@ -0,0 +1,262 @@
+'use client';
+import CreateEditor from '@/app/admin/(dashboard)/_com/createEditor';
+import stateGallery from '@/app/admin/(dashboard)/_state/desa/gallery';
+import colors from '@/con/colors';
+import ApiFetch from '@/lib/api-fetch';
+import {
+ ActionIcon,
+ Box,
+ Button,
+ Group,
+ Paper,
+ Stack,
+ Text,
+ TextInput,
+ Title,
+ Loader,
+ Image
+} from '@mantine/core';
+import { Dropzone } from '@mantine/dropzone';
+import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { toast } from 'react-toastify';
+import { useProxy } from 'valtio/utils';
+
+function CreateFoto() {
+ const FotoState = useProxy(stateGallery.foto);
+ const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const [previewImage, setPreviewImage] = useState(null);
+ const [file, setFile] = useState(null);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ FotoState.create.form.name?.trim() !== '' &&
+ !isHtmlEmpty(FotoState.create.form.deskripsi) &&
+ file !== null
+ );
+ };
+
+ const resetForm = () => {
+ FotoState.create.form = {
+ name: '',
+ deskripsi: '',
+ imagesId: '',
+ };
+ setPreviewImage(null);
+ setFile(null);
+ };
+
+ const handleSubmit = async () => {
+ if (!FotoState.create.form.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(FotoState.create.form.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ if (!file) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+ if (!file) {
+ return toast.warn('Silakan pilih file gambar terlebih dahulu');
+ }
+
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
+
+ const uploaded = res.data?.data;
+ if (!uploaded?.id) {
+ return toast.error('Gagal mengunggah gambar, silakan coba lagi');
+ }
+
+ FotoState.create.form.imagesId = uploaded.id;
+
+ await FotoState.create.create();
+
+ resetForm();
+ router.push('/admin/desa/gallery/foto');
+ } catch (error) {
+ console.error('Error creating foto:', error);
+ toast.error('Terjadi kesalahan saat membuat foto');
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ return (
+
+ {/* Header Back Button + Title */}
+
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
+
+ Tambah Foto
+
+
+
+ {/* Card Form */}
+
+
+ {/* Judul */}
+ {
+ FotoState.create.form.name = e.currentTarget.value;
+ }}
+ required
+ />
+
+
+
+ Gambar Berita
+
+ {
+ const selectedFile = files[0];
+ if (selectedFile) {
+ setFile(selectedFile);
+ setPreviewImage(URL.createObjectURL(selectedFile));
+ }
+ }}
+ onReject={() => toast.error('File tidak valid, gunakan format gambar')}
+ maxSize={5 * 1024 ** 2}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
+ radius="md"
+ p="xl"
+ >
+
+
+
+
+
+
+
+
+
+
+
+
+ Seret gambar atau klik untuk memilih file (maks 5MB)
+
+
+
+ {previewImage && (
+
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
+
+ )}
+
+
+ {/* Deskripsi */}
+
+
+ Deskripsi Foto
+
+ {
+ FotoState.create.form.deskripsi = val;
+ }}
+ />
+
+
+ {/* Button Submit */}
+
+
+ Reset
+
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
+
+
+
+
+ );
+}
+
+export default CreateFoto;
diff --git a/src/app/admin/(dashboard)/desa/gallery/foto/page.tsx b/src/app/admin/(dashboard)/desa/gallery/foto/page.tsx
index 3b08fbea..09a24742 100644
--- a/src/app/admin/(dashboard)/desa/gallery/foto/page.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/foto/page.tsx
@@ -1,160 +1,218 @@
-"use client";
-import stateFileStorage from "@/state/state-list-image";
+'use client'
+import colors from '@/con/colors';
import {
- ActionIcon,
Box,
- Card,
- Flex,
+ Button,
+ Center,
Group,
- Image,
Pagination,
Paper,
- SimpleGrid,
+ Skeleton,
Stack,
+ Table,
+ TableTbody,
+ TableTd,
+ TableTh,
+ TableThead,
+ TableTr,
Text,
- TextInput,
- Title,
- Tooltip,
-} from "@mantine/core";
-import { useShallowEffect } from "@mantine/hooks";
-import { IconSearch, IconTrash, IconX } from "@tabler/icons-react";
-import { motion } from "framer-motion";
-import toast from "react-simple-toasts";
-import { useSnapshot } from "valtio";
-
-export default function ListImage() {
- const { list, total } = useSnapshot(stateFileStorage);
-
- useShallowEffect(() => {
- stateFileStorage.load();
- }, []);
-
- let timeOut: NodeJS.Timer;
+ Title
+} from '@mantine/core';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
+import { IconDeviceImac, IconPlus, IconSearch } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
+import HeaderSearch from '../../../_com/header';
+import stateGallery from '../../../_state/desa/gallery';
+function Foto() {
+ const [search, setSearch] = useState("");
return (
-
-
-
- Galeri Foto
-
- }
- rightSection={
- stateFileStorage.load()}
- >
-
-
- }
- onChange={(e) => {
- if (timeOut) clearTimeout(timeOut);
- timeOut = setTimeout(() => {
- stateFileStorage.load({ search: e.target.value });
- }, 300);
- }}
- />
-
-
-
- {list && list.length > 0 ? (
-
- {list.map((v, k) => (
-
-
- {
- navigator.clipboard.writeText(v.url);
- toast("Tautan foto berhasil disalin");
- }}
- whileHover={{ scale: 1.05 }}
- whileTap={{ scale: 0.95 }}
- style={{ cursor: "pointer" }}
- >
-
-
-
-
-
- {v.name}
-
-
-
-
-
- {
- stateFileStorage
- .del({ id: v.id })
- .finally(() => toast("Foto berhasil dihapus"));
- }}
- >
-
-
-
-
-
-
- ))}
-
- ) : (
-
-
-
- Belum ada foto yang tersedia
-
-
- )}
-
-
- {total && total > 1 && (
-
- {
- stateFileStorage.load({ page });
- }}
- />
-
-
- )}
-
+
+ }
+ value={search}
+ onChange={(e) => setSearch(e.currentTarget.value)}
+ />
+
+
);
}
+
+function ListFoto({ search }: { search: string }) {
+ const FotoState = useProxy(stateGallery.foto)
+ const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
+
+ const {
+ data,
+ page,
+ totalPages,
+ loading,
+ load,
+ } = FotoState.findMany;
+
+ useShallowEffect(() => {
+ load(page, 10, debouncedSearch)
+ }, [page, debouncedSearch])
+
+ const filteredData = data || []
+
+ if (loading || !data) {
+ return (
+
+
+
+ )
+ }
+
+ return (
+
+
+
+ Daftar Foto
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/gallery/foto/create')}
+ >
+ Tambah Baru
+
+
+
+ {/* Desktop Table */}
+
+
+
+
+ Judul Foto
+ Tanggal
+ Deskripsi
+ Aksi
+
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ {item.name}
+
+
+
+
+ {new Date(item.createdAt).toLocaleDateString('id-ID', {
+ day: 'numeric',
+ month: 'long',
+ year: 'numeric',
+ })}
+
+
+
+
+
+
+ router.push(`/admin/desa/gallery/foto/${item.id}`)}
+ >
+
+ Detail
+
+
+
+ ))
+ ) : (
+
+
+
+ Tidak ada foto yang cocok
+
+
+
+ )}
+
+
+
+
+ {/* Mobile Card View */}
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Judul Foto
+ {item.name}
+
+
+ Tanggal
+
+ {new Date(item.createdAt).toLocaleDateString('id-ID', {
+ day: 'numeric',
+ month: 'long',
+ year: 'numeric',
+ })}
+
+
+
+ Deskripsi
+
+
+ router.push(`/admin/desa/gallery/foto/${item.id}`)}
+ >
+
+ Detail
+
+
+
+ ))
+ ) : (
+
+ Tidak ada foto yang cocok
+
+ )}
+
+
+
+
+
+ {
+ load(newPage, 10)
+ window.scrollTo({ top: 0, behavior: 'smooth' })
+ }}
+ total={totalPages}
+ mt="md"
+ mb="md"
+ color="blue"
+ radius="md"
+ />
+
+
+ );
+}
+
+export default Foto;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/gallery/layout.tsx b/src/app/admin/(dashboard)/desa/gallery/layout.tsx
index fbaf56c0..1f40b5ef 100644
--- a/src/app/admin/(dashboard)/desa/gallery/layout.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/layout.tsx
@@ -1,7 +1,29 @@
'use client'
+import { usePathname } from "next/navigation";
import LayoutTabsGallery from "./lib/layoutTabs"
+import { Box } from "@mantine/core";
export default function Layout({ children }: { children: React.ReactNode }) {
+const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/berita/semua → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
+
return (
{children}
diff --git a/src/app/admin/(dashboard)/desa/gallery/lib/layoutTabs.tsx b/src/app/admin/(dashboard)/desa/gallery/lib/layoutTabs.tsx
index 803a6884..43a029bb 100644
--- a/src/app/admin/(dashboard)/desa/gallery/lib/layoutTabs.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/lib/layoutTabs.tsx
@@ -1,10 +1,10 @@
/* eslint-disable react-hooks/exhaustive-deps */
'use client'
import colors from '@/con/colors';
-import { ScrollArea, Stack, Tabs, TabsList, TabsPanel, TabsTab, Title, Tooltip } from '@mantine/core';
+import { ScrollArea, Stack, Tabs, TabsList, TabsPanel, TabsTab, Title } from '@mantine/core';
+import { IconPhoto, IconVideo } from '@tabler/icons-react';
import { usePathname, useRouter } from 'next/navigation';
import React, { useEffect, useState } from 'react';
-import { IconPhoto, IconVideo } from '@tabler/icons-react';
function LayoutTabsGallery({ children }: { children: React.ReactNode }) {
const router = useRouter()
@@ -14,15 +14,13 @@ function LayoutTabsGallery({ children }: { children: React.ReactNode }) {
label: "Foto",
value: "foto",
href: "/admin/desa/gallery/foto",
- icon: ,
- tooltip: "Kelola foto-foto galeri desa"
+ icon:
},
{
label: "Video",
value: "video",
href: "/admin/desa/gallery/video",
- icon: ,
- tooltip: "Kelola video galeri desa"
+ icon:
},
];
@@ -70,25 +68,18 @@ function LayoutTabsGallery({ children }: { children: React.ReactNode }) {
}}
>
{tabs.map((tab, i) => (
-
-
- {tab.label}
-
-
+ {tab.label}
+
))}
diff --git a/src/app/admin/(dashboard)/desa/gallery/video/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/gallery/video/[id]/edit/page.tsx
index 1b4260fd..40dfb76f 100644
--- a/src/app/admin/(dashboard)/desa/gallery/video/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/video/[id]/edit/page.tsx
@@ -4,6 +4,7 @@ import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
import stateGallery from '@/app/admin/(dashboard)/_state/desa/gallery';
import colors from '@/con/colors';
import {
+ ActionIcon,
Box,
Button,
Group,
@@ -11,11 +12,11 @@ import {
Stack,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
-import { IconArrowBack } from '@tabler/icons-react';
+import { IconArrowBack, IconX } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
-import { useEffect, useState, useCallback } from 'react';
+import { useCallback, useEffect, useState } from 'react';
import { toast } from 'react-toastify';
import { useProxy } from 'valtio/utils';
import { convertYoutubeUrlToEmbed } from '../../../lib/youtube-utils';
@@ -25,6 +26,32 @@ function EditVideo() {
const videoState = useProxy(stateGallery.video);
const params = useParams();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ const embedLink = convertYoutubeUrlToEmbed(formData.linkVideo);
+ return (
+ formData.name?.trim() !== '' &&
+ formData.linkVideo?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ embedLink !== null // Make sure the embed link is valid
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ name: "",
+ deskripsi: "",
+ linkVideo: "",
+ });
+
const [formData, setFormData] = useState({
name: '',
deskripsi: '',
@@ -45,6 +72,11 @@ function EditVideo() {
deskripsi: data.deskripsi ?? '',
linkVideo: data.linkVideo ?? '',
});
+ setOriginalData({
+ name: data.name ?? '',
+ deskripsi: data.deskripsi ?? '',
+ linkVideo: data.linkVideo ?? '',
+ });
}
} catch (error) {
console.error('Error loading video:', error);
@@ -62,43 +94,73 @@ function EditVideo() {
[]
);
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ deskripsi: originalData.deskripsi,
+ linkVideo: originalData.linkVideo,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
const handleSubmit = async () => {
- const converted = convertYoutubeUrlToEmbed(formData.linkVideo);
- if (!converted) {
- toast.error("Link YouTube tidak valid. Pastikan formatnya benar.");
+ if (!formData.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!formData.linkVideo?.trim()) {
+ toast.error('Link YouTube wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
return;
}
try {
- videoState.update.form = {
- name: formData.name,
- deskripsi: formData.deskripsi,
- linkVideo: formData.linkVideo,
- };
- await videoState.update.update();
- toast.success('Video berhasil diperbarui!');
- router.push('/admin/desa/gallery/video');
+ setIsSubmitting(true);
+ const converted = convertYoutubeUrlToEmbed(formData.linkVideo);
+ if (!converted) {
+ toast.error("Link YouTube tidak valid. Pastikan formatnya benar.");
+ return;
+ }
+
+ try {
+ videoState.update.form = {
+ name: formData.name,
+ deskripsi: formData.deskripsi,
+ linkVideo: formData.linkVideo,
+ };
+ await videoState.update.update();
+ toast.success('Video berhasil diperbarui!');
+ router.push('/admin/desa/gallery/video');
+ } catch (error) {
+ console.error('Error updating video:', error);
+ toast.error('Terjadi kesalahan saat memperbarui video');
+ }
} catch (error) {
console.error('Error updating video:', error);
toast.error('Terjadi kesalahan saat memperbarui video');
+ } finally {
+ setIsSubmitting(false);
}
};
const embedLink = convertYoutubeUrlToEmbed(formData.linkVideo);
return (
-
+
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Video
@@ -130,7 +192,7 @@ function EditVideo() {
required
/>
{embedLink && (
-
+
+ />
+ {
+ setFormData({
+ ...formData,
+ linkVideo: '',
+ });
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -154,17 +236,31 @@ function EditVideo() {
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/gallery/video/[id]/page.tsx b/src/app/admin/(dashboard)/desa/gallery/video/[id]/page.tsx
index 65a579a0..5400f6b0 100644
--- a/src/app/admin/(dashboard)/desa/gallery/video/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/video/[id]/page.tsx
@@ -2,7 +2,7 @@
import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
import stateGallery from '@/app/admin/(dashboard)/_state/desa/gallery';
import colors from '@/con/colors';
-import { Box, Button, Group, Paper, Skeleton, Stack, Text, Tooltip } from '@mantine/core';
+import { Box, Button, Group, Paper, Skeleton, Stack, Text } from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -40,7 +40,7 @@ function DetailVideo() {
const data = videoState.findUnique.data;
return (
-
+
{/* Tombol Kembali */}
-
{
@@ -124,9 +123,7 @@ function DetailVideo() {
>
-
-
@@ -138,7 +135,6 @@ function DetailVideo() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/gallery/video/create/page.tsx b/src/app/admin/(dashboard)/desa/gallery/video/create/page.tsx
index 9badd230..ae0f3870 100644
--- a/src/app/admin/(dashboard)/desa/gallery/video/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/video/create/page.tsx
@@ -3,6 +3,7 @@ import CreateEditor from '@/app/admin/(dashboard)/_com/createEditor';
import stateGallery from '@/app/admin/(dashboard)/_state/desa/gallery';
import colors from '@/con/colors';
import {
+ ActionIcon,
Box,
Button,
Group,
@@ -11,9 +12,9 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader
} from '@mantine/core';
-import { IconArrowBack } from '@tabler/icons-react';
+import { IconArrowBack, IconX } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
import { toast } from 'react-toastify';
@@ -25,6 +26,24 @@ function CreateVideo() {
const router = useRouter();
const [link, setLink] = useState('');
const embedLink = convertYoutubeUrlToEmbed(link);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ videoState.create.form.name?.trim() !== '' &&
+ link.trim() !== '' &&
+ !isHtmlEmpty(videoState.create.form.deskripsi) &&
+ embedLink !== null // Make sure the embed link is valid
+ );
+ };
const resetForm = () => {
videoState.create.form = {
@@ -36,31 +55,57 @@ function CreateVideo() {
};
const handleSubmit = async () => {
+ if (!videoState.create.form.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!link.trim()) {
+ toast.error('Link YouTube wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(videoState.create.form.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
if (!embedLink) {
toast.error('Link YouTube tidak valid. Pastikan formatnya benar.');
return;
}
- videoState.create.form.linkVideo = embedLink;
- await videoState.create.create();
- resetForm();
- router.push('/admin/desa/gallery/video');
+ try {
+ setIsSubmitting(true);
+ if (!embedLink) {
+ toast.error('Link YouTube tidak valid. Pastikan formatnya benar.');
+ return;
+ }
+
+ videoState.create.form.linkVideo = embedLink;
+ await videoState.create.create();
+ resetForm();
+ router.push('/admin/desa/gallery/video');
+ } catch (error) {
+ console.error("Error creating video:", error);
+ toast.error("Terjadi kesalahan saat menambahkan video");
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header Back Button + Title */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Tambah Video
@@ -80,7 +125,7 @@ function CreateVideo() {
{
videoState.create.form.name = e.currentTarget.value;
}}
@@ -91,14 +136,14 @@ function CreateVideo() {
setLink(e.currentTarget.value)}
required
/>
{/* Preview Video */}
{embedLink && (
-
+
+ />
+ {
+ setLink('');
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -128,17 +190,31 @@ function CreateVideo() {
{/* Button Submit */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/gallery/video/page.tsx b/src/app/admin/(dashboard)/desa/gallery/video/page.tsx
index 232204ff..80c297bb 100644
--- a/src/app/admin/(dashboard)/desa/gallery/video/page.tsx
+++ b/src/app/admin/(dashboard)/desa/gallery/video/page.tsx
@@ -16,10 +16,9 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
import { IconDeviceImac, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
@@ -46,6 +45,7 @@ function Video() {
function ListVideo({ search }: { search: string }) {
const videoState = useProxy(stateGallery.video)
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const {
data,
@@ -56,77 +56,79 @@ function ListVideo({ search }: { search: string }) {
} = videoState.findMany;
useShallowEffect(() => {
- load(page, 10, search)
- }, [page, search])
+ load(page, 10, debouncedSearch)
+ }, [page, debouncedSearch])
const filteredData = data || []
if (loading || !data) {
return (
-
+
)
}
return (
-
-
-
- Daftar Video
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/gallery/video/create')}
- >
- Tambah Baru
-
-
+
+
+
+
+ Daftar Video
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/gallery/video/create')}
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Judul Video
- Tanggal
- Deskripsi
- Aksi
+ Judul Video
+ Tanggal
+ Deskripsi
+ Aksi
{filteredData.length > 0 ? (
filteredData.map((item) => (
-
-
- {item.name}
-
+
+
+ {item.name}
+
-
-
-
+
+
{new Date(item.createdAt).toLocaleDateString('id-ID', {
day: 'numeric',
month: 'long',
year: 'numeric',
})}
-
-
-
-
-
+
+
-
+
router.push(`/admin/desa/gallery/video/${item.id}`)}
+ fz="sm"
+ px="xs"
>
-
+
Detail
@@ -135,8 +137,10 @@ function ListVideo({ search }: { search: string }) {
) : (
-
- Tidak ada video yang cocok
+
+
+ Tidak ada video yang cocok
+
@@ -144,8 +148,60 @@ function ListVideo({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Judul Video
+
+ {item.name}
+
+
+
+ Tanggal
+
+ {new Date(item.createdAt).toLocaleDateString('id-ID', {
+ day: 'numeric',
+ month: 'long',
+ year: 'numeric',
+ })}
+
+
+
+ Deskripsi
+
+
+
+ router.push(`/admin/desa/gallery/video/${item.id}`)}
+ fz="sm"
+ >
+
+ Detail
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada video yang cocok
+
+
+ )}
+
-
+
+
+
{
@@ -153,8 +209,6 @@ function ListVideo({ search }: { search: string }) {
window.scrollTo({ top: 0, behavior: 'smooth' })
}}
total={totalPages}
- mt="md"
- mb="md"
color="blue"
radius="md"
/>
@@ -163,4 +217,4 @@ function ListVideo({ search }: { search: string }) {
);
}
-export default Video;
+export default Video;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/edit/page.tsx
index bd56c1c3..14f2605f 100644
--- a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/edit/page.tsx
@@ -6,12 +6,12 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Select,
Stack,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -24,6 +24,16 @@ function EditAjukanPermohonan() {
const params = useParams();
const stateAjukan = useProxy(stateLayananDesa.ajukanPermohonan);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ const [originalData, setOriginalData] = useState({
+ nama: "",
+ nik: "",
+ alamat: "",
+ nomorKk: "",
+ kategoriId: "",
+ });
+
// State lokal form
const [formData, setFormData] = useState({
nama: '',
@@ -51,6 +61,13 @@ function EditAjukanPermohonan() {
nomorKk: data.nomorKk || '',
kategoriId: data.kategoriId || '',
});
+ setOriginalData({
+ nama: data.nama || '',
+ nik: data.nik || '',
+ alamat: data.alamat || '',
+ nomorKk: data.nomorKk || '',
+ kategoriId: data.kategoriId || '',
+ });
}
} catch (error) {
console.error('Error loading ajukan:', error);
@@ -69,8 +86,20 @@ function EditAjukanPermohonan() {
}));
};
+ const handleResetForm = () => {
+ setFormData({
+ nama: originalData.nama,
+ nik: originalData.nik,
+ alamat: originalData.alamat,
+ nomorKk: originalData.nomorKk,
+ kategoriId: originalData.kategoriId,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
const handleSubmit = async () => {
try {
+ setIsSubmitting(true);
stateAjukan.edit.form = {
...stateAjukan.edit.form,
...formData,
@@ -80,18 +109,18 @@ function EditAjukanPermohonan() {
} catch (error) {
console.error('Error updating ajukan:', error);
toast.error('Terjadi kesalahan saat memperbarui ajukan');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Back Button */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Edit Ajukan Permohonan
@@ -156,6 +185,17 @@ function EditAjukanPermohonan() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/page.tsx b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/page.tsx
index 7d10ba08..9636c485 100644
--- a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/[id]/page.tsx
@@ -9,8 +9,7 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip
+ Text
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
@@ -49,7 +48,7 @@ function DetailAjukanPermohonan() {
const data = ajukanPermohonanState.findUnique.data;
return (
-
+
{/* Tombol Kembali */}
-
{
@@ -135,9 +133,7 @@ function DetailAjukanPermohonan() {
>
-
-
@@ -151,7 +147,6 @@ function DetailAjukanPermohonan() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/page.tsx b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/page.tsx
index b13d2a44..b008f6a8 100644
--- a/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/ajukan_permohonan/page.tsx
@@ -24,6 +24,7 @@ import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import stateLayananDesa from '../../../_state/desa/layananDesa';
+import { useDebouncedValue } from '@mantine/hooks';
function AjukanPermohonan() {
const [search, setSearch] = useState("");
@@ -44,6 +45,7 @@ function AjukanPermohonan() {
function ListAjukanPermohonan({ search }: { search: string }) {
const AjukanPermohonanState = useProxy(stateLayananDesa.ajukanPermohonan);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000); // 500ms delay
const {
data,
@@ -54,58 +56,58 @@ function ListAjukanPermohonan({ search }: { search: string }) {
} = AjukanPermohonanState.findMany;
useEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
// Loading state
if (loading || !data) {
return (
-
+
);
}
return (
-
-
- List Ajukan Permohonan
-
-
+
+
+
+ List Ajukan Permohonan
+
+
+ {/* Desktop Table */}
+
+
- Nama
- Alamat
- NIK
- Aksi
+ Nama
+ Alamat
+ NIK
+ Aksi
{data.length > 0 ? (
data.map((item) => (
-
-
-
- {item.nama}
-
-
+
+
+ {item.nama}
+
-
-
-
- {item.alamat}
-
-
+
+
+ {item.alamat}
+
-
-
-
- {item.nik}
-
-
+
+
+ {item.nik}
+
-
+
-
-
- Tidak ada data ajukan permohonan yang cocok
+
+
+
+ Tidak ada data ajukan permohonan yang cocok
+
@@ -133,23 +137,69 @@ function ListAjukanPermohonan({ search }: { search: string }) {
+
+ {/* Mobile Card View */}
+
+
+ {data.length > 0 ? (
+ data.map((item) => (
+
+
+
+ Nama
+ {item.nama}
+
+
+ Alamat
+ {item.alamat}
+
+
+ NIK
+ {item.nik}
+
+
+ }
+ onClick={() =>
+ router.push(`/admin/desa/layanan/ajukan_permohonan/${item.id}`)
+ }
+ fullWidth
+ >
+ Detail
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data ajukan permohonan yang cocok
+
+
+ )}
+
+
-
- {
- load(newPage, 10, search);
- window.scrollTo({ top: 0, behavior: 'smooth' });
- }}
- total={totalPages}
- mt="md"
- mb="md"
- color="blue"
- radius="md"
- />
-
+
+
+ {
+ load(newPage, 10, search);
+ window.scrollTo({ top: 0, behavior: 'smooth' });
+ }}
+ total={totalPages}
+ color="blue"
+ radius="md"
+ />
+
);
}
-export default AjukanPermohonan;
+export default AjukanPermohonan;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/layout.tsx b/src/app/admin/(dashboard)/desa/layanan/layout.tsx
index 7113ca7e..c4413f87 100644
--- a/src/app/admin/(dashboard)/desa/layanan/layout.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/layout.tsx
@@ -1,10 +1,31 @@
'use client'
+import { usePathname } from "next/navigation";
import LayoutTabsLayanan from "../_com/layoutTabLayanan";
+import { Box } from "@mantine/core";
export default function Layout({children} : {children: React.ReactNode}) {
+ const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/layanan/semua → panjang 5 → list
+ // - /darmasaba/desa/layanan/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/layanan/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
return (
-
+
{children}
-
- )
+
+ );
+ }
+
+ return (
+
+ {children}
+
+ );
}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/[id]/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/[id]/page.tsx
index 7f26b666..3b3a6e81 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/[id]/page.tsx
@@ -8,12 +8,12 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -33,6 +33,29 @@ function EditPelayananPendudukNonPermanent() {
deskripsi: '',
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi)
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ name: '',
+ deskripsi: '',
+ });
+
+
// Load data sekali dari backend
useEffect(() => {
const loadData = async () => {
@@ -46,6 +69,10 @@ function EditPelayananPendudukNonPermanent() {
name: data.name || '',
deskripsi: data.deskripsi || '',
});
+ setOriginalData({
+ name: data.name || '',
+ deskripsi: data.deskripsi || '',
+ });
}
} catch (error) {
console.error('Error loading data:', error);
@@ -58,41 +85,65 @@ function EditPelayananPendudukNonPermanent() {
const handleChange =
(field: keyof typeof formData) =>
- (value: string) => {
- setFormData((prev) => ({
- ...prev,
- [field]: value,
- }));
- };
+ (value: string) => {
+ setFormData((prev) => ({
+ ...prev,
+ [field]: value,
+ }));
+ };
+
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ deskripsi: originalData.deskripsi,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
const handleSubmit = async () => {
- if (!statePendudukNonPermanent.findById.data) return;
+ if (!formData.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
- // Update global state hanya di submit
- const updated = {
- ...statePendudukNonPermanent.findById.data,
- name: formData.name,
- deskripsi: formData.deskripsi,
- };
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
- await statePendudukNonPermanent.update.update(updated);
- router.push('/admin/desa/layanan/pelayanan_penduduk_non_permanent');
+ try {
+ setIsSubmitting(true);
+ if (!statePendudukNonPermanent.findById.data) return;
+
+ // Update global state hanya di submit
+ const updated = {
+ ...statePendudukNonPermanent.findById.data,
+ name: formData.name,
+ deskripsi: formData.deskripsi,
+ };
+
+ await statePendudukNonPermanent.update.update(updated);
+ router.push('/admin/desa/layanan/pelayanan_penduduk_non_permanent');
+ } catch (error) {
+ console.error('Error updating data:', error);
+ toast.error('Gagal memuat data pelayanan penduduk non permanent');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Pelayanan Penduduk Non Permanent
@@ -130,25 +181,34 @@ function EditPelayananPendudukNonPermanent() {
{/* Submit Button */}
-
-
- {statePendudukNonPermanent.update.loading
- ? 'Menyimpan...'
- : 'Simpan Perubahan'}
-
-
+
+ {/* Tombol Batal */}
router.back()}
- disabled={statePendudukNonPermanent.update.loading}
+ color="gray"
+ radius="md"
+ size="md"
+ onClick={handleResetForm}
>
Batal
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/page.tsx
index e5ce63bd..48a3e4dd 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_penduduk_non_permanent/page.tsx
@@ -5,14 +5,12 @@ import {
Button,
Center,
Divider,
- Grid,
- GridCol,
+ Group,
Paper,
Skeleton,
Stack,
Text,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconEdit } from '@tabler/icons-react';
@@ -44,43 +42,42 @@ function PelayananPendudukNonPermanent() {
{/* Header */}
-
-
-
- Preview Pelayanan Penduduk Non Permanen
-
-
-
-
- }
- radius="md"
- onClick={() =>
- router.push(
- `/admin/desa/layanan/pelayanan_penduduk_non_permanent/${data.id}`
- )
- }
- >
- Edit
-
-
-
-
+
+
+
+ Preview Pelayanan Penduduk Non Permanen
+
+ }
+ radius="md"
+ onClick={() =>
+ router.push(
+ `/admin/desa/layanan/pelayanan_penduduk_non_permanent/${data.id}`
+ )
+ }
+ >
+ Edit
+
+
{/* Content */}
-
{data.name}
-
+
@@ -89,9 +86,11 @@ function PelayananPendudukNonPermanent() {
@@ -101,4 +100,4 @@ function PelayananPendudukNonPermanent() {
);
}
-export default PelayananPendudukNonPermanent;
+export default PelayananPendudukNonPermanent;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/[id]/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/[id]/page.tsx
index e8f4a8b6..6c10fc9f 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/[id]/page.tsx
@@ -8,12 +8,12 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Skeleton,
Stack,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -35,13 +35,37 @@ function EditPelayananPerizinanBerusaha() {
link: '',
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi)
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ id: '',
+ name: '',
+ deskripsi: '',
+ link: '',
+ });
+
// Load data detail
useEffect(() => {
if (!id) {
toast.error("ID tidak valid");
return;
}
-
+
const loadData = async () => {
try {
setLoading(true);
@@ -53,6 +77,12 @@ function EditPelayananPerizinanBerusaha() {
deskripsi: data.deskripsi || "",
link: data.link || "",
});
+ setOriginalData({
+ id: data.id,
+ name: data.name || "",
+ deskripsi: data.deskripsi || "",
+ link: data.link || "",
+ });
} else {
toast.error("Data tidak ditemukan");
}
@@ -63,10 +93,10 @@ function EditPelayananPerizinanBerusaha() {
setLoading(false);
}
};
-
+
loadData();
}, [id]);
-
+
const handleChange =
(field: keyof typeof formData) =>
@@ -77,13 +107,41 @@ function EditPelayananPerizinanBerusaha() {
}));
};
+ const handleResetForm = () => {
+ setFormData({
+ id: originalData.id,
+ name: originalData.name,
+ deskripsi: originalData.deskripsi,
+ link: originalData.link,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
const handleSubmit = async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ if (!formData.link?.trim()) {
+ toast.error('Link wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
await state.update.update(formData);
router.push('/admin/desa/layanan/pelayanan_perizinan_berusaha');
} catch (error) {
console.error('Error updating pelayanan perizinan berusaha:', error);
toast.error('Terjadi kesalahan saat update data');
+ } finally {
+ setIsSubmitting(false);
}
};
@@ -96,20 +154,18 @@ function EditPelayananPerizinanBerusaha() {
}
return (
-
+
{/* Header */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Pelayanan Perizinan Berusaha
@@ -150,23 +206,34 @@ function EditPelayananPerizinanBerusaha() {
/>
-
-
- {state.update.loading ? 'Menyimpan...' : 'Simpan Perubahan'}
-
-
+
+ {/* Tombol Batal */}
router.back()}
- disabled={state.update.loading}
+ color="gray"
+ radius="md"
+ size="md"
+ onClick={handleResetForm}
>
Batal
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/page.tsx
index 3edb28cd..712eadc1 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_perizinan_berusaha/page.tsx
@@ -6,8 +6,6 @@ import {
Button,
Center,
Divider,
- Grid,
- GridCol,
Group,
Paper,
Skeleton,
@@ -16,14 +14,13 @@ import {
StepperCompleted,
StepperStep,
Text,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { IconEdit } from '@tabler/icons-react';
-import { useEffect, useState } from 'react';
-import stateLayananDesa from '../../../_state/desa/layananDesa';
-import { useProxy } from 'valtio/utils';
import { useRouter } from 'next/navigation';
+import { useEffect, useState } from 'react';
+import { useProxy } from 'valtio/utils';
+import stateLayananDesa from '../../../_state/desa/layananDesa';
function PerizinanBerusaha() {
const router = useRouter();
@@ -42,8 +39,7 @@ function PerizinanBerusaha() {
const loadData = async () => {
try {
setLoading(true);
- // You should get the ID from your router query or params
- const id = 'edit'; // Replace with actual ID or get from URL params
+ const id = 'edit';
await pelayananPerizinanBerusaha.findById.load(id);
} catch (err) {
setError('Gagal memuat data');
@@ -67,7 +63,7 @@ function PerizinanBerusaha() {
if (error || !pelayananPerizinanBerusaha.findById.data) {
return (
- {error || 'Data tidak ditemukan'}
+ {error || 'Data tidak ditemukan'}
);
}
@@ -78,67 +74,63 @@ function PerizinanBerusaha() {
{/* Header */}
-
-
-
- Preview Pelayanan Perizinan Berusaha
-
-
-
-
- }
- radius="md"
- onClick={() =>
- router.push(
- `/admin/desa/layanan/pelayanan_perizinan_berusaha/${data.id}`
- )
- }
- >
- Edit
-
-
-
-
+
+
+ Preview Pelayanan Perizinan Berusaha
+
+ }
+ radius="md"
+ onClick={() =>
+ router.push(
+ `/admin/desa/layanan/pelayanan_perizinan_berusaha/${data.id}`
+ )
+ }
+ >
+ Edit
+
+
{/* Content */}
-
{data.name}
-
+
Proses pendaftaran NIB melalui OSS mencakup beberapa langkah
umum:
-
+
- Pendaftaran akun pada portal OSS
+
+ Pendaftaran akun pada portal OSS
+
- Mengisi informasi perusahaan, termasuk data pemegang saham, alamat perusahaan, dan lainnya
+
+ Mengisi informasi perusahaan, termasuk data pemegang saham, alamat perusahaan, dan lainnya
+
- Memilih KBLI dengan jenis usaha yang akan didaftarkan
+
+ Memilih KBLI dengan jenis usaha yang akan didaftarkan
+
- Mengunggah dokumen-dokumen yang diperlukan, seperti akta pendirian perusahaan, surat izin usaha, dan dokumen lainnya sesuai dengan ketentuan yang berlaku
+
+ Mengunggah dokumen-dokumen yang diperlukan, seperti akta pendirian perusahaan, surat izin usaha, dan dokumen lainnya sesuai dengan ketentuan yang berlaku
+
- Proses verifikasi dan persetujuan oleh instansi terkait
+
+ Proses verifikasi dan persetujuan oleh instansi terkait
+
- Jika proses sebelumnya berhasil, perusahaan akan menerima NIB sebagai identitas resmi usaha anda
+
+ Jika proses sebelumnya berhasil, perusahaan akan menerima NIB sebagai identitas resmi usaha anda
+
- Selesai, anda telah mengikuti proses pendaftaran NIB melalui OSS
+
+ Selesai, anda telah mengikuti proses pendaftaran NIB melalui OSS
+
+
+
+
+
+
+ Back
+
+ Next step
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
@@ -180,9 +259,10 @@ function PerizinanBerusaha() {
Penting untuk diingat bahwa prosedur dan persyaratan dapat
berubah seiring waktu. Untuk informasi yang lebih akurat dan
@@ -206,5 +286,4 @@ function PerizinanBerusaha() {
);
}
-export default PerizinanBerusaha;
-
+export default PerizinanBerusaha;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/edit/page.tsx
index 092eadcd..938488f2 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/edit/page.tsx
@@ -1,127 +1,311 @@
-'use client'
+/* eslint-disable @typescript-eslint/no-unused-vars */
+'use client';
import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
import stateLayananDesa from '@/app/admin/(dashboard)/_state/desa/layananDesa';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
import {
+ ActionIcon,
Box,
Button,
Group,
Image,
+ Loader,
Paper,
Stack,
Text,
TextInput,
Title,
- Tooltip,
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
-import { useEffect, useState, useCallback } from 'react';
+import { useCallback, useEffect, useState } from 'react';
import { toast } from 'react-toastify';
-import { useProxy } from 'valtio/utils';
+import * as React from 'react';
+// 🔹 Types
+interface FormData {
+ name: string;
+ deskripsi: string;
+ imageId: string;
+ image2Id: string;
+ imageUrl: string;
+ image2Url: string;
+}
+
+interface FileUploaderProps {
+ title: string;
+ file: File | null;
+ setFile: React.Dispatch>;
+ preview: string | null;
+ setPreview: React.Dispatch>;
+}
+
+// 🔹 File Uploader Component
+const FileUploader: React.FC = ({
+ title,
+ file,
+ setFile,
+ preview,
+ setPreview
+}) => {
+ const handleDrop = (files: File[]) => {
+ const selected = files[0];
+ if (selected) {
+ setFile(selected);
+ setPreview(URL.createObjectURL(selected));
+ }
+ };
+
+ const handleRemove = () => {
+ setPreview(null);
+ setFile(null);
+ };
+
+ return (
+
+
+ {title}
+
+ toast.error('File tidak valid, gunakan format gambar')}
+ maxSize={5 * 1024 ** 2}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
+ radius="md"
+ p="xl"
+ >
+
+
+
+
+
+
+
+
+
+
+
+
+ Seret gambar atau klik untuk memilih file
+
+
+ Maksimal 5MB, format .png, .jpg, .jpeg, .webp
+
+
+
+
+
+ {preview && (
+
+
+
+
+
+
+ )}
+
+ );
+};
+
+// 🔹 Main Component
function EditSuratKeterangan() {
const router = useRouter();
const params = useParams();
- const stateSurat = useProxy(stateLayananDesa.suratKeterangan);
- // state lokal untuk form
- const [formData, setFormData] = useState({
+ // 🧩 State
+ const [formData, setFormData] = useState({
name: '',
deskripsi: '',
imageId: '',
image2Id: '',
+ imageUrl: '',
+ image2Url: '',
});
-
- // state file upload
+ const [originalData, setOriginalData] = useState(formData);
const [file, setFile] = useState(null);
const [file2, setFile2] = useState(null);
-
- // state preview gambar
const [previewImage, setPreviewImage] = useState(null);
const [previewImage2, setPreviewImage2] = useState(null);
+ const [isSubmitting, setIsSubmitting] = useState(false);
- // load data awal
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ (previewImage !== null || originalData.imageId !== '') // Either a new file is selected or an existing image exists
+ );
+ };
+
+ // 🧭 Load Initial Data
useEffect(() => {
const loadSurat = async () => {
const id = params?.id as string;
if (!id) return;
try {
- const data = await stateSurat.edit.load(id);
+ const data = await stateLayananDesa.suratKeterangan.edit.load(id);
if (!data) return;
- setFormData((prev) => ({
- ...prev,
- ...{
- name: prev.name || data.name || "",
- deskripsi: prev.deskripsi || data.deskripsi || "",
- imageId: prev.imageId || data.imageId || "",
- image2Id: prev.image2Id || data.image2Id || "",
- },
- }));
+ const mapped: FormData = {
+ name: data.name || '',
+ deskripsi: data.deskripsi || '',
+ imageId: data.imageId || '',
+ image2Id: data.image2Id || '',
+ imageUrl: data.image?.link || '',
+ image2Url: data.image2?.link || ''
+ };
- if (data.image?.link && !previewImage) setPreviewImage(data.image.link);
- if (data.image2?.link && !previewImage2) setPreviewImage2(data.image2.link);
+ setFormData(mapped);
+ setOriginalData(mapped);
+
+ if (data.image?.link) setPreviewImage(data.image.link);
+ if (data.image2?.link) setPreviewImage2(data.image2.link);
} catch (error) {
- console.error("Error loading surat:", error);
- toast.error("Gagal memuat data surat");
+ console.error('Error loading surat:', error);
+ toast.error('Gagal memuat data surat');
}
};
loadSurat();
- // eslint-disable-next-line react-hooks/exhaustive-deps
}, [params?.id]);
-
-
- // handler untuk submit
- const handleSubmit = useCallback(async () => {
+ // 📤 Upload File Helper
+ const uploadFile = async (file: File): Promise => {
try {
- // update form global hanya saat submit
- stateSurat.edit.form = { ...stateSurat.edit.form, ...formData };
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
+ const uploaded = res.data?.data;
+ return uploaded?.id || null;
+ } catch (error) {
+ console.error('Error uploading file:', error);
+ return null;
+ }
+ };
- // upload file 1
+ // 🔁 Reset Form
+ const handleResetForm = () => {
+ setFormData(originalData);
+ setPreviewImage(originalData.imageUrl || null);
+ setPreviewImage2(originalData.image2Url || null);
+ setFile(null);
+ setFile2(null);
+ toast.info('Form dikembalikan ke data awal');
+ };
+
+ // 💾 Submit Handler
+ const handleSubmit = useCallback(async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Nama surat keterangan wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Konten wajib diisi');
+ return;
+ }
+
+ if (!previewImage && !originalData.imageId) {
+ toast.error('Gambar konten pelayanan wajib dipilih');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+
+ // ✅ Access original state directly (not proxy)
+ const originalState = stateLayananDesa.suratKeterangan;
+
+ // Update form data properties individually
+ originalState.edit.form.name = formData.name;
+ originalState.edit.form.deskripsi = formData.deskripsi;
+ originalState.edit.form.imageId = formData.imageId;
+ originalState.edit.form.image2Id = formData.image2Id;
+
+ // Upload file 1 if exists
if (file) {
- const res = await ApiFetch.api.fileStorage.create.post({ file, name: file.name });
- const uploaded = res.data?.data;
- if (!uploaded?.id) return toast.error('Gagal upload gambar');
- stateSurat.edit.form.imageId = uploaded.id;
+ const uploadedId = await uploadFile(file);
+ if (!uploadedId) {
+ toast.error('Gagal upload gambar pertama');
+ return;
+ }
+ originalState.edit.form.imageId = uploadedId;
}
- // upload file 2
+ // Upload file 2 if exists
if (file2) {
- const res = await ApiFetch.api.fileStorage.create.post({ file: file2, name: file2.name });
- const uploaded = res.data?.data;
- if (!uploaded?.id) return toast.error('Gagal upload gambar');
- stateSurat.edit.form.image2Id = uploaded.id;
+ const uploadedId = await uploadFile(file2);
+ if (!uploadedId) {
+ toast.error('Gagal upload gambar kedua');
+ return;
+ }
+ originalState.edit.form.image2Id = uploadedId;
}
- await stateSurat.edit.update();
+ // Submit update
+ await originalState.edit.update();
toast.success('Surat berhasil diperbarui!');
router.push('/admin/desa/layanan/pelayanan_surat_keterangan');
} catch (error) {
console.error('Error updating surat:', error);
toast.error('Terjadi kesalahan saat memperbarui surat');
+ } finally {
+ setIsSubmitting(false);
}
- }, [formData, file, file2, router, stateSurat.edit]);
+ }, [formData, file, file2, router, previewImage, originalData.imageId]);
+
+ // 📝 Form Field Handlers
+ const handleNameChange = (e: React.ChangeEvent) => {
+ setFormData(prev => ({ ...prev, name: e.target.value }));
+ };
+
+ const handleDeskripsiChange = (html: string) => {
+ setFormData(prev => ({ ...prev, deskripsi: html }));
+ };
return (
- {/* Back Button */}
+ {/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Edit Surat Keterangan
+ {/* Form */}
- {/* Input nama */}
+ {/* Nama Surat */}
setFormData((prev) => ({ ...prev, name: e.target.value }))}
+ onChange={handleNameChange}
required
/>
- {/* Input deskripsi */}
+ {/* Deskripsi */}
Konten
- setFormData((prev) => ({ ...prev, deskripsi: htmlContent }))
- }
+ onChange={handleDeskripsiChange}
/>
- {/* Upload Gambar 1 */}
-
-
- Gambar Konten Pelayanan
-
- {
- const selectedFile = files[0];
- if (selectedFile) {
- setFile(selectedFile);
- setPreviewImage(URL.createObjectURL(selectedFile));
- }
- }}
- onReject={() => toast.error('File tidak valid, gunakan format gambar')}
- maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
- radius="md"
- p="xl"
- >
-
-
-
-
-
-
-
-
-
-
-
-
- Seret gambar atau klik untuk memilih file
-
-
- Maksimal 5MB, format gambar wajib
-
-
-
-
+ {/* Gambar 1 */}
+
- {previewImage && (
-
-
-
- )}
-
-
- {/* Upload Gambar 2 */}
-
-
- Gambar Alur Pelayanan Surat
-
- {
- const selectedFile = files[0];
- if (selectedFile) {
- setFile2(selectedFile);
- setPreviewImage2(URL.createObjectURL(selectedFile));
- }
- }}
- onReject={() => toast.error('File tidak valid, gunakan format gambar')}
- maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
- radius="md"
- p="xl"
- >
-
-
-
-
-
-
-
-
-
-
-
-
- Seret gambar atau klik untuk memilih file
-
-
- Maksimal 5MB, format gambar wajib
-
-
-
-
-
- {previewImage2 && (
-
-
-
- )}
-
+ {/* Gambar 2 */}
+
+ {/* Action Buttons */}
+
+ Batal
+
- Simpan
+ {isSubmitting ? : 'Simpan'}
@@ -287,4 +385,4 @@ function EditSuratKeterangan() {
);
}
-export default EditSuratKeterangan;
+export default EditSuratKeterangan;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/page.tsx
index 59d992ac..06b5af26 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/[id]/page.tsx
@@ -10,8 +10,7 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip,
+ Text
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
@@ -50,7 +49,7 @@ function DetailSuratKeterangan() {
const data = suratKeteranganState.findUnique.data;
return (
-
+
{/* Tombol Kembali */}
-
+
Nama
{data?.name || '-'}
-
+
-
+
Deskripsi
-
+
-
+
+
-
+
Gambar Konten Pelayanan
@@ -118,7 +119,7 @@ function DetailSuratKeterangan() {
Tidak ada gambar
)}
-
+
@@ -142,7 +143,6 @@ function DetailSuratKeterangan() {
-
{
@@ -156,9 +156,7 @@ function DetailSuratKeterangan() {
>
-
-
@@ -172,7 +170,6 @@ function DetailSuratKeterangan() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/create/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/create/page.tsx
index ba52a647..642f9449 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/create/page.tsx
@@ -5,16 +5,17 @@ import stateLayananDesa from '@/app/admin/(dashboard)/_state/desa/layananDesa';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
import {
+ ActionIcon,
Box,
Button,
Group,
Image,
+ Loader,
Paper,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
@@ -28,6 +29,23 @@ function CreateSuratKeterangan() {
const [previewImage2, setPreviewImage2] = useState<{ preview: string; file: File } | null>(null);
const [previewImage, setPreviewImage] = useState<{ preview: string; file: File } | null>(null);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ stateSurat.create.form.name?.trim() !== '' &&
+ !isHtmlEmpty(stateSurat.create.form.deskripsi) &&
+ previewImage !== null
+ );
+ };
const resetForm = () => {
stateSurat.create.form = {
@@ -41,11 +59,23 @@ function CreateSuratKeterangan() {
};
const handleSubmit = async () => {
+ if (!stateSurat.create.form.name?.trim()) {
+ toast.error('Nama surat keterangan wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(stateSurat.create.form.deskripsi)) {
+ toast.error('Konten wajib diisi');
+ return;
+ }
+
if (!previewImage) {
- return toast.warn('Pilih file gambar utama terlebih dahulu');
+ toast.error('Gambar konten pelayanan wajib dipilih');
+ return;
}
try {
+ setIsSubmitting(true);
// Upload gambar utama
const res1 = await ApiFetch.api.fileStorage.create.post({
file: previewImage.file,
@@ -78,18 +108,18 @@ function CreateSuratKeterangan() {
} catch (error) {
console.error('Error creating surat keterangan:', error);
toast.error('Terjadi kesalahan saat menambahkan surat keterangan');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Tambah Surat Keterangan
@@ -106,7 +136,7 @@ function CreateSuratKeterangan() {
{/* Nama Surat */}
(stateSurat.create.form.name = val.target.value)}
label="Nama Surat Keterangan"
placeholder="Masukkan nama surat keterangan"
@@ -143,7 +173,7 @@ function CreateSuratKeterangan() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -164,7 +194,7 @@ function CreateSuratKeterangan() {
{previewImage && (
-
+
+ {
+ setPreviewImage(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -193,7 +240,7 @@ function CreateSuratKeterangan() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -214,7 +261,7 @@ function CreateSuratKeterangan() {
{previewImage2 ? (
-
+
+ {
+ setPreviewImage2(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
) : (
@@ -232,17 +296,31 @@ function CreateSuratKeterangan() {
{/* Tombol Simpan */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/page.tsx
index e4948edf..8a67870a 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_surat_keterangan/page.tsx
@@ -18,7 +18,6 @@ import {
TableTr,
Text,
Title,
- Tooltip
} from '@mantine/core';
import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
@@ -26,9 +25,10 @@ import { useEffect, useMemo, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import stateLayananDesa from '../../../_state/desa/layananDesa';
+import { useDebouncedValue } from '@mantine/hooks';
function SuratKeterangan() {
- const [search, setSearch] = useState("");
+ const [search, setSearch] = useState('');
return (
{
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const filteredData = useMemo(() => {
if (!data) return [];
- const keyword = search.toLowerCase();
- return data.filter(item =>
- item.name?.toLowerCase().includes(keyword) ||
- item.deskripsi?.toLowerCase().includes(keyword)
+ const keyword = debouncedSearch.toLowerCase();
+ return data.filter(
+ (item) =>
+ item.name?.toLowerCase().includes(keyword) ||
+ item.deskripsi?.toLowerCase().includes(keyword)
);
- }, [data, search]);
+ }, [data, debouncedSearch]);
- // Loading state
if (loading || !data) {
return (
-
+
);
}
return (
-
-
-
- List Surat Keterangan
-
- }
- color="blue"
- variant="light"
- onClick={() =>
- router.push('/admin/desa/layanan/pelayanan_surat_keterangan/create')
- }
- >
- Tambah Baru
-
-
+
+
+
+
+ List Surat Keterangan
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/desa/layanan/pelayanan_surat_keterangan/create')
+ }
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Nama
- Deskripsi
- Aksi
-
-
-
- {filteredData.length > 0 ? (
- filteredData.map((item) => (
-
-
-
-
+
+ Nama
+
+
+ Deskripsi
+
+
+ Aksi
+
+
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
{item.name}
-
+
+
+
+
+
+ }
+ onClick={() =>
+ router.push(
+ `/admin/desa/layanan/pelayanan_surat_keterangan/${item.id}`
+ )
+ }
+ >
+ Detail
+
+
+
+ ))
+ ) : (
+
+
+
+
+ Tidak ada data surat keterangan yang cocok
+
+
-
-
-
+ )}
+
+
+
+
+ {/* Mobile Cards */}
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+
+ Nama
+
+
+ {item.name}
+
+
+
+
+ Deskripsi
+
+
+
-
-
+
+
}
onClick={() =>
- router.push(`/admin/desa/layanan/pelayanan_surat_keterangan/${item.id}`)
+ router.push(
+ `/admin/desa/layanan/pelayanan_surat_keterangan/${item.id}`
+ )
}
>
Detail
-
-
- ))
- ) : (
-
-
-
- Tidak ada data surat keterangan yang cocok
-
-
-
- )}
-
-
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data surat keterangan yang cocok
+
+
+ )}
+
+
{
+ return (
+ formData.name?.trim() !== '' &&
+ formData.deskripsi?.trim() !== '' &&
+ formData.link?.trim() !== ''
+ );
+ };
const [formData, setFormData] = useState({
name: '',
@@ -29,6 +39,12 @@ function EditPelayananTelunjukSakti() {
link: '',
});
+ const [originalData, setOriginalData] = useState({
+ name: '',
+ deskripsi: '',
+ link: '',
+ });
+
// Load data awal hanya sekali (pas ada id)
useEffect(() => {
const loadData = async () => {
@@ -43,6 +59,11 @@ function EditPelayananTelunjukSakti() {
deskripsi: data.deskripsi ?? '',
link: data.link ?? '',
});
+ setOriginalData({
+ name: data.name ?? '',
+ deskripsi: data.deskripsi ?? '',
+ link: data.link ?? '',
+ });
}
} catch (error) {
console.error('Error loading pelayanan telunjuk sakti:', error);
@@ -61,9 +82,34 @@ function EditPelayananTelunjukSakti() {
[]
);
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ deskripsi: originalData.deskripsi,
+ link: originalData.link,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
// Submit: update global state hanya saat simpan
const handleSubmit = async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Nama pelayanan wajib diisi');
+ return;
+ }
+
+ if (!formData.deskripsi?.trim()) {
+ toast.error('Judul link wajib diisi');
+ return;
+ }
+
+ if (!formData.link?.trim()) {
+ toast.error('Link wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
stateTelunjukDesa.edit.form = {
...stateTelunjukDesa.edit.form,
...formData,
@@ -74,18 +120,18 @@ function EditPelayananTelunjukSakti() {
} catch (error) {
console.error('Error updating pelayanan telunjuk sakti:', error);
toast.error('Terjadi kesalahan saat memperbarui pelayanan telunjuk sakti');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Back Button + Title */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Edit Pelayanan Telunjuk Sakti Desa
@@ -128,17 +174,31 @@ function EditPelayananTelunjukSakti() {
{/* Tombol Simpan */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/[id]/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/[id]/page.tsx
index 04b7a37f..c45e9e33 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/[id]/page.tsx
@@ -9,8 +9,7 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip,
+ Text
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
@@ -51,7 +50,7 @@ function DetailPelayananTelunjukSakti() {
const data = telunjukSaktiState.findUnique.data;
return (
-
+
{/* Tombol Kembali */}
-
{
@@ -144,9 +142,7 @@ function DetailPelayananTelunjukSakti() {
>
-
-
@@ -160,7 +156,6 @@ function DetailPelayananTelunjukSakti() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/create/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/create/page.tsx
index 2fd23ace..f96b381a 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/create/page.tsx
@@ -6,20 +6,31 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Stack,
TextInput,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
+import { useState } from 'react';
import { toast } from 'react-toastify';
import { useProxy } from 'valtio/utils';
function CreatePelayananTelunjukDesa() {
const stateTelunjukDesa = useProxy(stateLayananDesa.pelayananTelunjukSaktiDesa);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ stateTelunjukDesa.create.form.name?.trim() !== '' &&
+ stateTelunjukDesa.create.form.deskripsi?.trim() !== '' &&
+ stateTelunjukDesa.create.form.link?.trim() !== ''
+ );
+ };
const resetForm = () => {
stateTelunjukDesa.create.form = {
@@ -30,7 +41,23 @@ function CreatePelayananTelunjukDesa() {
};
const handleSubmit = async () => {
+ if (!stateTelunjukDesa.create.form.name?.trim()) {
+ toast.error('Nama pelayanan wajib diisi');
+ return;
+ }
+
+ if (!stateTelunjukDesa.create.form.deskripsi?.trim()) {
+ toast.error('Judul link wajib diisi');
+ return;
+ }
+
+ if (!stateTelunjukDesa.create.form.link?.trim()) {
+ toast.error('Link wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
await stateTelunjukDesa.create.create();
resetForm();
toast.success('Data pelayanan telunjuk sakti berhasil ditambahkan');
@@ -38,18 +65,18 @@ function CreatePelayananTelunjukDesa() {
} catch (error) {
console.error('Error create pelayanan telunjuk sakti:', error);
toast.error('Terjadi kesalahan saat menambahkan data');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Header */}
-
router.back()} p="xs" radius="md">
-
Tambah Pelayanan Telunjuk Sakti Desa
@@ -67,7 +94,7 @@ function CreatePelayananTelunjukDesa() {
{/* Nama */}
{
stateTelunjukDesa.create.form.name = val.target.value;
}}
@@ -78,7 +105,7 @@ function CreatePelayananTelunjukDesa() {
{/* Deskripsi */}
{
stateTelunjukDesa.create.form.deskripsi = val.target.value;
}}
@@ -89,7 +116,7 @@ function CreatePelayananTelunjukDesa() {
{/* Link */}
{
stateTelunjukDesa.create.form.link = val.target.value;
}}
@@ -100,17 +127,31 @@ function CreatePelayananTelunjukDesa() {
{/* Tombol Simpan */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/page.tsx b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/page.tsx
index 03752f68..591f8c3e 100644
--- a/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/page.tsx
+++ b/src/app/admin/(dashboard)/desa/layanan/pelayanan_telunjuk_sakti_desa/page.tsx
@@ -1,161 +1,5 @@
-// /* eslint-disable react-hooks/exhaustive-deps */
-// 'use client'
-// import colors from '@/con/colors';
-// import { Box, Button, Center, Pagination, Paper, Skeleton, Stack, Table, TableTbody, TableTd, TableTh, TableThead, TableTr, Text } from '@mantine/core';
-// import { IconDeviceImac, IconSearch } from '@tabler/icons-react';
-// import { useRouter } from 'next/navigation';
-// import { useEffect, useMemo, useState } from 'react';
-// import { useProxy } from 'valtio/utils';
-// import HeaderSearch from '../../../_com/header';
-// import JudulList from '../../../_com/judulList';
-// import stateLayananDesa from '../../../_state/desa/layananDesa';
-
-// function PelayananTelunjukSakti() {
-// const [search, setSearch] = useState("");
-// return (
-//
-// }
-// value={search}
-// onChange={(e) => setSearch(e.currentTarget.value)}
-// />
-//
-//
-// );
-// }
-
-// function ListPelayananTelunjukSakti({ search }: { search: string }) {
-// const telunjukSaktiState = useProxy(stateLayananDesa.pelayananTelunjukSaktiDesa)
-// const router = useRouter()
-
-// const {
-// data,
-// page,
-// totalPages,
-// loading,
-// load,
-// } = telunjukSaktiState.findMany;
-
-// useEffect(() => {
-// load(page, 10)
-// }, [])
-
-// const filteredData = useMemo(() => {
-// if (!data) return [];
-// return data.filter(item => {
-// const keyword = search.toLowerCase();
-// return (
-// item.name?.toLowerCase().includes(keyword) ||
-// item.link?.toLowerCase().includes(keyword) ||
-// item.deskripsi?.toLowerCase().includes(keyword)
-// );
-// })
-// .sort((a, b) => a.posisi?.hierarki - b.posisi?.hierarki);
-// }, [data, search]);
-
-// if (loading || !data) {
-// return (
-//
-//
-//
-// );
-// }
-
-// if (data.length === 0) {
-// return (
-//
-//
-//
-//
-//
-//
-// Nama
-// Link
-// Detail
-//
-//
-//
-//
-//
-//
-// Tidak ada data
-//
-//
-//
-//
-//
-//
-//
-// );
-// }
-
-// return (
-//
-//
-//
-//
-//
-//
-// Nama
-// Link
-// Detail
-//
-//
-//
-// {filteredData.map((item) => (
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-//
-// router.push(`/admin/desa/layanan/pelayanan_telunjuk_sakti_desa/${item.id}`)}>
-//
-//
-//
-//
-//
-// ))}
-//
-//
-//
-//
-// {
-// load(newPage, 10);
-// window.scrollTo(0, 0);
-// }}
-// total={totalPages}
-// mt="md"
-// mb="md"
-// />
-//
-//
-// );
-// }
-
-// export default PelayananTelunjukSakti;
-
/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
+'use client';
import colors from '@/con/colors';
import {
@@ -175,7 +19,6 @@ import {
TableTr,
Text,
Title,
- Tooltip,
} from '@mantine/core';
import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
@@ -183,9 +26,10 @@ import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import stateLayananDesa from '../../../_state/desa/layananDesa';
+import { useDebouncedValue } from '@mantine/hooks';
function PelayananTelunjukSakti() {
- const [search, setSearch] = useState("");
+ const [search, setSearch] = useState('');
return (
{
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const filteredData = data || [];
if (loading || !data) {
return (
-
+
);
}
return (
-
-
+
+
- Daftar Pelayanan Telunjuk Sakti
-
- }
- color="blue"
- variant="light"
- onClick={() =>
- router.push('/admin/desa/layanan/pelayanan_telunjuk_sakti_desa/create')
- }
- >
- Tambah Baru
-
-
+
+ Daftar Pelayanan Telunjuk Sakti
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/desa/layanan/pelayanan_telunjuk_sakti_desa/create')
+ }
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Nama
- Link
- Detail
+
+ Nama
+
+
+ Link
+
+
+ Detail
+
@@ -252,18 +107,19 @@ function ListPelayananTelunjukSakti({ search }: { search: string }) {
filteredData.map((item) => (
-
-
- {item.name}
-
-
+
+ {item.name}
+
-
-
-
-
-
+
+
+
- Detail
+
+ Detail
+
@@ -284,8 +142,8 @@ function ListPelayananTelunjukSakti({ search }: { search: string }) {
) : (
-
-
+
+
Tidak ada data layanan yang cocok
@@ -295,17 +153,68 @@ function ListPelayananTelunjukSakti({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+
+ {filteredData.map((item) => (
+
+
+
+ Nama
+
+
+ {item.name}
+
+
+
+
+ Link
+
+
+ {item.deskripsi}
+
+
+
+ router.push(
+ `/admin/desa/layanan/pelayanan_telunjuk_sakti_desa/${item.id}`
+ )
+ }
+ >
+
+
+ Detail
+
+
+
+ ))}
+
+ ) : (
+
+
+ Tidak ada data layanan yang cocok
+
+
+ )}
+
+
{
load(newPage, 10, search);
window.scrollTo({ top: 0, behavior: 'smooth' });
- }}
+ }
+ }
total={totalPages}
mt="md"
- mb="md"
color="blue"
radius="md"
/>
@@ -314,5 +223,4 @@ function ListPelayananTelunjukSakti({ search }: { search: string }) {
);
}
-export default PelayananTelunjukSakti;
-
+export default PelayananTelunjukSakti;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/penghargaan/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/penghargaan/[id]/edit/page.tsx
index 2f68bc4e..7545a147 100644
--- a/src/app/admin/(dashboard)/desa/penghargaan/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/penghargaan/[id]/edit/page.tsx
@@ -5,16 +5,17 @@ import penghargaanState from '@/app/admin/(dashboard)/_state/desa/penghargaan';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
import {
+ ActionIcon,
Box,
Button,
Group,
Image,
+ Loader,
Paper,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
@@ -30,6 +31,32 @@ function EditPenghargaan() {
const [previewImage, setPreviewImage] = useState(null);
const [file, setFile] = useState(null);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ formData.juara?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ (file !== null || originalData.imageId !== '') // Either a new file is selected or an existing image exists
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ name: "",
+ juara: "",
+ deskripsi: "",
+ imageId: "",
+ imageUrl: "",
+ });
// Lokal formData
const [formData, setFormData] = useState({
@@ -47,17 +74,25 @@ function EditPenghargaan() {
try {
const data = await statePenghargaan.edit.load(id);
+
if (data) {
- setFormData({
- name: data.name || '',
- juara: data.juara || '',
- deskripsi: data.deskripsi || '',
- imageId: data.imageId || '',
+ const newForm = {
+ name: data.name || "",
+ juara: data.juara || "",
+ deskripsi: data.deskripsi || "",
+ imageId: data.imageId || "",
+ };
+ setFormData(newForm);
+
+ // simpan juga versi original
+ const imageUrl = data.image?.link || "";
+
+ setOriginalData({
+ ...newForm,
+ imageUrl: imageUrl,
});
- if (data?.image?.link) {
- setPreviewImage(data.image.link);
- }
+ setPreviewImage(imageUrl || null);
}
} catch (error) {
console.error('Error loading penghargaan:', error);
@@ -68,45 +103,79 @@ function EditPenghargaan() {
loadPenghargaan();
}, [params?.id]);
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ juara: originalData.juara,
+ deskripsi: originalData.deskripsi,
+ imageId: originalData.imageId,
+ });
+ setPreviewImage(originalData.imageUrl || null);
+ setFile(null);
+ toast.info("Form dikembalikan ke data awal");
+ };
+
// Submit
const handleSubmit = async () => {
- try {
- // Sync ke global state saat submit
- statePenghargaan.edit.form = {
- ...statePenghargaan.edit.form,
- ...formData,
- };
+ if (!formData.name?.trim()) {
+ toast.error('Nama penghargaan wajib diisi');
+ return;
+ }
- // Upload file baru (kalau ada)
+ if (!formData.juara?.trim()) {
+ toast.error('Juara wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ if (!file && !originalData.imageId) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+ // Sync ke global state saat submit
+ let imageId = formData.imageId;
if (file) {
const res = await ApiFetch.api.fileStorage.create.post({ file, name: file.name });
const uploaded = res.data?.data;
-
if (!uploaded?.id) {
- return toast.error('Gagal upload gambar');
+ return toast.error("Gagal upload gambar");
}
-
- statePenghargaan.edit.form.imageId = uploaded.id;
+ imageId = uploaded.id;
}
+ // Update global state form (baru di sini)
+ statePenghargaan.edit.form = {
+ ...statePenghargaan.edit.form,
+ name: formData.name,
+ juara: formData.juara,
+ deskripsi: formData.deskripsi,
+ imageId,
+ }
await statePenghargaan.edit.update();
toast.success('Penghargaan berhasil diperbarui!');
router.push('/admin/desa/penghargaan');
} catch (error) {
console.error('Error updating penghargaan:', error);
toast.error('Terjadi kesalahan saat memperbarui penghargaan');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Tombol Back + Title */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Edit Penghargaan
@@ -155,7 +224,7 @@ function EditPenghargaan() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -174,25 +243,47 @@ function EditPenghargaan() {
Seret gambar atau klik untuk memilih file
- Maksimal 5MB, format gambar wajib
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
{previewImage && (
-
-
+
+
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
}}
- loading="lazy"
- />
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -212,17 +303,31 @@ function EditPenghargaan() {
{/* Tombol Simpan */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/penghargaan/[id]/page.tsx b/src/app/admin/(dashboard)/desa/penghargaan/[id]/page.tsx
index 05a5774a..ab55c5db 100644
--- a/src/app/admin/(dashboard)/desa/penghargaan/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/penghargaan/[id]/page.tsx
@@ -1,9 +1,5 @@
'use client'
-import React, { useState } from 'react';
-import penghargaanState from '../../../_state/desa/penghargaan';
-import { useProxy } from 'valtio/utils';
-import { useParams, useRouter } from 'next/navigation';
-import { useShallowEffect } from '@mantine/hooks';
+import colors from '@/con/colors';
import {
Box,
Button,
@@ -12,12 +8,15 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip,
+ Text
} from '@mantine/core';
-import colors from '@/con/colors';
+import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
+import penghargaanState from '../../../_state/desa/penghargaan';
function DetailPenghargaan() {
const statePenghargaan = useProxy(penghargaanState);
@@ -50,7 +49,7 @@ function DetailPenghargaan() {
const data = statePenghargaan.findUnique.data;
return (
-
+
router.back()}
@@ -127,34 +126,30 @@ function DetailPenghargaan() {
-
- {
- setSelectedId(data.id);
- setModalHapus(true);
- }}
- variant="light"
- radius="md"
- size="md"
- >
-
-
-
+ {
+ setSelectedId(data.id);
+ setModalHapus(true);
+ }}
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
-
-
- router.push(`/admin/desa/penghargaan/${data.id}/edit`)
- }
- variant="light"
- radius="md"
- size="md"
- >
-
-
-
+
+ router.push(`/admin/desa/penghargaan/${data.id}/edit`)
+ }
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
diff --git a/src/app/admin/(dashboard)/desa/penghargaan/create/page.tsx b/src/app/admin/(dashboard)/desa/penghargaan/create/page.tsx
index dbfb717c..00748810 100644
--- a/src/app/admin/(dashboard)/desa/penghargaan/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/penghargaan/create/page.tsx
@@ -2,16 +2,17 @@
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
import {
+ ActionIcon,
Box,
Button,
Group,
Image,
+ Loader,
Paper,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
@@ -27,6 +28,24 @@ function CreatePenghargaan() {
const [previewImage, setPreviewImage] = useState(null);
const [file, setFile] = useState(null);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ statePenghargaan.create.form.name?.trim() !== '' &&
+ statePenghargaan.create.form.juara?.trim() !== '' &&
+ !isHtmlEmpty(statePenghargaan.create.form.deskripsi) &&
+ file !== null
+ );
+ };
const resetForm = () => {
statePenghargaan.create.form = {
@@ -40,37 +59,63 @@ function CreatePenghargaan() {
};
const handleSubmit = async () => {
+ if (!statePenghargaan.create.form.name?.trim()) {
+ toast.error('Nama penghargaan wajib diisi');
+ return;
+ }
+
+ if (!statePenghargaan.create.form.juara?.trim()) {
+ toast.error('Juara wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(statePenghargaan.create.form.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
if (!file) {
- return toast.warn('Silakan pilih file gambar terlebih dahulu');
+ toast.error('Gambar wajib dipilih');
+ return;
}
- const res = await ApiFetch.api.fileStorage.create.post({
- file,
- name: file.name,
- });
+ try {
+ setIsSubmitting(true);
+ if (!file) {
+ return toast.warn('Silakan pilih file gambar terlebih dahulu');
+ }
- const uploaded = res.data?.data;
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
- if (!uploaded?.id) {
- return toast.error('Gagal mengunggah gambar, silakan coba lagi');
+ const uploaded = res.data?.data;
+
+ if (!uploaded?.id) {
+ return toast.error('Gagal mengunggah gambar, silakan coba lagi');
+ }
+
+ statePenghargaan.create.form.imageId = uploaded.id;
+
+ await statePenghargaan.create.create();
+ resetForm();
+ router.push('/admin/desa/penghargaan');
+ } catch (error) {
+ console.error('Error creating penghargaan:', error);
+ toast.error('Terjadi kesalahan saat menambahkan penghargaan');
+ } finally {
+ setIsSubmitting(false);
}
-
- statePenghargaan.create.form.imageId = uploaded.id;
-
- await statePenghargaan.create.create();
- resetForm();
- router.push('/admin/desa/penghargaan');
};
return (
-
+
{/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Tambah Penghargaan
@@ -87,7 +132,7 @@ function CreatePenghargaan() {
>
(statePenghargaan.create.form.name = val.target.value)}
label="Nama Penghargaan"
placeholder="Masukkan nama penghargaan"
@@ -95,7 +140,7 @@ function CreatePenghargaan() {
/>
(statePenghargaan.create.form.juara = val.target.value)}
label="Juara"
placeholder="Masukkan juara"
@@ -125,7 +170,7 @@ function CreatePenghargaan() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -146,7 +191,7 @@ function CreatePenghargaan() {
{previewImage && (
-
+
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
{/* Button Submit */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/penghargaan/page.tsx b/src/app/admin/(dashboard)/desa/penghargaan/page.tsx
index ac4de300..fe7be296 100644
--- a/src/app/admin/(dashboard)/desa/penghargaan/page.tsx
+++ b/src/app/admin/(dashboard)/desa/penghargaan/page.tsx
@@ -18,14 +18,14 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../_com/header';
+import { useDebouncedValue } from '@mantine/hooks';
function Penghargaan() {
const [search, setSearch] = useState("");
@@ -46,47 +46,50 @@ function Penghargaan() {
function ListPenghargaan({ search }: { search: string }) {
const state = useProxy(penghargaanState);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const { data, page, totalPages, loading, load } = state.findMany;
useEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
- const filteredData = data || []
+ const filteredData = data || [];
// Loading state
if (loading || !data) {
return (
-
-
+
+
);
}
return (
-
+
-
+
List Penghargaan
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/penghargaan/create')}
- >
- Tambah Baru
-
-
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/penghargaan/create')}
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Nama
- Deskripsi
- Aksi
+ Nama
+ Deskripsi
+ Aksi
@@ -94,31 +97,27 @@ function ListPenghargaan({ search }: { search: string }) {
filteredData.map((item) => (
-
-
- {item.name}
-
-
+
+ {item.name}
+
-
-
-
+
}
+ size="xs"
+ radius="md"
+ variant="light"
+ color="blue"
+ leftSection={ }
onClick={() =>
router.push(`/admin/desa/penghargaan/${item.id}`)
}
@@ -130,9 +129,9 @@ function ListPenghargaan({ search }: { search: string }) {
))
) : (
-
-
-
+
+
+
Tidak ada data penghargaan yang cocok
@@ -142,7 +141,54 @@ function ListPenghargaan({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Nama
+
+
+ {item.name}
+
+
+
+
+ Deskripsi
+
+
+
+
+
+
+ }
+ onClick={() =>
+ router.push(`/admin/desa/penghargaan/${item.id}`)
+ }
+ >
+ Detail
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data penghargaan yang cocok
+
+
+ )}
+
+
,
- tooltip: "Lihat semua daftar pengumuman"
+ icon:
},
{
label: "Kategori Pengumuman",
value: "kategoripengumuman",
href: "/admin/desa/pengumuman/kategori-pengumuman",
- icon: ,
- tooltip: "Kelola kategori pengumuman"
+ icon:
},
];
@@ -56,36 +54,76 @@ function LayoutTabsLayanan({ children }: { children: React.ReactNode }) {
keepMounted={false}
>
{/* ✅ Scroll horizontal wrapper */}
-
-
- {tabs.map((tab, i) => (
-
+
+
+
+ {tabs.map((tab, i) => (
{tab.label}
-
- ))}
-
-
+ ))}
+
+
+
+
+
+
+
+
+ {tabs.map((tab, i) => (
+
+ {tab.label}
+
+ ))}
+
+
+
{tabs.map((tab, i) => (
{
+ return (
+ formData.name?.trim() !== ''
+ );
+ };
// Load data awal sekali aja
useEffect(() => {
@@ -36,6 +44,7 @@ function EditKategoriPengumuman() {
const data = await editState.update.load(id);
if (data) {
setFormData({ name: data.name || '' });
+ setOriginalData({ name: data.name || '' });
}
} catch (error) {
console.error('Error loading kategori Pengumuman:', error);
@@ -54,7 +63,13 @@ function EditKategoriPengumuman() {
};
const handleSubmit = async () => {
+ if (!formData.name?.trim()) {
+ toast.error('Nama kategori pengumuman wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
// Update global state hanya di sini
editState.update.form = {
...editState.update.form,
@@ -67,14 +82,23 @@ function EditKategoriPengumuman() {
} catch (error) {
console.error('Error updating kategori Pengumuman:', error);
toast.error('Terjadi kesalahan saat memperbarui kategori Pengumuman');
+ } finally {
+ setIsSubmitting(false);
}
};
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
+
return (
-
+
{/* Header */}
-
router.back()}
@@ -83,7 +107,6 @@ function EditKategoriPengumuman() {
>
-
Edit Kategori Pengumuman
@@ -108,17 +131,31 @@ function EditKategoriPengumuman() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/create/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/create/page.tsx
index c5e4bac0..c3eac62d 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/create/page.tsx
@@ -9,15 +9,25 @@ import {
Stack,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useProxy } from 'valtio/utils';
+import { useState } from 'react';
+import { toast } from 'react-toastify';
function CreateKategoriPengumuman() {
const createState = useProxy(stateDesaPengumuman.category);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ createState.create.form.name?.trim() !== ''
+ );
+ };
const resetForm = () => {
createState.create.form = {
@@ -26,16 +36,27 @@ function CreateKategoriPengumuman() {
};
const handleSubmit = async () => {
- await createState.create.create();
- resetForm();
- router.push('/admin/desa/pengumuman/kategori-pengumuman');
+ if (!createState.create.form.name?.trim()) {
+ toast.error('Nama kategori pengumuman wajib diisi');
+ return;
+ }
+
+ try {
+ await createState.create.create();
+ resetForm();
+ router.push('/admin/desa/pengumuman/kategori-pengumuman');
+ } catch (error) {
+ console.error(error);
+ toast.error('Gagal menambahkan kategori pengumuman');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header dengan back button */}
-
router.back()}
@@ -44,7 +65,6 @@ function CreateKategoriPengumuman() {
>
-
Tambah Kategori Pengumuman
@@ -63,23 +83,37 @@ function CreateKategoriPengumuman() {
(createState.create.form.name = e.target.value)}
required
/>
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/page.tsx
index 541c0cae..3f364287 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/kategori-pengumuman/page.tsx
@@ -2,17 +2,30 @@
'use client'
import colors from '@/con/colors';
import {
- Box, Button, Center, Paper, Skeleton, Stack,
- Table, TableTbody, TableTd, TableTh, TableThead, TableTr,
- Text, Title, Tooltip, Pagination
+ Box,
+ Button,
+ Center,
+ Pagination,
+ Paper,
+ Skeleton,
+ Stack,
+ Table,
+ TableTbody,
+ TableTd,
+ TableTh,
+ TableThead,
+ TableTr,
+ Text,
+ Title,
} from '@mantine/core';
-import { IconEdit, IconSearch, IconTrash, IconPlus } from '@tabler/icons-react';
+import { IconEdit, IconPlus, IconSearch, IconTrash } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
import stateDesaPengumuman from '../../../_state/desa/pengumuman';
+import { useDebouncedValue } from '@mantine/hooks';
function KategoriPengumuman() {
const [search, setSearch] = useState('');
@@ -31,105 +44,147 @@ function KategoriPengumuman() {
}
function ListKategoriPengumuman({ search }: { search: string }) {
- const listDataState = useProxy(stateDesaPengumuman.category)
+ const listDataState = useProxy(stateDesaPengumuman.category);
const router = useRouter();
- const [modalHapus, setModalHapus] = useState(false)
- const [selectedId, setSelectedId] = useState(null)
+ const [modalHapus, setModalHapus] = useState(false);
+ const [selectedId, setSelectedId] = useState(null);
+ const [debouncedSearch] = useDebouncedValue(search, 500);
const { data, page, totalPages, loading, load } = listDataState.findMany;
useEffect(() => {
- load(1, 10, search)
- }, [search])
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const handleDelete = () => {
if (selectedId) {
- listDataState.delete.delete(selectedId)
- setModalHapus(false)
- setSelectedId(null)
- load(page, 10, search)
+ listDataState.delete.delete(selectedId);
+ setModalHapus(false);
+ setSelectedId(null);
+ load(page, 10, search);
}
- }
+ };
- const filteredData = data || []
+ const filteredData = data || [];
if (loading || !data) {
return (
-
+
- )
+ );
}
return (
-
-
-
-
- List Kategori Pengumuman
-
+
+
+
+
+
+ List Kategori Pengumuman
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/desa/pengumuman/kategori-pengumuman/create')
+ }
+ >
+ Tambah Baru
+
+
+
+
+
+
+ List Kategori Pengumuman
+
}
color="blue"
variant="light"
- onClick={() => router.push('/admin/desa/pengumuman/kategori-pengumuman/create')}
+ onClick={() =>
+ router.push('/admin/desa/pengumuman/kategori-pengumuman/create')
+ }
+ fullWidth
>
Tambah Baru
-
+
-
-
+
+
- No
- Nama
- Edit
- Hapus
+
+
+ Nama
+
+
+
+
+ Edit
+
+
+
+
+ Hapus
+
+
{filteredData.length > 0 ? (
- filteredData.map((item, index) => (
+ filteredData.map((item) => (
- {(page - 1) * 10 + index + 1}
+
+ {item.name}
+
- {item.name}
+
+ router.push(
+ `/admin/desa/pengumuman/kategori-pengumuman/${item.id}`
+ )
+ }
+ size="compact-sm"
+ >
+
+
-
- router.push(`/admin/desa/pengumuman/kategori-pengumuman/${item.id}`)}
- >
-
-
-
-
-
-
- {
- setSelectedId(item.id)
- setModalHapus(true)
- }}>
-
-
-
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ size="compact-sm"
+ >
+
+
))
) : (
-
- Tidak ada data kategori pengumuman yang cocok
+
+
+ Tidak ada data kategori pengumuman yang cocok
+
@@ -137,6 +192,71 @@ function ListKategoriPengumuman({ search }: { search: string }) {
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+
+ Nama
+
+
+ {item.name}
+
+
+
+
+ router.push(
+ `/admin/desa/pengumuman/kategori-pengumuman/${item.id}`
+ )
+ }
+ size="compact-xs"
+ >
+
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ size="compact-xs"
+ >
+
+
+
+
+
+ ))
+ ) : (
+
+
+
+ Tidak ada data kategori pengumuman yang cocok
+
+
+
+ )}
+
@@ -157,7 +277,7 @@ function ListKategoriPengumuman({ search }: { search: string }) {
text='Apakah anda yakin ingin menghapus kategori Pengumuman ini?'
/>
- )
+ );
}
-export default KategoriPengumuman;
+export default KategoriPengumuman;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/layout.tsx b/src/app/admin/(dashboard)/desa/pengumuman/layout.tsx
index 4a4acdbd..6ecbb342 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/layout.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/layout.tsx
@@ -1,7 +1,29 @@
+'use client'
import React from 'react';
import LayoutTabs from './_com/layoutTabs';
+import { usePathname } from 'next/navigation';
+import { Box } from '@mantine/core';
function Layout({ children }: { children: React.ReactNode }) {
+ const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/pengumuman/semua → panjang 5 → list
+ // - /darmasaba/desa/pengumuman/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/pengumuman/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
return (
{children}
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/edit/page.tsx
index 9e7a622f..5e41d683 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/edit/page.tsx
@@ -14,7 +14,7 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader
} from "@mantine/core";
import { IconArrowBack } from "@tabler/icons-react";
import { useParams, useRouter } from "next/navigation";
@@ -34,6 +34,32 @@ function EditPengumuman() {
content: "",
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.judul?.trim() !== '' &&
+ formData.deskripsi?.trim() !== '' &&
+ formData.categoryPengumumanId !== '' &&
+ !isHtmlEmpty(formData.content)
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ judul: "",
+ deskripsi: "",
+ categoryPengumumanId: "",
+ content: "",
+ });
+
// Load kategori & pengumuman by id saat pertama kali
useEffect(() => {
editState.category.findMany.load();
@@ -51,6 +77,12 @@ function EditPengumuman() {
categoryPengumumanId: data.categoryPengumumanId || "",
content: data.content || "",
});
+ setOriginalData({
+ judul: data.judul || "",
+ deskripsi: data.deskripsi || "",
+ categoryPengumumanId: data.categoryPengumumanId || "",
+ content: data.content || "",
+ });
}
} catch (error) {
console.error("Error loading pengumuman:", error);
@@ -66,7 +98,28 @@ function EditPengumuman() {
};
const handleSubmit = async () => {
+ if (!formData.judul?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!formData.deskripsi?.trim()) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
+ if (!formData.categoryPengumumanId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.content)) {
+ toast.error('Konten lengkap wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
// update global state hanya sekali pas submit
editState.pengumuman.edit.form = {
...editState.pengumuman.edit.form,
@@ -79,22 +132,32 @@ function EditPengumuman() {
} catch (error) {
console.error("Error updating pengumuman:", error);
toast.error("Terjadi kesalahan saat memperbarui pengumuman");
+ } finally {
+ setIsSubmitting(false);
}
};
+ const handleResetForm = () => {
+ setFormData({
+ judul: originalData.judul,
+ deskripsi: originalData.deskripsi,
+ categoryPengumumanId: originalData.categoryPengumumanId,
+ content: originalData.content,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
return (
-
+
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Pengumuman
@@ -155,17 +218,32 @@ function EditPengumuman() {
+ {/* Tombol Batal */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/page.tsx
index e1847094..fb768490 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/[id]/page.tsx
@@ -1,5 +1,7 @@
'use client'
+import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
+import stateDesaPengumuman from '@/app/admin/(dashboard)/_state/desa/pengumuman';
import colors from '@/con/colors';
import {
Box,
@@ -8,16 +10,13 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip,
+ Text
} from '@mantine/core';
+import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
-import { useRouter, useParams } from 'next/navigation';
+import { useParams, useRouter } from 'next/navigation';
import { useState } from 'react';
import { useProxy } from 'valtio/utils';
-import { useShallowEffect } from '@mantine/hooks';
-import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
-import stateDesaPengumuman from '@/app/admin/(dashboard)/_state/desa/pengumuman';
export default function DetailPengumuman() {
const pengumumanState = useProxy(stateDesaPengumuman);
@@ -50,7 +49,7 @@ export default function DetailPengumuman() {
const data = pengumumanState.pengumuman.findUnique.data;
return (
-
+
router.back()}
@@ -62,7 +61,7 @@ export default function DetailPengumuman() {
-
-
- Kategori
-
-
- {data?.CategoryPengumuman?.name || '-'}
-
-
@@ -93,6 +84,15 @@ export default function DetailPengumuman() {
+
+
+ Kategori
+
+
+ {data?.CategoryPengumuman?.name || '-'}
+
+
+
Deskripsi
@@ -117,7 +117,6 @@ export default function DetailPengumuman() {
-
{
@@ -130,9 +129,7 @@ export default function DetailPengumuman() {
>
-
-
@@ -146,7 +143,6 @@ export default function DetailPengumuman() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/create/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/create/page.tsx
index d2d62314..52eaf9d3 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/create/page.tsx
@@ -13,25 +13,73 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { toast } from 'react-toastify';
import { useProxy } from 'valtio/utils';
function CreatePengumuman() {
const pengumumanState = useProxy(stateDesaPengumuman);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ pengumumanState.pengumuman.create.form.judul?.trim() !== '' &&
+ pengumumanState.pengumuman.create.form.categoryPengumumanId !== '' &&
+ pengumumanState.pengumuman.create.form.deskripsi?.trim() !== '' &&
+ !isHtmlEmpty(pengumumanState.pengumuman.create.form.content)
+ );
+ };
useShallowEffect(() => {
pengumumanState.category.findMany.load();
}, []);
const handleSubmit = async () => {
- await pengumumanState.pengumuman.create.create();
- resetForm();
- router.push('/admin/desa/pengumuman/list-pengumuman');
+ if (!pengumumanState.pengumuman.create.form.judul?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (!pengumumanState.pengumuman.create.form.categoryPengumumanId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (!pengumumanState.pengumuman.create.form.deskripsi?.trim()) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(pengumumanState.pengumuman.create.form.content)) {
+ toast.error('Konten lengkap wajib diisi');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+ await pengumumanState.pengumuman.create.create();
+ resetForm();
+ router.push('/admin/desa/pengumuman/list-pengumuman');
+ } catch (error) {
+ console.error('Error creating pengumuman:', error);
+ toast.error('Terjadi kesalahan saat membuat pengumuman');
+ } finally {
+ setIsSubmitting(false);
+ }
};
const resetForm = () => {
@@ -44,14 +92,12 @@ function CreatePengumuman() {
};
return (
-
+
{/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Tambah Pengumuman
@@ -68,7 +114,7 @@ function CreatePengumuman() {
{/* Judul */}
(pengumumanState.pengumuman.create.form.judul = val.target.value)}
label="Judul"
placeholder="Masukkan judul pengumuman"
@@ -79,21 +125,32 @@ function CreatePengumuman() {
{
- pengumumanState.pengumuman.create.form.categoryPengumumanId = val ?? "";
- }}
data={pengumumanState.category.findMany.data?.map((item) => ({
label: item.name,
value: item.id,
- }))}
+ })) || []}
+ value={pengumumanState.pengumuman.create.form.categoryPengumumanId || null}
+ onChange={(val: string | null) => {
+ if (val) {
+ const selected = pengumumanState.category.findMany.data?.find(
+ (item) => item.id === val
+ );
+ if (selected) {
+ pengumumanState.pengumuman.create.form.categoryPengumumanId = selected.id;
+ }
+ } else {
+ pengumumanState.pengumuman.create.form.categoryPengumumanId = '';
+ }
+ }}
searchable
+ clearable
nothingFoundMessage="Tidak ditemukan"
+ required
/>
{/* Deskripsi Singkat */}
(pengumumanState.pengumuman.create.form.deskripsi = val.target.value)}
label="Deskripsi Singkat"
placeholder="Masukkan deskripsi singkat"
@@ -115,17 +172,31 @@ function CreatePengumuman() {
{/* Tombol Submit */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/page.tsx b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/page.tsx
index e4c819e4..a309b68d 100644
--- a/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/page.tsx
+++ b/src/app/admin/(dashboard)/desa/pengumuman/list-pengumuman/page.tsx
@@ -17,10 +17,9 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
import { IconCircleDashedPlus, IconDeviceImacCog, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
@@ -47,46 +46,58 @@ function Pengumuman() {
function ListPengumuman({ search }: { search: string }) {
const pengumumanState = useProxy(stateDesaPengumuman);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const { data, page, totalPages, loading, load } = pengumumanState.pengumuman.findMany;
useShallowEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const filteredData = data || [];
if (loading || !data) {
return (
-
+
);
}
return (
-
-
-
- Daftar Pengumuman
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/pengumuman/list-pengumuman/create')}
- >
- Tambah Baru
-
-
+
+
+
+
+ Daftar Pengumuman
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/pengumuman/list-pengumuman/create')}
+ fz={{ base: 'sm', md: 'md' }}
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Judul
- Kategori
- Detail
+
+ Judul
+
+
+ Kategori
+
+
+ Detail
+
@@ -94,14 +105,12 @@ function ListPengumuman({ search }: { search: string }) {
filteredData.map((item) => (
-
-
- {item.judul}
-
-
+
+ {item.judul}
+
-
+
{item.CategoryPengumuman?.name || '-'}
@@ -112,9 +121,12 @@ function ListPengumuman({ search }: { search: string }) {
onClick={() =>
router.push(`/admin/desa/pengumuman/list-pengumuman/${item.id}`)
}
+ fz="sm"
+ px="sm"
+ py="xs"
>
-
- Detail
+
+ Detail
@@ -122,8 +134,10 @@ function ListPengumuman({ search }: { search: string }) {
) : (
-
- Tidak ada pengumuman yang cocok
+
+
+ Tidak ada pengumuman yang cocok
+
@@ -131,7 +145,59 @@ function ListPengumuman({ search }: { search: string }) {
+
+ {/* Mobile Card List */}
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+
+ Judul
+
+
+ {item.judul}
+
+
+
+
+ Kategori
+
+
+ {item.CategoryPengumuman?.name || '-'}
+
+
+
+
+ router.push(`/admin/desa/pengumuman/list-pengumuman/${item.id}`)
+ }
+ fullWidth
+ fz="sm"
+ mt="xs"
+ >
+
+ Detail
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada pengumuman yang cocok
+
+
+ )}
+
+
+
,
- tooltip: "Lihat semua potensi desa"
+ icon:
},
{
label: "Kategori Potensi",
value: "kategori_potensi",
href: "/admin/desa/potensi/kategori-potensi",
- icon: ,
- tooltip: "Kelola kategori potensi"
+ icon:
},
];
@@ -70,19 +68,18 @@ function LayoutTabsPotensi({ children }: { children: React.ReactNode }) {
}}
>
{tabs.map((tab, i) => (
-
-
- {tab.label}
-
-
+
+ {tab.label}
+
))}
diff --git a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/[id]/page.tsx b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/[id]/page.tsx
index 16d0d951..66809e6c 100644
--- a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/[id]/page.tsx
@@ -10,7 +10,7 @@ import {
Stack,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -27,6 +27,19 @@ function EditKategoriPotensi() {
nama: '',
});
+ const [originalData, setOriginalData] = useState({
+ nama: '',
+ });
+
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.nama?.trim() !== ''
+ );
+ };
+
// Load data dari backend -> isi ke formData lokal
useEffect(() => {
const loadKategori = async () => {
@@ -39,6 +52,9 @@ function EditKategoriPotensi() {
setFormData({
nama: data.nama || '',
});
+ setOriginalData({
+ nama: data.nama || '',
+ });
}
} catch (error) {
console.error('Error loading kategori potensi:', error);
@@ -56,8 +72,21 @@ function EditKategoriPotensi() {
}));
};
+ const handleResetForm = () => {
+ setFormData({
+ nama: originalData.nama,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
const handleSubmit = async () => {
+ if (!formData.nama?.trim()) {
+ toast.error('Nama kategori potensi wajib diisi');
+ return;
+ }
+
try {
+ setIsSubmitting(true);
// Update global state hanya pas submit
editState.update.form = {
...editState.update.form,
@@ -70,13 +99,14 @@ function EditKategoriPotensi() {
} catch (error) {
console.error('Error updating kategori potensi:', error);
toast.error('Terjadi kesalahan saat memperbarui kategori potensi');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
-
router.back()}
@@ -85,7 +115,6 @@ function EditKategoriPotensi() {
>
-
Edit Kategori Potensi
@@ -109,17 +138,31 @@ function EditKategoriPotensi() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/create/page.tsx b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/create/page.tsx
index 68f63d0a..72659e74 100644
--- a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/create/page.tsx
@@ -9,15 +9,24 @@ import {
Stack,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
+import { useState } from 'react';
import { useProxy } from 'valtio/utils';
function CreateKategoriPotensi() {
const createState = useProxy(potensiDesaState.kategoriPotensi);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ createState.create.form.nama?.trim() !== ''
+ );
+ };
const resetForm = () => {
createState.create.form = {
@@ -26,25 +35,35 @@ function CreateKategoriPotensi() {
};
const handleSubmit = async () => {
- await createState.create.create();
- resetForm();
- router.push('/admin/desa/potensi/kategori-potensi');
+ if (!createState.create.form.nama?.trim()) {
+ alert('Nama kategori potensi wajib diisi');
+ return;
+ }
+
+ try {
+ setIsSubmitting(true);
+ await createState.create.create();
+ resetForm();
+ router.push('/admin/desa/potensi/kategori-potensi');
+ } catch (error) {
+ console.error('Error creating kategori potensi:', error);
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header dengan back button */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Tambah Kategori Potensi
@@ -63,23 +82,37 @@ function CreateKategoriPotensi() {
(createState.create.form.nama = e.target.value)}
required
/>
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/page.tsx b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/page.tsx
index 3d408f8f..8f4af700 100644
--- a/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/kategori-potensi/page.tsx
@@ -1,14 +1,32 @@
/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
+'use client';
import colors from '@/con/colors';
-import { Box, Button, Center, Paper, Skeleton, Stack, Table, TableTbody, TableTd, TableTh, TableThead, TableTr, Text, Title, Tooltip, Pagination, Group } from '@mantine/core';
-import { IconEdit, IconSearch, IconTrash, IconPlus } from '@tabler/icons-react';
+import {
+ Box,
+ Button,
+ Center,
+ Group,
+ Pagination,
+ Paper,
+ Skeleton,
+ Stack,
+ Table,
+ TableTbody,
+ TableTd,
+ TableTh,
+ TableThead,
+ TableTr,
+ Text,
+ Title,
+} from '@mantine/core';
+import { IconEdit, IconPlus, IconSearch, IconTrash } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
-import potensiDesaState from '../../../_state/desa/potensi';
import { ModalKonfirmasiHapus } from '../../../_com/modalKonfirmasiHapus';
+import potensiDesaState from '../../../_state/desa/potensi';
+import { useDebouncedValue } from '@mantine/hooks';
function KategoriPotensi() {
const [search, setSearch] = useState('');
@@ -27,87 +45,112 @@ function KategoriPotensi() {
}
function ListKategoriPotensi({ search }: { search: string }) {
- const listDataState = useProxy(potensiDesaState.kategoriPotensi)
+ const listDataState = useProxy(potensiDesaState.kategoriPotensi);
const router = useRouter();
- const [modalHapus, setModalHapus] = useState(false)
- const [selectedId, setSelectedId] = useState(null)
+ const [modalHapus, setModalHapus] = useState(false);
+ const [selectedId, setSelectedId] = useState(null);
const { data, page, totalPages, loading, load } = listDataState.findMany;
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
useEffect(() => {
- load(1, 10, search)
- }, [search])
+ load(1, 10, debouncedSearch);
+ }, [debouncedSearch]);
const handleDelete = () => {
if (selectedId) {
- listDataState.delete.delete(selectedId)
- setModalHapus(false)
- setSelectedId(null)
- load(page, 10, search)
+ listDataState.delete.delete(selectedId);
+ setModalHapus(false);
+ setSelectedId(null);
+ load(page, 10, search);
}
- }
+ };
- const filteredData = data || []
+ const filteredData = data || [];
if (loading || !data) {
return (
-
+
- )
+ );
}
return (
-
+
-
-
- List Kategori Potensi
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/potensi/kategori-potensi/create')}
- >
- Tambah Baru
-
-
+
+
+
+ List Kategori Potensi
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/desa/potensi/kategori-potensi/create')
+ }
+ >
+ Tambah Baru
+
-
-
+ {/* Desktop Table */}
+
+
- No
- Nama
- Edit
- Hapus
+
+
+ Nama
+
+
+
+
+ Edit
+
+
+
+
+ Hapus
+
+
{filteredData.length > 0 ? (
- filteredData.map((item, index) => (
+ filteredData.map((item) => (
- {(page - 1) * 10 + index + 1}
+
+ {item.nama}
+
- {item.nama}
-
-
- router.push(`/admin/desa/potensi/kategori-potensi/${item.id}`)}>
+
+ router.push(
+ `/admin/desa/potensi/kategori-potensi/${item.id}`
+ )
+ }
+ >
{
- setSelectedId(item.id)
- setModalHapus(true)
- }}>
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ >
@@ -116,8 +159,10 @@ function ListKategoriPotensi({ search }: { search: string }) {
) : (
-
- Tidak ada data kategori potensi yang cocok
+
+
+ Tidak ada data kategori potensi yang cocok
+
@@ -125,10 +170,70 @@ function ListKategoriPotensi({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item, index) => (
+
+
+
+
+ No
+
+
+ {(page - 1) * 10 + index + 1}
+
+
+
+
+ Nama
+
+
+ {item.nama}
+
+
+
+
+ router.push(
+ `/admin/desa/potensi/kategori-potensi/${item.id}`
+ )
+ }
+ >
+
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ >
+
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data kategori potensi yang cocok
+
+
+ )}
+
-
+
load(newPage, 10, search)}
@@ -145,7 +250,7 @@ function ListKategoriPotensi({ search }: { search: string }) {
text='Apakah anda yakin ingin menghapus kategori Potensi ini?'
/>
- )
+ );
}
-export default KategoriPotensi;
+export default KategoriPotensi;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/potensi/layout.tsx b/src/app/admin/(dashboard)/desa/potensi/layout.tsx
index b677970c..7e83e086 100644
--- a/src/app/admin/(dashboard)/desa/potensi/layout.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/layout.tsx
@@ -1,8 +1,29 @@
'use client'
import React from 'react';
import LayoutTabsPotensi from './_lib/layoutTabs';
+import { usePathname } from 'next/navigation';
+import { Box } from '@mantine/core';
function Layout({ children }: { children: React.ReactNode }) {
+ const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/berita/semua → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
return (
{children}
diff --git a/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/edit/page.tsx
index 68a453b3..600fc2ef 100644
--- a/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/edit/page.tsx
@@ -16,7 +16,8 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader,
+ ActionIcon
} from "@mantine/core";
import { Dropzone } from "@mantine/dropzone";
import { IconArrowBack, IconPhoto, IconUpload, IconX } from "@tabler/icons-react";
@@ -39,6 +40,34 @@ function EditPotensi() {
content: "",
imageId: "",
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi) &&
+ formData.kategoriId !== '' &&
+ (file !== null || originalData.imageId !== '') && // Either a new file is selected or an existing image exists
+ !isHtmlEmpty(formData.content)
+ );
+ };
+
+ const [originalData, setOriginalData] = useState({
+ name: "",
+ deskripsi: "",
+ kategoriId: "",
+ content: "",
+ imageId: "",
+ imageUrl: "",
+ });
// handle input changes
const handleChange = (field: string, value: string) => {
@@ -47,11 +76,11 @@ function EditPotensi() {
useEffect(() => {
potensiDesaState.kategoriPotensi.findMany.load();
-
+
const loadPotensi = async () => {
const id = params?.id as string;
if (!id) return;
-
+
try {
const data = await potensiState.edit.load(id);
if (data) {
@@ -62,35 +91,70 @@ function EditPotensi() {
content: data.content || "",
imageId: data.imageId || "",
});
-
- // // merge, bukan replace
- // setFormData((prev) => ({
- // ...prev,
- // name: data.name ?? prev.name,
- // deskripsi: data.deskripsi ?? prev.deskripsi,
- // kategoriId: data.kategoriId ?? prev.kategoriId,
- // content: data.content ?? prev.content,
- // imageId: data.imageId ?? prev.imageId,
- // }));
-
- if (data?.image?.link) {
- setPreviewImage(data.image.link);
- }
+
+ setOriginalData({
+ name: data.name || "",
+ deskripsi: data.deskripsi || "",
+ kategoriId: data.kategoriId || "",
+ content: data.content || "",
+ imageId: data.imageId || "",
+ imageUrl: data.image?.link || "",
+ });
+ setPreviewImage(data.image.link);
+
}
} catch (error) {
console.error("Error loading potensi:", error);
toast.error("Gagal memuat data potensi");
}
};
-
+
loadPotensi();
}, [params?.id]);
-
+
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name || "",
+ deskripsi: originalData.deskripsi || "",
+ kategoriId: originalData.kategoriId || "",
+ content: originalData.content || "",
+ imageId: originalData.imageId || ""
+ });
+ setPreviewImage(originalData.imageUrl || null);
+ setFile(null);
+ toast.info("Form dikembalikan ke data awal");
+ };
+
const handleSubmit = async () => {
- try {
- let imageId = formData.imageId;
+ if (!formData.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
+ if (!formData.kategoriId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (!file && !originalData.imageId) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.content)) {
+ toast.error('Konten lengkap wajib diisi');
+ return;
+ }
+ try {
+ setIsSubmitting(true);
+ let imageId = formData.imageId;
if (file) {
const res = await ApiFetch.api.fileStorage.create.post({
file,
@@ -116,22 +180,22 @@ function EditPotensi() {
} catch (error) {
console.error("Error updating potensi:", error);
toast.error("Terjadi kesalahan saat memperbarui potensi");
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Potensi Desa
@@ -167,6 +231,32 @@ function EditPotensi() {
({
+ label: item.nama,
+ value: item.id,
+ })) || []}
+ value={formData.kategoriId || null}
+ onChange={(val: string | null) => {
+ if (val) {
+ const selected = potensiDesaState.kategoriPotensi.findMany.data?.find(
+ (item) => item.id === val
+ );
+ if (selected) {
+ handleChange("kategoriId", selected.id);
+ }
+ } else {
+ handleChange("kategoriId", "");
+ }
+ }}
+ searchable
+ clearable
+ nothingFoundMessage="Tidak ditemukan"
+ required
+ />
+
+ {/* handleChange("kategoriId", val || "")}
label="Kategori"
@@ -181,7 +271,7 @@ function EditPotensi() {
searchable
required
error={!formData.kategoriId ? "Pilih kategori" : undefined}
- />
+ /> */}
@@ -222,25 +312,45 @@ function EditPotensi() {
Seret gambar atau klik untuk memilih file
- Maksimal 5MB, format gambar wajib
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
{previewImage && (
-
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -258,17 +368,32 @@ function EditPotensi() {
+ {/* Tombol Batal */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/page.tsx b/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/page.tsx
index 73531d4e..35fe407a 100644
--- a/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/list-potensi/[id]/page.tsx
@@ -1,13 +1,14 @@
'use client'
-import colors from '@/con/colors';
-import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text, Tooltip } from '@mantine/core';
-import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
-import { useRouter, useParams } from 'next/navigation';
-import { useState } from 'react';
-import { useProxy } from 'valtio/utils';
-import { useShallowEffect } from '@mantine/hooks';
import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
import potensiDesaState from '@/app/admin/(dashboard)/_state/desa/potensi';
+import colors from '@/con/colors';
+import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text } from '@mantine/core';
+import { useShallowEffect } from '@mantine/hooks';
+import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
+import DOMPurify from 'dompurify';
export default function DetailPotensi() {
const router = useRouter();
@@ -40,7 +41,7 @@ export default function DetailPotensi() {
const data = potensiState.findUnique.data;
return (
-
+
router.back()}
@@ -52,7 +53,7 @@ export default function DetailPotensi() {
Deskripsi
-
+
@@ -102,13 +113,17 @@ export default function DetailPotensi() {
-
{
@@ -121,9 +136,7 @@ export default function DetailPotensi() {
>
-
-
@@ -135,7 +148,6 @@ export default function DetailPotensi() {
>
-
diff --git a/src/app/admin/(dashboard)/desa/potensi/list-potensi/create/page.tsx b/src/app/admin/(dashboard)/desa/potensi/list-potensi/create/page.tsx
index 6eeb29b4..3fc3fa66 100644
--- a/src/app/admin/(dashboard)/desa/potensi/list-potensi/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/list-potensi/create/page.tsx
@@ -15,7 +15,8 @@ import {
Text,
TextInput,
Title,
- Tooltip,
+ Loader,
+ ActionIcon
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
@@ -29,30 +30,82 @@ function CreatePotensi() {
const [previewImage, setPreviewImage] = useState(null);
const [file, setFile] = useState(null);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ potensiState.create.form.name?.trim() !== '' &&
+ !isHtmlEmpty(potensiState.create.form.deskripsi) &&
+ potensiState.create.form.kategoriId !== '' &&
+ file !== null &&
+ !isHtmlEmpty(potensiState.create.form.content)
+ );
+ };
useEffect(() => {
potensiDesaState.kategoriPotensi.findMany.load();
}, []);
const handleSubmit = async () => {
- if (!file) return toast.warn('Pilih file gambar terlebih dahulu');
-
- const res = await ApiFetch.api.fileStorage.create.post({
- file,
- name: file.name,
- });
-
- const uploaded = res.data?.data;
- if (!uploaded?.id) {
- return toast.error('Gagal upload gambar');
+ if (!potensiState.create.form.name?.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(potensiState.create.form.deskripsi)) {
+ toast.error('Deskripsi singkat wajib diisi');
+ return;
+ }
+
+ if (!potensiState.create.form.kategoriId) {
+ toast.error('Kategori wajib dipilih');
+ return;
+ }
+
+ if (!file) {
+ toast.error('Gambar wajib dipilih');
+ return;
+ }
+
+ if (isHtmlEmpty(potensiState.create.form.content)) {
+ toast.error('Konten lengkap wajib diisi');
+ return;
}
- potensiState.create.form.imageId = uploaded.id;
+ try {
+ setIsSubmitting(true);
+ if (!file) return toast.warn('Pilih file gambar terlebih dahulu');
- await potensiState.create.create();
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
- resetForm();
- router.push('/admin/desa/potensi/list-potensi');
+ const uploaded = res.data?.data;
+ if (!uploaded?.id) {
+ return toast.error('Gagal upload gambar');
+ }
+
+ potensiState.create.form.imageId = uploaded.id;
+
+ await potensiState.create.create();
+
+ resetForm();
+ router.push('/admin/desa/potensi/list-potensi');
+ } catch (error) {
+ console.error('Error creating potensi:', error);
+ toast.error('Terjadi kesalahan saat menambahkan potensi');
+ } finally {
+ setIsSubmitting(false);
+ }
};
const resetForm = () => {
@@ -69,14 +122,12 @@ function CreatePotensi() {
};
return (
-
+
{/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Tambah Potensi Desa
@@ -93,7 +144,7 @@ function CreatePotensi() {
{/* Judul */}
(potensiState.create.form.name = val.target.value)}
label="Judul"
placeholder="Masukkan judul potensi"
@@ -115,6 +166,32 @@ function CreatePotensi() {
{/* Kategori */}
({
+ label: item.nama,
+ value: item.id,
+ })) || []}
+ value={potensiState.create.form.kategoriId || null}
+ onChange={(val: string | null) => {
+ if (val) {
+ const selected = potensiDesaState.kategoriPotensi.findMany.data?.find(
+ (item) => item.id === val
+ );
+ if (selected) {
+ potensiState.create.form.kategoriId = selected.id;
+ }
+ } else {
+ potensiState.create.form.kategoriId = '';
+ }
+ }}
+ searchable
+ clearable
+ nothingFoundMessage="Tidak ditemukan"
+ required
+ />
+
+ {/*
+ /> */}
{/* Upload Gambar */}
@@ -142,7 +219,7 @@ function CreatePotensi() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -160,17 +237,44 @@ function CreatePotensi() {
Seret gambar atau klik untuk memilih file (maks 5MB)
+
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
+
{previewImage && (
-
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -190,17 +294,31 @@ function CreatePotensi() {
{/* Tombol Simpan */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/potensi/list-potensi/page.tsx b/src/app/admin/(dashboard)/desa/potensi/list-potensi/page.tsx
index 0599c55b..5ed853da 100644
--- a/src/app/admin/(dashboard)/desa/potensi/list-potensi/page.tsx
+++ b/src/app/admin/(dashboard)/desa/potensi/list-potensi/page.tsx
@@ -18,8 +18,7 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
@@ -27,6 +26,8 @@ import { useEffect, useState } from 'react';
import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import potensiDesaState from '../../../_state/desa/potensi';
+import { useDebouncedValue } from '@mantine/hooks';
+import DOMPurify from 'dompurify';
function Potensi() {
const [search, setSearch] = useState("");
@@ -47,6 +48,7 @@ function Potensi() {
function ListPotensi({ search }: { search: string }) {
const potensiState = useProxy(potensiDesaState);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const {
data,
@@ -58,43 +60,63 @@ function ListPotensi({ search }: { search: string }) {
useEffect(() => {
potensiState.kategoriPotensi.findMany.load();
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
- const filteredData = data || []
+ const filteredData = data || [];
if (loading || !data) {
return (
-
+
);
}
return (
-
-
-
- Daftar Potensi Desa
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/potensi/list-potensi/create')}
- >
- Tambah Baru
-
-
+
+
+
+
+ Daftar Potensi Desa
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/potensi/list-potensi/create')}
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Judul
- Kategori
- Deskripsi
- Detail
+
+
+ Judul
+
+
+
+
+ Kategori
+
+
+
+
+ Deskripsi
+
+
+
+
+ Detail
+
+
@@ -102,27 +124,28 @@ function ListPotensi({ search }: { search: string }) {
filteredData.map((item) => (
-
-
- {item.name}
-
-
+
+ {item.name}
+
-
- {item.kategori?.nama || '-'}
-
+
+ {item.kategori?.nama || '-'}
+
-
-
-
+
-
- Tidak ada data potensi yang cocok
+
+
+ Tidak ada data potensi yang cocok
+
@@ -150,7 +175,69 @@ function ListPotensi({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+
+ {filteredData.map((item) => (
+
+
+
+ Judul
+
+
+ {item.name}
+
+
+
+
+ Kategori
+
+
+ {item.kategori?.nama || '-'}
+
+
+
+
+ Deskripsi
+
+
+
+ }
+ onClick={() => router.push(`/admin/desa/potensi/list-potensi/${item.id}`)}
+ w="100%"
+ >
+ Detail
+
+
+ ))}
+
+ ) : (
+
+
+ Tidak ada data potensi yang cocok
+
+
+ )}
+
+
+ },
+ {
+ label: "Profil Perbekel",
+ value: "profilperbekel",
+ href: "/admin/desa/profil/profil-perbekel",
+ icon:
+ },
+ {
+ label: "Profil Perbekel Dari Masa Ke Masa",
+ value: "profilperbekeldarimasakemasa",
+ href: "/admin/desa/profil/profil-perbekel-dari-masa-ke-masa",
+ icon:
+ }
+ ];
+
+ const currentTab = tabs.find(tab => tab.href === pathname)
+ const [activeTab, setActiveTab] = useState(currentTab?.value || tabs[0].value);
+
+ const handleTabChange = (value: string | null) => {
+ const tab = tabs.find(t => t.value === value)
+ if (tab) {
+ router.push(tab.href)
+ }
+ setActiveTab(value)
+ }
+
+ useEffect(() => {
+ const match = tabs.find(tab => tab.href === pathname)
+ if (match) {
+ setActiveTab(match.value)
+ }
+ }, [pathname])
+
+ return (
+
+ Profile Desa
+
+ {/* ✅ Scroll horizontal wrapper */}
+
+
+
+ {tabs.map((tab, i) => (
+
+ {tab.label}
+
+ ))}
+
+
+
+
+
+
+
+
+ {tabs.map((tab, i) => (
+
+ {tab.label}
+
+ ))}
+
+
+
+
+ {tabs.map((tab, i) => (
+
+ {/* Konten dummy, bisa diganti sesuai routing */}
+ <>{children}>
+
+ ))}
+
+
+ );
+}
+
+export default LayoutTabsDetail;
diff --git a/src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsEdit.tsx b/src/app/admin/(dashboard)/desa/profil/_lib/layoutTabsEdit.tsx
similarity index 89%
rename from src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsEdit.tsx
rename to src/app/admin/(dashboard)/desa/profil/_lib/layoutTabsEdit.tsx
index 75201d3d..c384b326 100644
--- a/src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsEdit.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/_lib/layoutTabsEdit.tsx
@@ -12,22 +12,22 @@ function LayoutTabsEdit({ children }: { children: React.ReactNode }) {
{
label: "Sejarah Desa",
value: "sejarahdesa",
- href: "/admin/desa/profile/edit/sejarah_desa"
+ href: "/admin/desa/profil/edit/sejarah_desa"
},
{
label: "Visi Misi Desa",
value: "visimisidesa",
- href: "/admin/desa/profile/edit/visi_misi_desa"
+ href: "/admin/desa/profil/edit/visi_misi_desa"
},
{
label: "Lambang Desa",
value: "lambangdesa",
- href: "/admin/desa/profile/edit/lambang_desa"
+ href: "/admin/desa/profil/edit/lambang_desa"
},
{
label: "Maskot Desa",
value: "maskotdesa",
- href: "/admin/desa/profile/edit/maskot_desa"
+ href: "/admin/desa/profil/edit/maskot_desa"
},
];
const curentTab = tabs.find(tab => tab.href === pathname)
diff --git a/src/app/admin/(dashboard)/desa/profil/layout.tsx b/src/app/admin/(dashboard)/desa/profil/layout.tsx
new file mode 100644
index 00000000..475635df
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/layout.tsx
@@ -0,0 +1,33 @@
+'use client'
+
+import { usePathname } from "next/navigation";
+import LayoutTabsDetail from "./_lib/layoutTabsDetail"
+import { Box } from "@mantine/core";
+
+export default function Layout({ children }: { children: React.ReactNode }) {
+ const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/berita/semua → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
+
+ return (
+
+ {children}
+
+ )
+}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/lambang_desa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/lambang_desa/page.tsx
new file mode 100644
index 00000000..955ce10d
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/lambang_desa/page.tsx
@@ -0,0 +1,266 @@
+'use client';
+import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
+import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
+import colors from '@/con/colors';
+import {
+ Alert,
+ Box,
+ Button,
+ Center,
+ Group,
+ Loader,
+ Paper,
+ Stack,
+ Text,
+ TextInput,
+ Title,
+} from '@mantine/core';
+import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useEffect, useState } from 'react';
+import { toast } from 'react-toastify';
+
+// 🧩 Type untuk form
+interface FormData {
+ judul: string;
+ deskripsi: string;
+}
+
+// 🧩 Main Component
+function Page() {
+ const router = useRouter();
+ const params = useParams();
+
+ const [formData, setFormData] = useState({ judul: '', deskripsi: '' });
+ const [originalData, setOriginalData] = useState({ judul: '', deskripsi: '' });
+ const [isLoading, setIsLoading] = useState(true);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const [loadError, setLoadError] = useState(null);
+
+ // 🧭 Load data awal
+ useEffect(() => {
+ const loadData = async () => {
+ const id = params?.id as string;
+ if (!id) {
+ toast.error('ID tidak valid');
+ router.push('/admin/desa/profil/profil-desa');
+ return;
+ }
+
+ setIsLoading(true);
+ setLoadError(null);
+
+ try {
+ const data = await stateProfileDesa.lambangDesa.findUnique.load(id);
+
+ if (data) {
+ const initial: FormData = {
+ judul: data.judul || '',
+ deskripsi: data.deskripsi || '',
+ };
+ setFormData(initial);
+ setOriginalData(initial);
+
+ // Penting untuk isi id di state sebelum submit
+ stateProfileDesa.lambangDesa.update.initialize(data);
+ } else {
+ setLoadError('Data tidak ditemukan');
+ }
+ } catch (error) {
+ console.error('Error loading lambang:', error);
+ setLoadError('Gagal memuat data lambang desa');
+ toast.error('Gagal memuat data lambang desa');
+ } finally {
+ setIsLoading(false);
+ }
+ };
+
+ loadData();
+
+ return () => {
+ stateProfileDesa.lambangDesa.update.reset();
+ stateProfileDesa.lambangDesa.findUnique.reset();
+ };
+ }, [params?.id, router]);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.judul?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi)
+ );
+ };
+
+ // 🔁 Reset form
+ const handleResetForm = () => {
+ setFormData(originalData);
+ toast.info('Form dikembalikan ke data awal');
+ };
+
+ // 💾 Submit handler
+ const handleSubmit = async () => {
+ if (!formData.judul.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ setIsSubmitting(true);
+ try {
+ const state = stateProfileDesa.lambangDesa;
+ state.update.form.judul = formData.judul;
+ state.update.form.deskripsi = formData.deskripsi;
+
+ const success = await state.update.submit();
+
+ if (success) {
+ toast.success('Data berhasil disimpan');
+ router.push('/admin/desa/profil/profil-desa');
+ } else {
+ toast.error('Gagal menyimpan data');
+ }
+ } catch (error) {
+ console.error('Error update lambang desa:', error);
+ toast.error('Terjadi kesalahan saat update lambang desa');
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ // 📝 Handlers
+ const handleJudulChange = (e: React.ChangeEvent) => {
+ setFormData(prev => ({ ...prev, judul: e.target.value }));
+ };
+
+ const handleDeskripsiChange = (html: string) => {
+ setFormData(prev => ({ ...prev, deskripsi: html }));
+ };
+
+ const handleBack = () => router.back();
+
+ // 🔄 Loading
+ if (isLoading) {
+ return (
+
+
+
+
+
+ Memuat data lambang desa...
+
+
+
+
+ );
+ }
+
+ // ❌ Error
+ if (loadError) {
+ return (
+
+
+
+
+
+ } color="red" title="Terjadi Kesalahan" radius="md">
+ {loadError}
+
+ router.push('/admin/desa/profil/profil-desa')} variant="outline">
+ Kembali ke Halaman Utama
+
+
+
+ );
+ }
+
+ // 🧱 UI utama
+ return (
+
+
+ {/* Header */}
+
+
+
+
+
+ Edit Lambang Desa
+
+
+
+ {/* Form */}
+
+
+ {/* Judul */}
+ Judul}
+ placeholder="Masukkan judul lambang desa"
+ value={formData.judul}
+ onChange={handleJudulChange}
+ error={!formData.judul.trim() && 'Judul wajib diisi'}
+ required
+ size="md"
+ radius="md"
+ />
+
+ {/* Deskripsi */}
+
+
+ Deskripsi
+
+
+
+
+ {/* Tombol Aksi */}
+
+
+ Batal
+
+
+ {isSubmitting ? : 'Simpan'}
+
+
+
+
+
+
+ );
+}
+
+export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/maskot_desa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/maskot_desa/page.tsx
similarity index 72%
rename from src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/maskot_desa/page.tsx
rename to src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/maskot_desa/page.tsx
index 8ead63e3..f5dfff4d 100644
--- a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/maskot_desa/page.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/maskot_desa/page.tsx
@@ -5,9 +5,9 @@ import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
-import { Box, Button, Group, Image, Paper, SimpleGrid, Stack, Text, TextInput, Title, Tooltip, Center, Alert } from '@mantine/core';
+import { Alert, Box, Button, Center, Group, Image, Loader, Paper, SimpleGrid, Stack, Text, TextInput, Title } from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
-import { IconArrowBack, IconPhoto, IconUpload, IconX, IconAlertCircle } from '@tabler/icons-react';
+import { IconAlertCircle, IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
import { useEffect, useState } from 'react';
import { toast } from 'react-toastify';
@@ -19,8 +19,8 @@ function Page() {
const params = useParams();
const [images, setImages] = useState<
- Array<{ file: File | null; preview: string; label: string; imageId?: string }>
->([]);
+ Array<{ file: File | null; preview: string; label: string; imageId?: string }>
+ >([]);
const [formData, setFormData] = useState({
judul: '',
deskripsi: '',
@@ -28,13 +28,34 @@ function Page() {
});
const [isSubmitting, setIsSubmitting] = useState(false);
+ const [originalData, setOriginalData] = useState({
+ judul: "",
+ deskripsi: "",
+ images: [] as Array<{ label: string; imageId: string }>
+ });
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.judul?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi)
+ );
+ };
+
// Load data
useEffect(() => {
const loadData = async () => {
const id = params?.id as string;
if (!id) {
toast.error("ID tidak valid");
- router.push("/admin/desa/profile/profile-desa");
+ router.push("/admin/desa/profil/profil-desa");
return;
}
@@ -52,6 +73,17 @@ function Page() {
})),
});
+ setOriginalData({
+ judul: data.judul || '',
+ deskripsi: data.deskripsi || '',
+ images: (data.images || []).map((img: any) => ({
+ label: img.label,
+ imageId: img.image?.id ?? '',
+ preview: img.image?.link ?? '',
+ })),
+ });
+
+
if (data?.images?.length > 0 && data.images[0].image?.link) {
setImages(data.images.map((img: any) => ({
file: null,
@@ -77,15 +109,42 @@ function Page() {
const handleBack = () => router.back();
+ const handleResetForm = () => {
+ setFormData({
+ judul: originalData.judul,
+ deskripsi: originalData.deskripsi,
+ images: originalData.images.map((img) => ({
+ label: img.label,
+ imageId: img.imageId,
+ })),
+ });
+
+ setImages(
+ originalData.images.map((img: any) => ({
+ file: null,
+ preview: img.preview, // pakai preview masing-masing, bukan cuma satu
+ label: img.label,
+ imageId: img.imageId,
+ }))
+ );
+
+ toast.info("Form dikembalikan ke data awal");
+ };
+
+
const handleSubmit = async () => {
if (isSubmitting || !formData.judul.trim()) {
toast.error("Judul wajib diisi");
return;
}
-
- setIsSubmitting(true);
-
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error("Deskripsi wajib diisi");
+ return;
+ }
+
try {
+ setIsSubmitting(true);
const uploadedImages = [];
// Upload semua gambar baru
@@ -95,7 +154,7 @@ function Page() {
uploadedImages.push({ imageId: img.imageId, label: img.label });
continue;
}
-
+
// upload baru
const res = await ApiFetch.api.fileStorage.create.post({
file: img.file,
@@ -108,7 +167,7 @@ function Page() {
}
uploadedImages.push({ imageId: uploaded.id, label: img.label || "main" });
}
-
+
// Update ke global state
maskotState.update.updateField("judul", formData.judul);
@@ -119,7 +178,7 @@ function Page() {
if (success) {
toast.success("Maskot berhasil diperbarui!");
- router.push("/admin/desa/profile/profile-desa");
+ router.push("/admin/desa/profil/profil-desa");
}
} catch (error) {
console.error("Error update maskot:", error);
@@ -132,7 +191,7 @@ function Page() {
// Loading state
if (maskotState.findUnique.loading || maskotState.update.loading) {
return (
-
+
Memuat data...
@@ -143,7 +202,7 @@ function Page() {
// Error state
if (maskotState.findUnique.error) {
return (
-
+
@@ -158,14 +217,12 @@ function Page() {
}
return (
-
+
-
-
-
-
-
+
+
+
Edit Maskot Desa
@@ -177,7 +234,7 @@ function Page() {
Judul}
placeholder="Masukkan judul maskot"
- defaultValue={formData.judul}
+ value={formData.judul}
onChange={(e) => setFormData({ ...formData, judul: e.currentTarget.value })}
error={!formData.judul && "Judul wajib diisi"}
/>
@@ -233,7 +290,7 @@ function Page() {
setImages(updated);
}}
>
- Hapus
+
{/* Buttons */}
-
+
+ {/* Tombol Batal */}
- {isSubmitting ? 'Menyimpan...' : 'Simpan Perubahan'}
-
-
Batal
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/sejarah_desa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/sejarah_desa/page.tsx
new file mode 100644
index 00000000..10f3105e
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/sejarah_desa/page.tsx
@@ -0,0 +1,302 @@
+'use client';
+import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
+import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
+import colors from '@/con/colors';
+import {
+ Alert,
+ Box,
+ Button,
+ Center,
+ Group,
+ Loader,
+ Paper,
+ Stack,
+ Text,
+ TextInput,
+ Title
+} from '@mantine/core';
+import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useEffect, useState } from 'react';
+import { toast } from 'react-toastify';
+
+// 🔹 Types
+interface FormData {
+ judul: string;
+ deskripsi: string;
+}
+
+// 🔹 Main Component
+function Page() {
+ const router = useRouter();
+ const params = useParams();
+
+ // 🧩 Local State
+ const [formData, setFormData] = useState({
+ judul: '',
+ deskripsi: '',
+ });
+ const [originalData, setOriginalData] = useState({
+ judul: '',
+ deskripsi: '',
+ });
+ const [isLoading, setIsLoading] = useState(true);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const [loadError, setLoadError] = useState(null);
+
+ // 🧭 Load Initial Data
+ useEffect(() => {
+ const loadData = async () => {
+ const id = params?.id as string;
+ if (!id) {
+ toast.error('ID tidak valid');
+ router.push('/admin/desa/profil/profil-desa');
+ return;
+ }
+
+ setIsLoading(true);
+ setLoadError(null);
+
+ try {
+ const data = await stateProfileDesa.sejarahDesa.findUnique.load(id);
+
+ if (data) {
+ const initialData: FormData = {
+ judul: data.judul || '',
+ deskripsi: data.deskripsi || '',
+ };
+
+ setFormData(initialData);
+ setOriginalData(initialData);
+
+ stateProfileDesa.sejarahDesa.update.initialize(data);
+ } else {
+ setLoadError('Data tidak ditemukan');
+ }
+ } catch (error) {
+ console.error('Error loading sejarah:', error);
+ setLoadError('Gagal memuat data sejarah desa');
+ toast.error('Gagal memuat data sejarah desa');
+ } finally {
+ setIsLoading(false);
+ }
+ };
+
+ loadData();
+
+ return () => {
+ stateProfileDesa.sejarahDesa.update.reset();
+ stateProfileDesa.sejarahDesa.findUnique.reset();
+ };
+ }, [params?.id, router]);
+
+ // 🔄 Check if form has changes
+
+
+ // 🔁 Reset Form to Original Data
+ const handleResetForm = () => {
+ setFormData(originalData);
+ toast.info('Form dikembalikan ke data awal');
+ };
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.judul?.trim() !== '' &&
+ !isHtmlEmpty(formData.deskripsi)
+ );
+ };
+
+ // 💾 Submit Handler
+ const handleSubmit = async () => {
+ // Validation
+ if (!formData.judul.trim()) {
+ toast.error('Judul wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.deskripsi)) {
+ toast.error('Deskripsi wajib diisi');
+ return;
+ }
+
+ setIsSubmitting(true);
+
+ try {
+ // Access original state directly (not proxy)
+ const originalState = stateProfileDesa.sejarahDesa;
+
+ // Update form data
+ originalState.update.form.judul = formData.judul;
+ originalState.update.form.deskripsi = formData.deskripsi;
+
+ // Submit
+ const success = await originalState.update.submit();
+
+ if (success) {
+ toast.success('Data berhasil disimpan');
+ router.push('/admin/desa/profil/profil-desa');
+ } else {
+ toast.error('Gagal menyimpan data');
+ }
+ } catch (error) {
+ console.error('Error update sejarah desa:', error);
+ toast.error('Terjadi kesalahan saat update sejarah desa');
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ // 📝 Form Field Handlers
+ const handleJudulChange = (e: React.ChangeEvent) => {
+ setFormData(prev => ({ ...prev, judul: e.target.value }));
+ };
+
+ const handleDeskripsiChange = (html: string) => {
+ setFormData(prev => ({ ...prev, deskripsi: html }));
+ };
+
+ const handleBack = () => {
+ router.back();
+ };
+ // 🔄 Loading State
+ if (isLoading) {
+ return (
+
+
+
+
+
+ Memuat data...
+
+
+
+
+ );
+ }
+
+ // ❌ Error State
+ if (loadError) {
+ return (
+
+
+ router.back()} p="xs" radius="md">
+
+
+ }
+ color="red"
+ title="Terjadi Kesalahan"
+ radius="md"
+ >
+ {loadError}
+
+ router.push('/admin/desa/profil/profil-desa')}
+ variant="outline"
+ >
+ Kembali ke Halaman Utama
+
+
+
+ );
+ }
+
+ return (
+
+
+ {/* Header */}
+
+
+
+
+
+ Edit Sejarah Desa
+
+
+
+ {/* Form */}
+
+
+ {/* Form Title */}
+
+
+ Informasi Sejarah Desa
+
+
+ {/* Judul Field */}
+ Judul Sejarah}
+ placeholder="Masukkan judul sejarah desa"
+ value={formData.judul}
+ onChange={handleJudulChange}
+ error={!formData.judul.trim() && 'Judul wajib diisi'}
+ required
+ size="md"
+ radius="md"
+ />
+
+ {/* Deskripsi Field */}
+
+
+ Deskripsi Sejarah
+
+
+
+
+
+ {/* Action Buttons */}
+
+ {/* Tombol Batal */}
+
+ Batal
+
+
+ {/* Tombol Simpan */}
+
+ {isSubmitting ? : 'Simpan'}
+
+
+
+
+
+
+ );
+}
+
+export default Page;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/visi_misi_desa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/visi_misi_desa/page.tsx
new file mode 100644
index 00000000..7a743755
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-desa/[id]/visi_misi_desa/page.tsx
@@ -0,0 +1,269 @@
+'use client';
+
+import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
+import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
+import colors from '@/con/colors';
+import {
+ Alert,
+ Box,
+ Button,
+ Center,
+ Group,
+ Loader,
+ Paper,
+ Stack,
+ Text,
+ Title,
+} from '@mantine/core';
+import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
+import { useParams, useRouter } from 'next/navigation';
+import { useEffect, useState } from 'react';
+import { toast } from 'react-toastify';
+
+// 🔹 Types
+interface FormData {
+ visi: string;
+ misi: string;
+}
+
+// 🔹 Main Component
+function Page() {
+ const router = useRouter();
+ const params = useParams();
+ const [formData, setFormData] = useState({ visi: '', misi: '' });
+ const [originalData, setOriginalData] = useState({ visi: '', misi: '' });
+ const [isLoading, setIsLoading] = useState(true);
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const [loadError, setLoadError] = useState(null);
+
+ // 🧭 Load Data
+ useEffect(() => {
+ const loadData = async () => {
+ const id = params?.id as string;
+ if (!id) {
+ toast.error('ID tidak valid');
+ router.push('/admin/desa/profil/profil-desa');
+ return;
+ }
+
+ setIsLoading(true);
+ setLoadError(null);
+
+ try {
+ const data = await stateProfileDesa.visiMisiDesa.findUnique.load(id);
+ if (data) {
+ const initialData: FormData = {
+ visi: data.visi || '',
+ misi: data.misi || '',
+ };
+ setFormData(initialData);
+ setOriginalData(initialData);
+
+ // set id ke state agar submit pakai endpoint benar
+ stateProfileDesa.visiMisiDesa.update.initialize(data);
+ } else {
+ setLoadError('Data tidak ditemukan');
+ }
+ } catch (error) {
+ console.error('Error load visi misi:', error);
+ setLoadError('Gagal memuat data visi misi desa');
+ toast.error('Gagal memuat data visi misi desa');
+ } finally {
+ setIsLoading(false);
+ }
+ };
+
+ loadData();
+
+ return () => {
+ stateProfileDesa.visiMisiDesa.update.reset();
+ stateProfileDesa.visiMisiDesa.findUnique.reset();
+ };
+ }, [params?.id, router]);
+
+ // Helper function to check if HTML content is empty
+ const isHtmlEmpty = (html: string) => {
+ // Remove all HTML tags and check if there's any text content
+ const textContent = html.replace(/<[^>]*>/g, '').trim();
+ return textContent === '';
+ };
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ !isHtmlEmpty(formData.visi) &&
+ !isHtmlEmpty(formData.misi)
+ );
+ };
+
+ // 🔄 Reset Form
+ const handleResetForm = () => {
+ setFormData(originalData);
+ toast.info('Form dikembalikan ke data awal');
+ };
+
+ // 💾 Submit
+ const handleSubmit = async () => {
+ if (isHtmlEmpty(formData.visi)) {
+ toast.error('Visi wajib diisi');
+ return;
+ }
+
+ if (isHtmlEmpty(formData.misi)) {
+ toast.error('Misi wajib diisi');
+ return;
+ }
+
+ setIsSubmitting(true);
+ try {
+ const originalState = stateProfileDesa.visiMisiDesa;
+
+ // update data form ke state sebelum submit
+ originalState.update.form.visi = formData.visi;
+ originalState.update.form.misi = formData.misi;
+
+ const success = await originalState.update.submit();
+
+ if (success) {
+ toast.success('Data berhasil disimpan');
+ router.push('/admin/desa/profil/profil-desa');
+ } else {
+ toast.error('Gagal menyimpan data');
+ }
+ } catch (error) {
+ console.error('Error update visi misi desa:', error);
+ toast.error('Terjadi kesalahan saat update visi misi desa');
+ } finally {
+ setIsSubmitting(false);
+ }
+ };
+
+ // 🧭 Field handlers
+ const handleVisiChange = (html: string) => setFormData(prev => ({ ...prev, visi: html }));
+ const handleMisiChange = (html: string) => setFormData(prev => ({ ...prev, misi: html }));
+ const handleBack = () => router.back();
+
+ // ⏳ Loading
+ if (isLoading) {
+ return (
+
+
+
+
+
+ Memuat data...
+
+
+
+
+ );
+ }
+
+ // ❌ Error
+ if (loadError) {
+ return (
+
+
+
+
+
+ }
+ color="red"
+ title="Terjadi Kesalahan"
+ radius="md"
+ >
+ {loadError}
+
+ router.push('/admin/desa/profil/profil-desa')}
+ variant="outline"
+ >
+ Kembali ke Halaman Utama
+
+
+
+ );
+ }
+
+ // ✅ UI
+ return (
+
+
+ {/* Header */}
+
+
+
+
+
+ Edit Visi & Misi Desa
+
+
+
+ {/* Form */}
+
+
+
+
+ Informasi Visi & Misi Desa
+
+
+
+ {/* Visi */}
+
+
+ Visi
+
+
+
+
+ {/* Misi */}
+
+
+ Misi
+
+
+
+
+ {/* Actions */}
+
+
+ Batal
+
+
+ {isSubmitting ? : 'Simpan'}
+
+
+
+
+
+
+ );
+}
+
+export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-desa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-desa/page.tsx
new file mode 100644
index 00000000..5d7a166b
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-desa/page.tsx
@@ -0,0 +1,590 @@
+'use client'
+
+import colors from '@/con/colors';
+import { Box, Button, Card, Center, Divider, Group, Image, Paper, SimpleGrid, Stack, Text, Title } from '@mantine/core';
+import { IconEdit } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useEffect } from 'react';
+import { useSnapshot } from 'valtio';
+import stateProfileDesa from '../../../_state/desa/profile';
+
+function Page() {
+ const router = useRouter();
+ const snap = useSnapshot(stateProfileDesa);
+
+ useEffect(() => {
+ stateProfileDesa.sejarahDesa.findUnique.load("edit");
+ stateProfileDesa.visiMisiDesa.findUnique.load("edit");
+ stateProfileDesa.lambangDesa.findUnique.load("edit");
+ stateProfileDesa.maskotDesa.findUnique.load("edit");
+ }, []);
+
+ const sejarah = snap.sejarahDesa.findUnique.data;
+ const visiMisi = snap.visiMisiDesa.findUnique.data;
+ const lambang = snap.lambangDesa.findUnique.data;
+ const maskot = snap.maskotDesa.findUnique.data;
+
+ return (
+
+
+ Preview Profil Desa
+
+ {/* Sejarah Desa */}
+
+
+ {sejarah && (
+
+
+ Preview Sejarah Desa
+ }
+ radius="md"
+ onClick={() => router.push(`/admin/desa/profil/profil-desa/${sejarah.id}/sejarah_desa`)}
+ >
+ Edit
+
+
+
+
+
+
+
+
+
+
+
+ {sejarah.judul}
+
+
+
+
+
+
+
+
+
+
+ )}
+
+
+
+ {sejarah && (
+
+ {/* Header */}
+
+
+ Preview Sejarah Desa
+
+
+ }
+ onClick={() =>
+ router.push(`/admin/desa/profil/profil-desa/${sejarah.id}/sejarah_desa`)
+ }
+ >
+ Edit
+
+
+
+ {/* Content Wrapper */}
+
+
+ {/* Logo + Title */}
+
+
+
+
+
+ {sejarah.judul}
+
+
+
+
+
+
+ {/* Deskripsi */}
+
+
+
+
+
+
+ )}
+
+
+ {/* Visi Misi Desa */}
+
+ {visiMisi && (
+
+
+ Preview Visi Misi Desa
+ }
+ radius="md"
+ onClick={() => router.push(`/admin/desa/profil/profil-desa/${visiMisi.id}/visi_misi_desa`)}
+ >
+ Edit
+
+
+
+
+
+
+
+
+
+
+
+
+ Visi Misi Desa
+
+
+
+
+
+ Visi Desa
+
+
+
+ Misi Desa
+
+
+
+
+
+ )}
+
+
+
+ {visiMisi && (
+
+ {/* Header */}
+
+
+ Preview Visi Misi Desa
+
+
+ }
+ onClick={() => router.push(`/admin/desa/profil/profil-desa/${visiMisi.id}/visi_misi_desa`)}
+ >
+ Edit
+
+
+
+ {/* Content Wrapper */}
+
+
+ {/* Logo + Title */}
+
+
+
+
+
+ Visi Misi Desa
+
+
+
+
+
+
+
+ Visi Desa
+
+
+
+
+ Misi Desa
+
+
+
+
+
+ )}
+
+
+ {/* Lambang Desa */}
+
+ {lambang && (
+
+
+ Preview Lambang Desa
+ }
+ radius="md"
+ onClick={() => router.push(`/admin/desa/profil/profil-desa/${lambang.id}/lambang_desa`)}
+ >
+ Edit
+
+
+
+
+
+
+
+
+
+
+
+ {lambang.judul}
+
+
+
+
+
+
+
+
+
+
+ )}
+
+
+
+ {lambang && (
+
+ {/* Header */}
+
+
+ Preview Lambang Desa
+
+
+ }
+ onClick={() =>
+ router.push(`/admin/desa/profil/profil-desa/${lambang.id}/lambang_desa`)
+ }
+ >
+ Edit
+
+
+
+ {/* Content Wrapper */}
+
+
+ {/* Logo + Title */}
+
+
+
+
+
+ {lambang.judul}
+
+
+
+
+
+
+ {/* Deskripsi */}
+
+
+
+
+
+
+ )}
+
+
+ {/* Maskot Desa */}
+
+ {maskot && (
+
+
+ Preview Maskot Desa
+ }
+ radius="md"
+ onClick={() => router.push(`/admin/desa/profil/profil-desa/${maskot.id}/maskot_desa`)}
+ >
+ Edit
+
+
+
+
+
+
+
+
+
+
+
+ Maskot Desa
+
+
+
+
+
+
+
+
+
+ {maskot.images.map((img, idx) => (
+
+
+
+
+ {img.label}
+
+ ))}
+
+
+
+
+
+ )}
+
+
+
+ {maskot && (
+
+ {/* Header */}
+
+
+ Preview Maskot Desa
+
+
+ }
+ onClick={() =>
+ router.push(`/admin/desa/profil/profil-desa/${maskot.id}/maskot_desa`)
+ }
+ >
+ Edit
+
+
+
+ {/* Content Wrapper */}
+
+
+ {/* Logo + Title */}
+
+
+
+
+
+ Maskot Desa
+
+
+
+
+
+
+ {/* Deskripsi */}
+
+
+
+
+
+ {maskot.images.map((img, idx) => (
+
+
+
+
+ {img.label}
+
+ ))}
+
+
+
+
+
+ )}
+
+
+
+ );
+}
+
+export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx
similarity index 70%
rename from src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx
rename to src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx
index 56f2c2e2..c83faf16 100644
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/edit/page.tsx
@@ -4,6 +4,7 @@ import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
import {
+ ActionIcon,
Box,
Button,
Group,
@@ -13,7 +14,7 @@ import {
Text,
TextInput,
Title,
- Tooltip
+ Loader
} from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
@@ -36,6 +37,16 @@ function EditPerbekelDariMasaKeMasa() {
imageId: ''
});
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ const [originalData, setOriginalData] = useState({
+ nama: '',
+ daerah: '',
+ periode: '',
+ imageId: '',
+ imageUrl: "",
+ });
+
// load data pertama kali
useEffect(() => {
const loadFoto = async () => {
@@ -50,9 +61,14 @@ function EditPerbekelDariMasaKeMasa() {
periode: data.periode || '',
imageId: data.imageId || ''
});
- if (data?.imageGalleryFoto?.link) {
- setPreviewImage(data.imageGalleryFoto.link);
- }
+ setOriginalData({
+ nama: data.nama || '',
+ daerah: data.daerah || '',
+ periode: data.periode || '',
+ imageId: data.imageId || '',
+ imageUrl: data.image.link || '',
+ })
+ setPreviewImage(data.image.link);
}
} catch (error) {
console.error('Error loading foto:', error);
@@ -70,8 +86,22 @@ function EditPerbekelDariMasaKeMasa() {
}));
};
+ const handleResetForm = () => {
+ setFormData({
+ nama: originalData.nama,
+ daerah: originalData.daerah,
+ periode: originalData.periode,
+ imageId: originalData.imageId,
+ });
+ setPreviewImage(originalData.imageUrl || null);
+ setFile(null);
+ toast.info("Form dikembalikan ke data awal");
+ };
+
+
const handleSubmit = async () => {
try {
+ setIsSubmitting(true);
// update global state hanya sekali pas submit
state.update.form = { ...state.update.form, ...formData };
@@ -87,21 +117,21 @@ function EditPerbekelDariMasaKeMasa() {
await state.update.update();
toast.success('Perbekel dari masa ke masa berhasil diperbarui!');
- router.push('/admin/desa/profile/profile-perbekel-dari-masa-ke-masa');
+ router.push('/admin/desa/profil/profil-perbekel-dari-masa-ke-masa');
} catch (error) {
console.error('Error updating perbekel dari masa ke masa:', error);
toast.error('Terjadi kesalahan saat memperbarui perbekel dari masa ke masa');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Edit Perbekel Dari Masa Ke Masa
@@ -138,7 +168,7 @@ function EditPerbekelDariMasaKeMasa() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -157,25 +187,45 @@ function EditPerbekelDariMasaKeMasa() {
Seret gambar atau klik untuk memilih file
- Maksimal 5MB, format gambar wajib
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
{previewImage && (
-
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
@@ -197,6 +247,17 @@ function EditPerbekelDariMasaKeMasa() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/page.tsx
similarity index 87%
rename from src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/page.tsx
rename to src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/page.tsx
index 048d8494..95653279 100644
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/[id]/page.tsx
@@ -2,9 +2,9 @@
import { ModalKonfirmasiHapus } from '@/app/admin/(dashboard)/_com/modalKonfirmasiHapus';
import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
import colors from '@/con/colors';
-import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text, Tooltip } from '@mantine/core';
+import { Box, Button, Group, Image, Paper, Skeleton, Stack, Text } from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
-import { IconArrowBack, IconEdit, IconX } from '@tabler/icons-react';
+import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
import { useState } from 'react';
import { useProxy } from 'valtio/utils';
@@ -25,7 +25,7 @@ function DetailPerbekelDariMasa() {
state.delete.byId(selectedId);
setModalHapus(false);
setSelectedId(null);
- router.push("/admin/desa/profile/profile-perbekel-dari-masa-ke-masa");
+ router.push("/admin/desa/profil/profil-perbekel-dari-masa-ke-masa");
}
};
@@ -40,7 +40,7 @@ function DetailPerbekelDariMasa() {
const data = state.findUnique.data;
return (
-
+
router.back()}
@@ -52,7 +52,7 @@ function DetailPerbekelDariMasa() {
-
{
@@ -109,21 +108,18 @@ function DetailPerbekelDariMasa() {
radius="md"
size="md"
>
-
+
-
-
router.push(`/admin/desa/profile/profile-perbekel-dari-masa-ke-masa/${data.id}/edit`)}
+ onClick={() => router.push(`/admin/desa/profil/profil-perbekel-dari-masa-ke-masa/${data.id}/edit`)}
variant="light"
radius="md"
size="md"
>
-
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/create/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/create/page.tsx
similarity index 59%
rename from src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/create/page.tsx
rename to src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/create/page.tsx
index d2c5ca5e..aa4f519b 100644
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/create/page.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/create/page.tsx
@@ -2,7 +2,7 @@
import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
-import { Box, Button, Group, Image, Paper, Stack, Text, TextInput, Title, Tooltip } from '@mantine/core';
+import { Loader, ActionIcon, Box, Button, Group, Image, Paper, Stack, Text, TextInput, Title } from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
import { IconArrowBack, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
@@ -14,6 +14,7 @@ function CreatePerbekelDariMasaKeMasa() {
const state = useProxy(stateProfileDesa.mantanPerbekel);
const router = useRouter();
const [previewImage, setPreviewImage] = useState(null);
+ const [isSubmitting, setIsSubmitting] = useState(false);
const [file, setFile] = useState(null);
const resetForm = () => {
@@ -28,33 +29,39 @@ function CreatePerbekelDariMasaKeMasa() {
};
const handleSubmit = async () => {
- if (!file) {
- return toast.warn('Pilih file gambar terlebih dahulu');
+ try {
+ setIsSubmitting(true);
+ if (!file) {
+ return toast.warn('Pilih file gambar terlebih dahulu');
+ }
+
+ const res = await ApiFetch.api.fileStorage.create.post({
+ file,
+ name: file.name,
+ });
+
+ const uploaded = res.data?.data;
+ if (!uploaded?.id) return toast.error('Gagal upload gambar');
+
+ state.create.form.imageId = uploaded.id;
+ await state.create.create();
+ resetForm();
+ router.push('/admin/desa/profil/profil-perbekel-dari-masa-ke-masa');
+ } catch (error) {
+ console.error(error);
+ toast.error('Gagal menambahkan perbekel dari masa ke masa');
+ } finally {
+ setIsSubmitting(false);
}
-
- const res = await ApiFetch.api.fileStorage.create.post({
- file,
- name: file.name,
- });
-
- const uploaded = res.data?.data;
- if (!uploaded?.id) return toast.error('Gagal upload gambar');
-
- state.create.form.imageId = uploaded.id;
- await state.create.create();
- resetForm();
- router.push('/admin/desa/profile/profile-perbekel-dari-masa-ke-masa');
};
return (
-
+
{/* Back button + Title */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Create Perbekel Dari Masa Ke Masa
@@ -72,21 +79,21 @@ function CreatePerbekelDariMasaKeMasa() {
(state.create.form.nama = e.target.value)}
required
/>
(state.create.form.daerah = e.target.value)}
required
/>
(state.create.form.periode = e.target.value)}
required
/>
@@ -104,7 +111,7 @@ function CreatePerbekelDariMasaKeMasa() {
}}
onReject={() => toast.error('File tidak valid, gunakan format gambar')}
maxSize={5 * 1024 ** 2}
- accept={{ 'image/*': [] }}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
radius="md"
p="xl"
>
@@ -120,25 +127,63 @@ function CreatePerbekelDariMasaKeMasa() {
- Seret gambar atau klik untuk memilih file (maks 5MB)
+ Seret gambar atau klik untuk memilih file
+
+
+ Maksimal 5MB, format gambar .png, .jpg, .jpeg, webp
{previewImage && (
-
+
+
+ {/* Tombol hapus (pojok kanan atas) */}
+ {
+ setPreviewImage(null);
+ setFile(null);
+ }}
+ style={{
+ boxShadow: '0 2px 6px rgba(0,0,0,0.15)',
+ }}
+ >
+
+
)}
{/* Submit */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/page.tsx
new file mode 100644
index 00000000..7065da65
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel-dari-masa-ke-masa/page.tsx
@@ -0,0 +1,193 @@
+'use client'
+import colors from '@/con/colors';
+import {
+ Box,
+ Button,
+ Center,
+ Group,
+ Pagination,
+ Paper,
+ Skeleton,
+ Stack,
+ Table,
+ TableTbody,
+ TableTd,
+ TableTh,
+ TableThead,
+ TableTr,
+ Text,
+ Title,
+} from '@mantine/core';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
+import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { useProxy } from 'valtio/utils';
+import HeaderSearch from '../../../_com/header';
+import stateProfileDesa from '../../../_state/desa/profile';
+
+function PerbekelDariMasaKeMasa() {
+ const [search, setSearch] = useState("");
+ return (
+
+ }
+ value={search}
+ onChange={(e) => setSearch(e.currentTarget.value)}
+ />
+
+
+ );
+}
+
+function ListPerbekelDariMasaKeMasa({ search }: { search: string }) {
+ const state = useProxy(stateProfileDesa.mantanPerbekel)
+ const router = useRouter();
+ const { data, page, totalPages, loading, load } = state.findMany;
+ const [debouncedSearch] = useDebouncedValue(search, 1000); // 500ms delay
+
+ useShallowEffect(() => {
+ load(page, 10, debouncedSearch)
+ }, [page, debouncedSearch]);
+
+ const filteredData = data || [];
+
+ if (loading || !data) {
+ return (
+
+
+
+ );
+ }
+
+ return (
+
+
+
+
+ List Perbekel Dari Masa Ke Masa
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() => router.push('/admin/desa/profil/profil-perbekel-dari-masa-ke-masa/create')}
+ >
+ Tambah Baru
+
+
+
+ {/* Desktop Table */}
+
+
+
+
+ Nama Perbekel
+ Periode
+ Aksi
+
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+ {item.nama}
+
+
+ {item.periode}
+
+
+ }
+ onClick={() => router.push(`/admin/desa/profil/profil-perbekel-dari-masa-ke-masa/${item.id}`)}
+ >
+ Detail
+
+
+
+ ))
+ ) : (
+
+
+
+
+ Tidak ada data perbekel yang cocok
+
+
+
+
+ )}
+
+
+
+
+ {/* Mobile Card View */}
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Nama Perbekel
+ {item.nama}
+
+
+ Periode
+ {item.periode}
+
+
+ }
+ onClick={() => router.push(`/admin/desa/profil/profil-perbekel-dari-masa-ke-masa/${item.id}`)}
+ fullWidth
+ >
+ Detail
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data perbekel yang cocok
+
+
+ )}
+
+
+
+
+
+ {
+ load(newPage, 10);
+ window.scrollTo({ top: 0, behavior: 'smooth' });
+ }}
+ total={totalPages}
+ mt={{ base: 'sm', md: 'md' }}
+ mb={{ base: 'sm', md: 'md' }}
+ color="blue"
+ radius="md"
+ />
+
+
+ );
+}
+
+export default PerbekelDariMasaKeMasa;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel/[id]/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel/[id]/page.tsx
similarity index 93%
rename from src/app/admin/(dashboard)/desa/profile/profile-perbekel/[id]/page.tsx
rename to src/app/admin/(dashboard)/desa/profil/profil-perbekel/[id]/page.tsx
index 43eefc8a..a135b37e 100644
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel/[id]/page.tsx
@@ -4,9 +4,9 @@ import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
import colors from '@/con/colors';
import ApiFetch from '@/lib/api-fetch';
-import { Box, Button, Center, Group, Image, Paper, Stack, Text, Title, Tooltip } from '@mantine/core';
+import { Box, Button, Center, Group, Image, Paper, Stack, Text, Title } from '@mantine/core';
import { Dropzone } from '@mantine/dropzone';
-import { IconArrowBack, IconPhoto, IconUpload, IconX, IconImageInPicture } from '@tabler/icons-react';
+import { IconArrowBack, IconImageInPicture, IconPhoto, IconUpload, IconX } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
import { useEffect, useState } from 'react';
import { toast } from 'react-toastify';
@@ -25,7 +25,7 @@ function ProfilePerbekel() {
const id = params?.id as string;
if (!id) {
toast.error("ID tidak valid");
- router.push("/admin/desa/profile/profile-perbekel");
+ router.push("/admin/desa/profil/profil-perbekel");
return;
}
@@ -74,7 +74,7 @@ function ProfilePerbekel() {
const success = await perbekelState.edit.submit()
if (success) {
toast.success("Data berhasil disimpan");
- router.push("/admin/desa/profile/profile-perbekel");
+ router.push("/admin/desa/profil/profil-perbekel");
}
} catch (error) {
console.error("Error update sejarah desa:", error);
@@ -97,15 +97,13 @@ function ProfilePerbekel() {
}
return (
-
+
{/* Header */}
-
-
Edit Profil Perbekel
@@ -128,8 +126,8 @@ function ProfilePerbekel() {
handleFileChange(files[0])}
onReject={() => toast.error('File tidak valid.')}
- maxSize={5 * 1024 ** 2} // 5MB
- accept={{ 'image/*': [] }}
+ maxSize={5 * 1024 ** 2}
+ accept={{ 'image/*': ['.jpeg', '.jpg', '.png', '.webp'] }}
>
diff --git a/src/app/admin/(dashboard)/desa/profil/profil-perbekel/page.tsx b/src/app/admin/(dashboard)/desa/profil/profil-perbekel/page.tsx
new file mode 100644
index 00000000..e79103d6
--- /dev/null
+++ b/src/app/admin/(dashboard)/desa/profil/profil-perbekel/page.tsx
@@ -0,0 +1,147 @@
+'use client'
+
+import colors from '@/con/colors';
+import { Box, Button, Center, Divider, Grid, GridCol, Group, Image, Paper, Skeleton, Stack, Text, Title } from '@mantine/core';
+import { IconEdit } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useEffect } from 'react';
+import { useSnapshot } from 'valtio';
+import stateProfileDesa from '../../../_state/desa/profile';
+
+function Page() {
+ const router = useRouter();
+ const snap = useSnapshot(stateProfileDesa);
+
+ // Load data saat mount
+ useEffect(() => {
+ stateProfileDesa.profilPerbekel.findUnique.load("edit");
+ }, []);
+
+ const perbekel = snap.profilPerbekel.findUnique.data;
+
+ if (!perbekel) {
+ return (
+
+
+
+ );
+ }
+
+ return (
+
+
+ {/* Header + tombol edit */}
+
+ Profil Perbekel
+ }
+ radius="md"
+ onClick={() => router.push(`/admin/desa/profil/profil-perbekel/${perbekel.id}`)}
+ >
+ Edit
+
+
+
+ {/* Card Profil */}
+
+
+
+
+
+
+
+
+
+
+ Profil Pimpinan Badan Publik Desa Darmasaba
+
+
+
+
+
+
+
+
+ { e.currentTarget.src = "/perbekel.png"; }}
+ loading='lazy'
+ />
+
+
+
+ {perbekel.nama || "I.B. Surya Prabhawa Manuaba, S.H., M.H."}
+
+
+
+
+ {/* Biodata & Info */}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
+
+export default Page;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsDetail.tsx b/src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsDetail.tsx
deleted file mode 100644
index 7554a1c8..00000000
--- a/src/app/admin/(dashboard)/desa/profile/_lib/layoutTabsDetail.tsx
+++ /dev/null
@@ -1,117 +0,0 @@
-/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
-import colors from '@/con/colors';
-import { ScrollArea, Stack, Tabs, TabsList, TabsPanel, TabsTab, Title, Tooltip } from '@mantine/core';
-import { usePathname, useRouter } from 'next/navigation';
-import React, { useEffect, useState } from 'react';
-import { IconUser, IconUsers, IconCalendar } from '@tabler/icons-react';
-
-function LayoutTabsDetail({ children }: { children: React.ReactNode }) {
- const router = useRouter()
- const pathname = usePathname()
- const tabs = [
- {
- label: "Profile Desa",
- value: "profiledesa",
- href: "/admin/desa/profile/profile-desa",
- icon: ,
- tooltip: "Lihat dan kelola profil desa"
- },
- {
- label: "Profile Perbekel",
- value: "profileperbekel",
- href: "/admin/desa/profile/profile-perbekel",
- icon: ,
- tooltip: "Kelola data Perbekel"
- },
- {
- label: "Profile Perbekel Dari Masa Ke Masa",
- value: "profile-perbekel-dari-masa-ke-masa",
- href: "/admin/desa/profile/profile-perbekel-dari-masa-ke-masa",
- icon: ,
- tooltip: "Riwayat Perbekel dari masa ke masa"
- }
- ];
-
- const currentTab = tabs.find(tab => tab.href === pathname)
- const [activeTab, setActiveTab] = useState(currentTab?.value || tabs[0].value);
-
- const handleTabChange = (value: string | null) => {
- const tab = tabs.find(t => t.value === value)
- if (tab) {
- router.push(tab.href)
- }
- setActiveTab(value)
- }
-
- useEffect(() => {
- const match = tabs.find(tab => tab.href === pathname)
- if (match) {
- setActiveTab(match.value)
- }
- }, [pathname])
-
- return (
-
- Profile Desa
-
- {/* ✅ Scroll horizontal wrapper */}
-
-
- {tabs.map((tab, i) => (
-
-
- {tab.label}
-
-
- ))}
-
-
-
- {tabs.map((tab, i) => (
-
- {/* Konten dummy, bisa diganti sesuai routing */}
- <>{children}>
-
- ))}
-
-
- );
-}
-
- export default LayoutTabsDetail;
diff --git a/src/app/admin/(dashboard)/desa/profile/layout.tsx b/src/app/admin/(dashboard)/desa/profile/layout.tsx
deleted file mode 100644
index f82687f5..00000000
--- a/src/app/admin/(dashboard)/desa/profile/layout.tsx
+++ /dev/null
@@ -1,11 +0,0 @@
-'use client'
-
-import LayoutTabsDetail from "./_lib/layoutTabsDetail"
-
-export default function Layout({ children }: { children: React.ReactNode }) {
- return (
-
- {children}
-
- )
-}
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/lambang_desa/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/lambang_desa/page.tsx
deleted file mode 100644
index 03660631..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/lambang_desa/page.tsx
+++ /dev/null
@@ -1,154 +0,0 @@
-/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
-import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
-import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
-import colors from '@/con/colors';
-import { Alert, Box, Button, Center, Group, Paper, Stack, Text, TextInput, Title, Tooltip } from '@mantine/core';
-import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
-import { useParams, useRouter } from 'next/navigation';
-import { useEffect, useState } from 'react';
-import { toast } from 'react-toastify';
-import { useProxy } from 'valtio/utils';
-
-function Page() {
- const lambangState = useProxy(stateProfileDesa.lambangDesa)
- const router = useRouter()
- const params = useParams()
- const [isSubmitting, setIsSubmitting] = useState(false);
- // Load data
- useEffect(() => {
- const loadData = async () => {
- const id = params?.id as string;
- if (!id) {
- toast.error("ID tidak valid");
- router.push("/admin/desa/profile/profile-desa");
- return;
- }
-
- try {
- const data = await lambangState.findUnique.load(id);
- lambangState.update.initialize(data);
- } catch (error) {
- console.error("Error loading lambang:", error);
- toast.error("Gagal memuat data lambang desa");
- }
- };
-
- loadData();
-
- return () => {
- lambangState.update.reset();
- lambangState.findUnique.reset();
- };
- }, [params?.id, router]);
-
- const handleSubmit = async () => {
- if (isSubmitting || !lambangState.update.form.judul.trim()) {
- toast.error("Judul wajib diisi");
- return;
- }
-
- setIsSubmitting(true);
-
- try {
- const success = await lambangState.update.submit();
-
- if (success) {
- toast.success("Data berhasil disimpan");
- router.push("/admin/desa/profile/profile-desa");
- }
- } catch (error) {
- console.error("Error update lambang desa:", error);
- toast.error("Terjadi kesalahan saat update lambang desa");
- } finally {
- setIsSubmitting(false);
- }
- };
-
- const handleBack = () => router.back();
-
- // Loading state
- if (lambangState.findUnique.loading || lambangState.update.loading) {
- return (
-
-
- Memuat data...
-
-
- );
- }
-
- // Error state
- if (lambangState.findUnique.error) {
- return (
-
-
-
-
-
- } color="red">
- Error
- {lambangState.findUnique.error}
-
-
-
- );
- }
-
- return (
-
-
-
-
-
-
-
-
- Edit Lambang Desa
-
-
-
-
- Edit Lambang Desa
-
- {/* Judul */}
- Judul}
- placeholder="Judul lambang"
- defaultValue={lambangState.update.form.judul}
- onChange={(e) => lambangState.update.form.judul = e.currentTarget.value}
- error={!lambangState.update.form.judul && "Judul wajib diisi"}
- />
-
- {/* Deskripsi */}
-
- Deskripsi
- lambangState.update.form.deskripsi = val}
- />
-
-
- {/* Buttons */}
-
-
- {isSubmitting ? 'Menyimpan...' : 'Simpan Perubahan'}
-
-
-
- Batal
-
-
-
-
-
-
- );
-}
-
-export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/sejarah_desa/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/sejarah_desa/page.tsx
deleted file mode 100644
index c941dd4f..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/sejarah_desa/page.tsx
+++ /dev/null
@@ -1,155 +0,0 @@
-/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
-import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
-import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
-import colors from '@/con/colors';
-import { Alert, Box, Button, Center, Group, Paper, Stack, Text, TextInput, Title, Tooltip } from '@mantine/core';
-import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
-import { useParams, useRouter } from 'next/navigation';
-import { useEffect, useState } from 'react';
-import { toast } from 'react-toastify';
-import { useProxy } from 'valtio/utils';
-
-function Page() {
- const sejarahState = useProxy(stateProfileDesa.sejarahDesa)
- const router = useRouter()
- const params = useParams()
- const [isSubmitting, setIsSubmitting] = useState(false);
-
- // Load data
- useEffect(() => {
- const loadData = async () => {
- const id = params?.id as string;
- if (!id) {
- toast.error("ID tidak valid");
- router.push("/admin/desa/profile/profile-desa");
- return;
- }
-
- try {
- const data = await sejarahState.findUnique.load(id);
- if (data) {
- sejarahState.update.initialize(data);
- }
- } catch (error) {
- console.error("Error loading sejarah:", error);
- toast.error("Gagal memuat data sejarah desa");
- }
- };
-
- loadData();
-
- return () => {
- sejarahState.update.reset();
- sejarahState.findUnique.reset();
- };
- }, [params?.id, router]);
-
- const handleSubmit = async () => {
- if (isSubmitting || !sejarahState.update.form.judul.trim()) {
- toast.error("Judul wajib diisi");
- return;
- }
-
- setIsSubmitting(true);
- try {
- const success = await sejarahState.update.submit();
- if (success) {
- toast.success("Data berhasil disimpan");
- router.push("/admin/desa/profile/profile-desa");
- }
- } catch (error) {
- console.error("Error update sejarah desa:", error);
- toast.error("Terjadi kesalahan saat update sejarah desa");
- } finally {
- setIsSubmitting(false);
- }
- };
-
- const handleBack = () => router.back();
-
- // Loading state
- if (sejarahState.findUnique.loading || sejarahState.update.loading) {
- return (
-
-
- Memuat data...
-
-
- );
- }
-
- // Error state
- if (sejarahState.findUnique.error) {
- return (
-
-
-
-
-
- } color="red">
- Error
- {sejarahState.findUnique.error}
-
-
-
- );
- }
-
- return (
-
-
-
-
-
-
-
-
- Edit Sejarah Desa
-
-
-
-
- Edit Sejarah Desa
-
- {/* Judul */}
- Judul}
- placeholder="Judul sejarah"
- defaultValue={sejarahState.update.form.judul}
- onChange={(e) => sejarahState.update.form.judul = e.currentTarget.value}
- error={!sejarahState.update.form.judul && "Judul wajib diisi"}
- />
-
- {/* Deskripsi */}
-
- Deskripsi
- sejarahState.update.form.deskripsi = val}
- />
-
-
- {/* Buttons */}
-
-
- {isSubmitting ? 'Menyimpan...' : 'Simpan Perubahan'}
-
-
-
- Batal
-
-
-
-
-
-
- );
-}
-
-export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/visi_misi_desa/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/visi_misi_desa/page.tsx
deleted file mode 100644
index 0a06bc88..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-desa/[id]/visi_misi_desa/page.tsx
+++ /dev/null
@@ -1,155 +0,0 @@
-/* eslint-disable react-hooks/exhaustive-deps */
-'use client'
-import EditEditor from '@/app/admin/(dashboard)/_com/editEditor';
-import stateProfileDesa from '@/app/admin/(dashboard)/_state/desa/profile';
-import colors from '@/con/colors';
-import { Alert, Box, Button, Center, Group, Paper, Stack, Text, Title, Tooltip } from '@mantine/core';
-import { IconAlertCircle, IconArrowBack } from '@tabler/icons-react';
-import { useParams, useRouter } from 'next/navigation';
-import { useEffect, useState } from 'react';
-import { toast } from 'react-toastify';
-import { useProxy } from 'valtio/utils';
-
-function Page() {
- const visiMisiState = useProxy(stateProfileDesa.visiMisiDesa)
- const router = useRouter()
- const params = useParams()
- const [isSubmitting, setIsSubmitting] = useState(false);
-
- // Load data
- useEffect(() => {
- const loadData = async () => {
- const id = params?.id as string;
- if (!id) {
- toast.error("ID tidak valid");
- router.push("/admin/desa/profile/profile-desa");
- return;
- }
-
- try {
- const data = await visiMisiState.findUnique.load(id);
- visiMisiState.update.initialize(data);
- } catch (error) {
- console.error("Error loading visi misi:", error);
- toast.error("Gagal memuat data visi misi desa");
- }
- };
-
- loadData();
-
- return () => {
- visiMisiState.update.reset();
- visiMisiState.findUnique.reset();
- };
- }, [params?.id, router]);
-
- const handleSubmit = async () => {
- if (isSubmitting || !visiMisiState.update.form.visi.trim()) {
- toast.error("Visi wajib diisi");
- return;
- }
-
- setIsSubmitting(true);
-
- try {
- const success = await visiMisiState.update.submit();
-
- if (success) {
- toast.success("Data berhasil disimpan");
- router.push("/admin/desa/profile/profile-desa");
- }
- } catch (error) {
- console.error("Error update visi misi desa:", error);
- toast.error("Terjadi kesalahan saat update visi misi desa");
- } finally {
- setIsSubmitting(false);
- }
- };
-
- const handleBack = () => router.back();
-
- // Loading state
- if (visiMisiState.findUnique.loading || visiMisiState.update.loading) {
- return (
-
-
- Memuat data...
-
-
- );
- }
-
- // Error state
- if (visiMisiState.findUnique.error) {
- return (
-
-
-
-
-
- } color="red">
- Error
- {visiMisiState.findUnique.error}
-
-
-
- );
- }
-
- return (
-
-
-
-
-
-
-
-
- Edit Visi Misi Desa
-
-
-
-
- Edit Visi Misi Desa
-
- {/* Visi */}
-
- Visi
- visiMisiState.update.form.visi = val}
- />
-
-
- {/* Misi */}
-
- Misi
- visiMisiState.update.form.misi = val}
- />
-
-
- {/* Buttons */}
-
-
- {isSubmitting ? 'Menyimpan...' : 'Simpan Perubahan'}
-
-
-
- Batal
-
-
-
-
-
-
- );
-}
-
-export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-desa/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-desa/page.tsx
deleted file mode 100644
index 0dee1b80..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-desa/page.tsx
+++ /dev/null
@@ -1,248 +0,0 @@
-'use client'
-
-import colors from '@/con/colors';
-import { Box, Button, Card, Center, Divider, Grid, GridCol, Image, Paper, SimpleGrid, Stack, Text, Title, Tooltip } from '@mantine/core';
-import { useSnapshot } from 'valtio';
-import stateProfileDesa from '../../../_state/desa/profile';
-import { useEffect } from 'react';
-import { IconEdit } from '@tabler/icons-react';
-import { useRouter } from 'next/navigation';
-
-function Page() {
- const router = useRouter();
- const snap = useSnapshot(stateProfileDesa);
-
- useEffect(() => {
- stateProfileDesa.sejarahDesa.findUnique.load("edit");
- stateProfileDesa.visiMisiDesa.findUnique.load("edit");
- stateProfileDesa.lambangDesa.findUnique.load("edit");
- stateProfileDesa.maskotDesa.findUnique.load("edit");
- }, []);
-
- const sejarah = snap.sejarahDesa.findUnique.data;
- const visiMisi = snap.visiMisiDesa.findUnique.data;
- const lambang = snap.lambangDesa.findUnique.data;
- const maskot = snap.maskotDesa.findUnique.data;
-
- return (
-
-
- Preview Profile Desa
-
- {/* Sejarah Desa */}
- {sejarah && (
-
-
-
- Preview Sejarah Desa
-
-
-
- }
- radius="md"
- onClick={() => router.push(`/admin/desa/profile/profile-desa/${sejarah.id}/sejarah_desa`)}
- >
- Edit
-
-
-
-
-
-
-
-
-
-
-
-
-
- {sejarah.judul}
-
-
-
-
-
-
-
-
- )}
-
- {/* Visi Misi Desa */}
- {visiMisi && (
-
-
-
- Preview Visi Misi Desa
-
-
-
- }
- radius="md"
- onClick={() => router.push(`/admin/desa/profile/profile-desa/${visiMisi.id}/visi_misi_desa`)}
- >
- Edit
-
-
-
-
-
-
-
-
-
-
-
-
-
- Visi Misi Desa
-
-
-
-
- Visi Desa
-
- Misi Desa
-
-
-
-
- )}
-
- {/* Lambang Desa */}
- {lambang && (
-
-
-
- Preview Lambang Desa
-
-
-
- }
- radius="md"
- onClick={() => router.push(`/admin/desa/profile/profile-desa/${lambang.id}/lambang_desa`)}
- >
- Edit
-
-
-
-
-
-
-
-
-
-
-
-
-
- {lambang.judul}
-
-
-
-
-
-
-
-
- )}
-
- {/* Maskot Desa */}
- {maskot && (
-
-
-
- Preview Maskot Desa
-
-
-
- }
- radius="md"
- onClick={() => router.push(`/admin/desa/profile/profile-desa/${maskot.id}/maskot_desa`)}
- >
- Edit
-
-
-
-
-
-
-
-
-
-
-
-
-
- Maskot Desa
-
-
-
-
-
-
-
- {maskot.images.map((img, idx) => (
-
-
-
-
- {img.label}
-
- ))}
-
-
-
-
-
- )}
-
-
- );
-}
-
-export default Page;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/page.tsx
deleted file mode 100644
index 6957d8f5..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel-dari-masa-ke-masa/page.tsx
+++ /dev/null
@@ -1,135 +0,0 @@
-'use client'
-import colors from '@/con/colors';
-import { Box, Button, Center, Group, Pagination, Paper, Skeleton, Stack, Table, TableTbody, TableTd, TableTh, TableThead, TableTr, Text, Title, Tooltip } from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
-import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
-import { useRouter } from 'next/navigation';
-import { useState } from 'react';
-import { useProxy } from 'valtio/utils';
-import HeaderSearch from '../../../_com/header';
-import stateProfileDesa from '../../../_state/desa/profile';
-
-function PerbekelDariMasaKeMasa() {
- const [search, setSearch] = useState("");
- return (
-
- }
- value={search}
- onChange={(e) => setSearch(e.currentTarget.value)}
- />
-
-
- );
-}
-
-function ListPerbekelDariMasaKeMasa({ search }: { search: string }) {
- const state = useProxy(stateProfileDesa.mantanPerbekel)
- const router = useRouter();
- const { data, page, totalPages, loading, load } = state.findMany;
-
- useShallowEffect(() => {
- load(page, 10, search)
- }, [page, search]);
-
- const filteredData = data || [];
-
- if (loading || !data) {
- return (
-
-
-
- );
- }
-
- return (
-
-
-
- List Perbekel Dari Masa Ke Masa
-
- }
- color="blue"
- variant="light"
- onClick={() => router.push('/admin/desa/profile/profile-perbekel-dari-masa-ke-masa/create')}
- >
- Tambah Baru
-
-
-
-
-
-
-
-
- Nama Perbekel
- Periode
- Aksi
-
-
-
- {filteredData.length > 0 ? (
- filteredData.map((item) => (
-
-
-
- {item.nama}
-
-
-
-
- {item.periode}
-
-
-
-
- }
- onClick={() => router.push(`/admin/desa/profile/profile-perbekel-dari-masa-ke-masa/${item.id}`)}
- >
- Detail
-
-
-
-
- ))
- ) : (
-
-
-
- Tidak ada data perbekel yang cocok
-
-
-
- )}
-
-
-
-
-
-
- {
- load(newPage, 10);
- window.scrollTo({ top: 0, behavior: 'smooth' });
- }}
- total={totalPages}
- mt="md"
- mb="md"
- color="blue"
- radius="md"
- />
-
-
- );
-}
-
-export default PerbekelDariMasaKeMasa;
diff --git a/src/app/admin/(dashboard)/desa/profile/profile-perbekel/page.tsx b/src/app/admin/(dashboard)/desa/profile/profile-perbekel/page.tsx
deleted file mode 100644
index 73da958e..00000000
--- a/src/app/admin/(dashboard)/desa/profile/profile-perbekel/page.tsx
+++ /dev/null
@@ -1,117 +0,0 @@
-'use client'
-
-import colors from '@/con/colors';
-import { Box, Button, Center, Divider, Grid, GridCol, Image, Paper, Skeleton, Stack, Text, Title, Tooltip } from '@mantine/core';
-import { IconEdit } from '@tabler/icons-react';
-import { useRouter } from 'next/navigation';
-import { useEffect } from 'react';
-import { useSnapshot } from 'valtio';
-import stateProfileDesa from '../../../_state/desa/profile';
-
-function Page() {
- const router = useRouter();
- const snap = useSnapshot(stateProfileDesa);
-
- // Load data saat mount
- useEffect(() => {
- stateProfileDesa.profilPerbekel.findUnique.load("edit");
- }, []);
-
- const perbekel = snap.profilPerbekel.findUnique.data;
-
- if (!perbekel) {
- return (
-
-
-
- );
- }
-
- return (
-
-
- {/* Header + tombol edit */}
-
-
- Preview Profil PPID
-
-
-
- }
- radius="md"
- onClick={() => router.push(`/admin/desa/profile/profile-perbekel/${perbekel.id}`)}
- >
- Edit
-
-
-
-
-
- {/* Card Profil */}
-
-
-
-
-
-
-
-
-
-
- Profil Pimpinan Badan Publik Desa Darmasaba
-
-
-
-
-
-
-
-
- { e.currentTarget.src = "/perbekel.png"; }}
- loading='lazy'
- />
-
-
-
- I.B. Surya Prabhawa Manuaba, S.H., M.H.
-
-
-
-
- {/* Biodata & Info */}
-
- Biodata
-
-
- Pengalaman
-
-
- Pengalaman Organisasi
-
-
- Program Kerja Unggulan
-
-
-
-
-
- );
-}
-
-export default Page;
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/_lib/layoutTabs.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/_lib/layoutTabs.tsx
index dca2dfe2..e252c31d 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/_lib/layoutTabs.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/_lib/layoutTabs.tsx
@@ -2,23 +2,23 @@
'use client'
import colors from '@/con/colors';
import {
+ Box,
+ ScrollArea,
Stack,
Tabs,
TabsList,
TabsPanel,
TabsTab,
- Title,
- Tooltip,
- ScrollArea,
+ Title
} from '@mantine/core';
-import { usePathname, useRouter } from 'next/navigation';
-import React, { useEffect, useState } from 'react';
import {
- IconFileAnalytics,
IconCoins,
+ IconFileAnalytics,
IconShoppingCart,
IconWallet,
} from '@tabler/icons-react';
+import { usePathname, useRouter } from 'next/navigation';
+import React, { useEffect, useState } from 'react';
function LayoutTabs({ children }: { children: React.ReactNode }) {
const router = useRouter();
@@ -29,29 +29,25 @@ function LayoutTabs({ children }: { children: React.ReactNode }) {
label: "APB Desa",
value: "apbdesa",
href: "/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa",
- icon: ,
- tooltip: "Lihat ringkasan Anggaran Pendapatan dan Belanja Desa",
+ icon:
},
{
label: "Pendapatan",
value: "pendapatan",
href: "/admin/ekonomi/PADesa-pendapatan-asli-desa/pendapatan",
icon: ,
- tooltip: "Kelola data pendapatan desa",
},
{
label: "Belanja",
value: "belanja",
href: "/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja",
icon: ,
- tooltip: "Atur data belanja desa",
},
{
label: "Pembiayaan",
value: "pembiayaan",
href: "/admin/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan",
icon: ,
- tooltip: "Kelola data pembiayaan desa",
},
];
@@ -90,43 +86,76 @@ function LayoutTabs({ children }: { children: React.ReactNode }) {
keepMounted={false}
>
{/* ✅ Scroll horizontal wrapper */}
-
-
- {tabs.map((tab, i) => (
-
+
+
+
+ {tabs.map((tab, i) => (
{tab.label}
-
- ))}
-
-
+ ))}
+
+
+
+
+
+
+
+
+ {tabs.map((tab, i) => (
+
+ {tab.label}
+
+ ))}
+
+
+
{tabs.map((tab, i) => (
{
+ if (!Array.isArray(arr)) return [];
+ return arr
+ .filter(item => item != null && item !== '')
+ .map(item => String(item))
+ .filter(item => item.trim() !== '');
+};
+
+const createEmptyForm = () => ({
+ tahun: '',
+ pendapatanIds: [] as string[],
+ belanjaIds: [] as string[],
+ pembiayaanIds: [] as string[],
+});
+
+// ==================== COMPONENT ====================
function EditAPBDesa() {
const apbState = useProxy(PendapatanAsliDesa.ApbDesa);
const router = useRouter();
const params = useParams();
- const [formData, setFormData] = useState({
- tahun: '',
- pendapatanIds: [] as string[],
- belanjaIds: [] as string[],
- pembiayaanIds: [] as string[],
- });
+ const [formData, setFormData] = useState(createEmptyForm());
+ const [originalData, setOriginalData] = useState(createEmptyForm());
+ const [isSubmitting, setIsSubmitting] = useState(false);
+ const [isLoading, setIsLoading] = useState(true);
- // Load APB desa by id → hanya update formData, bukan global state
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.tahun?.trim() !== '' &&
+ Number(formData.tahun) > 0 &&
+ formData.pendapatanIds.length > 0 &&
+ formData.belanjaIds.length > 0 &&
+ formData.pembiayaanIds.length > 0
+ );
+ };
+
+ // ==================== LOAD DATA ====================
useEffect(() => {
const loadAPBdesa = async () => {
const id = params?.id as string;
- if (!id) return;
+ if (!id) {
+ toast.error('ID tidak valid');
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa');
+ return;
+ }
try {
+ setIsLoading(true);
const data = await apbState.update.load(id);
- if (data) {
- setFormData({
- tahun: String(data.tahun || ''),
- pendapatanIds: data.pendapatan?.map((p: any) => p.id) || [],
- belanjaIds: data.belanja?.map((b: any) => b.id) || [],
- pembiayaanIds: data.pembiayaan?.map((p: any) => p.id) || [],
- });
+
+ if (!data) {
+ toast.error('Data APB Desa tidak ditemukan');
+ return;
}
- } catch (error) {
- console.error("Error loading APBdesa:", error);
- toast.error("Gagal memuat data APBdesa");
+
+ const normalized = {
+ tahun: String(data.tahun || ''),
+ pendapatanIds: safeStringArray(data.pendapatan?.map((p: any) => p.id) || []),
+ belanjaIds: safeStringArray(data.belanja?.map((b: any) => b.id) || []),
+ pembiayaanIds: safeStringArray(data.pembiayaan?.map((p: any) => p.id) || []),
+ };
+
+ setFormData(normalized);
+ setOriginalData(normalized);
+ } catch (err) {
+ console.error('Error loading APBdesa:', err);
+ toast.error('Gagal memuat data APB Desa');
+ } finally {
+ setIsLoading(false);
}
};
loadAPBdesa();
}, [params?.id]);
+ // ==================== HANDLERS ====================
const handleChange = (field: keyof typeof formData, value: any) => {
setFormData(prev => ({ ...prev, [field]: value }));
};
+ const handleResetForm = () => {
+ setFormData(originalData);
+ toast.info('Form dikembalikan ke data awal');
+ };
+
const handleSubmit = async () => {
try {
- // update global state cuma pas submit
+ setIsSubmitting(true);
+
+ if (!formData.tahun.trim()) {
+ toast.warning('Tahun harus diisi');
+ return;
+ }
+
apbState.update.form = {
...apbState.update.form,
tahun: Number(formData.tahun),
- pendapatanIds: formData.pendapatanIds,
- belanjaIds: formData.belanjaIds,
- pembiayaanIds: formData.pembiayaanIds,
+ pendapatanIds: safeStringArray(formData.pendapatanIds),
+ belanjaIds: safeStringArray(formData.belanjaIds),
+ pembiayaanIds: safeStringArray(formData.pembiayaanIds),
};
await apbState.update.update();
- toast.success("APB Desa berhasil diperbarui!");
- router.push("/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa");
+ toast.success('APB Desa berhasil diperbarui!');
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa');
} catch (error) {
- console.error("Error updating APBdesa:", error);
- toast.error("Terjadi kesalahan saat memperbarui APBdesa");
+ console.error('Error updating APBdesa:', error);
+ toast.error('Terjadi kesalahan saat memperbarui APB Desa');
+ } finally {
+ setIsSubmitting(false);
}
};
+ // ==================== UI ====================
+ if (isLoading) {
+ return (
+
+
+ Memuat data APB Desa...
+
+ );
+ }
+
return (
-
+
{/* Header */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit APB Desa
@@ -117,151 +180,128 @@ function EditAPBDesa() {
handleChange("tahun", e.target.value)}
+ onChange={(e) => handleChange('tahun', e.target.value)}
label={Tahun }
placeholder="Masukkan tahun anggaran"
required
/>
{/* Selects */}
- handleChange("pendapatanIds", ids)}
+ onSelectionChange={(ids) => handleChange('pendapatanIds', ids)}
/>
- handleChange("belanjaIds", ids)}
+ onSelectionChange={(ids) => handleChange('belanjaIds', ids)}
/>
- handleChange("pembiayaanIds", ids)}
+ onSelectionChange={(ids) => handleChange('pembiayaanIds', ids)}
/>
{/* Save Button */}
-
+
+
+ Batal
+
+
- Simpan
+ {isSubmitting ? : 'Simpan'}
);
+}
- /* --- Sub Components --- */
+// ==================== SUB COMPONENT ====================
+function SelectAPBItem({
+ label,
+ state,
+ selectedIds,
+ onSelectionChange,
+}: {
+ label: string;
+ state: any;
+ selectedIds: string[];
+ onSelectionChange: (ids: string[]) => void;
+}) {
+ const proxyState = useProxy(state);
- function SelectPendapatan({
- selectedIds,
- onSelectionChange,
- }: {
- selectedIds: string[];
- onSelectionChange: (ids: string[]) => void;
- }) {
- const pendapatanState = useProxy(PendapatanAsliDesa.pendapatan);
+ useShallowEffect(() => {
+ proxyState.findMany.load();
+ }, []);
- useShallowEffect(() => {
- pendapatanState.findMany.load();
- }, []);
+ const data = proxyState.findMany.data;
+ const isLoading = !data;
- if (!pendapatanState.findMany.data) {
- return ;
- }
+ const options =
+ data
+ ?.filter((item: any) => item?.id)
+ .map((item: any) => ({
+ value: String(item.id),
+ label: String(item?.name || '(Tanpa Nama)'),
+ })) || [];
+ if (isLoading) {
return (
- Pendapatan}
- data={pendapatanState.findMany.data.map((p: any) => ({
- value: p.id,
- label: p.name,
- }))}
- value={selectedIds}
- onChange={onSelectionChange}
- searchable
- clearable
- placeholder="Pilih pendapatan..."
- nothingFoundMessage="Tidak ditemukan"
- />
+
+ {label}
+
+
);
}
- function SelectBelanja({
- selectedIds,
- onSelectionChange,
- }: {
- selectedIds: string[];
- onSelectionChange: (ids: string[]) => void;
- }) {
- const belanjaState = useProxy(PendapatanAsliDesa.belanja);
-
- useShallowEffect(() => {
- belanjaState.findMany.load();
- }, []);
-
- if (!belanjaState.findMany.data) {
- return ;
- }
-
+ if (options.length === 0) {
return (
- Belanja}
- data={belanjaState.findMany.data.map((b: any) => ({
- value: b.id,
- label: b.name,
- }))}
- value={selectedIds}
- onChange={onSelectionChange}
- searchable
- clearable
- placeholder="Pilih belanja..."
- nothingFoundMessage="Tidak ditemukan"
- />
+
+
+ Tidak ada data {label.toLowerCase()} tersedia.
+
+
);
}
- function SelectPembiayaan({
- selectedIds,
- onSelectionChange,
- }: {
- selectedIds: string[];
- onSelectionChange: (ids: string[]) => void;
- }) {
- const pembiayaanState = useProxy(PendapatanAsliDesa.pembiayaan);
-
- useShallowEffect(() => {
- pembiayaanState.findMany.load();
- }, []);
-
- if (!pembiayaanState.findMany.data) {
- return ;
- }
-
- return (
- Pembiayaan}
- data={pembiayaanState.findMany.data.map((p: any) => ({
- value: p.id,
- label: p.name,
- }))}
- value={selectedIds}
- onChange={onSelectionChange}
- searchable
- clearable
- placeholder="Pilih pembiayaan..."
- nothingFoundMessage="Tidak ditemukan"
- />
- );
- }
+ return (
+ {label}}
+ data={options}
+ value={selectedIds}
+ onChange={(ids) => onSelectionChange(safeStringArray(ids))}
+ searchable
+ clearable
+ placeholder={`Pilih ${label.toLowerCase()}...`}
+ nothingFoundMessage="Tidak ditemukan"
+ />
+ );
}
export default EditAPBDesa;
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/[id]/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/[id]/page.tsx
index ee57dc50..3075f2e1 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/[id]/page.tsx
@@ -9,8 +9,7 @@ import {
Paper,
Skeleton,
Stack,
- Text,
- Tooltip
+ Text
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack, IconEdit, IconTrash } from '@tabler/icons-react';
@@ -58,7 +57,7 @@ function DetailAPBDesa() {
const data = apbState.findUnique.data;
return (
-
+
router.back()}
@@ -81,7 +80,7 @@ function DetailAPBDesa() {
Detail APB Desa
-
+
@@ -159,36 +158,32 @@ function DetailAPBDesa() {
-
- {
- setSelectedId(data.id);
- setModalHapus(true);
- }}
- variant="light"
- radius="md"
- size="md"
- >
-
-
-
+ {
+ setSelectedId(data.id);
+ setModalHapus(true);
+ }}
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
-
-
- router.push(
- `/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/${data.id}/edit`
- )
- }
- variant="light"
- radius="md"
- size="md"
- >
-
-
-
+
+ router.push(
+ `/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/${data.id}/edit`
+ )
+ }
+ variant="light"
+ radius="md"
+ size="md"
+ >
+
+
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create/page.tsx
index 797cc92f..cc237b70 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create/page.tsx
@@ -6,23 +6,37 @@ import {
Box,
Button,
Group,
+ Loader,
MultiSelect,
Paper,
Skeleton,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { useShallowEffect } from '@mantine/hooks';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
+import { useState } from 'react';
+import { toast } from 'react-toastify';
import { useProxy } from 'valtio/utils';
function CreateAPBDesa() {
const apbDesaState = useProxy(PendapatanAsliDesa.ApbDesa);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ apbDesaState.create.form.tahun !== null &&
+ apbDesaState.create.form.tahun > 0 &&
+ apbDesaState.create.form.pendapatanIds.length > 0 &&
+ apbDesaState.create.form.belanjaIds.length > 0 &&
+ apbDesaState.create.form.pembiayaanIds.length > 0
+ );
+ };
const resetForm = () => {
apbDesaState.create.form = {
@@ -34,20 +48,26 @@ function CreateAPBDesa() {
};
const handleSubmit = async () => {
- await apbDesaState.create.submit();
- resetForm();
- router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa');
+ try {
+ setIsSubmitting(true);
+ await apbDesaState.create.submit();
+ resetForm();
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa');
+ } catch (error) {
+ console.error('Error creating APB Desa:', error);
+ toast.error('Gagal membuat APB Desa');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header */}
-
- router.back()} p="xs" radius="md">
-
-
-
+ router.back()} p="xs" radius="md">
+
+
Tambah APB Desa
@@ -65,7 +85,7 @@ function CreateAPBDesa() {
{
apbDesaState.create.form.tahun = Number(val.target.value);
}}
@@ -97,17 +117,31 @@ function CreateAPBDesa() {
{/* Action */}
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/page.tsx
index ff0c6464..7c4792d4 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/page.tsx
@@ -5,8 +5,8 @@ import {
Button,
Center,
Group,
- Paper,
Pagination,
+ Paper,
Skeleton,
Stack,
Table,
@@ -16,13 +16,13 @@ import {
TableThead,
TableTr,
Text,
- Tooltip,
+ Title
} from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
import { IconDeviceImacCog, IconPlus, IconSearch } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
-import { useProxy } from 'valtio/utils';
+ import { useProxy } from 'valtio/utils';
import HeaderSearch from '../../../_com/header';
import PendapatanAsliDesa from '../../../_state/ekonomi/PADesa';
@@ -45,6 +45,7 @@ function APBDesa() {
function ListAPBDesa({ search }: { search: string }) {
const apbDesaState = useProxy(PendapatanAsliDesa.ApbDesa);
const router = useRouter();
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const {
data,
@@ -62,96 +63,122 @@ function ListAPBDesa({ search }: { search: string }) {
}).format(value);
useShallowEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const filteredData = data || [];
if (loading || !data) {
return (
-
+
);
}
return (
-
-
-
-
+
+
+
+
List APB Desa
-
-
- }
- color="blue"
- variant="light"
- onClick={() =>
- router.push(
- "/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create"
- )
- }
- >
- Tambah Baru
-
-
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push(
+ "/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/create"
+ )
+ }
+ >
+ Tambah Baru
+
-
-
+
+ {/* Desktop Table */}
+
+
- Tahun
- Pembiayaan
- Belanja
- Pendapatan
- Aksi
+
+ Tahun
+
+
+ Pembiayaan
+
+
+ Belanja
+
+
+ Pendapatan
+
+
+ Aksi
+
{filteredData.length > 0 ? (
filteredData.map((item) => (
- {item.tahun}
- {formatRupiah(
- item.pembiayaan.reduce(
- (sum, val) => sum + Number(val.value),
- 0
- )
- )}
+ {item.tahun}
- {formatRupiah(
- item.belanja.reduce(
- (sum, val) => sum + Number(val.value),
- 0
- )
- )}
+
+ {formatRupiah(
+ item.pembiayaan.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
- {formatRupiah(
- item.pendapatan.reduce(
- (sum, val) => sum + Number(val.value),
- 0
- )
- )}
+
+ {formatRupiah(
+ item.belanja.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
-
-
- router.push(
- `/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/${item.id}`
- )
- }
- >
-
- Detail
-
-
+
+ {formatRupiah(
+ item.pendapatan.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
+
+
+
+ router.push(
+ `/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/${item.id}`
+ )
+ }
+ size="compact-sm"
+ >
+
+
+ Detail
+
+
))
@@ -159,7 +186,7 @@ function ListAPBDesa({ search }: { search: string }) {
-
+
Tidak ada data APB Desa yang cocok
@@ -169,7 +196,81 @@ function ListAPBDesa({ search }: { search: string }) {
+
+ {/* Mobile Cards */}
+
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+ Tahun
+ {item.tahun}
+
+
+ Pembiayaan
+
+ {formatRupiah(
+ item.pembiayaan.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
+
+
+ Belanja
+
+ {formatRupiah(
+ item.belanja.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
+
+
+ Pendapatan
+
+ {formatRupiah(
+ item.pendapatan.reduce(
+ (sum, val) => sum + Number(val.value),
+ 0
+ )
+ )}
+
+
+
+ router.push(
+ `/admin/ekonomi/PADesa-pendapatan-asli-desa/apbdesa/${item.id}`
+ )
+ }
+ size="sm"
+ >
+
+
+ Detail
+
+
+
+
+ ))
+ ) : (
+
+
+ Tidak ada data APB Desa yang cocok
+
+
+ )}
+
+
+
{
+ return (
+ formData.name?.trim() !== '' &&
+ formData.value !== '' &&
+ Number(formData.value) > 0
+ );
+ };
+
// format angka ke rupiah
const formatRupiah = (value: number | string) => {
const number =
@@ -59,6 +74,10 @@ function EditBelanja() {
name: data.name || '',
value: String(data.value || ''),
});
+ setOriginalData({
+ name: data.name || '',
+ value: String(data.value || ''),
+ });
}
} catch (error) {
console.error("Error loading belanja:", error);
@@ -71,6 +90,7 @@ function EditBelanja() {
const handleSubmit = async () => {
try {
+ setIsSubmitting(true);
belanjaState.update.form = {
...belanjaState.update.form,
name: formData.name,
@@ -83,23 +103,31 @@ function EditBelanja() {
} catch (error) {
console.error("Error updating jenis belanja:", error);
toast.error("Terjadi kesalahan saat memperbarui jenis belanja");
+ } finally {
+ setIsSubmitting(false);
}
};
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ value: originalData.value,
+ });
+ toast.info("Form dikembalikan ke data awal");
+ };
+
return (
-
+
{/* Header */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Jenis Belanja
@@ -138,17 +166,31 @@ function EditBelanja() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/create/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/create/page.tsx
index 7cc80cf9..d5fede9b 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/create/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/create/page.tsx
@@ -6,21 +6,32 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Stack,
Text,
TextInput,
- Title,
- Tooltip,
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
-import { useProxy } from 'valtio/utils';
+import { useState } from 'react';
import { toast } from 'react-toastify';
+import { useProxy } from 'valtio/utils';
function CreateBelanja() {
const belanjaState = useProxy(PendapatanAsliDesa.belanja);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ belanjaState.create.form.name?.trim() !== '' &&
+ belanjaState.create.form.value !== null &&
+ belanjaState.create.form.value > 0
+ );
+ };
const formatRupiah = (value: number | string) => {
const number =
@@ -48,25 +59,31 @@ function CreateBelanja() {
return toast.warn('Lengkapi semua field terlebih dahulu');
}
- await belanjaState.create.submit();
- resetForm();
- router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja');
+ try {
+ setIsSubmitting(true);
+ await belanjaState.create.submit();
+ resetForm();
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja');
+ } catch (error) {
+ console.error('Error creating belanja:', error);
+ toast.error('Gagal menambahkan jenis belanja');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header dengan back button */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Tambah Jenis Belanja
@@ -85,7 +102,7 @@ function CreateBelanja() {
Nama Jenis Belanja}
placeholder="Masukkan nama jenis belanja"
- defaultValue={belanjaState.create.form.name}
+ value={belanjaState.create.form.name}
onChange={(e) => (belanjaState.create.form.name = e.target.value)}
required
/>
@@ -94,7 +111,7 @@ function CreateBelanja() {
type="text"
label={Nilai }
placeholder="Masukkan nilai belanja"
- defaultValue={formatRupiah(belanjaState.create.form.value)}
+ value={formatRupiah(belanjaState.create.form.value)}
onChange={(e) => {
const raw = e.currentTarget.value;
belanjaState.create.form.value = unformatRupiah(raw);
@@ -103,17 +120,31 @@ function CreateBelanja() {
/>
+
+ Reset
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/page.tsx
index 467c8142..dc185abc 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/belanja/page.tsx
@@ -17,10 +17,9 @@ import {
TableThead,
TableTr,
Text,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
-import { useShallowEffect } from '@mantine/hooks';
+import { useDebouncedValue, useShallowEffect } from '@mantine/hooks';
import { IconEdit, IconPlus, IconSearch, IconTrash } from '@tabler/icons-react';
import { useRouter } from 'next/navigation';
import { useState } from 'react';
@@ -32,7 +31,7 @@ import PendapatanAsliDesa from '../../../_state/ekonomi/PADesa';
function Belanja() {
const [search, setSearch] = useState("");
return (
-
+
setSearch(e.currentTarget.value)}
/>
-
+
);
}
@@ -50,6 +49,7 @@ function ListBelanja({ search }: { search: string }) {
const router = useRouter();
const [modalHapus, setModalHapus] = useState(false);
const [selectedId, setSelectedId] = useState(null);
+ const [debouncedSearch] = useDebouncedValue(search, 1000);
const {
data,
@@ -73,51 +73,73 @@ function ListBelanja({ search }: { search: string }) {
belanjaState.delete.byId(selectedId);
setModalHapus(false);
setSelectedId(null);
- load(page, 10, search);
+ load(page, 10, debouncedSearch);
}
};
useShallowEffect(() => {
- load(page, 10, search);
- }, [page, search]);
+ load(page, 10, debouncedSearch);
+ }, [page, debouncedSearch]);
const filteredData = data || [];
if (loading || !data) {
return (
-
+
);
}
return (
-
+
+ {/* Desktop Table */}
- Daftar Belanja
-
- }
- color="blue"
- variant="light"
- onClick={() =>
- router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja/create')
- }
- >
- Tambah Baru
-
-
+
+ Daftar Belanja
+
+ }
+ color="blue"
+ variant="light"
+ onClick={() =>
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja/create')
+ }
+ >
+ Tambah Baru
+
-
-
+
+
- Nama
- Nilai
- Persentase
- Aksi
+
+ Nama
+
+
+ Nilai
+
+
+ Edit
+
+
+ Delete
+
@@ -126,64 +148,66 @@ function ListBelanja({ search }: { search: string }) {
{filteredData.map((item) => (
-
+
{item.name}
- {formatRupiah(item.value)}
- {totalBelanja > 0
- ? ((item.value / totalBelanja) * 100).toFixed(0) + '%'
- : '0%'}
+ {formatRupiah(item.value)}
+
+
+
+ {totalBelanja > 0
+ ? ((item.value / totalBelanja) * 100).toFixed(0) + '%'
+ : '0%'}
+
-
-
- router.push(
- `/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja/${item.id}`
- )
- }
- >
-
-
-
-
- {
- setSelectedId(item.id);
- setModalHapus(true);
- }}
- >
-
-
-
+
+ router.push(
+ `/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja/${item.id}`
+ )
+ }
+ >
+
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ >
+
+
))}
- Total
+ Total
- {formatRupiah(totalBelanja)}
+ {formatRupiah(totalBelanja)}
>
) : (
-
- Tidak ada data belanja yang cocok
+
+
+ Tidak ada data belanja yang cocok
+
@@ -193,21 +217,107 @@ function ListBelanja({ search }: { search: string }) {
+ {/* Mobile Cards */}
+
+ {filteredData.length > 0 ? (
+ filteredData.map((item) => (
+
+
+
+
+ Nama
+
+
+ {item.name}
+
+
+
+
+ Nilai
+
+
+ {formatRupiah(item.value)}
+
+
+
+
+ Persentase
+
+
+ {totalBelanja > 0
+ ? ((item.value / totalBelanja) * 100).toFixed(0) + '%'
+ : '0%'}
+
+
+
+
+ router.push(
+ `/admin/ekonomi/PADesa-pendapatan-asli-desa/belanja/${item.id}`
+ )
+ }
+ >
+
+
+ {
+ setSelectedId(item.id);
+ setModalHapus(true);
+ }}
+ >
+
+
+
+
+
+ ))
+ ) : (
+
+
+
+ Tidak ada data belanja yang cocok
+
+
+
+ )}
+
+ {filteredData.length > 0 && (
+
+
+
+ Total
+
+
+ {formatRupiah(totalBelanja)}
+
+
+
+ )}
+
+
{/* Pagination */}
-
- {
- load(newPage, 10, search);
- window.scrollTo({ top: 0, behavior: 'smooth' });
- }}
- total={totalPages}
- mt="md"
- mb="md"
- color="blue"
- radius="md"
- />
-
+ {(totalPages > 1 || page > 1) && (
+
+ {
+ load(newPage, 10, search);
+ window.scrollTo({ top: 0, behavior: 'smooth' });
+ }}
+ total={totalPages}
+ mt="md"
+ color="blue"
+ radius="md"
+ />
+
+ )}
{/* Modal Konfirmasi Hapus */}
-
+
);
}
-export default Belanja;
+export default Belanja;
\ No newline at end of file
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/layout.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/layout.tsx
index 3a414c77..34fe9efa 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/layout.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/layout.tsx
@@ -1,7 +1,30 @@
+'use client'
import React from 'react';
import LayoutTabs from './_lib/layoutTabs';
+import { usePathname } from 'next/navigation';
+import { Box } from '@mantine/core';
function Layout({ children }: { children: React.ReactNode }) {
+const pathname = usePathname();
+
+ // Contoh path:
+ // - /darmasaba/desa/berita/semua → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan → panjang 5 → list
+ // - /darmasaba/desa/berita/Pemerintahan/123 → panjang 6 → detail
+
+ const segments = pathname.split('/').filter(Boolean);
+ const isDetailPage = segments.length >= 5;
+
+ if (isDetailPage) {
+ // Tampilkan tanpa tab menu
+ return (
+
+ {children}
+
+ );
+ }
+
+
return (
{children}
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/[id]/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/[id]/page.tsx
index 25df9c5b..378a34fd 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/[id]/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/[id]/page.tsx
@@ -6,11 +6,11 @@ import {
Box,
Button,
Group,
+ Loader,
Paper,
Stack,
TextInput,
- Title,
- Tooltip
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
import { useParams, useRouter } from 'next/navigation';
@@ -22,12 +22,27 @@ function EditPembiayaan() {
const pembiayaanState = useProxy(PendapatanAsliDesa.pembiayaan);
const router = useRouter();
const params = useParams();
+ const [isSubmitting, setIsSubmitting] = useState(false);
const [formData, setFormData] = useState({
name: '',
value: '',
});
+ const [originalData, setOriginalData] = useState({
+ name: '',
+ value: '',
+ });
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ formData.name?.trim() !== '' &&
+ formData.value !== '' &&
+ Number(formData.value) > 0
+ );
+ };
+
const formatRupiah = (value: number | string) => {
const number =
typeof value === 'number'
@@ -56,6 +71,10 @@ function EditPembiayaan() {
name: data.name || '',
value: String(data.value || ''),
});
+ setOriginalData({
+ name: data.name || '',
+ value: String(data.value || ''),
+ });
}
} catch (error) {
console.error('Error loading pembiayaan:', error);
@@ -66,8 +85,17 @@ function EditPembiayaan() {
loadPembiayaan();
}, [params?.id]);
+ const handleResetForm = () => {
+ setFormData({
+ name: originalData.name,
+ value: originalData.value,
+ });
+ toast.info('Form dikembalikan ke data awal');
+ };
+
const handleSubmit = async () => {
try {
+ setIsSubmitting(true);
pembiayaanState.update.form = {
...pembiayaanState.update.form,
name: formData.name,
@@ -80,23 +108,23 @@ function EditPembiayaan() {
} catch (error) {
console.error('Error updating jenis pembiayaan:', error);
toast.error('Terjadi kesalahan saat memperbarui jenis pembiayaan');
+ } finally {
+ setIsSubmitting(false);
}
};
return (
-
+
{/* Header */}
-
- router.back()}
- p="xs"
- radius="md"
- >
-
-
-
+ router.back()}
+ p="xs"
+ radius="md"
+ >
+
+
Edit Jenis Pembiayaan
@@ -135,17 +163,31 @@ function EditPembiayaan() {
/>
+
+ Batal
+
+
+ {/* Tombol Simpan */}
- Simpan
+ {isSubmitting ? : 'Simpan'}
diff --git a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/create/page.tsx b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/create/page.tsx
index a432a431..63897a2f 100644
--- a/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/create/page.tsx
+++ b/src/app/admin/(dashboard)/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan/create/page.tsx
@@ -1,26 +1,36 @@
'use client';
-import React from 'react';
-import { useProxy } from 'valtio/utils';
import PendapatanAsliDesa from '@/app/admin/(dashboard)/_state/ekonomi/PADesa';
-import { useRouter } from 'next/navigation';
import colors from '@/con/colors';
import {
Box,
Button,
Group,
+ Loader,
Paper,
Stack,
- Title,
- TextInput,
Text,
- Tooltip,
+ TextInput,
+ Title
} from '@mantine/core';
import { IconArrowBack } from '@tabler/icons-react';
+import { useRouter } from 'next/navigation';
+import { useState } from 'react';
import { toast } from 'react-toastify';
+import { useProxy } from 'valtio/utils';
function CreatePembiayaan() {
const pembiayaanState = useProxy(PendapatanAsliDesa.pembiayaan);
const router = useRouter();
+ const [isSubmitting, setIsSubmitting] = useState(false);
+
+ // Check if form is valid
+ const isFormValid = () => {
+ return (
+ pembiayaanState.create.form.name?.trim() !== '' &&
+ pembiayaanState.create.form.value !== null &&
+ pembiayaanState.create.form.value > 0
+ );
+ };
const formatRupiah = (value: number | string) => {
const number =
@@ -44,29 +54,35 @@ function CreatePembiayaan() {
};
const handleSubmit = async () => {
- if (!pembiayaanState.create.form.name || !pembiayaanState.create.form.value) {
- return toast.warn('Nama dan nilai wajib diisi');
- }
+ try {
+ setIsSubmitting(true);
+ if (!pembiayaanState.create.form.name || !pembiayaanState.create.form.value) {
+ return toast.warn('Nama dan nilai wajib diisi');
+ }
- await pembiayaanState.create.submit();
- resetForm();
- router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan');
+ await pembiayaanState.create.submit();
+ resetForm();
+ router.push('/admin/ekonomi/PADesa-pendapatan-asli-desa/pembiayaan');
+ } catch (error) {
+ console.error('Error creating pembiayaan:', error);
+ toast.error('Gagal menambahkan jenis pembiayaan');
+ } finally {
+ setIsSubmitting(false);
+ }
};
return (
-
+
{/* Header */}
-
-