adjustment on picker dialog procedure

refactor(procedure-picker): improve component structure and add readonly support

- Reorganize imports and add type imports section
- Replace custom Button with ButtonAction component
- Add readonly state handling for fields and buttons
- Improve type safety with ProcedureSrc type casting

feat(usePaginatedList): add syncToUrl option for nested components

Add syncToUrl option to disable URL synchronization when used in modals or nested components to prevent overriding parent page URL. Default remains true for backward compatibility.

Also includes minor formatting improvements in procedure-list.vue template.
This commit is contained in:
Khafid Prayoga
2025-11-28 12:45:45 +07:00
parent 1fbd20d9ae
commit ccefb69f0c
4 changed files with 77 additions and 55 deletions
+28 -8
View File
@@ -2,7 +2,6 @@ import type { DataTableLoader } from '~/components/pub/my-ui/data-table/type'
import type { PaginationMeta } from '~/components/pub/my-ui/pagination/pagination.type'
import { refDebounced, useUrlSearchParams } from '@vueuse/core'
import * as z from 'zod'
import { is } from "date-fns/locale"
// Default query schema yang bisa digunakan semua list
export const defaultQuerySchema = z.object({
@@ -38,10 +37,23 @@ interface UsePaginatedListOptions<T = any> {
}>
// Nama endpoint untuk logging error
entityName: string
/**
* Apakah state harus disinkronkan ke URL Browser?
* Set `false` jika digunakan di dalam Modal, Drawer, atau Nested Component
* agar tidak menimpa URL halaman induk.
* @default true
*/
syncToUrl?: boolean
}
export function usePaginatedList<T = any>(options: UsePaginatedListOptions<T>) {
const { querySchema = defaultQuerySchema, defaultQuery = defaultQueryParams, fetchFn, entityName } = options
const {
querySchema = defaultQuerySchema,
defaultQuery = defaultQueryParams,
fetchFn,
entityName,
syncToUrl = true, // Default true agar behavior lama tetap jalan
} = options
// State management
const data = ref<T[]>([])
@@ -49,11 +61,19 @@ export function usePaginatedList<T = any>(options: UsePaginatedListOptions<T>) {
isTableLoading: false,
})
// URL state management
const queryParams = useUrlSearchParams('history', {
initialValue: defaultQuery,
removeFalsyValues: true,
})
let queryParams: any
if (syncToUrl) {
// Mode Halaman Utama: Sync ke URL
queryParams = useUrlSearchParams('history', {
initialValue: defaultQuery,
removeFalsyValues: true,
write: false,
})
} else {
// Mode Nested/Modal: Local Reactive State
queryParams = reactive({ ...defaultQuery })
}
const params = computed(() => {
const result = querySchema.safeParse(queryParams)
@@ -168,7 +188,7 @@ export function usePaginatedList<T = any>(options: UsePaginatedListOptions<T>) {
}
}
export function transform(endpoint: string ,params: any): string {
export function transform(endpoint: string, params: any): string {
const urlParams = new URLSearchParams()
Object.entries(params).forEach(([key, value]) => {