refactor(stage,editor): stage销毁后释放内部变量

This commit is contained in:
roymondchen 2024-08-22 16:45:06 +08:00 committed by roymondchen
parent 6754c3a8a5
commit c82e3257c1
19 changed files with 171 additions and 135 deletions

View File

@ -1,4 +1,4 @@
import { computed, watch } from 'vue'; import { computed, onBeforeUnmount, watch } from 'vue';
import type { MNode } from '@tmagic/schema'; import type { MNode } from '@tmagic/schema';
import StageCore, { GuidesType, RemoveEventData, SortEventData, UpdateEventData } from '@tmagic/stage'; import StageCore, { GuidesType, RemoveEventData, SortEventData, UpdateEventData } from '@tmagic/stage';
@ -62,7 +62,7 @@ export const useStage = (stageOptions: StageOptions) => {
}, },
); );
stage.mask.setGuides([ stage.mask?.setGuides([
getGuideLineFromCache(getGuideLineKey(H_GUIDE_LINE_STORAGE_KEY)), getGuideLineFromCache(getGuideLineKey(H_GUIDE_LINE_STORAGE_KEY)),
getGuideLineFromCache(getGuideLineKey(V_GUIDE_LINE_STORAGE_KEY)), getGuideLineFromCache(getGuideLineKey(V_GUIDE_LINE_STORAGE_KEY)),
]); ]);
@ -131,5 +131,9 @@ export const useStage = (stageOptions: StageOptions) => {
} }
}); });
onBeforeUnmount(() => {
stage.destroy();
});
return stage; return stage;
}; };

View File

@ -242,7 +242,7 @@ export const initServiceEvents = (
const getApp = () => { const getApp = () => {
const stage = editorService.get('stage'); const stage = editorService.get('stage');
return stage?.renderer.runtime?.getApp?.(); return stage?.renderer?.runtime?.getApp?.();
}; };
const updateDataSourceSchema = () => { const updateDataSourceSchema = () => {

View File

@ -111,7 +111,7 @@ const dragendHandler = () => {
globalThis.clearTimeout(timeout); globalThis.clearTimeout(timeout);
timeout = undefined; timeout = undefined;
} }
const doc = stage.value?.renderer.getDocument(); const doc = stage.value?.renderer?.getDocument();
if (doc && stageOptions?.containerHighlightClassName) { if (doc && stageOptions?.containerHighlightClassName) {
removeClassNameByClassName(doc, stageOptions.containerHighlightClassName); removeClassNameByClassName(doc, stageOptions.containerHighlightClassName);
} }

View File

@ -61,7 +61,7 @@ const unWatch = watch(
nextTick(() => unWatch()); nextTick(() => unWatch());
stage.on('select', (el: HTMLElement, event: MouseEvent) => { stage.on('select', (el: HTMLElement, event: MouseEvent) => {
const els = stage.renderer.getElementsFromPoint(event) || []; const els = stage.renderer?.getElementsFromPoint(event) || [];
const ids = els.map((el) => getIdFromEl()(el)).filter((id) => Boolean(id)) as string[]; const ids = els.map((el) => getIdFromEl()(el)).filter((id) => Boolean(id)) as string[];
buttonVisible.value = ids.length > 3; buttonVisible.value = ids.length > 3;

View File

@ -196,7 +196,7 @@ const dropHandler = async (e: DragEvent) => {
e.preventDefault(); e.preventDefault();
const doc = stage?.renderer.contentWindow?.document; const doc = stage?.renderer?.contentWindow?.document;
const parentEl: HTMLElement | null | undefined = doc?.querySelector(`.${stageOptions?.containerHighlightClassName}`); const parentEl: HTMLElement | null | undefined = doc?.querySelector(`.${stageOptions?.containerHighlightClassName}`);
let parent: MContainer | undefined | null = page.value; let parent: MContainer | undefined | null = page.value;
@ -209,7 +209,7 @@ const dropHandler = async (e: DragEvent) => {
const layout = await services?.editorService.getLayout(parent); const layout = await services?.editorService.getLayout(parent);
const containerRect = stageContainer.value.getBoundingClientRect(); const containerRect = stageContainer.value.getBoundingClientRect();
const { scrollTop, scrollLeft } = stage.mask; const { scrollTop, scrollLeft } = stage.mask!;
const { style = {} } = config.data; const { style = {} } = config.data;
let top = 0; let top = 0;

View File

@ -34,7 +34,7 @@ const style = computed(() => ({
watch(stage, (stage) => { watch(stage, (stage) => {
if (stage) { if (stage) {
stage.on('dblclick', async (event: MouseEvent) => { stage.on('dblclick', async (event: MouseEvent) => {
const el = await stage.actionManager.getElementFromPoint(event); const el = (await stage.actionManager?.getElementFromPoint(event)) || null;
services?.stageOverlayService.openOverlay(el); services?.stageOverlayService.openOverlay(el);
}); });
} else { } else {
@ -53,8 +53,8 @@ watch(stageOverlay, (stageOverlay) => {
const { mask, renderer } = subStage; const { mask, renderer } = subStage;
const { contentWindow } = renderer; const { contentWindow } = renderer!;
mask.showRule(false); mask?.showRule(false);
services?.stageOverlayService.updateOverlay(); services?.stageOverlayService.updateOverlay();

View File

@ -267,7 +267,7 @@ class Editor extends BaseService {
if (node?.id) { if (node?.id) {
this.get('stage') this.get('stage')
?.renderer.runtime?.getApp?.() ?.renderer?.runtime?.getApp?.()
?.page?.emit( ?.page?.emit(
'editor:select', 'editor:select',
{ {
@ -737,7 +737,7 @@ class Editor extends BaseService {
public async doPaste(config: MNode[], position: PastePosition = {}): Promise<MNode[]> { public async doPaste(config: MNode[], position: PastePosition = {}): Promise<MNode[]> {
propsService.clearRelateId(); propsService.clearRelateId();
const doc = this.get('stage')?.renderer.contentWindow?.document; const doc = this.get('stage')?.renderer?.contentWindow?.document;
const pasteConfigs = beforePaste(position, cloneDeep(config), doc); const pasteConfigs = beforePaste(position, cloneDeep(config), doc);
return pasteConfigs; return pasteConfigs;
} }
@ -756,7 +756,7 @@ class Editor extends BaseService {
if (!node.style) return config; if (!node.style) return config;
const stage = this.get('stage'); const stage = this.get('stage');
const doc = stage?.renderer.contentWindow?.document; const doc = stage?.renderer?.contentWindow?.document;
if (doc) { if (doc) {
const el = getElById()(doc, node.id); const el = getElById()(doc, node.id);

View File

@ -64,6 +64,12 @@ class StageOverlay extends BaseService {
const subStage = this.get('stage'); const subStage = this.get('stage');
const wrapDiv = this.get('wrapDiv'); const wrapDiv = this.get('wrapDiv');
subStage?.destroy(); subStage?.destroy();
for (let i = 0, l = wrapDiv.children.length; i < l; i++) {
const child = wrapDiv.children[i];
child.remove();
}
wrapDiv.remove(); wrapDiv.remove();
this.set('stage', null); this.set('stage', null);
@ -97,7 +103,7 @@ class StageOverlay extends BaseService {
render: async (stage: StageCore) => { render: async (stage: StageCore) => {
this.copyDocumentElement(); this.copyDocumentElement();
const rootEls = stage.renderer.getDocument()?.body.children; const rootEls = stage.renderer?.getDocument()?.body.children;
if (rootEls) { if (rootEls) {
Array.from(rootEls).forEach((element) => { Array.from(rootEls).forEach((element) => {
if (['SCRIPT', 'STYLE'].includes(element.tagName)) { if (['SCRIPT', 'STYLE'].includes(element.tagName)) {
@ -135,8 +141,8 @@ class StageOverlay extends BaseService {
const subStage = this.get('stage'); const subStage = this.get('stage');
const stage = editorService.get('stage'); const stage = editorService.get('stage');
const doc = subStage?.renderer.getDocument(); const doc = subStage?.renderer?.getDocument();
const documentElement = stage?.renderer.getDocument()?.documentElement; const documentElement = stage?.renderer?.getDocument()?.documentElement;
if (doc && documentElement) { if (doc && documentElement) {
doc.replaceChild(documentElement.cloneNode(true), doc.documentElement); doc.replaceChild(documentElement.cloneNode(true), doc.documentElement);
@ -160,13 +166,15 @@ class StageOverlay extends BaseService {
background-color: #fff; background-color: #fff;
`; `;
Array.from(wrapDiv.children).forEach((element) => { for (let i = 0, l = wrapDiv.children.length; i < l; i++) {
element.remove(); const child = wrapDiv.children[i];
}); child.remove();
}
wrapDiv.appendChild(contentEl); wrapDiv.appendChild(contentEl);
setTimeout(() => { setTimeout(() => {
subStage?.renderer.contentWindow?.magic.onPageElUpdate(wrapDiv); subStage?.renderer?.contentWindow?.magic.onPageElUpdate(wrapDiv);
}); });
if (await stageOptions?.canSelect?.(contentEl)) { if (await stageOptions?.canSelect?.(contentEl)) {

View File

@ -47,10 +47,10 @@ export const usePasteMenu = (menu?: Ref<InstanceType<typeof ContentMenu> | undef
const rect = menu.value.$el.getBoundingClientRect(); const rect = menu.value.$el.getBoundingClientRect();
const parentRect = stage?.container?.getBoundingClientRect(); const parentRect = stage?.container?.getBoundingClientRect();
const initialLeft = const initialLeft =
calcValueByFontsize(stage?.renderer.getDocument(), (rect.left || 0) - (parentRect?.left || 0)) / calcValueByFontsize(stage?.renderer?.getDocument(), (rect.left || 0) - (parentRect?.left || 0)) /
services.uiService.get('zoom'); services.uiService.get('zoom');
const initialTop = const initialTop =
calcValueByFontsize(stage?.renderer.getDocument(), (rect.top || 0) - (parentRect?.top || 0)) / calcValueByFontsize(stage?.renderer?.getDocument(), (rect.top || 0) - (parentRect?.top || 0)) /
services.uiService.get('zoom'); services.uiService.get('zoom');
services?.editorService?.paste({ left: initialLeft, top: initialTop }); services?.editorService?.paste({ left: initialLeft, top: initialTop });
} else { } else {

View File

@ -109,12 +109,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; let wrapperHeightDeal = parentHeight;
const wrapperHeightDeal = calcValueByFontsize(stage.renderer.getDocument()!, wrapperHeight); if (stage.mask && stage.renderer) {
const scrollTopDeal = calcValueByFontsize(stage.renderer.getDocument()!, scrollTop); // wrapperHeight 是未 calcValue的高度, 所以要将其calcValueByFontsize一下, 否则在pad or pc端计算的结果有误
if (isPage(parentNode)) { const { scrollTop = 0, wrapperHeight } = stage.mask;
return (wrapperHeightDeal - height) / 2 + scrollTopDeal; wrapperHeightDeal = calcValueByFontsize(stage.renderer.getDocument()!, wrapperHeight);
const scrollTopDeal = calcValueByFontsize(stage.renderer.getDocument()!, scrollTop);
if (isPage(parentNode)) {
return (wrapperHeightDeal - height) / 2 + scrollTopDeal;
}
} }
// 如果容器的元素高度大于当前视口高度的2倍, 添加的元素居中位置也会看不见, 所以要取最小值计算 // 如果容器的元素高度大于当前视口高度的2倍, 添加的元素居中位置也会看不见, 所以要取最小值计算
@ -263,7 +267,7 @@ export const fixNodePosition = (config: MNode, parent: MContainer, stage: StageC
return { return {
...(config.style || {}), ...(config.style || {}),
top: getMiddleTop(config, parent, stage), top: getMiddleTop(config, parent, stage),
left: fixNodeLeft(config, parent, stage?.renderer.contentWindow?.document), left: fixNodeLeft(config, parent, stage?.renderer?.contentWindow?.document),
}; };
}; };

View File

@ -65,9 +65,9 @@ const defaultContainerHighlightDuration = 800;
* @extends EventEmitter * @extends EventEmitter
*/ */
export default class ActionManager extends EventEmitter { export default class ActionManager extends EventEmitter {
private dr: StageDragResize; private dr: StageDragResize | null = null;
private multiDr?: StageMultiDragResize; private multiDr: StageMultiDragResize | null = null;
private highlightLayer: StageHighlight; private highlightLayer: StageHighlight | null = null;
/** 单选、多选、高亮的容器蒙层的content */ /** 单选、多选、高亮的容器蒙层的content */
private container: HTMLElement; private container: HTMLElement;
/** 当前选中的节点 */ /** 当前选中的节点 */
@ -143,7 +143,7 @@ export default class ActionManager extends EventEmitter {
this.disabledMultiSelect = true; this.disabledMultiSelect = true;
if (this.multiDr) { if (this.multiDr) {
this.multiDr.destroy(); this.multiDr.destroy();
this.multiDr = undefined; this.multiDr = null;
} }
} }
@ -161,7 +161,7 @@ export default class ActionManager extends EventEmitter {
* @param guidelines 线 * @param guidelines 线
*/ */
public setGuidelines(type: GuidesType, guidelines: number[]): void { public setGuidelines(type: GuidesType, guidelines: number[]): void {
this.dr.setGuidelines(type, guidelines); this.dr?.setGuidelines(type, guidelines);
this.multiDr?.setGuidelines(type, guidelines); this.multiDr?.setGuidelines(type, guidelines);
} }
@ -169,7 +169,7 @@ export default class ActionManager extends EventEmitter {
* 线 * 线
*/ */
public clearGuides(): void { public clearGuides(): void {
this.dr.clearGuides(); this.dr?.clearGuides();
this.multiDr?.clearGuides(); this.multiDr?.clearGuides();
} }
@ -178,7 +178,7 @@ export default class ActionManager extends EventEmitter {
* @param el * @param el
*/ */
public updateMoveable(el?: HTMLElement): void { public updateMoveable(el?: HTMLElement): void {
this.dr.updateMoveable(el); this.dr?.updateMoveable(el);
// 多选时不可配置元素因此不存在多选元素变更不需要传el // 多选时不可配置元素因此不存在多选元素变更不需要传el
this.multiDr?.updateMoveable(); this.multiDr?.updateMoveable();
} }
@ -204,7 +204,7 @@ export default class ActionManager extends EventEmitter {
} }
public getMoveableOption<K extends keyof MoveableOptions>(key: K): MoveableOptions[K] | undefined { public getMoveableOption<K extends keyof MoveableOptions>(key: K): MoveableOptions[K] | undefined {
if (this.dr.getTarget()) { if (this.dr?.getTarget()) {
return this.dr.getOption(key); return this.dr.getOption(key);
} }
if (this.multiDr?.targetList.length) { if (this.multiDr?.targetList.length) {
@ -271,7 +271,7 @@ export default class ActionManager extends EventEmitter {
public select(el: HTMLElement | null, event?: MouseEvent): void { public select(el: HTMLElement | null, event?: MouseEvent): void {
this.setSelectedEl(el); this.setSelectedEl(el);
this.clearSelectStatus(SelectStatus.MULTI_SELECT); this.clearSelectStatus(SelectStatus.MULTI_SELECT);
this.dr.select(el, event); this.dr?.select(el, event);
} }
public multiSelect(ids: Id[]): void { public multiSelect(ids: Id[]): void {
@ -310,14 +310,14 @@ export default class ActionManager extends EventEmitter {
} }
if (el === this.highlightedEl || !el) return; if (el === this.highlightedEl || !el) return;
this.highlightLayer.highlight(el); this.highlightLayer?.highlight(el);
this.highlightedEl = el; this.highlightedEl = el;
this.emit('highlight', el); this.emit('highlight', el);
} }
public clearHighlight(): void { public clearHighlight(): void {
this.setHighlightEl(undefined); this.setHighlightEl(undefined);
this.highlightLayer.clearHighlight(); this.highlightLayer?.clearHighlight();
} }
/** /**
@ -329,7 +329,7 @@ export default class ActionManager extends EventEmitter {
this.multiDr?.clearSelectStatus(); this.multiDr?.clearSelectStatus();
this.selectedElList = []; this.selectedElList = [];
} else { } else {
this.dr.clearSelectStatus(); this.dr?.clearSelectStatus();
} }
} }
@ -373,7 +373,7 @@ export default class ActionManager extends EventEmitter {
} }
public getDragStatus() { public getDragStatus() {
return this.dr.getDragStatus(); return this.dr?.getDragStatus();
} }
public destroy(): void { public destroy(): void {
@ -382,9 +382,16 @@ export default class ActionManager extends EventEmitter {
this.container.removeEventListener('mouseleave', this.mouseLeaveHandler); this.container.removeEventListener('mouseleave', this.mouseLeaveHandler);
this.container.removeEventListener('wheel', this.mouseWheelHandler); this.container.removeEventListener('wheel', this.mouseWheelHandler);
this.container.removeEventListener('dblclick', this.dblclickHandler); this.container.removeEventListener('dblclick', this.dblclickHandler);
this.dr.destroy(); this.selectedEl = null;
this.selectedElList = [];
this.dr?.destroy();
this.multiDr?.destroy(); this.multiDr?.destroy();
this.highlightLayer.destroy(); this.highlightLayer?.destroy();
this.dr = null;
this.multiDr = null;
this.highlightLayer = null;
} }
public on<Name extends keyof ActionManagerEvents, Param extends ActionManagerEvents[Name]>( public on<Name extends keyof ActionManagerEvents, Param extends ActionManagerEvents[Name]>(
@ -431,7 +438,7 @@ export default class ActionManager extends EventEmitter {
this.emit('select-parent'); this.emit('select-parent');
}) })
.on(AbleActionEventType.REMOVE, () => { .on(AbleActionEventType.REMOVE, () => {
const drTarget = this.dr.getTarget(); const drTarget = this.dr?.getTarget();
if (!drTarget) return; if (!drTarget) return;
const data: RemoveEventData = { const data: RemoveEventData = {
data: [{ el: drTarget }], data: [{ el: drTarget }],

View File

@ -50,7 +50,7 @@ export default class DragResizeHelper {
/** 目标节点在蒙层上的占位节点,用于跟鼠标交互,避免鼠标事件直接作用到目标节点 */ /** 目标节点在蒙层上的占位节点,用于跟鼠标交互,避免鼠标事件直接作用到目标节点 */
private targetShadow: TargetShadow; private targetShadow: TargetShadow;
/** 要操作的原始目标节点 */ /** 要操作的原始目标节点 */
private target!: HTMLElement; private target: HTMLElement | null = null;
/** 多选:目标节点组 */ /** 多选:目标节点组 */
private targetList: HTMLElement[] = []; private targetList: HTMLElement[] = [];
/** targetShadowdom /** targetShadowdom
@ -84,6 +84,8 @@ export default class DragResizeHelper {
} }
public destroy(): void { public destroy(): void {
this.target = null;
this.targetList = [];
this.targetShadow.destroy(); this.targetShadow.destroy();
this.destroyGhostEl(); this.destroyGhostEl();
this.moveableHelper.clear(); this.moveableHelper.clear();
@ -114,8 +116,8 @@ export default class DragResizeHelper {
public onResizeStart(e: OnResizeStart): void { public onResizeStart(e: OnResizeStart): void {
this.moveableHelper.onResizeStart(e); this.moveableHelper.onResizeStart(e);
this.frameSnapShot.top = this.target.offsetTop; this.frameSnapShot.top = this.target!.offsetTop;
this.frameSnapShot.left = this.target.offsetLeft; this.frameSnapShot.left = this.target!.offsetLeft;
} }
public onResize(e: OnResize): void { public onResize(e: OnResize): void {
@ -123,33 +125,33 @@ export default class DragResizeHelper {
const { beforeTranslate } = drag; const { beforeTranslate } = drag;
// 流式布局 // 流式布局
if (this.mode === Mode.SORTABLE) { if (this.mode === Mode.SORTABLE) {
this.target.style.top = '0px'; this.target!.style.top = '0px';
if (this.targetShadow.el) { if (this.targetShadow.el) {
this.targetShadow.el.style.width = `${width}px`; this.targetShadow.el.style.width = `${width}px`;
this.targetShadow.el.style.height = `${height}px`; this.targetShadow.el.style.height = `${height}px`;
} }
} else { } else {
this.moveableHelper.onResize(e); this.moveableHelper.onResize(e);
const { marginLeft, marginTop } = getMarginValue(this.target); const { marginLeft, marginTop } = getMarginValue(this.target!);
this.target.style.left = `${this.frameSnapShot.left + beforeTranslate[0] - marginLeft}px`; this.target!.style.left = `${this.frameSnapShot.left + beforeTranslate[0] - marginLeft}px`;
this.target.style.top = `${this.frameSnapShot.top + beforeTranslate[1] - marginTop}px`; this.target!.style.top = `${this.frameSnapShot.top + beforeTranslate[1] - marginTop}px`;
} }
const { borderLeftWidth, borderRightWidth, borderTopWidth, borderBottomWidth } = getBorderWidth(this.target); const { borderLeftWidth, borderRightWidth, borderTopWidth, borderBottomWidth } = getBorderWidth(this.target!);
this.target.style.width = `${width + borderLeftWidth + borderRightWidth}px`; this.target!.style.width = `${width + borderLeftWidth + borderRightWidth}px`;
this.target.style.height = `${height + borderTopWidth + borderBottomWidth}px`; this.target!.style.height = `${height + borderTopWidth + borderBottomWidth}px`;
} }
public onDragStart(e: OnDragStart): void { public onDragStart(e: OnDragStart): void {
this.moveableHelper.onDragStart(e); this.moveableHelper.onDragStart(e);
if (this.mode === Mode.SORTABLE) { if (this.mode === Mode.SORTABLE) {
this.ghostEl = this.generateGhostEl(this.target); this.ghostEl = this.generateGhostEl(this.target!);
} }
this.frameSnapShot.top = this.target.offsetTop; this.frameSnapShot.top = this.target!.offsetTop;
this.frameSnapShot.left = this.target.offsetLeft; this.frameSnapShot.left = this.target!.offsetLeft;
} }
public onDrag(e: OnDrag): void { public onDrag(e: OnDrag): void {
@ -161,10 +163,10 @@ export default class DragResizeHelper {
this.moveableHelper.onDrag(e); this.moveableHelper.onDrag(e);
const { marginLeft, marginTop } = getMarginValue(this.target); const { marginLeft, marginTop } = getMarginValue(this.target!);
this.target.style.left = `${this.frameSnapShot.left + e.beforeTranslate[0] - marginLeft}px`; this.target!.style.left = `${this.frameSnapShot.left + e.beforeTranslate[0] - marginLeft}px`;
this.target.style.top = `${this.frameSnapShot.top + e.beforeTranslate[1] - marginTop}px`; this.target!.style.top = `${this.frameSnapShot.top + e.beforeTranslate[1] - marginTop}px`;
} }
public onRotateStart(e: OnRotateStart): void { public onRotateStart(e: OnRotateStart): void {
@ -174,7 +176,7 @@ export default class DragResizeHelper {
public onRotate(e: OnRotate): void { public onRotate(e: OnRotate): void {
this.moveableHelper.onRotate(e); this.moveableHelper.onRotate(e);
const frame = this.moveableHelper.getFrame(e.target); const frame = this.moveableHelper.getFrame(e.target);
this.target.style.transform = frame?.toCSSObject().transform || ''; this.target!.style.transform = frame?.toCSSObject().transform || '';
} }
public onScaleStart(e: OnScaleStart): void { public onScaleStart(e: OnScaleStart): void {
@ -184,7 +186,7 @@ export default class DragResizeHelper {
public onScale(e: OnScale): void { public onScale(e: OnScale): void {
this.moveableHelper.onScale(e); this.moveableHelper.onScale(e);
const frame = this.moveableHelper.getFrame(e.target); const frame = this.moveableHelper.getFrame(e.target);
this.target.style.transform = frame?.toCSSObject().transform || ''; this.target!.style.transform = frame?.toCSSObject().transform || '';
} }
public getGhostEl(): HTMLElement | undefined { public getGhostEl(): HTMLElement | undefined {

View File

@ -47,9 +47,9 @@ import type {
*/ */
export default class StageCore extends EventEmitter { export default class StageCore extends EventEmitter {
public container?: HTMLDivElement; public container?: HTMLDivElement;
public renderer: StageRender; public renderer: StageRender | null = null;
public mask: StageMask; public mask: StageMask | null = null;
public actionManager: ActionManager; public actionManager: ActionManager | null = null;
private pageResizeObserver: ResizeObserver | null = null; private pageResizeObserver: ResizeObserver | null = null;
private autoScrollIntoView: boolean | undefined; private autoScrollIntoView: boolean | undefined;
@ -87,17 +87,18 @@ export default class StageCore extends EventEmitter {
* @param id id * @param id id
*/ */
public async select(id: Id, event?: MouseEvent): Promise<void> { public async select(id: Id, event?: MouseEvent): Promise<void> {
const el = this.renderer.getTargetElement(id); const el = this.renderer?.getTargetElement(id) || null;
if (el === this.actionManager.getSelectedEl()) return; if (el === this.actionManager?.getSelectedEl()) return;
await this.renderer.select([id]); await this.renderer?.select([id]);
el && this.mask.setLayout(el); if (el) {
this.mask?.setLayout(el);
this.actionManager.select(el, event); }
this.actionManager?.select(el, event);
if (el && (this.autoScrollIntoView || el.dataset.autoScrollIntoView)) { if (el && (this.autoScrollIntoView || el.dataset.autoScrollIntoView)) {
this.mask.observerIntersection(el); this.mask?.observerIntersection(el);
} }
} }
@ -106,20 +107,20 @@ export default class StageCore extends EventEmitter {
* @param ids id列表 * @param ids id列表
*/ */
public async multiSelect(ids: Id[]): Promise<void> { public async multiSelect(ids: Id[]): Promise<void> {
const els = ids.map((id) => this.renderer.getTargetElement(id)).filter((el) => Boolean(el)); const els = ids.map((id) => this.renderer?.getTargetElement(id)).filter((el) => Boolean(el));
if (els.length === 0) return; if (els.length === 0) return;
const lastEl = els[els.length - 1]; const lastEl = els[els.length - 1];
// 是否减少了组件选择 // 是否减少了组件选择
const isReduceSelect = els.length < this.actionManager.getSelectedElList().length; const isReduceSelect = els.length < this.actionManager!.getSelectedElList().length;
await this.renderer.select(ids); await this.renderer?.select(ids);
lastEl && this.mask.setLayout(lastEl); lastEl && this.mask?.setLayout(lastEl);
this.actionManager.multiSelect(ids); this.actionManager?.multiSelect(ids);
if (lastEl && (this.autoScrollIntoView || lastEl.dataset.autoScrollIntoView) && !isReduceSelect) { if (lastEl && (this.autoScrollIntoView || lastEl.dataset.autoScrollIntoView) && !isReduceSelect) {
this.mask.observerIntersection(lastEl); this.mask?.observerIntersection(lastEl);
} }
} }
@ -128,11 +129,11 @@ export default class StageCore extends EventEmitter {
* @param el * @param el
*/ */
public highlight(id: Id): void { public highlight(id: Id): void {
this.actionManager.highlight(id); this.actionManager?.highlight(id);
} }
public clearHighlight(): void { public clearHighlight(): void {
this.actionManager.clearHighlight(); this.actionManager?.clearHighlight();
} }
/** /**
@ -142,13 +143,13 @@ export default class StageCore extends EventEmitter {
public async update(data: UpdateData): Promise<void> { public async update(data: UpdateData): Promise<void> {
const { config } = data; const { config } = data;
await this.renderer.update(data); await this.renderer?.update(data);
// 通过setTimeout等画布中组件完成渲染更新 // 通过setTimeout等画布中组件完成渲染更新
setTimeout(() => { setTimeout(() => {
const el = this.renderer.getTargetElement(`${config.id}`); const el = this.renderer?.getTargetElement(`${config.id}`);
if (el && this.actionManager.isSelectedEl(el)) { if (el && this.actionManager?.isSelectedEl(el)) {
// 更新了组件的布局需要重新设置mask是否可以滚动 // 更新了组件的布局需要重新设置mask是否可以滚动
this.mask.setLayout(el); this.mask?.setLayout(el);
// 组件有更新需要set // 组件有更新需要set
this.actionManager.setSelectedEl(el); this.actionManager.setSelectedEl(el);
this.actionManager.updateMoveable(el); this.actionManager.updateMoveable(el);
@ -161,7 +162,7 @@ export default class StageCore extends EventEmitter {
* @param data * @param data
*/ */
public async add(data: UpdateData): Promise<void> { public async add(data: UpdateData): Promise<void> {
return await this.renderer.add(data); return await this.renderer?.add(data);
} }
/** /**
@ -169,11 +170,11 @@ export default class StageCore extends EventEmitter {
* @param data * @param data
*/ */
public async remove(data: RemoveData): Promise<void> { public async remove(data: RemoveData): Promise<void> {
return await this.renderer.remove(data); return await this.renderer?.remove(data);
} }
public setZoom(zoom: number = DEFAULT_ZOOM): void { public setZoom(zoom: number = DEFAULT_ZOOM): void {
this.renderer.setZoom(zoom); this.renderer?.setZoom(zoom);
} }
/** /**
@ -184,8 +185,8 @@ export default class StageCore extends EventEmitter {
this.container = el; this.container = el;
const { mask, renderer } = this; const { mask, renderer } = this;
await renderer.mount(el); await renderer?.mount(el);
mask.mount(el); mask?.mount(el);
this.emit('mounted'); this.emit('mounted');
} }
@ -194,8 +195,8 @@ export default class StageCore extends EventEmitter {
* 线 * 线
*/ */
public clearGuides() { public clearGuides() {
this.mask.clearGuides(); this.mask?.clearGuides();
this.actionManager.clearGuides(); this.actionManager?.clearGuides();
} }
/** /**
@ -216,23 +217,23 @@ export default class StageCore extends EventEmitter {
* @returns timeoutIdtimeout * @returns timeoutIdtimeout
*/ */
public delayedMarkContainer(event: MouseEvent, excludeElList: Element[] = []): NodeJS.Timeout | undefined { public delayedMarkContainer(event: MouseEvent, excludeElList: Element[] = []): NodeJS.Timeout | undefined {
return this.actionManager.delayedMarkContainer(event, excludeElList); return this.actionManager?.delayedMarkContainer(event, excludeElList);
} }
public getMoveableOption<K extends keyof MoveableOptions>(key: K): MoveableOptions[K] | undefined { public getMoveableOption<K extends keyof MoveableOptions>(key: K): MoveableOptions[K] | undefined {
return this.actionManager.getMoveableOption(key); return this.actionManager?.getMoveableOption(key);
} }
public getDragStatus() { public getDragStatus() {
return this.actionManager.getDragStatus(); return this.actionManager?.getDragStatus();
} }
public disableMultiSelect() { public disableMultiSelect() {
this.actionManager.disableMultiSelect(); this.actionManager?.disableMultiSelect();
} }
public enableMultiSelect() { public enableMultiSelect() {
this.actionManager.enableMultiSelect(); this.actionManager?.enableMultiSelect();
} }
/** /**
@ -241,14 +242,18 @@ export default class StageCore extends EventEmitter {
public destroy(): void { public destroy(): void {
const { mask, renderer, actionManager, pageResizeObserver } = this; const { mask, renderer, actionManager, pageResizeObserver } = this;
renderer.destroy(); renderer?.destroy();
mask.destroy(); mask?.destroy();
actionManager.destroy(); actionManager?.destroy();
pageResizeObserver?.disconnect(); pageResizeObserver?.disconnect();
this.removeAllListeners(); this.removeAllListeners();
this.container = undefined; this.container = undefined;
this.renderer = null;
this.mask = null;
this.actionManager = null;
this.pageResizeObserver = null;
} }
public on<Name extends keyof CoreEvents, Param extends CoreEvents[Name]>( public on<Name extends keyof CoreEvents, Param extends CoreEvents[Name]>(
@ -272,8 +277,8 @@ export default class StageCore extends EventEmitter {
if (typeof ResizeObserver !== 'undefined') { if (typeof ResizeObserver !== 'undefined') {
this.pageResizeObserver = new ResizeObserver((entries) => { this.pageResizeObserver = new ResizeObserver((entries) => {
this.mask.pageResize(entries); this.mask?.pageResize(entries);
this.actionManager.updateMoveable(); this.actionManager?.updateMoveable();
}); });
this.pageResizeObserver.observe(page); this.pageResizeObserver.observe(page);
@ -286,26 +291,26 @@ export default class StageCore extends EventEmitter {
containerHighlightDuration: config.containerHighlightDuration, containerHighlightDuration: config.containerHighlightDuration,
containerHighlightType: config.containerHighlightType, containerHighlightType: config.containerHighlightType,
moveableOptions: config.moveableOptions, moveableOptions: config.moveableOptions,
container: this.mask.content, container: this.mask!.content,
disabledDragStart: config.disabledDragStart, disabledDragStart: config.disabledDragStart,
disabledMultiSelect: config.disabledMultiSelect, disabledMultiSelect: config.disabledMultiSelect,
canSelect: config.canSelect, canSelect: config.canSelect,
isContainer: config.isContainer, isContainer: config.isContainer,
updateDragEl: config.updateDragEl, updateDragEl: config.updateDragEl,
getRootContainer: () => this.container, getRootContainer: () => this.container,
getRenderDocument: () => this.renderer.getDocument(), getRenderDocument: () => this.renderer!.getDocument(),
getTargetElement: (id: Id) => this.renderer.getTargetElement(id), getTargetElement: (id: Id) => this.renderer!.getTargetElement(id),
getElementsFromPoint: (point: Point) => this.renderer.getElementsFromPoint(point), getElementsFromPoint: (point: Point) => this.renderer!.getElementsFromPoint(point),
}; };
return actionManagerConfig; return actionManagerConfig;
} }
private initRenderEvent(): void { private initRenderEvent(): void {
this.renderer.on('runtime-ready', (runtime: Runtime) => { this.renderer?.on('runtime-ready', (runtime: Runtime) => {
this.emit('runtime-ready', runtime); this.emit('runtime-ready', runtime);
}); });
this.renderer.on('page-el-update', (el: HTMLElement) => { this.renderer?.on('page-el-update', (el: HTMLElement) => {
this.mask?.observe(el); this.mask?.observe(el);
this.observePageResize(el); this.observePageResize(el);
@ -314,8 +319,8 @@ export default class StageCore extends EventEmitter {
} }
private initMaskEvent(): void { private initMaskEvent(): void {
this.mask.on('change-guides', (data: GuidesEventData) => { this.mask?.on('change-guides', (data: GuidesEventData) => {
this.actionManager.setGuidelines(data.type, data.guides); this.actionManager?.setGuidelines(data.type, data.guides);
this.emit('change-guides', data); this.emit('change-guides', data);
}); });
} }
@ -336,7 +341,7 @@ export default class StageCore extends EventEmitter {
*/ */
private initActionManagerEvent(): void { private initActionManagerEvent(): void {
this.actionManager this.actionManager
.on('before-select', (el: HTMLElement, event?: MouseEvent) => { ?.on('before-select', (el: HTMLElement, event?: MouseEvent) => {
const id = getIdFromEl()(el); const id = getIdFromEl()(el);
id && this.select(id, event); id && this.select(id, event);
}) })
@ -359,7 +364,7 @@ export default class StageCore extends EventEmitter {
*/ */
private initDrEvent(): void { private initDrEvent(): void {
this.actionManager this.actionManager
.on('update', (data: UpdateEventData) => { ?.on('update', (data: UpdateEventData) => {
this.emit('update', data); this.emit('update', data);
}) })
.on('sort', (data: SortEventData) => { .on('sort', (data: SortEventData) => {
@ -379,11 +384,11 @@ export default class StageCore extends EventEmitter {
private initMulDrEvent(): void { private initMulDrEvent(): void {
this.actionManager this.actionManager
// 多选切换到单选 // 多选切换到单选
.on('change-to-select', (id: Id, e: MouseEvent) => { ?.on('change-to-select', (id: Id, e: MouseEvent) => {
this.select(id); this.select(id);
// 先保证画布内完成渲染,再通知外部更新 // 先保证画布内完成渲染,再通知外部更新
setTimeout(() => { setTimeout(() => {
const el = this.renderer.getTargetElement(id); const el = this.renderer?.getTargetElement(id);
el && this.emit('select', el, e); el && this.emit('select', el, e);
}); });
}) })
@ -396,7 +401,7 @@ export default class StageCore extends EventEmitter {
* Highlight类通过ActionManager抛出来的事件监听 * Highlight类通过ActionManager抛出来的事件监听
*/ */
private initHighlightEvent(): void { private initHighlightEvent(): void {
this.actionManager.on('highlight', (highlightEl: HTMLElement) => { this.actionManager?.on('highlight', (highlightEl: HTMLElement) => {
this.emit('highlight', highlightEl); this.emit('highlight', highlightEl);
}); });
} }
@ -406,7 +411,7 @@ export default class StageCore extends EventEmitter {
*/ */
private initMouseEvent(): void { private initMouseEvent(): void {
this.actionManager this.actionManager
.on('mousemove', (event: MouseEvent) => { ?.on('mousemove', (event: MouseEvent) => {
this.emit('mousemove', event); this.emit('mousemove', event);
}) })
.on('mouseleave', (event: MouseEvent) => { .on('mouseleave', (event: MouseEvent) => {

View File

@ -125,6 +125,7 @@ export default class StageDragResize extends MoveableOptionsManager {
* *
*/ */
public destroy(): void { public destroy(): void {
this.target = null;
this.moveable?.destroy(); this.moveable?.destroy();
this.dragResizeHelper.destroy(); this.dragResizeHelper.destroy();
this.dragStatus = StageDragStatus.END; this.dragStatus = StageDragStatus.END;

View File

@ -81,6 +81,7 @@ export default class StageHighlight extends EventEmitter {
* *
*/ */
public destroy(): void { public destroy(): void {
this.target = undefined;
this.moveable?.destroy(); this.moveable?.destroy();
this.targetShadow?.destroy(); this.targetShadow?.destroy();
this.moveable = undefined; this.moveable = undefined;

View File

@ -1,5 +1,5 @@
{ {
"version": "1.1.1", "version": "1.1.2",
"name": "@tmagic/tmagic-form-runtime", "name": "@tmagic/tmagic-form-runtime",
"type": "module", "type": "module",
"main": "dist/tmagic-form-runtime.umd.cjs", "main": "dist/tmagic-form-runtime.umd.cjs",

View File

@ -23,8 +23,8 @@ const { mForm, formConfig, config, values } = useFormConfig(props);
watch(formConfig, async () => { watch(formConfig, async () => {
setTimeout(() => { setTimeout(() => {
const page = props.stage.renderer.getDocument()?.querySelector<HTMLElement>('.m-form'); const page = props.stage.renderer?.getDocument()?.querySelector<HTMLElement>('.m-form');
page && props.stage.renderer.contentWindow?.magic.onPageElUpdate(page); page && props.stage.renderer?.contentWindow?.magic.onPageElUpdate(page);
}); });
}); });
</script> </script>

View File

@ -24,21 +24,25 @@ export const useRuntime = ({
fillConfig?: (config: FormConfig, mForm: any) => FormConfig; fillConfig?: (config: FormConfig, mForm: any) => FormConfig;
} = {}) => { } = {}) => {
const render = (stage: StageCore) => { const render = (stage: StageCore) => {
injectStyle(stage.renderer.getDocument()!, cssStyle); const doc = stage.renderer?.getDocument();
injectStyle(
stage.renderer.getDocument()!, if (doc) {
`html, injectStyle(doc, cssStyle);
body, injectStyle(
#app { doc,
width: 100%; `html,
height: 100%; body,
margin: 0; #app {
} width: 100%;
::-webkit-scrollbar { height: 100%;
width: 0; margin: 0;
} }
`, ::-webkit-scrollbar {
); width: 0;
}
`,
);
}
const el: HTMLDivElement = globalThis.document.createElement('div'); const el: HTMLDivElement = globalThis.document.createElement('div');
el.id = 'app'; el.id = 'app';

View File

@ -9,7 +9,7 @@ import { getElById, getNodePath, replaceChildNode } from '@tmagic/utils';
import { AppProps } from './types'; import { AppProps } from './types';
export const useFormConfig = (props: AppProps) => { export const useFormConfig = (props: AppProps) => {
const { contentWindow } = props.stage.renderer; const { contentWindow } = props.stage.renderer!;
const mForm = ref<InstanceType<typeof MForm>>(); const mForm = ref<InstanceType<typeof MForm>>();
const root = ref<MApp>(); const root = ref<MApp>();