152 lines
4.2 KiB
TypeScript

import { computed, nextTick, onBeforeUnmount, reactive, ref } from 'vue';
import type { Id, MApp, MNode } from '@tmagic/core';
import TMagicApp from '@tmagic/core';
import type { FormConfig, MForm, RemoveData, UpdateData } from '@tmagic/editor';
import { getElById, getNodePath, initValue, replaceChildNode } from '@tmagic/editor';
import { AppProps } from './types';
export const useFormConfig = (props: AppProps) => {
const { contentWindow } = props.stage.renderer!;
const mForm = ref<InstanceType<typeof MForm>>();
const root = ref<MApp>();
const values = ref({});
const curPageId = ref<Id>();
const selectedId = ref<Id>();
const config = computed(
() => root.value?.items?.find((item: MNode) => item.id === curPageId.value) || root.value?.items?.[0],
);
// @ts-ignore
const formConfig = computed(() => props.fillConfig((config.value?.items || []) as FormConfig, mForm));
let app: TMagicApp | undefined = new TMagicApp({
ua: contentWindow?.navigator.userAgent,
disabledFlexible: true,
platform: 'editor',
});
// @ts-ignore
app.mForm = mForm;
const resetValues = () => {
initValue(mForm.value?.formState, {
initValues: {},
config: formConfig.value,
}).then((value) => {
values.value = value;
});
};
const runtimeReadyHandler = ({ data }: any) => {
if (!data.tmagicRuntimeReady) {
return;
}
contentWindow?.magic?.onRuntimeReady({
getApp() {
return app;
},
updateRootConfig(config: MApp) {
root.value = config;
app?.setConfig(config, curPageId.value);
},
updatePageId(id: Id) {
curPageId.value = id;
app?.setPage(id);
},
select(id: Id) {
selectedId.value = id;
if (app?.getPage(id)) {
this.updatePageId?.(id);
}
const el = getElById()(document, `${id}`);
if (el) return el;
// 未在当前文档下找到目标元素,可能是还未渲染,等待渲染完成后再尝试获取
return nextTick().then(() => getElById()(document, `${id}`) as HTMLElement);
},
add({ config, parentId }: UpdateData) {
if (!root.value) throw new Error('error');
if (!selectedId.value) throw new Error('error');
if (!parentId) throw new Error('error');
const parent = getNodePath(parentId, [root.value]).pop();
if (!parent) throw new Error('未找到父节点');
if (config.type !== 'page') {
const parentNode = app?.page?.getNode(parent.id);
parentNode && app?.page?.initNode(config, parentNode);
}
if (parent.id !== selectedId.value) {
const index = parent.items?.findIndex((child: MNode) => child.id === selectedId.value);
parent.items?.splice(index + 1, 0, config);
} else {
// 新增节点添加到配置中
parent.items?.push(config);
}
resetValues();
},
update({ config, parentId }: UpdateData) {
if (!root.value || !app) throw new Error('error');
const newNode = app.dataSourceManager?.compiledNode(config) || config;
replaceChildNode(reactive(newNode), [root.value], parentId);
const nodeInstance = app.page?.getNode(config.id);
if (nodeInstance) {
nodeInstance.setData(config);
}
resetValues();
},
remove({ id, parentId }: RemoveData) {
if (!root.value) throw new Error('error');
const node = getNodePath(id, [root.value]).pop();
if (!node) throw new Error('未找到目标元素');
const parent = getNodePath(parentId, [root.value]).pop();
if (!parent) throw new Error('未找到父元素');
if (node.type === 'page') {
app?.deletePage();
} else {
app?.page?.deleteNode(node.id);
}
const index = parent.items?.findIndex((child: MNode) => child.id === node.id);
parent.items.splice(index, 1);
resetValues();
},
});
};
contentWindow?.addEventListener('message', runtimeReadyHandler);
onBeforeUnmount(() => {
app?.destroy();
app = undefined;
contentWindow?.removeEventListener('message', runtimeReadyHandler);
});
return {
mForm,
config,
formConfig,
values,
};
};