import { reactive, watch } from "vue";
import { throttle } from "throttle-debounce";
import type { Form, GetFormResultType } from "@/scripts/form";
import * as os from "@/os";
import { deepClone } from "@/scripts/clone";
export interface Widget
> {
id: string;
data: Partial
;
}
export interface WidgetComponentProps
> {
widget?: Widget
;
}
export interface WidgetComponentEmits
> {
(ev: "updateProps", props: P);
}
export interface WidgetComponentExpose {
name: string;
id: string | null;
configure: () => void;
}
export const useWidgetPropsManager = <
F extends Form & Record,
>(
name: string,
propsDef: F,
props: Readonly>>,
emit: WidgetComponentEmits>,
): {
widgetProps: GetFormResultType;
save: () => void;
configure: () => void;
} => {
const widgetProps = reactive(
props.widget ? deepClone(props.widget.data) : {},
);
const mergeProps = () => {
for (const prop of Object.keys(propsDef)) {
if (typeof widgetProps[prop] === "undefined") {
widgetProps[prop] = propsDef[prop].default;
}
}
};
watch(
widgetProps,
() => {
mergeProps();
},
{ deep: true, immediate: true },
);
const save = throttle(3000, () => {
emit("updateProps", widgetProps);
});
const configure = async () => {
const form = deepClone(propsDef);
for (const item of Object.keys(form)) {
form[item].default = widgetProps[item];
}
const { canceled, result } = await os.form(name, form);
if (canceled) return;
for (const key of Object.keys(result)) {
widgetProps[key] = result[key];
}
save();
};
return {
widgetProps,
save,
configure,
};
};