2023-01-13 05:40:33 +01:00
|
|
|
import { reactive, watch } from "vue";
|
|
|
|
import { throttle } from "throttle-debounce";
|
2023-09-02 01:27:33 +02:00
|
|
|
import type { Form, GetFormResultType } from "@/scripts/form";
|
2023-01-13 05:40:33 +01:00
|
|
|
import * as os from "@/os";
|
|
|
|
import { deepClone } from "@/scripts/clone";
|
2022-01-08 12:30:01 +01:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
export interface Widget<P extends Record<string, unknown>> {
|
2022-01-08 12:30:01 +01:00
|
|
|
id: string;
|
|
|
|
data: Partial<P>;
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2022-01-08 12:30:01 +01:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
export interface WidgetComponentProps<P extends Record<string, unknown>> {
|
2022-01-08 12:30:01 +01:00
|
|
|
widget?: Widget<P>;
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2022-01-08 12:30:01 +01:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
export interface WidgetComponentEmits<P extends Record<string, unknown>> {
|
2023-01-13 05:40:33 +01:00
|
|
|
(ev: "updateProps", props: P);
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2022-01-08 12:30:01 +01:00
|
|
|
|
2023-09-02 01:27:33 +02:00
|
|
|
export interface WidgetComponentExpose {
|
2022-01-08 12:30:01 +01:00
|
|
|
name: string;
|
|
|
|
id: string | null;
|
|
|
|
configure: () => void;
|
2023-09-02 01:27:33 +02:00
|
|
|
}
|
2022-01-08 12:30:01 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export const useWidgetPropsManager = <
|
|
|
|
F extends Form & Record<string, { default: any }>,
|
|
|
|
>(
|
2022-01-08 12:30:01 +01:00
|
|
|
name: string,
|
|
|
|
propsDef: F,
|
|
|
|
props: Readonly<WidgetComponentProps<GetFormResultType<F>>>,
|
|
|
|
emit: WidgetComponentEmits<GetFormResultType<F>>,
|
|
|
|
): {
|
|
|
|
widgetProps: GetFormResultType<F>;
|
|
|
|
save: () => void;
|
|
|
|
configure: () => void;
|
|
|
|
} => {
|
2023-01-13 05:40:33 +01:00
|
|
|
const widgetProps = reactive(
|
|
|
|
props.widget ? deepClone(props.widget.data) : {},
|
|
|
|
);
|
2022-01-08 12:30:01 +01:00
|
|
|
|
|
|
|
const mergeProps = () => {
|
|
|
|
for (const prop of Object.keys(propsDef)) {
|
2023-01-13 05:40:33 +01:00
|
|
|
if (typeof widgetProps[prop] === "undefined") {
|
2022-07-04 16:39:04 +02:00
|
|
|
widgetProps[prop] = propsDef[prop].default;
|
|
|
|
}
|
2022-01-08 12:30:01 +01:00
|
|
|
}
|
|
|
|
};
|
2023-01-13 05:40:33 +01:00
|
|
|
watch(
|
|
|
|
widgetProps,
|
|
|
|
() => {
|
|
|
|
mergeProps();
|
|
|
|
},
|
|
|
|
{ deep: true, immediate: true },
|
|
|
|
);
|
2022-01-08 12:30:01 +01:00
|
|
|
|
|
|
|
const save = throttle(3000, () => {
|
2023-01-13 05:40:33 +01:00
|
|
|
emit("updateProps", widgetProps);
|
2022-01-08 12:30:01 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
const configure = async () => {
|
2022-11-17 01:31:07 +01:00
|
|
|
const form = deepClone(propsDef);
|
2022-01-08 12:30:01 +01:00
|
|
|
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,
|
|
|
|
};
|
|
|
|
};
|