mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-12-10 18:02:53 +00:00
185 lines
4.4 KiB
Vue
185 lines
4.4 KiB
Vue
<template>
|
|
<div class="editor-app">
|
|
<TMagicEditor
|
|
v-model="value"
|
|
ref="editor"
|
|
:menu="menu"
|
|
:runtime-url="runtimeUrl"
|
|
:props-configs="propsConfigs"
|
|
:props-values="propsValues"
|
|
:event-method-list="eventMethodList"
|
|
:datasource-event-method-list="datasourceEventMethodList"
|
|
:datasource-configs="datasourceConfigs"
|
|
:datasource-values="datasourceValues"
|
|
:component-group-list="componentGroupList"
|
|
:datasource-list="datasourceList"
|
|
:default-selected="defaultSelected"
|
|
:moveable-options="moveableOptions"
|
|
:auto-scroll-into-view="true"
|
|
:stage-rect="stageRect"
|
|
:layerContentMenu="contentMenuData"
|
|
:stageContentMenu="contentMenuData"
|
|
@props-submit-error="propsSubmitErrorHandler"
|
|
>
|
|
<template #workspace-content>
|
|
<DeviceGroup ref="deviceGroup" v-model="stageRect"></DeviceGroup>
|
|
</template>
|
|
</TMagicEditor>
|
|
|
|
<TMagicDialog
|
|
v-model="previewVisible"
|
|
close-onClick-modal
|
|
destroy-on-close
|
|
class="pre-viewer"
|
|
title="预览"
|
|
:width="stageRect?.width"
|
|
>
|
|
<iframe
|
|
v-if="previewVisible"
|
|
ref="iframe"
|
|
width="100%"
|
|
style="border: none"
|
|
:height="stageRect?.height"
|
|
:src="previewUrl"
|
|
></iframe>
|
|
</TMagicDialog>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts" setup>
|
|
import { computed, onBeforeUnmount, ref, shallowRef, toRaw } from 'vue';
|
|
import serialize from 'serialize-javascript';
|
|
|
|
import type { MApp, MContainer, MNode } from '@tmagic/core';
|
|
import type { DatasourceTypeOption } from '@tmagic/editor';
|
|
import { editorService, propsService, TMagicDialog, TMagicEditor, tMagicMessage } from '@tmagic/editor';
|
|
|
|
import DeviceGroup from '../components/DeviceGroup.vue';
|
|
import componentGroupList from '../configs/componentGroupList';
|
|
import dsl from '../configs/dsl';
|
|
|
|
import { useEditorContentMenuData } from './composables/use-editor-content-menu-data';
|
|
import { useEditorMenu } from './composables/use-editor-menu';
|
|
import { useEditorMoveableOptions } from './composables/use-editor-moveable-options';
|
|
import { useEditorRes } from './composables/use-editor-res';
|
|
|
|
const { VITE_RUNTIME_PATH } = import.meta.env;
|
|
|
|
const datasourceList: DatasourceTypeOption[] = [];
|
|
const runtimeUrl = `${VITE_RUNTIME_PATH}/playground/index.html`;
|
|
|
|
const { propsValues, propsConfigs, eventMethodList, datasourceConfigs, datasourceValues, datasourceEventMethodList } =
|
|
useEditorRes();
|
|
const { contentMenuData } = useEditorContentMenuData();
|
|
|
|
const editor = shallowRef<InstanceType<typeof TMagicEditor>>();
|
|
const value = ref<MApp>(dsl);
|
|
const defaultSelected = ref(dsl.items[0].id);
|
|
|
|
const stageRect = ref({
|
|
width: 375,
|
|
height: 817,
|
|
});
|
|
|
|
const previewUrl = computed(
|
|
() => `${VITE_RUNTIME_PATH}/page/index.html?localPreview=1&page=${editor.value?.editorService.get('page')?.id}`,
|
|
);
|
|
|
|
const { moveableOptions } = useEditorMoveableOptions(editor);
|
|
|
|
const save = () => {
|
|
localStorage.setItem(
|
|
'magicDSL',
|
|
serialize(toRaw(value.value), {
|
|
space: 2,
|
|
unsafe: true,
|
|
}).replace(/"(\w+)":\s/g, '$1: '),
|
|
);
|
|
editor.value?.editorService.resetModifiedNodeId();
|
|
};
|
|
|
|
const { menu, deviceGroup, iframe, previewVisible } = useEditorMenu(value, save);
|
|
|
|
try {
|
|
// eslint-disable-next-line no-eval
|
|
const magicDSL = eval(`(${localStorage.getItem('magicDSL')})`);
|
|
if (!magicDSL) {
|
|
save();
|
|
} else {
|
|
value.value = magicDSL;
|
|
}
|
|
} catch (e) {
|
|
console.error(e);
|
|
save();
|
|
}
|
|
|
|
editorService.usePlugin({
|
|
beforeDoAdd: (config: MNode, parent: MContainer) => {
|
|
if (config.type === 'overlay') {
|
|
config.style = {
|
|
...config.style,
|
|
left: 0,
|
|
top: 0,
|
|
};
|
|
|
|
return [config, editorService.get('page') as MContainer];
|
|
}
|
|
|
|
return [config, parent];
|
|
},
|
|
});
|
|
|
|
propsService.usePlugin({
|
|
beforeFillConfig: (config) => [config, '100px'],
|
|
});
|
|
|
|
onBeforeUnmount(() => {
|
|
editorService.removeAllPlugins();
|
|
});
|
|
|
|
const propsSubmitErrorHandler = async (e: any) => {
|
|
console.error(e);
|
|
tMagicMessage.closeAll();
|
|
tMagicMessage.error(e.message);
|
|
};
|
|
</script>
|
|
|
|
<style lang="scss">
|
|
html {
|
|
overflow: hidden;
|
|
}
|
|
#app {
|
|
width: 100%;
|
|
height: 100%;
|
|
display: flex;
|
|
}
|
|
|
|
.editor-app {
|
|
width: 100%;
|
|
height: 100%;
|
|
|
|
.m-editor {
|
|
flex: 1;
|
|
height: 100%;
|
|
}
|
|
|
|
.el-overlay-dialog {
|
|
display: flex;
|
|
}
|
|
|
|
.pre-viewer {
|
|
margin: auto;
|
|
|
|
.el-dialog__body {
|
|
padding: 0;
|
|
}
|
|
}
|
|
|
|
.menu-left {
|
|
.menu-item-text {
|
|
margin-left: 10px;
|
|
}
|
|
}
|
|
}
|
|
</style>
|