fix(editor,stage,utils): 修复在ipad等大屏场景下编辑画布中元素位置计算偏差问题 (#598)

This commit is contained in:
Evan Wu 2024-04-25 20:13:21 +08:00 committed by GitHub
parent 71c90d1d4d
commit e39a7d140f
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 45 additions and 30 deletions

View File

@ -45,7 +45,8 @@ import { computed, inject, markRaw, nextTick, onBeforeUnmount, onMounted, ref, t
import { cloneDeep } from 'lodash-es'; import { cloneDeep } from 'lodash-es';
import type { MApp, MContainer } from '@tmagic/schema'; import type { MApp, MContainer } from '@tmagic/schema';
import StageCore, { calcValueByFontsize, getOffset, Runtime } from '@tmagic/stage'; import StageCore, { getOffset, Runtime } from '@tmagic/stage';
import { calcValueByFontsize } from '@tmagic/utils';
import ScrollViewer from '@editor/components/ScrollViewer.vue'; import ScrollViewer from '@editor/components/ScrollViewer.vue';
import { useStage } from '@editor/hooks/use-stage'; import { useStage } from '@editor/hooks/use-stage';
@ -223,18 +224,18 @@ const dropHandler = async (e: DragEvent) => {
top = e.clientY - containerRect.top + scrollTop; top = e.clientY - containerRect.top + scrollTop;
left = e.clientX - containerRect.left + scrollLeft; left = e.clientX - containerRect.left + scrollLeft;
if (parentEl && doc) { if (parentEl) {
const { left: parentLeft, top: parentTop } = getOffset(parentEl); const { left: parentLeft, top: parentTop } = getOffset(parentEl);
left = left - calcValueByFontsize(doc, parentLeft) * zoom.value; left = left - parentLeft * zoom.value;
top = top - calcValueByFontsize(doc, parentTop) * zoom.value; top = top - parentTop * zoom.value;
} }
} }
config.data.style = { config.data.style = {
...style, ...style,
position, position,
top: top / zoom.value, top: calcValueByFontsize(doc, top / zoom.value),
left: left / zoom.value, left: calcValueByFontsize(doc, left / zoom.value),
}; };
config.data.inputEvent = e; config.data.inputEvent = e;

View File

@ -23,7 +23,7 @@ import { Writable } from 'type-fest';
import { DepTargetType } from '@tmagic/dep'; import { DepTargetType } from '@tmagic/dep';
import type { Id, MApp, MComponent, MContainer, MNode, MPage, MPageFragment } from '@tmagic/schema'; import type { Id, MApp, MComponent, MContainer, MNode, MPage, MPageFragment } from '@tmagic/schema';
import { NodeType } from '@tmagic/schema'; import { NodeType } from '@tmagic/schema';
import { getNodePath, isNumber, isPage, isPageFragment, isPop } from '@tmagic/utils'; import { calcValueByFontsize, getNodePath, isNumber, isPage, isPageFragment, isPop } from '@tmagic/utils';
import BaseService from '@editor/services//BaseService'; import BaseService from '@editor/services//BaseService';
import propsService from '@editor/services//props'; import propsService from '@editor/services//props';
@ -741,7 +741,7 @@ class Editor extends BaseService {
const el = doc.getElementById(`${node.id}`); const el = doc.getElementById(`${node.id}`);
const parentEl = layout === Layout.FIXED ? doc.body : el?.offsetParent; const parentEl = layout === Layout.FIXED ? doc.body : el?.offsetParent;
if (parentEl && el) { if (parentEl && el) {
node.style.left = (parentEl.clientWidth - el.clientWidth) / 2; node.style.left = calcValueByFontsize(doc, (parentEl.clientWidth - el.clientWidth) / 2);
node.style.right = ''; node.style.right = '';
} }
} else if (parent.style && isNumber(parent.style?.width) && isNumber(node.style?.width)) { } else if (parent.style && isNumber(parent.style?.width) && isNumber(node.style?.width)) {

View File

@ -2,7 +2,7 @@ import { computed, markRaw, Ref } from 'vue';
import { CopyDocument, Delete, DocumentCopy } from '@element-plus/icons-vue'; import { CopyDocument, Delete, DocumentCopy } from '@element-plus/icons-vue';
import { Id, MContainer, NodeType } from '@tmagic/schema'; import { Id, MContainer, NodeType } from '@tmagic/schema';
import { isPage, isPageFragment } from '@tmagic/utils'; import { calcValueByFontsize, isPage, isPageFragment } from '@tmagic/utils';
import ContentMenu from '@editor/components/ContentMenu.vue'; import ContentMenu from '@editor/components/ContentMenu.vue';
import type { MenuButton, Services } from '@editor/type'; import type { MenuButton, Services } from '@editor/type';
@ -46,8 +46,12 @@ export const usePasteMenu = (menu?: Ref<InstanceType<typeof ContentMenu> | undef
const stage = services?.editorService?.get('stage'); const stage = services?.editorService?.get('stage');
const rect = menu.value.$el.getBoundingClientRect(); const rect = menu.value.$el.getBoundingClientRect();
const parentRect = stage?.container?.getBoundingClientRect(); const parentRect = stage?.container?.getBoundingClientRect();
const initialLeft = (rect.left || 0) - (parentRect?.left || 0); const initialLeft =
const initialTop = (rect.top || 0) - (parentRect?.top || 0); calcValueByFontsize(stage?.renderer.getDocument(), (rect.left || 0) - (parentRect?.left || 0)) /
services.uiService.get('zoom');
const initialTop =
calcValueByFontsize(stage?.renderer.getDocument(), (rect.top || 0) - (parentRect?.top || 0)) /
services.uiService.get('zoom');
services?.editorService?.paste({ left: initialLeft, top: initialTop }); services?.editorService?.paste({ left: initialLeft, top: initialTop });
} else { } else {
services?.editorService?.paste(); services?.editorService?.paste();

View File

@ -21,7 +21,7 @@ import serialize from 'serialize-javascript';
import type { Id, MApp, MContainer, MNode, MPage, MPageFragment } from '@tmagic/schema'; import type { Id, MApp, MContainer, MNode, MPage, MPageFragment } from '@tmagic/schema';
import { NodeType } from '@tmagic/schema'; import { NodeType } from '@tmagic/schema';
import type StageCore from '@tmagic/stage'; import type StageCore from '@tmagic/stage';
import { getNodePath, isNumber, isPage, isPageFragment, isPop } from '@tmagic/utils'; import { calcValueByFontsize, getNodePath, isNumber, isPage, isPageFragment, isPop } from '@tmagic/utils';
import { Layout } from '@editor/type'; import { Layout } from '@editor/type';
export const COPY_STORAGE_KEY = '$MagicEditorCopyData'; export const COPY_STORAGE_KEY = '$MagicEditorCopyData';
@ -106,13 +106,16 @@ const getMiddleTop = (node: MNode, parentNode: MNode, stage: StageCore | null) =
} }
const { height: parentHeight } = parentNode.style; const { height: parentHeight } = parentNode.style;
// wrapperHeight 是未 calcValue的高度, 所以要将其calcValueByFontsize一下, 否则在pad or pc端计算的结果有误
const { scrollTop = 0, wrapperHeight } = stage.mask;
const wrapperHeightDeal = calcValueByFontsize(stage.renderer.getDocument()!, wrapperHeight);
const scrollTopDeal = calcValueByFontsize(stage.renderer.getDocument()!, scrollTop);
if (isPage(parentNode)) { if (isPage(parentNode)) {
const { scrollTop = 0, wrapperHeight } = stage.mask; return (wrapperHeightDeal - height) / 2 + scrollTopDeal;
return (wrapperHeight - height) / 2 + scrollTop;
} }
return (parentHeight - height) / 2; // 如果容器的元素高度大于当前视口高度的2倍, 添加的元素居中位置也会看不见, 所以要取最小值计算
return (Math.min(parentHeight, wrapperHeightDeal) - height) / 2;
}; };
export const getInitPositionStyle = (style: Record<string, any> = {}, layout: Layout) => { export const getInitPositionStyle = (style: Record<string, any> = {}, layout: Layout) => {
@ -238,8 +241,12 @@ export const fixNodeLeft = (config: MNode, parent: MContainer, doc?: Document) =
const parentEl = doc.getElementById(`${parent.id}`); const parentEl = doc.getElementById(`${parent.id}`);
const left = Number(config.style?.left) || 0; const left = Number(config.style?.left) || 0;
if (el && parentEl && el.offsetWidth + left > parentEl.offsetWidth) { if (el && parentEl) {
return parentEl.offsetWidth - el.offsetWidth; const calcParentOffsetWidth = calcValueByFontsize(doc, parentEl.offsetWidth);
const calcElOffsetWidth = calcValueByFontsize(doc, el.offsetWidth);
if (calcElOffsetWidth + left > calcParentOffsetWidth) {
return calcParentOffsetWidth - calcElOffsetWidth;
}
} }
return config.style.left; return config.style.left;

View File

@ -32,10 +32,12 @@ import type {
} from 'moveable'; } from 'moveable';
import MoveableHelper from 'moveable-helper'; import MoveableHelper from 'moveable-helper';
import { calcValueByFontsize } from '@tmagic/utils';
import { DRAG_EL_ID_PREFIX, GHOST_EL_ID_PREFIX, Mode, ZIndex } from './const'; import { DRAG_EL_ID_PREFIX, GHOST_EL_ID_PREFIX, Mode, ZIndex } from './const';
import TargetShadow from './TargetShadow'; import TargetShadow from './TargetShadow';
import type { DragResizeHelperConfig, Rect, TargetElement } from './types'; import type { DragResizeHelperConfig, Rect, TargetElement } from './types';
import { calcValueByFontsize, getAbsolutePosition, getBorderWidth, getMarginValue, getOffset } from './util'; import { getAbsolutePosition, getBorderWidth, getMarginValue, getOffset } from './util';
/** /**
* /moveable会抛出各种状态事件DragResizeHelper负责响应这些事件target和拖拽节点targetShadow进行修改 * /moveable会抛出各种状态事件DragResizeHelper负责响应这些事件target和拖拽节点targetShadow进行修改

View File

@ -161,17 +161,6 @@ export const addSelectedClassName = (el: Element, doc: Document) => {
}); });
}; };
export const calcValueByFontsize = (doc: Document, value: number) => {
const { fontSize } = doc.documentElement.style;
if (fontSize) {
const times = globalThis.parseFloat(fontSize) / 100;
return Number((value / times).toFixed(2));
}
return value;
};
/** /**
* *
* @param {number} deltaTop * @param {number} deltaTop

View File

@ -110,3 +110,15 @@ export const createDiv = ({ className, cssText }: { className: string; cssText:
}; };
export const getDocument = () => globalThis.document; export const getDocument = () => globalThis.document;
export const calcValueByFontsize = (doc: Document | undefined, value: number) => {
if (!doc) return value;
const { fontSize } = doc.documentElement.style;
if (fontSize) {
const times = globalThis.parseFloat(fontSize) / 100;
return Number((value / times).toFixed(2));
}
return value;
};