feat(editor): 优化代码块编辑弹窗

This commit is contained in:
roymondchen 2024-03-15 15:24:16 +08:00
parent 36a1a18615
commit c83e76e641
14 changed files with 186 additions and 135 deletions

View File

@ -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>();

View File

@ -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>

View File

@ -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';

View File

@ -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: [],
}; };

View 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,
};
};

View File

@ -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,

View 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,
};
};

View File

@ -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 => {

View File

@ -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"

View File

@ -17,9 +17,4 @@
.el-drawer__body { .el-drawer__body {
padding: 10px 20px; padding: 10px 20px;
} }
&.m-form-box {
width: 100%;
min-width: 872px;
}
} }

View File

@ -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;
} }
} }

View File

@ -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 {

View File

@ -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,

View File

@ -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;
} }
} }