feat(create): add realisasi awal input di create page

Features:
- Add realisasiAwal field to ItemForm type
- Add NumberInput for realisasi awal (optional)
- Update table preview to show realisasi awal
- Update state to send realisasiAwal to API
- Update API create to handle realisasiAwal:
  * Create APBDesItem with totalRealisasi = realisasiAwal
  * Auto-create first RealisasiItem if realisasiAwal > 0
  * Auto-calculate selisih and persentase

UX Improvements:
- User can input initial realization during create
- Optional field with clear label and description
- Auto-calculation of percentages on backend
- Single transaction for item + first realisasi

Co-authored-by: Qwen-Coder <qwen-coder@alibabacloud.com>
This commit is contained in:
2026-03-03 15:13:58 +08:00
parent 63682e47b6
commit e0436cc384
3 changed files with 59 additions and 13 deletions

View File

@@ -10,6 +10,7 @@ type APBDesItemInput = {
anggaran: number;
level: number;
tipe?: string | null;
realisasiAwal?: number; // Realisasi pertama saat create
};
type FormCreate = {
@@ -55,14 +56,15 @@ export default async function apbdesCreate(context: Context) {
},
});
// Create items dengan auto-calculate totalRealisasi=0, selisih, persentase
// Create items dengan auto-calculate totalRealisasi, selisih, persentase
const items = await Promise.all(
body.items.map(item => {
body.items.map(async item => {
const anggaran = item.anggaran;
const totalRealisasi = 0; // Belum ada realisasi saat create
const selisih = totalRealisasi - anggaran;
const persentase = anggaran > 0 ? (totalRealisasi / anggaran) * 100 : 0;
const realisasiAwal = item.realisasiAwal || 0;
// Jika ada realisasiAwal, buat realisasi item pertama
let totalRealisasi = realisasiAwal;
const itemData = {
kode: item.kode,
uraian: item.uraian,
@@ -70,15 +72,29 @@ export default async function apbdesCreate(context: Context) {
level: item.level,
tipe: item.tipe || null,
totalRealisasi,
selisih,
persentase,
selisih: totalRealisasi - anggaran,
persentase: anggaran > 0 ? (totalRealisasi / anggaran) * 100 : 0,
apbdesId: apbdes.id,
};
return prisma.aPBDesItem.create({
const createdItem = await prisma.aPBDesItem.create({
data: itemData,
select: { id: true, kode: true },
});
// Jika ada realisasiAwal, buat realisasi item pertama
if (realisasiAwal > 0) {
await prisma.realisasiItem.create({
data: {
apbdesItemId: createdItem.id,
jumlah: realisasiAwal,
tanggal: new Date(),
keterangan: 'Realisasi awal saat create APBDes',
},
});
}
return createdItem;
})
);