mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-06 03:57:56 +08:00
feat(editor): 优化代码块编辑弹窗
This commit is contained in:
parent
36a1a18615
commit
c83e76e641
@ -1,8 +1,15 @@
|
|||||||
<template>
|
<template>
|
||||||
<!-- 代码块编辑区 -->
|
<!-- 代码块编辑区 -->
|
||||||
<FloatingBox v-model:visible="boxVisible" title="代码编辑" :position="boxPosition" :before-close="beforeClose">
|
<FloatingBox
|
||||||
|
v-model:visible="boxVisible"
|
||||||
|
v-model:width="width"
|
||||||
|
v-model:height="codeBlockEditorHeight"
|
||||||
|
:title="content.name ? `${disabled ? '查看' : '编辑'}${content.name}` : '新增代码'"
|
||||||
|
:position="boxPosition"
|
||||||
|
:before-close="beforeClose"
|
||||||
|
>
|
||||||
<template #body>
|
<template #body>
|
||||||
<div ref="floatingBoxBody"></div>
|
<div ref="floatingBoxBody" style="height: 100%"></div>
|
||||||
</template>
|
</template>
|
||||||
</FloatingBox>
|
</FloatingBox>
|
||||||
|
|
||||||
@ -13,18 +20,17 @@
|
|||||||
label-width="80px"
|
label-width="80px"
|
||||||
:close-on-press-escape="false"
|
:close-on-press-escape="false"
|
||||||
:title="content.name"
|
:title="content.name"
|
||||||
:width="size"
|
|
||||||
:config="functionConfig"
|
:config="functionConfig"
|
||||||
:values="content"
|
:values="content"
|
||||||
:disabled="disabled"
|
:disabled="disabled"
|
||||||
|
:height="floatingBoxBody?.clientHeight"
|
||||||
@change="changeHandler"
|
@change="changeHandler"
|
||||||
@submit="submitForm"
|
@submit="submitForm"
|
||||||
@error="errorHandler"
|
@error="errorHandler"
|
||||||
@open="openHandler"
|
|
||||||
@closed="closedHandler"
|
@closed="closedHandler"
|
||||||
>
|
>
|
||||||
<template #left>
|
<template #left>
|
||||||
<TMagicButton type="primary" link @click="difVisible = true">查看修改</TMagicButton>
|
<TMagicButton v-if="!disabled" type="primary" link @click="difVisible = true">查看修改</TMagicButton>
|
||||||
</template>
|
</template>
|
||||||
</MFormBox>
|
</MFormBox>
|
||||||
</Teleport>
|
</Teleport>
|
||||||
@ -42,7 +48,7 @@
|
|||||||
language="json"
|
language="json"
|
||||||
:initValues="content.content"
|
:initValues="content.content"
|
||||||
:modifiedValues="formBox?.form?.values.content"
|
:modifiedValues="formBox?.form?.values.content"
|
||||||
:style="`height: ${height - 200}px`"
|
:style="`height: ${windowRect.height - 150}px`"
|
||||||
></CodeEditor>
|
></CodeEditor>
|
||||||
|
|
||||||
<template #footer>
|
<template #footer>
|
||||||
@ -55,13 +61,15 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { computed, inject, nextTick, onBeforeUnmount, ref } from 'vue';
|
import { computed, inject, nextTick, ref } from 'vue';
|
||||||
|
|
||||||
import { TMagicButton, TMagicDialog, tMagicMessage, tMagicMessageBox, TMagicTag } from '@tmagic/design';
|
import { TMagicButton, TMagicDialog, tMagicMessage, tMagicMessageBox, TMagicTag } from '@tmagic/design';
|
||||||
import { ColumnConfig, FormConfig, FormState, MFormBox } from '@tmagic/form';
|
import { ColumnConfig, FormConfig, FormState, MFormBox } from '@tmagic/form';
|
||||||
import type { CodeBlockContent } from '@tmagic/schema';
|
import type { CodeBlockContent } from '@tmagic/schema';
|
||||||
|
|
||||||
import FloatingBox from '@editor/components/FloatingBox.vue';
|
import FloatingBox from '@editor/components/FloatingBox.vue';
|
||||||
|
import { useEditorContentHeight } from '@editor/hooks/use-editor-content-height';
|
||||||
|
import { useWindowRect } from '@editor/hooks/use-window-rect';
|
||||||
import CodeEditor from '@editor/layouts/CodeEditor.vue';
|
import CodeEditor from '@editor/layouts/CodeEditor.vue';
|
||||||
import type { Services, SlideType } from '@editor/type';
|
import type { Services, SlideType } from '@editor/type';
|
||||||
import { getConfig } from '@editor/utils/config';
|
import { getConfig } from '@editor/utils/config';
|
||||||
@ -70,6 +78,9 @@ defineOptions({
|
|||||||
name: 'MEditorCodeBlockEditor',
|
name: 'MEditorCodeBlockEditor',
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const width = defineModel<number>('width', { default: 670 });
|
||||||
|
const boxVisible = defineModel<boolean>('visible', { default: false });
|
||||||
|
|
||||||
const props = defineProps<{
|
const props = defineProps<{
|
||||||
content: CodeBlockContent;
|
content: CodeBlockContent;
|
||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
@ -84,18 +95,10 @@ const emit = defineEmits<{
|
|||||||
|
|
||||||
const services = inject<Services>('services');
|
const services = inject<Services>('services');
|
||||||
|
|
||||||
|
const { height: codeBlockEditorHeight } = useEditorContentHeight();
|
||||||
|
|
||||||
const difVisible = ref(false);
|
const difVisible = ref(false);
|
||||||
const height = ref(globalThis.innerHeight);
|
const { rect: windowRect } = useWindowRect();
|
||||||
|
|
||||||
const windowResizeHandler = () => {
|
|
||||||
height.value = globalThis.innerHeight;
|
|
||||||
};
|
|
||||||
|
|
||||||
globalThis.addEventListener('resize', windowResizeHandler);
|
|
||||||
|
|
||||||
onBeforeUnmount(() => {
|
|
||||||
globalThis.removeEventListener('resize', windowResizeHandler);
|
|
||||||
});
|
|
||||||
|
|
||||||
const magicVsEditor = ref<InstanceType<typeof CodeEditor>>();
|
const magicVsEditor = ref<InstanceType<typeof CodeEditor>>();
|
||||||
|
|
||||||
@ -109,13 +112,6 @@ const diffChange = () => {
|
|||||||
difVisible.value = false;
|
difVisible.value = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
const columnWidth = computed(() => services?.uiService.get('columnWidth'));
|
|
||||||
const size = computed(() =>
|
|
||||||
columnWidth.value ? columnWidth.value.center + columnWidth.value.right - (props.isDataSource ? 100 : 0) : 600,
|
|
||||||
);
|
|
||||||
|
|
||||||
const codeEditorHeight = ref('600px');
|
|
||||||
|
|
||||||
const defaultParamColConfig: ColumnConfig = {
|
const defaultParamColConfig: ColumnConfig = {
|
||||||
type: 'row',
|
type: 'row',
|
||||||
label: '参数类型',
|
label: '参数类型',
|
||||||
@ -199,7 +195,7 @@ const functionConfig = computed<FormConfig>(() => [
|
|||||||
name: 'content',
|
name: 'content',
|
||||||
type: 'vs-code',
|
type: 'vs-code',
|
||||||
options: inject('codeOptions', {}),
|
options: inject('codeOptions', {}),
|
||||||
height: codeEditorHeight.value,
|
height: '500px',
|
||||||
onChange: (formState: FormState | undefined, code: string) => {
|
onChange: (formState: FormState | undefined, code: string) => {
|
||||||
try {
|
try {
|
||||||
// 检测js代码是否存在语法错误
|
// 检测js代码是否存在语法错误
|
||||||
@ -226,15 +222,6 @@ const errorHandler = (error: any) => {
|
|||||||
|
|
||||||
const formBox = ref<InstanceType<typeof MFormBox>>();
|
const formBox = ref<InstanceType<typeof MFormBox>>();
|
||||||
|
|
||||||
const openHandler = () => {
|
|
||||||
setTimeout(() => {
|
|
||||||
if (formBox.value) {
|
|
||||||
const height = formBox.value?.bodyHeight - 348 - (props.isDataSource ? 50 : 0);
|
|
||||||
codeEditorHeight.value = `${height > 100 ? height : 600}px`;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
const changedValue = ref<CodeBlockContent>();
|
const changedValue = ref<CodeBlockContent>();
|
||||||
const changeHandler = (values: CodeBlockContent) => {
|
const changeHandler = (values: CodeBlockContent) => {
|
||||||
changedValue.value = values;
|
changedValue.value = values;
|
||||||
@ -270,7 +257,6 @@ const closedHandler = () => {
|
|||||||
changedValue.value = undefined;
|
changedValue.value = undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
const boxVisible = ref<boolean>(false);
|
|
||||||
const editVisible = ref<boolean>(false);
|
const editVisible = ref<boolean>(false);
|
||||||
const floatingBoxBody = ref<HTMLDivElement>();
|
const floatingBoxBody = ref<HTMLDivElement>();
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<Teleport to="body" v-if="visible">
|
<Teleport to="body" v-if="visible">
|
||||||
<div ref="target" class="m-editor-float-box" :style="{ ...style, zIndex: curZIndex }" @mousedown="nextZIndex">
|
<div ref="target" class="m-editor-float-box" :style="{ ...style, zIndex: curZIndex }" @mousedown="nextZIndex">
|
||||||
<div ref="dragTarget" class="m-editor-float-box-title">
|
<div ref="titleEl" class="m-editor-float-box-title">
|
||||||
<slot name="title">
|
<slot name="title">
|
||||||
<span>{{ title }}</span>
|
<span>{{ title }}</span>
|
||||||
</slot>
|
</slot>
|
||||||
@ -9,7 +9,7 @@
|
|||||||
<TMagicButton link size="small" :icon="Close" @click="closeHandler"></TMagicButton>
|
<TMagicButton link size="small" :icon="Close" @click="closeHandler"></TMagicButton>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="m-editor-float-box-body">
|
<div class="m-editor-float-box-body" :style="{ height: `${bodyHeight}px` }">
|
||||||
<slot name="body"></slot>
|
<slot name="body"></slot>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -17,7 +17,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { computed, nextTick, onBeforeUnmount, ref, watch, watchEffect } from 'vue';
|
import { computed, nextTick, onBeforeUnmount, ref, watch } from 'vue';
|
||||||
import { Close } from '@element-plus/icons-vue';
|
import { Close } from '@element-plus/icons-vue';
|
||||||
import VanillaMoveable from 'moveable';
|
import VanillaMoveable from 'moveable';
|
||||||
|
|
||||||
@ -28,54 +28,46 @@ interface Position {
|
|||||||
top: number;
|
top: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Rect {
|
const width = defineModel<number>('width', { default: 0 });
|
||||||
width: number | string;
|
const height = defineModel<number>('height', { default: 0 });
|
||||||
height: number | string;
|
const visible = defineModel<boolean>('visible', { default: false });
|
||||||
}
|
|
||||||
|
|
||||||
const props = withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
visible: boolean;
|
|
||||||
position?: Position;
|
position?: Position;
|
||||||
rect?: Rect;
|
|
||||||
title?: string;
|
title?: string;
|
||||||
beforeClose?: (done: (cancel?: boolean) => void) => void;
|
beforeClose?: (done: (cancel?: boolean) => void) => void;
|
||||||
}>(),
|
}>(),
|
||||||
{
|
{
|
||||||
visible: false,
|
|
||||||
title: '',
|
title: '',
|
||||||
position: () => ({ left: 0, top: 0 }),
|
position: () => ({ left: 0, top: 0 }),
|
||||||
rect: () => ({ width: 'auto', height: 'auto' }),
|
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
const emit = defineEmits<{
|
|
||||||
'update:visible': [boolean];
|
|
||||||
}>();
|
|
||||||
|
|
||||||
const target = ref<HTMLDivElement>();
|
const target = ref<HTMLDivElement>();
|
||||||
const dragTarget = ref<HTMLDivElement>();
|
const titleEl = ref<HTMLDivElement>();
|
||||||
|
|
||||||
const zIndex = useZIndex();
|
const zIndex = useZIndex();
|
||||||
const curZIndex = ref<number>(zIndex.nextZIndex());
|
const curZIndex = ref<number>(zIndex.nextZIndex());
|
||||||
|
|
||||||
const rect = ref({
|
const titleHeight = ref(0);
|
||||||
width: props.rect.width,
|
const bodyHeight = computed(() => {
|
||||||
height: props.rect.height,
|
if (height.value) {
|
||||||
});
|
return height.value - titleHeight.value;
|
||||||
|
}
|
||||||
|
|
||||||
watchEffect(() => {
|
if (target.value) {
|
||||||
rect.value = {
|
return target.value.clientHeight - titleHeight.value;
|
||||||
width: props.rect.width,
|
}
|
||||||
height: props.rect.height,
|
|
||||||
};
|
return 'auto';
|
||||||
});
|
});
|
||||||
|
|
||||||
const style = computed(() => ({
|
const style = computed(() => ({
|
||||||
left: `${props.position.left}px`,
|
left: `${props.position.left}px`,
|
||||||
top: `${props.position.top}px`,
|
top: `${props.position.top}px`,
|
||||||
width: typeof rect.value.width === 'string' ? rect.value.width : `${rect.value.width}px`,
|
width: width.value ? `${width.value}px` : 'auto',
|
||||||
height: typeof rect.value.height === 'string' ? rect.value.height : `${rect.value.height}px`,
|
height: height.value ? `${height.value}px` : 'auto',
|
||||||
}));
|
}));
|
||||||
|
|
||||||
let moveable: VanillaMoveable | null = null;
|
let moveable: VanillaMoveable | null = null;
|
||||||
@ -90,7 +82,7 @@ const initMoveable = () => {
|
|||||||
keepRatio: false,
|
keepRatio: false,
|
||||||
origin: false,
|
origin: false,
|
||||||
snappable: true,
|
snappable: true,
|
||||||
dragTarget: dragTarget.value,
|
dragTarget: titleEl.value,
|
||||||
dragTargetSelf: false,
|
dragTargetSelf: false,
|
||||||
linePadding: 10,
|
linePadding: 10,
|
||||||
controlPadding: 10,
|
controlPadding: 10,
|
||||||
@ -98,12 +90,14 @@ const initMoveable = () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
moveable.on('drag', (e) => {
|
moveable.on('drag', (e) => {
|
||||||
|
width.value = e.width;
|
||||||
|
height.value = e.height;
|
||||||
e.target.style.transform = e.transform;
|
e.target.style.transform = e.transform;
|
||||||
});
|
});
|
||||||
|
|
||||||
moveable.on('resize', (e) => {
|
moveable.on('resize', (e) => {
|
||||||
rect.value.width = e.width;
|
width.value = e.width;
|
||||||
rect.value.height = e.height;
|
height.value = e.height;
|
||||||
e.target.style.width = `${e.width}px`;
|
e.target.style.width = `${e.width}px`;
|
||||||
e.target.style.height = `${e.height}px`;
|
e.target.style.height = `${e.height}px`;
|
||||||
e.target.style.transform = e.drag.transform;
|
e.target.style.transform = e.drag.transform;
|
||||||
@ -116,11 +110,22 @@ const destroyMoveable = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => props.visible,
|
visible,
|
||||||
async (visible) => {
|
async (visible) => {
|
||||||
if (visible) {
|
if (visible) {
|
||||||
await nextTick();
|
await nextTick();
|
||||||
initMoveable();
|
|
||||||
|
const targetRect = target.value?.getBoundingClientRect();
|
||||||
|
if (targetRect) {
|
||||||
|
width.value = targetRect.width;
|
||||||
|
height.value = targetRect.height;
|
||||||
|
initMoveable();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (titleEl.value) {
|
||||||
|
const titleRect = titleEl.value.getBoundingClientRect();
|
||||||
|
titleHeight.value = titleRect.height;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
destroyMoveable();
|
destroyMoveable();
|
||||||
}
|
}
|
||||||
@ -136,7 +141,7 @@ onBeforeUnmount(() => {
|
|||||||
|
|
||||||
const hide = (cancel?: boolean) => {
|
const hide = (cancel?: boolean) => {
|
||||||
if (cancel !== false) {
|
if (cancel !== false) {
|
||||||
emit('update:visible', false);
|
visible.value = false;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -153,6 +158,8 @@ const nextZIndex = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
defineExpose({
|
defineExpose({
|
||||||
|
bodyHeight,
|
||||||
target,
|
target,
|
||||||
|
titleEl,
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
@ -20,3 +20,5 @@ export * from './use-code-block-edit';
|
|||||||
export * from './use-data-source-method';
|
export * from './use-data-source-method';
|
||||||
export * from './use-stage';
|
export * from './use-stage';
|
||||||
export * from './use-float-box';
|
export * from './use-float-box';
|
||||||
|
export * from './use-window-rect';
|
||||||
|
export * from './use-editor-content-height';
|
||||||
|
@ -20,7 +20,7 @@ export const useCodeBlockEdit = (codeBlockService?: CodeBlockService) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
codeConfig.value = {
|
codeConfig.value = {
|
||||||
name: '代码块',
|
name: '',
|
||||||
content: `({app, params}) => {\n // place your code here\n}`,
|
content: `({app, params}) => {\n // place your code here\n}`,
|
||||||
params: [],
|
params: [],
|
||||||
};
|
};
|
||||||
|
20
packages/editor/src/hooks/use-editor-content-height.ts
Normal file
20
packages/editor/src/hooks/use-editor-content-height.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import { computed, inject, ref, watchEffect } from 'vue';
|
||||||
|
|
||||||
|
import type { Services } from '@editor/type';
|
||||||
|
|
||||||
|
export const useEditorContentHeight = () => {
|
||||||
|
const services = inject<Services>('services');
|
||||||
|
const frameworkHeight = computed(() => services?.uiService.get('frameworkRect').height || 0);
|
||||||
|
const navMenuHeight = computed(() => services?.uiService.get('navMenuRect').height || 0);
|
||||||
|
const editorContentHeight = computed(() => frameworkHeight.value - navMenuHeight.value);
|
||||||
|
|
||||||
|
const height = ref(0);
|
||||||
|
watchEffect(() => {
|
||||||
|
if (height.value > 0) return;
|
||||||
|
height.value = editorContentHeight.value;
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
height,
|
||||||
|
};
|
||||||
|
};
|
@ -26,34 +26,34 @@ export const useFloatBox = (slideKeys: ComputedRef<string[]>) => {
|
|||||||
const showingBoxKeys = computed(() =>
|
const showingBoxKeys = computed(() =>
|
||||||
Object.keys(floatBoxStates.value).filter((key) => floatBoxStates.value[key].status),
|
Object.keys(floatBoxStates.value).filter((key) => floatBoxStates.value[key].status),
|
||||||
);
|
);
|
||||||
const isDraging = ref(false);
|
const isDragging = ref(false);
|
||||||
|
|
||||||
const dragstartHandler = () => (isDraging.value = true);
|
const dragstartHandler = () => (isDragging.value = true);
|
||||||
const dragendHandler = (key: string, e: DragEvent) => {
|
const dragendHandler = (key: string, e: DragEvent) => {
|
||||||
floatBoxStates.value[key] = {
|
floatBoxStates.value[key] = {
|
||||||
left: e.clientX,
|
left: e.clientX,
|
||||||
top: e.clientY,
|
top: e.clientY,
|
||||||
status: true,
|
status: true,
|
||||||
};
|
};
|
||||||
isDraging.value = false;
|
isDragging.value = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
document.body.addEventListener('dragover', (e: DragEvent) => {
|
document.body.addEventListener('dragover', (e: DragEvent) => {
|
||||||
if (!isDraging.value) return;
|
if (!isDragging.value) return;
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
});
|
});
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => slideKeys.value,
|
() => slideKeys.value,
|
||||||
() => {
|
(slideKeys) => {
|
||||||
for (const key in slideKeys.value) {
|
slideKeys.forEach((key) => {
|
||||||
if (floatBoxStates.value[key]) continue;
|
if (floatBoxStates.value[key]) return;
|
||||||
floatBoxStates.value[key] = {
|
floatBoxStates.value[key] = {
|
||||||
status: false,
|
status: false,
|
||||||
top: 0,
|
top: 0,
|
||||||
left: 0,
|
left: 0,
|
||||||
};
|
};
|
||||||
}
|
});
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
deep: true,
|
deep: true,
|
||||||
|
20
packages/editor/src/hooks/use-window-rect.ts
Normal file
20
packages/editor/src/hooks/use-window-rect.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
import { onBeforeUnmount, reactive } from 'vue';
|
||||||
|
|
||||||
|
export const useWindowRect = () => {
|
||||||
|
const rect = reactive({ width: globalThis.innerWidth, height: globalThis.innerHeight });
|
||||||
|
|
||||||
|
const windowResizeHandler = () => {
|
||||||
|
rect.width = globalThis.innerWidth;
|
||||||
|
rect.height = globalThis.innerHeight;
|
||||||
|
};
|
||||||
|
|
||||||
|
globalThis.addEventListener('resize', windowResizeHandler);
|
||||||
|
|
||||||
|
onBeforeUnmount(() => {
|
||||||
|
globalThis.removeEventListener('resize', windowResizeHandler);
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
rect,
|
||||||
|
};
|
||||||
|
};
|
@ -111,6 +111,8 @@
|
|||||||
:key="config.$key ?? index"
|
:key="config.$key ?? index"
|
||||||
v-if="floatBoxStates[config.$key]?.status"
|
v-if="floatBoxStates[config.$key]?.status"
|
||||||
v-model:visible="floatBoxStates[config.$key].status"
|
v-model:visible="floatBoxStates[config.$key].status"
|
||||||
|
:width="columnLeftWitch"
|
||||||
|
:height="600"
|
||||||
:title="config.text"
|
:title="config.text"
|
||||||
:position="{
|
:position="{
|
||||||
left: floatBoxStates[config.$key].left,
|
left: floatBoxStates[config.$key].left,
|
||||||
@ -139,14 +141,15 @@ import { Coin, EditPen, Goods, List } from '@element-plus/icons-vue';
|
|||||||
import FloatingBox from '@editor/components/FloatingBox.vue';
|
import FloatingBox from '@editor/components/FloatingBox.vue';
|
||||||
import MIcon from '@editor/components/Icon.vue';
|
import MIcon from '@editor/components/Icon.vue';
|
||||||
import { useFloatBox } from '@editor/hooks/use-float-box';
|
import { useFloatBox } from '@editor/hooks/use-float-box';
|
||||||
import type {
|
import {
|
||||||
MenuButton,
|
ColumnLayout,
|
||||||
MenuComponent,
|
type MenuButton,
|
||||||
Services,
|
type MenuComponent,
|
||||||
SideBarData,
|
type Services,
|
||||||
SidebarSlots,
|
type SideBarData,
|
||||||
SideComponent,
|
type SidebarSlots,
|
||||||
SideItem,
|
type SideComponent,
|
||||||
|
type SideItem,
|
||||||
} from '@editor/type';
|
} from '@editor/type';
|
||||||
|
|
||||||
import CodeBlockListPanel from './code-block/CodeBlockListPanel.vue';
|
import CodeBlockListPanel from './code-block/CodeBlockListPanel.vue';
|
||||||
@ -173,6 +176,8 @@ const props = withDefaults(
|
|||||||
|
|
||||||
const services = inject<Services>('services');
|
const services = inject<Services>('services');
|
||||||
|
|
||||||
|
const columnLeftWitch = computed(() => services?.uiService.get('columnWidth')[ColumnLayout.LEFT] || 0);
|
||||||
|
|
||||||
const activeTabName = ref(props.data?.status);
|
const activeTabName = ref(props.data?.status);
|
||||||
|
|
||||||
const getItemConfig = (data: SideItem): SideComponent => {
|
const getItemConfig = (data: SideItem): SideComponent => {
|
||||||
|
@ -2,6 +2,7 @@
|
|||||||
<TMagicTooltip v-if="page && buttonVisible" content="点击查看当前位置下的组件">
|
<TMagicTooltip v-if="page && buttonVisible" content="点击查看当前位置下的组件">
|
||||||
<div ref="button" class="m-editor-stage-float-button" @click="visible = true">可选组件</div>
|
<div ref="button" class="m-editor-stage-float-button" @click="visible = true">可选组件</div>
|
||||||
</TMagicTooltip>
|
</TMagicTooltip>
|
||||||
|
|
||||||
<FloatingBox
|
<FloatingBox
|
||||||
v-if="page && nodeStatusMap && buttonVisible"
|
v-if="page && nodeStatusMap && buttonVisible"
|
||||||
ref="box"
|
ref="box"
|
||||||
|
@ -17,9 +17,4 @@
|
|||||||
.el-drawer__body {
|
.el-drawer__body {
|
||||||
padding: 10px 20px;
|
padding: 10px 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
&.m-form-box {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 872px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -21,9 +21,9 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.m-editor-float-box-body {
|
.m-editor-float-box-body {
|
||||||
padding: 5px;
|
|
||||||
flex: 1;
|
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
|
flex: 1;
|
||||||
|
padding: 0 16px;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -246,6 +246,12 @@ export interface UiState {
|
|||||||
width: number;
|
width: number;
|
||||||
height: number;
|
height: number;
|
||||||
};
|
};
|
||||||
|
frameworkRect: {
|
||||||
|
left: number;
|
||||||
|
top: number;
|
||||||
|
width: number;
|
||||||
|
height: number;
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface EditorNodeInfo {
|
export interface EditorNodeInfo {
|
||||||
|
@ -1,56 +1,57 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="m-form-box">
|
<div class="m-form-box" :style="style">
|
||||||
<div ref="boxBody" class="m-box-body">
|
<div class="m-box-body" :style="bodyHeight ? { height: `${bodyHeight}px` } : {}">
|
||||||
<Form
|
<TMagicScrollbar>
|
||||||
ref="form"
|
<Form
|
||||||
:size="size"
|
ref="form"
|
||||||
:disabled="disabled"
|
:size="size"
|
||||||
:config="config"
|
:disabled="disabled"
|
||||||
:init-values="values"
|
:config="config"
|
||||||
:parent-values="parentValues"
|
:init-values="values"
|
||||||
:label-width="labelWidth"
|
:parent-values="parentValues"
|
||||||
:label-position="labelPosition"
|
:label-width="labelWidth"
|
||||||
:inline="inline"
|
:label-position="labelPosition"
|
||||||
@change="changeHandler"
|
:inline="inline"
|
||||||
></Form>
|
@change="changeHandler"
|
||||||
<slot></slot>
|
></Form>
|
||||||
|
<slot></slot>
|
||||||
|
</TMagicScrollbar>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<TMagicRow class="dialog-footer">
|
<div class="dialog-footer" :style="`height: ${footerHeight}px`">
|
||||||
<TMagicCol :span="12" style="text-align: left">
|
<div>
|
||||||
<div style="min-height: 1px">
|
<slot name="left"></slot>
|
||||||
<slot name="left"></slot>
|
</div>
|
||||||
</div>
|
<div>
|
||||||
</TMagicCol>
|
|
||||||
<TMagicCol :span="12">
|
|
||||||
<slot name="footer">
|
<slot name="footer">
|
||||||
<TMagicButton type="primary" :disabled="disabled" :loading="saveFetch" @click="submitHandler">{{
|
<TMagicButton type="primary" :size="size" :disabled="disabled" :loading="saveFetch" @click="submitHandler">{{
|
||||||
confirmText
|
confirmText
|
||||||
}}</TMagicButton>
|
}}</TMagicButton>
|
||||||
</slot>
|
</slot>
|
||||||
</TMagicCol>
|
</div>
|
||||||
</TMagicRow>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, watchEffect } from 'vue';
|
import { computed, ref, watchEffect } from 'vue';
|
||||||
|
|
||||||
import { TMagicButton, TMagicCol, TMagicRow } from '@tmagic/design';
|
import { TMagicButton, TMagicScrollbar } from '@tmagic/design';
|
||||||
|
|
||||||
import Form from './Form.vue';
|
import Form from './Form.vue';
|
||||||
import type { FormConfig } from './schema';
|
import type { FormConfig } from './schema';
|
||||||
|
|
||||||
defineOptions({
|
defineOptions({
|
||||||
name: 'MFormDialog',
|
name: 'MFormBox',
|
||||||
});
|
});
|
||||||
|
|
||||||
withDefaults(
|
const props = withDefaults(
|
||||||
defineProps<{
|
defineProps<{
|
||||||
config?: FormConfig;
|
config?: FormConfig;
|
||||||
values?: Object;
|
values?: Object;
|
||||||
parentValues?: Object;
|
parentValues?: Object;
|
||||||
width?: string | number;
|
width?: number;
|
||||||
|
height?: number;
|
||||||
labelWidth?: string;
|
labelWidth?: string;
|
||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
size?: 'small' | 'default' | 'large';
|
size?: 'small' | 'default' | 'large';
|
||||||
@ -67,15 +68,29 @@ withDefaults(
|
|||||||
|
|
||||||
const emit = defineEmits(['submit', 'change', 'error']);
|
const emit = defineEmits(['submit', 'change', 'error']);
|
||||||
|
|
||||||
|
const footerHeight = 60;
|
||||||
|
|
||||||
|
const style = computed(() => {
|
||||||
|
const style: { width?: string; height?: string } = {};
|
||||||
|
if (typeof props.width === 'number') {
|
||||||
|
style.width = `${props.width}px`;
|
||||||
|
}
|
||||||
|
if (typeof props.height === 'number') {
|
||||||
|
style.height = `${props.height}px`;
|
||||||
|
}
|
||||||
|
return style;
|
||||||
|
});
|
||||||
|
|
||||||
const form = ref<InstanceType<typeof Form>>();
|
const form = ref<InstanceType<typeof Form>>();
|
||||||
const boxBody = ref<HTMLDivElement>();
|
|
||||||
const saveFetch = ref(false);
|
const saveFetch = ref(false);
|
||||||
|
|
||||||
const bodyHeight = ref(0);
|
const bodyHeight = ref(0);
|
||||||
|
|
||||||
watchEffect(() => {
|
watchEffect(() => {
|
||||||
if (boxBody.value) {
|
if (!props.height) {
|
||||||
bodyHeight.value = boxBody.value.clientHeight;
|
return;
|
||||||
}
|
}
|
||||||
|
bodyHeight.value = props.height - footerHeight;
|
||||||
});
|
});
|
||||||
|
|
||||||
const submitHandler = async () => {
|
const submitHandler = async () => {
|
||||||
@ -98,7 +113,6 @@ const hide = () => {};
|
|||||||
defineExpose({
|
defineExpose({
|
||||||
form,
|
form,
|
||||||
saveFetch,
|
saveFetch,
|
||||||
bodyHeight,
|
|
||||||
|
|
||||||
show,
|
show,
|
||||||
hide,
|
hide,
|
||||||
|
@ -1,18 +1,13 @@
|
|||||||
.m-form-box {
|
.m-form-box {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
padding: 16px;
|
|
||||||
box-sizing: border-box;
|
|
||||||
.el-box__header {
|
.el-box__header {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.m-box-body {
|
|
||||||
flex: 1;
|
|
||||||
overflow-y: auto;
|
|
||||||
overflow-x: hidden;
|
|
||||||
}
|
|
||||||
.dialog-footer {
|
.dialog-footer {
|
||||||
margin-top: 16px;
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user