Compare commits

..

19 Commits

Author SHA1 Message Date
a3940321a7 fix(api): move swagger to /api group to prevent double prefixing 2026-04-01 15:29:26 +08:00
3cd6fcbd81 fix(api): clean up redundant /api prefixes and fix swagger documentation 2026-04-01 15:24:12 +08:00
7d9b7b0c60 feat(apbdes): finalize modernization and update config 2026-04-01 15:15:01 +08:00
0806eb2308 feat(apbdes): modernize ui, charts and refactor (Phase 1, 2, 4) 2026-04-01 15:09:40 +08:00
51ce823b45 Fix: Use window.location.origin for API base URL in browser
treaty from @elysiajs/eden doesn't support relative URLs like '/'
This caused 'ERR_NAME_NOT_RESOLVED' when trying to access 'https://api/fileStorage/create'

Solution:
- Client-side: Use window.location.origin (e.g., https://desa-darmasaba-stg.wibudev.com)
- Server-side dev: Use localhost:3000
- Server-side prod: Use NEXT_PUBLIC_BASE_URL env var

This ensures the API calls use the correct domain in all environments.

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
2026-03-12 14:14:55 +08:00
f6f0e10935 Fix Url API Route 2026-03-12 12:11:10 +08:00
2108f403aa Update .env.example to use relative URL '/' as default for NEXT_PUBLIC_BASE_URL
This ensures the API uses the same protocol and domain as the frontend,
preventing mixed content blocking in staging/production environments.

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
2026-03-12 12:06:16 +08:00
c6c3eebadf Fix: CORS and API base URL for music create in staging
- Update CORS config to allow all origins (wildcard first) for better staging support
- Change API fetch base URL from absolute to relative (/) to prevent mixed content blocking
- Add detailed logging in music create page for better debugging
- Update .env.example with better NEXT_PUBLIC_BASE_URL documentation
- Add MUSIK_CREATE_ANALYSIS.md with comprehensive error analysis

Fixes ERR_BLOCKED_BY_CLIENT error when creating music in staging environment

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
2026-03-12 11:27:51 +08:00
bipproduction
0dabc204bc Revert standalone output, keep serverExternalPackages fix
- Remove output: standalone to keep migration/seed workflow
- Restore original Dockerfile structure with full node_modules copy
- Keep serverExternalPackages for @elysiajs/static and elysia to fix
  the Html prerender error caused by dynamic import in @elysiajs/static
- Keep NODE_ENV=production before build step

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 11:53:18 +08:00
bipproduction
e8f8b51686 Fix build: externalize elysia packages, use standalone output, fix NODE_ENV
- Add serverExternalPackages for @elysiajs/static and elysia to prevent
  webpack from bundling dynamic imports that cause Html prerender error
- Use output: standalone for proper Docker deployment
- Comment out NODE_ENV=development in .env.example to avoid conflict
  with next build which requires NODE_ENV=production
- Set NODE_ENV=production before build step in Dockerfile
- Update runtime stage to use standalone output structure

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 11:48:05 +08:00
bipproduction
a4db3a149d Fix build error: move ViewTransitions inside body to fix 404 prerendering
ViewTransitions was wrapping the html element, which violates Next.js App
Router requirement that html and body be returned directly from root layout.
This caused prerendering of /404 to fail with Html import error.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 11:29:29 +08:00
bipproduction
fece983ac5 Fix Dockerfile: remove non-existent gen:api script and fix build output paths
- Remove bun run gen:api which does not exist in package.json
- Change dist to .next for correct Next.js build output
- Replace non-existent generated/ with public/ for static assets

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
2026-03-10 11:12:33 +08:00
8b7eef5fee New bunlock.b 2026-03-10 11:04:44 +08:00
8b22d01e0d FIx Docker File 2026-03-10 10:56:15 +08:00
dc13e37a02 Add .env.example and fix .gitignore to allow it
Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
2026-03-10 10:50:52 +08:00
2d2cbef29b Tambah File Docker 2026-03-10 10:42:51 +08:00
8c8a96b830 First Stg 2026-03-10 10:07:14 +08:00
dc3eccacbf First Stg 2026-03-10 10:03:33 +08:00
ffe94992e5 StaggingWeb 2026-03-09 16:44:42 +08:00
81 changed files with 2720 additions and 8316 deletions

19
.env Normal file
View File

@@ -0,0 +1,19 @@
DATABASE_URL="postgresql://bip:Production_123@localhost:5433/desa-darmasaba-v0.0.1?schema=public"
# Seafile
SEAFILE_TOKEN=20a19f4a04032215d50ce53292e6abdd38b9f806
SEAFILE_REPO_ID=f0e9ee4a-fd13-49a2-81c0-f253951d063a
SEAFILE_URL=https://cld-dkr-makuro-seafile.wibudev.com
SEAFILE_PUBLIC_SHARE_TOKEN=3a9a9ecb5e244f4da8ae
# Upload
WIBU_UPLOAD_DIR=uploads
WIBU_DOWNLOAD_DIR="./download"
NEXT_PUBLIC_BASE_URL="http://localhost:3000"
EMAIL_USER=nicoarya20@gmail.com
EMAIL_PASS=hymmfpcaqzqkfgbh
BASE_SESSION_KEY=kp9sGx91as0Kj2Ls81nAsl2Kdj13KsxP
BASE_TOKEN_KEY=Qm82JsA92lMnKw0291mxKaaP02KjslaA
# BOT-TELE
BOT_TOKEN=8479423145:AAE9ArrOgTD3DyVxYSVs3IXN40u_sL6c9sw
CHAT_ID=-1003368982298

41
.env.example Normal file
View File

@@ -0,0 +1,41 @@
# Database Configuration
DATABASE_URL="postgresql://username:password@localhost:5432/desa-darmasaba?schema=public"
# Seafile Configuration (File Storage)
SEAFILE_TOKEN=your_seafile_token
SEAFILE_REPO_ID=your_seafile_repo_id
SEAFILE_URL=https://your-seafile-instance.com
SEAFILE_PUBLIC_SHARE_TOKEN=your_seafile_public_share_token
# Upload Configuration
WIBU_UPLOAD_DIR=uploads
WIBU_DOWNLOAD_DIR=./download
# Application Configuration
# IMPORTANT: For staging/production, set this to your actual domain
# Local development: NEXT_PUBLIC_BASE_URL=http://localhost:3000
# Staging: NEXT_PUBLIC_BASE_URL=https://desa-darmasaba-stg.wibudev.com
# Production: NEXT_PUBLIC_BASE_URL=https://your-production-domain.com
# Or use relative URL '/' for automatic protocol/domain detection (recommended)
NEXT_PUBLIC_BASE_URL=/
# Email Configuration (for notifications/subscriptions)
EMAIL_USER=your_email@gmail.com
EMAIL_PASS=your_email_app_password
# Session Configuration
BASE_SESSION_KEY=your_session_key_generate_secure_random_string
BASE_TOKEN_KEY=your_jwt_secret_key_generate_secure_random_string
# Telegram Bot Configuration (for notifications)
BOT_TOKEN=your_telegram_bot_token
CHAT_ID=your_telegram_chat_id
# Session Password (for iron-session)
SESSION_PASSWORD="your_session_password_min_32_characters_long_secure"
# ElevenLabs API Key (for TTS features - optional)
ELEVENLABS_API_KEY=your_elevenlabs_api_key
# Environment (optional, defaults to development)
# NODE_ENV=development

View File

@@ -1,43 +1,52 @@
#!/usr/bin/env bun
import { readFileSync } from "node:fs";
import { readFileSync, existsSync } from "node:fs";
import { join } from "node:path";
// 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);
// Function to manually load .env from project root if process.env is missing keys
function loadEnv() {
const envPath = join(process.cwd(), ".env");
if (existsSync(envPath)) {
const envContent = readFileSync(envPath, "utf-8");
const lines = envContent.split("\n");
for (const line of lines) {
if (line && !line.startsWith("#")) {
const [key, ...valueParts] = line.split("=");
if (key && valueParts.length > 0) {
const value = valueParts.join("=").trim().replace(/^["']|["']$/g, "");
process.env[key.trim()] = value;
}
}
}
};
search(obj);
return longest;
}
}
async function run() {
try {
// Ensure environment variables are loaded
loadEnv();
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));
let finalText = "";
let sessionId = "web-desa-darmasaba";
try {
// Try parsing as JSON first
const input = JSON.parse(inputRaw);
sessionId = input.session_id || "web-desa-darmasaba";
finalText = typeof input === "string" ? input : (input.response || input.text || JSON.stringify(input));
} catch {
// If not JSON, use raw text
finalText = inputRaw;
}
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(", ");
if (!BOT_TOKEN || !CHAT_ID) {
console.error("Missing BOT_TOKEN or CHAT_ID in environment variables");
return;
}
const message =
@@ -45,7 +54,7 @@ async function run() {
`🆔 Session: \`${sessionId}\` \n\n` +
`🧠 Output:\n${finalText.substring(0, 3500)}`;
await fetch(`https://api.telegram.org/bot${BOT_TOKEN}/sendMessage`, {
const res = await fetch(`https://api.telegram.org/bot${BOT_TOKEN}/sendMessage`, {
method: "POST",
headers: { "Content-Type": "application/json" },
body: JSON.stringify({
@@ -55,6 +64,13 @@ async function run() {
}),
});
if (!res.ok) {
const errorData = await res.json();
console.error("Telegram API Error:", errorData);
} else {
console.log("Notification sent successfully!");
}
process.stdout.write(JSON.stringify({ status: "continue" }));
} catch (err) {
console.error("Hook Error:", err);

9
.gitignore vendored
View File

@@ -29,7 +29,12 @@ yarn-error.log*
.pnpm-debug.log*
# env
.env*
# env local files (keep .env.example)
.env.local
.env*.local
.env.production
.env.development
!.env.example
# QC
QC
@@ -52,7 +57,5 @@ next-env.d.ts
.github/
.env.*
*.tar.gz

View File

@@ -10,7 +10,7 @@ Desa Darmasaba is a Next.js 15 application for village management services in Ba
- **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
- **State Management**: Jotai for global state
- **Authentication**: JWT with iron-session
## Build Commands
@@ -105,39 +105,11 @@ import { MyComponent } from '@/components/my-component'
- Add loading states and error states for async operations
### State Management
- Use Valtio for global state (proxy pattern)
- State dibagi menjadi admin dan public domains
- Use Jotai atoms for global state
- Keep local state in components when possible
- Use SWR for server state caching
- Use React Query (SWR) for server state caching
- Implement optimistic updates for better UX
**State Structure:**
```
src/state/
├── admin/ # Admin dashboard state
│ ├── adminNavState.ts
│ ├── adminAuthState.ts
│ ├── adminFormState.ts
│ └── adminModuleState.ts
├── public/ # Public pages state
│ ├── publicNavState.ts
│ └── publicMusicState.ts
├── darkModeStore.ts # Dark mode state
└── index.ts # Central exports
```
**Usage Examples:**
```typescript
// Import state
import { adminNavState, useAdminNav } from '@/state';
// In non-React code
adminNavState.mobileOpen = true;
// In React components
const { mobileOpen, toggleMobile } = useAdminNav();
```
### Styling
- Primary: Mantine UI components
- Use Mantine theme system for customization
@@ -155,13 +127,9 @@ const { mobileOpen, toggleMobile } = useAdminNav();
```
src/
├── app/ # Next.js app router pages
├── components/ # Reusable React components
├── components/ # Reusable React components
├── lib/ # Utility functions and configurations
├── state/ # Valtio state management
│ ├── admin/ # Admin domain state
│ ├── public/ # Public domain state
│ └── index.ts # Central exports
├── store/ # Legacy store (deprecated)
├── state/ # Jotai atoms and state management
├── types/ # TypeScript type definitions
└── con/ # Constants and static data
```

View File

@@ -1,255 +0,0 @@
# 🐛 DEBUGGING GUIDE - Music State
## Problem: `window.publicMusicState` is undefined
### Possible Causes & Solutions
---
### 1⃣ **Debug Utility Not Loaded**
**Check:** Open browser console and look for:
```
[Debug] State exposed to window object:
✅ window.publicMusicState
✅ window.adminNavState
✅ window.adminAuthState
```
**If NOT visible:**
- Debug utility not imported
- Check `src/app/layout.tsx` has: `import '@/lib/debug-state';`
---
### 2⃣ **Timing Issue - Console.log Too Early**
**Problem:** You're checking `window.publicMusicState` before it's exposed.
**Solution:** Wait for page to fully load, then check:
```javascript
// In browser console, type:
window.publicMusicState
```
**Expected Output:**
```javascript
{
isPlaying: false,
currentSong: null,
currentSongIndex: -1,
musikData: [],
currentTime: 0,
duration: 0,
volume: 70,
isMuted: false,
isRepeat: false,
isShuffle: false,
isLoading: true,
isPlayerOpen: false,
error: null,
playSong: ƒ,
togglePlayPause: ƒ,
// ... all methods
}
```
---
### 3⃣ **Alternative Debug Methods**
If `window.publicMusicState` still undefined, try these:
#### Method 1: Use Helper Function
```javascript
// In browser console:
window.getMusicState()
```
#### Method 2: Import Directly (in console)
```javascript
// This won't work in console, but you can add to your component:
import { publicMusicState } from '@/state/public/publicMusicState';
console.log('Music State:', publicMusicState);
```
#### Method 3: Check from Component
Add to any component:
```typescript
useEffect(() => {
console.log('Music State:', window.publicMusicState);
}, []);
```
---
### 4⃣ **Verify Import Chain**
Check if all files are properly imported:
```
src/app/layout.tsx
└─ import '@/lib/debug-state'
└─ import { publicMusicState } from '@/state/public/publicMusicState'
└─ Exports proxy state
```
---
### 5⃣ **Check Browser Console for Errors**
Look for errors like:
-`Cannot find module '@/state/public/publicMusicState'`
-`publicMusicState is not defined`
-`Failed to load module`
**If you see these:**
- Check TypeScript compilation: `bunx tsc --noEmit`
- Check file paths are correct
- Restart dev server: `bun run dev`
---
### 6⃣ **Manual Test - Add to Component**
Temporarily add to any page component:
```typescript
'use client';
import { publicMusicState } from '@/state/public/publicMusicState';
import { useEffect } from 'react';
export default function TestPage() {
useEffect(() => {
console.log('🎵 Music State:', publicMusicState);
console.log('🎵 Is Playing:', publicMusicState.isPlaying);
console.log('🎵 Current Song:', publicMusicState.currentSong);
}, []);
return <div>Check console</div>;
}
```
---
### 7⃣ **Quick Fix - Re-import in Layout**
If still undefined, add explicit import in `src/app/layout.tsx`:
```typescript
import '@/lib/debug-state'; // Debug state exposure
// Add this AFTER imports
if (typeof window !== 'undefined') {
import('@/state/public/publicMusicState').then(({ publicMusicState }) => {
(window as any).publicMusicState = publicMusicState.publicMusicState;
console.log('✅ Music state manually exposed!');
});
}
```
---
### 8⃣ **Verify State is Working**
Test state reactivity:
```javascript
// In browser console:
window.publicMusicState.volume = 80
console.log(window.publicMusicState.volume) // Should log: 80
// Change state
window.publicMusicState.togglePlayer()
console.log(window.publicMusicState.isPlayerOpen) // Should log: true
```
---
### 9⃣ **Check Valtio Installation**
Ensure Valtio is installed:
```bash
bun list valtio
```
Should show: `valtio@1.x.x`
If not installed:
```bash
bun install valtio
```
---
### 🔟 **Nuclear Option - Re-export**
Create new file `src/lib/music-debug.ts`:
```typescript
'use client';
import { publicMusicState } from '@/state/public/publicMusicState';
if (typeof window !== 'undefined') {
(window as any).publicMusicState = publicMusicState;
console.log('🎵 Music state exposed!');
}
export { publicMusicState };
```
Then import in layout:
```typescript
import '@/lib/music-debug';
```
---
## ✅ Working Checklist
- [ ] Debug utility imported in layout.tsx
- [ ] Console shows "[Debug] State exposed" message
- [ ] No TypeScript errors
- [ ] No console errors about missing modules
- [ ] `window.publicMusicState` returns object (not undefined)
- [ ] State has all properties (isPlaying, currentSong, etc.)
- [ ] State methods are functions (playSong, togglePlayPause, etc.)
---
## 🎯 Expected Console Output
When page loads, you should see:
```
[Debug] State exposed to window object:
✅ window.publicMusicState
✅ window.adminNavState
✅ window.adminAuthState
Type "window.publicMusicState" in console to check state
[MusicState] Loading musik data...
[MusicState] API response: {...}
[MusicState] Loaded 2 active songs
[MusicState] First song: {judul: 'Celengan Rindu', ...}
```
---
## 📞 Still Having Issues?
If `window.publicMusicState` still undefined after trying all above:
1. **Clear browser cache** - Hard refresh (Ctrl+Shift+R)
2. **Restart dev server** - `bun run dev`
3. **Check file permissions** - Ensure files are readable
4. **Check Next.js config** - Ensure path aliases work
5. **Try incognito mode** - Rule out extensions interfering
---
Last updated: March 9, 2026

60
Dockerfile Normal file
View File

@@ -0,0 +1,60 @@
# Stage 1: Build
FROM oven/bun:1.3 AS build
# Install build dependencies for native modules
RUN apt-get update && apt-get install -y \
python3 \
make \
g++ \
&& rm -rf /var/lib/apt/lists/*
# Set the working directory
WORKDIR /app
# Copy package files
COPY package.json bun.lock* ./
# Install dependencies
RUN bun install --frozen-lockfile
# Copy the rest of the application code
COPY . .
# Use .env.example as default env for build
RUN cp .env.example .env
# Generate Prisma client
RUN bun x prisma generate
# Build the application frontend
ENV NODE_ENV=production
RUN bun run build
# Stage 2: Runtime
FROM oven/bun:1.3-slim AS runtime
# Set environment variables
ENV NODE_ENV=production
# Install runtime dependencies
RUN apt-get update && apt-get install -y \
postgresql-client \
&& rm -rf /var/lib/apt/lists/*
# Set the working directory
WORKDIR /app
# Copy necessary files from build stage
COPY --from=build /app/package.json ./
COPY --from=build /app/tsconfig.json ./
COPY --from=build /app/.next ./.next
COPY --from=build /app/public ./public
COPY --from=build /app/src ./src
COPY --from=build /app/node_modules ./node_modules
COPY --from=build /app/prisma ./prisma
# Expose the port
EXPOSE 3000
# Start the application
CMD ["bun", "start"]

173
MUSIK_CREATE_ANALYSIS.md Normal file
View File

@@ -0,0 +1,173 @@
# Musik Desa - Create Feature Analysis
## Error Summary
**Error**: `ERR_BLOCKED_BY_CLIENT` saat create musik di staging environment
## Root Cause Analysis
### 1. **CORS Configuration Issue** (Primary)
File: `src/app/api/[[...slugs]]/route.ts`
The CORS configuration has specific origins listed:
```typescript
const corsConfig = {
origin: [
"http://localhost:3000",
"http://localhost:3001",
"https://cld-dkr-desa-darmasaba-stg.wibudev.com",
"https://cld-dkr-staging-desa-darmasaba.wibudev.com",
"*",
],
// ...
}
```
**Problem**: The wildcard `*` is at the end, but some browsers don't respect it when `credentials: true` is set.
### 2. **API Fetch Base URL** (Secondary)
File: `src/lib/api-fetch.ts`
```typescript
const BASE_URL = process.env.NEXT_PUBLIC_BASE_URL || 'http://localhost:3000'
```
**Problem**:
- In staging, this might still default to `http://localhost:3000`
- Mixed content (HTTPS frontend → HTTP API) gets blocked by browsers
- The `NEXT_PUBLIC_BASE_URL` environment variable might not be set in staging
### 3. **File Storage Upload Path** (Tertiary)
File: `src/app/api/[[...slugs]]/_lib/fileStorage/_lib/create.ts`
```typescript
const UPLOAD_DIR = process.env.WIBU_UPLOAD_DIR;
```
**Problem**: If `WIBU_UPLOAD_DIR` is not set or points to a non-writable location, uploads will fail silently.
## Solution
### Fix 1: Update CORS Configuration
**File**: `src/app/api/[[...slugs]]/route.ts`
```typescript
// Move wildcard to first position and ensure it works with credentials
const corsConfig = {
origin: [
"*", // Allow all origins (for staging flexibility)
"http://localhost:3000",
"http://localhost:3001",
"https://cld-dkr-desa-darmasaba-stg.wibudev.com",
"https://cld-dkr-staging-desa-darmasaba.wibudev.com",
"https://desa-darmasaba-stg.wibudev.com"
],
methods: ["GET", "POST", "PATCH", "DELETE", "PUT", "OPTIONS"] as HTTPMethod[],
allowedHeaders: ["Content-Type", "Authorization", "Accept"],
exposedHeaders: ["Content-Range", "X-Content-Range"],
maxAge: 86400, // 24 hours
credentials: true,
};
```
### Fix 2: Add Environment Variable Validation
**File**: `.env.example` (update)
```bash
# Application Configuration
NEXT_PUBLIC_BASE_URL=http://localhost:3000
# For staging/production, set this to your actual domain
# NEXT_PUBLIC_BASE_URL=https://cld-dkr-desa-darmasaba-stg.wibudev.com
```
### Fix 3: Update API Fetch to Handle Relative URLs
**File**: `src/lib/api-fetch.ts`
```typescript
import { AppServer } from '@/app/api/[[...slugs]]/route'
import { treaty } from '@elysiajs/eden'
// Use relative URL for better deployment flexibility
const BASE_URL = process.env.NEXT_PUBLIC_BASE_URL || '/'
const ApiFetch = treaty<AppServer>(BASE_URL)
export default ApiFetch
```
### Fix 4: Add Error Handling in Create Page
**File**: `src/app/admin/(dashboard)/musik/create/page.tsx`
Add better error logging to diagnose issues:
```typescript
const handleSubmit = async () => {
// ... validation ...
try {
setIsSubmitting(true);
// Upload cover image
const coverRes = await ApiFetch.api.fileStorage.create.post({
file: coverFile,
name: coverFile.name,
});
if (!coverRes.data?.data?.id) {
console.error('Cover upload failed:', coverRes);
return toast.error('Gagal mengunggah cover, silakan coba lagi');
}
// ... rest of the code ...
} catch (error) {
console.error('Error creating musik:', {
error,
message: error instanceof Error ? error.message : 'Unknown error',
stack: error instanceof Error ? error.stack : undefined,
});
toast.error('Terjadi kesalahan saat membuat musik');
} finally {
setIsSubmitting(false);
}
};
```
## Testing Checklist
### Local Development
- [ ] Test create musik with cover image and audio file
- [ ] Verify CORS headers in browser DevTools Network tab
- [ ] Check that file uploads are saved to correct directory
### Staging Environment
- [ ] Set `NEXT_PUBLIC_BASE_URL` to staging domain
- [ ] Verify HTTPS is used for all API calls
- [ ] Check browser console for mixed content warnings
- [ ] Verify `WIBU_UPLOAD_DIR` is set and writable
- [ ] Test create musik end-to-end
## Additional Notes
### ERR_BLOCKED_BY_CLIENT Common Causes:
1. **CORS policy blocking** - Most likely cause
2. **Ad blockers** - Can block certain API endpoints
3. **Mixed content** - HTTPS page making HTTP requests
4. **Content Security Policy (CSP)** - Restrictive CSP headers
5. **Browser extensions** - Privacy/security extensions blocking requests
### Debugging Steps:
1. Open browser DevTools → Network tab
2. Try to create musik
3. Look for failed requests (red status)
4. Check the "Headers" tab for:
- Request URL (should be correct domain)
- Response headers (should have `Access-Control-Allow-Origin`)
- Status code (4xx/5xx indicates server-side issue)
5. Check browser console for CORS errors
## Recommended Next Steps
1. **Immediate**: Update CORS configuration to allow staging domain
2. **Short-term**: Add proper environment variable validation
3. **Long-term**: Implement proper error boundaries and logging

File diff suppressed because it is too large Load Diff

View File

@@ -1,269 +0,0 @@
# Security Fixes Implementation
**Date:** March 9, 2026
**Issue:** SECURITY VULNERABILITIES - CRITICAL (from QUALITY_CONTROL_REPORT.md)
**Status:** ✅ COMPLETED
---
## 🔒 Security Vulnerabilities Fixed
### 3.1 ✅ OTP Sent via POST Request (Not GET)
**Problem:** OTP code was exposed in URL query strings, which are:
- Logged by web servers and proxies
- Visible in browser history
- Potentially intercepted in man-in-the-middle attacks
**Solution:** Created secure WhatsApp service that uses POST request
**Files Changed:**
1. `src/lib/whatsapp.ts` - ✅ NEW - Secure WhatsApp OTP service
2. `src/app/api/[[...slugs]]/_lib/auth/login/route.ts` - Updated to use new service
**Implementation:**
```typescript
// OLD (Insecure) - GET with OTP in URL
const waRes = await fetch(
`https://wa.wibudev.com/code?nom=${nomor}&text=Kode OTP: ${codeOtp}`
);
// NEW (Secure) - POST with OTP reference
const waResult = await sendWhatsAppOTP({
nomor: nomor,
otpId: otpRecord.id, // Send reference, not actual OTP
message: formatOTPMessage(codeOtp),
});
```
**Benefits:**
- ✅ OTP not exposed in URL
- ✅ Not logged by servers/proxies
- ✅ Not visible in browser history
- ✅ Uses proper HTTP method for sensitive operations
---
### 3.2 ✅ Strong Session Password Enforcement
**Problem:** Default fallback password in production creates security vulnerability
**Solution:** Enforce SESSION_PASSWORD environment variable with validation
**Files Changed:**
- `src/lib/session.ts` - Added runtime validation
**Implementation:**
```typescript
// Validate SESSION_PASSWORD environment variable
if (!process.env.SESSION_PASSWORD) {
throw new Error(
'SESSION_PASSWORD environment variable is required. ' +
'Please set a strong password (min 32 characters) in your .env file.'
);
}
// Validate password length for security
if (process.env.SESSION_PASSWORD.length < 32) {
throw new Error(
'SESSION_PASSWORD must be at least 32 characters long for security. ' +
'Please use a strong random password.'
);
}
```
**Benefits:**
- ✅ No default/fallback password
- ✅ Enforces strong password (min 32 chars)
- ✅ Fails fast on startup if not configured
- ✅ Clear error messages for developers
**Migration:**
Add to your `.env.local`:
```bash
# Generate a strong random password (min 32 characters)
SESSION_PASSWORD="your-super-secure-random-password-at-least-32-chars"
```
---
### 3.3 ✅ Input Validation with Zod
**Problem:** No input validation - direct type casting without sanitization
**Solution:** Comprehensive Zod validation schemas with HTML sanitization
**Files Created:**
1. `src/lib/validations/index.ts` - ✅ NEW - Centralized validation schemas
2. `src/lib/sanitizer.ts` - ✅ NEW - HTML/content sanitization utilities
**Files Changed:**
- `src/app/api/[[...slugs]]/_lib/desa/berita/create.ts` - Added validation + sanitization
**Validation Schemas:**
```typescript
// Berita validation
export const createBeritaSchema = z.object({
judul: z.string().min(5).max(255),
deskripsi: z.string().min(10).max(500),
content: z.string().min(50),
kategoriBeritaId: z.string().cuid(),
imageId: z.string().cuid(),
imageIds: z.array(z.string().cuid()).optional(),
linkVideo: z.string().url().optional().or(z.literal('')),
});
// Login validation
export const loginRequestSchema = z.object({
nomor: z.string().min(10).max(15).regex(/^[0-9]+$/),
});
// OTP verification
export const otpVerificationSchema = z.object({
nomor: z.string().min(10).max(15),
kodeId: z.string().cuid(),
otp: z.string().length(6).regex(/^[0-9]+$/),
});
```
**Sanitization:**
```typescript
// HTML sanitization to prevent XSS
const sanitizedContent = sanitizeHtml(validated.content);
// YouTube URL sanitization
const sanitizedLinkVideo = validated.linkVideo
? sanitizeYouTubeUrl(validated.linkVideo)
: null;
```
**Benefits:**
- ✅ Type-safe validation with Zod
- ✅ Clear error messages for users
- ✅ HTML sanitization prevents XSS attacks
- ✅ URL validation prevents malicious links
- ✅ Centralized schemas for consistency
---
## 📋 Additional Security Improvements
### Error Handling
All API endpoints now properly handle validation errors:
```typescript
try {
const validated = createBeritaSchema.parse(context.body);
// ... process data
} catch (error) {
if (error instanceof Error && error.constructor.name === 'ZodError') {
const zodError = error as import('zod').ZodError;
return {
success: false,
message: "Validasi gagal",
errors: zodError.errors.map(e => ({
field: e.path.join('.'),
message: e.message,
})),
};
}
throw error;
}
```
### Cleanup on Failure
OTP records are cleaned up if WhatsApp delivery fails:
```typescript
if (waResult.status !== "success") {
await prisma.kodeOtp.delete({
where: { id: otpRecord.id },
}).catch(() => {});
return NextResponse.json(
{ success: false, message: "Gagal mengirim kode verifikasi" },
{ status: 400 }
);
}
```
---
## 🧪 Testing
Run TypeScript check to ensure no errors:
```bash
bunx tsc --noEmit
```
---
## 📊 Security Metrics
| Metric | Before | After | Improvement |
|--------|--------|-------|-------------|
| OTP in URL | ✅ Yes | ❌ No | ✅ 100% |
| Session Password | ⚠️ Optional | ✅ Required | ✅ 100% |
| Input Validation | ❌ None | ✅ Zod | ✅ 100% |
| HTML Sanitization | ❌ None | ✅ Yes | ✅ 100% |
| Validation Schemas | ❌ None | ✅ 7 schemas | ✅ New |
---
## 🚀 Next Steps
### Immediate (Recommended)
1. **Update other auth routes** - Apply same pattern to:
- `src/app/api/auth/register/route.ts`
- `src/app/api/auth/resend/route.ts`
- `src/app/api/auth/send-otp-register/route.ts`
2. **Add more validation schemas** for:
- Update berita
- Delete operations
- Other CRUD endpoints
3. **Add rate limiting** for:
- Login attempts
- OTP requests
- Password reset
### Short-term
1. **Add CSRF protection** for state-changing operations
2. **Implement request logging** for security audits
3. **Add security headers** (CSP, X-Frame-Options, etc.)
4. **Set up security monitoring** (failed login attempts, etc.)
---
## 📚 Documentation
New documentation files created:
- `src/lib/whatsapp.ts` - WhatsApp service documentation
- `src/lib/validations/index.ts` - Validation schemas documentation
- `src/lib/sanitizer.ts` - Sanitization utilities documentation
---
## ✅ Checklist
- [x] OTP transmission secured (POST instead of GET)
- [x] Session password enforced (no fallback)
- [x] Input validation implemented (Zod)
- [x] HTML sanitization added (XSS prevention)
- [x] Error handling improved
- [x] TypeScript compilation passes
- [x] Documentation updated
---
**Security Status:** 🟢 SIGNIFICANTLY IMPROVED
All critical security vulnerabilities identified in the quality control report have been addressed. The application now follows security best practices for:
- Sensitive data transmission
- Session management
- Input validation
- XSS prevention

View File

@@ -1,244 +0,0 @@
# State Management Refactoring Summary
**Date:** March 9, 2026
**Issue:** STATE MANAGEMENT CHAOS - CRITICAL (from QUALITY_CONTROL_REPORT.md)
**Status:** ✅ COMPLETED
---
## Problem Statement
The codebase had multiple state management solutions used inconsistently:
- Valtio (primary but not documented)
- React Context (MusicContext)
- AGENTS.md mentioned Jotai (incorrect documentation)
- No clear separation between admin and public state
- Tight coupling between domains
---
## Changes Made
### 1. **Created Organized State Structure**
```
src/state/
├── admin/ # Admin dashboard state
│ ├── index.ts # Admin state exports
│ ├── adminNavState.ts # ✅ NEW - Navigation state
│ ├── adminAuthState.ts # ✅ NEW - Authentication state
│ ├── adminFormState.ts # ✅ NEW - Form/image state
│ └── adminModuleState.ts # ✅ NEW - Module-specific state
├── public/ # Public pages state
│ ├── index.ts # Public state exports
│ ├── publicNavState.ts # ✅ NEW - Navigation state
│ └── publicMusicState.ts # ✅ NEW - Music player state
├── darkModeStore.ts # Existing (kept as-is)
└── index.ts # ✅ NEW - Central exports
```
### 2. **Refactored MusicContext to Valtio**
**Before:**
```typescript
// Pure React Context with useState
const [isPlaying, setIsPlaying] = useState(false);
const [currentSong, setCurrentSong] = useState<Musik | null>(null);
// ... 300+ lines of Context logic
```
**After:**
```typescript
// Valtio state with React Context wrapper
export const publicMusicState = proxy<{
isPlaying: boolean;
currentSong: Musik | null;
// ... all state
playSong: (song: Musik) => void;
togglePlayPause: () => void;
// ... all methods
}>({...});
// Backward compatible Context wrapper
export function MusicProvider({ children }) {
// Uses Valtio state internally
}
```
**Files Changed:**
- `src/app/context/MusicContext.tsx` - Refactored to use Valtio
- `src/app/context/MusicContext.ts` - ✅ NEW - Compatibility layer
- `src/app/context/MusicProvider.tsx` - ✅ NEW - Provider implementation
- `src/state/public/publicMusicState.ts` - ✅ NEW - Valtio state
### 3. **Updated Legacy Files for Backward Compatibility**
All existing state files now re-export from new structure:
```typescript
// src/state/state-nav.ts (OLD - kept for compatibility)
import { adminNavState } from './admin/adminNavState';
export const stateNav = adminNavState;
export default stateNav;
// src/store/authStore.ts (OLD - kept for compatibility)
import { adminAuthState } from '../state/admin/adminAuthState';
export const authStore = adminAuthState;
export default authStore;
// src/state/state-list-image.ts (OLD - kept for compatibility)
import { adminFormState } from './admin/adminFormState';
export const stateListImage = adminFormState;
export default stateListImage;
```
### 4. **Fixed Documentation Mismatch**
**Updated AGENTS.md:**
- ✅ Changed "Jotai" to "Valtio"
- ✅ Added state structure diagram
- ✅ Added usage examples
- ✅ Updated file organization
### 5. **Created Comprehensive Documentation**
**New File:** `docs/STATE_MANAGEMENT.md`
Contains:
- Overview of Valtio usage
- State structure explanation
- Basic usage examples
- Domain-specific state guide
- Async operations pattern
- Best practices (DO/DON'T)
- Migration guide from legacy state
- Troubleshooting tips
---
## Benefits
### ✅ Clear Separation of Concerns
- Admin state: `/admin` routes only
- Public state: `/darmasaba` routes only
- No more cross-domain coupling
### ✅ Consistent Pattern
- All state uses Valtio
- Same pattern across entire codebase
- Methods defined within state objects
### ✅ Backward Compatible
- All existing imports still work
- No breaking changes to existing code
- Gradual migration path
### ✅ Better Documentation
- AGENTS.md now accurate (Valtio, not Jotai)
- Comprehensive guide in docs/STATE_MANAGEMENT.md
- Clear usage examples
### ✅ Type Safe
- Full TypeScript support
- All state properly typed
- No `any` types in new code
---
## Migration Guide
### For New Code
```typescript
// Import admin state
import { adminNavState, useAdminNav } from '@/state';
// Use in component
function MyComponent() {
const { mobileOpen, toggleMobile } = useAdminNav();
return <Button onClick={toggleMobile}>Menu</Button>;
}
// Use outside component
adminNavState.mobileOpen = true;
```
### For Existing Code
No changes needed! All existing imports continue to work:
```typescript
// Still works
import stateNav from '@/state/state-nav';
import { authStore } from '@/store/authStore';
import { useMusic } from '@/app/context/MusicContext';
```
---
## Testing
All TypeScript checks pass:
```bash
bunx tsc --noEmit
# ✅ No errors
```
---
## Files Created
1. `src/state/admin/index.ts`
2. `src/state/admin/adminNavState.ts`
3. `src/state/admin/adminAuthState.ts`
4. `src/state/admin/adminFormState.ts`
5. `src/state/admin/adminModuleState.ts`
6. `src/state/public/index.ts`
7. `src/state/public/publicNavState.ts`
8. `src/state/public/publicMusicState.ts`
9. `src/state/index.ts`
10. `src/app/context/MusicContext.ts`
11. `src/app/context/MusicProvider.tsx`
12. `docs/STATE_MANAGEMENT.md`
13. `STATE_REFACTORING_SUMMARY.md` (this file)
---
## Files Modified
1. `src/state/state-nav.ts` - Re-export from new structure
2. `src/store/authStore.ts` - Re-export from new structure
3. `src/state/state-list-image.ts` - Re-export from new structure
4. `src/state/state-layanan.ts` - Simplified
5. `src/state/darkModeStore.ts` - Updated docs
6. `src/app/context/MusicContext.tsx` - Refactored to use Valtio
7. `AGENTS.md` - Fixed Jotai → Valtio documentation
---
## Next Steps (Optional)
Future improvements that can be made:
1. **Gradually migrate** old state files to new structure
2. **Remove legacy files** once all usages are updated
3. **Add unit tests** for state management
4. **Add state persistence** for admin preferences
5. **Implement state hydration** for SSR optimization
---
## Conclusion
The state management refactoring is **COMPLETE**. All issues identified in the quality control report have been addressed:
- ✅ Single state management solution (Valtio)
- ✅ Clear separation between admin and public domains
- ✅ Documentation updated (AGENTS.md)
- ✅ Comprehensive guide created (docs/STATE_MANAGEMENT.md)
- ✅ Backward compatible (no breaking changes)
- ✅ TypeScript compilation passes
The codebase now has a **consistent, well-documented, and maintainable** state management structure.

View File

@@ -1,400 +0,0 @@
---
🧪 TESTING GUIDE
1⃣ STATE MANAGEMENT REFACTORING
A. Music Player State (Valtio)
Page: http://localhost:3000/darmasaba/musik/musik-desa
Test Steps:
1. Buka halaman musik desa
2. Klik lagu untuk memutar
3. Test tombol play/pause
4. Test next/previous
5. Test volume control
6. Test shuffle/repeat
7. Refresh page - state harus tetap ada
Expected Result:
- ✅ Musik bisa diputar
- ✅ Semua kontrol berfungsi
- ✅ State reactive (UI update otomatis)
- ✅ Tidak ada error di console
Console Check:
1 // Buka browser console, ketik:
2 window.publicMusicState
3 // Harus bisa akses state langsung
---
B. Admin Navigation State
Page: http://localhost:3000/admin/dashboard
Test Steps:
1. Login ke admin panel
2. Test toggle sidebar (collapse/expand)
3. Test mobile menu (hamburger menu)
4. Test hover menu items
5. Test search functionality
6. Navigate antar module
Expected Result:
- ✅ Sidebar bisa collapse/expand
- ✅ Mobile menu berfungsi
- ✅ Menu hover responsive
- ✅ State persist saat navigate
---
2⃣ SECURITY FIXES
A. OTP via POST (Not GET) - CRITICAL ⚠️
Page: http://localhost:3000/admin/login
Test Steps:
1. Buka halaman login admin
2. Masukkan nomor WhatsApp valid
3. Klik "Kirim Kode OTP"
4. Check Network tab di browser DevTools
Network Tab Check:
1 ❌ BEFORE (Insecure):
2 Request URL: https://wa.wibudev.com/code?nom=08123456789&text=Kode OTP: 123456
3 Method: GET
4
5 ✅ AFTER (Secure):
6 Request URL: https://wa.wibudev.com/send
7 Method: POST
8 Request Payload: {
9 "nomor": "08123456789",
10 "otpId": "clxxx...",
11 "message": "Website Desa Darmasaba..."
12 }
Expected Result:
- ✅ Request ke WhatsApp menggunakan POST
- ✅ OTP TIDAK terlihat di URL
- ✅ OTP hanya ada di message body
- ✅ Dapat OTP via WhatsApp
Browser History Check:
- Buka browser history
- Cari URL dengan "wa.wibudev.com"
- ✅ TIDAK BOLEH ADA OTP di URL
---
B. Session Password Enforcement
File: .env.local
Test 1 - Tanpa SESSION_PASSWORD:
1 # Hapus atau comment SESSION_PASSWORD di .env.local
2 # SESSION_PASSWORD=""
Restart server:
1 bun run dev
Expected Result:
- ❌ Server GAGAL start
- ✅ Error message: "SESSION_PASSWORD environment variable is required"
---
Test 2 - Password Pendek (< 32 chars):
1 # Password terlalu pendek
2 SESSION_PASSWORD="short"
Restart server:
1 bun run dev
Expected Result:
- Server GAGAL start
- Error message: "SESSION_PASSWORD must be at least 32 characters long"
---
Test 3 - Password Valid (≥ 32 chars):
1 # Generate password kuat (min 32 chars)
2 SESSION_PASSWORD="this-is-a-very-secure-password-with-more-than-32-characters"
Restart server:
1 bun run dev
Expected Result:
- Server BERHASIL start
- Tidak ada error
- Bisa login ke admin panel
---
C. Input Validation (Zod)
Page: http://localhost:3000/admin/desa/berita/list-berita/create
Test 1 - Judul Pendek (< 5 chars):
1 Judul: "abc"
Expected:
- Error: "Judul minimal 5 karakter"
---
Test 2 - Judul Terlalu Panjang (> 255 chars):
1 Judul: "abc..." (300 chars) ❌
Expected:
- ✅ Error: "Judul maksimal 255 karakter"
---
Test 3 - Deskripsi Pendek (< 10 chars):
1 Judul: "Judul Valid"
2 Deskripsi: "abc"
Expected:
- Error: "Deskripsi minimal 10 karakter"
---
Test 4 - Konten Pendek (< 50 chars):
1 Judul: "Judul Valid"
2 Deskripsi: "Deskripsi yang cukup panjang"
3 Konten: "abc"
Expected:
- Error: "Konten minimal 50 karakter"
---
Test 5 - YouTube URL Invalid:
1 Link Video: "https://youtube.com"
Expected:
- Error: "Format URL YouTube tidak valid"
---
Test 6 - XSS Attempt:
1 Konten: "<script>alert('XSS')</script>Content yang valid..." ❌
Expected:
- ✅ Script tag dihapus
- ✅ Content tersimpan tanpa <script>
- ✅ Data tersimpan dengan aman
Verify di Database:
1 SELECT content FROM berita ORDER BY "createdAt" DESC LIMIT 1;
2 -- Harus tanpa <script> tag
---
Test 7 - Data Valid (Semua Field Benar):
1 Judul: "Berita Testing" ✅ (5-255 chars)
2 Deskripsi: "Deskripsi lengkap berita" ✅ (10-500 chars)
3 Konten: "Konten berita yang lengkap dan valid..." ✅ (>50 chars)
4 Kategori: [Pilih kategori] ✅
5 Featured Image: [Upload image] ✅
6 Link Video: "https://www.youtube.com/watch?v=dQw4w9WgXcQ" ✅
Expected:
- ✅ Berhasil simpan
- ✅ Redirect ke list berita
- ✅ Data tampil dengan benar
---
3⃣ ADDITIONAL PAGES TO TEST
Music Player Integration
┌────────────┬─────────────────────────────┬───────────────────────────────┐
│ Page │ URL │ Test │
├────────────┼─────────────────────────────┼───────────────────────────────┤
│ Musik Desa │ /darmasaba/musik/musik-desa │ Full player functionality │
│ Home │ /darmasaba │ Fixed player bar (if enabled) │
└────────────┴─────────────────────────────┴───────────────────────────────┘
---
Admin Pages (State Management)
┌───────────────┬───────────────────────────────────────┬───────────────────────────┐
│ Page │ URL │ Test │
├───────────────┼───────────────────────────────────────┼───────────────────────────┤
│ Login │ /admin/login │ Session state │
│ Dashboard │ /admin/dashboard │ Navigation state │
│ Berita List │ /admin/desa/berita/list-berita │ Form state │
│ Create Berita │ /admin/desa/berita/list-berita/create │ Validation + sanitization │
└───────────────┴───────────────────────────────────────┴───────────────────────────┘
---
4⃣ BROWSER CONSOLE TESTS
Test State Management Directly
Buka browser console dan test:
1 // Test 1: Access public music state
2 import { publicMusicState } from '@/state/public/publicMusicState';
3 console.log('Music State:', publicMusicState);
4
5 // Test 2: Access admin nav state
6 import { adminNavState } from '@/state/admin/adminNavState';
7 console.log('Admin Nav:', adminNavState);
8
9 // Test 3: Change state manually
10 adminNavState.mobileOpen = true;
11 console.log('Mobile Open:', adminNavState.mobileOpen);
12
13 // Test 4: Music state methods
14 publicMusicState.togglePlayer();
15 console.log('Player Open:', publicMusicState.isPlayerOpen);
---
5⃣ NETWORK TAB CHECKS
OTP Login Flow
1. Buka DevTools → Network tab
2. Login page: /admin/login
3. Submit nomor
4. Cari request ke wa.wibudev.com
Check:
1 ✅ CORRECT:
2 - Method: POST
3 - URL: https://wa.wibudev.com/send
4 - Body: { nomor, otpId, message }
5 - NO OTP in URL
6
7 ❌ WRONG:
8 - Method: GET
9 - URL: https://wa.wibudev.com/code?nom=...&text=...OTP...
10 - OTP visible in URL
---
6⃣ DATABASE CHECKS
Verify Sanitization
1 -- Check berita content setelah input XSS attempt
2 SELECT
3 id,
4 judul,
5 content,
6 "linkVideo",
7 "createdAt"
8 FROM "Berita"
9 ORDER BY "createdAt" DESC
10 LIMIT 5;
11
12 -- Content TIDAK BOLEH mengandung:
13 -- <script>, javascript:, onerror=, onclick=, dll
---
✅ TESTING CHECKLIST
1 STATE MANAGEMENT:
2 [ ] Music player works (play/pause/next/prev)
3 [ ] Volume control works
4 [ ] Shuffle/repeat works
5 [ ] State persists after refresh
6 [ ] Admin navigation works
7 [ ] Sidebar toggle works
8 [ ] Mobile menu works
9
10 SECURITY - OTP:
11 [ ] Login request uses POST (not GET)
12 [ ] OTP NOT visible in Network tab URL
13 [ ] OTP NOT in browser history
14 [ ] WhatsApp receives OTP correctly
15 [ ] Login flow completes successfully
16
17 SECURITY - SESSION:
18 [ ] Server fails without SESSION_PASSWORD
19 [ ] Server fails with short password
20 [ ] Server starts with valid password
21 [ ] Can login to admin panel
22 [ ] Session persists across pages
23
24 SECURITY - VALIDATION:
25 [ ] Short judul rejected
26 [ ] Long judul rejected
27 [ ] Short deskripsi rejected
28 [ ] Short content rejected
29 [ ] Invalid YouTube URL rejected
30 [ ] XSS attempt sanitized
31 [ ] Valid data accepted
32
33 CLEANUP:
34 [ ] No console errors
35 [ ] No TypeScript errors
36 [ ] All pages load correctly
---
🐛 TROUBLESHOOTING
Issue: "SESSION_PASSWORD environment variable is required"
Fix:
1 # Tambahkan ke .env.local
2 SESSION_PASSWORD="your-secure-password-at-least-32-characters-long"
---
Issue: WhatsApp OTP tidak terkirim
Check:
1. Network tab - apakah POST request berhasil?
2. Check logs - apakah ada error dari WhatsApp API?
3. Check nomor WhatsApp format (harus valid)
---
Issue: Validasi error tidak muncul
Check:
1. Browser console - apakah ada Zod error?
2. Network tab - check request body
3. Check schema di src/lib/validations/index.ts
---
Issue: Music player tidak berfungsi
Check:
1. Browser console - ada error?
2. Check publicMusicState di console
3. Reload page - state ter-initialize?
---
Selamat testing! Jika ada issue, check console logs dan network tab untuk debugging. 🎉

View File

@@ -1,350 +0,0 @@
# Testing Implementation Summary
## Overview
This document summarizes the comprehensive testing implementation for the Desa Darmasaba project, addressing the critically low testing coverage identified in the Quality Control Report (Issue #4).
## Implementation Date
March 9, 2026
## Test Files Created
### Unit Tests (Vitest)
#### 1. Validation Schema Tests
**File:** `__tests__/lib/validations.test.ts`
**Coverage:** 7 validation schemas with 60+ test cases
- `createBeritaSchema` - News creation validation
- `updateBeritaSchema` - News update validation
- `loginRequestSchema` - Login request validation
- `otpVerificationSchema` - OTP verification validation
- `uploadFileSchema` - File upload validation
- `registerUserSchema` - User registration validation
- `paginationSchema` - Pagination validation
**Test Cases Include:**
- Valid data acceptance
- Invalid data rejection
- Edge cases (min/max lengths, wrong formats)
- Error message validation
#### 2. Sanitizer Utility Tests
**File:** `__tests__/lib/sanitizer.test.ts`
**Coverage:** 4 sanitizer functions with 40+ test cases
- `sanitizeHtml()` - HTML sanitization for XSS prevention
- `sanitizeText()` - Plain text extraction
- `sanitizeUrl()` - URL validation and sanitization
- `sanitizeYouTubeUrl()` - YouTube URL validation
**Test Cases Include:**
- Script tag removal
- Event handler removal
- Protocol validation
- Edge cases and malformed input
#### 3. WhatsApp Service Tests
**File:** `__tests__/lib/whatsapp.test.ts`
**Coverage:** Complete WhatsApp OTP service with 25+ test cases
- `formatOTPMessage()` - OTP message formatting
- `formatOTPMessageWithReference()` - Reference-based message formatting
- `sendWhatsAppOTP()` - OTP sending functionality
**Test Cases Include:**
- Successful OTP sending
- Invalid input handling
- Error response handling
- Security verification (POST vs GET, URL exposure)
### Component Tests (Vitest + React Testing Library)
#### 4. UnifiedTypography Tests
**File:** `__tests__/components/admin/UnifiedTypography.test.tsx`
**Coverage:** 3 components with 40+ test cases
- `UnifiedTitle` - Heading component
- `UnifiedText` - Text component
- `UnifiedPageHeader` - Page header component
**Test Cases Include:**
- Prop validation
- Rendering behavior
- Style application
- Accessibility features
#### 5. UnifiedSurface Tests
**File:** `__tests__/components/admin/UnifiedSurface.test.tsx`
**Coverage:** 4 components with 35+ test cases
- `UnifiedCard` - Card container
- `UnifiedCard.Header` - Card header section
- `UnifiedCard.Body` - Card body section
- `UnifiedCard.Footer` - Card footer section
- `UnifiedDivider` - Divider component
**Test Cases Include:**
- Composition patterns
- Prop validation
- Styling consistency
- Section rendering
### E2E Tests (Playwright)
#### 6. Admin Authentication Tests
**File:** `__tests__/e2e/admin/auth.spec.ts`
**Coverage:** Complete authentication flow
- Login page rendering
- Form validation
- OTP verification flow
- Session management
- Navigation protection
**Test Cases Include:**
- Empty form validation
- Phone number validation
- OTP validation
- Successful login flow
- Responsive design
#### 7. Public Pages Tests
**File:** `__tests__/e2e/public/pages.spec.ts`
**Coverage:** Public-facing pages
- Homepage redirect
- Navigation functionality
- Section pages (PPID, Health, Education, etc.)
- News/Berita section
- Footer content
- Search functionality
- Accessibility features
- Performance metrics
**Test Cases Include:**
- Page rendering
- Navigation links
- Content verification
- Accessibility compliance
- Performance benchmarks
## Configuration Files
### Vitest Configuration
**File:** `vitest.config.ts`
```typescript
{
test: {
globals: true,
environment: 'jsdom',
setupFiles: ['./__tests__/setup.ts'],
include: ['__tests__/**/*.test.ts'],
coverage: {
provider: 'v8',
thresholds: {
branches: 50,
functions: 50,
lines: 50,
statements: 50,
},
},
},
}
```
### Test Setup
**File:** `__tests__/setup.ts`
- MSW server setup for API mocking
- window.matchMedia mock for Mantine
- IntersectionObserver mock
- Global test utilities
### Playwright Configuration
**File:** `playwright.config.ts`
- Test directory configuration
- Browser setup (Chromium)
- Web server configuration
- Retry logic for CI
## Test Statistics
| Category | Count | Status |
|----------|-------|--------|
| **Unit Test Files** | 3 | ✅ Complete |
| **Component Test Files** | 2 | ✅ Complete |
| **E2E Test Files** | 2 | ✅ Complete |
| **Total Test Files** | 7 | ✅ |
| **Total Test Cases** | 200+ | ✅ |
| **Passing Tests** | 115 | ✅ 100% |
## Coverage Areas
### Critical Files Tested
1. **Security & Validation**
- `src/lib/validations/index.ts`
- `src/lib/sanitizer.ts`
- `src/lib/whatsapp.ts`
2. **Core Components**
- `src/components/admin/UnifiedTypography.tsx`
- `src/components/admin/UnifiedSurface.tsx`
3. **API Integration**
- `src/app/api/fileStorage/*`
4. **User Flows**
- Admin authentication
- Public page navigation
## Running Tests
### All Tests
```bash
bun run test
```
### Unit Tests Only
```bash
bun run test:api
```
### E2E Tests Only
```bash
bun run test:e2e
```
### Watch Mode
```bash
bunx vitest
```
### With Coverage
```bash
bunx vitest run --coverage
```
## Test Coverage Improvement
### Before Implementation
- **Coverage:** ~2% (Critical)
- **Test Files:** 2
- **Test Cases:** <20
### After Implementation
- **Coverage:** 50%+ target achieved
- **Test Files:** 7 new files
- **Test Cases:** 200+ test cases
- **Status:** All tests passing
## Documentation
### Testing Guide
**File:** `docs/TESTING.md`
Comprehensive guide covering:
- Testing stack overview
- Test structure and organization
- Writing guidelines
- Best practices
- Common patterns
- Troubleshooting
### Quality Control Report
**File:** `QUALITY_CONTROL_REPORT.md`
Updated to reflect:
- Testing coverage improvements
- Remaining recommendations
- Future testing priorities
## Security Testing
### OTP Security Tests
- POST method verification (not GET)
- OTP not exposed in URL
- Reference ID usage
- Input validation
- Error handling
### Input Validation Tests
- XSS prevention
- SQL injection prevention
- Type validation
- Length validation
- Format validation
## Future Recommendations
### Phase 2 (Next Sprint)
1. Add tests for remaining utility functions
2. Test database operations
3. Add more E2E scenarios for admin features
4. Test state management (Valtio stores)
### Phase 3 (Future)
1. Integration tests for API endpoints
2. Performance tests
3. Load tests
4. Visual regression tests
### Coverage Goals
- **Short-term:** 50% coverage (✅ Achieved)
- **Medium-term:** 70% coverage
- **Long-term:** 80%+ coverage
## Test Quality Metrics
### Unit Tests
- Fast execution (<1s)
- Isolated tests
- Comprehensive mocking
- Clear assertions
### Component Tests
- Render testing
- Prop validation
- User interaction testing
- Accessibility testing
### E2E Tests
- Real browser testing
- Full user flows
- Responsive design
- Performance monitoring
## Continuous Integration
### GitHub Actions Workflow
Tests run automatically on:
- Pull requests
- Push to main branch
- Manual trigger
### Test Requirements
- All new features must include tests
- Bug fixes should include regression tests
- Coverage should not decrease
## Conclusion
The testing implementation has successfully addressed the critically low testing coverage identified in the Quality Control Report. The project now has:
1. **Comprehensive unit tests** for critical utilities and validation
2. **Component tests** for shared UI components
3. **E2E tests** for key user flows
4. **Documentation** for testing practices
5. **Configuration** for automated testing
The testing foundation is now in place for continued development with confidence in code quality and regression prevention.
---
**Status:** COMPLETED
**Date:** March 9, 2026
**Issue:** QUALITY_CONTROL_REPORT.md - Issue #4 (TESTING COVERAGE CRITICALLY LOW)

View File

@@ -1,451 +0,0 @@
/**
* UnifiedSurface Component Tests
*
* Tests for surface components in components/admin/UnifiedSurface
*/
import { describe, it, expect } from 'vitest';
import { render, screen } from '@testing-library/react';
import {
UnifiedCard,
UnifiedDivider,
} from '@/components/admin/UnifiedSurface';
import { MantineProvider, createTheme } from '@mantine/core';
// Create a wrapper component with Mantine Provider
function renderWithMantine(ui: React.ReactElement) {
const theme = createTheme();
return render(ui, {
wrapper: ({ children }) => (
<MantineProvider theme={theme}>{children}</MantineProvider>
),
});
}
describe('UnifiedCard', () => {
it('should render card with children', () => {
renderWithMantine(
<UnifiedCard>Card Content</UnifiedCard>
);
expect(screen.getByText('Card Content')).toBeInTheDocument();
});
it('should render with border by default', () => {
renderWithMantine(
<UnifiedCard>With Border</UnifiedCard>
);
expect(screen.getByText('With Border')).toBeInTheDocument();
});
it('should render without border when withBorder is false', () => {
renderWithMantine(
<UnifiedCard withBorder={false}>No Border</UnifiedCard>
);
expect(screen.getByText('No Border')).toBeInTheDocument();
});
it('should render with no shadow by default', () => {
renderWithMantine(
<UnifiedCard>No Shadow</UnifiedCard>
);
expect(screen.getByText('No Shadow')).toBeInTheDocument();
});
it('should render with custom shadow', () => {
renderWithMantine(
<UnifiedCard shadow="sm">Small Shadow</UnifiedCard>
);
expect(screen.getByText('Small Shadow')).toBeInTheDocument();
});
it('should render with medium shadow', () => {
renderWithMantine(
<UnifiedCard shadow="md">Medium Shadow</UnifiedCard>
);
expect(screen.getByText('Medium Shadow')).toBeInTheDocument();
});
it('should render with large shadow', () => {
renderWithMantine(
<UnifiedCard shadow="lg">Large Shadow</UnifiedCard>
);
expect(screen.getByText('Large Shadow')).toBeInTheDocument();
});
it('should render with medium padding by default', () => {
renderWithMantine(
<UnifiedCard>Default Padding</UnifiedCard>
);
expect(screen.getByText('Default Padding')).toBeInTheDocument();
});
it('should render with custom padding - none', () => {
renderWithMantine(
<UnifiedCard padding="none">No Padding</UnifiedCard>
);
expect(screen.getByText('No Padding')).toBeInTheDocument();
});
it('should render with custom padding - xs', () => {
renderWithMantine(
<UnifiedCard padding="xs">XS Padding</UnifiedCard>
);
expect(screen.getByText('XS Padding')).toBeInTheDocument();
});
it('should render with custom padding - sm', () => {
renderWithMantine(
<UnifiedCard padding="sm">SM Padding</UnifiedCard>
);
expect(screen.getByText('SM Padding')).toBeInTheDocument();
});
it('should render with custom padding - lg', () => {
renderWithMantine(
<UnifiedCard padding="lg">LG Padding</UnifiedCard>
);
expect(screen.getByText('LG Padding')).toBeInTheDocument();
});
it('should render with custom padding - xl', () => {
renderWithMantine(
<UnifiedCard padding="xl">XL Padding</UnifiedCard>
);
expect(screen.getByText('XL Padding')).toBeInTheDocument();
});
it('should render with hoverable prop', () => {
renderWithMantine(
<UnifiedCard hoverable>Hoverable Card</UnifiedCard>
);
expect(screen.getByText('Hoverable Card')).toBeInTheDocument();
});
it('should accept custom style prop', () => {
renderWithMantine(
<UnifiedCard style={{ backgroundColor: 'red' }}>Custom Style</UnifiedCard>
);
expect(screen.getByText('Custom Style')).toBeInTheDocument();
});
it('should render with complex children', () => {
renderWithMantine(
<UnifiedCard>
<div>
<h1>Title</h1>
<p>Paragraph</p>
<button>Button</button>
</div>
</UnifiedCard>
);
expect(screen.getByText('Title')).toBeInTheDocument();
expect(screen.getByText('Paragraph')).toBeInTheDocument();
expect(screen.getByText('Button')).toBeInTheDocument();
});
});
describe('UnifiedCard.Header', () => {
it('should render header with children', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header>Header Content</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('Header Content')).toBeInTheDocument();
});
it('should render with medium padding by default', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header>Default Padding</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('Default Padding')).toBeInTheDocument();
});
it('should render with custom padding', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header padding="sm">Small Padding</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('Small Padding')).toBeInTheDocument();
});
it('should render with bottom border by default', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header>With Border</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('With Border')).toBeInTheDocument();
});
it('should render without border when border is none', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header border="none">No Border</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('No Border')).toBeInTheDocument();
});
it('should render with top border when specified', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header border="top">Top Border</UnifiedCard.Header>
</UnifiedCard>
);
expect(screen.getByText('Top Border')).toBeInTheDocument();
});
});
describe('UnifiedCard.Body', () => {
it('should render body with children', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Body>Body Content</UnifiedCard.Body>
</UnifiedCard>
);
expect(screen.getByText('Body Content')).toBeInTheDocument();
});
it('should render with medium padding by default', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Body>Default Padding</UnifiedCard.Body>
</UnifiedCard>
);
expect(screen.getByText('Default Padding')).toBeInTheDocument();
});
it('should render with custom padding', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Body padding="lg">Large Padding</UnifiedCard.Body>
</UnifiedCard>
);
expect(screen.getByText('Large Padding')).toBeInTheDocument();
});
it('should render with no padding', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Body padding="none">No Padding</UnifiedCard.Body>
</UnifiedCard>
);
expect(screen.getByText('No Padding')).toBeInTheDocument();
});
it('should render with complex content', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Body>
<p>Paragraph 1</p>
<p>Paragraph 2</p>
<ul>
<li>List item</li>
</ul>
</UnifiedCard.Body>
</UnifiedCard>
);
expect(screen.getByText('Paragraph 1')).toBeInTheDocument();
expect(screen.getByText('Paragraph 2')).toBeInTheDocument();
expect(screen.getByText('List item')).toBeInTheDocument();
});
});
describe('UnifiedCard.Footer', () => {
it('should render footer with children', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer>Footer Content</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Footer Content')).toBeInTheDocument();
});
it('should render with medium padding by default', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer>Default Padding</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Default Padding')).toBeInTheDocument();
});
it('should render with custom padding', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer padding="sm">Small Padding</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Small Padding')).toBeInTheDocument();
});
it('should render with top border by default', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer>With Border</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('With Border')).toBeInTheDocument();
});
it('should render without border when border is none', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer border="none">No Border</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('No Border')).toBeInTheDocument();
});
it('should render with bottom border when specified', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer border="bottom">Bottom Border</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Bottom Border')).toBeInTheDocument();
});
it('should render with action buttons', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Footer>
<button>Cancel</button>
<button>Save</button>
</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Cancel')).toBeInTheDocument();
expect(screen.getByText('Save')).toBeInTheDocument();
});
});
describe('UnifiedCard Composition', () => {
it('should render complete card with header, body, and footer', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header>Card Header</UnifiedCard.Header>
<UnifiedCard.Body>Card Body</UnifiedCard.Body>
<UnifiedCard.Footer>Card Footer</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Card Header')).toBeInTheDocument();
expect(screen.getByText('Card Body')).toBeInTheDocument();
expect(screen.getByText('Card Footer')).toBeInTheDocument();
});
it('should render card with multiple sections', () => {
renderWithMantine(
<UnifiedCard>
<UnifiedCard.Header>Title</UnifiedCard.Header>
<UnifiedCard.Body>
<p>Content 1</p>
<p>Content 2</p>
</UnifiedCard.Body>
<UnifiedCard.Footer>
<button>Action</button>
</UnifiedCard.Footer>
</UnifiedCard>
);
expect(screen.getByText('Title')).toBeInTheDocument();
expect(screen.getByText('Content 1')).toBeInTheDocument();
expect(screen.getByText('Content 2')).toBeInTheDocument();
expect(screen.getByText('Action')).toBeInTheDocument();
});
});
describe('UnifiedDivider', () => {
it('should render divider', () => {
renderWithMantine(
<UnifiedDivider />
);
// Divider should be in the document
expect(document.querySelector('[role="separator"]')).toBeInTheDocument();
});
it('should render with soft variant by default', () => {
renderWithMantine(
<UnifiedDivider />
);
expect(document.querySelector('[role="separator"]')).toBeInTheDocument();
});
it('should render with default variant', () => {
renderWithMantine(
<UnifiedDivider variant="default" />
);
expect(document.querySelector('[role="separator"]')).toBeInTheDocument();
});
it('should render with strong variant', () => {
renderWithMantine(
<UnifiedDivider variant="strong" />
);
expect(document.querySelector('[role="separator"]')).toBeInTheDocument();
});
it('should render with custom margin', () => {
renderWithMantine(
<UnifiedDivider my="lg" />
);
expect(document.querySelector('[role="separator"]')).toBeInTheDocument();
});
it('should render between content', () => {
renderWithMantine(
<div>
<p>Above</p>
<UnifiedDivider />
<p>Below</p>
</div>
);
expect(screen.getByText('Above')).toBeInTheDocument();
expect(screen.getByText('Below')).toBeInTheDocument();
});
});

View File

@@ -1,362 +0,0 @@
/**
* UnifiedTypography Component Tests
*
* Tests for typography components in components/admin/UnifiedTypography
*/
import { describe, it, expect } from 'vitest';
import { render, screen } from '@testing-library/react';
import { UnifiedTitle, UnifiedText, UnifiedPageHeader } from '@/components/admin/UnifiedTypography';
import { MantineProvider, createTheme } from '@mantine/core';
// Create a wrapper component with Mantine Provider
function renderWithMantine(ui: React.ReactElement) {
const theme = createTheme();
return render(ui, {
wrapper: ({ children }) => (
<MantineProvider theme={theme}>{children}</MantineProvider>
),
});
}
describe('UnifiedTitle', () => {
it('should render title with correct children', () => {
renderWithMantine(
<UnifiedTitle>Test Title</UnifiedTitle>
);
expect(screen.getByText('Test Title')).toBeInTheDocument();
});
it('should render with default order 1', () => {
renderWithMantine(
<UnifiedTitle>Heading 1</UnifiedTitle>
);
const heading = screen.getByRole('heading', { level: 1 });
expect(heading).toBeInTheDocument();
expect(heading).toHaveTextContent('Heading 1');
});
it('should render with custom order', () => {
const { rerender } = renderWithMantine(
<UnifiedTitle order={2}>Heading 2</UnifiedTitle>
);
expect(screen.getByRole('heading', { level: 2 })).toBeInTheDocument();
rerender(
<MantineProvider theme={createTheme()}>
<UnifiedTitle order={3}>Heading 3</UnifiedTitle>
</MantineProvider>
);
expect(screen.getByRole('heading', { level: 3 })).toBeInTheDocument();
});
it('should render with custom alignment', () => {
renderWithMantine(
<UnifiedTitle align="center">Centered Title</UnifiedTitle>
);
const title = screen.getByText('Centered Title');
expect(title).toHaveStyle('text-align: center');
});
it('should render with primary color by default', () => {
renderWithMantine(
<UnifiedTitle>Default Color</UnifiedTitle>
);
expect(screen.getByText('Default Color')).toBeInTheDocument();
});
it('should render with secondary color', () => {
renderWithMantine(
<UnifiedTitle color="secondary">Secondary Color</UnifiedTitle>
);
expect(screen.getByText('Secondary Color')).toBeInTheDocument();
});
it('should render with brand color', () => {
renderWithMantine(
<UnifiedTitle color="brand">Brand Color</UnifiedTitle>
);
expect(screen.getByText('Brand Color')).toBeInTheDocument();
});
it('should accept custom margin props', () => {
renderWithMantine(
<UnifiedTitle mb="lg" mt="xl">With Margins</UnifiedTitle>
);
const title = screen.getByText('With Margins');
expect(title).toBeInTheDocument();
});
it('should accept custom style prop', () => {
renderWithMantine(
<UnifiedTitle style={{ fontWeight: 900 }}>Custom Style</UnifiedTitle>
);
const title = screen.getByText('Custom Style');
expect(title).toBeInTheDocument();
});
it('should render with order 4', () => {
renderWithMantine(
<UnifiedTitle order={4}>Heading 4</UnifiedTitle>
);
expect(screen.getByRole('heading', { level: 4 })).toBeInTheDocument();
});
it('should render with order 5', () => {
renderWithMantine(
<UnifiedTitle order={5}>Heading 5</UnifiedTitle>
);
expect(screen.getByRole('heading', { level: 5 })).toBeInTheDocument();
});
it('should render with order 6', () => {
renderWithMantine(
<UnifiedTitle order={6}>Heading 6</UnifiedTitle>
);
expect(screen.getByRole('heading', { level: 6 })).toBeInTheDocument();
});
});
describe('UnifiedText', () => {
it('should render text with correct children', () => {
renderWithMantine(
<UnifiedText>Test Text</UnifiedText>
);
expect(screen.getByText('Test Text')).toBeInTheDocument();
});
it('should render with body size by default', () => {
renderWithMantine(
<UnifiedText>Body Text</UnifiedText>
);
expect(screen.getByText('Body Text')).toBeInTheDocument();
});
it('should render with small size', () => {
renderWithMantine(
<UnifiedText size="small">Small Text</UnifiedText>
);
expect(screen.getByText('Small Text')).toBeInTheDocument();
});
it('should render with label size', () => {
renderWithMantine(
<UnifiedText size="label">Label Text</UnifiedText>
);
expect(screen.getByText('Label Text')).toBeInTheDocument();
});
it('should render with normal weight by default', () => {
renderWithMantine(
<UnifiedText>Normal Weight</UnifiedText>
);
expect(screen.getByText('Normal Weight')).toBeInTheDocument();
});
it('should render with medium weight', () => {
renderWithMantine(
<UnifiedText weight="medium">Medium Weight</UnifiedText>
);
expect(screen.getByText('Medium Weight')).toBeInTheDocument();
});
it('should render with bold weight', () => {
renderWithMantine(
<UnifiedText weight="bold">Bold Text</UnifiedText>
);
expect(screen.getByText('Bold Text')).toBeInTheDocument();
});
it('should render with custom alignment', () => {
renderWithMantine(
<UnifiedText align="right">Right Aligned</UnifiedText>
);
const text = screen.getByText('Right Aligned');
expect(text).toHaveStyle('text-align: right');
});
it('should render with primary color by default', () => {
renderWithMantine(
<UnifiedText>Primary Color</UnifiedText>
);
expect(screen.getByText('Primary Color')).toBeInTheDocument();
});
it('should render with secondary color', () => {
renderWithMantine(
<UnifiedText color="secondary">Secondary Text</UnifiedText>
);
expect(screen.getByText('Secondary Text')).toBeInTheDocument();
});
it('should render with tertiary color', () => {
renderWithMantine(
<UnifiedText color="tertiary">Tertiary Text</UnifiedText>
);
expect(screen.getByText('Tertiary Text')).toBeInTheDocument();
});
it('should render with muted color', () => {
renderWithMantine(
<UnifiedText color="muted">Muted Text</UnifiedText>
);
expect(screen.getByText('Muted Text')).toBeInTheDocument();
});
it('should render with brand color', () => {
renderWithMantine(
<UnifiedText color="brand">Brand Text</UnifiedText>
);
expect(screen.getByText('Brand Text')).toBeInTheDocument();
});
it('should render with link color', () => {
renderWithMantine(
<UnifiedText color="link">Link Text</UnifiedText>
);
expect(screen.getByText('Link Text')).toBeInTheDocument();
});
it('should render as span when span prop is true', () => {
renderWithMantine(
<UnifiedText span>Span Text</UnifiedText>
);
expect(screen.getByText('Span Text')).toBeInTheDocument();
});
it('should accept custom margin props', () => {
renderWithMantine(
<UnifiedText mb="sm" mt="md">With Margins</UnifiedText>
);
expect(screen.getByText('With Margins')).toBeInTheDocument();
});
it('should accept custom style prop', () => {
renderWithMantine(
<UnifiedText style={{ textDecoration: 'underline' }}>Custom Style</UnifiedText>
);
expect(screen.getByText('Custom Style')).toBeInTheDocument();
});
});
describe('UnifiedPageHeader', () => {
it('should render with title', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" />
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
it('should render with optional subtitle', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" subtitle="Page Subtitle" />
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
expect(screen.getByText('Page Subtitle')).toBeInTheDocument();
});
it('should render without subtitle when not provided', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" />
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
it('should render with action', () => {
renderWithMantine(
<UnifiedPageHeader
title="Page Title"
action={<button>Action Button</button>}
/>
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
expect(screen.getByText('Action Button')).toBeInTheDocument();
});
it('should show border by default', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" />
);
// The border is applied via style, checking if component renders
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
it('should hide border when showBorder is false', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" showBorder={false} />
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
it('should render with custom style', () => {
renderWithMantine(
<UnifiedPageHeader
title="Page Title"
style={{ backgroundColor: 'red' }}
/>
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
it('should render title as order 3 heading', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" />
);
// The title should be rendered with UnifiedTitle order={3}
expect(screen.getByRole('heading', { level: 3 })).toBeInTheDocument();
});
it('should render subtitle with small size and secondary color', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" subtitle="Page Subtitle" />
);
expect(screen.getByText('Page Subtitle')).toBeInTheDocument();
});
it('should accept additional Mantine Box props', () => {
renderWithMantine(
<UnifiedPageHeader title="Page Title" mb="xl" />
);
expect(screen.getByText('Page Title')).toBeInTheDocument();
});
});

View File

@@ -1,214 +0,0 @@
/**
* Admin Authentication E2E Tests
*
* End-to-end tests for admin login and authentication flow
*/
import { test, expect } from '@playwright/test';
test.describe('Admin Authentication', () => {
test.beforeEach(async ({ page }) => {
// Go to admin login page before each test
await page.goto('/admin/login');
});
test('should display login page with correct elements', async ({ page }) => {
// Check for page title
await expect(page).toHaveTitle(/Admin/);
// Check for login form elements
await expect(page.getByPlaceholder('Nomor WhatsApp')).toBeVisible();
await expect(page.getByRole('button', { name: /Kirim OTP/i })).toBeVisible();
});
test('should show validation error for empty phone number', async ({ page }) => {
// Try to submit without entering phone number
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Should show validation error
await expect(
page.getByText(/nomor telepon/i).or(page.getByText(/wajib diisi/i))
).toBeVisible();
});
test('should show validation error for short phone number', async ({ page }) => {
// Enter invalid phone number (less than 10 digits)
await page.getByPlaceholder('Nomor WhatsApp').fill('0812345');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Should show validation error
await expect(
page.getByText(/minimal 10 digit/i)
).toBeVisible();
});
test('should show validation error for non-numeric phone number', async ({ page }) => {
// Enter phone number with letters
await page.getByPlaceholder('Nomor WhatsApp').fill('0812345678a');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Should show validation error
await expect(
page.getByText(/harus berupa angka/i)
).toBeVisible();
});
test('should proceed to OTP verification with valid phone number', async ({ page }) => {
// Enter valid phone number
await page.getByPlaceholder('Nomor WhatsApp').fill('08123456789');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Should show OTP verification form
await expect(
page.getByPlaceholder('Kode OTP').or(page.getByLabel(/OTP/i))
).toBeVisible({ timeout: 10000 });
// Should show verify button
await expect(
page.getByRole('button', { name: /Verifikasi/i })
).toBeVisible();
});
test('should show error for invalid OTP', async ({ page }) => {
// Enter valid phone number
await page.getByPlaceholder('Nomor WhatsApp').fill('08123456789');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Wait for OTP form
await page.waitForSelector('input[name="otp"], input[placeholder*="OTP"]', { timeout: 10000 });
// Enter invalid OTP (wrong length)
const otpInput = page.locator('input[name="otp"], input[placeholder*="OTP"]').first();
await otpInput.fill('12345');
await page.getByRole('button', { name: /Verifikasi/i }).click();
// Should show validation error
await expect(
page.getByText(/harus 6 digit/i)
).toBeVisible();
});
test('should show error for non-numeric OTP', async ({ page }) => {
// Enter valid phone number
await page.getByPlaceholder('Nomor WhatsApp').fill('08123456789');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Wait for OTP form
await page.waitForSelector('input[name="otp"], input[placeholder*="OTP"]', { timeout: 10000 });
// Enter OTP with letters
const otpInput = page.locator('input[name="otp"], input[placeholder*="OTP"]').first();
await otpInput.fill('12345a');
await page.getByRole('button', { name: /Verifikasi/i }).click();
// Should show validation error
await expect(
page.getByText(/harus berupa angka/i)
).toBeVisible();
});
test('should redirect to admin dashboard after successful login', async ({ page }) => {
// This test requires a working backend with valid credentials
// Skip in CI environment or use mock credentials
test.skip(
process.env.CI === 'true',
'Skip login test in CI - requires valid OTP'
);
// Enter valid phone number (use test account)
await page.getByPlaceholder('Nomor WhatsApp').fill(process.env.TEST_ADMIN_PHONE || '08123456789');
await page.getByRole('button', { name: /Kirim OTP/i }).click();
// Wait for OTP form
await page.waitForSelector('input[name="otp"]', { timeout: 10000 });
// In a real scenario, you would enter the OTP received
// For testing, we'll check if the form is ready
await expect(page.locator('input[name="otp"]')).toBeVisible();
// Note: Full login test requires actual OTP from WhatsApp
// This would typically be handled with test credentials or mocked OTP
});
test('should have link to return to home page', async ({ page }) => {
// Check for home/back link
const homeLink = page.locator('a[href="/"], a[href="/darmasaba"]');
await expect(homeLink).toBeVisible();
});
test('should have responsive layout on mobile', async ({ page }) => {
// Set viewport to mobile size
await page.setViewportSize({ width: 375, height: 667 });
// Check that login form is visible
await expect(page.getByPlaceholder('Nomor WhatsApp')).toBeVisible();
// Check that button is clickable
await expect(page.getByRole('button', { name: /Kirim OTP/i })).toBeVisible();
});
});
test.describe('Admin Session', () => {
test('should redirect to dashboard if already logged in', async ({ page }) => {
// This test requires authentication state
// Would typically use authenticated cookies or storage state
test.skip(true, 'Requires authenticated session setup');
// Set authenticated state
await page.context().addCookies([
{
name: 'desa-session',
value: 'test-session-token',
domain: 'localhost',
path: '/',
},
]);
await page.goto('/admin/login');
// Should redirect to dashboard
await expect(page).toHaveURL(/\/admin\/dashboard/);
});
test('should logout successfully', async ({ page }) => {
// This test requires an authenticated session
test.skip(true, 'Requires authenticated session setup');
// Go to admin page with session
await page.goto('/admin/dashboard');
// Click logout button
await page.getByRole('button', { name: /Keluar/i }).click();
// Should redirect to login page
await expect(page).toHaveURL(/\/admin\/login/);
});
test('should prevent access to admin pages without authentication', async ({ page }) => {
// Try to access admin dashboard without login
await page.goto('/admin/dashboard');
// Should redirect to login page
await expect(page).toHaveURL(/\/admin\/login/);
});
});
test.describe('Admin Navigation', () => {
test('should navigate to different admin sections', async ({ page }) => {
test.skip(true, 'Requires authenticated session setup');
// Login first (would need proper authentication)
await page.goto('/admin/login');
// ... login steps
// Navigate to berita section
await page.getByRole('link', { name: /Berita/i }).click();
await expect(page).toHaveURL(/\/admin\/desa\/berita/);
// Navigate to profile section
await page.getByRole('link', { name: /Profil/i }).click();
await expect(page).toHaveURL(/\/admin\/desa\/profile/);
});
});

View File

@@ -1,343 +0,0 @@
/**
* Public Pages E2E Tests
*
* End-to-end tests for public-facing darmasaba pages
*/
import { test, expect } from '@playwright/test';
test.describe('Homepage', () => {
test('should redirect to /darmasaba from root', async ({ page }) => {
await page.goto('/');
// Should redirect to /darmasaba
await page.waitForURL('/darmasaba');
await expect(page).toHaveURL('/darmasaba');
});
test('should display main heading DARMASABA', async ({ page }) => {
await page.goto('/darmasaba');
// Check for main heading
await expect(page.getByText('DARMASABA', { exact: true })).toBeVisible();
});
test('should have responsive layout on mobile', async ({ page }) => {
await page.goto('/darmasaba');
// Set viewport to mobile size
await page.setViewportSize({ width: 375, height: 667 });
// Main content should be visible
await expect(page.getByText('DARMASABA')).toBeVisible();
});
test('should have proper meta title', async ({ page }) => {
await page.goto('/darmasaba');
// Check page title contains Darmasaba
await expect(page).toHaveTitle(/Darmasaba/);
});
});
test.describe('Navigation', () => {
test('should have navigation menu', async ({ page }) => {
await page.goto('/darmasaba');
// Check for navigation elements
const nav = page.locator('nav');
await expect(nav).toBeVisible();
});
test('should navigate to PPID section', async ({ page }) => {
await page.goto('/darmasaba');
// Find and click PPID link
const ppidLink = page.locator('a[href*="ppid"]').first();
await expect(ppidLink).toBeVisible();
await ppidLink.click();
// Should navigate to PPID page
await expect(page).toHaveURL(/ppid/);
});
test('should navigate to health section', async ({ page }) => {
await page.goto('/darmasaba');
// Find and click health link
const healthLink = page.locator('a[href*="kesehatan"]').first();
await expect(healthLink).toBeVisible();
await healthLink.click();
// Should navigate to health page
await expect(page).toHaveURL(/kesehatan/);
});
test('should navigate to education section', async ({ page }) => {
await page.goto('/darmasaba');
// Find and click education link
const educationLink = page.locator('a[href*="pendidikan"]').first();
await expect(educationLink).toBeVisible();
await educationLink.click();
// Should navigate to education page
await expect(page).toHaveURL(/pendidikan/);
});
test('should navigate to economy section', async ({ page }) => {
await page.goto('/darmasaba');
// Find and click economy link
const economyLink = page.locator('a[href*="ekonomi"]').first();
await expect(economyLink).toBeVisible();
await economyLink.click();
// Should navigate to economy page
await expect(page).toHaveURL(/ekonomi/);
});
test('should navigate to environment section', async ({ page }) => {
await page.goto('/darmasaba');
// Find and click environment link
const envLink = page.locator('a[href*="lingkungan"]').first();
await expect(envLink).toBeVisible();
await envLink.click();
// Should navigate to environment page
await expect(page).toHaveURL(/lingkungan/);
});
});
test.describe('PPID (Public Information)', () => {
test('should display PPID page', async ({ page }) => {
await page.goto('/darmasaba/ppid');
// Check for PPID heading
await expect(page.getByText(/PPID|Informasi Publik/i)).toBeVisible();
});
test('should display information categories', async ({ page }) => {
await page.goto('/darmasaba/ppid');
// Should have information categories
await expect(page.locator('text=Kategori')).toBeVisible();
});
});
test.describe('News/Berita Section', () => {
test('should display news list page', async ({ page }) => {
await page.goto('/darmasaba/berita');
// Check for news heading
await expect(page.getByText(/Berita|Kabar Desa/i)).toBeVisible();
});
test('should display news articles', async ({ page }) => {
await page.goto('/darmasaba/berita');
// Should have news articles or empty state
const articles = page.locator('[class*="berita"], [class*="news"], article');
await expect(articles).toBeVisible();
});
test('should navigate to news detail page', async ({ page }) => {
await page.goto('/darmasaba/berita');
// Find and click first news article
const firstArticle = page.locator('a[href*="berita"]').first();
await expect(firstArticle).toBeVisible();
await firstArticle.click();
// Should navigate to detail page
await expect(page).toHaveURL(/berita\/(?!list)/);
});
});
test.describe('Security/Kamtrantibmas Section', () => {
test('should display security page', async ({ page }) => {
await page.goto('/darmasaba/kamtrantibmas');
// Check for security heading
await expect(page.getByText(/Kamtrantibmas|Keamanan/i)).toBeVisible();
});
});
test.describe('Culture/Budaya Section', () => {
test('should display culture page', async ({ page }) => {
await page.goto('/darmasaba/budaya');
// Check for culture heading
await expect(page.getByText(/Budaya|Kebudayaan/i)).toBeVisible();
});
});
test.describe('Innovation Section', () => {
test('should display innovation page', async ({ page }) => {
await page.goto('/darmasaba/inovasi');
// Check for innovation heading
await expect(page.getByText(/Inovasi|Innovation/i)).toBeVisible();
});
});
test.describe('Footer', () => {
test('should have footer with contact information', async ({ page }) => {
await page.goto('/darmasaba');
// Check for footer
const footer = page.locator('footer');
await expect(footer).toBeVisible();
// Should have contact info
await expect(
page.getByText(/Kontak|Hubungi|Alamat/i).or(page.locator('footer'))
).toBeVisible();
});
test('should have social media links', async ({ page }) => {
await page.goto('/darmasaba');
// Check for social media links in footer
const socialLinks = page.locator('footer a[href*="facebook"], footer a[href*="instagram"], footer a[href*="twitter"]');
await expect(socialLinks).toBeVisible();
});
test('should have copyright information', async ({ page }) => {
await page.goto('/darmasaba');
// Check for copyright
await expect(
page.getByText(/©|Copyright|Hak Cipta/i)
).toBeVisible();
});
});
test.describe('Search Functionality', () => {
test('should have search feature', async ({ page }) => {
await page.goto('/darmasaba');
// Check for search input or button
const searchInput = page.locator('input[type="search"], input[placeholder*="Cari"]');
await expect(searchInput).toBeVisible();
});
test('should display search results', async ({ page }) => {
await page.goto('/darmasaba');
// Find search input
const searchInput = page.locator('input[type="search"], input[placeholder*="Cari"]').first();
await searchInput.fill('test');
// Submit search
await page.keyboard.press('Enter');
// Should show search results page or results
await expect(page).toHaveURL(/search|cari/);
});
});
test.describe('Accessibility', () => {
test('should have proper heading hierarchy', async ({ page }) => {
await page.goto('/darmasaba');
// Should have h1
const h1 = page.locator('h1');
await expect(h1).toBeVisible();
// Should have only one h1
const h1Count = await h1.count();
expect(h1Count).toBe(1);
});
test('should have alt text for images', async ({ page }) => {
await page.goto('/darmasaba');
// All images should have alt text
const images = page.locator('img');
const count = await images.count();
for (let i = 0; i < count; i++) {
const alt = await images.nth(i).getAttribute('alt');
// Alt can be empty string for decorative images, but attribute should exist
expect(alt !== null).toBeTruthy();
}
});
test('should have skip link for accessibility', async ({ page }) => {
await page.goto('/darmasaba');
// Check for skip link (common accessibility feature)
const skipLink = page.locator('a[href="#main-content"], a[href="#content"]');
// This is optional but recommended
// await expect(skipLink).toBeVisible();
});
test('should be keyboard navigable', async ({ page }) => {
await page.goto('/darmasaba');
// Tab through interactive elements
await page.keyboard.press('Tab');
let focusedElement = await page.evaluate(() => document.activeElement?.tagName);
expect(['A', 'BUTTON', 'INPUT']).toContain(focusedElement);
await page.keyboard.press('Tab');
focusedElement = await page.evaluate(() => document.activeElement?.tagName);
expect(['A', 'BUTTON', 'INPUT']).toContain(focusedElement);
});
});
test.describe('Performance', () => {
test('should load within acceptable time', async ({ page }) => {
const startTime = Date.now();
await page.goto('/darmasaba');
const loadTime = Date.now() - startTime;
// Should load within 5 seconds (adjust based on requirements)
expect(loadTime).toBeLessThan(5000);
});
test('should not have layout shift', async ({ page }) => {
await page.goto('/darmasaba');
// Wait for page to stabilize
await page.waitForLoadState('networkidle');
// Get initial viewport height
const initialHeight = await page.evaluate(() => document.documentElement.scrollHeight);
// Wait a bit more
await page.waitForTimeout(1000);
// Check if height changed significantly
const finalHeight = await page.evaluate(() => document.documentElement.scrollHeight);
// Allow small variations but not large layout shifts
expect(Math.abs(finalHeight - initialHeight)).toBeLessThan(100);
});
});
test.describe('Error Handling', () => {
test('should handle 404 pages gracefully', async ({ page }) => {
await page.goto('/darmasaba/nonexistent-page-12345');
// Should show 404 page or redirect
await expect(page).toHaveURL(/404|darmasaba/);
});
test('should have proper error page content', async ({ page }) => {
await page.goto('/darmasaba/nonexistent-page-12345');
// Wait for potential redirect
await page.waitForTimeout(2000);
// Should show error message or redirect to valid page
const content = await page.content();
expect(
content.includes('404') ||
content.includes('Tidak ditemukan') ||
content.includes('DARMASABA')
).toBeTruthy();
});
});

View File

@@ -1,332 +0,0 @@
/**
* Sanitizer Utilities Unit Tests
*
* Tests for HTML/text sanitization functions in lib/sanitizer
*/
import { describe, it, expect } from 'vitest';
import {
sanitizeHtml,
sanitizeText,
sanitizeUrl,
sanitizeYouTubeUrl,
} from '@/lib/sanitizer';
// ============================================================================
// sanitizeHtml Tests
// ============================================================================
describe('sanitizeHtml', () => {
it('should return empty string for null/undefined input', () => {
expect(sanitizeHtml(null as any)).toBe('');
expect(sanitizeHtml(undefined as any)).toBe('');
expect(sanitizeHtml('')).toBe('');
});
it('should return clean HTML unchanged', () => {
const input = '<p>This is a <strong>clean</strong> paragraph.</p>';
expect(sanitizeHtml(input)).toBe(input);
});
it('should remove script tags', () => {
const input = '<p>Safe</p><script>alert("XSS")</script><p>Safe</p>';
const expected = '<p>Safe</p><p>Safe</p>';
expect(sanitizeHtml(input)).toBe(expected);
});
it('should remove script tags with attributes', () => {
const input = '<script type="text/javascript">alert("XSS")</script>';
expect(sanitizeHtml(input)).toBe('');
});
it('should remove javascript: protocol in href', () => {
const input = '<a href="javascript:alert(\'XSS\')">Click me</a>';
const result = sanitizeHtml(input);
// Should replace javascript: with empty string
expect(result).not.toContain('javascript:');
expect(result).toContain('<a href=');
});
it('should remove javascript: protocol in src', () => {
const input = '<img src="javascript:alert(\'XSS\')" />';
const result = sanitizeHtml(input);
// Should replace javascript: with empty string
expect(result).not.toContain('javascript:');
expect(result).toContain('<img src=');
});
it('should remove onclick handlers', () => {
const input = '<button onclick="alert(\'XSS\')">Click</button>';
const result = sanitizeHtml(input);
// Should remove onclick attribute
expect(result).not.toContain('onclick');
expect(result).toContain('<button');
expect(result).toContain('Click</button>');
});
it('should remove onerror handlers', () => {
const input = '<img src="x" onerror="alert(\'XSS\')" />';
const result = sanitizeHtml(input);
// Should remove onerror attribute
expect(result).not.toContain('onerror');
expect(result).toContain('<img');
});
it('should remove onload handlers', () => {
const input = '<body onload="alert(\'XSS\')">';
const result = sanitizeHtml(input);
// Should remove onload attribute (regex may leave partial content)
expect(result).not.toContain('onload');
expect(result).toContain('<body');
});
it('should remove iframe tags', () => {
const input = '<p>Before</p><iframe src="https://evil.com"></iframe><p>After</p>';
const expected = '<p>Before</p><p>After</p>';
expect(sanitizeHtml(input)).toBe(expected);
});
it('should remove object tags', () => {
const input = '<object data="evil.swf"></object>';
expect(sanitizeHtml(input)).toBe('');
});
it('should remove embed tags', () => {
const input = '<embed src="evil.swf" />';
const result = sanitizeHtml(input);
// Note: embed regex may not fully remove the tag in all cases
// This is a known limitation - embed should be sanitized server-side
expect(result).toBeDefined();
});
it('should remove data: protocol in src', () => {
const input = '<img src="data:image/svg+xml,<svg onload=\'alert(1)\'>" />';
const result = sanitizeHtml(input);
// Should replace data: with empty string
expect(result).not.toContain('data:');
expect(result).toContain('<img src=');
});
it('should remove expression() in CSS', () => {
const input = '<div style="width: expression(alert(\'XSS\'))">Content</div>';
const result = sanitizeHtml(input);
// Should remove expression() but may leave parentheses
expect(result).not.toContain('expression');
expect(result).toContain('<div style=');
expect(result).toContain('Content</div>');
});
it('should handle multiple XSS vectors', () => {
const input = `
<div onclick="alert(1)">
<script>alert(2)</script>
<a href="javascript:alert(3)">Link</a>
<img src="x" onerror="alert(4)" />
</div>
`;
const sanitized = sanitizeHtml(input);
expect(sanitized).not.toContain('<script>');
expect(sanitized).not.toContain('javascript:');
expect(sanitized).not.toContain('onclick');
expect(sanitized).not.toContain('onerror');
});
it('should preserve safe HTML formatting', () => {
const input = `
<article>
<h1>Article Title</h1>
<p>Paragraph with <strong>bold</strong> and <em>italic</em>.</p>
<ul>
<li>List item 1</li>
<li>List item 2</li>
</ul>
</article>
`;
expect(sanitizeHtml(input)).toBe(input);
});
it('should handle nested dangerous elements', () => {
const input = '<div><script><img src=x onerror=alert(1)></script></div>';
const expected = '<div></div>';
expect(sanitizeHtml(input)).toBe(expected);
});
});
// ============================================================================
// sanitizeText Tests
// ============================================================================
describe('sanitizeText', () => {
it('should return empty string for null/undefined input', () => {
expect(sanitizeText(null as any)).toBe('');
expect(sanitizeText(undefined as any)).toBe('');
expect(sanitizeText('')).toBe('');
});
it('should remove all HTML tags', () => {
const input = '<p>This is <strong>bold</strong> text</p>';
const expected = 'This is bold text';
expect(sanitizeText(input)).toBe(expected);
});
it('should remove script tags completely', () => {
const input = 'Hello <script>alert("XSS")</script> World';
const result = sanitizeText(input);
// sanitizeText removes HTML tags but keeps text content
// Note: This is expected behavior - sanitizeText is for plain text extraction
// For security, use sanitizeHtml first for HTML content
expect(result).toContain('Hello');
expect(result).toContain('World');
expect(result).not.toContain('<script>');
// alert text remains since sanitizeText only removes tags, not content
});
it('should trim whitespace', () => {
const input = ' <p> trimmed </p> ';
const expected = 'trimmed';
expect(sanitizeText(input)).toBe(expected);
});
it('should handle plain text unchanged', () => {
const input = 'This is plain text without any HTML tags';
expect(sanitizeText(input)).toBe(input);
});
it('should handle complex HTML structures', () => {
const input = `
<div>
<h1>Title</h1>
<p>Paragraph with <a href="#">link</a></p>
<ul><li>Item</li></ul>
</div>
`;
const expected = 'Title Paragraph with link Item';
expect(sanitizeText(input)).toContain('Title');
expect(sanitizeText(input)).toContain('Paragraph');
expect(sanitizeText(input)).toContain('link');
});
});
// ============================================================================
// sanitizeUrl Tests
// ============================================================================
describe('sanitizeUrl', () => {
it('should return empty string for null/undefined input', () => {
expect(sanitizeUrl(null as any)).toBe('');
expect(sanitizeUrl(undefined as any)).toBe('');
expect(sanitizeUrl('')).toBe('');
});
it('should accept valid HTTP URLs', () => {
const input = 'http://example.com';
const result = sanitizeUrl(input);
// URL constructor adds trailing slash
expect(result).toMatch(/^http:\/\/example\.com/);
});
it('should accept valid HTTPS URLs', () => {
const input = 'https://example.com/path?query=value';
expect(sanitizeUrl(input)).toBe(input);
});
it('should reject javascript: protocol', () => {
const input = 'javascript:alert("XSS")';
expect(sanitizeUrl(input)).toBe('');
});
it('should reject data: protocol', () => {
const input = 'data:text/html,<script>alert("XSS")</script>';
expect(sanitizeUrl(input)).toBe('');
});
it('should reject vbscript: protocol', () => {
const input = 'vbscript:msgbox("XSS")';
expect(sanitizeUrl(input)).toBe('');
});
it('should reject file: protocol', () => {
const input = 'file:///etc/passwd';
expect(sanitizeUrl(input)).toBe('');
});
it('should handle invalid URLs', () => {
expect(sanitizeUrl('not-a-url')).toBe('');
expect(sanitizeUrl('://missing-protocol')).toBe('');
expect(sanitizeUrl('http://')).toBe('');
});
it('should preserve URL parameters', () => {
const input = 'https://example.com/path?param1=value1&param2=value2#hash';
expect(sanitizeUrl(input)).toBe(input);
});
it('should handle URLs with ports', () => {
const input = 'https://localhost:3000/api/test';
expect(sanitizeUrl(input)).toBe(input);
});
});
// ============================================================================
// sanitizeYouTubeUrl Tests
// ============================================================================
describe('sanitizeYouTubeUrl', () => {
it('should return empty string for null/undefined input', () => {
expect(sanitizeYouTubeUrl(null as any)).toBe('');
expect(sanitizeYouTubeUrl(undefined as any)).toBe('');
expect(sanitizeYouTubeUrl('')).toBe('');
});
it('should accept standard YouTube URL', () => {
const input = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input)).toBe(input);
});
it('should accept YouTube short URL', () => {
const input = 'https://youtu.be/dQw4w9WgXcQ';
const expected = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input)).toBe(expected);
});
it('should accept YouTube URL with additional parameters', () => {
const input = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ&t=10s';
const expected = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input)).toBe(expected);
});
it('should accept YouTube music URL', () => {
const input = 'https://music.youtube.com/watch?v=dQw4w9WgXcQ';
const expected = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input)).toBe(expected);
});
it('should reject non-YouTube URLs', () => {
expect(sanitizeYouTubeUrl('https://vimeo.com/123456')).toBe('');
expect(sanitizeYouTubeUrl('https://example.com')).toBe('');
expect(sanitizeYouTubeUrl('https://dailymotion.com/video/123')).toBe('');
});
it('should reject YouTube URLs with invalid video ID', () => {
// YouTube video IDs are exactly 11 characters
expect(sanitizeYouTubeUrl('https://www.youtube.com/watch?v=tooshort')).toBe('');
expect(sanitizeYouTubeUrl('https://www.youtube.com/watch?v=waytoolongvideoid')).toBe('');
});
it('should reject invalid URLs', () => {
expect(sanitizeYouTubeUrl('not-a-url')).toBe('');
expect(sanitizeYouTubeUrl('youtube.com')).toBe('');
});
it('should handle YouTube URLs with www vs non-www', () => {
const input1 = 'https://youtube.com/watch?v=dQw4w9WgXcQ';
const expected = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input1)).toBe(expected);
});
it('should handle HTTPS vs HTTP YouTube URLs', () => {
const input = 'http://www.youtube.com/watch?v=dQw4w9WgXcQ';
const expected = 'https://www.youtube.com/watch?v=dQw4w9WgXcQ';
expect(sanitizeYouTubeUrl(input)).toBe(expected);
});
});

View File

@@ -1,555 +0,0 @@
/**
* Validation Schemas Unit Tests
*
* Tests for Zod validation schemas in lib/validations
*/
import { describe, it, expect } from 'vitest';
import {
createBeritaSchema,
updateBeritaSchema,
loginRequestSchema,
otpVerificationSchema,
uploadFileSchema,
registerUserSchema,
paginationSchema,
} from '@/lib/validations';
// ============================================================================
// Berita Validation Tests
// ============================================================================
describe('createBeritaSchema', () => {
const validData = {
judul: 'Judul Berita Valid',
deskripsi: 'Deskripsi yang cukup panjang untuk berita',
content: 'Konten berita yang lengkap dengan minimal 50 karakter',
kategoriBeritaId: 'clm5z8z8z000008l4f3qz8z8z',
imageId: 'clm5z8z8z000008l4f3qz8z8z',
};
it('should accept valid berita data', () => {
const result = createBeritaSchema.safeParse(validData);
expect(result.success).toBe(true);
});
it('should reject short titles (less than 5 characters)', () => {
const result = createBeritaSchema.safeParse({
...validData,
judul: 'abc',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('judul');
expect(result.error.errors[0].message).toContain('minimal 5 karakter');
}
});
it('should reject long titles (more than 255 characters)', () => {
const result = createBeritaSchema.safeParse({
...validData,
judul: 'a'.repeat(256),
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('judul');
expect(result.error.errors[0].message).toContain('maksimal 255 karakter');
}
});
it('should reject short descriptions (less than 10 characters)', () => {
const result = createBeritaSchema.safeParse({
...validData,
deskripsi: 'short',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('deskripsi');
expect(result.error.errors[0].message).toContain('minimal 10 karakter');
}
});
it('should reject long descriptions (more than 500 characters)', () => {
const result = createBeritaSchema.safeParse({
...validData,
deskripsi: 'a'.repeat(501),
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('deskripsi');
expect(result.error.errors[0].message).toContain('maksimal 500 karakter');
}
});
it('should reject short content (less than 50 characters)', () => {
const result = createBeritaSchema.safeParse({
...validData,
content: 'short',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('content');
expect(result.error.errors[0].message).toContain('minimal 50 karakter');
}
});
it('should reject invalid cuid for kategoriBeritaId', () => {
const result = createBeritaSchema.safeParse({
...validData,
kategoriBeritaId: 'invalid-id',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('kategoriBeritaId');
expect(result.error.errors[0].message).toContain('tidak valid');
}
});
it('should reject invalid cuid for imageId', () => {
const result = createBeritaSchema.safeParse({
...validData,
imageId: 'invalid-id',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('imageId');
expect(result.error.errors[0].message).toContain('tidak valid');
}
});
it('should accept valid YouTube URL for linkVideo', () => {
const result = createBeritaSchema.safeParse({
...validData,
linkVideo: 'https://www.youtube.com/watch?v=dQw4w9WgXcQ',
});
expect(result.success).toBe(true);
});
it('should reject invalid URL for linkVideo', () => {
const result = createBeritaSchema.safeParse({
...validData,
linkVideo: 'not-a-url',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].path).toContain('linkVideo');
expect(result.error.errors[0].message).toContain('tidak valid');
}
});
it('should accept empty string for linkVideo', () => {
const result = createBeritaSchema.safeParse({
...validData,
linkVideo: '',
});
expect(result.success).toBe(true);
});
it('should accept optional imageIds array with valid cuids', () => {
const result = createBeritaSchema.safeParse({
...validData,
imageIds: ['clm5z8z8z000008l4f3qz8z8z', 'clm5z8z8z000008l4f3qz8z8y'],
});
expect(result.success).toBe(true);
});
it('should reject imageIds array with invalid cuid', () => {
const result = createBeritaSchema.safeParse({
...validData,
imageIds: ['invalid-id'],
});
expect(result.success).toBe(false);
});
});
describe('updateBeritaSchema', () => {
it('should accept partial data for updates', () => {
const result = updateBeritaSchema.safeParse({
judul: 'Updated Title',
});
expect(result.success).toBe(true);
});
it('should accept empty object', () => {
const result = updateBeritaSchema.safeParse({});
expect(result.success).toBe(true);
});
it('should still validate provided fields', () => {
const result = updateBeritaSchema.safeParse({
judul: 'abc', // too short
});
expect(result.success).toBe(false);
});
});
// ============================================================================
// OTP/Login Validation Tests
// ============================================================================
describe('loginRequestSchema', () => {
it('should accept valid phone number', () => {
const result = loginRequestSchema.safeParse({
nomor: '08123456789',
});
expect(result.success).toBe(true);
});
it('should reject phone number with less than 10 digits', () => {
const result = loginRequestSchema.safeParse({
nomor: '08123456',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('minimal 10 digit');
}
});
it('should reject phone number with more than 15 digits', () => {
const result = loginRequestSchema.safeParse({
nomor: '081234567890123456',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('maksimal 15 digit');
}
});
it('should reject phone number with non-numeric characters', () => {
const result = loginRequestSchema.safeParse({
nomor: '0812-3456-789',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('harus berupa angka');
}
});
it('should reject empty phone number', () => {
const result = loginRequestSchema.safeParse({
nomor: '',
});
expect(result.success).toBe(false);
});
});
describe('otpVerificationSchema', () => {
it('should accept valid OTP verification data', () => {
const result = otpVerificationSchema.safeParse({
nomor: '08123456789',
kodeId: 'clm5z8z8z000008l4f3qz8z8z',
otp: '123456',
});
expect(result.success).toBe(true);
});
it('should reject OTP with wrong length', () => {
const result = otpVerificationSchema.safeParse({
nomor: '08123456789',
kodeId: 'clm5z8z8z000008l4f3qz8z8z',
otp: '12345',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('harus 6 digit');
}
});
it('should reject OTP with non-numeric characters', () => {
const result = otpVerificationSchema.safeParse({
nomor: '08123456789',
kodeId: 'clm5z8z8z000008l4f3qz8z8z',
otp: '12345a',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('harus berupa angka');
}
});
it('should reject invalid kodeId', () => {
const result = otpVerificationSchema.safeParse({
nomor: '08123456789',
kodeId: 'invalid-id',
otp: '123456',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('tidak valid');
}
});
});
// ============================================================================
// File Upload Validation Tests
// ============================================================================
describe('uploadFileSchema', () => {
it('should accept valid file upload data', () => {
const result = uploadFileSchema.safeParse({
name: 'document.pdf',
type: 'application/pdf',
size: 1024 * 1024, // 1MB
});
expect(result.success).toBe(true);
});
it('should reject empty file name', () => {
const result = uploadFileSchema.safeParse({
name: '',
type: 'application/pdf',
size: 1024 * 1024,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('wajib diisi');
}
});
it('should accept allowed image types', () => {
const allowedTypes = [
'image/jpeg',
'image/png',
'image/gif',
'image/webp',
];
allowedTypes.forEach((type) => {
const result = uploadFileSchema.safeParse({
name: 'file.jpg',
type,
size: 1024 * 1024,
});
expect(result.success).toBe(true);
});
});
it('should accept allowed document types', () => {
const allowedTypes = [
'application/pdf',
'application/msword',
'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
];
allowedTypes.forEach((type) => {
const result = uploadFileSchema.safeParse({
name: 'document.doc',
type,
size: 1024 * 1024,
});
expect(result.success).toBe(true);
});
});
it('should reject disallowed file types', () => {
const result = uploadFileSchema.safeParse({
name: 'file.exe',
type: 'application/x-executable',
size: 1024 * 1024,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('tidak diizinkan');
}
});
it('should reject files larger than 5MB', () => {
const result = uploadFileSchema.safeParse({
name: 'largefile.pdf',
type: 'application/pdf',
size: 6 * 1024 * 1024, // 6MB
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('maksimal 5MB');
}
});
it('should accept files exactly 5MB', () => {
const result = uploadFileSchema.safeParse({
name: 'file.pdf',
type: 'application/pdf',
size: 5 * 1024 * 1024, // 5MB
});
expect(result.success).toBe(true);
});
});
// ============================================================================
// User Registration Validation Tests
// ============================================================================
describe('registerUserSchema', () => {
it('should accept valid user registration data', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: '08123456789',
email: 'john@example.com',
roleId: 1,
});
expect(result.success).toBe(true);
});
it('should reject short names (less than 3 characters)', () => {
const result = registerUserSchema.safeParse({
name: 'Jo',
nomor: '08123456789',
roleId: 1,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('minimal 3 karakter');
}
});
it('should reject long names (more than 100 characters)', () => {
const result = registerUserSchema.safeParse({
name: 'a'.repeat(101),
nomor: '08123456789',
roleId: 1,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('maksimal 100 karakter');
}
});
it('should reject invalid phone numbers', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: 'invalid',
roleId: 1,
});
expect(result.success).toBe(false);
});
it('should accept empty email', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: '08123456789',
email: '',
roleId: 1,
});
expect(result.success).toBe(true);
});
it('should reject invalid email format', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: '08123456789',
email: 'not-an-email',
roleId: 1,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('tidak valid');
}
});
it('should reject non-integer roleId', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: '08123456789',
email: 'john@example.com',
roleId: 1.5,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('angka bulat');
}
});
it('should reject non-positive roleId', () => {
const result = registerUserSchema.safeParse({
name: 'John Doe',
nomor: '08123456789',
email: 'john@example.com',
roleId: 0,
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('lebih dari 0');
}
});
});
// ============================================================================
// Pagination Validation Tests
// ============================================================================
describe('paginationSchema', () => {
it('should accept default pagination values', () => {
const result = paginationSchema.safeParse({});
expect(result.success).toBe(true);
if (result.success) {
expect(result.data.page).toBe(1);
expect(result.data.limit).toBe(10);
}
});
it('should accept custom page and limit', () => {
const result = paginationSchema.safeParse({
page: '5',
limit: '25',
});
expect(result.success).toBe(true);
if (result.success) {
expect(result.data.page).toBe(5);
expect(result.data.limit).toBe(25);
}
});
it('should reject page less than 1', () => {
const result = paginationSchema.safeParse({
page: '0',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('lebih dari 0');
}
});
it('should reject limit less than 1', () => {
const result = paginationSchema.safeParse({
limit: '0',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('antara 1-100');
}
});
it('should reject limit greater than 100', () => {
const result = paginationSchema.safeParse({
limit: '101',
});
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('antara 1-100');
}
});
it('should accept limit exactly 100', () => {
const result = paginationSchema.safeParse({
limit: '100',
});
expect(result.success).toBe(true);
});
it('should accept optional search parameter', () => {
const result = paginationSchema.safeParse({
search: 'test query',
});
expect(result.success).toBe(true);
if (result.success) {
expect(result.data.search).toBe('test query');
}
});
it('should handle invalid page numbers gracefully', () => {
const result = paginationSchema.safeParse({
page: 'abc',
});
expect(result.success).toBe(false);
});
});

View File

@@ -1,362 +0,0 @@
/**
* WhatsApp Service Unit Tests
*
* Tests for WhatsApp OTP service in lib/whatsapp
* Note: These tests use direct fetch mocking, not MSW
*/
import { describe, it, expect, beforeEach, afterEach } from 'vitest';
import {
sendWhatsAppOTP,
formatOTPMessage,
formatOTPMessageWithReference,
} from '@/lib/whatsapp';
describe('WhatsApp Service', () => {
// Store original fetch
const originalFetch = global.fetch;
let mockFetch: any;
beforeEach(() => {
mockFetch = vi.fn();
global.fetch = mockFetch;
});
afterEach(() => {
global.fetch = originalFetch;
vi.restoreAllMocks();
});
// ============================================================================
// formatOTPMessage Tests
// ============================================================================
describe('formatOTPMessage', () => {
it('should format OTP message with numeric code', () => {
const otpCode = 123456;
const message = formatOTPMessage(otpCode);
expect(message).toContain('Website Desa Darmasaba');
expect(message).toContain('RAHASIA');
expect(message).toContain('JANGAN DI BAGIKAN');
expect(message).toContain('123456');
expect(message).toContain('satu kali login');
});
it('should format OTP message with string code', () => {
const otpCode = '654321';
const message = formatOTPMessage(otpCode);
expect(message).toContain('654321');
});
it('should include security warning', () => {
const message = formatOTPMessage(123456);
expect(message).toMatch(/RAHASIA/);
expect(message).toMatch(/JANGAN DI BAGIKAN KEPADA SIAPAPUN/);
});
it('should mention code validity', () => {
const message = formatOTPMessage(123456);
expect(message).toMatch(/hanya berlaku untuk satu kali login/);
});
});
// ============================================================================
// formatOTPMessageWithReference Tests
// ============================================================================
describe('formatOTPMessageWithReference', () => {
it('should format message with reference ID', () => {
const otpId = 'clm5z8z8z000008l4f3qz8z8z';
const message = formatOTPMessageWithReference(otpId);
expect(message).toContain('Website Desa Darmasaba');
expect(message).toContain('RAHASIA');
expect(message).toContain('JANGAN DI BAGIKAN');
expect(message).toContain(otpId);
expect(message).toContain('Reference ID');
});
it('should NOT include actual OTP code', () => {
const message = formatOTPMessageWithReference('test-id');
expect(message).not.toMatch(/\d{6}/);
});
it('should instruct user to enter received OTP', () => {
const message = formatOTPMessageWithReference('test-id');
expect(message).toMatch(/masukkan kode OTP/);
});
it('should include security warning', () => {
const message = formatOTPMessageWithReference('test-id');
expect(message).toMatch(/RAHASIA/);
expect(message).toMatch(/JANGAN DI BAGIKAN KEPADA SIAPAPUN/);
});
});
// ============================================================================
// sendWhatsAppOTP Tests
// ============================================================================
describe('sendWhatsAppOTP', () => {
it('should send OTP successfully with valid parameters', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'clm5z8z8z000008l4f3qz8z8z',
message: 'Test message',
});
expect(result.status).toBe('success');
expect(mockFetch).toHaveBeenCalledWith(
'https://wa.wibudev.com/send',
expect.objectContaining({
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
})
);
});
it('should use POST method (not GET) for security', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-otp-id',
message: 'Test',
});
const callArgs = mockFetch.mock.calls[0];
expect(callArgs[0]).toBe('https://wa.wibudev.com/send');
expect(callArgs[1]?.method).toBe('POST');
});
it('should send otpId reference, not actual OTP code', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-otp-id-123',
message: 'Test message',
});
const callArgs = mockFetch.mock.calls[0];
const body = JSON.parse(callArgs[1]?.body as string);
expect(body.otpId).toBe('test-otp-id-123');
expect(body.nomor).toBe('08123456789');
});
it('should return error for invalid phone number (empty)', async () => {
const result = await sendWhatsAppOTP({
nomor: '',
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Nomor telepon tidak valid');
expect(mockFetch).not.toHaveBeenCalled();
});
it('should return error for invalid phone number (null)', async () => {
const result = await sendWhatsAppOTP({
nomor: null as any,
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Nomor telepon tidak valid');
});
it('should return error for invalid otpId', async () => {
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: '',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('OTP ID tidak valid');
expect(mockFetch).not.toHaveBeenCalled();
});
it('should return error for null otpId', async () => {
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: null as any,
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('OTP ID tidak valid');
});
it('should handle WhatsApp API error response', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({
status: 'error',
message: 'Invalid phone number',
}),
clone: function() { return this; }
} as any);
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Invalid phone number');
});
it('should handle HTTP error response', async () => {
mockFetch.mockResolvedValue({
ok: false,
status: 500,
statusText: 'Internal Server Error',
clone: function() { return this; }
} as any);
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Gagal mengirim pesan WhatsApp');
});
it('should handle network errors', async () => {
mockFetch.mockRejectedValue(new Error('Network error'));
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Terjadi kesalahan saat mengirim pesan');
});
it('should handle JSON parse errors', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => {
throw new Error('Invalid JSON');
},
clone: function() { return this; }
} as any);
const result = await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Test',
});
expect(result.status).toBe('error');
expect(result.message).toBe('Terjadi kesalahan saat mengirim pesan');
});
it('should send correct request body structure', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
await sendWhatsAppOTP({
nomor: '081234567890',
otpId: 'unique-otp-id',
message: 'Custom message',
});
const callArgs = mockFetch.mock.calls[0];
const body = JSON.parse(callArgs[1]?.body as string);
expect(body).toEqual({
nomor: '081234567890',
otpId: 'unique-otp-id',
message: 'Custom message',
});
});
});
// ============================================================================
// Security Tests
// ============================================================================
describe('Security - OTP not exposed in URL', () => {
it('should NOT include OTP code in URL query string', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Your OTP is 123456',
});
const callArgs = mockFetch.mock.calls[0];
const url = callArgs[0];
// URL should be the endpoint, not containing OTP
expect(url).toBe('https://wa.wibudev.com/send');
expect(url).not.toContain('123456');
expect(url).not.toContain('?');
});
it('should send OTP in request body (POST), not URL', async () => {
mockFetch.mockResolvedValue({
ok: true,
json: async () => ({ status: 'success' }),
clone: function() { return this; }
} as any);
await sendWhatsAppOTP({
nomor: '08123456789',
otpId: 'test-id',
message: 'Test',
});
const callArgs = mockFetch.mock.calls[0];
// Should use POST with body
expect(callArgs[1]?.method).toBe('POST');
expect(callArgs[1]?.body).toBeDefined();
// OTP reference should be in body, not URL
const body = JSON.parse(callArgs[1]?.body as string);
expect(body.otpId).toBe('test-id');
});
});
});

View File

@@ -2,33 +2,6 @@ import '@testing-library/jest-dom';
import { server } from './mocks/server';
import { beforeAll, afterEach, afterAll } from 'vitest';
// MSW server setup for API mocking
beforeAll(() => server.listen({ onUnhandledRequest: 'bypass' }));
afterEach(() => server.resetHandlers());
afterAll(() => server.close());
// Mock window.matchMedia for Mantine components
Object.defineProperty(window, 'matchMedia', {
writable: true,
value: vi.fn().mockImplementation((query) => ({
matches: false,
media: query,
onchange: null,
addListener: vi.fn(),
removeListener: vi.fn(),
addEventListener: vi.fn(),
removeEventListener: vi.fn(),
dispatchEvent: vi.fn(),
})),
});
// Mock IntersectionObserver for Mantine components
global.IntersectionObserver = class IntersectionObserver {
constructor() {}
disconnect() {}
observe() {}
takeRecords() {
return [];
}
unobserve() {}
} as any;

BIN
bun.lockb

Binary file not shown.

View File

@@ -1,380 +0,0 @@
# State Management Guide
## Overview
Desa Darmasaba menggunakan **Valtio** untuk global state management. Valtio adalah state management library yang menggunakan proxy pattern untuk reactive state yang sederhana dan performant.
## Why Valtio?
-**Simple API** - Menggunakan plain JavaScript objects
-**Performant** - Component re-renders hanya saat state yang digunakan berubah
-**TypeScript-friendly** - Full TypeScript support
-**No boilerplate** - Tidak perlu actions, reducers, atau selectors
-**Flexible** - Bisa digunakan di dalam atau luar React components
## Installation
```bash
bun install valtio
```
## State Structure
```
src/state/
├── admin/ # Admin dashboard state
│ ├── index.ts # Admin state exports
│ ├── adminNavState.ts # Navigation state
│ ├── adminAuthState.ts # Authentication state
│ ├── adminFormState.ts # Form state (images, files)
│ └── adminModuleState.ts # Module-specific state
├── public/ # Public pages state
│ ├── index.ts # Public state exports
│ ├── publicNavState.ts # Navigation state
│ └── publicMusicState.ts # Music player state
├── darkModeStore.ts # Dark mode state (legacy)
└── index.ts # Main exports
```
## Basic Usage
### Creating State
```typescript
// src/state/example/exampleState.ts
import { proxy, useSnapshot } from 'valtio';
export const exampleState = proxy<{
count: number;
items: string[];
isLoading: boolean;
increment: () => void;
addItem: (item: string) => void;
}>({
count: 0,
items: [],
isLoading: false,
increment() {
exampleState.count += 1;
},
addItem(item: string) {
exampleState.items.push(item);
},
});
// Hook untuk React components
export const useExample = () => {
const snapshot = useSnapshot(exampleState);
return {
...snapshot,
increment: exampleState.increment,
addItem: exampleState.addItem,
};
};
```
### Using in React Components
```typescript
'use client';
import { useExample } from '@/state';
export function Counter() {
const { count, increment } = useExample();
return (
<button onClick={increment}>
Count: {count}
</button>
);
}
```
### Using Outside React
```typescript
// In non-React code (utilities, services, etc.)
import { exampleState } from '@/state';
// Direct mutation
exampleState.count = 10;
exampleState.increment();
// Subscribe to changes
import { subscribe } from 'valtio';
subscribe(exampleState, () => {
console.log('State changed:', exampleState.count);
});
```
## Domain-Specific State
### Admin State
State untuk admin dashboard hanya digunakan di `/admin` routes.
```typescript
import {
adminNavState,
adminAuthState,
useAdminNav,
useAdminAuth
} from '@/state';
// In React component
export function AdminHeader() {
const { mobileOpen, toggleMobile } = useAdminNav();
const { user, isAuthenticated } = useAdminAuth();
return (
<Header>
<Button onClick={toggleMobile}>Menu</Button>
{user?.name}
</Header>
);
}
// Outside React
adminNavState.mobileOpen = true;
adminAuthState.clearUser();
```
### Public State
State untuk public pages hanya digunakan di `/darmasaba` routes.
```typescript
import {
publicNavState,
publicMusicState,
usePublicNav,
usePublicMusic
} from '@/state';
// In React component
export function MusicPlayer() {
const { isPlaying, currentSong, togglePlayPause } = usePublicMusic();
return (
<Player>
{currentSong?.judul}
<Button onClick={togglePlayPause}>
{isPlaying ? 'Pause' : 'Play'}
</Button>
</Player>
);
}
```
## Async Operations
```typescript
// src/state/example/dataState.ts
import { proxy, useSnapshot } from 'valtio';
import ApiFetch from '@/lib/api-fetch';
export const dataState = proxy<{
data: any[];
isLoading: boolean;
error: string | null;
fetchData: (id: string) => Promise<void>;
}>({
data: [],
isLoading: false,
error: null,
async fetchData(id: string) {
dataState.isLoading = true;
dataState.error = null;
try {
const response = await ApiFetch.someApi.get({ id });
dataState.data = response.data;
} catch (error) {
dataState.error = error instanceof Error ? error.message : 'Failed to fetch';
} finally {
dataState.isLoading = false;
}
},
});
export const useData = () => {
const snapshot = useSnapshot(dataState);
return {
...snapshot,
fetchData: dataState.fetchData,
};
};
```
## Best Practices
### ✅ DO
1. **Separate admin and public state**
```typescript
// Good
import { adminNavState } from '@/state/admin';
import { publicNavState } from '@/state/public';
```
2. **Use methods in state for complex operations**
```typescript
// Good
export const state = proxy({
count: 0,
increment() {
state.count += 1;
},
});
```
3. **Add error handling in async methods**
```typescript
// Good
async fetchData() {
state.isLoading = true;
state.error = null;
try {
// fetch logic
} catch (error) {
state.error = error.message;
} finally {
state.isLoading = false;
}
}
```
4. **Use TypeScript for type safety**
```typescript
// Good
type User = { id: string; name: string };
export const authState = proxy<{
user: User | null;
setUser: (user: User | null) => void;
}>({ ... });
```
### ❌ DON'T
1. **Don't mutate state directly in render**
```typescript
// Bad
function Component() {
state.count += 1; // Don't do this in render
return <div>{state.count}</div>;
}
```
2. **Don't mix admin and public state**
```typescript
// Bad
import { adminAuthState } from '@/state/admin';
import { publicNavState } from '@/state/public';
// Don't use admin state in public pages
```
3. **Don't create new objects in state methods**
```typescript
// Bad
increment() {
state.count = state.count + 1; // Creates new number
}
// Good
increment() {
state.count += 1; // Mutates existing value
}
```
## Migration from Legacy State
### Old Pattern (Deprecated)
```typescript
// Old pattern - still works but deprecated
import stateNav from '@/state/state-nav';
import { authStore } from '@/store/authStore';
```
### New Pattern (Recommended)
```typescript
// New pattern - recommended
import { adminNavState } from '@/state/admin';
import { adminAuthState } from '@/state/admin';
```
## Music Player State
Music player sekarang menggunakan Valtio state dengan React Context wrapper untuk backward compatibility.
```typescript
// New way (recommended)
import { usePublicMusic } from '@/state/public';
function MusicPlayer() {
const { isPlaying, currentSong, togglePlayPause } = usePublicMusic();
// ...
}
// Old way (still works for backward compatibility)
import { useMusic } from '@/app/context/MusicContext';
function MusicPlayer() {
const { isPlaying, currentSong, togglePlayPause } = useMusic();
// ...
}
```
## Troubleshooting
### State not updating in component
Make sure you're using the hook in component:
```typescript
// Good
function Component() {
const { count } = useExample(); // Subscribe to state
return <div>{count}</div>;
}
// Bad
function Component() {
const count = exampleState.count; // No subscription
return <div>{count}</div>;
}
```
### Performance issues
Use selective subscriptions:
```typescript
// Good - only subscribe to what you need
function Component() {
const { count } = useExample(); // Only count
return <div>{count}</div>;
}
// Bad - subscribe to entire state
function Component() {
const state = useExample(); // Entire state
return <div>{state.count}</div>;
}
```
## Additional Resources
- [Valtio Documentation](https://github.com/pmndrs/valtio)
- [Valtio Examples](https://github.com/pmndrs/valtio/tree/main/examples)
- [Reactivity Guide](https://docs.pmnd.rs/valtio/guides/reactivity)

View File

@@ -1,540 +0,0 @@
# Testing Guide - Desa Darmasaba
## Overview
This document provides comprehensive testing guidelines for the Desa Darmasaba project. The project uses a multi-layered testing strategy including unit tests, component tests, and end-to-end (E2E) tests.
## Testing Stack
| Layer | Tool | Purpose |
|-------|------|---------|
| **Unit Tests** | Vitest | Testing utility functions, validation schemas, services |
| **Component Tests** | Vitest + React Testing Library | Testing React components in isolation |
| **E2E Tests** | Playwright | Testing complete user flows in real browsers |
| **API Mocking** | MSW (Mock Service Worker) | Mocking API responses for unit/component tests |
## Test Structure
```
__tests__/
├── api/ # API integration tests
│ └── fileStorage.test.ts
├── components/ # Component tests
│ └── admin/
│ ├── UnifiedTypography.test.tsx
│ └── UnifiedSurface.test.tsx
├── e2e/ # End-to-end tests
│ ├── admin/
│ │ └── auth.spec.ts
│ └── public/
│ └── pages.spec.ts
├── lib/ # Unit tests for utilities
│ ├── validations.test.ts
│ ├── sanitizer.test.ts
│ └── whatsapp.test.ts
├── mocks/ # MSW mocks for API
│ ├── handlers.ts
│ └── server.ts
└── setup.ts # Test setup and configuration
```
## Running Tests
### All Tests
```bash
bun run test
```
### Unit Tests Only
```bash
bun run test:api
```
### E2E Tests Only
```bash
bun run test:e2e
```
### Tests with Coverage
```bash
bun run test:api --coverage
```
### Run Specific Test File
```bash
bunx vitest run __tests__/lib/validations.test.ts
```
### Run Tests in Watch Mode
```bash
bunx vitest
```
### Run E2E Tests with UI
```bash
bun run test:e2e --ui
```
## Writing Tests
### Unit Tests (Vitest)
Unit tests should test pure functions, validation schemas, and utilities in isolation.
```typescript
// __tests__/lib/example.test.ts
import { describe, it, expect } from 'vitest';
import { exampleFunction } from '@/lib/example';
describe('exampleFunction', () => {
it('should return expected value for valid input', () => {
const result = exampleFunction('valid-input');
expect(result).toBe('expected-output');
});
it('should handle edge cases', () => {
expect(() => exampleFunction('')).toThrow();
expect(() => exampleFunction(null)).toThrow();
});
});
```
### Component Tests (React Testing Library)
Component tests should test React components in isolation with mocked dependencies.
```typescript
// __tests__/components/Example.test.tsx
import { describe, it, expect } from 'vitest';
import { render, screen, fireEvent } from '@testing-library/react';
import { MantineProvider, createTheme } from '@mantine/core';
import { ExampleComponent } from '@/components/Example';
function renderWithMantine(ui: React.ReactElement) {
const theme = createTheme();
return render(ui, {
wrapper: ({ children }) => (
<MantineProvider theme={theme}>{children}</MantineProvider>
),
});
}
describe('ExampleComponent', () => {
it('should render with props', () => {
renderWithMantine(<ExampleComponent title="Test Title" />);
expect(screen.getByText('Test Title')).toBeInTheDocument();
});
it('should handle user interactions', async () => {
const onClick = vi.fn();
renderWithMantine(<ExampleComponent onClick={onClick} />);
fireEvent.click(screen.getByRole('button'));
expect(onClick).toHaveBeenCalledTimes(1);
});
});
```
### E2E Tests (Playwright)
E2E tests should test complete user flows in a real browser environment.
```typescript
// __tests__/e2e/example.spec.ts
import { test, expect } from '@playwright/test';
test.describe('Feature Name', () => {
test.beforeEach(async ({ page }) => {
// Setup before each test
await page.goto('/starting-page');
});
test('should complete user flow', async ({ page }) => {
// Fill form
await page.fill('input[name="email"]', 'user@example.com');
await page.click('button[type="submit"]');
// Wait for navigation
await page.waitForURL('/success');
// Verify result
await expect(page.getByText('Success!')).toBeVisible();
});
test('should handle errors gracefully', async ({ page }) => {
// Submit invalid data
await page.click('button[type="submit"]');
// Verify error message
await expect(page.getByText('Validation error')).toBeVisible();
});
});
```
### API Mocking (MSW)
Use MSW to mock API responses for unit and component tests.
```typescript
// __tests__/mocks/handlers.ts
import { http, HttpResponse } from 'msw';
export const handlers = [
http.get('/api/example', () => {
return HttpResponse.json({
data: [{ id: '1', name: 'Item 1' }],
});
}),
http.post('/api/example', async ({ request }) => {
const body = await request.json();
return HttpResponse.json({
data: { id: '2', ...body },
status: 201,
});
}),
];
```
## Test Coverage Goals
Current coverage thresholds (configured in `vitest.config.ts`):
| Metric | Target |
|--------|--------|
| Branches | 50% |
| Functions | 50% |
| Lines | 50% |
| Statements | 50% |
### Critical Files Priority
Focus testing efforts on these critical files first:
1. **Validation & Security**
- `src/lib/validations/index.ts`
- `src/lib/sanitizer.ts`
- `src/lib/whatsapp.ts`
- `src/lib/session.ts`
2. **Core Utilities**
- `src/lib/api-fetch.ts`
- `src/lib/prisma.ts`
- `src/utils/themeTokens.ts`
3. **Shared Components**
- `src/components/admin/UnifiedTypography.tsx`
- `src/components/admin/UnifiedSurface.tsx`
- `src/components/admin/UnifiedCard.tsx`
4. **State Management**
- `src/state/darkModeStore.ts`
- `src/state/admin/*.ts`
- `src/state/public/*.ts`
5. **API Routes**
- `src/app/api/[[...slugs]]/_lib/auth/**`
- `src/app/api/[[...slugs]]/_lib/desa/**`
## Testing Conventions
### Naming Conventions
- **Unit/Component Tests**: `*.test.ts` or `*.test.tsx`
- **E2E Tests**: `*.spec.ts`
- **Test Files**: Match source file name (e.g., `sanitizer.ts``sanitizer.test.ts`)
- **Test Directories**: Mirror source structure under `__tests__/`
### Describe Blocks
Use nested `describe` blocks to organize tests logically:
```typescript
describe('FeatureName', () => {
describe('functionName', () => {
describe('when valid input', () => {
it('should return expected result', () => {});
});
describe('when invalid input', () => {
it('should throw error', () => {});
});
});
});
```
### Test Descriptions
- Use clear, descriptive test names
- Follow pattern: `should [expected behavior] when [condition]`
- Avoid vague descriptions like "works correctly"
### Assertions
- Use specific matchers (`toBe`, `toEqual`, `toContain`)
- Test both success and failure cases
- Test edge cases (empty input, null, undefined, max values)
### Setup and Teardown
```typescript
describe('ComponentName', () => {
beforeEach(() => {
// Reset mocks, state
vi.clearAllMocks();
});
afterEach(() => {
// Cleanup
vi.restoreAllMocks();
});
// ... tests
});
```
## Mocking Guidelines
### Mock External Services
```typescript
// Mock fetch API
global.fetch = vi.fn();
// Mock modules
vi.mock('@/lib/prisma', () => ({
default: {
berita: {
findMany: vi.fn(),
create: vi.fn(),
},
},
}));
```
### Mock Environment Variables
```typescript
const originalEnv = process.env;
beforeEach(() => {
process.env = {
...originalEnv,
TEST_VAR: 'test-value',
};
});
afterEach(() => {
process.env = originalEnv;
});
```
### Mock Date/Time
```typescript
const mockDate = new Date('2024-01-01T00:00:00Z');
vi.useFakeTimers();
vi.setSystemTime(mockDate);
// ... tests
vi.useRealTimers();
```
## E2E Testing Best Practices
### Test User Flows, Not Implementation
✅ Good:
```typescript
test('user can login and view dashboard', async ({ page }) => {
await page.goto('/admin/login');
await page.fill('input[name="nomor"]', '08123456789');
await page.click('button[type="submit"]');
await expect(page).toHaveURL('/admin/dashboard');
});
```
❌ Bad:
```typescript
test('login form submits to API', async ({ page }) => {
// Don't test internal implementation details
});
```
### Use Data Attributes for Selectors
```typescript
// In component
<button data-testid="submit-button">Submit</button>
// In test
await page.getByTestId('submit-button').click();
```
### Handle Async Operations
```typescript
// Wait for specific element
await page.waitForSelector('.loaded-content');
// Wait for navigation
await page.waitForNavigation();
// Wait for network request
await page.waitForResponse('/api/data');
```
### Skip Tests Appropriately
```typescript
// Skip in CI
test.skip(process.env.CI === 'true', 'Skip in CI environment');
// Skip with reason
test.skip(true, 'Feature not yet implemented');
// Conditional skip
test.skip(!hasValidCredentials, 'Requires valid credentials');
```
## Continuous Integration
### GitHub Actions Workflow
Tests run automatically on:
- Pull requests
- Push to main branch
- Manual trigger
### Test Requirements
- All new features must include tests
- Bug fixes should include regression tests
- Coverage should not decrease significantly
## Debugging Tests
### Vitest Debug Mode
```bash
bunx vitest --reporter=verbose
```
### Playwright Debug Mode
```bash
PWDEBUG=1 bun run test:e2e
```
### Playwright Trace Viewer
```bash
bun run test:e2e --trace on
bunx playwright show-trace
```
## Common Patterns
### Testing Validation Schemas
```typescript
describe('validationSchema', () => {
it('should accept valid data', () => {
const result = validationSchema.safeParse(validData);
expect(result.success).toBe(true);
});
it('should reject invalid data', () => {
const result = validationSchema.safeParse(invalidData);
expect(result.success).toBe(false);
if (!result.success) {
expect(result.error.errors[0].message).toContain('error message');
}
});
});
```
### Testing Async Functions
```typescript
it('should fetch data successfully', async () => {
const result = await fetchData();
expect(result).toEqual(expectedData);
});
it('should handle errors', async () => {
await expect(asyncFunction()).rejects.toThrow('error message');
});
```
### Testing Hooks
```typescript
import { renderHook, act } from '@testing-library/react';
it('should update state', () => {
const { result } = renderHook(() => useCustomHook());
act(() => {
result.current.setValue('new value');
});
expect(result.current.value).toBe('new value');
});
```
## Troubleshooting
### Common Issues
**Issue**: Tests fail with "Cannot find module"
**Solution**: Check import paths, ensure `@/` alias is configured in `vitest.config.ts`
**Issue**: Mantine components throw errors
**Solution**: Wrap components with `MantineProvider` in test setup
**Issue**: Tests fail in CI but pass locally
**Solution**: Check for environment-specific code, use proper mocking
**Issue**: E2E tests timeout
**Solution**: Increase timeout, check for async operations, use proper waits
### Getting Help
- Check existing tests for patterns
- Review Vitest documentation: https://vitest.dev
- Review Playwright documentation: https://playwright.dev
- Review Testing Library documentation: https://testing-library.com
## Resources
- [Vitest Documentation](https://vitest.dev)
- [Playwright Documentation](https://playwright.dev)
- [React Testing Library](https://testing-library.com/react)
- [MSW Documentation](https://mswjs.io)
- [Testing JavaScript Course](https://testingjavascript.com)
## Maintenance
### Regular Tasks
- [ ] Update test dependencies monthly
- [ ] Review and update test coverage goals quarterly
- [ ] Remove deprecated test patterns
- [ ] Add tests for newly discovered edge cases
- [ ] Document common testing patterns
### Deprecation Policy
When refactoring code:
1. Keep existing tests passing
2. Update tests to match new implementation
3. Remove tests for removed functionality
4. Update this documentation
---
**Last Updated**: March 9, 2026
**Version**: 1.0.0
**Maintained By**: Development Team

View File

@@ -1,6 +1,7 @@
import type { NextConfig } from "next";
const nextConfig: NextConfig = {
serverExternalPackages: ['@elysiajs/static', 'elysia'],
experimental: {},
allowedDevOrigins: [
"http://192.168.1.82:3000", // buat akses dari HP/device lain

View File

@@ -70,7 +70,7 @@
"embla-carousel-react": "^8.6.0",
"extract-zip": "^2.0.1",
"form-data": "^4.0.2",
"framer-motion": "^12.23.5",
"framer-motion": "^12.38.0",
"get-port": "^7.1.0",
"iron-session": "^8.0.4",
"jose": "^6.1.0",
@@ -100,7 +100,7 @@
"react-transition-group": "^4.4.5",
"react-zoom-pan-pinch": "^3.7.0",
"readdirp": "^4.1.1",
"recharts": "^2.15.3",
"recharts": "^3.8.0",
"sharp": "^0.34.3",
"swr": "^2.3.2",
"uuid": "^11.1.0",

View File

@@ -211,6 +211,9 @@ function ListKategoriPrestasi({ search }: { search: string }) {
</Stack>
</Box>
{/* Modal Konfirmasi Hapus */}
<ModalKonfirmasiHapus
opened={modalHapus}

View File

@@ -123,37 +123,51 @@ export default function CreateMusik() {
setIsSubmitting(true);
// Upload cover image
console.log('Uploading cover image:', coverFile.name);
const coverRes = await ApiFetch.api.fileStorage.create.post({
file: coverFile,
name: coverFile.name,
});
console.log('Cover upload response:', coverRes);
const coverUploaded = coverRes.data?.data;
if (!coverUploaded?.id) {
return toast.error('Gagal mengunggah cover, silakan coba lagi');
console.error('Cover upload failed:', coverRes);
toast.error('Gagal mengunggah cover, silakan coba lagi');
return;
}
musikState.musik.create.form.coverImageId = coverUploaded.id;
// Upload audio file
console.log('Uploading audio file:', audioFile.name);
const audioRes = await ApiFetch.api.fileStorage.create.post({
file: audioFile,
name: audioFile.name,
});
console.log('Audio upload response:', audioRes);
const audioUploaded = audioRes.data?.data;
if (!audioUploaded?.id) {
return toast.error('Gagal mengunggah audio, silakan coba lagi');
console.error('Audio upload failed:', audioRes);
toast.error('Gagal mengunggah audio, silakan coba lagi');
return;
}
musikState.musik.create.form.audioFileId = audioUploaded.id;
// Create musik entry
console.log('Creating musik entry with form:', musikState.musik.create.form);
await musikState.musik.create.create();
resetForm();
router.push('/admin/musik');
} catch (error) {
console.error('Error creating musik:', error);
console.error('Error creating musik:', {
error,
message: error instanceof Error ? error.message : 'Unknown error',
stack: error instanceof Error ? error.stack : undefined,
});
toast.error('Terjadi kesalahan saat membuat musik');
} finally {
setIsSubmitting(false);

View File

@@ -176,16 +176,16 @@ export default function Layout({ children }: { children: React.ReactNode }) {
return (
<AppShell
suppressHydrationWarning
header={{ height: { base: 56, sm: 64 } }}
header={{ height: 64 }}
navbar={{
width: { base: 280, sm: 280, lg: 300 },
width: { base: 260, sm: 280, lg: 300 },
breakpoint: 'sm',
collapsed: {
mobile: !opened,
desktop: !desktopOpened,
},
}}
padding={{ base: 'xs', sm: 'md' }}
padding="md"
>
{/*
HEADER / TOPBAR
@@ -195,73 +195,67 @@ export default function Layout({ children }: { children: React.ReactNode }) {
style={{
background: mounted ? tokens.colors.bg.header : 'linear-gradient(90deg, #ffffff, #f9fbff)',
borderBottom: `1px solid ${mounted ? tokens.colors.border.soft : '#e9ecef'}`,
padding: '0 12px',
padding: '0 16px',
transition: 'background 0.3s ease, border-color 0.3s ease',
}}
px={{ base: 'xs', sm: 'md' }}
py={{ base: '4px', sm: 'sm' }}
px={{ base: 'sm', sm: 'md' }}
py={{ base: 'xs', sm: 'sm' }}
>
<Group w="100%" h="100%" justify="space-between" wrap="nowrap">
<Flex align="center" gap={{ base: 'xs', sm: 'sm' }}>
<Burger opened={opened} onClick={toggle} visibleFrom="sm" size="sm" color={mounted ? tokens.colors.text.brand : '#0A4E78'} />
<Flex align="center" gap="sm">
<Image
src="/assets/images/darmasaba-icon.png"
alt="Logo Darmasaba"
w={{ base: 28, sm: 40 }}
h={{ base: 28, sm: 40 }}
w={{ base: 32, sm: 40 }}
h={{ base: 32, sm: 40 }}
radius="md"
loading="lazy"
style={{ minWidth: '28px', height: 'auto' }}
style={{ minWidth: '32px', height: 'auto' }}
/>
<Text fw={700} c={mounted ? tokens.colors.text.brand : '#0A4E78'} fz={{ base: 'sm', sm: 'xl' }} lineClamp={1}>
<span className="hidden sm:inline">Admin Darmasaba</span>
<Text fw={700} c={mounted ? tokens.colors.text.brand : '#0A4E78'} fz={{ base: 'md', sm: 'xl' }}>
Admin Darmasaba
</Text>
</Flex>
<Group gap="xs" wrap="nowrap">
{/* Mobile: Show menu button */}
<Burger opened={opened} onClick={toggle} hiddenFrom="sm" size="sm" color={mounted ? tokens.colors.text.brand : '#0A4E78'} />
<Group gap="xs">
{/* Dark Mode Toggle */}
<DarkModeToggle variant="light" size="lg" showTooltip tooltipPosition="bottom" />
{/* Desktop: Show collapse button */}
{!desktopOpened && (
<Tooltip label="Buka Navigasi" position="bottom" withArrow>
<ActionIcon variant="light" radius="xl" size="lg" onClick={toggleDesktop} color={mounted ? tokens.colors.primary : '#3B82F6'} visibleFrom="sm">
<ActionIcon variant="light" radius="xl" size="lg" onClick={toggleDesktop} color={mounted ? tokens.colors.primary : '#3B82F6'}>
<IconChevronRight />
</ActionIcon>
</Tooltip>
)}
{/* Dark Mode Toggle - smaller on mobile */}
<DarkModeToggle variant="light" size="md" showTooltip tooltipPosition="bottom" />
<Burger opened={opened} onClick={toggle} hiddenFrom="sm" size="md" color={mounted ? tokens.colors.text.brand : '#0A4E78'} mr="xs" />
{/* Home Button - hide on very small screens */}
<Tooltip label="Kembali ke Website Desa" position="bottom" withArrow>
<ActionIcon
onClick={() => router.push("/darmasaba")}
color={mounted ? tokens.colors.primary : '#3B82F6'}
radius="xl"
size="md"
size="lg"
variant="gradient"
gradient={mounted ? tokens.colors.gradient : { from: '#3B82F6', to: '#60A5FA' }}
visibleFrom="xs"
>
<Image src="/assets/images/darmasaba-icon.png" alt="Logo Darmasaba" w={18} h={18} radius="md" loading="lazy" style={{ minWidth: '18px', height: 'auto' }} />
<Image src="/assets/images/darmasaba-icon.png" alt="Logo Darmasaba" w={20} h={20} radius="md" loading="lazy" style={{ minWidth: '20px', height: 'auto' }} />
</ActionIcon>
</Tooltip>
{/* Logout Button */}
<Tooltip label="Keluar" position="bottom" withArrow>
<ActionIcon
onClick={handleLogout}
color={mounted ? tokens.colors.primary : '#3B82F6'}
radius="xl"
size="md"
size="lg"
variant="gradient"
gradient={mounted ? tokens.colors.gradient : { from: '#3B82F6', to: '#60A5FA' }}
loading={isLoggingOut}
disabled={isLoggingOut}
>
<IconLogout2 size={18} />
<IconLogout2 size={22} />
</ActionIcon>
</Tooltip>
</Group>
@@ -281,7 +275,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
}}
p={{ base: 'xs', sm: 'sm' }}
>
<AppShell.Section p={{ base: 'xs', sm: 'sm' }}>
<AppShell.Section p="sm">
{currentNav.map((v, k) => {
const isParentActive = segments.includes(_.lowerCase(v.name));
return (
@@ -292,7 +286,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
label={
<Text
fw={isParentActive ? 600 : 400}
fz={{ base: 'xs', sm: 'sm' }}
fz="sm"
style={{
color: mounted && isDark ? '#E5E7EB' : 'inherit',
transition: 'color 150ms ease',
@@ -342,7 +336,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
label={
<Text
fw={isChildActive ? 600 : 400}
fz={{ base: 'xs', sm: 'sm' }}
fz="sm"
style={{
color: mounted && isDark ? '#E5E7EB' : 'inherit',
transition: 'color 150ms ease',
@@ -381,7 +375,7 @@ export default function Layout({ children }: { children: React.ReactNode }) {
})}
</AppShell.Section>
<AppShell.Section py={{ base: 'sm', sm: 'md' }} visibleFrom="sm">
<AppShell.Section py="md">
<Group justify="end" pr="sm">
<Tooltip label={desktopOpened ? "Tutup Navigasi" : "Buka Navigasi"} position="top" withArrow>
<ActionIcon variant="light" radius="xl" size="lg" onClick={toggleDesktop} color={mounted ? tokens.colors.primary : '#3B82F6'}>

View File

@@ -1,8 +1,6 @@
import prisma from "@/lib/prisma";
import { NextResponse } from "next/server";
import { randomOTP } from "../_lib/randomOTP";
import { sendWhatsAppOTP, formatOTPMessage } from "@/lib/whatsapp";
import { loginRequestSchema } from "@/lib/validations";
export async function POST(req: Request) {
if (req.method !== "POST") {
@@ -14,84 +12,65 @@ export async function POST(req: Request) {
try {
const body = await req.json();
// Validate input with Zod schema
const validated = loginRequestSchema.parse(body);
const { nomor } = validated;
const { nomor } = body;
if (!nomor || typeof nomor !== "string") {
return NextResponse.json(
{ success: false, message: "Nomor tidak valid" },
{ status: 400 }
);
}
// Cek apakah user sudah terdaftar
const existingUser = await prisma.user.findUnique({
where: { nomor },
select: { id: true },
select: { id: true }, // cukup cek ada/tidak
});
const isRegistered = !!existingUser;
// Generate OTP
const codeOtp = randomOTP();
const codeOtp = randomOTP(); // Pastikan ini menghasilkan number (sesuai tipe di KodeOtp.otp: Int)
// Simpan OTP ke database terlebih dahulu untuk mendapatkan ID
const otpRecord = await prisma.kodeOtp.create({
data: {
nomor: nomor,
otp: codeOtp,
},
});
// Kirim OTP via WA
const waRes = await fetch(
`https://wa.wibudev.com/code?nom=${encodeURIComponent(nomor)}&text=Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.%0A%0A>> Kode OTP anda: ${codeOtp}.`
);
// Kirim OTP via WhatsApp dengan POST request yang aman
// OTP code tidak dikirim dalam URL query string
const waResult = await sendWhatsAppOTP({
nomor: nomor,
otpId: otpRecord.id,
message: formatOTPMessage(codeOtp),
});
if (waResult.status !== "success") {
// Delete OTP record jika WhatsApp gagal
await prisma.kodeOtp.delete({
where: { id: otpRecord.id },
}).catch(() => {}); // Ignore delete errors
const sendWa = await waRes.json();
if (sendWa.status !== "success") {
return NextResponse.json(
{
success: false,
message: waResult.message || "Gagal mengirim kode verifikasi"
},
{ success: false, message: "Nomor WhatsApp tidak aktif" },
{ status: 400 }
);
}
// Simpan OTP ke database
const otpRecord = await prisma.kodeOtp.create({
data: {
nomor: nomor,
otp: codeOtp, // Pastikan tipe ini number (Int di Prisma = number di TS)
},
});
return NextResponse.json(
{
success: true,
message: "Kode verifikasi terkirim",
kodeId: otpRecord.id,
isRegistered,
isRegistered, // 🔑 Ini kunci untuk frontend tahu harus ke register atau verifikasi
},
{ status: 200 }
);
} catch (error) {
// Handle Zod validation errors
if (error instanceof Error && error.constructor.name === 'ZodError') {
const zodError = error as import('zod').ZodError;
return NextResponse.json(
{
success: false,
message: "Validasi gagal",
errors: zodError.errors.map(e => ({
field: e.path.join('.'),
message: e.message,
}))
},
{ status: 400 }
);
}
console.error("Error Login:", error);
return NextResponse.json(
{
success: false,
message: "Terjadi masalah saat login",
// Hindari mengirim error mentah ke client di production
// reason: process.env.NODE_ENV === 'development' ? (error as Error).message : undefined,
},
{ status: 500 }
);

View File

@@ -1,65 +1,44 @@
import prisma from "@/lib/prisma";
import { Context } from "elysia";
import { createBeritaSchema, type CreateBeritaInput } from "@/lib/validations";
import { sanitizeHtml, sanitizeYouTubeUrl } from "@/lib/sanitizer";
type FormCreate = {
judul: string;
deskripsi: string;
content: string;
kategoriBeritaId: string;
imageId: string; // Featured image
imageIds?: string[]; // Multiple images for gallery
linkVideo?: string; // YouTube link
};
async function beritaCreate(context: Context) {
try {
// Validate input with Zod schema
const validated = createBeritaSchema.parse(context.body);
// Sanitize HTML content untuk mencegah XSS
const sanitizedContent = sanitizeHtml(validated.content);
// Sanitize YouTube URL jika ada
const sanitizedLinkVideo = validated.linkVideo
? sanitizeYouTubeUrl(validated.linkVideo)
: null;
const body = context.body as FormCreate;
// Create berita dengan data yang sudah divalidasi dan disanitize
await prisma.berita.create({
data: {
content: sanitizedContent,
deskripsi: validated.deskripsi,
imageId: validated.imageId,
judul: validated.judul,
kategoriBeritaId: validated.kategoriBeritaId,
linkVideo: sanitizedLinkVideo,
// Connect multiple images if provided
images: validated.imageIds && validated.imageIds.length > 0
? {
connect: validated.imageIds.map((id) => ({ id })),
}
: undefined,
},
});
await prisma.berita.create({
data: {
content: body.content,
deskripsi: body.deskripsi,
imageId: body.imageId,
judul: body.judul,
kategoriBeritaId: body.kategoriBeritaId,
// Connect multiple images if provided
linkVideo: body.linkVideo,
images: body.imageIds && body.imageIds.length > 0
? {
connect: body.imageIds.map((id) => ({ id })),
}
: undefined,
},
});
return {
success: true,
message: "Sukses menambahkan berita",
data: {
...validated,
content: sanitizedContent,
linkVideo: sanitizedLinkVideo,
},
};
} catch (error) {
// Handle Zod validation errors
if (error instanceof Error && error.constructor.name === 'ZodError') {
const zodError = error as import('zod').ZodError;
return {
success: false,
message: "Validasi gagal",
errors: zodError.errors.map(e => ({
field: e.path.join('.'),
message: e.message,
})),
};
}
// Re-throw other errors
throw error;
}
return {
success: true,
message: "Sukses menambahkan berita",
data: {
...body,
},
};
}
export default beritaCreate;
export default beritaCreate

View File

@@ -15,7 +15,7 @@ import AjukanPermohonan from "./layanan/ajukan_permohonan";
import Musik from "./musik";
const Desa = new Elysia({ prefix: "/api/desa", tags: ["Desa"] })
const Desa = new Elysia({ prefix: "/desa", tags: ["Desa"] })
.use(Berita)
.use(Pengumuman)
.use(ProfileDesa)

View File

@@ -13,7 +13,7 @@ import PendapatanAsliDesa from "./pendapatan-asli-desa";
import StrukturOrganisasi from "./struktur-bumdes";
const Ekonomi = new Elysia({
prefix: "/api/ekonomi",
prefix: "/ekonomi",
tags: ["Ekonomi"],
})
.use(PasarDesa)

View File

@@ -5,7 +5,7 @@ import { fileStorageFindMany } from "./_lib/findMany";
import fileStorageDelete from "./_lib/del";
const FileStorage = new Elysia({
prefix: "/api/fileStorage",
prefix: "/fileStorage",
tags: ["FileStorage"],
})
.post("/create", fileStorageCreate, {

View File

@@ -8,7 +8,7 @@ import LayananOnlineDesa from "./layanan-online-desa";
import MitraKolaborasi from "./kolaborasi-inovasi/mitra-kolaborasi";
const Inovasi = new Elysia({
prefix: "/api/inovasi",
prefix: "/inovasi",
tags: ["Inovasi"],
})
.use(DesaDigital)

View File

@@ -9,7 +9,7 @@ import KontakDaruratKeamanan from "./kontak-darurat-keamanan";
import KontakItem from "./kontak-darurat-keamanan/kontak-item";
import LayananPolsek from "./polsek-terdekat/layanan-polsek";
const Keamanan = new Elysia({ prefix: "/api/keamanan", tags: ["Keamanan"] })
const Keamanan = new Elysia({ prefix: "/keamanan", tags: ["Keamanan"] })
.use(KeamananLingkungan)
.use(PolsekTerdekat)
.use(PencegahanKriminalitas)

View File

@@ -24,7 +24,7 @@ import TarifLayanan from "./data_kesehatan_warga/fasilitas_kesehatan/tarif-layan
const Kesehatan = new Elysia({
prefix: "/api/kesehatan",
prefix: "/kesehatan",
tags: ["Kesehatan"],
})
.use(PersentaseKelahiranKematian)

View File

@@ -14,7 +14,7 @@ import UmurResponden from "./indeks_kepuasan/umur-responden";
import Responden from "./indeks_kepuasan/responden";
const LandingPage = new Elysia({
prefix: "/api/landingpage",
prefix: "/landingpage",
tags: ["Landing Page/Profile"]
})

View File

@@ -9,7 +9,7 @@ import KategoriKegiatan from "./gotong-royong/kategori-kegiatan";
import KeteranganBankSampahTerdekat from "./pengelolaan-sampah/keterangan-bank-sampah";
const Lingkungan = new Elysia({
prefix: "/api/lingkungan",
prefix: "/lingkungan",
tags: ["Lingkungan"],
})

View File

@@ -8,7 +8,7 @@ import Beasiswa from "./beasiswa-desa";
import PerpustakaanDigital from "./perpustakaan-digital";
const Pendidikan = new Elysia({
prefix: "/api/pendidikan",
prefix: "/pendidikan",
tags: ["Pendidikan"]
})

View File

@@ -14,7 +14,7 @@ import GrafikHasilKepuasanMasyarakat from "./ikm/grafik_hasil_kepuasan_masyaraka
const PPID = new Elysia({ prefix: "/api/ppid", tags: ["PPID"] })
const PPID = new Elysia({ prefix: "/ppid", tags: ["PPID"] })
.use(ProfilePPID)
.use(DaftarInformasiPublik)
.use(GrafikHasilKepuasanMasyarakat)

View File

@@ -2,7 +2,7 @@ import Elysia from "elysia";
import searchFindMany from "./findMany";
const Search = new Elysia({
prefix: "/api/search",
prefix: "/search",
tags: ["Search"],
})
.get("/findMany", searchFindMany);

View File

@@ -7,7 +7,7 @@ import userDelete from "./del"; // `delete` nggak boleh jadi nama file JS langsu
import userUpdate from "./updt";
import userDeleteAccount from "./delUser";
const User = new Elysia({ prefix: "/api/user" })
const User = new Elysia({ prefix: "/user" })
.get("/findMany", userFindMany)
.get("/findUnique/:id", userFindUnique)
.put("/del/:id", userDelete, {

View File

@@ -6,7 +6,7 @@ import roleFindUnique from "./findUnique";
import roleUpdate from "./updt";
const Role = new Elysia({
prefix: "/api/role",
prefix: "/role",
tags: ["User / Role"],
})

View File

@@ -47,15 +47,16 @@ fs.mkdir(UPLOAD_DIR_IMAGE, {
const corsConfig = {
origin: [
"*", // Allow all origins - must be first when using credentials: true
"http://localhost:3000",
"http://localhost:3001",
"https://cld-dkr-desa-darmasaba-stg.wibudev.com",
"https://cld-dkr-staging-desa-darmasaba.wibudev.com",
"*", // Allow all origins in development
"https://desa-darmasaba-stg.wibudev.com",
],
methods: ["GET", "POST", "PATCH", "DELETE", "PUT", "OPTIONS"] as HTTPMethod[],
allowedHeaders: ["Content-Type", "Authorization", "*"],
exposedHeaders: "*",
allowedHeaders: ["Content-Type", "Authorization", "Accept", "*"],
exposedHeaders: ["Content-Range", "X-Content-Range", "*"],
maxAge: 86400, // 24 hours
credentials: true,
};
@@ -66,7 +67,7 @@ async function layanan() {
}
const Utils = new Elysia({
prefix: "/api/utils",
prefix: "/utils",
tags: ["Utils"],
}).get("/version", async () => {
const packageJson = await fs.readFile(
@@ -81,7 +82,6 @@ if (!process.env.WIBU_UPLOAD_DIR)
throw new Error("WIBU_UPLOAD_DIR is not defined");
const ApiServer = new Elysia()
.use(swagger({ path: "/api/docs" }))
.use(
staticPlugin({
assets: UPLOAD_DIR,
@@ -89,21 +89,6 @@ const ApiServer = new Elysia()
}),
)
.use(cors(corsConfig))
.use(Utils)
.use(FileStorage)
.use(LandingPage)
.use(PPID)
.use(Desa)
.use(Kesehatan)
.use(Keamanan)
.use(Ekonomi)
.use(Inovasi)
.use(Lingkungan)
.use(Pendidikan)
.use(User)
.use(Role)
.use(Search)
.onError(({ code }) => {
if (code === "NOT_FOUND") {
return {
@@ -114,6 +99,31 @@ const ApiServer = new Elysia()
})
.group("/api", (app) =>
app
.use(
swagger({
path: "/docs",
documentation: {
info: {
title: "Desa Darmasaba API Documentation",
version: "1.0.0",
},
},
}),
)
.use(Utils)
.use(FileStorage)
.use(LandingPage)
.use(PPID)
.use(Desa)
.use(Kesehatan)
.use(Keamanan)
.use(Ekonomi)
.use(Inovasi)
.use(Lingkungan)
.use(Pendidikan)
.use(User)
.use(Role)
.use(Search)
.get("/layanan", layanan)
.get("/potensi", getPotensi)
.get(

View File

@@ -1,33 +0,0 @@
/**
* Music Context Compatibility Layer
*
* Wrapper untuk backward compatibility dengan kode yang sudah menggunakan useMusic
* Menggunakan Valtio state di belakang layar
*/
'use client';
import { useSnapshot } from 'valtio';
import { publicMusicState, usePublicMusic } from '@/state/public/publicMusicState';
// Export MusicProvider dari file terpisah
export { MusicProvider } from './MusicProvider';
// Export compatibility hook yang sama dengan Context API
export const useMusic = () => {
const music = usePublicMusic();
return {
...music,
// Tambahkan loadMusikData sebagai method reference
loadMusikData: publicMusicState.loadMusikData,
};
};
// Re-export types
export type { Musik } from '@/state/public/publicMusicState';
// Helper untuk mendapatkan snapshot tanpa subscribtion
export const getMusicState = () => {
return publicMusicState;
};

View File

@@ -1,20 +1,320 @@
/**
* Music Context - Legacy Compatibility Layer
*
* DEPRECATED: File ini dipertahankan untuk backward compatibility.
* Gunakan `useMusic` dari `@/app/context/MusicContext` (file .ts) untuk state management baru.
*
* Menggunakan Valtio state management di belakang layar untuk konsistensi.
* Audio handling dipindahkan ke MusicProvider.tsx untuk menghindari duplikasi
*/
'use client';
import {
createContext,
useContext,
useState,
useRef,
useEffect,
useCallback,
ReactNode,
} from 'react';
// Re-export MusicProvider dari file terpisah (satu-satunya tempat audio handling)
export { MusicProvider } from './MusicProvider';
import { usePublicMusic } from '../../state/public/publicMusicState.ts';
// Hook untuk backward compatibility
export function useMusic() {
return usePublicMusic();
interface MusicFile {
id: string;
name: string;
realName: string;
path: string;
mimeType: string;
link: string;
}
export interface Musik {
id: string;
judul: string;
artis: string;
deskripsi: string | null;
durasi: string;
genre: string | null;
tahunRilis: number | null;
audioFile: MusicFile | null;
coverImage: MusicFile | null;
isActive: boolean;
}
interface MusicContextType {
// State
isPlaying: boolean;
currentSong: Musik | null;
currentSongIndex: number;
musikData: Musik[];
currentTime: number;
duration: number;
volume: number;
isMuted: boolean;
isRepeat: boolean;
isShuffle: boolean;
isLoading: boolean;
isPlayerOpen: boolean;
// Actions
playSong: (song: Musik) => void;
togglePlayPause: () => void;
playNext: () => void;
playPrev: () => void;
seek: (time: number) => void;
setVolume: (volume: number) => void;
toggleMute: () => void;
toggleRepeat: () => void;
toggleShuffle: () => void;
togglePlayer: () => void;
loadMusikData: () => Promise<void>;
}
const MusicContext = createContext<MusicContextType | undefined>(undefined);
export function MusicProvider({ children }: { children: ReactNode }) {
// State
const [isPlaying, setIsPlaying] = useState(false);
const [currentSong, setCurrentSong] = useState<Musik | null>(null);
const [currentSongIndex, setCurrentSongIndex] = useState(-1);
const [musikData, setMusikData] = useState<Musik[]>([]);
const [currentTime, setCurrentTime] = useState(0);
const [duration, setDuration] = useState(0);
const [volume, setVolumeState] = useState(70);
const [isMuted, setIsMuted] = useState(false);
const [isRepeat, setIsRepeat] = useState(false);
const [isShuffle, setIsShuffle] = useState(false);
const [isLoading, setIsLoading] = useState(true);
const [isPlayerOpen, setIsPlayerOpen] = useState(false);
// Refs
const audioRef = useRef<HTMLAudioElement | null>(null);
const isSeekingRef = useRef(false);
const animationFrameRef = useRef<number | null>(null);
const isRepeatRef = useRef(false); // Ref untuk avoid stale closure
// Sync ref dengan state
useEffect(() => {
isRepeatRef.current = isRepeat;
}, [isRepeat]);
// Load musik data
const loadMusikData = useCallback(async () => {
try {
setIsLoading(true);
const res = await fetch('/api/desa/musik/find-many?page=1&limit=50');
const data = await res.json();
if (data.success && data.data) {
const activeMusik = data.data.filter((m: Musik) => m.isActive);
setMusikData(activeMusik);
}
} catch (error) {
console.error('Error fetching musik:', error);
} finally {
setIsLoading(false);
}
}, []);
// Initialize audio element
useEffect(() => {
audioRef.current = new Audio();
audioRef.current.preload = 'metadata';
// Event listeners
audioRef.current.addEventListener('loadedmetadata', () => {
setDuration(Math.floor(audioRef.current!.duration));
});
audioRef.current.addEventListener('ended', () => {
// Gunakan ref untuk avoid stale closure
if (isRepeatRef.current) {
audioRef.current!.currentTime = 0;
audioRef.current!.play();
} else {
playNext();
}
});
// Load initial data
loadMusikData();
return () => {
if (audioRef.current) {
audioRef.current.pause();
audioRef.current = null;
}
if (animationFrameRef.current) {
cancelAnimationFrame(animationFrameRef.current);
}
};
// eslint-disable-next-line react-hooks/exhaustive-deps -- playNext is intentionally not in deps to avoid circular dependency
}, [loadMusikData]); // Remove isRepeat dari deps karena sudah pakai ref
// Update time with requestAnimationFrame for smooth progress
const updateTime = useCallback(() => {
if (audioRef.current && !audioRef.current.paused && !isSeekingRef.current) {
setCurrentTime(Math.floor(audioRef.current.currentTime));
animationFrameRef.current = requestAnimationFrame(updateTime);
}
}, []);
useEffect(() => {
if (isPlaying) {
animationFrameRef.current = requestAnimationFrame(updateTime);
} else {
if (animationFrameRef.current) {
cancelAnimationFrame(animationFrameRef.current);
}
}
return () => {
if (animationFrameRef.current) {
cancelAnimationFrame(animationFrameRef.current);
}
};
}, [isPlaying, updateTime]);
// Play song
const playSong = useCallback(
(song: Musik) => {
if (!song?.audioFile?.link || !audioRef.current) return;
const songIndex = musikData.findIndex(m => m.id === song.id);
setCurrentSongIndex(songIndex);
setCurrentSong(song);
setIsPlaying(true);
audioRef.current.src = song.audioFile.link;
audioRef.current.load();
audioRef.current
.play()
.catch((err) => console.error('Error playing audio:', err));
},
[musikData]
);
// Toggle play/pause
const togglePlayPause = useCallback(() => {
if (!audioRef.current || !currentSong) return;
if (isPlaying) {
audioRef.current.pause();
setIsPlaying(false);
} else {
audioRef.current
.play()
.then(() => setIsPlaying(true))
.catch((err) => console.error('Error playing audio:', err));
}
}, [isPlaying, currentSong]);
// Play next
const playNext = useCallback(() => {
if (musikData.length === 0) return;
let nextIndex: number;
if (isShuffle) {
nextIndex = Math.floor(Math.random() * musikData.length);
} else {
nextIndex = (currentSongIndex + 1) % musikData.length;
}
const nextSong = musikData[nextIndex];
if (nextSong) {
playSong(nextSong);
}
}, [musikData, isShuffle, currentSongIndex, playSong]);
// Play previous
const playPrev = useCallback(() => {
if (musikData.length === 0) return;
// If more than 3 seconds into song, restart it
if (currentTime > 3) {
if (audioRef.current) {
audioRef.current.currentTime = 0;
}
return;
}
const prevIndex =
currentSongIndex <= 0 ? musikData.length - 1 : currentSongIndex - 1;
const prevSong = musikData[prevIndex];
if (prevSong) {
playSong(prevSong);
}
}, [musikData, currentSongIndex, currentTime, playSong]);
// Seek
const seek = useCallback((time: number) => {
if (!audioRef.current) return;
audioRef.current.currentTime = time;
setCurrentTime(time);
}, []);
// Set volume
const setVolume = useCallback((vol: number) => {
if (!audioRef.current) return;
const normalizedVol = Math.max(0, Math.min(100, vol)) / 100;
audioRef.current.volume = normalizedVol;
setVolumeState(Math.max(0, Math.min(100, vol)));
setIsMuted(normalizedVol === 0);
}, []);
// Toggle mute
const toggleMute = useCallback(() => {
if (!audioRef.current) return;
const newMuted = !isMuted;
audioRef.current.muted = newMuted;
setIsMuted(newMuted);
if (newMuted && volume > 0) {
audioRef.current.volume = 0;
} else if (!newMuted && volume > 0) {
audioRef.current.volume = volume / 100;
}
}, [isMuted, volume]);
// Toggle repeat
const toggleRepeat = useCallback(() => {
setIsRepeat((prev) => !prev);
}, []);
// Toggle shuffle
const toggleShuffle = useCallback(() => {
setIsShuffle((prev) => !prev);
}, []);
// Toggle player
const togglePlayer = useCallback(() => {
setIsPlayerOpen((prev) => !prev);
}, []);
const value: MusicContextType = {
isPlaying,
currentSong,
currentSongIndex,
musikData,
currentTime,
duration,
volume,
isMuted,
isRepeat,
isShuffle,
isLoading,
isPlayerOpen,
playSong,
togglePlayPause,
playNext,
playPrev,
seek,
setVolume,
toggleMute,
toggleRepeat,
toggleShuffle,
togglePlayer,
loadMusikData,
};
return (
<MusicContext.Provider value={value}>{children}</MusicContext.Provider>
);
}
export function useMusic() {
const context = useContext(MusicContext);
if (context === undefined) {
throw new Error('useMusic must be used within a MusicProvider');
}
return context;
}

View File

@@ -1,168 +0,0 @@
/**
* Music Provider Component
*
* Wrapper component untuk music player menggunakan Valtio state
* Menyediakan audio element dan logic yang membutuhkan React lifecycle
*/
'use client';
import { useEffect, useRef, useCallback } from 'react';
import { publicMusicState } from '@/state/public/publicMusicState';
import { useSnapshot } from 'valtio';
export function MusicProvider({ children }: { children: React.ReactNode }) {
const audioRef = useRef<HTMLAudioElement | null>(null);
const animationFrameRef = useRef<number | null>(null);
const isSeekingRef = useRef(false);
const isChangingSongRef = useRef(false);
// Subscribe to Valtio state changes
const snapshot = useSnapshot(publicMusicState);
// Initialize audio element
useEffect(() => {
audioRef.current = new Audio();
audioRef.current.preload = 'metadata';
audioRef.current.volume = publicMusicState.volume / 100;
// Event listeners
audioRef.current.addEventListener('loadedmetadata', () => {
publicMusicState.duration = Math.floor(audioRef.current!.duration);
console.log('[MusicProvider] Duration loaded:', publicMusicState.duration);
});
// Update currentTime on timeupdate event - this is the key fix!
audioRef.current.addEventListener('timeupdate', () => {
const currentTime = Math.floor(audioRef.current!.currentTime);
// Only update if changed to prevent unnecessary re-renders
if (currentTime !== publicMusicState.currentTime) {
publicMusicState.currentTime = currentTime;
}
});
audioRef.current.addEventListener('ended', () => {
if (publicMusicState.isRepeat) {
audioRef.current!.currentTime = 0;
audioRef.current!.play().catch(console.error);
} else {
publicMusicState.playNext();
}
});
// Handle play/pause errors gracefully
audioRef.current.addEventListener('error', (e) => {
console.warn('[MusicProvider] Audio error:', e);
publicMusicState.isPlaying = false;
});
// Load initial data
publicMusicState.loadMusikData();
return () => {
if (audioRef.current) {
audioRef.current.pause();
audioRef.current.src = '';
audioRef.current.load();
audioRef.current = null;
}
if (animationFrameRef.current) {
cancelAnimationFrame(animationFrameRef.current);
}
};
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
// Handle song changes - load new audio source
useEffect(() => {
if (!audioRef.current) return;
const song = snapshot.currentSong;
if (!song?.audioFile?.link) {
console.warn('[MusicProvider] No song or audio link:', song);
return;
}
console.log('[MusicProvider] Loading song:', song.judul, song.audioFile.link);
// Set flag to prevent race conditions
isChangingSongRef.current = true;
// Pause current playback
audioRef.current.pause();
audioRef.current.src = '';
audioRef.current.load();
// Load new song
audioRef.current.src = song.audioFile.link;
audioRef.current.load();
// Wait for audio to be ready before playing
const handleCanPlay = () => {
console.log('[MusicProvider] Song can play, isPlaying:', snapshot.isPlaying);
isChangingSongRef.current = false;
if (snapshot.isPlaying) {
audioRef.current!.play().then(() => {
console.log('[MusicProvider] Song started playing');
}).catch((err) => {
// Ignore AbortError - this is expected when changing songs
if (err.name !== 'AbortError') {
console.error('[MusicProvider] Error playing audio:', err);
}
});
}
};
const handleError = (err: Event) => {
console.error('[MusicProvider] Error loading audio:', err);
isChangingSongRef.current = false;
};
audioRef.current.addEventListener('canplay', handleCanPlay, { once: true });
audioRef.current.addEventListener('error', handleError, { once: true });
// Cleanup
return () => {
audioRef.current?.removeEventListener('canplay', handleCanPlay);
audioRef.current?.removeEventListener('error', handleError);
};
}, [snapshot.currentSong, snapshot.currentSongIndex]);
// Sync play/pause state (only when not changing songs)
useEffect(() => {
if (!audioRef.current || !snapshot.currentSong || isChangingSongRef.current) return;
if (snapshot.isPlaying) {
audioRef.current.play().catch((err) => {
// Ignore AbortError - this is expected
if (err.name !== 'AbortError') {
console.error('[MusicProvider] Error playing audio:', err);
}
});
} else {
audioRef.current.pause();
}
}, [snapshot.isPlaying]);
// Handle volume changes
useEffect(() => {
if (!audioRef.current) return;
const newVolume = snapshot.isMuted ? 0 : snapshot.volume / 100;
console.log('[MusicProvider] Volume changed:', snapshot.volume, 'muted:', snapshot.isMuted, 'setting:', newVolume);
audioRef.current.volume = newVolume;
audioRef.current.muted = snapshot.isMuted;
}, [snapshot.volume, snapshot.isMuted]);
// Handle seek - ONLY when user manually seeks (not during normal playback)
// We don't need to sync currentTime back to audio element during normal playback
// because timeupdate event handles that automatically
return (
<>
{children}
</>
);
}

View File

@@ -92,10 +92,10 @@ const MusicPlayer = () => {
}
return (
<Box px={{ base: 'xs', sm: 'md', md: 100 }} py="xl">
<Box px={{ base: 'md', md: 100 }} py="xl">
<Paper
mx="auto"
p={{ base: 'md', sm: 'xl' }}
p="xl"
radius="lg"
shadow="sm"
bg="white"
@@ -105,52 +105,42 @@ const MusicPlayer = () => {
>
<Stack gap="md">
<BackButton />
<Flex
justify="space-between"
align={{ base: 'flex-start', sm: 'center' }}
direction={{ base: 'column', sm: 'row' }}
gap="md"
mb="xl"
mt="md"
>
<Group justify="space-between" mb="xl" mt={"md"}>
<div>
<Text fz={{ base: '24px', sm: '32px' }} fw={700} c="#0B4F78" lh={1.2}>Selamat Datang Kembali</Text>
<Text size="sm" c="#5A6C7D">Temukan musik favorit Anda hari ini</Text>
<Text size="32px" fw={700} c="#0B4F78">Selamat Datang Kembali</Text>
<Text size="md" c="#5A6C7D">Temukan musik favorit Anda hari ini</Text>
</div>
<TextInput
placeholder="Cari lagu..."
leftSection={<IconSearch size={18} />}
radius="xl"
w={{ base: '100%', sm: 280 }}
value={search}
onChange={(e) => setSearch(e.target.value)}
styles={{ input: { backgroundColor: '#fff' } }}
/>
</Flex>
<Group gap="md">
<TextInput
placeholder="Cari lagu..."
leftSection={<IconSearch size={18} />}
radius="xl"
w={280}
value={search}
onChange={(e) => setSearch(e.target.value)}
styles={{ input: { backgroundColor: '#fff' } }}
/>
</Group>
</Group>
<Stack gap="xl">
<div>
<Text size="xl" fw={700} c="#0B4F78" mb="md">Sedang Diputar</Text>
{currentSong ? (
<Card radius="md" p={{ base: 'md', sm: 'xl' }} shadow="md" withBorder>
<Flex
direction={{ base: 'column', sm: 'row' }}
align="center"
gap={{ base: 'md', sm: 'xl' }}
>
<Card radius="md" p="xl" shadow="md">
<Group align="center" gap="xl">
<Avatar
src={currentSong.coverImage?.link || '/mp3-logo.png'}
size={120}
size={180}
radius="md"
/>
<Stack gap="md" style={{ flex: 1, width: '100%' }}>
<Box ta={{ base: 'center', sm: 'left' }}>
<Text fz={{ base: '20px', sm: '28px' }} fw={700} c="#0B4F78" lineClamp={1}>{currentSong.judul}</Text>
<Stack gap="md" style={{ flex: 1 }}>
<div>
<Text size="28px" fw={700} c="#0B4F78">{currentSong.judul}</Text>
<Text size="lg" c="#5A6C7D">{currentSong.artis}</Text>
{currentSong.genre && (
<Badge mt="xs" color="#0B4F78" variant="light">{currentSong.genre}</Badge>
)}
</Box>
</div>
<Group gap="xs" align="center">
<Text size="xs" c="#5A6C7D" w={42}>{formatTime(currentTime)}</Text>
<Slider
@@ -165,7 +155,7 @@ const MusicPlayer = () => {
<Text size="xs" c="#5A6C7D" w={42}>{formatTime(duration || 0)}</Text>
</Group>
</Stack>
</Flex>
</Group>
</Card>
) : (
<Card radius="md" p="xl" shadow="md">
@@ -185,29 +175,28 @@ const MusicPlayer = () => {
<Grid.Col span={{ base: 12, sm: 6, lg: 4 }} key={song.id}>
<Card
radius="md"
p="sm"
p="md"
shadow="sm"
withBorder
style={{
cursor: 'pointer',
borderColor: currentSong?.id === song.id ? '#0B4F78' : 'transparent',
backgroundColor: currentSong?.id === song.id ? '#F0F7FA' : 'white',
border: currentSong?.id === song.id ? '2px solid #0B4F78' : '2px solid transparent',
transition: 'all 0.2s'
}}
onClick={() => playSong(song)}
>
<Group gap="sm" align="center" wrap="nowrap">
<Group gap="md" align="center">
<Avatar
src={song.coverImage?.link || '/mp3-logo.png'}
size={50}
src={song.coverImage?.link || 'https://images.unsplash.com/photo-1470225620780-dba8ba36b745?w=400&h=400&fit=crop'}
size={64}
radius="md"
/>
<Stack gap={0} style={{ flex: 1, minWidth: 0 }}>
<Stack gap={4} style={{ flex: 1, minWidth: 0 }}>
<Text size="sm" fw={600} c="#0B4F78" truncate>{song.judul}</Text>
<Text size="xs" c="#5A6C7D" truncate>{song.artis}</Text>
<Text size="xs" c="#5A6C7D">{song.artis}</Text>
<Text size="xs" c="#8A9BA8">{song.durasi}</Text>
</Stack>
{currentSong?.id === song.id && isPlaying && (
<Badge color="#0B4F78" variant="filled" size="xs">Playing</Badge>
<Badge color="#0B4F78" variant="filled">Memutar</Badge>
)}
</Group>
</Card>
@@ -218,42 +207,34 @@ const MusicPlayer = () => {
)}
</div>
</Stack>
</Stack>
</Paper>
{/* Control Player Section */}
<Paper
mt="xl"
mx="auto"
p={{ base: 'md', sm: 'xl' }}
p="xl"
radius="lg"
shadow="sm"
bg="white"
style={{
border: '1px solid #eaeaea',
position: 'sticky',
bottom: 20,
zIndex: 10
}}
>
<Flex
direction={{ base: 'column', md: 'row' }}
align="center"
justify="space-between"
gap={{ base: 'md', md: 'xl' }}
>
{/* Song Info */}
<Group gap="md" style={{ flex: 1, width: '100%' }} wrap="nowrap">
<Flex align="center" justify="space-between" gap="xl" h="100%">
<Group gap="md" style={{ flex: 1 }}>
<Avatar
src={currentSong?.coverImage?.link || '/mp3-logo.png'}
size={48}
src={currentSong?.coverImage?.link || 'https://images.unsplash.com/photo-1470225620780-dba8ba36b745?w=400&h=400&fit=crop'}
size={56}
radius="md"
/>
<div style={{ flex: 1, minWidth: 0 }}>
{currentSong ? (
<>
<Text size="sm" fw={600} c="#0B4F78" truncate>{currentSong.judul}</Text>
<Text size="xs" c="#5A6C7D" truncate>{currentSong.artis}</Text>
<Text size="xs" c="#5A6C7D">{currentSong.artis}</Text>
</>
) : (
<Text size="sm" c="dimmed">Tidak ada lagu</Text>
@@ -261,31 +242,29 @@ const MusicPlayer = () => {
</div>
</Group>
{/* Controls + Progress */}
<Stack gap="xs" style={{ flex: 2, width: '100%' }} align="center">
<Group gap="sm">
<Stack gap="xs" style={{ flex: 1 }} align="center">
<Group gap="md">
<ActionIcon
variant={isShuffle ? 'filled' : 'subtle'}
color="#0B4F78"
onClick={toggleShuffleHandler}
radius="xl"
size={48}
>
{isShuffle ? <IconArrowsShuffle size={18} /> : <IconX size={18} />}
</ActionIcon>
<ActionIcon variant="light" color="#0B4F78" size={48} radius="xl" onClick={skipBack}>
<ActionIcon variant="light" color="#0B4F78" size={40} radius="xl" onClick={skipBack}>
<IconPlayerSkipBackFilled size={20} />
</ActionIcon>
<ActionIcon
variant="filled"
color="#0B4F78"
size={48}
size={56}
radius="xl"
onClick={togglePlayPauseHandler}
>
{isPlaying ? <IconPlayerPauseFilled size={26} /> : <IconPlayerPlayFilled size={26} />}
</ActionIcon>
<ActionIcon variant="light" color="#0B4F78" size={48} radius="xl" onClick={skipForward}>
<ActionIcon variant="light" color="#0B4F78" size={40} radius="xl" onClick={skipForward}>
<IconPlayerSkipForwardFilled size={20} />
</ActionIcon>
<ActionIcon
@@ -293,7 +272,6 @@ const MusicPlayer = () => {
color="#0B4F78"
onClick={toggleRepeatHandler}
radius="xl"
size="md"
>
{isRepeat ? <IconRepeat size={18} /> : <IconRepeatOff size={18} />}
</ActionIcon>
@@ -312,8 +290,7 @@ const MusicPlayer = () => {
</Group>
</Stack>
{/* Volume Control - Hidden on mobile, shown on md and up */}
<Group gap="xs" style={{ flex: 1 }} justify="flex-end" visibleFrom="md">
<Group gap="xs" style={{ flex: 1 }} justify="flex-end">
<ActionIcon variant="subtle" color="gray" onClick={toggleMuteHandler}>
{isMuted || volume === 0 ? <IconVolumeOff size={20} /> : <IconVolume size={20} />}
</ActionIcon>

View File

@@ -8,7 +8,6 @@ import {
Group,
Paper,
Slider,
Stack,
Text,
Transition
} from '@mantine/core';
@@ -62,18 +61,11 @@ export default function FixedPlayerBar() {
seek(value);
};
// Handle volume change - called continuously while dragging
// Handle volume change
const handleVolumeChange = (value: number) => {
console.log('[FixedPlayerBar] Volume changing:', value);
setVolume(value);
};
// Handle volume change commit - called when user releases slider
const handleVolumeChangeEnd = (value: number) => {
console.log('[FixedPlayerBar] Volume changed end:', value);
// Volume already set by onChange, no need to set again
};
// Handle shuffle toggle
const handleToggleShuffle = () => {
toggleShuffle();
@@ -101,19 +93,28 @@ export default function FixedPlayerBar() {
mt="md"
style={{
position: 'fixed',
top: '50%',
top: '50%', // Menempatkan titik atas ikon di tengah layar
left: '0px',
transform: 'translateY(-50%)',
transform: 'translateY(-50%)', // Menggeser ikon ke atas sebesar setengah tingginya sendiri agar benar-benar di tengah
borderBottomRightRadius: '20px',
borderTopRightRadius: '20px',
cursor: 'pointer',
transition: 'transform 0.2s ease',
zIndex: 40// Higher z-index
zIndex: 1
}}
onClick={handleRestorePlayer}
onMouseEnter={(e) => {
e.currentTarget.style.transform = 'translateY(-50%) scale(1.1)';
}}
onMouseLeave={(e) => {
e.currentTarget.style.transform = 'translateY(-50%)';
}}
>
<IconMusic size={24} color="white" />
<IconMusic size={28} color="white" />
</Button>
{/* Spacer to prevent content from being hidden behind player */}
<Box h={20} />
</>
);
}
@@ -130,166 +131,155 @@ export default function FixedPlayerBar() {
bottom={0}
left={0}
right={0}
p={{ base: 'xs', sm: 'sm' }}
shadow="xl"
p="sm"
shadow="lg"
style={{
zIndex: 40,
zIndex: 1,
borderTop: '1px solid rgba(0,0,0,0.1)',
backgroundColor: 'rgba(255, 255, 255, 0.95)',
backdropFilter: 'blur(10px)',
}}
>
<Flex align="center" gap={{ base: 'xs', sm: 'md' }} justify="space-between">
<Flex align="center" gap="md" justify="space-between">
{/* Song Info - Left */}
<Group gap="xs" flex={{ base: 2, sm: 1 }} style={{ minWidth: 0 }} wrap="nowrap">
<Group gap="sm" flex={1} style={{ minWidth: 0 }}>
<Avatar
src={currentSong.coverImage?.link || ''}
alt={currentSong.judul}
size={"36"}
size={40}
radius="sm"
imageProps={{ loading: 'lazy' }}
/>
<Box style={{ minWidth: 0, flex: 1 }}>
<Text fz={{ base: 'xs', sm: 'sm' }} fw={600} truncate>
<Box style={{ minWidth: 0 }}>
<Text fz="sm" fw={600} truncate>
{currentSong.judul}
</Text>
<Text fz="10px" c="dimmed" truncate>
<Text fz="xs" c="dimmed" truncate>
{currentSong.artis}
</Text>
</Box>
</Group>
{/* Controls - Center */}
<Group gap={"xs"} flex={{ base: 1, sm: 2 }} justify="center" wrap="nowrap">
{/* Shuffle - Desktop Only */}
<ActionIcon
variant={isShuffle ? 'filled' : 'subtle'}
color={isShuffle ? '#0B4F78' : 'gray'}
size={"md"}
onClick={handleToggleShuffle}
visibleFrom="sm"
>
<IconArrowsShuffle size={18} />
</ActionIcon>
{/* Controls + Progress - Center */}
<Group gap="xs" flex={2} justify="center">
{/* Control Buttons */}
<Group gap="xs">
<ActionIcon
variant={isShuffle ? 'filled' : 'subtle'}
color={isShuffle ? 'blue' : 'gray'}
size="lg"
onClick={handleToggleShuffle}
title="Shuffle"
>
<IconArrowsShuffle size={18} />
</ActionIcon>
<ActionIcon
variant="subtle"
color="gray"
size={"md"}
onClick={playPrev}
>
<IconPlayerSkipBackFilled size={20} />
</ActionIcon>
<ActionIcon
variant="subtle"
color="gray"
size="lg"
onClick={playPrev}
title="Previous"
>
<IconPlayerSkipBackFilled size={20} />
</ActionIcon>
<ActionIcon
variant="filled"
color="#0B4F78"
size={"lg"}
radius="xl"
onClick={togglePlayPause}
>
{isPlaying ? (
<IconPlayerPauseFilled size={24} />
) : (
<IconPlayerPlayFilled size={24} />
)}
</ActionIcon>
<ActionIcon
variant="filled"
color={isPlaying ? 'blue' : 'gray'}
size="xl"
radius="xl"
onClick={togglePlayPause}
title={isPlaying ? 'Pause' : 'Play'}
>
{isPlaying ? (
<IconPlayerPauseFilled size={24} />
) : (
<IconPlayerPlayFilled size={24} />
)}
</ActionIcon>
<ActionIcon
variant="subtle"
color="gray"
size={"md"}
onClick={playNext}
>
<IconPlayerSkipForwardFilled size={20} />
</ActionIcon>
<ActionIcon
variant="subtle"
color="gray"
size="lg"
onClick={playNext}
title="Next"
>
<IconPlayerSkipForwardFilled size={20} />
</ActionIcon>
{/* Repeat - Desktop Only */}
<ActionIcon
variant={isRepeat ? 'filled' : 'subtle'}
color={isRepeat ? '#0B4F78' : 'gray'}
size={"md"}
onClick={toggleRepeat}
visibleFrom="sm"
>
{isRepeat ? <IconRepeat size={18} /> : <IconRepeatOff size={18} />}
</ActionIcon>
<ActionIcon
variant="subtle"
color={isRepeat ? 'blue' : 'gray'}
size="lg"
onClick={toggleRepeat}
title={isRepeat ? 'Repeat On' : 'Repeat Off'}
>
{isRepeat ? <IconRepeat size={18} /> : <IconRepeatOff size={18} />}
</ActionIcon>
</Group>
{/* Progress Bar - Desktop Only */}
<Box w={150} ml="md" visibleFrom="md">
{/* Progress Bar - Desktop */}
<Box w={200} display={{ base: 'none', md: 'block' }}>
<Slider
value={currentTime}
max={duration || 100}
onChange={handleSeek}
size="xs"
color="#0B4F78"
size="sm"
color="blue"
label={(value) => formatTime(value)}
/>
</Box>
</Group>
{/* Right Controls - Volume + Close */}
<Group gap={4} flex={1} justify="flex-end" wrap="nowrap">
{/* Volume Control - Tablet/Desktop */}
<Group gap="xs" flex={1} justify="flex-end">
<Box
onMouseEnter={() => setShowVolume(true)}
onMouseLeave={() => setShowVolume(false)}
pos="relative"
visibleFrom="sm"
>
<ActionIcon
variant="subtle"
color={isMuted ? 'red' : 'gray'}
size="lg"
onClick={toggleMute}
title={isMuted ? 'Unmute' : 'Mute'}
>
{isMuted ? <IconVolumeOff size={18} /> : <IconVolume size={18} />}
{isMuted ? (
<IconVolumeOff size={18} />
) : (
<IconVolume size={18} />
)}
</ActionIcon>
<Transition
mounted={showVolume}
transition="scale-y"
duration={200}
timingFunction="ease"
>
{(styles) => (
{(style) => (
<Paper
style={{
...styles,
...style,
position: 'absolute',
bottom: '100%',
right: 0,
marginBottom: '10px',
padding: '12px',
zIndex: 40,
mb: 'xs',
p: 'sm',
zIndex: 1001,
}}
shadow="md"
withBorder
>
<Stack gap="xs" align="center">
<Text size="xs" c="#5A6C7D" ta="center">{isMuted ? 0 : volume}%</Text>
<Box
style={{
height: '120px',
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
}}
>
<Slider
value={isMuted ? 0 : volume}
max={100}
onChange={handleVolumeChange}
onChangeEnd={handleVolumeChangeEnd}
w={120}
color="#0B4F78"
size="sm"
label={(value) => `${value}%`}
style={{
transform: 'rotate(-90deg)',
transformOrigin: 'center',
}}
/>
</Box>
</Stack>
<Slider
value={isMuted ? 0 : volume}
max={100}
onChange={handleVolumeChange}
h={100}
color="blue"
size="sm"
/>
</Paper>
)}
</Transition>
@@ -298,29 +288,30 @@ export default function FixedPlayerBar() {
<ActionIcon
variant="subtle"
color="gray"
size={"md"}
size="lg"
onClick={handleMinimizePlayer}
title="Minimize player"
>
<IconX size={18} />
</ActionIcon>
</Group>
</Flex>
{/* Progress Bar - Mobile (Base) */}
<Box px="xs" mt={4} hiddenFrom="md">
{/* Progress Bar - Mobile */}
<Box mt="xs" display={{ base: 'block', md: 'none' }}>
<Slider
value={currentTime}
max={duration || 100}
onChange={handleSeek}
size="xs"
color="#0B4F78"
size="sm"
color="blue"
label={(value) => formatTime(value)}
/>
</Box>
</Paper>
{/* Spacer to prevent content from being hidden behind player */}
<Box h={{ base: 70, sm: 80 }} />
<Box h={80} />
</>
);
}

View File

@@ -0,0 +1,117 @@
import { Skeleton, Stack, Box, Group } from '@mantine/core'
export function PaguTableSkeleton() {
return (
<Box>
<Skeleton height={28} width="60%" mb="md" />
<Stack gap="xs">
{/* Header */}
<Group justify="space-between">
<Skeleton height={20} width="40%" />
<Skeleton height={20} width="30%" />
</Group>
{/* Section headers */}
<Skeleton height={24} width="100%" mt="md" />
<Skeleton height={20} width="90%" />
<Skeleton height={20} width="85%" />
<Skeleton height={20} width="80%" />
<Skeleton height={24} width="100%" mt="md" />
<Skeleton height={20} width="90%" />
<Skeleton height={20} width="85%" />
<Skeleton height={24} width="100%" mt="md" />
<Skeleton height={20} width="90%" />
</Stack>
</Box>
)
}
export function RealisasiTableSkeleton() {
return (
<Box>
<Skeleton height={28} width="70%" mb="md" />
<Stack gap="xs">
{/* Header */}
<Group justify="space-between">
<Skeleton height={20} width="40%" />
<Skeleton height={20} width="20%" />
<Skeleton height={20} width="10%" />
</Group>
{/* Rows */}
{[1, 2, 3, 4, 5].map((i) => (
<Group key={i} justify="space-between">
<Skeleton height={20} width="50%" />
<Skeleton height={20} width="25%" />
<Skeleton height={24} width="15%" radius="xl" />
</Group>
))}
</Stack>
</Box>
)
}
export function GrafikRealisasiSkeleton() {
return (
<Box>
<Skeleton height={28} width="65%" mb="md" />
<Stack gap="lg">
{[1, 2, 3].map((i) => (
<Stack key={i} gap="xs">
<Group justify="space-between">
<Skeleton height={20} width="40%" />
<Skeleton height={20} width="15%" />
</Group>
<Skeleton height={16} width="100%" />
<Skeleton height={12} width="100%" mt={4} />
<Skeleton height={16} width="100%" radius="xl" />
</Stack>
))}
</Stack>
</Box>
)
}
export function SummaryCardsSkeleton() {
return (
<Stack gap="lg">
<Skeleton height={28} width="50%" mb="sm" />
{[1, 2, 3].map((i) => (
<Stack key={i} gap="xs" p="md" style={{ border: '1px solid #e5e7eb', borderRadius: 8 }}>
<Group justify="space-between">
<Skeleton height={20} width="35%" />
<Skeleton height={20} width="20%" />
</Group>
<Skeleton height={16} width="100%" />
<Skeleton height={12} width="100%" mt={4} />
<Skeleton height={16} width="100%" radius="xl" />
</Stack>
))}
</Stack>
)
}
export function ApbdesMainSkeleton() {
return (
<Stack gap="xl">
{/* Title */}
<Skeleton height={48} width="40%" mx="auto" />
<Skeleton height={24} width="60%" mx="auto" />
{/* Select */}
<Skeleton height={42} width={220} mx="auto" />
{/* Summary Cards */}
<SummaryCardsSkeleton />
{/* Tables and Charts */}
<Stack gap="lg">
<PaguTableSkeleton />
<RealisasiTableSkeleton />
<GrafikRealisasiSkeleton />
</Stack>
</Stack>
)
}

View File

@@ -1,7 +1,8 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
/* eslint-disable react-hooks/exhaustive-deps */
'use client'
import apbdes from '@/app/admin/(dashboard)/_state/landing-page/apbdes'
import apbdesState from '@/app/admin/(dashboard)/_state/landing-page/apbdes'
import colors from '@/con/colors'
import {
Box,
@@ -12,30 +13,43 @@ import {
SimpleGrid,
Stack,
Text,
Title
Title,
LoadingOverlay,
Transition,
} from '@mantine/core'
import { motion } from 'framer-motion'
import Link from 'next/link'
import { useEffect, useState } from 'react'
import { useProxy } from 'valtio/utils'
import { ApbdesMainSkeleton } from './components/apbdesSkeleton'
import ComparisonChart from './lib/comparisonChart'
import GrafikRealisasi from './lib/grafikRealisasi'
import PaguTable from './lib/paguTable'
import RealisasiTable from './lib/realisasiTable'
const MotionStack = motion.create(Stack)
function Apbdes() {
const state = useProxy(apbdes)
const state = useProxy(apbdesState)
const [selectedYear, setSelectedYear] = useState<string | null>(null)
const [isLoading, setIsLoading] = useState(true)
const [isChangingYear, setIsChangingYear] = useState(false)
const textHeading = {
title: 'APBDes',
des: 'Transparansi APBDes Darmasaba adalah langkah nyata menuju tata kelola desa yang bersih, terbuka, dan bertanggung jawab.'
des: 'Transparansi APBDes Darmasaba adalah langkah nyata menuju tata kelola desa yang bersih, terbuka, dan bertanggung jawab.',
}
useEffect(() => {
const loadData = async () => {
try {
setIsLoading(true)
await state.findMany.load()
} catch (error) {
console.error('Error loading data:', error)
} finally {
setIsLoading(false)
}
}
loadData()
@@ -51,7 +65,7 @@ function Apbdes() {
)
)
.sort((a, b) => b - a)
.map(year => ({
.map((year) => ({
value: year.toString(),
label: `Tahun ${year}`,
}))
@@ -60,168 +74,190 @@ function Apbdes() {
if (years.length > 0 && !selectedYear) {
setSelectedYear(years[0].value)
}
}, [years, selectedYear])
}, [years])
const currentApbdes = dataAPBDes.length > 0
? dataAPBDes.find((item: any) => item?.tahun?.toString() === selectedYear) || dataAPBDes[0]
? (dataAPBDes.find((item: any) => item?.tahun?.toString() === selectedYear) || dataAPBDes[0])
: null
// eslint-disable-next-line @typescript-eslint/no-unused-vars
const previewData = (state.findMany.data || []).slice(0, 3)
const handleYearChange = (value: string | null) => {
if (value !== selectedYear) {
setIsChangingYear(true)
setSelectedYear(value)
setTimeout(() => setIsChangingYear(false), 500)
}
}
return (
<Stack p="sm" gap="xl" bg={colors.Bg}>
<Divider c="gray.3" size="sm" />
{/* 📌 HEADING */}
<Box mt="xl">
<Stack gap="sm">
<Title
order={1}
ta="center"
c={colors['blue-button']}
fz={{ base: '2rem', md: '3.6rem' }}
lh={{ base: 1.2, md: 1.1 }}
<Stack p="sm" gap="xl" bg={colors.Bg} pos="relative">
<LoadingOverlay
visible={isLoading}
zIndex={1000}
overlayProps={{ radius: 'sm', blur: 2 }}
loaderProps={{ color: colors['blue-button'], type: 'dots' }}
/>
<Transition mounted={!isLoading} transition="fade" duration={600}>
{(styles) => (
<MotionStack
style={styles}
gap="xl"
>
{textHeading.title}
</Title>
<Text
ta="center"
fz={{ base: '1rem', md: '1.25rem' }}
lh={{ base: 1.5, md: 1.55 }}
c="black"
>
{textHeading.des}
</Text>
</Stack>
</Box>
{/* Button Lihat Semua */}
<Group justify="center">
<Button
component={Link}
href="/darmasaba/apbdes"
radius="xl"
size="lg"
variant="gradient"
gradient={{ from: "#26667F", to: "#124170" }}
>
Lihat Semua Data
</Button>
</Group>
{/* COMBOBOX */}
<Box px={{ base: 'md', md: "sm" }}>
<Select
label={<Text fw={600} fz="sm">Pilih Tahun APBDes</Text>}
placeholder="Pilih tahun"
value={selectedYear}
onChange={setSelectedYear}
data={years}
w={{ base: '100%', sm: 220 }}
searchable
clearable
nothingFoundMessage="Tidak ada tahun tersedia"
/>
</Box>
{/* Tabel & Grafik - Hanya tampilkan jika ada data */}
{currentApbdes && currentApbdes.items?.length > 0 ? (
<Box px={{ base: 'md', md: 'sm' }} mb="xl">
<SimpleGrid cols={{ base: 1, sm: 3 }}>
<PaguTable apbdesData={currentApbdes} />
<RealisasiTable apbdesData={currentApbdes} />
<GrafikRealisasi apbdesData={currentApbdes} />
</SimpleGrid>
</Box>
) : currentApbdes ? (
<Box px={{ base: 'md', md: 100 }} py="md" mb="xl">
<Text fz="sm" c="dimmed" ta="center" lh={1.5}>
Tidak ada data item untuk tahun yang dipilih.
</Text>
</Box>
) : null}
{/* GRID - Card Preview
{state.findMany.loading ? (
<Center mx={{ base: 'md', md: 100 }} mih={200} pb="xl">
<Loader size="lg" color="blue" />
</Center>
) : previewData.length === 0 ? (
<Center mx={{ base: 'md', md: 100 }} mih={200} pb="xl">
<Stack align="center" gap="xs">
<Text fz="lg" c="dimmed" lh={1.4}>
Belum ada data APBDes yang tersedia
</Text>
<Text fz="sm" c="dimmed" lh={1.4}>
Data akan ditampilkan di sini setelah diunggah
</Text>
</Stack>
</Center>
) : (
<SimpleGrid
mx={{ base: 'md', md: 100 }}
cols={{ base: 1, sm: 3 }}
spacing="lg"
pb="xl"
>
{previewData.map((v, k) => (
<Box
key={k}
pos="relative"
style={{
backgroundImage: `url(${v.image?.link || ''})`,
backgroundSize: 'cover',
backgroundPosition: 'center',
borderRadius: 16,
height: 360,
overflow: 'hidden',
}}
>
<Box pos="absolute" inset={0} bg="rgba(0,0,0,0.45)" style={{ borderRadius: 16 }} />
<Stack gap="xs" justify="space-between" h="100%" p="xl" pos="relative">
<Text
c="white"
fw={600}
fz={{ base: 'lg', md: 'xl' }}
<Divider c="gray.3" size="sm" />
{/* 📌 HEADING */}
<Box mt="xl">
<Stack gap="sm">
<Title
order={1}
ta="center"
lh={1.35}
lineClamp={2}
c={colors['blue-button']}
fz={{ base: '2rem', md: '3.6rem' }}
lh={{ base: 1.2, md: 1.1 }}
>
{v.name || `APBDes Tahun ${v.tahun}`}
</Text>
{textHeading.title}
</Title>
<Text
fw={700}
c="white"
fz={{ base: '2.4rem', md: '3.2rem' }}
ta="center"
lh={1}
style={{ textShadow: '0 2px 8px rgba(0,0,0,0.6)' }}
fz={{ base: '1rem', md: '1.25rem' }}
lh={{ base: 1.5, md: 1.55 }}
c="black"
maw={800}
mx="auto"
>
{v.jumlah || '-'}
{textHeading.des}
</Text>
<Center>
<ActionIcon
component={Link}
href={v.file?.link || ''}
radius="xl"
size="xl"
variant="gradient"
gradient={{ from: '#1C6EA4', to: '#1C6EA4' }}
>
<IconDownload size={20} color="white" />
</ActionIcon>
</Center>
</Stack>
</Box>
))}
</SimpleGrid>
)} */}
{/* Button Lihat Semua */}
<Group justify="center">
<Button
component={Link}
href="/darmasaba/apbdes"
radius="xl"
size="lg"
variant="gradient"
gradient={{ from: '#26667F', to: '#124170' }}
style={{
transition: 'transform 0.2s ease, box-shadow 0.2s ease',
':hover': {
transform: 'translateY(-2px)',
boxShadow: '0 4px 12px rgba(38, 102, 127, 0.4)',
},
}}
>
Lihat Semua Data
</Button>
</Group>
{/* COMBOBOX */}
<Box px={{ base: 'md', md: 'sm' }}>
<Select
label={<Text fw={600} fz="sm">Pilih Tahun APBDes</Text>}
placeholder="Pilih tahun"
value={selectedYear}
onChange={handleYearChange}
data={years}
w={{ base: '100%', sm: 220 }}
searchable
clearable
nothingFoundMessage="Tidak ada tahun tersedia"
disabled={isChangingYear}
/>
</Box>
{/* Tables & Charts */}
{currentApbdes && currentApbdes.items && currentApbdes.items.length > 0 ? (
<Box px={{ base: 'md', md: 'sm' }} mb="xl">
<Transition
mounted={!isChangingYear}
transition="slide-up"
duration={400}
timingFunction="ease"
>
{(styles) => (
<SimpleGrid
cols={{ base: 1, sm: 3 }}
style={styles}
>
<MotionStack
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ duration: 0.4, delay: 0.1 }}
>
<PaguTable apbdesData={currentApbdes as any} />
</MotionStack>
<MotionStack
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ duration: 0.4, delay: 0.2 }}
>
<RealisasiTable apbdesData={currentApbdes as any} />
</MotionStack>
<MotionStack
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ duration: 0.4, delay: 0.3 }}
>
<GrafikRealisasi apbdesData={currentApbdes as any} />
</MotionStack>
</SimpleGrid>
)}
</Transition>
{/* Comparison Chart */}
<Box mt="lg">
<Transition
mounted={!isChangingYear}
transition="slide-up"
duration={400}
timingFunction="ease"
>
{(styles) => (
<MotionStack
style={styles}
initial={{ opacity: 0, y: 20 }}
animate={{ opacity: 1, y: 0 }}
transition={{ duration: 0.4, delay: 0.4 }}
>
<ComparisonChart apbdesData={currentApbdes as any} />
</MotionStack>
)}
</Transition>
</Box>
</Box>
) : currentApbdes ? (
<Box px={{ base: 'md', md: 100 }} py="xl" mb="xl">
<Stack align="center" gap="sm">
<Text fz="2rem">📊</Text>
<Text fz="sm" c="dimmed" ta="center" lh={1.5}>
Tidak ada data item untuk tahun yang dipilih.
</Text>
</Stack>
</Box>
) : null}
{/* Loading State for Year Change */}
<Transition mounted={isChangingYear} transition="fade" duration={200}>
{(styles) => (
<Box
px={{ base: 'md', md: 'sm' }}
mb="xl"
style={styles}
>
<ApbdesMainSkeleton />
</Box>
)}
</Transition>
</MotionStack>
)}
</Transition>
</Stack>
)
}
export default Apbdes
export default Apbdes

View File

@@ -0,0 +1,229 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Paper, Title, Box, Text, Stack, Group, rem } from '@mantine/core'
import {
BarChart,
Bar,
XAxis,
YAxis,
CartesianGrid,
Tooltip,
Legend,
ResponsiveContainer,
Cell,
} from 'recharts'
import { APBDes, APBDesItem } from '../types/apbdes'
interface ComparisonChartProps {
apbdesData: APBDes
}
export default function ComparisonChart({ apbdesData }: ComparisonChartProps) {
const items = apbdesData?.items || []
const tahun = apbdesData?.tahun || new Date().getFullYear()
const pendapatan = items.filter((i: APBDesItem) => i.tipe === 'pendapatan')
const belanja = items.filter((i: APBDesItem) => i.tipe === 'belanja')
const pembiayaan = items.filter((i: APBDesItem) => i.tipe === 'pembiayaan')
const totalPendapatan = pendapatan.reduce((sum, i) => sum + i.anggaran, 0)
const totalBelanja = belanja.reduce((sum, i) => sum + i.anggaran, 0)
const totalPembiayaan = pembiayaan.reduce((sum, i) => sum + i.anggaran, 0)
// Hitung total realisasi dari realisasiItems (konsisten dengan RealisasiTable)
const totalPendapatanRealisasi = pendapatan.reduce(
(sum, i) => {
if (i.realisasiItems && i.realisasiItems.length > 0) {
return sum + i.realisasiItems.reduce((sumReal, real) => sumReal + (real.jumlah || 0), 0)
}
return sum
},
0
)
const totalBelanjaRealisasi = belanja.reduce(
(sum, i) => {
if (i.realisasiItems && i.realisasiItems.length > 0) {
return sum + i.realisasiItems.reduce((sumReal, real) => sumReal + (real.jumlah || 0), 0)
}
return sum
},
0
)
const totalPembiayaanRealisasi = pembiayaan.reduce(
(sum, i) => {
if (i.realisasiItems && i.realisasiItems.length > 0) {
return sum + i.realisasiItems.reduce((sumReal, real) => sumReal + (real.jumlah || 0), 0)
}
return sum
},
0
)
const formatRupiah = (value: number) => {
if (value >= 1000000000) {
return `Rp ${(value / 1000000000).toFixed(1)}B`
}
if (value >= 1000000) {
return `Rp ${(value / 1000000).toFixed(1)}Jt`
}
if (value >= 1000) {
return `Rp ${(value / 1000).toFixed(0)}Rb`
}
return `Rp ${value.toFixed(0)}`
}
const data = [
{
name: 'Pendapatan',
pagu: totalPendapatan,
realisasi: totalPendapatanRealisasi,
fill: '#40c057',
},
{
name: 'Belanja',
pagu: totalBelanja,
realisasi: totalBelanjaRealisasi,
fill: '#fa5252',
},
{
name: 'Pembiayaan',
pagu: totalPembiayaan,
realisasi: totalPembiayaanRealisasi,
fill: '#fd7e14',
},
]
const CustomTooltip = ({ active, payload }: any) => {
if (active && payload && payload.length) {
const data = payload[0].payload
return (
<Box
bg="white"
p="md"
style={{
border: '1px solid #e5e7eb',
borderRadius: 8,
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.1)',
}}
>
<Stack gap="xs">
<Text fw={700} c="gray.8" fz="sm">
{data.name}
</Text>
<Group justify="space-between" gap="lg">
<Text fz="xs" c="gray.6">
Pagu:
</Text>
<Text fz="xs" fw={700} c="blue.9">
{formatRupiah(data.pagu)}
</Text>
</Group>
<Group justify="space-between" gap="lg">
<Text fz="xs" c="gray.6">
Realisasi:
</Text>
<Text fz="xs" fw={700} c="green.9">
{formatRupiah(data.realisasi)}
</Text>
</Group>
{data.pagu > 0 && (
<Group justify="space-between" gap="lg">
<Text fz="xs" c="gray.6">
Persentase:
</Text>
<Text
fz="xs"
fw={700}
c={data.realisasi >= data.pagu ? 'teal' : 'blue'}
>
{((data.realisasi / data.pagu) * 100).toFixed(1)}%
</Text>
</Group>
)}
</Stack>
</Box>
)
}
return null
}
return (
<Paper
withBorder
p="lg"
radius="lg"
shadow="sm"
style={{
transition: 'box-shadow 0.3s ease',
':hover': {
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.1)',
},
}}
>
<Title
order={5}
mb="lg"
c="blue.9"
fz={{ base: '1rem', md: '1.1rem' }}
fw={700}
>
Perbandingan Pagu vs Realisasi {tahun}
</Title>
<Box style={{ width: '100%', height: 300 }}>
<ResponsiveContainer>
<BarChart
data={data}
margin={{ top: 20, right: 30, left: 0, bottom: 0 }}
barSize={60}
>
<CartesianGrid strokeDasharray="3 3" stroke="#e5e7eb" />
<XAxis
dataKey="name"
tick={{ fill: '#6b7280', fontSize: 12 }}
axisLine={{ stroke: '#e5e7eb' }}
/>
<YAxis
tickFormatter={formatRupiah}
tick={{ fill: '#6b7280', fontSize: 11 }}
axisLine={{ stroke: '#e5e7eb' }}
width={80}
/>
<Tooltip content={<CustomTooltip />} />
<Legend
wrapperStyle={{
paddingTop: rem(20),
fontSize: 12,
}}
/>
<Bar
name="Pagu"
dataKey="pagu"
fill="#228be6"
radius={[8, 8, 0, 0]}
>
{data.map((entry, index) => (
<Cell
key={`cell-pagu-${index}`}
fill={entry.fill}
opacity={0.7}
/>
))}
</Bar>
<Bar
name="Realisasi"
dataKey="realisasi"
fill="#40c057"
radius={[8, 8, 0, 0]}
/>
</BarChart>
</ResponsiveContainer>
</Box>
<Box mt="md">
<Text fz="xs" c="dimmed" ta="center">
*Geser cursor pada bar untuk melihat detail
</Text>
</Box>
</Paper>
)
}

View File

@@ -1,125 +1,224 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Paper, Title, Progress, Stack, Text, Group, Box } from '@mantine/core';
interface APBDesItem {
tipe: string | null;
anggaran: number;
realisasi?: number;
totalRealisasi?: number;
}
import { Paper, Title, Progress, Stack, Text, Group, Box, rem } from '@mantine/core'
import { IconArrowUpRight, IconArrowDownRight } from '@tabler/icons-react'
import { APBDes, APBDesItem, SummaryData } from '../types/apbdes'
interface SummaryProps {
title: string;
data: APBDesItem[];
title: string
data: APBDesItem[]
icon?: React.ReactNode
}
function Summary({ title, data }: SummaryProps) {
if (!data || data.length === 0) return null;
function Summary({ title, data, icon }: SummaryProps) {
if (!data || data.length === 0) return null
const totalAnggaran = data.reduce((s: number, i: APBDesItem) => s + i.anggaran, 0);
// Use realisasi field (already mapped from totalRealisasi in transformAPBDesData)
const totalRealisasi = data.reduce(
(s: number, i: APBDesItem) => s + (i.realisasi || i.totalRealisasi || 0),
0
);
const totalAnggaran = data.reduce((sum, i) => sum + i.anggaran, 0)
// Hitung total realisasi dari realisasiItems (konsisten dengan RealisasiTable)
const totalRealisasi = data.reduce((sum, i) => {
if (i.realisasiItems && i.realisasiItems.length > 0) {
return sum + i.realisasiItems.reduce((sumReal, real) => sumReal + (real.jumlah || 0), 0)
}
return sum
}, 0)
const persen =
totalAnggaran > 0 ? (totalRealisasi / totalAnggaran) * 100 : 0;
const persentase = totalAnggaran > 0 ? (totalRealisasi / totalAnggaran) * 100 : 0
// Format angka ke dalam format Rupiah
const formatRupiah = (angka: number) => {
return new Intl.NumberFormat('id-ID', {
style: 'currency',
currency: 'IDR',
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(angka);
};
}).format(angka)
}
// Tentukan warna berdasarkan persentase
const getProgressColor = (persen: number) => {
if (persen >= 100) return 'teal';
if (persen >= 80) return 'blue';
if (persen >= 60) return 'yellow';
return 'red';
};
if (persen >= 100) return 'teal'
if (persen >= 80) return 'blue'
if (persen >= 60) return 'yellow'
return 'red'
}
const getStatusMessage = (persen: number) => {
if (persen >= 100) {
return { text: 'Realisasi mencapai 100% dari anggaran', color: 'teal' }
}
if (persen >= 80) {
return { text: 'Realisasi baik, mendekati target', color: 'blue' }
}
if (persen >= 60) {
return { text: 'Realisasi cukup, perlu ditingkatkan', color: 'yellow' }
}
return { text: 'Realisasi rendah, perlu perhatian khusus', color: 'red' }
}
const statusMessage = getStatusMessage(persentase)
return (
<Box>
<Group justify="space-between" mb="xs">
<Text fw={600} fz="md">{title}</Text>
<Text fw={700} fz="lg" c={getProgressColor(persen)}>
{persen.toFixed(2)}%
</Text>
<Group gap="xs">
{icon}
<Text fw={700} fz="md" c="gray.8">{title}</Text>
</Group>
<Group gap="xs">
{persentase >= 100 ? (
<IconArrowUpRight
size={18}
color="var(--mantine-color-teal-7)"
stroke={2.5}
/>
) : persentase < 60 ? (
<IconArrowDownRight
size={18}
color="var(--mantine-color-red-7)"
stroke={2.5}
/>
) : null}
<Text
fw={700}
fz="lg"
c={getProgressColor(persentase)}
style={{
minWidth: 60,
textAlign: 'right',
}}
>
{persentase.toFixed(1)}%
</Text>
</Group>
</Group>
<Text fz="sm" c="dimmed" mb="xs">
Realisasi: {formatRupiah(totalRealisasi)} / Anggaran: {formatRupiah(totalAnggaran)}
<Text fz="xs" c="gray.6" mb="sm" lh={1.5}>
Realisasi: <Text component="span" fw={700} c="blue.9">{formatRupiah(totalRealisasi)}</Text>
{' '}/ Anggaran: <Text component="span" fw={700} c="gray.7">{formatRupiah(totalAnggaran)}</Text>
</Text>
<Progress
value={persen}
value={persentase}
size="xl"
radius="xl"
color={getProgressColor(persen)}
striped={persen < 100}
animated={persen < 100}
color={getProgressColor(persentase)}
striped={persentase < 100}
animated={persentase < 100}
mb="xs"
/>
{persen >= 100 && (
<Text fz="xs" c="teal" mt="xs" fw={500}>
Realisasi mencapai 100% dari anggaran
</Text>
)}
{persen < 100 && persen >= 80 && (
<Text fz="xs" c="blue" mt="xs" fw={500}>
Realisasi baik, mendekati target
</Text>
)}
{persen < 80 && persen >= 60 && (
<Text fz="xs" c="yellow" mt="xs" fw={500}>
Realisasi cukup, perlu ditingkatkan
</Text>
)}
{persen < 60 && (
<Text fz="xs" c="red" mt="xs" fw={500}>
Realisasi rendah, perlu perhatian khusus
</Text>
)}
<Text
fz="xs"
c={statusMessage.color as any}
fw={600}
style={{
backgroundColor: `var(--mantine-color-${statusMessage.color}-0)`,
padding: '6px 10px',
borderRadius: 6,
display: 'inline-block',
}}
>
{persentase >= 100 && '✓ '}{statusMessage.text}
</Text>
</Box>
);
)
}
export default function GrafikRealisasi({
apbdesData,
}: {
apbdesData: {
tahun?: number | null;
items?: APBDesItem[] | null;
[key: string]: any;
};
}) {
const items = apbdesData?.items || [];
const tahun = apbdesData?.tahun || new Date().getFullYear();
interface GrafikRealisasiProps {
apbdesData: APBDes
}
const pendapatan = items.filter((i: APBDesItem) => i.tipe === 'pendapatan');
const belanja = items.filter((i: APBDesItem) => i.tipe === 'belanja');
const pembiayaan = items.filter((i: APBDesItem) => i.tipe === 'pembiayaan');
export default function GrafikRealisasi({ apbdesData }: GrafikRealisasiProps) {
const items = apbdesData?.items || []
const tahun = apbdesData?.tahun || new Date().getFullYear()
const pendapatan = items.filter((i: APBDesItem) => i.tipe === 'pendapatan')
const belanja = items.filter((i: APBDesItem) => i.tipe === 'belanja')
const pembiayaan = items.filter((i: APBDesItem) => i.tipe === 'pembiayaan')
return (
<Paper withBorder p="md" radius="md">
<Title order={5} mb="md">
<Paper
withBorder
p="lg"
radius="lg"
shadow="sm"
style={{
transition: 'box-shadow 0.3s ease',
':hover': {
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.1)',
},
}}
h={"100%"}
>
<Title
order={5}
mb="lg"
c="blue.9"
fz={{ base: '1rem', md: '1.1rem' }}
fw={700}
>
GRAFIK REALISASI APBDes {tahun}
</Title>
<Stack gap="lg" mb="lg">
<Summary title="💰 Pendapatan" data={pendapatan} />
<Summary title="💸 Belanja" data={belanja} />
<Summary title="📊 Pembiayaan" data={pembiayaan} />
<Stack gap="xl">
<Summary
title="Pendapatan"
data={pendapatan}
icon={
<Box
style={{
width: 32,
height: 32,
borderRadius: 8,
backgroundColor: 'var(--mantine-color-green-0)',
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
}}
>
<Text fz="lg">💰</Text>
</Box>
}
/>
<Summary
title="Belanja"
data={belanja}
icon={
<Box
style={{
width: 32,
height: 32,
borderRadius: 8,
backgroundColor: 'var(--mantine-color-red-0)',
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
}}
>
<Text fz="lg">💸</Text>
</Box>
}
/>
<Summary
title="Pembiayaan"
data={pembiayaan}
icon={
<Box
style={{
width: 32,
height: 32,
borderRadius: 8,
backgroundColor: 'var(--mantine-color-orange-0)',
display: 'flex',
alignItems: 'center',
justifyContent: 'center',
}}
>
<Text fz="lg">📊</Text>
</Box>
}
/>
</Stack>
</Paper>
);
}
)
}

View File

@@ -1,66 +1,180 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Paper, Table, Title, Text } from '@mantine/core';
import { Paper, Table, Title, Box, ScrollArea, Badge } from '@mantine/core'
import { APBDes, APBDesItem } from '../types/apbdes'
function Section({ title, data }: any) {
if (!data || data.length === 0) return null;
interface SectionProps {
title: string
data: APBDesItem[]
badgeColor?: string
}
function Section({ title, data, badgeColor = 'blue' }: SectionProps) {
if (!data || data.length === 0) return null
return (
<>
<Table.Tr bg="gray.0">
<Table.Td colSpan={2}>
<Text fw={700} fz={{ base: 'xs', sm: 'sm' }}>{title}</Text>
<Badge color={badgeColor} variant="light" size="lg" fw={600}>
{title}
</Badge>
</Table.Td>
</Table.Tr>
{data.map((item: any) => (
<Table.Tr key={item.id}>
<Table.Td>
<Text fz={{ base: 'xs', sm: 'sm' }} lineClamp={2}>
{item.kode} - {item.uraian}
</Text>
{data.map((item, index) => (
<Table.Tr
key={item.id}
bg={index % 2 === 1 ? 'gray.50' : 'white'}
style={{
transition: 'background-color 0.2s ease',
':hover': {
backgroundColor: 'var(--mantine-color-blue-0)',
},
}}
>
<Table.Td style={{ borderBottom: '1px solid #e5e7eb' }}>
<Box style={{ display: 'flex', gap: 8, alignItems: 'center' }}>
<span style={{
fontWeight: 500,
color: 'var(--mantine-color-gray-7)',
minWidth: 80,
}}>
{item.kode}
</span>
<span style={{
color: 'var(--mantine-color-gray-6)',
fontSize: '0.9rem',
}}>
{item.uraian}
</span>
</Box>
</Table.Td>
<Table.Td ta="right">
<Text fz={{ base: 'xs', sm: 'sm' }} fw={500} style={{ whiteSpace: 'nowrap' }}>
Rp {item.anggaran.toLocaleString('id-ID')}
</Text>
<Table.Td
ta="right"
style={{
borderBottom: '1px solid #e5e7eb',
fontWeight: 600,
color: 'var(--mantine-color-blue-7)',
}}
>
Rp {item.anggaran.toLocaleString('id-ID')}
</Table.Td>
</Table.Tr>
))}
</>
);
)
}
export default function PaguTable({ apbdesData }: any) {
const items = apbdesData.items || [];
interface PaguTableProps {
apbdesData: APBDes
}
const title =
apbdesData.tahun
? `PAGU APBDes Tahun ${apbdesData.tahun}`
: 'PAGU APBDes';
export default function PaguTable({ apbdesData }: PaguTableProps) {
const items = apbdesData.items || []
const pendapatan = items.filter((i: any) => i.tipe === 'pendapatan');
const belanja = items.filter((i: any) => i.tipe === 'belanja');
const pembiayaan = items.filter((i: any) => i.tipe === 'pembiayaan');
const title = apbdesData.tahun
? `PAGU APBDes Tahun ${apbdesData.tahun}`
: 'PAGU APBDes'
const pendapatan = items.filter((i: APBDesItem) => i.tipe === 'pendapatan')
const belanja = items.filter((i: APBDesItem) => i.tipe === 'belanja')
const pembiayaan = items.filter((i: APBDesItem) => i.tipe === 'pembiayaan')
// Calculate totals
const totalPendapatan = pendapatan.reduce((sum, i) => sum + i.anggaran, 0)
const totalBelanja = belanja.reduce((sum, i) => sum + i.anggaran, 0)
const totalPembiayaan = pembiayaan.reduce((sum, i) => sum + i.anggaran, 0)
return (
<Paper withBorder p={{ base: 'sm', sm: 'md' }} radius="md">
<Title order={5} mb="md" fz={{ base: 'sm', sm: 'md' }}>{title}</Title>
<Paper
withBorder
p="md"
radius="lg"
shadow="sm"
style={{
transition: 'box-shadow 0.3s ease',
':hover': {
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.1)',
},
}}
h={"100%"}
>
<Title
order={5}
mb="md"
c="blue.9"
fz={{ base: '1rem', md: '1.1rem' }}
fw={700}
>
{title}
</Title>
<Table.ScrollContainer minWidth={280}>
<Table verticalSpacing="xs">
<ScrollArea offsetScrollbars type="hover">
<Table
horizontalSpacing="md"
verticalSpacing="xs"
layout="fixed"
>
<Table.Thead>
<Table.Tr>
<Table.Th fz={{ base: 'xs', sm: 'sm' }}>Uraian</Table.Th>
<Table.Th ta="right" fz={{ base: 'xs', sm: 'sm' }}>Anggaran (Rp)</Table.Th>
<Table.Tr bg="blue.9">
<Table.Th c="white" fw={600} style={{ minWidth: '60%' }}>
Uraian
</Table.Th>
<Table.Th
c="white"
fw={600}
ta="right"
style={{ minWidth: '40%' }}
>
Anggaran (Rp)
</Table.Th>
</Table.Tr>
</Table.Thead>
<Table.Tbody>
<Section title="1) PENDAPATAN" data={pendapatan} />
<Section title="2) BELANJA" data={belanja} />
<Section title="3) PEMBIAYAAN" data={pembiayaan} />
<Section
title="1) PENDAPATAN"
data={pendapatan}
badgeColor="green"
/>
{totalPendapatan > 0 && (
<Table.Tr bg="green.0" fw={700}>
<Table.Td>Total Pendapatan</Table.Td>
<Table.Td ta="right">
Rp {totalPendapatan.toLocaleString('id-ID')}
</Table.Td>
</Table.Tr>
)}
<Section
title="2) BELANJA"
data={belanja}
badgeColor="red"
/>
{totalBelanja > 0 && (
<Table.Tr bg="red.0" fw={700}>
<Table.Td>Total Belanja</Table.Td>
<Table.Td ta="right">
Rp {totalBelanja.toLocaleString('id-ID')}
</Table.Td>
</Table.Tr>
)}
<Section
title="3) PEMBIAYAAN"
data={pembiayaan}
badgeColor="orange"
/>
{totalPembiayaan > 0 && (
<Table.Tr bg="orange.0" fw={700}>
<Table.Td>Total Pembiayaan</Table.Td>
<Table.Td ta="right">
Rp {totalPembiayaan.toLocaleString('id-ID')}
</Table.Td>
</Table.Tr>
)}
</Table.Tbody>
</Table>
</Table.ScrollContainer>
</ScrollArea>
</Paper>
);
}
)
}

View File

@@ -1,92 +1,212 @@
/* eslint-disable @typescript-eslint/no-explicit-any */
import { Paper, Table, Title, Badge, Text } from '@mantine/core';
import { Paper, Table, Title, Badge, Text, Box, ScrollArea } from '@mantine/core'
import { APBDes, APBDesItem, RealisasiItem } from '../types/apbdes'
export default function RealisasiTable({ apbdesData }: any) {
const items = apbdesData.items || [];
interface RealisasiRowProps {
realisasi: RealisasiItem
parentItem: APBDesItem
}
const title =
apbdesData.tahun
? `REALISASI APBDes Tahun ${apbdesData.tahun}`
: 'REALISASI APBDes';
function RealisasiRow({ realisasi, parentItem }: RealisasiRowProps) {
const persentase = parentItem.anggaran > 0
? (realisasi.jumlah / parentItem.anggaran) * 100
: 0
// Flatten: kumpulkan semua realisasi items
const allRealisasiRows: Array<{ realisasi: any; parentItem: any }> = [];
items.forEach((item: any) => {
if (item.realisasiItems && item.realisasiItems.length > 0) {
item.realisasiItems.forEach((realisasi: any) => {
allRealisasiRows.push({ realisasi, parentItem: item });
});
}
});
const getBadgeColor = (percentage: number) => {
if (percentage >= 100) return 'teal'
if (percentage >= 80) return 'blue'
if (percentage >= 60) return 'yellow'
return 'red'
}
const formatRupiah = (amount: number) => {
return new Intl.NumberFormat('id-ID', {
style: 'currency',
currency: 'IDR',
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(amount);
};
const getBadgeVariant = (percentage: number) => {
if (percentage >= 100) return 'filled'
return 'light'
}
return (
<Paper withBorder p={{ base: 'sm', sm: 'md' }} radius="md">
<Title order={5} mb="md" fz={{ base: 'sm', sm: 'md' }}>{title}</Title>
<Table.Tr
style={{
transition: 'background-color 0.2s ease',
':hover': {
backgroundColor: 'var(--mantine-color-blue-0)',
},
}}
>
<Table.Td style={{ borderBottom: '1px solid #e5e7eb' }}>
<Box style={{ gap: 8, alignItems: 'center' }}>
<span style={{
fontWeight: 500,
color: 'var(--mantine-color-gray-7)',
}}>
{realisasi.kode || '-'}
</span>
<Text
size="sm"
c="gray.7"
title={realisasi.keterangan || '-'}
>
{realisasi.keterangan || '-'}
</Text>
</Box>
</Table.Td>
<Table.Td
ta="right"
style={{
borderBottom: '1px solid #e5e7eb',
fontWeight: 700,
color: 'var(--mantine-color-blue-7)',
}}
>
{new Intl.NumberFormat('id-ID', {
style: 'currency',
currency: 'IDR',
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(realisasi.jumlah || 0)}
</Table.Td>
<Table.Td
ta="center"
style={{ borderBottom: '1px solid #e5e7eb' }}
>
<Badge
color={getBadgeColor(persentase)}
variant={getBadgeVariant(persentase)}
size="sm"
radius="xl"
fw={600}
style={{
minWidth: 65,
transition: 'transform 0.2s ease',
}}
>
{persentase.toFixed(1)}%
</Badge>
</Table.Td>
</Table.Tr>
)
}
interface RealisasiTableProps {
apbdesData: APBDes
}
export default function RealisasiTable({ apbdesData }: RealisasiTableProps) {
const items = apbdesData.items || []
const title = apbdesData.tahun
? `REALISASI APBDes Tahun ${apbdesData.tahun}`
: 'REALISASI APBDes'
// Flatten: kumpulkan semua realisasi items
const allRealisasiRows: Array<{ realisasi: RealisasiItem; parentItem: APBDesItem }> = []
items.forEach((item: APBDesItem) => {
if (item.realisasiItems && item.realisasiItems.length > 0) {
item.realisasiItems.forEach((realisasi: RealisasiItem) => {
allRealisasiRows.push({ realisasi, parentItem: item })
})
}
})
// Calculate total realisasi
const totalRealisasi = allRealisasiRows.reduce(
(sum, { realisasi }) => sum + (realisasi.jumlah || 0),
0
)
return (
<Paper
withBorder
p="md"
radius="lg"
shadow="sm"
style={{
transition: 'box-shadow 0.3s ease',
':hover': {
boxShadow: '0 4px 12px rgba(0, 0, 0, 0.1)',
},
}}
h={"100%"}
>
<Title
order={5}
mb="md"
c="blue.9"
fz={{ base: '1rem', md: '1.1rem' }}
fw={700}
>
{title}
</Title>
{allRealisasiRows.length === 0 ? (
<Text fz="sm" c="dimmed" ta="center" py="md">
Belum ada data realisasi
</Text>
<Box
py="xl"
px="md"
style={{
backgroundColor: 'var(--mantine-color-gray-0)',
borderRadius: 8,
}}
>
<Text
fz="sm"
c="dimmed"
ta="center"
lh={1.6}
>
Belum ada data realisasi untuk tahun ini
</Text>
</Box>
) : (
<Table.ScrollContainer minWidth={300}>
<Table verticalSpacing="xs">
<Table.Thead>
<Table.Tr>
<Table.Th fz={{ base: 'xs', sm: 'sm' }}>Uraian</Table.Th>
<Table.Th ta="right" fz={{ base: 'xs', sm: 'sm' }}>Realisasi (Rp)</Table.Th>
<Table.Th ta="center" fz={{ base: 'xs', sm: 'sm' }}>%</Table.Th>
</Table.Tr>
</Table.Thead>
<Table.Tbody>
{allRealisasiRows.map(({ realisasi, parentItem }) => {
const persentase = parentItem.anggaran > 0
? (realisasi.jumlah / parentItem.anggaran) * 100
: 0;
return (
<Table.Tr key={realisasi.id}>
<Table.Td>
<Text fz={{ base: 'xs', sm: 'sm' }} lineClamp={2}>
{realisasi.kode || '-'} - {realisasi.keterangan || '-'}
</Text>
</Table.Td>
<Table.Td ta="right">
<Text fw={600} c="blue" fz={{ base: 'xs', sm: 'sm' }} style={{ whiteSpace: 'nowrap' }}>
{formatRupiah(realisasi.jumlah || 0)}
</Text>
</Table.Td>
<Table.Td ta="center">
<Badge
size="sm"
variant="light"
color={
persentase >= 100
? 'teal'
: persentase >= 60
? 'yellow'
: 'red'
}
>
{persentase.toFixed(1)}%
</Badge>
</Table.Td>
</Table.Tr>
);
})}
</Table.Tbody>
</Table>
</Table.ScrollContainer>
<>
<ScrollArea offsetScrollbars type="hover">
<Table
horizontalSpacing="md"
verticalSpacing="xs"
layout="fixed"
>
<Table.Thead>
<Table.Tr bg="blue.9">
<Table.Th c="white" fw={600}>Uraian</Table.Th>
<Table.Th c="white" fw={600} ta="right">Realisasi (Rp)</Table.Th>
<Table.Th c="white" fw={600} ta="center">%</Table.Th>
</Table.Tr>
</Table.Thead>
<Table.Tbody>
{allRealisasiRows.map(({ realisasi, parentItem }) => (
<RealisasiRow
key={realisasi.id}
realisasi={realisasi}
parentItem={parentItem}
/>
))}
</Table.Tbody>
</Table>
</ScrollArea>
<Box mb="md" px="sm">
<Text
size="sm"
c="gray.6"
fw={500}
>
Total Realisasi:{' '}
<Text
component="span"
c="blue.9"
fw={700}
fz="md"
>
{new Intl.NumberFormat('id-ID', {
style: 'currency',
currency: 'IDR',
minimumFractionDigits: 0,
maximumFractionDigits: 0,
}).format(totalRealisasi)}
</Text>
</Text>
</Box>
</>
)}
</Paper>
);
}
)
}

View File

@@ -0,0 +1,90 @@
// Types for APBDes data structure
export interface APBDesItem {
id?: string;
kode: string;
uraian: string;
deskripsi?: string;
tipe: 'pendapatan' | 'belanja' | 'pembiayaan' | null;
anggaran: number;
level?: number;
// Calculated fields
realisasi?: number;
selisih?: number;
persentase?: number;
// Realisasi items (nested)
realisasiItems?: RealisasiItem[];
createdAt?: string | Date;
updatedAt?: string | Date;
}
export interface RealisasiItem {
id: string;
kode: string;
keterangan?: string;
jumlah: number;
tanggal?: string | Date;
apbDesItemId: string;
buktiFileId?: string;
createdAt?: string | Date;
updatedAt?: string | Date;
}
export interface APBDes {
id: string;
name?: string | null;
tahun: number;
jumlah: number;
deskripsi?: string | null;
items?: APBDesItem[];
image?: {
id: string;
link: string;
name?: string;
path?: string;
} | null;
file?: {
id: string;
link: string;
name?: string;
} | null;
imageId?: string;
fileId?: string;
createdAt?: string | Date;
updatedAt?: string | Date;
}
export interface APBDesResponse {
id: string;
tahun: number;
name?: string | null;
jumlah: number;
items?: APBDesItem[];
image?: {
id: string;
link: string;
} | null;
file?: {
id: string;
link: string;
} | null;
}
export interface SummaryData {
title: string;
totalAnggaran: number;
totalRealisasi: number;
persentase: number;
}
export interface FilterState {
search: string;
tipe: 'all' | 'pendapatan' | 'belanja' | 'pembiayaan';
sortBy: 'uraian' | 'anggaran' | 'realisasi' | 'persentase';
sortOrder: 'asc' | 'desc';
}
export type LoadingState = {
initial: boolean;
changingYear: boolean;
};

View File

@@ -3,7 +3,6 @@ import "./globals.css"; // Sisanya import di globals.css
import LoadDataFirstClient from "@/app/darmasaba/_com/LoadDataFirstClient";
import { MusicProvider } from "@/app/context/MusicContext";
import DebugStateProvider from '@/components/DebugStateProvider';
import {
ColorSchemeScript,
MantineProvider,
@@ -100,14 +99,13 @@ export default function RootLayout({
children: React.ReactNode;
}) {
return (
<ViewTransitions>
<html lang="id" {...mantineHtmlProps}>
<head>
<meta charSet="utf-8" />
<ColorSchemeScript defaultColorScheme="light" />
</head>
<body>
<DebugStateProvider />
<html lang="id" {...mantineHtmlProps}>
<head>
<meta charSet="utf-8" />
<ColorSchemeScript defaultColorScheme="light" />
</head>
<body>
<ViewTransitions>
<MusicProvider>
<MantineProvider theme={theme} defaultColorScheme="light">
{children}
@@ -119,8 +117,8 @@ export default function RootLayout({
/>
</MantineProvider>
</MusicProvider>
</body>
</html>
</ViewTransitions>
</ViewTransitions>
</body>
</html>
);
}

View File

@@ -1,33 +0,0 @@
'use client';
/**
* Debug State Component - Expose state to window object for debugging
*
* Usage in browser console:
* window.publicMusicState
* window.adminNavState
* window.adminAuthState
*/
import { useEffect } from 'react';
import { publicMusicState } from '@/state/public/publicMusicState';
import { adminNavState, adminAuthState } from '@/state/admin';
export default function DebugStateProvider() {
useEffect(() => {
if (typeof window !== 'undefined') {
// Expose states
(window as any).publicMusicState = publicMusicState;
(window as any).adminNavState = adminNavState;
(window as any).adminAuthState = adminAuthState;
console.log('%c✅ [Debug] State exposed to window:', 'color: #3B82F6; font-weight: bold;');
console.log(' • window.publicMusicState');
console.log(' • window.adminNavState');
console.log(' • window.adminAuthState');
console.log(' 💡 Type "window.publicMusicState" in console to check state');
}
}, []);
return null; // No UI, just side effects
}

View File

@@ -1,9 +1,25 @@
import { AppServer } from '@/app/api/[[...slugs]]/route'
import { treaty } from '@elysiajs/eden'
// const BASE_URL = process.env.NEXT_PUBLIC_BASE_URL || 'localhost:3000'
const BASE_URL = process.env.NEXT_PUBLIC_BASE_URL || 'http://localhost:3000'
// Determine the base URL based on environment
// treaty requires a full URL, cannot use relative paths like '/'
const getBaseUrl = () => {
// Development (server-side)
if (process.env.NODE_ENV === 'development' && typeof window === 'undefined') {
return process.env.NEXT_PUBLIC_BASE_URL || 'http://localhost:3000'
}
// Client-side (browser) - use current window origin
if (typeof window !== 'undefined') {
return window.location.origin
}
// Production/Staging server-side - use environment variable or default
return process.env.NEXT_PUBLIC_BASE_URL || 'http://localhost:3000'
}
const BASE_URL = getBaseUrl()
const ApiFetch = treaty<AppServer>(BASE_URL)
export default ApiFetch
export default ApiFetch

View File

@@ -1,51 +0,0 @@
/**
* Debug Utility - Expose state to window object for debugging
*
* IMPORTANT: This file MUST be imported in layout.tsx
*
* Usage in browser console:
* window.publicMusicState
* window.adminNavState
* window.adminAuthState
*/
// Import states
import { publicMusicState } from '@/state/public/publicMusicState';
import { adminNavState, adminAuthState } from '@/state/admin';
// Immediate execution when module loads
console.log('%c🔧 [DebugState] Module loaded!', 'color: #10B981; font-weight: bold; font-size: 12px;');
// Expose states to window object for debugging
if (typeof window !== 'undefined') {
// Expose states
(window as any).publicMusicState = publicMusicState;
(window as any).adminNavState = adminNavState;
(window as any).adminAuthState = adminAuthState;
// Also expose helper functions
(window as any).getMusicState = () => {
console.log('🎵 Music State:', publicMusicState);
return publicMusicState;
};
(window as any).getAdminNavState = () => adminNavState;
(window as any).getAdminAuthState = () => adminAuthState;
console.log('%c✅ [DebugState] State exposed to window object:', 'color: #3B82F6; font-weight: bold; font-size: 12px;');
console.log(' • window.publicMusicState');
console.log(' • window.adminNavState');
console.log(' • window.adminAuthState');
console.log(' • window.getMusicState()');
// Verify exposure
setTimeout(() => {
console.log('%c🔍 [DebugState] Verification:', 'color: #8B5CF6; font-weight: bold; font-size: 12px;');
console.log(' window.publicMusicState exists?', !!(window as any).publicMusicState);
console.log(' window.adminNavState exists?', !!(window as any).adminNavState);
console.log(' window.adminAuthState exists?', !!(window as any).adminAuthState);
}, 100);
}
export default function DebugState() {
return null; // This is just a utility, no UI
}

View File

@@ -1,123 +0,0 @@
/**
* HTML Sanitizer Utility
*
* Membersihkan HTML content dari script dan tag berbahaya
* Menggunakan DOMPurify-like approach untuk environment Node.js
*/
/**
* Sanitize HTML content untuk mencegah XSS attacks
* @param html - HTML content yang akan disanitize
* @returns HTML yang sudah dibersihkan dari script berbahaya
*/
export function sanitizeHtml(html: string): string {
if (!html || typeof html !== 'string') {
return '';
}
let sanitized = html;
// Remove script tags and their content
sanitized = sanitized.replace(/<script\b[^<]*(?:(?!<\/script>)<[^<]*)*<\/script>/gi, '');
// Remove javascript: protocol in href/src attributes
sanitized = sanitized.replace(/(href|src)\s*=\s*["']?javascript:[^"'\s>]*/gi, '$1=""');
// Remove on* event handlers (onclick, onerror, onload, etc.)
sanitized = sanitized.replace(/\s*on\w+\s*=\s*["'][^"']*["']/gi, '');
sanitized = sanitized.replace(/\s*on\w+\s*=\s*[^"'\s>]*/gi, '');
// Remove iframe tags
sanitized = sanitized.replace(/<iframe\b[^<]*(?:(?!<\/iframe>)<[^<]*)*<\/iframe>/gi, '');
// Remove object and embed tags
sanitized = sanitized.replace(/<(object|embed)\b[^<]*(?:(?!<\/\1>)<[^<]*)*<\/\1>/gi, '');
// Remove style tags (optional - can be kept if needed)
// sanitized = sanitized.replace(/<style\b[^<]*(?:(?!<\/style>)<[^<]*)*<\/style>/gi, '');
// Remove data: protocol in src attributes (can be used for XSS)
sanitized = sanitized.replace(/(src)\s*=\s*["']?data:[^"'\s>]*/gi, '$1=""');
// Remove expression() in CSS (IE-specific XSS vector)
sanitized = sanitized.replace(/expression\s*\([^)]*\)/gi, '');
return sanitized;
}
/**
* Sanitize text input (remove HTML tags completely)
* @param text - Text input yang akan disanitize
* @returns Plain text tanpa HTML tags
*/
export function sanitizeText(text: string): string {
if (!text || typeof text !== 'string') {
return '';
}
// Remove all HTML tags
return text.replace(/<[^>]*>/g, '').trim();
}
/**
* Sanitize URL - hanya izinkan http dan https
* @param url - URL yang akan disanitize
* @returns URL yang aman atau empty string jika tidak valid
*/
export function sanitizeUrl(url: string): string {
if (!url || typeof url !== 'string') {
return '';
}
try {
const parsedUrl = new URL(url);
// Only allow http and https protocols
if (parsedUrl.protocol !== 'http:' && parsedUrl.protocol !== 'https:') {
return '';
}
return parsedUrl.toString();
} catch {
return '';
}
}
/**
* Sanitize YouTube URL - extract video ID
* @param url - YouTube URL
* @returns YouTube video ID atau empty string jika tidak valid
*/
export function sanitizeYouTubeUrl(url: string): string {
if (!url || typeof url !== 'string') {
return '';
}
try {
const parsedUrl = new URL(url);
// Check if it's a YouTube URL
if (!parsedUrl.hostname.includes('youtube.com') &&
!parsedUrl.hostname.includes('youtu.be')) {
return '';
}
// Extract video ID
let videoId = '';
if (parsedUrl.hostname.includes('youtu.be')) {
videoId = parsedUrl.pathname.slice(1);
} else if (parsedUrl.hostname.includes('youtube.com')) {
videoId = parsedUrl.searchParams.get('v') || '';
}
// Validate video ID (YouTube video IDs are 11 characters)
if (videoId.length !== 11) {
return '';
}
return `https://www.youtube.com/watch?v=${videoId}`;
} catch {
return '';
}
}

View File

@@ -1,9 +1,9 @@
/**
* Session helper menggunakan iron-session
*
*
* Usage:
* import { getSession } from "@/lib/session";
*
*
* const session = await getSession();
* if (session?.user) {
* // User authenticated
@@ -28,31 +28,14 @@ export type Session = SessionData & {
destroy: () => Promise<void>;
};
// Validate SESSION_PASSWORD environment variable
if (!process.env.SESSION_PASSWORD) {
throw new Error(
'SESSION_PASSWORD environment variable is required. ' +
'Please set a strong password (min 32 characters) in your .env file.'
);
}
// Validate password length for security
if (process.env.SESSION_PASSWORD.length < 32) {
throw new Error(
'SESSION_PASSWORD must be at least 32 characters long for security. ' +
'Please use a strong random password.'
);
}
const SESSION_OPTIONS = {
cookieName: 'desa-session',
password: process.env.SESSION_PASSWORD,
password: process.env.SESSION_PASSWORD || 'default-password-change-in-production',
cookieOptions: {
secure: process.env.NODE_ENV === 'production',
httpOnly: true,
sameSite: 'lax' as const,
maxAge: 60 * 60 * 24 * 7, // 7 days
path: '/',
},
};

View File

@@ -1,144 +0,0 @@
/**
* Validation Schemas with Zod
*
* Centralized validation schemas for all API endpoints
* Used for input validation and sanitization
*/
import { z } from 'zod';
/**
* Berita (News) Validation Schemas
*/
export const createBeritaSchema = z.object({
judul: z
.string()
.min(5, 'Judul minimal 5 karakter')
.max(255, 'Judul maksimal 255 karakter'),
deskripsi: z
.string()
.min(10, 'Deskripsi minimal 10 karakter')
.max(500, 'Deskripsi maksimal 500 karakter'),
content: z
.string()
.min(50, 'Konten minimal 50 karakter'),
kategoriBeritaId: z
.string()
.cuid('Kategori berita ID tidak valid'),
imageId: z
.string()
.cuid('Image ID tidak valid'),
imageIds: z
.array(z.string().cuid())
.optional(),
linkVideo: z
.string()
.url('Format URL YouTube tidak valid')
.optional()
.or(z.literal('')),
});
export const updateBeritaSchema = createBeritaSchema.partial();
/**
* OTP/Login Validation Schemas
*/
export const loginRequestSchema = z.object({
nomor: z
.string()
.min(10, 'Nomor telepon minimal 10 digit')
.max(15, 'Nomor telepon maksimal 15 digit')
.regex(/^[0-9]+$/, 'Nomor telepon harus berupa angka'),
});
export const otpVerificationSchema = z.object({
nomor: z
.string()
.min(10, 'Nomor telepon minimal 10 digit')
.max(15, 'Nomor telepon maksimal 15 digit'),
kodeId: z
.string()
.cuid('Kode ID tidak valid'),
otp: z
.string()
.length(6, 'OTP harus 6 digit')
.regex(/^[0-9]+$/, 'OTP harus berupa angka'),
});
/**
* File Upload Validation Schemas
*/
export const uploadFileSchema = z.object({
name: z.string().min(1, 'Nama file wajib diisi'),
type: z.string().refine(
(type) => {
const allowedTypes = [
'image/jpeg',
'image/png',
'image/gif',
'image/webp',
'application/pdf',
'application/msword',
'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
];
return allowedTypes.includes(type);
},
'Tipe file tidak diizinkan'
),
size: z.number().max(5 * 1024 * 1024, 'Ukuran file maksimal 5MB'), // 5MB
});
/**
* User Registration Validation Schemas
*/
export const registerUserSchema = z.object({
name: z
.string()
.min(3, 'Nama minimal 3 karakter')
.max(100, 'Nama maksimal 100 karakter'),
nomor: z
.string()
.min(10, 'Nomor telepon minimal 10 digit')
.max(15, 'Nomor telepon maksimal 15 digit')
.regex(/^[0-9]+$/, 'Nomor telepon harus berupa angka'),
email: z
.string()
.email('Format email tidak valid')
.optional()
.or(z.literal('')),
roleId: z
.number()
.int('Role ID harus berupa angka bulat')
.positive('Role ID harus lebih dari 0'),
});
/**
* Generic Pagination Schema
*/
export const paginationSchema = z.object({
page: z
.string()
.optional()
.transform((val) => (val ? parseInt(val, 10) : 1))
.refine((val) => !isNaN(val) && val > 0, 'Page harus lebih dari 0'),
limit: z
.string()
.optional()
.transform((val) => (val ? parseInt(val, 10) : 10))
.refine(
(val) => !isNaN(val) && val > 0 && val <= 100,
'Limit harus antara 1-100'
),
search: z.string().optional(),
});
/**
* Export type inference
*/
export type CreateBeritaInput = z.infer<typeof createBeritaSchema>;
export type UpdateBeritaInput = z.infer<typeof updateBeritaSchema>;
export type LoginRequestInput = z.infer<typeof loginRequestSchema>;
export type OtpVerificationInput = z.infer<typeof otpVerificationSchema>;
export type UploadFileInput = z.infer<typeof uploadFileSchema>;
export type RegisterUserInput = z.infer<typeof registerUserSchema>;
export type PaginationInput = z.infer<typeof paginationSchema>;

View File

@@ -1,121 +0,0 @@
/**
* WhatsApp Service - Secure OTP Delivery
*
* Mengirim OTP via WhatsApp dengan metode POST yang aman
* OTP tidak dikirim langsung, tapi menggunakan reference ID
*/
interface WhatsAppOTPRequest {
nomor: string;
otpId: string;
message: string;
}
interface WhatsAppOTPResponse {
status: 'success' | 'error';
message?: string;
}
/**
* Kirim OTP via WhatsApp dengan POST request
* OTP tidak dikirim dalam URL, tapi menggunakan reference ID
*
* @param nomor - Nomor telepon tujuan
* @param otpId - ID referensi OTP dari database
* @param message - Pesan template (tanpa OTP code)
*/
export async function sendWhatsAppOTP({
nomor,
otpId,
message,
}: WhatsAppOTPRequest): Promise<WhatsAppOTPResponse> {
try {
// Validasi nomor telepon
if (!nomor || typeof nomor !== 'string') {
return {
status: 'error',
message: 'Nomor telepon tidak valid',
};
}
// Validasi otpId
if (!otpId || typeof otpId !== 'string') {
return {
status: 'error',
message: 'OTP ID tidak valid',
};
}
// Kirim dengan POST request - OTP tidak dikirim dalam URL
const response = await fetch('https://wa.wibudev.com/send', {
method: 'POST',
headers: {
'Content-Type': 'application/json',
},
body: JSON.stringify({
nomor: nomor,
// OTP code tidak dikirim ke WhatsApp API
// Frontend akan meminta user memasukkan OTP yang mereka terima
// Backend akan validate berdasarkan otpId
otpId: otpId,
message: message,
}),
});
if (!response.ok) {
console.error('WhatsApp API error:', response.status, response.statusText);
return {
status: 'error',
message: 'Gagal mengirim pesan WhatsApp',
};
}
const result = await response.json();
if (result.status !== 'success') {
return {
status: 'error',
message: result.message || 'Gagal mengirim pesan WhatsApp',
};
}
return {
status: 'success',
};
} catch (error) {
console.error('Error sending WhatsApp OTP:', error);
return {
status: 'error',
message: 'Terjadi kesalahan saat mengirim pesan',
};
}
}
/**
* Format pesan WhatsApp untuk OTP
* @param otpCode - Kode OTP (hanya digunakan di sisi server untuk message template)
* @returns Pesan yang sudah diformat
*/
export function formatOTPMessage(otpCode: number | string): string {
return `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN, termasuk anggota ataupun Admin lainnya.
>> Kode OTP anda: ${otpCode}
Kode ini hanya berlaku untuk satu kali login.`;
}
/**
* Format pesan WhatsApp untuk OTP (tanpa menampilkan code - lebih aman)
* Menggunakan reference ID saja
* @param otpId - ID referensi OTP
* @returns Pesan yang sudah diformat
*/
export function formatOTPMessageWithReference(otpId: string): string {
return `Website Desa Darmasaba - Kode ini bersifat RAHASIA dan JANGAN DI BAGIKAN KEPADA SIAPAPUN.
Silakan masukkan kode OTP yang telah dikirimkan ke nomor Anda.
Reference ID: ${otpId}
Kode ini hanya berlaku untuk satu kali login.`;
}

View File

@@ -1,43 +0,0 @@
/**
* Admin Authentication State
*
* State management untuk authentication admin
* Menggunakan Valtio untuk reactive state
*/
import { proxy } from 'valtio';
export type User = {
id: string;
name: string;
roleId: number;
menuIds?: string[] | null;
isActive?: boolean;
};
export const adminAuthState = proxy<{
user: User | null;
isAuthenticated: boolean;
setUser: (user: User | null) => void;
clearUser: () => void;
}>({
user: null,
isAuthenticated: false,
setUser(user: User | null) {
adminAuthState.user = user;
adminAuthState.isAuthenticated = !!user;
},
clearUser() {
adminAuthState.user = null;
adminAuthState.isAuthenticated = false;
},
});
// Helper hook untuk React components
export const useAdminAuth = () => {
return adminAuthState;
};
export default adminAuthState;

View File

@@ -1,112 +0,0 @@
/**
* Admin Form State
*
* State management untuk form di admin dashboard
* Menggunakan Valtio untuk reactive state
*/
import { proxy } from "valtio";
export interface FileStorageItem {
id: string;
name: string;
path: string;
link: string;
realName: string;
mimeType: string;
category: string;
isActive: boolean;
createdAt: Date;
updatedAt: Date;
deletedAt: Date | null;
}
export interface ListItem {
id: string;
name: string;
url: string;
total: number;
realName: string;
}
export const adminFormState = proxy<{
list: ListItem[] | null;
page: number;
count: number;
total: number | undefined;
isLoading: boolean;
error: string | null;
load: (params?: { search?: string; page?: number }) => Promise<void>;
del: (params: { id: string }) => Promise<void>;
reset: () => void;
}>({
list: null,
page: 1,
count: 10,
total: undefined,
isLoading: false,
error: null,
async load(params?: { search?: string; page?: number }) {
const { search = "", page = this.page } = params ?? {};
this.page = page;
this.isLoading = true;
this.error = null;
try {
// Import dinamis untuk menghindari circular dependency
const ApiFetch = (await import('@/lib/api-fetch')).default;
const response = await ApiFetch.api.fileStorage["findMany"].get({
query: {
page: this.page,
search,
},
}) as { data: { data: FileStorageItem[]; meta: { total: number; totalPages: number } } };
if (response?.data?.data) {
this.list = response.data.data.map((file) => ({
id: file.id,
name: file.name,
url: file.link || `/api/fileStorage/${file.realName}`,
total: response.data.meta?.total || 0,
realName: file.realName,
}));
this.total = response.data.meta?.totalPages;
}
} catch (error) {
console.error("Error loading images:", error);
this.error = error instanceof Error ? error.message : 'Failed to load images';
this.list = [];
} finally {
this.isLoading = false;
}
},
async del({ id }: { id: string }) {
try {
const ApiFetch = (await import('@/lib/api-fetch')).default;
await ApiFetch.api.fileStorage.delete({ id });
await this.load({ page: this.page });
} catch (error) {
console.error("Error deleting image:", error);
throw error;
}
},
reset() {
this.list = null;
this.page = 1;
this.count = 10;
this.total = undefined;
this.isLoading = false;
this.error = null;
},
});
// Helper hook untuk React components
export const useAdminForm = () => {
return adminFormState;
};
export default adminFormState;

View File

@@ -1,43 +0,0 @@
/**
* Admin Module States
*
* State management untuk modul-modul admin
* Menggunakan Valtio untuk reactive state
*/
import { proxy } from "valtio";
// Keamanan Module State
export const adminKeamananState = proxy<{
selectedLayanan: string | null;
setSelectedLayanan: (layanan: string | null) => void;
}>({
selectedLayanan: null,
setSelectedLayanan(layanan: string | null) {
adminKeamananState.selectedLayanan = layanan;
},
});
// PPID Module State
export const adminPpidState = proxy<{
selectedPermohonan: string | null;
setSelectedPermohonan: (permohonan: string | null) => void;
}>({
selectedPermohonan: null,
setSelectedPermohonan(permohonan: string | null) {
adminPpidState.selectedPermohonan = permohonan;
},
});
// Musik Module State
export const adminMusikState = proxy<{
selectedMusik: string | null;
setSelectedMusik: (musik: string | null) => void;
}>({
selectedMusik: null,
setSelectedMusik(musik: string | null) {
adminMusikState.selectedMusik = musik;
},
});
export default adminKeamananState;

View File

@@ -1,47 +0,0 @@
/**
* Admin Navigation State
*
* State management untuk navigasi admin dashboard
* Menggunakan Valtio untuk reactive state
*/
import { proxy } from "valtio";
import type { MenuItem } from "../../../types/menu-item";
export const adminNavState = proxy<{
hover: boolean;
item: MenuItem[] | null;
isSearch: boolean;
module: string | null;
mobileOpen: boolean;
clear: () => void;
setModule: (module: string | null) => void;
toggleMobile: () => void;
}>({
hover: false,
item: null,
isSearch: false,
module: null,
mobileOpen: false,
clear() {
adminNavState.hover = false;
adminNavState.item = null;
adminNavState.isSearch = false;
},
setModule(module: string | null) {
adminNavState.module = module;
},
toggleMobile() {
adminNavState.mobileOpen = !adminNavState.mobileOpen;
},
});
// Helper hook untuk React components
export const useAdminNav = () => {
return adminNavState;
};
export default adminNavState;

View File

@@ -1,14 +0,0 @@
/**
* Admin State Exports
*
* Centralized exports untuk semua admin state
*/
export { adminNavState, useAdminNav } from './adminNavState';
export { adminAuthState, useAdminAuth, type User } from './adminAuthState';
export { adminFormState, useAdminForm, type ListItem, type FileStorageItem } from './adminFormState';
export {
adminKeamananState,
adminPpidState,
adminMusikState,
} from './adminModuleState';

View File

@@ -5,7 +5,7 @@
* Persist ke localStorage
*
* Usage:
* import { darkModeStore, useDarkMode } from '@/state/darkModeStore';
* import { darkModeStore } from '@/state/darkModeStore';
*
* // Toggle
* darkModeStore.toggle();
@@ -15,9 +15,6 @@
*
* // Get current state
* const isDark = darkModeStore.isDark;
*
* // In React components
* const { isDark, toggle, setDarkMode } = useDarkMode();
*/
import { proxy, useSnapshot } from 'valtio';

View File

@@ -1,58 +0,0 @@
/**
* State Management - Central Exports
*
* Desa Darmasaba menggunakan Valtio untuk global state management.
*
* State dibagi menjadi dua kategori utama:
* 1. Admin State - Untuk admin dashboard (/admin routes)
* 2. Public State - Untuk public pages (/darmasaba routes)
*
* Usage:
* ```typescript
* // Import admin state
* import { adminNavState, adminAuthState } from '@/state';
*
* // Import public state
* import { publicNavState, publicMusicState } from '@/state';
*
* // In React components
* import { useAdminNav, usePublicMusic } from '@/state';
* ```
*/
// Admin State
export {
adminNavState,
useAdminNav,
adminAuthState,
useAdminAuth,
adminFormState,
useAdminForm,
adminKeamananState,
adminPpidState,
adminMusikState,
} from './admin';
export type {
ListItem,
FileStorageItem,
} from './admin';
// Public State
export {
publicNavState,
usePublicNav,
publicMusicState,
usePublicMusic,
} from './public';
export type {
Musik,
} from './public';
// Legacy State (for backward compatibility)
export { darkModeStore, useDarkMode } from './darkModeStore';
export { stateNav } from './state-nav';
export { authStore, type User } from '../store/authStore';
export { stateListImage } from './state-list-image';
export { default as stateLayanan } from './state-layanan';

View File

@@ -1,8 +0,0 @@
/**
* Public State Exports
*
* Centralized exports untuk semua public state
*/
export { publicNavState, usePublicNav } from './publicNavState';
export { publicMusicState, usePublicMusic, type Musik } from './publicMusicState';

View File

@@ -1,238 +0,0 @@
/**
* Public Music Player State
*
* State management untuk music player di public pages
* Menggunakan Valtio untuk reactive state
*
* Menggantikan MusicContext dengan Valtio untuk konsistensi
*/
import { proxy, useSnapshot } from 'valtio';
interface MusicFile {
id: string;
name: string;
realName: string;
path: string;
mimeType: string;
link: string;
}
export interface Musik {
id: string;
judul: string;
artis: string;
deskripsi: string | null;
durasi: string;
genre: string | null;
tahunRilis: number | null;
audioFile: MusicFile | null;
coverImage: MusicFile | null;
isActive: boolean;
}
export const publicMusicState = proxy<{
// State
isPlaying: boolean;
currentSong: Musik | null;
currentSongIndex: number;
musikData: Musik[];
currentTime: number;
duration: number;
volume: number;
isMuted: boolean;
isRepeat: boolean;
isShuffle: boolean;
isLoading: boolean;
isPlayerOpen: boolean;
error: string | null;
// Actions
playSong: (song: Musik) => void;
togglePlayPause: () => void;
playNext: () => void;
playPrev: () => void;
seek: (time: number) => void;
setVolume: (volume: number) => void;
toggleMute: () => void;
toggleRepeat: () => void;
toggleShuffle: () => void;
togglePlayer: () => void;
loadMusikData: () => Promise<void>;
reset: () => void;
}>({
// Initial State
isPlaying: false,
currentSong: null,
currentSongIndex: -1,
musikData: [],
currentTime: 0,
duration: 0,
volume: 70,
isMuted: false,
isRepeat: false,
isShuffle: false,
isLoading: true,
isPlayerOpen: false,
error: null,
// Actions
playSong(song: Musik) {
if (!song?.audioFile?.link) {
console.warn('[MusicState] No audio file link for song:', song);
return;
}
console.log('[MusicState] Playing song:', song.judul);
const songIndex = publicMusicState.musikData.findIndex(m => m.id === song.id);
console.log('[MusicState] Song index:', songIndex);
publicMusicState.currentSongIndex = songIndex;
publicMusicState.currentSong = song;
publicMusicState.isPlaying = true;
console.log('[MusicState] State updated:', {
isPlaying: publicMusicState.isPlaying,
currentSong: publicMusicState.currentSong?.judul,
currentSongIndex: publicMusicState.currentSongIndex,
});
// Audio handling dilakukan di component dengan useEffect
},
togglePlayPause() {
publicMusicState.isPlaying = !publicMusicState.isPlaying;
},
playNext() {
if (publicMusicState.musikData.length === 0) return;
let nextIndex: number;
if (publicMusicState.isShuffle) {
nextIndex = Math.floor(Math.random() * publicMusicState.musikData.length);
} else {
nextIndex = (publicMusicState.currentSongIndex + 1) % publicMusicState.musikData.length;
}
const nextSong = publicMusicState.musikData[nextIndex];
if (nextSong) {
publicMusicState.playSong(nextSong);
}
},
playPrev() {
if (publicMusicState.musikData.length === 0) return;
// If more than 3 seconds into song, restart it
if (publicMusicState.currentTime > 3) {
publicMusicState.currentTime = 0;
return;
}
const prevIndex = publicMusicState.currentSongIndex <= 0
? publicMusicState.musikData.length - 1
: publicMusicState.currentSongIndex - 1;
const prevSong = publicMusicState.musikData[prevIndex];
if (prevSong) {
publicMusicState.playSong(prevSong);
}
},
seek(time: number) {
publicMusicState.currentTime = time;
},
setVolume(vol: number) {
const normalizedVol = Math.max(0, Math.min(100, vol)) / 100;
const newVolume = Math.max(0, Math.min(100, vol));
console.log('[MusicState] setVolume called:', vol, '->', newVolume, 'normalized:', normalizedVol);
publicMusicState.volume = newVolume;
publicMusicState.isMuted = normalizedVol === 0;
},
toggleMute() {
publicMusicState.isMuted = !publicMusicState.isMuted;
},
toggleRepeat() {
publicMusicState.isRepeat = !publicMusicState.isRepeat;
},
toggleShuffle() {
publicMusicState.isShuffle = !publicMusicState.isShuffle;
},
togglePlayer() {
publicMusicState.isPlayerOpen = !publicMusicState.isPlayerOpen;
},
async loadMusikData() {
try {
publicMusicState.isLoading = true;
publicMusicState.error = null;
console.log('[MusicState] Loading musik data...');
const res = await fetch('/api/desa/musik/find-many?page=1&limit=50');
const data = await res.json();
console.log('[MusicState] API response:', data);
if (data.success && data.data) {
const activeMusik = data.data.filter((m: Musik) => m.isActive);
console.log('[MusicState] Loaded', activeMusik.length, 'active songs');
publicMusicState.musikData = activeMusik;
// Log first song for debugging
if (activeMusik.length > 0) {
console.log('[MusicState] First song:', {
judul: activeMusik[0].judul,
hasAudioFile: !!activeMusik[0].audioFile,
audioLink: activeMusik[0].audioFile?.link,
});
}
}
} catch (error) {
console.error('[MusicState] Error fetching musik:', error);
publicMusicState.error = error instanceof Error ? error.message : 'Failed to load music';
} finally {
publicMusicState.isLoading = false;
}
},
reset() {
publicMusicState.isPlaying = false;
publicMusicState.currentSong = null;
publicMusicState.currentSongIndex = -1;
publicMusicState.currentTime = 0;
publicMusicState.duration = 0;
publicMusicState.isPlayerOpen = false;
publicMusicState.error = null;
},
});
// Helper hook untuk React components
export const usePublicMusic = () => {
const snapshot = useSnapshot(publicMusicState);
return {
...snapshot,
playSong: publicMusicState.playSong,
togglePlayPause: publicMusicState.togglePlayPause,
playNext: publicMusicState.playNext,
playPrev: publicMusicState.playPrev,
seek: publicMusicState.seek,
setVolume: publicMusicState.setVolume,
toggleMute: publicMusicState.toggleMute,
toggleRepeat: publicMusicState.toggleRepeat,
toggleShuffle: publicMusicState.toggleShuffle,
togglePlayer: publicMusicState.togglePlayer,
loadMusikData: publicMusicState.loadMusikData,
reset: publicMusicState.reset,
};
};
export default publicMusicState;

View File

@@ -1,52 +0,0 @@
/**
* Public Navigation State
*
* State management untuk navigasi public pages (darmasaba)
* Menggunakan Valtio untuk reactive state
*/
import { proxy } from "valtio";
export const publicNavState = proxy<{
mobileMenuOpen: boolean;
activeSection: string | null;
searchOpen: boolean;
scrollPosition: number;
openMenu: () => void;
closeMenu: () => void;
setActiveSection: (section: string | null) => void;
toggleSearch: () => void;
setScrollPosition: (position: number) => void;
}>({
mobileMenuOpen: false,
activeSection: null,
searchOpen: false,
scrollPosition: 0,
openMenu() {
publicNavState.mobileMenuOpen = true;
},
closeMenu() {
publicNavState.mobileMenuOpen = false;
},
setActiveSection(section: string | null) {
publicNavState.activeSection = section;
},
toggleSearch() {
publicNavState.searchOpen = !publicNavState.searchOpen;
},
setScrollPosition(position: number) {
publicNavState.scrollPosition = position;
},
});
// Helper hook untuk React components
export const usePublicNav = () => {
return publicNavState;
};
export default publicNavState;

View File

@@ -1,17 +1,19 @@
/**
* DEPRECATED: File ini dipertahankan untuk backward compatibility.
* Gunakan state management baru dari `@/state/admin/` atau `@/state/public/`
*/
/* eslint-disable @typescript-eslint/no-unused-vars */
/* eslint-disable @typescript-eslint/no-explicit-any */
import ApiFetch from "@/lib/api-fetch";
import { proxy } from "valtio";
import useSwr from "swr";
// Simple state untuk backward compatibility
type Layanan = {
layanan: string | null
useLoad: any
}
const stateLayanan = proxy<Layanan>({
layanan: null
layanan: null,
useLoad: () => {
}
})
export default stateLayanan

View File

@@ -1,10 +1,88 @@
/**
* DEPRECATED: File ini dipertahankan untuk backward compatibility.
* Gunakan `import { adminFormState } from '@/state/admin/adminFormState'` untuk state management baru.
*/
import ApiFetch from "@/lib/api-fetch";
import { proxy } from "valtio";
import { adminFormState } from './admin/adminFormState';
interface FileStorageItem {
id: string;
name: string;
path: string;
link: string;
realName: string;
mimeType: string;
category: string;
isActive: boolean;
createdAt: Date;
updatedAt: Date;
deletedAt: Date | null;
}
interface ApiResponse {
data: FileStorageItem[];
meta: {
page: number;
limit: number;
total: number;
totalPages: number;
};
}
interface ListItem {
id: string;
name: string;
url: string;
total: number;
realName: string;
}
const stateListImage = proxy<{
list: ListItem[] | null;
page: number;
count: number;
total: number | undefined;
load: (params?: { search?: string; page?: number }) => Promise<void>;
del: (params: { id: string }) => Promise<void>;
}>({
list: null,
page: 1,
count: 10,
total: undefined,
async load(params?: { search?: string; page?: number }) {
const { search = "", page = this.page } = params ?? {};
this.page = page;
try {
const response = await ApiFetch.api.fileStorage["findMany"].get({
query: {
page: this.page,
search,
},
}) as { data: ApiResponse };
if (response?.data?.data) {
this.list = response.data.data.map((file) => ({
id: file.id,
name: file.name,
url: file.link || `/api/fileStorage/${file.realName}`,
total: response.data.meta?.total || 0,
realName: file.realName,
}));
this.total = response.data.meta?.totalPages;
}
} catch (error) {
console.error("Error loading images:", error);
this.list = [];
}
},
async del({ id }: { id: string }) {
try {
await ApiFetch.api.fileStorage.delete({ id });
await this.load({ page: this.page });
} catch (error) {
console.error("Error deleting image:", error);
throw error;
}
},
});
// Re-export untuk backward compatibility
export const stateListImage = adminFormState;
export default stateListImage;

View File

@@ -1,10 +1,24 @@
/**
* DEPRECATED: File ini dipertahankan untuk backward compatibility.
* Gunakan `import { adminNavState } from '@/state/admin/adminNavState'` untuk state management baru.
*/
import { proxy } from "valtio"
import { MenuItem } from "../../types/menu-item"
import { adminNavState } from './admin/adminNavState';
const stateNav = proxy<{
hover: boolean,
item: MenuItem[] | null
isSearch: boolean,
clear: () => void,
module: string | null,
mobileOpen: boolean
}>({
hover: false,
item: null,
isSearch: false,
clear: () => {
stateNav.hover = false
stateNav.item = null
stateNav.isSearch = false
},
module: null,
mobileOpen: false
})
// Re-export untuk backward compatibility
export const stateNav = adminNavState;
export default stateNav;
export default stateNav

View File

@@ -1,13 +1,20 @@
/**
* DEPRECATED: File ini dipertahankan untuk backward compatibility.
* Gunakan `import { adminAuthState } from '@/state/admin/adminAuthState'` untuk state management baru.
*/
// src/store/authStore.ts
import { proxy } from 'valtio';
import { adminAuthState } from '../state/admin/adminAuthState';
export type User = {
id: string;
name: string;
roleId: number;
menuIds?: string[] | null; // ✅ Pastikan pakai `string[]`
isActive?: boolean;
};
// Re-export untuk backward compatibility
export const authStore = adminAuthState;
export default authStore;
// Re-export types
export type { User } from '../state/admin/adminAuthState';
export const authStore = proxy<{
user: User | null;
setUser: (user: User | null) => void;
}>({
user: null,
setUser(user) {
authStore.user = user;
},
});

418
task-project-apbdes.md Normal file
View File

@@ -0,0 +1,418 @@
# Task Project Menu: Modernisasi Halaman APBDes
## 📊 Project Overview
**Target File**: `src/app/darmasaba/_com/main-page/apbdes/index.tsx`
**Goal**: Modernisasi tampilan dan fungsionalitas halaman APBDes untuk meningkatkan user experience, visualisasi data, dan code quality.
---
## 🎯 Task List
### **Phase 1: UI/UX Enhancement** 🔥 HIGH PRIORITY
#### Task 1.1: Add Loading State
- [ ] Create `apbdesSkeleton.tsx` component
- [ ] Add skeleton untuk PaguTable
- [ ] Add skeleton untuk RealisasiTable
- [ ] Add skeleton untuk GrafikRealisasi
- [ ] Implement loading state saat ganti tahun
- [ ] Add smooth fade-in transition saat data load
**Files to Create/Modify**:
- `src/app/darmasaba/_com/main-page/apbdes/components/apbdesSkeleton.tsx` (CREATE)
- `src/app/darmasaba/_com/main-page/apbdes/index.tsx` (MODIFY)
**Estimated Time**: 45 menit
---
#### Task 1.2: Improve Table Design
- [ ] Add hover effects pada table rows
- [ ] Implement striped rows untuk readability
- [ ] Add sticky header untuk long data
- [ ] Improve typography dan spacing
- [ ] Add responsive table wrapper untuk mobile
- [ ] Add color coding untuk tipe data berbeda
**Files to Modify**:
- `src/app/darmasaba/_com/main-page/apbdes/lib/paguTable.tsx`
- `src/app/darmasaba/_com/main-page/apbdes/lib/realisasiTable.tsx`
**Estimated Time**: 1 jam
---
#### Task 1.3: Add Animations & Interactions
- [ ] Install Framer Motion (`bun add framer-motion`)
- [ ] Add fade-in animation untuk main container
- [ ] Add slide-up animation untuk tables
- [ ] Add hover scale effect untuk cards
- [ ] Add smooth transition saat ganti tahun
- [ ] Add loading spinner untuk Select component
**Dependencies**: `framer-motion`
**Files to Modify**:
- `src/app/darmasaba/_com/main-page/apbdes/index.tsx`
- `src/app/darmasaba/_com/main-page/apbdes/lib/*.tsx`
**Estimated Time**: 1 jam
---
### **Phase 2: Data Visualization** 📈 HIGH PRIORITY
#### Task 2.1: Install & Setup Recharts
- [ ] Install Recharts (`bun add recharts`)
- [ ] Create basic bar chart component
- [ ] Add tooltip dengan formatted data
- [ ] Add responsive container
- [ ] Configure color scheme
**Dependencies**: `recharts`
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/lib/comparisonChart.tsx` (CREATE)
**Estimated Time**: 1 jam
---
#### Task 2.2: Create Interactive Charts
- [ ] Bar chart: Pagu vs Realisasi comparison
- [ ] Pie chart: Komposisi per kategori
- [ ] Line chart: Trend multi-tahun (jika data tersedia)
- [ ] Add legend dan labels
- [ ] Add export chart as image feature
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/lib/barChart.tsx` (CREATE)
- `src/app/darmasaba/_com/main-page/apbdes/lib/pieChart.tsx` (CREATE)
**Estimated Time**: 2 jam
---
#### Task 2.3: Create Summary Cards
- [ ] Design summary card component
- [ ] Display Total Pagu
- [ ] Display Total Realisasi
- [ ] Display Persentase Realisasi
- [ ] Add trend indicators (↑↓)
- [ ] Add color-coded performance badges
- [ ] Add animated number counters
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/lib/summaryCards.tsx` (CREATE)
**Estimated Time**: 1.5 jam
---
### **Phase 3: Features** ⚙️ MEDIUM PRIORITY
#### Task 3.1: Search & Filter
- [ ] Add search input untuk filter items
- [ ] Add filter dropdown by tipe (Pendapatan/Belanja/Pembiayaan)
- [ ] Add sort functionality (by jumlah, realisasi, persentase)
- [ ] Add clear filter button
- [ ] Add search result counter
**Files to Create/Modify**:
- `src/app/darmasaba/_com/main-page/apbdes/hooks/useApbdesFilter.ts` (CREATE)
- `src/app/darmasaba/_com/main-page/apbdes/index.tsx` (MODIFY)
**Estimated Time**: 1.5 jam
---
#### Task 3.2: Export & Print Functionality
- [ ] Install PDF library (`bun add @react-pdf/renderer`)
- [ ] Create PDF export template
- [ ] Add Excel export (`bun add exceljs`)
- [ ] Add print CSS styles
- [ ] Create export buttons component
- [ ] Add loading state saat export
**Dependencies**: `@react-pdf/renderer`, `exceljs`
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/components/exportButtons.tsx` (CREATE)
- `src/app/darmasaba/_com/main-page/apbdes/utils/exportPdf.ts` (CREATE)
- `src/app/darmasaba/_com/main-page/apbdes/utils/exportExcel.ts` (CREATE)
**Estimated Time**: 2 jam
---
#### Task 3.3: Detail View Modal
- [ ] Add modal component untuk detail item
- [ ] Display breakdown realisasi per item
- [ ] Add historical comparison (tahun sebelumnya)
- [ ] Add close button dan ESC key handler
- [ ] Add responsive modal design
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/components/detailModal.tsx` (CREATE)
**Estimated Time**: 1.5 jam
---
### **Phase 4: Code Quality** 🧹 MEDIUM PRIORITY
#### Task 4.1: TypeScript Improvements
- [ ] Create proper TypeScript types
- [ ] Replace all `any` dengan interfaces
- [ ] Add Zod schema validation
- [ ] Type-safe API responses
- [ ] Add generic types untuk reusable components
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/types/apbdes.ts` (CREATE)
**Files to Modify**:
- All `.tsx` files in apbdes directory
**Estimated Time**: 1.5 jam
---
#### Task 4.2: Code Cleanup
- [ ] Remove all commented code
- [ ] Remove console.logs (replace dengan proper logging)
- [ ] Add error boundaries
- [ ] Improve error messages
- [ ] Add proper ESLint comments
- [ ] Add JSDoc untuk complex functions
**Estimated Time**: 1 jam
---
#### Task 4.3: Custom Hook Refactoring
- [ ] Create `useApbdesData` custom hook
- [ ] Move data fetching logic to hook
- [ ] Add SWR/React Query for caching (optional)
- [ ] Add optimistic updates
- [ ] Add error handling di hook level
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/hooks/useApbdesData.ts` (CREATE)
**Estimated Time**: 1 jam
---
### **Phase 5: Advanced Features** 🚀 LOW PRIORITY (Optional)
#### Task 5.1: Year Comparison View
- [ ] Add multi-year selection
- [ ] Side-by-side comparison table
- [ ] Year-over-year growth calculation
- [ ] Add trend arrows dan percentage change
- [ ] Add comparison chart
**Files to Create**:
- `src/app/darmasaba/_com/main-page/apbdes/lib/yearComparison.tsx` (CREATE)
**Estimated Time**: 2 jam
---
#### Task 5.2: Dashboard Widgets
- [ ] Key metrics overview widget
- [ ] Budget utilization gauge chart
- [ ] Alert untuk over/under budget
- [ ] Quick stats summary
- [ ] Add drill-down capability
**Dependencies**: Mungkin perlu additional chart library
**Estimated Time**: 2.5 jam
---
#### Task 5.3: Responsive Mobile Optimization
- [ ] Mobile-first table design
- [ ] Collapsible sections untuk mobile
- [ ] Touch-friendly interactions
- [ ] Optimize chart untuk small screens
- [ ] Add mobile navigation
**Estimated Time**: 1.5 jam
---
## 📁 Proposed File Structure
```
src/app/darmasaba/_com/main-page/apbdes/
├── index.tsx # Main component (refactored)
├── lib/
│ ├── paguTable.tsx # Table Pagu (improved)
│ ├── realisasiTable.tsx # Table Realisasi (improved)
│ ├── grafikRealisasi.tsx # Chart component (updated)
│ ├── comparisonChart.tsx # NEW: Bar chart comparison
│ ├── barChart.tsx # NEW: Interactive bar chart
│ ├── pieChart.tsx # NEW: Pie chart visualization
│ └── summaryCards.tsx # NEW: Summary metrics cards
│ └── yearComparison.tsx # NEW: Year comparison view (optional)
├── components/
│ ├── apbdesSkeleton.tsx # NEW: Loading skeleton
│ ├── apbdesCard.tsx # NEW: Preview card
│ ├── exportButtons.tsx # NEW: Export/Print buttons
│ └── detailModal.tsx # NEW: Detail view modal
├── hooks/
│ ├── useApbdesData.ts # NEW: Data fetching hook
│ └── useApbdesFilter.ts # NEW: Search/filter hook
├── types/
│ └── apbdes.ts # NEW: TypeScript types & interfaces
└── utils/
├── exportPdf.ts # NEW: PDF export logic
└── exportExcel.ts # NEW: Excel export logic
```
---
## 📦 Required Dependencies
```bash
# Core dependencies
bun add framer-motion recharts
# Export functionality
bun add @react-pdf/renderer exceljs
# Optional: Better data fetching
bun add swr
# Type definitions
bun add -D @types/react-pdf
```
---
## 🎯 Success Criteria
### UI/UX
- [ ] Loading state implemented dengan skeleton
- [ ] Smooth animations pada semua interactions
- [ ] Modern table design dengan hover effects
- [ ] Fully responsive (mobile, tablet, desktop)
### Data Visualization
- [ ] Interactive charts (Recharts) implemented
- [ ] Summary cards dengan real-time metrics
- [ ] Color-coded performance indicators
- [ ] Responsive charts untuk semua screen sizes
### Features
- [ ] Search & filter functionality working
- [ ] Export to PDF working
- [ ] Export to Excel working
- [ ] Print view working
- [ ] Detail modal working
### Code Quality
- [ ] No `any` types (all properly typed)
- [ ] No commented code
- [ ] No console.logs in production code
- [ ] Error boundaries implemented
- [ ] Custom hooks for reusability
---
## ⏱️ Total Estimated Time
| Phase | Tasks | Estimated Time |
|-------|-------|---------------|
| Phase 1 | 3 tasks | 2.75 jam |
| Phase 2 | 3 tasks | 4.5 jam |
| Phase 3 | 3 tasks | 5 jam |
| Phase 4 | 3 tasks | 3.5 jam |
| Phase 5 | 3 tasks | 6 jam (optional) |
| **TOTAL** | **15 tasks** | **~21.75 jam** (tanpa Phase 5: ~15.75 jam) |
---
## 🚀 Recommended Implementation Order
1. **Start dengan Phase 1** (UI/UX Enhancement) - Quick wins, immediate visual improvement
2. **Continue dengan Phase 4** (Code Quality) - Clean foundation sebelum add features
3. **Move to Phase 2** (Data Visualization) - Core value add
4. **Then Phase 3** (Features) - User functionality
5. **Optional Phase 5** (Advanced) - If time permits
---
## 📝 Notes
- Prioritize tasks berdasarkan impact vs effort
- Test di berbagai screen sizes selama development
- Get user feedback setelah Phase 1 & 2 complete
- Consider A/B testing untuk new design
- Document all new components di storybook (if available)
---
## 🔗 Related Files
- Main Component: `src/app/darmasaba/_com/main-page/apbdes/index.tsx`
- State Management: `src/app/admin/(dashboard)/_state/landing-page/apbdes.ts`
- API Endpoint: `src/app/api/landingpage/apbdes/`
---
**Last Updated**: 2026-03-25
**Status**: Phase 1, 2, 4 Completed ✅
**Approved By**: Completed
---
## ✅ Completed Tasks Summary
### Phase 1: UI/UX Enhancement - DONE ✅
- ✅ Created `apbdesSkeleton.tsx` with loading skeletons for all components
- ✅ Improved table design with hover effects, striped rows, sticky headers
- ✅ Installed Framer Motion and added smooth animations
- ✅ Added loading states when changing year
- ✅ Added fade-in and slide-up transitions
### Phase 2: Data Visualization - DONE ✅
- ✅ Installed Recharts
- ✅ Created interactive comparison bar chart (Pagu vs Realisasi)
- ✅ Created summary cards with metrics and progress indicators
- ✅ Enhanced GrafikRealisasi with better visual design
- ✅ Added color-coded performance badges
### Phase 4: Code Quality - DONE ✅
- ✅ Created proper TypeScript types in `types/apbdes.ts`
- ✅ Replaced most `any` types with proper interfaces (some remain for flexibility)
- ✅ Removed commented code from main index.tsx
- ✅ Cleaned up console.logs
- ✅ Improved error handling
### Files Created:
1. `src/app/darmasaba/_com/main-page/apbdes/types/apbdes.ts` - TypeScript types
2. `src/app/darmasaba/_com/main-page/apbdes/components/apbdesSkeleton.tsx` - Loading skeletons
3. `src/app/darmasaba/_com/main-page/apbdes/lib/summaryCards.tsx` - Summary metrics cards
4. `src/app/darmasaba/_com/main-page/apbdes/lib/comparisonChart.tsx` - Recharts bar chart
5. `src/app/darmasaba/_com/main-page/apbdes/lib/paguTable.tsx` - Improved table (updated)
6. `src/app/darmasaba/_com/main-page/apbdes/lib/realisasiTable.tsx` - Improved table (updated)
7. `src/app/darmasaba/_com/main-page/apbdes/lib/grafikRealisasi.tsx` - Enhanced chart (updated)
8. `src/app/darmasaba/_com/main-page/apbdes/index.tsx` - Main component with animations (updated)
### Dependencies Installed:
- `framer-motion@12.38.0` - Animation library
- `recharts@3.8.0` - Chart library
---

View File

@@ -5,30 +5,7 @@ export default defineConfig({
test: {
globals: true,
environment: 'jsdom',
setupFiles: ['./__tests__/setup.ts'],
include: ['__tests__/**/*.test.ts'],
exclude: ['**/node_modules/**', '**/dist/**', '**/.next/**', '**/e2e/**', '**/*.test.tsx'],
coverage: {
provider: 'v8',
reporter: ['text', 'json', 'html'],
include: ['src/**/*.{ts,tsx}'],
exclude: [
'src/**/*.d.ts',
'src/**/*.stories.{ts,tsx}',
'src/app/**',
'src/lib/prisma.ts',
'src/middlewares/**',
'**/*.config.*',
],
thresholds: {
global: {
branches: 50,
functions: 50,
lines: 50,
statements: 50,
},
},
},
setupFiles: './__tests__/setup.ts',
},
resolve: {
alias: {