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 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(V_GUIDE_LINE_STORAGE_KEY)),
]);
@ -131,5 +131,9 @@ export const useStage = (stageOptions: StageOptions) => {
}
});
onBeforeUnmount(() => {
stage.destroy();
});
return stage;
};

View File

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

View File

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

View File

@ -61,7 +61,7 @@ const unWatch = watch(
nextTick(() => unWatch());
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[];
buttonVisible.value = ids.length > 3;

View File

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

View File

@ -34,7 +34,7 @@ const style = computed(() => ({
watch(stage, (stage) => {
if (stage) {
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);
});
} else {
@ -53,8 +53,8 @@ watch(stageOverlay, (stageOverlay) => {
const { mask, renderer } = subStage;
const { contentWindow } = renderer;
mask.showRule(false);
const { contentWindow } = renderer!;
mask?.showRule(false);
services?.stageOverlayService.updateOverlay();

View File

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

View File

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

View File

@ -109,12 +109,16 @@ const getMiddleTop = (node: MNode, parentNode: MNode, stage: StageCore | null) =
}
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)) {
return (wrapperHeightDeal - height) / 2 + scrollTopDeal;
let wrapperHeightDeal = parentHeight;
if (stage.mask && stage.renderer) {
// wrapperHeight 是未 calcValue的高度, 所以要将其calcValueByFontsize一下, 否则在pad or pc端计算的结果有误
const { scrollTop = 0, wrapperHeight } = stage.mask;
wrapperHeightDeal = calcValueByFontsize(stage.renderer.getDocument()!, wrapperHeight);
const scrollTopDeal = calcValueByFontsize(stage.renderer.getDocument()!, scrollTop);
if (isPage(parentNode)) {
return (wrapperHeightDeal - height) / 2 + scrollTopDeal;
}
}
// 如果容器的元素高度大于当前视口高度的2倍, 添加的元素居中位置也会看不见, 所以要取最小值计算
@ -263,7 +267,7 @@ export const fixNodePosition = (config: MNode, parent: MContainer, stage: StageC
return {
...(config.style || {}),
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
*/
export default class ActionManager extends EventEmitter {
private dr: StageDragResize;
private multiDr?: StageMultiDragResize;
private highlightLayer: StageHighlight;
private dr: StageDragResize | null = null;
private multiDr: StageMultiDragResize | null = null;
private highlightLayer: StageHighlight | null = null;
/** 单选、多选、高亮的容器蒙层的content */
private container: HTMLElement;
/** 当前选中的节点 */
@ -143,7 +143,7 @@ export default class ActionManager extends EventEmitter {
this.disabledMultiSelect = true;
if (this.multiDr) {
this.multiDr.destroy();
this.multiDr = undefined;
this.multiDr = null;
}
}
@ -161,7 +161,7 @@ export default class ActionManager extends EventEmitter {
* @param guidelines 线
*/
public setGuidelines(type: GuidesType, guidelines: number[]): void {
this.dr.setGuidelines(type, guidelines);
this.dr?.setGuidelines(type, guidelines);
this.multiDr?.setGuidelines(type, guidelines);
}
@ -169,7 +169,7 @@ export default class ActionManager extends EventEmitter {
* 线
*/
public clearGuides(): void {
this.dr.clearGuides();
this.dr?.clearGuides();
this.multiDr?.clearGuides();
}
@ -178,7 +178,7 @@ export default class ActionManager extends EventEmitter {
* @param el
*/
public updateMoveable(el?: HTMLElement): void {
this.dr.updateMoveable(el);
this.dr?.updateMoveable(el);
// 多选时不可配置元素因此不存在多选元素变更不需要传el
this.multiDr?.updateMoveable();
}
@ -204,7 +204,7 @@ export default class ActionManager extends EventEmitter {
}
public getMoveableOption<K extends keyof MoveableOptions>(key: K): MoveableOptions[K] | undefined {
if (this.dr.getTarget()) {
if (this.dr?.getTarget()) {
return this.dr.getOption(key);
}
if (this.multiDr?.targetList.length) {
@ -271,7 +271,7 @@ export default class ActionManager extends EventEmitter {
public select(el: HTMLElement | null, event?: MouseEvent): void {
this.setSelectedEl(el);
this.clearSelectStatus(SelectStatus.MULTI_SELECT);
this.dr.select(el, event);
this.dr?.select(el, event);
}
public multiSelect(ids: Id[]): void {
@ -310,14 +310,14 @@ export default class ActionManager extends EventEmitter {
}
if (el === this.highlightedEl || !el) return;
this.highlightLayer.highlight(el);
this.highlightLayer?.highlight(el);
this.highlightedEl = el;
this.emit('highlight', el);
}
public clearHighlight(): void {
this.setHighlightEl(undefined);
this.highlightLayer.clearHighlight();
this.highlightLayer?.clearHighlight();
}
/**
@ -329,7 +329,7 @@ export default class ActionManager extends EventEmitter {
this.multiDr?.clearSelectStatus();
this.selectedElList = [];
} else {
this.dr.clearSelectStatus();
this.dr?.clearSelectStatus();
}
}
@ -373,7 +373,7 @@ export default class ActionManager extends EventEmitter {
}
public getDragStatus() {
return this.dr.getDragStatus();
return this.dr?.getDragStatus();
}
public destroy(): void {
@ -382,9 +382,16 @@ export default class ActionManager extends EventEmitter {
this.container.removeEventListener('mouseleave', this.mouseLeaveHandler);
this.container.removeEventListener('wheel', this.mouseWheelHandler);
this.container.removeEventListener('dblclick', this.dblclickHandler);
this.dr.destroy();
this.selectedEl = null;
this.selectedElList = [];
this.dr?.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]>(
@ -431,7 +438,7 @@ export default class ActionManager extends EventEmitter {
this.emit('select-parent');
})
.on(AbleActionEventType.REMOVE, () => {
const drTarget = this.dr.getTarget();
const drTarget = this.dr?.getTarget();
if (!drTarget) return;
const data: RemoveEventData = {
data: [{ el: drTarget }],

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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

View File

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