From 3fb880d09b2b3bcf00b5058b1387aa7ecaaf86f3 Mon Sep 17 00:00:00 2001 From: oceanzhu Date: Thu, 24 Nov 2022 21:19:56 +0800 Subject: [PATCH] =?UTF-8?q?refactor(stage):=E9=87=8D=E6=9E=84=E9=AD=94?= =?UTF-8?q?=E6=96=B9=E7=BC=96=E8=BE=91=E5=99=A8stage=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Squash merge branch 'feature/oc_actionbox_879360293' into 'master' 对魔方编辑器核心代码stage进行重构,这部分代码主要是负责编辑器中间画布区域的处理,包括渲染一个所见即所得的画布,支持组件的增删改查和拖拽、高亮操作。 旧代码存在的问题以及解决方案: 1、过多暴露属性和循环引用,导致stageCore、stageDragResize、StageMultiDragResize、StageRender、StageMask、StageHighlight之间形成复杂的网状依赖,非常不可控。StageCore负责创建后面5个类的实例,并把这些实例作为自己的公共属性,同时core把自己的this传给这些实例,这些实例就会通过core传进来的this,通过core间接的访问其它实例的方法和属性,比如在stageDragResize中可能存在这样的一个访问:this.core.stageRender.contentWindow.document 解决方案: 1)、属性尽量设置为私有,对外暴露方法,不暴露属性; 2)、core避免向其它类传递this,改为传递接口,需要什么就传什么 2、事件传递较多,跳来跳去,定位问题较为困难 解决方案: 重新梳理各个类的职责,尽量在类中闭环,减少事件传递。 新增了actionManager类,core负责管理render、mask、actionManager三个类; actionManager负责管理单选、多选、高亮三个类,同时将mask中的事件监听,转移到actionManager监听,actionManager形成单选、多选、高亮行为后,直接调动单选、多选、高亮完成功能。 3、存在一些重复代码 主要是拖拽框的代码在单选、多选、高亮中各自维护,改为统一维护 4、多选不支持辅助线对齐 将单选中的moveableOption管理逻辑抽取出来成为单选和多选的父类,使多选支持辅助线对齐 本次改动取消了一些对外暴露的属性,moveableOption回调函数签名也有变化,详细情况如下: 删除stageCore公共属性: public selectedDom: HTMLElement | undefined; public selectedDomList: HTMLElement[] = []; public highlightedDom: Element | undefined; public dr: StageDragResize; public multiDr: StageMultiDragResize; public highlightLayer: StageHighlight; public config: StageCoreConfig; public zoom = DEFAULT_ZOOM; public containerHighlightClassName: string; public containerHighlightDuration: number; public containerHighlightType?: ContainerHighlightType; public isContainer: IsContainer; stageCore入参改动: 这两个参数原本定义: moveableOptions?: ((core?: StageCore) => MoveableOptions) | MoveableOptions; multiMoveableOptions?: ((core?: StageCore) => MoveableOptions) | MoveableOptions; 修改后定义: moveableOptions?: CustomizeMoveableOptions; multiMoveableOptions?: CustomizeMoveableOptions; CustomizeMoveableOptions = | ((config?: CustomizeMoveableOptionsCallbackConfig) => MoveableOptions) | MoveableOptions | undefined; export interface CustomizeMoveableOptionsCallbackConfig { targetElId?: string; } --- docs/src/api/editor/editor.md | 2 +- magic-admin/web/src/views/editor.vue | 6 +- packages/core/src/index.ts | 2 + packages/editor/src/Editor.vue | 17 +- packages/editor/src/type.ts | 11 +- packages/editor/src/utils/stage.ts | 4 +- packages/stage/README.md | 58 +- packages/stage/src/ActionManager.ts | 534 ++++++++++++++++++ packages/stage/src/MoveableOptionsManager.ts | 249 ++++++++ .../stage/src/MoveableSelectParentAble.ts | 8 +- packages/stage/src/Rule.ts | 8 +- packages/stage/src/StageCore.ts | 483 +++++++--------- packages/stage/src/StageDragResize.ts | 312 +++------- packages/stage/src/StageHighlight.ts | 33 +- packages/stage/src/StageMask.ts | 121 +--- packages/stage/src/StageMultiDragResize.ts | 198 +++---- packages/stage/src/StageRender.ts | 95 +++- packages/stage/src/TargetCalibrate.ts | 119 ---- packages/stage/src/TargetShadow.ts | 120 ++++ packages/stage/src/const.ts | 2 +- packages/stage/src/types.ts | 106 +++- packages/stage/src/util.ts | 29 +- packages/utils/src/dom.ts | 2 + playground/src/pages/Editor.vue | 6 +- 24 files changed, 1593 insertions(+), 932 deletions(-) create mode 100644 packages/stage/src/ActionManager.ts create mode 100644 packages/stage/src/MoveableOptionsManager.ts delete mode 100644 packages/stage/src/TargetCalibrate.ts create mode 100644 packages/stage/src/TargetShadow.ts diff --git a/docs/src/api/editor/editor.md b/docs/src/api/editor/editor.md index 8d041dbd..a761ead5 100644 --- a/docs/src/api/editor/editor.md +++ b/docs/src/api/editor/editor.md @@ -292,7 +292,7 @@ icon使用的是[element-plus icon](https://element-plus.org/zh-CN/component/ico ### moveableOptions -- **类型:** ((core: StageCore) => MoveableOptions) | [MoveableOptions](https://daybrush.com/moveable/release/latest/doc/) +- **类型:** ((config?: CustomizeMoveableOptionsCallbackConfig) => MoveableOptions) | [MoveableOptions](https://daybrush.com/moveable/release/latest/doc/) - **默认值:** {} diff --git a/magic-admin/web/src/views/editor.vue b/magic-admin/web/src/views/editor.vue index 0c7cb87b..7e2856b7 100644 --- a/magic-admin/web/src/views/editor.vue +++ b/magic-admin/web/src/views/editor.vue @@ -21,7 +21,7 @@ import { Edit, FolderOpened, SwitchButton, Tickets } from '@element-plus/icons-v import type { MoveableOptions } from '@tmagic/editor'; import { ComponentGroup } from '@tmagic/editor'; import { NodeType } from '@tmagic/schema'; -import StageCore from '@tmagic/stage'; +import { CustomizeMoveableOptionsCallbackConfig } from '@tmagic/stage'; import { asyncLoadJs } from '@tmagic/utils'; import editorApi from '@src/api/editor'; @@ -89,10 +89,10 @@ export default defineComponent({ magicPresetConfigs, magicPresetEvents, editorDefaultSelected, - moveableOptions: (core?: StageCore): MoveableOptions => { + moveableOptions: (config?: CustomizeMoveableOptionsCallbackConfig): MoveableOptions => { const options: MoveableOptions = {}; - const id = core?.dr?.target?.id; + const id = config?.targetElId; if (!id || !editor.value) return options; const node = editor.value.editorService.getNodeById(id); diff --git a/packages/core/src/index.ts b/packages/core/src/index.ts index 2ff1619d..a3179fde 100644 --- a/packages/core/src/index.ts +++ b/packages/core/src/index.ts @@ -22,4 +22,6 @@ import './resetcss.css'; export * from './events'; +export { default as Env } from './Env'; + export default App; diff --git a/packages/editor/src/Editor.vue b/packages/editor/src/Editor.vue index 7d0431d5..3a590f3b 100644 --- a/packages/editor/src/Editor.vue +++ b/packages/editor/src/Editor.vue @@ -69,8 +69,13 @@ import { defineComponent, onUnmounted, PropType, provide, reactive, toRaw, watch import { EventOption } from '@tmagic/core'; import type { FormConfig } from '@tmagic/form'; import type { MApp, MNode } from '@tmagic/schema'; -import type StageCore from '@tmagic/stage'; -import { CONTAINER_HIGHLIGHT_CLASS, ContainerHighlightType, MoveableOptions } from '@tmagic/stage'; +import { + CONTAINER_HIGHLIGHT_CLASS_NAME, + ContainerHighlightType, + CustomizeMoveableOptionsCallbackConfig, + MoveableOptions, + UpdateDragEl, +} from '@tmagic/stage'; import Framework from './layouts/Framework.vue'; import NavMenu from './layouts/NavMenu.vue'; @@ -164,7 +169,9 @@ export default defineComponent({ /** 画布中组件选中框的移动范围 */ moveableOptions: { - type: [Object, Function] as PropType MoveableOptions)>, + type: [Object, Function] as PropType< + MoveableOptions | ((config?: CustomizeMoveableOptionsCallbackConfig) => MoveableOptions) + >, }, /** 编辑器初始化时默认选中的组件ID */ @@ -184,7 +191,7 @@ export default defineComponent({ containerHighlightClassName: { type: String, - default: CONTAINER_HIGHLIGHT_CLASS, + default: CONTAINER_HIGHLIGHT_CLASS_NAME, }, containerHighlightDuration: { @@ -207,7 +214,7 @@ export default defineComponent({ }, updateDragEl: { - type: Function as PropType<(el: HTMLDivElement, target: HTMLElement) => void>, + type: Function as PropType, }, }, diff --git a/packages/editor/src/type.ts b/packages/editor/src/type.ts index 2711f6a9..70bb77f7 100644 --- a/packages/editor/src/type.ts +++ b/packages/editor/src/type.ts @@ -21,7 +21,12 @@ import type { Component } from 'vue'; import type { FormConfig } from '@tmagic/form'; import type { CodeBlockContent, CodeBlockDSL, Id, MApp, MContainer, MNode, MPage } from '@tmagic/schema'; import type StageCore from '@tmagic/stage'; -import type { ContainerHighlightType, MoveableOptions } from '@tmagic/stage'; +import type { + ContainerHighlightType, + CustomizeMoveableOptionsCallbackConfig, + MoveableOptions, + UpdateDragEl, +} from '@tmagic/stage'; import type { CodeBlockService } from './services/codeBlock'; import type { ComponentListService } from './services/componentList'; @@ -57,10 +62,10 @@ export interface StageOptions { containerHighlightDuration: number; containerHighlightType: ContainerHighlightType; render: () => HTMLDivElement; - moveableOptions: MoveableOptions | ((core?: StageCore) => MoveableOptions); + moveableOptions: MoveableOptions | ((config?: CustomizeMoveableOptionsCallbackConfig) => MoveableOptions); canSelect: (el: HTMLElement) => boolean | Promise; isContainer: (el: HTMLElement) => boolean | Promise; - updateDragEl: (el: HTMLDivElement) => void; + updateDragEl: UpdateDragEl; } export interface StoreState { diff --git a/packages/editor/src/utils/stage.ts b/packages/editor/src/utils/stage.ts index ff21ba75..7ea2f295 100644 --- a/packages/editor/src/utils/stage.ts +++ b/packages/editor/src/utils/stage.ts @@ -54,7 +54,7 @@ export const useStage = (stageOptions: StageOptions) => { editorService.highlight(el.id); }); - stage.on('multiSelect', (els: HTMLElement[]) => { + stage.on('multi-select', (els: HTMLElement[]) => { editorService.multiSelect(els.map((el) => el.id)); }); @@ -79,7 +79,7 @@ export const useStage = (stageOptions: StageOptions) => { editorService.get('stage').select(parent.id); }); - stage.on('changeGuides', (e) => { + stage.on('change-guides', (e) => { uiService.set('showGuides', true); if (!root.value || !page.value) return; diff --git a/packages/stage/README.md b/packages/stage/README.md index 0c425cb8..4b6d8153 100644 --- a/packages/stage/README.md +++ b/packages/stage/README.md @@ -1 +1,57 @@ -# [文档](https://tencent.github.io/tmagic-editor/docs/) \ No newline at end of file +# 画布功能介绍 +画布是编辑器中最核心的功能,处理组件拖拽和所见即所得的展示。 +## 画布整体功能示意图 +![魔方编辑器](https://vfiles.gtimg.cn/vupload/20221113/78b8ab1668310500232.png) +如上图所示,中间粉色区域及其周边的标尺,是画布区域,就是这个模块代码要处理的内容。

+## 已选组件的组件树 +![已选组件列表](https://vfiles.gtimg.cn/vupload/20221113/c3816e1668311041998.png) +已选组件列表,组件列表也可以单选、多选、高亮、删除、拖拽组件到容器内

+ +## 画布支持的功能 +- 渲染runtime +- 从编辑器增加组件,可以在左侧组件列表中通过单击/拖拽往画布中加入组件 +- 删除组件,在画布中右键单击组件,在弹出菜单中删除;或者在左侧已选组件的组件树中右键删除组件 +- 单选拖拽组件,可以在画布中选中组件,也可以在左侧目录中 +- 多选拖拽组件,通过按住ctrl健选中多个组件 +- 拖拽改变组件大小 +- 旋转组件 +- 高亮组件,在画布中mousemove经过组件的时候,或者在组件树中mousemove经过组件的时候,高亮组件 +- 配置组件,单选选中组件之后,右侧表单区域对组件进行配置,并更新组件的渲染 +- 添加/删除/隐藏/显示参考线,通过在标尺中往画布中拖拽,给画布添加参考线,图中两条竖向和一天横向的红色线条就是参考线 +- 辅助对齐,单选和多选都支持拖拽过程中会辅助对齐其它组件,并在靠近参考线时吸附到参考线 +- 拖拽组件进入容器,支持通过在画布中单选,或者在组件树中单选,将组件拖拽进入容器 +

+# 核心类介绍 +## StageCore +- 负责统一对外接口,编辑器通过StageCore传入runtime、添加/删除组件、缩放画布、更新参考线和标尺等;同时StageCore也会对外抛出事件,比如组件选中、多选、高亮、更新,runtimeReady等。 +- 管理三个核心类:StageRender、StageMask、ActionManager +

+## StageRender +基于iframe加载传入进来的runtimeUrl,并支持增删改查组件。还提供了一个核心API:getElementsFromPoint,该API负责获取指定坐标下所有dom节点。 +

+## StageMask +mask是一个盖在画布区域的一个蒙层,主要作用是隔离鼠标事件,避免鼠标事件直接作用于runtime中的组件,从而避免触发组件本身的点击事件(比如链接组件会跳走)。mask在滚动画布时,需要保证同步大小。 +

+## ActionManager +- 负责监听鼠标和键盘事件,基于这些事件,形成单选、多选、高亮行为。主要监听的是蒙层上的鼠标事件,通过StageRender.getElementsFromPoint计算获得鼠标下方的组件,实现事件监听和实际组件的解构。 +- 向上负责跟StageCore双向通信,提供接口供core调用,并向core抛出事件 +- 向下管理StageDragResize、StageMultiDragResize、StageHightlight这三个单选、多选、高亮类,让它们协同工作 +

+## StageDragResize +负责单选相关逻辑,拖拽、改变大小、旋转等行为是依赖于开源库Moveable实现的,这些行为并不是直接作用于组件本身,而是在蒙层上创建了一个跟组件同等大小的边框div,实际拖拽的是边框div,在拖拽过程中同步更新组件。 +这个类的主要工作包括: +- 初始化组件操作边框,初始化moveable参数 +- 更新moveable参数,比如增加了参考线、缩放了大小、表单改变了组件,都需要更新 +- 接收moveable的回调函数,同步去更新实际组件的渲染 +

+## StageMultiDragResize +功能跟StageDragResize类似,只是这个类是负责多选操作的,通过ctrl健选中多个组件,多选状态下不支持通过表单配置组件。 +

+## StageHightlight +在鼠标经过画布中的组件、或者鼠标经过组件目录树中的组件时,会触发组件高亮,高亮也是通过moveable实现的,这个类主要负责初始化moveable并管理高亮状态。 +

+## MoveableOptionsManager +StageDragResize、StageMultiDragResize的父类,负责管理Moveable的配置 +

+## TargetShadow +统一管理拖拽和高亮框,包括创建、更新、销毁。 diff --git a/packages/stage/src/ActionManager.ts b/packages/stage/src/ActionManager.ts new file mode 100644 index 00000000..bf8a606c --- /dev/null +++ b/packages/stage/src/ActionManager.ts @@ -0,0 +1,534 @@ +/* + * Tencent is pleased to support the open source community by making TMagicEditor available. + * + * Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import EventEmitter from 'events'; + +import KeyController from 'keycon'; +import { throttle } from 'lodash-es'; + +import { Env } from '@tmagic/core'; +import { Id } from '@tmagic/schema'; +import { addClassName, getDocument, removeClassNameByClassName } from '@tmagic/utils'; + +import { CONTAINER_HIGHLIGHT_CLASS_NAME, GHOST_EL_ID_PREFIX, GuidesType, MouseButton, PAGE_CLASS } from './const'; +import StageDragResize from './StageDragResize'; +import StageHighlight from './StageHighlight'; +import StageMultiDragResize from './StageMultiDragResize'; +import { + ActionManagerConfig, + CanSelect, + ContainerHighlightType, + CustomizeMoveableOptions, + CustomizeMoveableOptionsCallbackConfig, + GetElementsFromPoint, + GetRenderDocument, + GetTargetElement, + IsContainer, + Point, + SelectStatus, + StageDragStatus, + UpdateEventData, +} from './types'; +import { isMoveableButton } from './util'; + +const throttleTime = 100; +const defaultContainerHighlightDuration = 800; + +/** + * 管理蒙层mask之上的操作:1、监听键盘鼠标事件,判断形成单选、多选、高亮操作;2、管理单选、多选、高亮三个类协同工作。 + * @extends EventEmitter + */ +export default class ActionManager extends EventEmitter { + private dr: StageDragResize; + private multiDr: StageMultiDragResize; + private highlightLayer: StageHighlight; + /** 单选、多选、高亮的容器(蒙层的content) */ + private container: HTMLElement; + /** 当前选中的节点 */ + private selectedEl: HTMLElement | undefined; + /** 多选选中的节点组 */ + private selectedElList: HTMLElement[] = []; + /** 当前高亮的节点 */ + private highlightedEl: HTMLElement | undefined; + /** 当前是否处于多选状态 */ + private isMultiSelectStatus = false; + /** 当拖拽组件到容器上方进入可加入容器状态时,给容器添加的一个class名称 */ + private containerHighlightClassName: string; + /** 当拖拽组件到容器上方时,需要悬停多久才能将组件加入容器 */ + private containerHighlightDuration: number; + /** 将组件加入容器的操作方式 */ + private containerHighlightType?: ContainerHighlightType; + private isAltKeydown = false; + private getTargetElement: GetTargetElement; + private getElementsFromPoint: GetElementsFromPoint; + private canSelect: CanSelect; + private isContainer: IsContainer; + private getRenderDocument: GetRenderDocument; + + private mouseMoveHandler = throttle(async (event: MouseEvent): Promise => { + const el = await this.getElementFromPoint(event); + if (!el) { + this.clearHighlight(); + return; + } + this.highlight(el); + }, throttleTime); + + constructor(config: ActionManagerConfig) { + super(); + this.container = config.container; + this.containerHighlightClassName = config.containerHighlightClassName || CONTAINER_HIGHLIGHT_CLASS_NAME; + this.containerHighlightDuration = config.containerHighlightDuration || defaultContainerHighlightDuration; + this.containerHighlightType = config.containerHighlightType; + this.getTargetElement = config.getTargetElement; + this.getElementsFromPoint = config.getElementsFromPoint; + this.canSelect = config.canSelect || ((el: HTMLElement) => !!el.id); + this.getRenderDocument = config.getRenderDocument; + this.isContainer = config.isContainer; + + this.dr = new StageDragResize({ + container: config.container, + getRootContainer: config.getRootContainer, + getRenderDocument: config.getRenderDocument, + updateDragEl: config.updateDragEl, + markContainerEnd: () => this.markContainerEnd(), + delayedMarkContainer: (event: MouseEvent, exclude: Element[]) => { + if (this.canAddToContainer()) { + return this.delayedMarkContainer(event, exclude); + } + return undefined; + }, + moveableOptions: this.changeCallback(config.moveableOptions), + }); + this.multiDr = new StageMultiDragResize({ + container: config.container, + multiMoveableOptions: config.multiMoveableOptions, + getRootContainer: config.getRootContainer, + getRenderDocument: config.getRenderDocument, + updateDragEl: config.updateDragEl, + }); + this.highlightLayer = new StageHighlight({ + container: config.container, + updateDragEl: config.updateDragEl, + getRootContainer: config.getRootContainer, + }); + + this.initMouseEvent(); + this.initKeyEvent(); + this.initActionEvent(); + } + + /** + * 设置水平/垂直参考线 + * @param type 参考线类型 + * @param guidelines 参考线坐标数组 + */ + public setGuidelines(type: GuidesType, guidelines: number[]): void { + this.dr.setGuidelines(type, guidelines); + this.multiDr.setGuidelines(type, guidelines); + } + + /** + * 清空所有参考线 + */ + public clearGuides(): void { + this.dr.clearGuides(); + this.multiDr.clearGuides(); + } + + /** + * 更新moveable,外部主要调用场景是元素配置变更、页面大小变更 + * @param el 变更的元素 + */ + public updateMoveable(el?: HTMLElement): void { + this.dr.updateMoveable(el); + // 多选时不可配置元素,因此不存在多选元素变更,不需要传el + this.multiDr.updateMoveable(); + } + + /** + * 判断是否单选选中的元素 + */ + public isSelectedEl(el: HTMLElement): boolean { + // 有可能dom已经重新渲染,不再是原来的dom了,所以这里判断id,而不是判断el === this.selectedDom + return el.id === this.selectedEl?.id; + } + + public setSelectedEl(el: HTMLElement): void { + this.selectedEl = el; + } + + public getSelectedEl(): HTMLElement | undefined { + return this.selectedEl; + } + + public getSelectedElList(): HTMLElement[] { + return this.selectedElList; + } + + /** + * 获取鼠标下方第一个可选中元素,如果元素层叠,返回到是最上层元素 + * @param event 鼠标事件 + * @returns 鼠标下方第一个可选中元素 + */ + public async getElementFromPoint(event: MouseEvent): Promise { + const els = this.getElementsFromPoint(event as Point); + + let stopped = false; + const stop = () => (stopped = true); + for (const el of els) { + if (!el.id.startsWith(GHOST_EL_ID_PREFIX) && (await this.isElCanSelect(el, event, stop))) { + if (stopped) break; + return el; + } + } + } + + /** + * 判断一个元素能否在当前场景被选中 + * @param el 被判断的元素 + * @param event 鼠标事件 + * @param stop 通过该元素如果得知剩下的元素都不可被选中,通知调用方终止对剩下元素的判断 + * @returns 能否选中 + */ + public async isElCanSelect(el: HTMLElement, event: MouseEvent, stop: () => boolean): Promise { + // 执行业务方传入的判断逻辑 + const canSelectByProp = await this.canSelect(el, event, stop); + if (!canSelectByProp) return false; + // 多选规则 + if (this.isMultiSelectStatus) { + return this.canMultiSelect(el, stop); + } + return true; + } + + /** + * 判断一个元素是否可以被多选,如果当前元素是page,则调stop函数告诉调用方不必继续判断其它元素了 + */ + public canMultiSelect(el: HTMLElement, stop: () => boolean): boolean { + // 多选状态下不可以选中magic-ui-page,并停止继续向上层选中 + if (el.className.includes(PAGE_CLASS)) { + stop(); + return false; + } + const selectedEl = this.getSelectedEl(); + // 先单击选中了页面(magic-ui-page),再按住多选键多选时,任一元素均可选中 + if (selectedEl?.className.includes(PAGE_CLASS)) { + return true; + } + return this.multiDr.canSelect(el, selectedEl); + } + + public select(el: HTMLElement, event: MouseEvent | undefined): void { + this.selectedEl = el; + this.clearSelectStatus(SelectStatus.MULTI_SELECT); + this.dr.select(el, event); + } + + public multiSelect(idOrElList: HTMLElement[] | Id[]): void { + this.selectedElList = idOrElList.map((idOrEl) => this.getTargetElement(idOrEl)); + this.clearSelectStatus(SelectStatus.SELECT); + this.multiDr.multiSelect(this.selectedElList); + } + + public getHighlightEl(): HTMLElement | undefined { + return this.highlightedEl; + } + + public setHighlightEl(el: HTMLElement | undefined): void { + this.highlightedEl = el; + } + + public highlight(idOrEl: Id | HTMLElement): void { + let el; + try { + el = this.getTargetElement(idOrEl); + } catch (error) { + this.clearHighlight(); + return; + } + + // 选中组件不高亮、多选拖拽状态不高亮 + if (el === this.getSelectedEl() || this.multiDr.dragStatus === StageDragStatus.ING) { + this.clearHighlight(); + return; + } + if (el === this.highlightedEl || !el) return; + + this.highlightLayer.highlight(el); + this.highlightedEl = el; + this.emit('highlight', el); + } + + public clearHighlight(): void { + this.setHighlightEl(undefined); + this.highlightLayer.clearHighlight(); + } + + /** + * 用于在切换选择模式时清除上一次的状态 + * @param selectType 需要清理的选择模式 + */ + public clearSelectStatus(selectType: SelectStatus): void { + if (selectType === SelectStatus.MULTI_SELECT) { + this.multiDr.clearSelectStatus(); + this.selectedElList = []; + } else { + this.dr.clearSelectStatus(); + } + } + + /** + * 找到鼠标下方的容器,通过添加className对容器进行标记 + * @param event 鼠标事件 + * @param excludeElList 计算鼠标点所在容器时要排除的元素列表 + */ + public async addContainerHighlightClassName(event: MouseEvent, excludeElList: Element[]): Promise { + const doc = this.getRenderDocument(); + if (!doc) return; + + const els = this.getElementsFromPoint(event); + + for (const el of els) { + if (!el.id.startsWith(GHOST_EL_ID_PREFIX) && (await this.isContainer(el)) && !excludeElList.includes(el)) { + addClassName(el, doc, this.containerHighlightClassName); + break; + } + } + } + + /** + * 鼠标拖拽着元素,在容器上方悬停,延迟一段时间后,对容器进行标记,如果悬停时间够长将标记成功,悬停时间短,调用方通过返回的timeoutId取消标记 + * 标记的作用:1、高亮容器,给用户一个加入容器的交互感知;2、释放鼠标后,通过标记的标志找到要加入的容器 + * @param event 鼠标事件 + * @param excludeElList 计算鼠标所在容器时要排除的元素列表 + * @returns timeoutId,调用方在鼠标移走时要取消该timeout,阻止标记 + */ + public delayedMarkContainer(event: MouseEvent, excludeElList: Element[] = []): NodeJS.Timeout { + return globalThis.setTimeout(() => { + this.addContainerHighlightClassName(event, excludeElList); + }, this.containerHighlightDuration); + } + + public destroy(): void { + this.container.removeEventListener('mousedown', this.mouseDownHandler); + this.container.removeEventListener('mousemove', this.mouseMoveHandler); + this.container.removeEventListener('mouseleave', this.mouseLeaveHandler); + this.container.removeEventListener('wheel', this.mouseWheelHandler); + this.dr.destroy(); + this.multiDr.destroy(); + this.highlightLayer.destroy(); + } + + private changeCallback(options: CustomizeMoveableOptions): CustomizeMoveableOptions { + // 在actionManager才能获取到各种参数,在这里传好参数有比较好的扩展性 + if (typeof options === 'function') { + return () => { + // 要再判断一次,不然过不了ts检查 + if (typeof options === 'function') { + const cfg: CustomizeMoveableOptionsCallbackConfig = { + targetElId: this.selectedEl?.id, + }; + return options(cfg); + } + return options; + }; + } + return options; + } + + /** + * 在执行多选逻辑前,先准备好多选选中元素 + * @param el 新选中的元素 + * @returns 多选选中的元素列表 + */ + private async beforeMultiSelect(event: MouseEvent): Promise { + const el = await this.getElementFromPoint(event); + if (!el) return; + + // 如果已有单选选中元素,不是magic-ui-page就可以加入多选列表 + if (this.selectedEl && !this.selectedEl.className.includes(PAGE_CLASS)) { + this.selectedElList.push(this.selectedEl as HTMLElement); + this.selectedEl = undefined; + } + // 判断元素是否已在多选列表 + const existIndex = this.selectedElList.findIndex((selectedDom) => selectedDom.id === el.id); + if (existIndex !== -1) { + // 再次点击取消选中 + this.selectedElList.splice(existIndex, 1); + } else { + this.selectedElList.push(el); + } + } + + /** + * 当前状态下能否将组件加入容器,默认是鼠标悬停一段时间加入,alt模式则是按住alt+鼠标悬停一段时间加入 + */ + private canAddToContainer(): boolean { + return ( + this.containerHighlightType === ContainerHighlightType.DEFAULT || + (this.containerHighlightType === ContainerHighlightType.ALT && this.isAltKeydown) + ); + } + + /** + * 结束对container的标记状态 + * @returns 标记的容器元素,没有标记的容器时返回null + */ + private markContainerEnd(): HTMLElement | null { + const doc = this.getRenderDocument(); + if (doc && this.canAddToContainer()) { + return removeClassNameByClassName(doc, this.containerHighlightClassName); + } + return null; + } + + private initMouseEvent(): void { + this.container.addEventListener('mousedown', this.mouseDownHandler); + this.container.addEventListener('mousemove', this.mouseMoveHandler); + this.container.addEventListener('mouseleave', this.mouseLeaveHandler); + this.container.addEventListener('wheel', this.mouseWheelHandler); + } + + /** + * 初始化键盘事件监听 + */ + private initKeyEvent(): void { + const { isMac } = new Env(); + const ctrl = isMac ? 'meta' : 'ctrl'; + + // 多选启用状态监听 + KeyController.global.keydown(ctrl, (e) => { + e.inputEvent.preventDefault(); + this.isMultiSelectStatus = true; + }); + // ctrl+tab切到其他窗口,需要将多选状态置为false + KeyController.global.on('blur', () => { + this.isMultiSelectStatus = false; + }); + KeyController.global.keyup(ctrl, (e) => { + e.inputEvent.preventDefault(); + this.isMultiSelectStatus = false; + }); + + // alt健监听,用于启用拖拽组件加入容器状态 + KeyController.global.keydown('alt', (e) => { + e.inputEvent.preventDefault(); + this.isAltKeydown = true; + }); + + KeyController.global.keyup('alt', (e) => { + e.inputEvent.preventDefault(); + this.markContainerEnd(); + this.isAltKeydown = false; + }); + } + + /** + * 处理单选、多选抛出来的事件 + */ + private initActionEvent(): void { + this.dr + .on('update', (data: UpdateEventData) => { + // 点击组件并立即拖动的场景,要保证select先被触发,延迟update通知 + setTimeout(() => this.emit('update', data)); + }) + .on('sort', (data: UpdateEventData) => { + // 点击组件并立即拖动的场景,要保证select先被触发,延迟update通知 + setTimeout(() => this.emit('sort', data)); + }) + .on('select-parent', () => { + this.emit('select-parent'); + }); + + this.multiDr + .on('update', (data: UpdateEventData, parentEl: HTMLElement | null) => { + this.emit('multi-update', data, parentEl); + }) + .on('change-to-select', async (id: Id) => { + // 如果还在多选状态,不触发切换到单选 + if (this.isMultiSelectStatus) return false; + const el = this.getTargetElement(id); + this.emit('change-to-select', el); + }); + } + + /** + * 在down事件中集中cpu处理画布中选中操作渲染,在up事件中再通知外面的编辑器更新 + */ + private mouseDownHandler = async (event: MouseEvent): Promise => { + this.clearHighlight(); + event.stopImmediatePropagation(); + event.stopPropagation(); + + if (this.isStopTriggerSelect(event)) return; + + // 点击状态下不触发高亮事件 + this.container.removeEventListener('mousemove', this.mouseMoveHandler); + + // 判断触发多选还是单选 + if (this.isMultiSelectStatus) { + await this.beforeMultiSelect(event); + if (this.selectedElList.length > 0) { + this.emit('before-multi-select', this.selectedElList); + } + } else { + const el = await this.getElementFromPoint(event); + if (!el) return; + this.emit('before-select', el, event); + } + getDocument().addEventListener('mouseup', this.mouseUpHandler); + }; + + private isStopTriggerSelect(event: MouseEvent): boolean { + if (event.button !== MouseButton.LEFT && event.button !== MouseButton.RIGHT) return true; + if (!event.target) return true; + + const targetClassList = (event.target as HTMLDivElement).classList; + + // 如果单击多选选中区域,则不需要再触发选中了,要支持此处单击后进行拖动 + if (!this.isMultiSelectStatus && targetClassList.contains('moveable-area')) { + return true; + } + // 点击对象如果是边框锚点,则可能是resize; 点击对象是功能按钮 + if (targetClassList.contains('moveable-control') || isMoveableButton(event.target as Element)) { + return true; + } + return false; + } + + /** + * 在up事件中负责对外通知选中事件,通知画布之外的编辑器更新 + */ + private mouseUpHandler = (): void => { + getDocument().removeEventListener('mouseup', this.mouseUpHandler); + this.container.addEventListener('mousemove', this.mouseMoveHandler); + if (this.isMultiSelectStatus) { + this.emit('multi-select', this.selectedElList); + } else { + this.emit('select', this.selectedEl); + } + }; + + private mouseLeaveHandler = () => { + setTimeout(() => this.clearHighlight(), throttleTime); + }; + + private mouseWheelHandler = () => { + this.clearHighlight(); + }; +} diff --git a/packages/stage/src/MoveableOptionsManager.ts b/packages/stage/src/MoveableOptionsManager.ts new file mode 100644 index 00000000..86432a17 --- /dev/null +++ b/packages/stage/src/MoveableOptionsManager.ts @@ -0,0 +1,249 @@ +/* + * Tencent is pleased to support the open source community by making TMagicEditor available. + * + * Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +import EventEmitter from 'events'; + +import { merge } from 'lodash-es'; +import { MoveableOptions } from 'moveable'; + +import { GuidesType, Mode } from './const'; +import selectParentAbles from './MoveableSelectParentAble'; +import { GetRootContainer, MoveableOptionsManagerConfig } from './types'; +import { getOffset } from './util'; + +/** + * 单选和多选的父类,用于管理moveableOptions + * @extends EventEmitter + */ +export default class MoveableOptionsManager extends EventEmitter { + /** 布局方式:流式布局、绝对定位、固定定位 */ + public mode: Mode = Mode.ABSOLUTE; + + /** 画布容器 */ + protected container: HTMLElement; + + /** 水平参考线 */ + private horizontalGuidelines: number[] = []; + /** 垂直参考线 */ + private verticalGuidelines: number[] = []; + /** 对齐元素集合 */ + private elementGuidelines: HTMLElement[] = []; + /** 由外部调用方(编辑器)传入进来的moveable默认参数,可以为空,也可以是一个回调函数 */ + private customizedOptions?: (() => MoveableOptions) | MoveableOptions; + /** 获取整个画布的根元素(在StageCore的mount函数中挂载的container) */ + private getRootContainer: GetRootContainer; + + constructor(config: MoveableOptionsManagerConfig) { + super(); + this.customizedOptions = config.moveableOptions; + this.container = config.container; + this.getRootContainer = config.getRootContainer; + } + + /** + * 设置水平/垂直参考线 + * @param type 参考线类型 + * @param guidelines 参考线坐标数组 + */ + public setGuidelines(type: GuidesType, guidelines: number[]): void { + if (type === GuidesType.HORIZONTAL) { + this.horizontalGuidelines = guidelines; + } else if (type === GuidesType.VERTICAL) { + this.verticalGuidelines = guidelines; + } + + this.emit('update-moveable'); + } + + /** + * 清除横向和纵向的参考线 + */ + public clearGuides(): void { + this.horizontalGuidelines = []; + this.verticalGuidelines = []; + + this.emit('update-moveable'); + } + + /** + * 设置有哪些元素要辅助对齐 + * @param selectedElList 选中的元素列表,需要排除在对齐元素之外 + * @param allElList 全部元素列表 + */ + protected setElementGuidelines(selectedElList: HTMLElement[], allElList: HTMLElement[]): void { + this.elementGuidelines.forEach((node) => { + node.remove(); + }); + this.elementGuidelines = []; + + if (this.mode === Mode.ABSOLUTE) { + this.container.append(this.createGuidelineElements(selectedElList, allElList)); + } + } + + /** + * 获取moveable参数 + * @param isMultiSelect 是否多选模式 + * @param runtimeOptions 调用时实时传进来的的moveable参数 + * @returns moveable所需参数 + */ + protected getOptions(isMultiSelect: boolean, runtimeOptions: MoveableOptions = {}): MoveableOptions { + const defaultOptions = this.getDefaultOptions(isMultiSelect); + const customizedOptions = this.getCustomizeOptions(); + + return merge(defaultOptions, customizedOptions, runtimeOptions); + } + + /** + * 获取单选和多选的moveable公共参数 + * @returns moveable公共参数 + */ + private getDefaultOptions(isMultiSelect: boolean): MoveableOptions { + const isSortable = this.mode === Mode.SORTABLE; + + const commonOptions = { + draggable: true, + resizable: true, + rootContainer: this.getRootContainer(), + zoom: 1, + throttleDrag: 0, + snappable: true, + horizontalGuidelines: this.horizontalGuidelines, + verticalGuidelines: this.verticalGuidelines, + elementGuidelines: this.elementGuidelines, + bounds: { + top: 0, + // 设置0的话无法移动到left为0,所以只能设置为-1 + left: -1, + right: this.container.clientWidth - 1, + bottom: isSortable ? undefined : this.container.clientHeight, + }, + }; + const differenceOptions = isMultiSelect ? this.getMultiOptions() : this.getSingleOptions(); + + return merge(commonOptions, differenceOptions); + } + + /** + * 获取单选下的差异化参数 + * @returns {MoveableOptions} moveable options参数 + */ + private getSingleOptions(): MoveableOptions { + const isAbsolute = this.mode === Mode.ABSOLUTE; + const isFixed = this.mode === Mode.FIXED; + + return { + origin: false, + dragArea: false, + scalable: false, + rotatable: false, + snapGap: isAbsolute || isFixed, + snapThreshold: 5, + snapDigit: 0, + isDisplaySnapDigit: isAbsolute, + snapDirections: { + top: isAbsolute, + right: isAbsolute, + bottom: isAbsolute, + left: isAbsolute, + center: isAbsolute, + middle: isAbsolute, + }, + elementSnapDirections: { + top: isAbsolute, + right: isAbsolute, + bottom: isAbsolute, + left: isAbsolute, + }, + isDisplayInnerSnapDigit: true, + + props: { + selectParent: true, + }, + + ables: [selectParentAbles(this.selectParentHandler.bind(this))], + }; + } + + /** + * 获取多选下的差异化参数 + * @returns {MoveableOptions} moveable options参数 + */ + private getMultiOptions(): MoveableOptions { + return { + defaultGroupRotate: 0, + defaultGroupOrigin: '50% 50%', + startDragRotate: 0, + throttleDragRotate: 0, + origin: true, + padding: { left: 0, top: 0, right: 0, bottom: 0 }, + }; + } + + /** + * 获取业务方自定义的moveable参数 + */ + private getCustomizeOptions(): MoveableOptions | undefined { + if (typeof this.customizedOptions === 'function') { + return this.customizedOptions(); + } + return this.customizedOptions; + } + + /** + * 这是给selectParentAbles的回调函数,用于触发选中父元素事件 + */ + private selectParentHandler(): void { + this.emit('select-parent'); + } + + /** + * 为需要辅助对齐的元素创建div + * @param selectedElList 选中的元素列表,需要排除在对齐元素之外 + * @param allElList 全部元素列表 + * @returns frame 辅助对齐元素集合的页面片 + */ + private createGuidelineElements(selectedElList: HTMLElement[], allElList: HTMLElement[]): DocumentFragment { + const frame = globalThis.document.createDocumentFragment(); + + for (const node of allElList) { + const { width, height } = node.getBoundingClientRect(); + if (this.isInElementList(node, selectedElList)) continue; + const { left, top } = getOffset(node); + const elementGuideline = globalThis.document.createElement('div'); + elementGuideline.style.cssText = `position: absolute;width: ${width}px;height: ${height}px;top: ${top}px;left: ${left}px`; + this.elementGuidelines.push(elementGuideline); + frame.append(elementGuideline); + } + + return frame; + } + + /** + * 判断一个元素是否在元素列表里面 + * @param ele 元素 + * @param eleList 元素列表 + * @returns 是否在元素列表里面 + */ + private isInElementList(ele: HTMLElement, eleList: HTMLElement[]): boolean { + for (const eleItem of eleList) { + if (ele === eleItem) return true; + } + return false; + } +} diff --git a/packages/stage/src/MoveableSelectParentAble.ts b/packages/stage/src/MoveableSelectParentAble.ts index 8a4e105e..6fa973b2 100644 --- a/packages/stage/src/MoveableSelectParentAble.ts +++ b/packages/stage/src/MoveableSelectParentAble.ts @@ -1,9 +1,7 @@ import { MoveableManagerInterface, Renderer } from 'moveable'; -import type StageDragResize from './StageDragResize'; - -export default (dr: StageDragResize) => ({ - name: 'selectParent', +export default (selectParentHandler: () => void) => ({ + name: 'select-parent', props: {}, events: {}, render(moveable: MoveableManagerInterface, React: Renderer) { @@ -51,7 +49,7 @@ export default (dr: StageDragResize) => ({ className: 'moveable-button', title: '选中父组件', onClick: () => { - dr.emit('select-parent'); + selectParentHandler(); }, }, React.createElement( diff --git a/packages/stage/src/Rule.ts b/packages/stage/src/Rule.ts index dfd54ae8..fd213cc6 100644 --- a/packages/stage/src/Rule.ts +++ b/packages/stage/src/Rule.ts @@ -60,12 +60,12 @@ export default class Rule extends EventEmitter { defaultGuides: vLines, }); - this.emit('changeGuides', { + this.emit('change-guides', { type: GuidesType.HORIZONTAL, guides: hLines, }); - this.emit('changeGuides', { + this.emit('change-guides', { type: GuidesType.VERTICAL, guides: vLines, }); @@ -143,7 +143,7 @@ export default class Rule extends EventEmitter { private hGuidesChangeGuidesHandler = (e: GuidesEvents['changeGuides']) => { this.horizontalGuidelines = e.guides; - this.emit('changeGuides', { + this.emit('change-guides', { type: GuidesType.HORIZONTAL, guides: this.horizontalGuidelines, }); @@ -151,7 +151,7 @@ export default class Rule extends EventEmitter { private vGuidesChangeGuidesHandler = (e: GuidesEvents['changeGuides']) => { this.verticalGuidelines = e.guides; - this.emit('changeGuides', { + this.emit('change-guides', { type: GuidesType.VERTICAL, guides: this.verticalGuidelines, }); diff --git a/packages/stage/src/StageCore.ts b/packages/stage/src/StageCore.ts index 7a18b65e..feac9806 100644 --- a/packages/stage/src/StageCore.ts +++ b/packages/stage/src/StageCore.ts @@ -18,303 +18,149 @@ import { EventEmitter } from 'events'; -import type { Id } from '@tmagic/schema'; -import { addClassName } from '@tmagic/utils'; +import { Id } from '@tmagic/schema'; -import { CONTAINER_HIGHLIGHT_CLASS, DEFAULT_ZOOM, GHOST_EL_ID_PREFIX, PAGE_CLASS } from './const'; -import StageDragResize from './StageDragResize'; -import StageHighlight from './StageHighlight'; +import ActionManager from './ActionManager'; +import { DEFAULT_ZOOM } from './const'; import StageMask from './StageMask'; -import StageMultiDragResize from './StageMultiDragResize'; import StageRender from './StageRender'; import { - CanSelect, - ContainerHighlightType, + ActionManagerConfig, + CustomizeRender, GuidesEventData, - IsContainer, + Point, RemoveData, Runtime, - SortEventData, StageCoreConfig, - StageDragStatus, UpdateData, UpdateEventData, } from './types'; -import { addSelectedClassName, removeSelectedClassName } from './util'; +/** + * 负责管理画布,管理renderer、mask、actionManager三个核心类,并负责统一对外通信,包括提供接口和抛事件 + */ export default class StageCore extends EventEmitter { public container?: HTMLDivElement; - // 当前选中的节点 - public selectedDom: HTMLElement | undefined; - // 多选选中的节点组 - public selectedDomList: HTMLElement[] = []; - public highlightedDom: Element | undefined; public renderer: StageRender; public mask: StageMask; - public dr: StageDragResize; - public multiDr: StageMultiDragResize; - public highlightLayer: StageHighlight; - public config: StageCoreConfig; - public zoom = DEFAULT_ZOOM; - public containerHighlightClassName: string; - public containerHighlightDuration: number; - public containerHighlightType?: ContainerHighlightType; - public isContainer: IsContainer; - private canSelect: CanSelect; + private actionManager: ActionManager; private pageResizeObserver: ResizeObserver | null = null; + private autoScrollIntoView: boolean | undefined; + private customizedRender?: CustomizeRender; constructor(config: StageCoreConfig) { super(); - this.config = config; + this.autoScrollIntoView = config.autoScrollIntoView; + this.customizedRender = config.render; - this.setZoom(config.zoom); - this.canSelect = config.canSelect || ((el: HTMLElement) => !!el.id); - this.isContainer = config.isContainer; - this.containerHighlightClassName = config.containerHighlightClassName || CONTAINER_HIGHLIGHT_CLASS; - this.containerHighlightDuration = config.containerHighlightDuration || 800; - this.containerHighlightType = config.containerHighlightType; - - this.renderer = new StageRender({ runtimeUrl: config.runtimeUrl, render: this.render.bind(this) }); + this.renderer = new StageRender({ + runtimeUrl: config.runtimeUrl, + zoom: config.zoom, + customizedRender: async (): Promise => { + if (this?.customizedRender) { + return await this.customizedRender(this); + } + return null; + }, + }); this.mask = new StageMask(); - this.dr = new StageDragResize({ core: this, container: this.mask.content, mask: this.mask }); - this.multiDr = new StageMultiDragResize({ core: this, container: this.mask.content, mask: this.mask }); - this.highlightLayer = new StageHighlight({ core: this, container: this.mask.wrapper }); + this.actionManager = new ActionManager(this.getActionManagerConfig(config)); - this.renderer.on('runtime-ready', (runtime: Runtime) => { - this.emit('runtime-ready', runtime); - }); - this.renderer.on('page-el-update', (el: HTMLElement) => { - this.mask?.observe(el); - this.observePageResize(el); - }); - - this.mask - .on('beforeSelect', async (event: MouseEvent) => { - this.clearSelectStatus('multiSelect'); - const el = await this.getElementFromPoint(event); - if (!el) return; - this.select(el, event); - }) - .on('select', () => { - this.emit('select', this.selectedDom); - }) - .on('changeGuides', (data: GuidesEventData) => { - this.dr.setGuidelines(data.type, data.guides); - this.emit('changeGuides', data); - }) - .on('highlight', async (event: MouseEvent) => { - const el = await this.getElementFromPoint(event); - if (!el) return; - // 如果多选组件处于拖拽状态时不进行组件高亮 - if (this.multiDr.dragStatus === StageDragStatus.ING) return; - await this.highlight(el); - if (this.highlightedDom === this.selectedDom) { - this.highlightLayer.clearHighlight(); - return; - } - this.emit('highlight', this.highlightedDom); - }) - .on('clearHighlight', async () => { - this.highlightLayer.clearHighlight(); - }) - .on('beforeMultiSelect', async (event: MouseEvent) => { - const el = await this.getElementFromPoint(event); - if (!el) return; - // 如果已有单选选中元素,不是magic-ui-page就可以加入多选列表 - if (this.selectedDom && !this.selectedDom.className.includes(PAGE_CLASS)) { - this.selectedDomList.push(this.selectedDom as HTMLElement); - this.selectedDom = undefined; - } - // 判断元素是否已在多选列表 - const existIndex = this.selectedDomList.findIndex((selectedDom) => selectedDom.id === el.id); - if (existIndex !== -1) { - // 再次点击取消选中 - this.selectedDomList.splice(existIndex, 1); - } else { - this.selectedDomList.push(el); - } - this.multiSelect(this.selectedDomList); - this.emit('multiSelect', this.selectedDomList); - }); - - // 要先触发select,在触发update - this.dr - .on('update', (data: UpdateEventData) => { - setTimeout(() => this.emit('update', data)); - }) - .on('sort', (data: UpdateEventData) => { - setTimeout(() => this.emit('sort', data)); - }) - .on('select-parent', () => { - this.emit('select-parent'); - }); - - this.multiDr - .on('update', (data: UpdateEventData) => { - setTimeout(() => this.emit('update', data)); - }) - .on('select', async (id: Id) => { - const el = await this.getTargetElement(id); - this.select(el); // 选中 - setTimeout(() => this.emit('select', el)); // set node - }); - } - - public getElementsFromPoint(event: MouseEvent) { - const { renderer, zoom } = this; - - let x = event.clientX; - let y = event.clientY; - - if (renderer.iframe) { - const rect = renderer.iframe.getClientRects()[0]; - if (rect) { - x = x - rect.left; - y = y - rect.top; - } - } - - return renderer.getDocument()?.elementsFromPoint(x / zoom, y / zoom) as HTMLElement[]; - } - - public async getElementFromPoint(event: MouseEvent) { - const els = this.getElementsFromPoint(event); - let stopped = false; - const stop = () => (stopped = true); - for (const el of els) { - if (!el.id.startsWith(GHOST_EL_ID_PREFIX) && (await this.isElCanSelect(el, event, stop))) { - if (stopped) break; - return el; - } - } - } - - public async isElCanSelect(el: HTMLElement, event: MouseEvent, stop: () => boolean): Promise { - // 执行业务方传入的判断逻辑 - const canSelectByProp = await this.canSelect(el, event, stop); - if (!canSelectByProp) return false; - // 多选规则 - if (this.mask.isMultiSelectStatus) { - return this.multiDr.canSelect(el, stop); - } - return true; + this.initRenderEvent(); + this.initActionEvent(); + this.initMaskEvent(); } /** - * 选中组件 - * @param idOrEl 组件Dom节点的id属性,或者Dom节点 + * 单选选中元素 + * @param idOrEl 选中的id或者元素 */ public async select(idOrEl: Id | HTMLElement, event?: MouseEvent): Promise { - this.clearSelectStatus('multiSelect'); - const el = await this.getTargetElement(idOrEl); + const el = this.renderer.getTargetElement(idOrEl); + if (el === this.actionManager.getSelectedEl()) return; - if (el === this.selectedDom) return; - - const runtime = await this.renderer.getRuntime(); - - await runtime?.select?.(el.id); - - if (runtime?.beforeSelect) { - await runtime.beforeSelect(el); - } + await this.renderer.select([el]); this.mask.setLayout(el); - this.dr.select(el, event); - if (this.config.autoScrollIntoView || el.dataset.autoScrollIntoView) { + this.actionManager.select(el, event); + + if (this.autoScrollIntoView || el.dataset.autoScrollIntoView) { this.mask.observerIntersection(el); } + } - this.selectedDom = el; + /** + * 多选选中多个元素 + * @param idOrElList 选中元素的id或元素列表 + */ + public async multiSelect(idOrElList: HTMLElement[] | Id[]): Promise { + const els = idOrElList.map((idOrEl) => this.renderer.getTargetElement(idOrEl)); + if (els.length === 0) return; - const doc = this.renderer.getDocument(); - if (doc) { - removeSelectedClassName(doc); - if (this.selectedDom) { - addSelectedClassName(this.selectedDom, doc); - } + const lastEl = els[els.length - 1]; + // 是否减少了组件选择 + const isReduceSelect = els.length < this.actionManager.getSelectedElList().length; + await this.renderer.select(els); + + this.mask.setLayout(lastEl); + + this.actionManager.multiSelect(idOrElList); + + if ((this.autoScrollIntoView || lastEl.dataset.autoScrollIntoView) && !isReduceSelect) { + this.mask.observerIntersection(lastEl); } } /** - * 多选 - * @param domList 多选节点 + * 高亮选中元素 + * @param el 要高亮的元素 */ - public async multiSelect(idOrElList: HTMLElement[] | Id[]): Promise { - this.clearSelectStatus('select'); - this.selectedDomList = await Promise.all(idOrElList.map(async (idOrEl) => await this.getTargetElement(idOrEl))); - this.multiDr.multiSelect(this.selectedDomList); + public highlight(idOrEl: Id | HTMLElement): void { + this.actionManager.highlight(idOrEl); } /** - * 更新选中的节点 - * @param data 更新的数据 + * 更新组件 + * @param data 更新组件的数据 */ - public update(data: UpdateData): Promise { + public async update(data: UpdateData): Promise { const { config } = data; - return this.renderer?.getRuntime().then((runtime) => { - runtime?.update?.(data); - // 更新配置后,需要等组件渲染更新 - setTimeout(() => { - const el = this.renderer.getDocument()?.getElementById(`${config.id}`); - // 有可能dom已经重新渲染,不再是原来的dom了,所以这里判断id,而不是判断el === this.selectedDom - if (el && el.id === this.selectedDom?.id) { - this.selectedDom = el; - // 更新了组件的布局,需要重新设置mask是否可以滚动 - this.mask.setLayout(el); - this.dr.updateMoveable(el); - } - }, 0); + await this.renderer.update(data); + // 通过setTimeout等画布中组件完成渲染更新 + setTimeout(() => { + const el = this.renderer.getTargetElement(`${config.id}`); + if (el && this.actionManager.isSelectedEl(el)) { + // 更新了组件的布局,需要重新设置mask是否可以滚动 + this.mask.setLayout(el); + // 组件有更新,需要set + this.actionManager.setSelectedEl(el); + this.actionManager.updateMoveable(el); + } }); } /** - * 高亮选中组件 - * @param el 页面Dom节点 + * 往画布增加一个组件 + * @param data 组件信息数据 */ - public async highlight(idOrEl: HTMLElement | Id): Promise { - let el; - try { - el = await this.getTargetElement(idOrEl); - } catch (error) { - this.highlightLayer.clearHighlight(); - return; - } - if (el === this.highlightedDom || !el) return; - this.highlightLayer.highlight(el); - this.highlightedDom = el; + public async add(data: UpdateData): Promise { + return await this.renderer.add(data); } - public sortNode(data: SortEventData): Promise { - return this.renderer?.getRuntime().then((runtime) => runtime?.sortNode?.(data)); - } - - public add(data: UpdateData): Promise { - return this.renderer?.getRuntime().then((runtime) => runtime?.add?.(data)); - } - - public remove(data: RemoveData): Promise { - return this.renderer?.getRuntime().then((runtime) => runtime?.remove?.(data)); + /** + * 从画布删除一个组件 + * @param data 组件信息数据 + */ + public async remove(data: RemoveData): Promise { + return await this.renderer.remove(data); } public setZoom(zoom: number = DEFAULT_ZOOM): void { - this.zoom = zoom; - } - - /** - * 用于在切换选择模式时清除上一次的状态 - * @param selectType 需要清理的选择模式 多选:multiSelect,单选:select - */ - public clearSelectStatus(selectType: String) { - if (selectType === 'multiSelect') { - this.multiDr.clearSelectStatus(); - this.selectedDomList = []; - } else { - this.dr.clearSelectStatus(); - } + this.renderer.setZoom(zoom); } /** @@ -336,40 +182,36 @@ export default class StageCore extends EventEmitter { */ public clearGuides() { this.mask.clearGuides(); - this.dr.clearGuides(); + this.actionManager.clearGuides(); } - public async addContainerHighlightClassName(event: MouseEvent, exclude: Element[]) { - const els = this.getElementsFromPoint(event); - const { renderer } = this; - const doc = renderer.getDocument(); - - if (!doc) return; - - for (const el of els) { - if (!el.id.startsWith(GHOST_EL_ID_PREFIX) && (await this.isContainer(el)) && !exclude.includes(el)) { - addClassName(el, doc, this.containerHighlightClassName); - break; - } - } + /** + * @deprecated 废弃接口,建议用delayedMarkContainer代替 + */ + public getAddContainerHighlightClassNameTimeout(event: MouseEvent, excludeElList: Element[] = []): NodeJS.Timeout { + return this.delayedMarkContainer(event, excludeElList); } - public getAddContainerHighlightClassNameTimeout(event: MouseEvent, exclude: Element[] = []): NodeJS.Timeout { - return globalThis.setTimeout(() => { - this.addContainerHighlightClassName(event, exclude); - }, this.containerHighlightDuration); + /** + * 鼠标拖拽着元素,在容器上方悬停,延迟一段时间后,对容器进行标记,如果悬停时间够长将标记成功,悬停时间短,调用方通过返回的timeoutId取消标记 + * 标记的作用:1、高亮容器,给用户一个加入容器的交互感知;2、释放鼠标后,通过标记的标志找到要加入的容器 + * @param event 鼠标事件 + * @param excludeElList 计算鼠标所在容器时要排除的元素列表 + * @returns timeoutId,调用方在鼠标移走时要取消该timeout,阻止标记 + */ + public delayedMarkContainer(event: MouseEvent, excludeElList: Element[] = []): NodeJS.Timeout { + return this.actionManager.delayedMarkContainer(event, excludeElList); } /** * 销毁实例 */ public destroy(): void { - const { mask, renderer, dr, highlightLayer, pageResizeObserver } = this; + const { mask, renderer, actionManager, pageResizeObserver } = this; renderer.destroy(); mask.destroy(); - dr.destroy(); - highlightLayer.destroy(); + actionManager.destroy(); pageResizeObserver?.disconnect(); this.removeAllListeners(); @@ -384,32 +226,117 @@ export default class StageCore extends EventEmitter { if (typeof ResizeObserver !== 'undefined') { this.pageResizeObserver = new ResizeObserver((entries) => { this.mask.pageResize(entries); - - if (this.dr.moveable) { - this.dr.updateMoveable(); - } + this.actionManager.updateMoveable(); }); this.pageResizeObserver.observe(page); } } - /** - * 传入stageRender供其回调,获取业务方自定义渲染画布页面渲染结果 - */ - private async render(): Promise { - if (this.config?.render) { - return await this.config.render(this); - } - return null; + private getActionManagerConfig(config: StageCoreConfig): ActionManagerConfig { + const actionManagerConfig: ActionManagerConfig = { + containerHighlightClassName: config.containerHighlightClassName, + containerHighlightDuration: config.containerHighlightDuration, + containerHighlightType: config.containerHighlightType, + moveableOptions: config.moveableOptions, + multiMoveableOptions: config.multiMoveableOptions, + container: this.mask.content, + canSelect: config.canSelect, + isContainer: config.isContainer, + updateDragEl: config.updateDragEl, + getRootContainer: () => this.container, + getRenderDocument: () => this.renderer.getDocument(), + getTargetElement: (idOrEl: Id | HTMLElement) => this.renderer.getTargetElement(idOrEl), + getElementsFromPoint: (point: Point) => this.renderer.getElementsFromPoint(point), + }; + + return actionManagerConfig; } - private async getTargetElement(idOrEl: Id | HTMLElement): Promise { - if (typeof idOrEl === 'string' || typeof idOrEl === 'number') { - const el = this.renderer.getDocument()?.getElementById(`${idOrEl}`); - if (!el) throw new Error(`不存在ID为${idOrEl}的元素`); - return el; - } - return idOrEl; + private initRenderEvent(): void { + this.renderer.on('runtime-ready', (runtime: Runtime) => { + this.emit('runtime-ready', runtime); + }); + this.renderer.on('page-el-update', (el: HTMLElement) => { + this.mask?.observe(el); + this.observePageResize(el); + }); + } + + private initMaskEvent(): void { + this.mask.on('change-guides', (data: GuidesEventData) => { + this.actionManager.setGuidelines(data.type, data.guides); + this.emit('change-guides', data); + }); + } + + /** + * 初始化操作相关事件监听 + */ + private initActionEvent(): void { + this.initActionManagerEvent(); + this.initDrEvent(); + this.initMulDrEvent(); + this.initHighlightEvent(); + } + + /** + * 初始化ActionManager类本身抛出来的事件监听 + */ + private initActionManagerEvent(): void { + this.actionManager + .on('before-select', (idOrEl: Id | HTMLElement, event?: MouseEvent) => { + this.select(idOrEl, event); + }) + .on('select', (selectedEl: HTMLElement) => { + this.emit('select', selectedEl); + }) + .on('before-multi-select', (idOrElList: HTMLElement[] | Id[]) => { + this.multiSelect(idOrElList); + }) + .on('multi-select', (selectedElList: HTMLElement[]) => { + this.emit('multi-select', selectedElList); + }); + } + + /** + * 初始化DragResize类通过ActionManager抛出来的事件监听 + */ + private initDrEvent(): void { + this.actionManager + .on('update', (data: UpdateEventData) => { + this.emit('update', data); + }) + .on('sort', (data: UpdateEventData) => { + this.emit('sort', data); + }) + .on('select-parent', () => { + this.emit('select-parent'); + }); + } + + /** + * 初始化MultiDragResize类通过ActionManager抛出来的事件监听 + */ + private initMulDrEvent(): void { + this.actionManager + // 多选切换到单选 + .on('change-to-select', (el: HTMLElement) => { + this.select(el); + // 先保证画布内完成渲染,再通知外部更新 + setTimeout(() => this.emit('select', el)); + }) + .on('multi-update', (data: UpdateEventData, parentEl: HTMLElement | null) => { + this.emit('update', { data, parentEl }); + }); + } + + /** + * 初始化Highlight类通过ActionManager抛出来的事件监听 + */ + private initHighlightEvent(): void { + this.actionManager.on('highlight', async (highlightEl: HTMLElement) => { + this.emit('highlight', highlightEl); + }); } } diff --git a/packages/stage/src/StageDragResize.ts b/packages/stage/src/StageDragResize.ts index ba0e1219..fb36fea3 100644 --- a/packages/stage/src/StageDragResize.ts +++ b/packages/stage/src/StageDragResize.ts @@ -17,79 +17,60 @@ */ /* eslint-disable no-param-reassign */ -import { EventEmitter } from 'events'; - -import KeyController from 'keycon'; -import type { MoveableOptions } from 'moveable'; -import Moveable from 'moveable'; +import Moveable, { MoveableOptions } from 'moveable'; import MoveableHelper from 'moveable-helper'; -import { removeClassNameByClassName } from '@tmagic/utils'; - -import { DRAG_EL_ID_PREFIX, GHOST_EL_ID_PREFIX, GuidesType, Mode, ZIndex } from './const'; -import selectParentAbles from './MoveableSelectParentAble'; -import StageCore from './StageCore'; -import StageMask from './StageMask'; -import type { StageDragResizeConfig } from './types'; -import { ContainerHighlightType, StageDragStatus } from './types'; -import { calcValueByFontsize, down, getAbsolutePosition, getMode, getOffset, getTargetElStyle, up } from './util'; +import { DRAG_EL_ID_PREFIX, GHOST_EL_ID_PREFIX, Mode, ZIndex } from './const'; +import MoveableOptionsManager from './MoveableOptionsManager'; +import TargetShadow from './TargetShadow'; +import type { DelayedMarkContainer, GetRenderDocument, MarkContainerEnd, StageDragResizeConfig } from './types'; +import { StageDragStatus } from './types'; +import { calcValueByFontsize, down, getAbsolutePosition, getMode, getOffset, up } from './util'; /** - * 选中框 + * 管理单选操作,响应选中操作,初始化moveableOption参数并初始化moveable,处理moveable回调事件对组件进行更新 + * @extends MoveableOptionsManager */ -export default class StageDragResize extends EventEmitter { - public core: StageCore; - public mask: StageMask; - /** 画布容器 */ - public container: HTMLElement; +export default class StageDragResize extends MoveableOptionsManager { /** 目标节点 */ - public target?: HTMLElement; + private target?: HTMLElement; /** 目标节点在蒙层中的占位节点 */ - public dragEl?: HTMLDivElement; + private targetShadow: TargetShadow; /** Moveable拖拽类实例 */ - public moveable?: Moveable; - /** 水平参考线 */ - public horizontalGuidelines: number[] = []; - /** 垂直参考线 */ - public verticalGuidelines: number[] = []; - /** 对齐元素集合 */ - public elementGuidelines: HTMLElement[] = []; - /** 布局方式:流式布局、绝对定位、固定定位 */ - public mode: Mode = Mode.ABSOLUTE; - - private moveableOptions: MoveableOptions = {}; + private moveable?: Moveable; /** 拖动状态 */ private dragStatus: StageDragStatus = StageDragStatus.END; /** 流式布局下,目标节点的镜像节点 */ private ghostEl: HTMLElement | undefined; private moveableHelper?: MoveableHelper; - private isContainerHighlight: Boolean = false; + private getRenderDocument: GetRenderDocument; + private markContainerEnd: MarkContainerEnd; + private delayedMarkContainer: DelayedMarkContainer; constructor(config: StageDragResizeConfig) { - super(); + super(config); - this.core = config.core; - this.container = config.container; - this.mask = config.mask; + this.getRenderDocument = config.getRenderDocument; + this.markContainerEnd = config.markContainerEnd; + this.delayedMarkContainer = config.delayedMarkContainer; - KeyController.global.keydown('alt', (e) => { - e.inputEvent.preventDefault(); - this.isContainerHighlight = true; + this.targetShadow = new TargetShadow({ + container: config.container, + updateDragEl: config.updateDragEl, + zIndex: ZIndex.DRAG_EL, + idPrefix: DRAG_EL_ID_PREFIX, }); - KeyController.global.keyup('alt', (e) => { - e.inputEvent.preventDefault(); - const doc = this.core.renderer.contentWindow?.document; - if (doc && this.canContainerHighlight()) { - removeClassNameByClassName(doc, this.core.containerHighlightClassName); + this.on('update-moveable', () => { + if (this.moveable) { + this.updateMoveable(); } - this.isContainerHighlight = false; }); } /** * 将选中框渲染并覆盖到选中的组件Dom节点上方 - * 当选中的节点是不是absolute时,会创建一个新的节点出来作为拖拽目标 + * 当选中的节点不是absolute时,会创建一个新的节点出来作为拖拽目标 * @param el 选中组件的Dom节点元素 * @param event 鼠标事件 */ @@ -97,23 +78,11 @@ export default class StageDragResize extends EventEmitter { const oldTarget = this.target; this.target = el; - if (!this.dragEl) { - this.dragEl = globalThis.document.createElement('div'); - this.container.append(this.dragEl); - } - // 从不能拖动到能拖动的节点之间切换,要重新创建moveable,不然dragStart不生效 if (!this.moveable || this.target !== oldTarget) { - this.init(el); - this.moveableHelper = MoveableHelper.create({ - useBeforeRender: true, - useRender: false, - createAuto: true, - }); - - this.initMoveable(); + this.initMoveable(el); } else { - this.updateMoveable(); + this.updateMoveable(el); } if (event) { @@ -125,120 +94,70 @@ export default class StageDragResize extends EventEmitter { * 初始化选中框并渲染出来 */ public updateMoveable(el = this.target): void { - if (!this.moveable) throw new Error('未初始化moveable'); + if (!this.moveable) return; if (!el) throw new Error('未选中任何节点'); this.target = el; - this.init(el); + const options: MoveableOptions = this.init(el); - Object.entries(this.moveableOptions).forEach(([key, value]) => { + Object.entries(options).forEach(([key, value]) => { (this.moveable as any)[key] = value; }); this.moveable.updateTarget(); } - public setGuidelines(type: GuidesType, guidelines: number[]): void { - if (type === GuidesType.HORIZONTAL) { - this.horizontalGuidelines = guidelines; - this.moveableOptions.horizontalGuidelines = guidelines; - } else if (type === GuidesType.VERTICAL) { - this.verticalGuidelines = guidelines; - this.moveableOptions.verticalGuidelines = guidelines; - } - - if (this.moveable) { - this.updateMoveable(); - } - } - - public clearGuides() { - this.horizontalGuidelines = []; - this.verticalGuidelines = []; - this.moveableOptions.horizontalGuidelines = []; - this.moveableOptions.verticalGuidelines = []; - this.updateMoveable(); - } - public clearSelectStatus(): void { if (!this.moveable) return; - this.destroyDragEl(); - this.dragEl = undefined; + this.targetShadow.destroyEl(); this.moveable.target = null; this.moveable.updateTarget(); } - public destroyDragEl(): void { - this.dragEl?.remove(); - } - /** * 销毁实例 */ public destroy(): void { this.moveable?.destroy(); this.destroyGhostEl(); - this.destroyDragEl(); + this.targetShadow.destroy(); this.dragStatus = StageDragStatus.END; this.removeAllListeners(); } - private init(el: HTMLElement): void { + private init(el: HTMLElement): MoveableOptions { // 如果有滚动条会导致resize时获取到width,height不准确 if (/(auto|scroll)/.test(el.style.overflow)) { el.style.overflow = 'hidden'; } + this.mode = getMode(el); this.destroyGhostEl(); - if (!this.dragEl) { - return; - } + this.targetShadow.update(el); - this.dragEl.style.cssText = getTargetElStyle(el); - this.dragEl.id = `${DRAG_EL_ID_PREFIX}${el.id}`; + // 设置选中元素的周围元素,用于选中元素跟周围元素对齐辅助 + const elementGuidelines: any = this.target?.parentElement?.children || []; + this.setElementGuidelines([this.target as HTMLElement], elementGuidelines); - if (typeof this.core.config.updateDragEl === 'function') { - this.core.config.updateDragEl(this.dragEl, el); - } - this.moveableOptions = this.getOptions({ - target: this.dragEl, + return this.getOptions(false, { + target: this.targetShadow.el, }); } - private setElementGuidelines(nodes: HTMLElement[]) { - this.elementGuidelines.forEach((node) => { - node.remove(); + private initMoveable(el: HTMLElement) { + const options: MoveableOptions = this.init(el); + this.moveableHelper = MoveableHelper.create({ + useBeforeRender: true, + useRender: false, + createAuto: true, }); - this.elementGuidelines = []; - if (this.mode === Mode.ABSOLUTE) { - this.container.append(this.createGuidelineElements(nodes)); - } - } - - private createGuidelineElements(nodes: HTMLElement[]) { - const frame = globalThis.document.createDocumentFragment(); - - for (const node of nodes) { - const { width, height } = node.getBoundingClientRect(); - if (node === this.target) continue; - const { left, top } = getOffset(node as HTMLElement); - const elementGuideline = globalThis.document.createElement('div'); - elementGuideline.style.cssText = `position: absolute;width: ${width}px;height: ${height}px;top: ${top}px;left: ${left}px`; - this.elementGuidelines.push(elementGuideline); - frame.append(elementGuideline); - } - - return frame; - } - - private initMoveable() { this.moveable?.destroy(); this.moveable = new Moveable(this.container, { - ...this.moveableOptions, + ...options, }); this.bindResizeEvent(); @@ -267,7 +186,7 @@ export default class StageDragResize extends EventEmitter { }) .on('resize', (e) => { const { width, height, drag } = e; - if (!this.moveable || !this.target || !this.dragEl) return; + if (!this.moveable || !this.target || !this.targetShadow.el) return; const { beforeTranslate } = drag; this.dragStatus = StageDragStatus.ING; @@ -275,8 +194,8 @@ export default class StageDragResize extends EventEmitter { // 流式布局 if (this.mode === Mode.SORTABLE) { this.target.style.top = '0px'; - this.dragEl.style.width = `${width}px`; - this.dragEl.style.height = `${height}px`; + this.targetShadow.el.style.width = `${width}px`; + this.targetShadow.el.style.height = `${height}px`; } else { this.moveableHelper?.onResize(e); this.target.style.left = `${frame.left + beforeTranslate[0]}px`; @@ -302,9 +221,6 @@ export default class StageDragResize extends EventEmitter { let timeout: NodeJS.Timeout | undefined; - const { contentWindow } = this.core.renderer; - const doc = contentWindow?.document; - this.moveable .on('dragStart', (e) => { if (!this.target) throw new Error('未选中组件'); @@ -321,16 +237,13 @@ export default class StageDragResize extends EventEmitter { frame.left = this.target.offsetLeft; }) .on('drag', (e) => { - if (!this.target || !this.dragEl) return; + if (!this.target || !this.targetShadow.el) return; if (timeout) { globalThis.clearTimeout(timeout); timeout = undefined; } - - if (this.canContainerHighlight()) { - timeout = this.core.getAddContainerHighlightClassNameTimeout(e.inputEvent, [this.target]); - } + timeout = this.delayedMarkContainer(e.inputEvent, [this.target]); this.dragStatus = StageDragStatus.ING; @@ -351,12 +264,7 @@ export default class StageDragResize extends EventEmitter { timeout = undefined; } - let parentEl: HTMLElement | null = null; - - if (doc && this.canContainerHighlight()) { - parentEl = removeClassNameByClassName(doc, this.core.containerHighlightClassName); - } - + const parentEl = this.markContainerEnd(); // 点击不拖动时会触发dragStart和dragEnd,但是不会有drag事件 if (this.dragStatus === StageDragStatus.ING) { if (parentEl) { @@ -386,7 +294,7 @@ export default class StageDragResize extends EventEmitter { this.moveableHelper?.onRotateStart(e); }) .on('rotate', (e) => { - if (!this.target || !this.dragEl) return; + if (!this.target || !this.targetShadow.el) return; this.dragStatus = StageDragStatus.ING; this.moveableHelper?.onRotate(e); const frame = this.moveableHelper?.getFrame(e.target); @@ -417,7 +325,7 @@ export default class StageDragResize extends EventEmitter { this.moveableHelper?.onScaleStart(e); }) .on('scale', (e) => { - if (!this.target || !this.dragEl) return; + if (!this.target || !this.targetShadow.el) return; this.dragStatus = StageDragStatus.ING; this.moveableHelper?.onScale(e); const frame = this.moveableHelper?.getFrame(e.target); @@ -461,8 +369,7 @@ export default class StageDragResize extends EventEmitter { private update(isResize = false, parentEl: HTMLElement | null = null): void { if (!this.target) return; - const { contentWindow } = this.core.renderer; - const doc = contentWindow?.document; + const doc = this.getRenderDocument(); if (!doc) return; @@ -474,15 +381,24 @@ export default class StageDragResize extends EventEmitter { const width = calcValueByFontsize(doc, this.target.clientWidth); const height = calcValueByFontsize(doc, this.target.clientHeight); - if (parentEl && this.mode === Mode.ABSOLUTE && this.dragEl) { - const [translateX, translateY] = this.moveableHelper?.getFrame(this.dragEl).properties.transform.translate.value; + if (parentEl && this.mode === Mode.ABSOLUTE && this.targetShadow.el) { + const targetShadowHtmlEl = this.targetShadow.el as HTMLElement; + const targetShadowElOffsetLeft = targetShadowHtmlEl.offsetLeft || 0; + const targetShadowElOffsetTop = targetShadowHtmlEl.offsetTop || 0; + + const frame = this.moveableHelper?.getFrame(this.targetShadow.el); + + const [translateX, translateY] = frame?.properties.transform.translate.value; const { left: parentLeft, top: parentTop } = getOffset(parentEl); + left = - calcValueByFontsize(doc, this.dragEl.offsetLeft) + + calcValueByFontsize(doc, targetShadowElOffsetLeft) + parseFloat(translateX) - calcValueByFontsize(doc, parentLeft); top = - calcValueByFontsize(doc, this.dragEl.offsetTop) + parseFloat(translateY) - calcValueByFontsize(doc, parentTop); + calcValueByFontsize(doc, targetShadowElOffsetTop) + + parseFloat(translateY) - + calcValueByFontsize(doc, parentTop); } this.emit('update', { @@ -530,86 +446,4 @@ export default class StageDragResize extends EventEmitter { this.ghostEl?.remove(); this.ghostEl = undefined; } - - private getOptions(options: MoveableOptions = {}): MoveableOptions { - if (!this.target) return {}; - - const isAbsolute = this.mode === Mode.ABSOLUTE; - const isFixed = this.mode === Mode.FIXED; - const isSortable = this.mode === Mode.SORTABLE; - - let { moveableOptions = {} } = this.core.config; - - if (typeof moveableOptions === 'function') { - moveableOptions = moveableOptions(this.core); - } - - const elementGuidelines: any = moveableOptions.elementGuidelines || this.target.parentElement?.children || []; - - this.setElementGuidelines(elementGuidelines); - - if (moveableOptions.elementGuidelines) { - delete moveableOptions.elementGuidelines; - } - - return { - origin: false, - rootContainer: this.core.container, - zoom: 1, - dragArea: false, - draggable: true, - resizable: true, - scalable: false, - rotatable: false, - snappable: true, - snapGap: isAbsolute || isFixed, - snapThreshold: 5, - snapDigit: 0, - throttleDrag: 0, - isDisplaySnapDigit: isAbsolute, - snapDirections: { - top: isAbsolute, - right: isAbsolute, - bottom: isAbsolute, - left: isAbsolute, - center: isAbsolute, - middle: isAbsolute, - }, - elementSnapDirections: { - top: isAbsolute, - right: isAbsolute, - bottom: isAbsolute, - left: isAbsolute, - }, - isDisplayInnerSnapDigit: true, - horizontalGuidelines: this.horizontalGuidelines, - verticalGuidelines: this.verticalGuidelines, - elementGuidelines: this.elementGuidelines, - - bounds: { - top: 0, - // 设置0的话无法移动到left为0,所以只能设置为-1 - left: -1, - right: this.container.clientWidth - 1, - bottom: isSortable ? undefined : this.container.clientHeight, - ...(moveableOptions.bounds || {}), - }, - - props: { - selectParent: true, - }, - - ables: [selectParentAbles(this)], - - ...options, - ...moveableOptions, - }; - } - - private canContainerHighlight() { - return ( - this.core.containerHighlightType === ContainerHighlightType.DEFAULT || - (this.core.containerHighlightType === ContainerHighlightType.ALT && this.isContainerHighlight) - ); - } } diff --git a/packages/stage/src/StageHighlight.ts b/packages/stage/src/StageHighlight.ts index 1426eeea..cafadea4 100644 --- a/packages/stage/src/StageHighlight.ts +++ b/packages/stage/src/StageHighlight.ts @@ -20,27 +20,28 @@ import { EventEmitter } from 'events'; import Moveable from 'moveable'; -import { HIGHLIGHT_EL_ID_PREFIX } from './const'; -import StageCore from './StageCore'; -import TargetCalibrate from './TargetCalibrate'; -import type { StageHighlightConfig } from './types'; +import { HIGHLIGHT_EL_ID_PREFIX, ZIndex } from './const'; +import TargetShadow from './TargetShadow'; +import type { GetRootContainer, StageHighlightConfig } from './types'; + export default class StageHighlight extends EventEmitter { - public core: StageCore; public container: HTMLElement; public target?: HTMLElement; public moveable?: Moveable; - public calibrationTarget: TargetCalibrate; + public targetShadow: TargetShadow; + private getRootContainer: GetRootContainer; constructor(config: StageHighlightConfig) { super(); - this.core = config.core; this.container = config.container; - this.calibrationTarget = new TargetCalibrate({ - parent: this.core.mask.content, - mask: this.core.mask, - dr: this.core.dr, - core: this.core, + this.getRootContainer = config.getRootContainer; + + this.targetShadow = new TargetShadow({ + container: config.container, + updateDragEl: config.updateDragEl, + zIndex: ZIndex.HIGHLIGHT_EL, + idPrefix: HIGHLIGHT_EL_ID_PREFIX, }); } @@ -54,9 +55,9 @@ export default class StageHighlight extends EventEmitter { this.moveable?.destroy(); this.moveable = new Moveable(this.container, { - target: this.calibrationTarget.update(el, HIGHLIGHT_EL_ID_PREFIX), + target: this.targetShadow.update(el), origin: false, - rootContainer: this.core.container, + rootContainer: this.getRootContainer(), zoom: 2, }); } @@ -65,7 +66,7 @@ export default class StageHighlight extends EventEmitter { * 清空高亮 */ public clearHighlight(): void { - if (!this.moveable) return; + if (!this.moveable || !this.target) return; this.target = undefined; this.moveable.target = null; this.moveable.updateTarget(); @@ -76,6 +77,6 @@ export default class StageHighlight extends EventEmitter { */ public destroy(): void { this.moveable?.destroy(); - this.calibrationTarget.destroy(); + this.targetShadow.destroy(); } } diff --git a/packages/stage/src/StageMask.ts b/packages/stage/src/StageMask.ts index 04bfd9a1..2dc88155 100644 --- a/packages/stage/src/StageMask.ts +++ b/packages/stage/src/StageMask.ts @@ -16,20 +16,16 @@ * limitations under the License. */ -import KeyController from 'keycon'; -import { throttle } from 'lodash-es'; +import { createDiv, getDocument, injectStyle } from '@tmagic/utils'; -import { createDiv, injectStyle } from '@tmagic/utils'; - -import { Mode, MouseButton, ZIndex } from './const'; +import { Mode, ZIndex } from './const'; import Rule from './Rule'; -import { getScrollParent, isFixedParent, isMoveableButton } from './util'; +import { getScrollParent, isFixedParent } from './util'; const wrapperClassName = 'editor-mask-wrapper'; -const throttleTime = 100; const hideScrollbar = () => { - injectStyle(globalThis.document, `.${wrapperClassName}::-webkit-scrollbar { width: 0 !important; display: none }`); + injectStyle(getDocument(), `.${wrapperClassName}::-webkit-scrollbar { width: 0 !important; display: none }`); }; const createContent = (): HTMLDivElement => @@ -78,36 +74,28 @@ export default class StageMask extends Rule { public wrapperWidth = 0; public maxScrollTop = 0; public maxScrollLeft = 0; - public isMultiSelectStatus: Boolean = false; private mode: Mode = Mode.ABSOLUTE; private pageScrollParent: HTMLElement | null = null; private intersectionObserver: IntersectionObserver | null = null; private wrapperResizeObserver: ResizeObserver | null = null; - /** - * 高亮事件处理函数 - * @param event 事件对象 - */ - private highlightHandler = throttle((event: MouseEvent): void => { - this.emit('highlight', event); - }, throttleTime); - constructor() { const wrapper = createWrapper(); super(wrapper); this.wrapper = wrapper; - this.initContentEventListener(); + this.content.addEventListener('wheel', this.mouseWheelHandler); this.wrapper.appendChild(this.content); - - this.initMultiSelectEvent(); } public setMode(mode: Mode) { this.mode = mode; this.scroll(); + + this.content.dataset.mode = mode; + if (mode === Mode.FIXED) { this.content.style.width = `${this.wrapperWidth}px`; this.content.style.height = `${this.wrapperHeight}px`; @@ -182,42 +170,9 @@ export default class StageMask extends Rule { this.pageScrollParent = null; this.wrapperResizeObserver?.disconnect(); - this.content.removeEventListener('mouseleave', this.mouseLeaveHandler); super.destroy(); } - /** - * 初始化content的事件监听 - */ - private initContentEventListener(): void { - this.content.addEventListener('mousedown', this.mouseDownHandler); - this.content.addEventListener('wheel', this.mouseWheelHandler); - this.content.addEventListener('mousemove', this.highlightHandler); - this.content.addEventListener('mouseleave', this.mouseLeaveHandler); - } - - /** - * 初始化多选事件监听 - */ - private initMultiSelectEvent(): void { - const isMac = /mac os x/.test(navigator.userAgent.toLowerCase()); - - const ctrl = isMac ? 'meta' : 'ctrl'; - - KeyController.global.keydown(ctrl, (e) => { - e.inputEvent.preventDefault(); - this.isMultiSelectStatus = true; - }); - // ctrl+tab切到其他窗口,需要将多选状态置为false - KeyController.global.on('blur', () => { - this.isMultiSelectStatus = false; - }); - KeyController.global.keyup(ctrl, (e) => { - e.inputEvent.preventDefault(); - this.isMultiSelectStatus = false; - }); - } - /** * 监听选中元素是否在画布可视区域内,如果目标元素不在可视区域内,通过滚动使该元素出现在可视区域 */ @@ -286,6 +241,17 @@ export default class StageMask extends Rule { private scrollTo(scrollLeft: number, scrollTop: number): void { this.content.style.transform = `translate3d(${-scrollLeft}px, ${-scrollTop}px, 0)`; + + const event = new CustomEvent<{ + scrollLeft: number; + scrollTop: number; + }>('customScroll', { + detail: { + scrollLeft: this.scrollLeft, + scrollTop: this.scrollTop, + }, + }); + this.content.dispatchEvent(event); } /** @@ -333,51 +299,7 @@ export default class StageMask extends Rule { if (this.maxScrollLeft < this.scrollLeft) this.scrollLeft = this.maxScrollLeft; } - /** - * 点击事件处理函数 - * @param event 事件对象 - */ - private mouseDownHandler = (event: MouseEvent): void => { - this.emit('clearHighlight'); - event.stopImmediatePropagation(); - event.stopPropagation(); - - if (event.button !== MouseButton.LEFT && event.button !== MouseButton.RIGHT) return; - if (!event.target) return; - - const targetClassList = (event.target as HTMLDivElement).classList; - - // 如果单击多选选中区域,则不需要再触发选中了,而可能是拖动行为 - if (!this.isMultiSelectStatus && targetClassList.contains('moveable-area')) { - return; - } - // 点击对象如果是边框锚点,则可能是resize; 点击对象是功能按钮 - if (targetClassList.contains('moveable-control') || isMoveableButton(event.target as Element)) { - return; - } - - this.content.removeEventListener('mousemove', this.highlightHandler); - - // 判断触发多选还是单选 - if (this.isMultiSelectStatus) { - this.emit('beforeMultiSelect', event); - } else { - this.emit('beforeSelect', event); - } - // 如果是右键点击,这里的mouseup事件监听没有效果 - globalThis.document.addEventListener('mouseup', this.mouseUpHandler); - }; - - private mouseUpHandler = (): void => { - globalThis.document.removeEventListener('mouseup', this.mouseUpHandler); - this.content.addEventListener('mousemove', this.highlightHandler); - if (!this.isMultiSelectStatus) { - this.emit('select'); - } - }; - private mouseWheelHandler = (event: WheelEvent) => { - this.emit('clearHighlight'); if (!this.page) throw new Error('page 未初始化'); const { deltaY, deltaX } = event; @@ -394,11 +316,6 @@ export default class StageMask extends Rule { } this.scroll(); - this.emit('scroll', event); }; - - private mouseLeaveHandler = () => { - setTimeout(() => this.emit('clearHighlight'), throttleTime); - }; } diff --git a/packages/stage/src/StageMultiDragResize.ts b/packages/stage/src/StageMultiDragResize.ts index 6a3f29e9..4c6038aa 100644 --- a/packages/stage/src/StageMultiDragResize.ts +++ b/packages/stage/src/StageMultiDragResize.ts @@ -16,39 +16,53 @@ * limitations under the License. */ -import { EventEmitter } from 'events'; - -import type { MoveableOptions, OnDragStart, OnResizeStart } from 'moveable'; +import type { OnDragStart, OnResizeStart } from 'moveable'; import Moveable from 'moveable'; import MoveableHelper from 'moveable-helper'; -import { DRAG_EL_ID_PREFIX, PAGE_CLASS } from './const'; -import StageCore from './StageCore'; -import StageMask from './StageMask'; -import { StageDragResizeConfig, StageDragStatus } from './types'; -import { calcValueByFontsize, getMode, getTargetElStyle } from './util'; +import { DRAG_EL_ID_PREFIX, Mode, ZIndex } from './const'; +import MoveableOptionsManager from './MoveableOptionsManager'; +import TargetShadow from './TargetShadow'; +import { GetRenderDocument, MoveableOptionsManagerConfig, StageDragStatus, StageMultiDragResizeConfig } from './types'; +import { calcValueByFontsize, getMode } from './util'; -export default class StageMultiDragResize extends EventEmitter { - public core: StageCore; - public mask: StageMask; +export default class StageMultiDragResize extends MoveableOptionsManager { /** 画布容器 */ public container: HTMLElement; /** 多选:目标节点组 */ public targetList: HTMLElement[] = []; /** 多选:目标节点在蒙层中的占位节点组 */ - public dragElList: HTMLDivElement[] = []; + public targetShadow: TargetShadow; /** Moveable多选拖拽类实例 */ public moveableForMulti?: Moveable; /** 拖动状态 */ public dragStatus: StageDragStatus = StageDragStatus.END; private multiMoveableHelper?: MoveableHelper; + private getRenderDocument: GetRenderDocument; - constructor(config: StageDragResizeConfig) { - super(); + constructor(config: StageMultiDragResizeConfig) { + const moveableOptionsManagerConfig: MoveableOptionsManagerConfig = { + container: config.container, + moveableOptions: config.multiMoveableOptions, + getRootContainer: config.getRootContainer, + }; + super(moveableOptionsManagerConfig); - this.core = config.core; this.container = config.container; - this.mask = config.mask; + this.getRenderDocument = config.getRenderDocument; + + this.targetShadow = new TargetShadow({ + container: config.container, + updateDragEl: config.updateDragEl, + zIndex: ZIndex.DRAG_EL, + idPrefix: DRAG_EL_ID_PREFIX, + }); + + this.on('update-moveable', () => { + if (this.moveableForMulti) { + this.updateMoveable(); + } + }); } /** @@ -56,27 +70,24 @@ export default class StageMultiDragResize extends EventEmitter { * @param els */ public multiSelect(els: HTMLElement[]): void { + if (els.length === 0) { + return; + } + this.mode = getMode(els[0]); this.targetList = els; - this.core.dr.destroyDragEl(); - this.destroyDragElList(); - // 生成虚拟多选节点 - this.dragElList = els.map((elItem) => { - const dragElDiv = globalThis.document.createElement('div'); - this.container.append(dragElDiv); - dragElDiv.style.cssText = getTargetElStyle(elItem); - dragElDiv.id = `${DRAG_EL_ID_PREFIX}${elItem.id}`; - // 业务方校准 - if (typeof this.core.config.updateDragEl === 'function') { - this.core.config.updateDragEl(dragElDiv, elItem); - } - return dragElDiv; - }); + + this.targetShadow.updateGroup(els); + + // 设置周围元素,用于选中元素跟周围元素的对齐辅助 + const elementGuidelines: any = this.targetList[0].parentElement?.children || []; + this.setElementGuidelines(this.targetList, elementGuidelines); + this.moveableForMulti?.destroy(); this.multiMoveableHelper?.clear(); this.moveableForMulti = new Moveable( this.container, - this.getOptions({ - target: this.dragElList, + this.getOptions(true, { + target: this.targetShadow.els, }), ); this.multiMoveableHelper = MoveableHelper.create({ @@ -177,47 +188,58 @@ export default class StageMultiDragResize extends EventEmitter { }) .on('clickGroup', (params) => { const { inputTarget, targets } = params; - // 如果此时mask不处于多选状态下,且有多个元素被选中,同时点击的元素在选中元素中的其中一项,代表多选态切换为该元素的单选态 - if (!this.mask.isMultiSelectStatus && targets.length > 1 && targets.includes(inputTarget)) { - this.emit('select', inputTarget.id.replace(DRAG_EL_ID_PREFIX, '')); + // 如果有多个元素被选中,同时点击的元素在选中元素中的其中一项,可能是多选态切换为该元素的单选态,抛事件给上一层继续判断是否切换 + if (targets.length > 1 && targets.includes(inputTarget)) { + this.emit('change-to-select', inputTarget.id.replace(DRAG_EL_ID_PREFIX, '')); } }); } - public canSelect(el: HTMLElement, stop: () => boolean): Boolean { - // 多选状态下不可以选中magic-ui-page,并停止继续向上层选中 - if (el.className.includes(PAGE_CLASS)) { - this.core.highlightedDom = undefined; - this.core.highlightLayer.clearHighlight(); - stop(); + public canSelect(el: HTMLElement, selectedEl: HTMLElement | undefined): boolean { + const currentTargetMode = getMode(el); + let selectedElMode = ''; + + // 流式布局不支持多选 + if (currentTargetMode === Mode.SORTABLE) { return false; } - const currentTargetMode = getMode(el); - let selectedDomMode = ''; - if (this.core.selectedDom?.className.includes(PAGE_CLASS)) { - // 先单击选中了页面(magic-ui-page),再按住多选键多选时,任一元素均可选中 - return true; - } - if (this.targetList.length === 0 && this.core.selectedDom) { + + if (this.targetList.length === 0 && selectedEl) { // 单选后添加到多选的情况 - selectedDomMode = getMode(this.core.selectedDom); + selectedElMode = getMode(selectedEl); } else if (this.targetList.length > 0) { // 已加入多选列表的布局模式是一样的,取第一个判断 - selectedDomMode = getMode(this.targetList[0]); + selectedElMode = getMode(this.targetList[0]); } // 定位模式不同,不可混选 - if (currentTargetMode !== selectedDomMode) { + if (currentTargetMode !== selectedElMode) { return false; } return true; } + public updateMoveable(eleList = this.targetList) { + if (!this.moveableForMulti) return; + if (!eleList) throw new Error('未选中任何节点'); + + this.targetList = eleList; + + const options = this.getOptions(true, { + target: this.targetShadow.els, + }); + + Object.entries(options).forEach(([key, value]) => { + (this.moveableForMulti as any)[key] = value; + }); + this.moveableForMulti.updateTarget(); + } + /** * 清除多选状态 */ public clearSelectStatus(): void { if (!this.moveableForMulti) return; - this.destroyDragElList(); + this.targetShadow.destroyEls(); this.moveableForMulti.target = null; this.moveableForMulti.updateTarget(); this.targetList = []; @@ -228,14 +250,7 @@ export default class StageMultiDragResize extends EventEmitter { */ public destroy(): void { this.moveableForMulti?.destroy(); - this.destroyDragElList(); - } - - /** - * 清除蒙层占位节点 - */ - public destroyDragElList(): void { - this.dragElList.forEach((dragElItem) => dragElItem?.remove()); + this.targetShadow.destroy(); } /** @@ -245,60 +260,19 @@ export default class StageMultiDragResize extends EventEmitter { private update(isResize = false): void { if (this.targetList.length === 0) return; - const { contentWindow } = this.core.renderer; - const doc = contentWindow?.document; + const doc = this.getRenderDocument(); if (!doc) return; - this.emit('update', { - data: this.targetList.map((targetItem) => { - const offset = { left: targetItem.offsetLeft, top: targetItem.offsetTop }; - const left = calcValueByFontsize(doc, offset.left); - const top = calcValueByFontsize(doc, offset.top); - const width = calcValueByFontsize(doc, targetItem.clientWidth); - const height = calcValueByFontsize(doc, targetItem.clientHeight); - return { - el: targetItem, - style: isResize ? { left, top, width, height } : { left, top }, - }; - }), - parentEl: null, + const data = this.targetList.map((targetItem) => { + const left = calcValueByFontsize(doc, targetItem.offsetLeft); + const top = calcValueByFontsize(doc, targetItem.offsetTop); + const width = calcValueByFontsize(doc, targetItem.clientWidth); + const height = calcValueByFontsize(doc, targetItem.clientHeight); + return { + el: targetItem, + style: isResize ? { left, top, width, height } : { left, top }, + }; }); - } - - /** - * 获取moveable options参数 - * @param {MoveableOptions} options - * @return {MoveableOptions} moveable options参数 - */ - private getOptions(options: MoveableOptions = {}): MoveableOptions { - let { multiMoveableOptions = {} } = this.core.config; - - if (typeof multiMoveableOptions === 'function') { - multiMoveableOptions = multiMoveableOptions(this.core); - } - - return { - defaultGroupRotate: 0, - defaultGroupOrigin: '50% 50%', - draggable: true, - resizable: true, - throttleDrag: 0, - startDragRotate: 0, - throttleDragRotate: 0, - zoom: 1, - origin: true, - padding: { left: 0, top: 0, right: 0, bottom: 0 }, - snappable: true, - bounds: { - top: 0, - // 设置0的话无法移动到left为0,所以只能设置为-1 - left: -1, - right: this.container.clientWidth - 1, - bottom: this.container.clientHeight, - ...(multiMoveableOptions.bounds || {}), - }, - ...options, - ...multiMoveableOptions, - }; + this.emit('update', data, null); } } diff --git a/packages/stage/src/StageRender.ts b/packages/stage/src/StageRender.ts index dc613ff4..2963e88c 100644 --- a/packages/stage/src/StageRender.ts +++ b/packages/stage/src/StageRender.ts @@ -18,28 +18,30 @@ import { EventEmitter } from 'events'; +import { Id } from '@tmagic/schema'; import { getHost, injectStyle, isSameDomain } from '@tmagic/utils'; +import { DEFAULT_ZOOM } from './const'; import style from './style.css?raw'; -import type { Runtime, RuntimeWindow, StageRenderConfig } from './types'; +import type { Point, RemoveData, Runtime, RuntimeWindow, StageRenderConfig, UpdateData } from './types'; +import { addSelectedClassName, removeSelectedClassName } from './util'; export default class StageRender extends EventEmitter { /** 组件的js、css执行的环境,直接渲染为当前window,iframe渲染则为iframe.contentWindow */ public contentWindow: RuntimeWindow | null = null; - public runtime: Runtime | null = null; - public iframe?: HTMLIFrameElement; - public runtimeUrl?: string; + private runtimeUrl?: string; + private zoom = DEFAULT_ZOOM; + private customizedRender?: () => Promise; - private render?: () => Promise; - - constructor({ runtimeUrl, render }: StageRenderConfig) { + constructor({ runtimeUrl, zoom, customizedRender }: StageRenderConfig) { super(); this.runtimeUrl = runtimeUrl || ''; - this.render = render; + this.customizedRender = customizedRender; + this.setZoom(zoom); this.iframe = globalThis.document.createElement('iframe'); // 同源,直接加载 @@ -63,6 +65,38 @@ export default class StageRender extends EventEmitter { }, }); + public async add(data: UpdateData): Promise { + const runtime = await this.getRuntime(); + return runtime?.add?.(data); + } + + public async remove(data: RemoveData): Promise { + const runtime = await this.getRuntime(); + return runtime?.remove?.(data); + } + + public async update(data: UpdateData): Promise { + const runtime = await this.getRuntime(); + // 更新画布中的组件 + runtime?.update?.(data); + } + + public async select(els: HTMLElement[]): Promise { + const runtime = await this.getRuntime(); + + for (const el of els) { + await runtime?.select?.(el.id); + if (runtime?.beforeSelect) { + await runtime.beforeSelect(el); + } + this.flagSelectedEl(el); + } + } + + public setZoom(zoom: number = DEFAULT_ZOOM): void { + this.zoom = zoom; + } + /** * 挂载Dom节点 * @param el 将页面挂载到该Dom节点上 @@ -101,6 +135,35 @@ export default class StageRender extends EventEmitter { return this.contentWindow?.document; } + /** + * 通过坐标获得坐标下所有HTML元素数组 + * @param point 坐标 + * @returns 坐标下方所有HTML元素数组,会包含父元素直至html,元素层叠时返回顺序是从上到下 + */ + public getElementsFromPoint(point: Point): HTMLElement[] { + let x = point.clientX; + let y = point.clientY; + + if (this.iframe) { + const rect = this.iframe.getClientRects()[0]; + if (rect) { + x = x - rect.left; + y = y - rect.top; + } + } + + return this.getDocument()?.elementsFromPoint(x / this.zoom, y / this.zoom) as HTMLElement[]; + } + + public getTargetElement(idOrEl: Id | HTMLElement): HTMLElement { + if (typeof idOrEl === 'string' || typeof idOrEl === 'number') { + const el = this.getDocument()?.getElementById(`${idOrEl}`); + if (!el) throw new Error(`不存在ID为${idOrEl}的元素`); + return el; + } + return idOrEl; + } + /** * 销毁实例 */ @@ -112,6 +175,18 @@ export default class StageRender extends EventEmitter { this.removeAllListeners(); } + /** + * 在runtime中对被选中的元素进行标记,部分组件有对选中态进行特殊显示的需求 + * @param el 被选中的元素 + */ + private flagSelectedEl(el: HTMLElement): void { + const doc = this.getDocument(); + if (doc) { + removeSelectedClassName(doc); + addSelectedClassName(el, doc); + } + } + private loadHandler = async () => { if (!this.contentWindow?.magic) { this.postTmagicRuntimeReady(); @@ -119,8 +194,8 @@ export default class StageRender extends EventEmitter { if (!this.contentWindow) return; - if (this.render) { - const el = await this.render(); + if (this.customizedRender) { + const el = await this.customizedRender(); if (el) { this.contentWindow.document?.body?.appendChild(el); } diff --git a/packages/stage/src/TargetCalibrate.ts b/packages/stage/src/TargetCalibrate.ts deleted file mode 100644 index 5731654f..00000000 --- a/packages/stage/src/TargetCalibrate.ts +++ /dev/null @@ -1,119 +0,0 @@ -/* - * Tencent is pleased to support the open source community by making TMagicEditor available. - * - * Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved. - * - * Licensed under the Apache License, Version 2.0 (the "License"); - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -import { EventEmitter } from 'events'; - -import { Mode, ZIndex } from './const'; -import StageCore from './StageCore'; -import StageDragResize from './StageDragResize'; -import StageMask from './StageMask'; -import type { Offset, TargetCalibrateConfig } from './types'; -import { getMode } from './util'; - -/** - * 将选中的节点修正定位后,添加一个操作节点到蒙层上 - */ -export default class TargetCalibrate extends EventEmitter { - public parent: HTMLElement; - public mask: StageMask; - public dr: StageDragResize; - public core: StageCore; - public operationEl: HTMLDivElement; - - constructor(config: TargetCalibrateConfig) { - super(); - - this.parent = config.parent; - this.mask = config.mask; - this.dr = config.dr; - this.core = config.core; - - this.operationEl = globalThis.document.createElement('div'); - this.parent.append(this.operationEl); - } - - public update(el: HTMLElement, prefix: String): HTMLElement { - const { left, top } = this.getOffset(el); - const { transform } = getComputedStyle(el); - this.operationEl.style.cssText = ` - position: absolute; - transform: ${transform}; - left: ${left}px; - top: ${top}px; - width: ${el.clientWidth}px; - height: ${el.clientHeight}px; - z-index: ${ZIndex.HIGHLIGHT_EL}; - `; - - this.operationEl.id = `${prefix}${el.id}`; - - if (typeof this.core.config.updateDragEl === 'function') { - this.core.config.updateDragEl(this.operationEl, el); - } - - return this.operationEl; - } - - public destroy(): void { - this.operationEl?.remove(); - } - - private getOffset(el: HTMLElement): Offset { - const { offsetParent } = el; - - const left = el.offsetLeft; - const top = el.offsetTop; - - if (offsetParent) { - const parentOffset = this.getOffset(offsetParent as HTMLElement); - return { - left: left + parentOffset.left, - top: top + parentOffset.top, - }; - } - - // 选中固定定位元素后editor-mask高度被置为视窗大小 - if (this.dr.mode === Mode.FIXED) { - // 弹窗的情况 - if (getMode(el) === Mode.FIXED) { - return { - left, - top, - }; - } - - return { - left: left - this.mask.scrollLeft, - top: top - this.mask.scrollTop, - }; - } - - // 无父元素的固定定位需按滚动值计算 - if (getMode(el) === Mode.FIXED) { - return { - left: left + this.mask.scrollLeft, - top: top + this.mask.scrollTop, - }; - } - - return { - left, - top, - }; - } -} diff --git a/packages/stage/src/TargetShadow.ts b/packages/stage/src/TargetShadow.ts new file mode 100644 index 00000000..168b52bb --- /dev/null +++ b/packages/stage/src/TargetShadow.ts @@ -0,0 +1,120 @@ +/* + * Tencent is pleased to support the open source community by making TMagicEditor available. + * + * Copyright (C) 2021 THL A29 Limited, a Tencent company. All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +import { Mode, ZIndex } from './const'; +import type { TargetElement, TargetShadowConfig, UpdateDragEl } from './types'; +import { getTargetElStyle, isFixedParent } from './util'; + +/** + * 将选中的节点修正定位后,添加一个操作节点到蒙层上 + */ +export default class TargetShadow { + public el?: TargetElement; + public els: TargetElement[] = []; + + private idPrefix = 'target_calibrate_'; + private container: HTMLElement; + private scrollLeft = 0; + private scrollTop = 0; + private zIndex?: ZIndex; + + private updateDragEl?: UpdateDragEl; + + constructor(config: TargetShadowConfig) { + this.container = config.container; + + if (config.updateDragEl) { + this.updateDragEl = config.updateDragEl; + } + + if (typeof config.zIndex !== 'undefined') { + this.zIndex = config.zIndex; + } + + if (config.idPrefix) { + this.idPrefix = config.idPrefix; + } + + this.container.addEventListener('customScroll', this.scrollHandler); + } + + public update(target: TargetElement): TargetElement { + this.el = this.updateEl(target, this.el); + + return this.el; + } + + public updateGroup(targetGroup: TargetElement[]): TargetElement[] { + if (this.els.length > targetGroup.length) { + this.els.slice(targetGroup.length - 1).forEach((el) => { + el.remove(); + }); + } + + this.els = targetGroup.map((target, index) => this.updateEl(target, this.els[index])); + + return this.els; + } + + public destroyEl(): void { + this.el?.remove(); + this.el = undefined; + } + + public destroyEls(): void { + this.els.forEach((el) => { + el.remove(); + }); + this.els = []; + } + + public destroy(): void { + this.container.removeEventListener('customScroll', this.scrollHandler); + this.destroyEl(); + this.destroyEls(); + } + + private updateEl(target: TargetElement, src?: TargetElement): TargetElement { + const el = src || globalThis.document.createElement('div'); + + el.id = `${this.idPrefix}${target.id}`; + + el.style.cssText = getTargetElStyle(target, this.zIndex); + + if (typeof this.updateDragEl === 'function') { + this.updateDragEl(el, target); + } + const isFixed = isFixedParent(target); + const mode = this.container.dataset.mode || Mode.ABSOLUTE; + if (isFixed && mode !== Mode.FIXED) { + el.style.transform = `translate3d(${this.scrollLeft}px, ${this.scrollTop}px, 0)`; + } else if (!isFixed && mode === Mode.FIXED) { + el.style.transform = `translate3d(${-this.scrollLeft}px, ${-this.scrollTop}px, 0)`; + } + + if (!globalThis.document.getElementById(el.id)) { + this.container.append(el); + } + + return el; + } + + private scrollHandler = (e: any) => { + this.scrollLeft = e.detail.scrollLeft; + this.scrollTop = e.detail.scrollTop; + }; +} diff --git a/packages/stage/src/const.ts b/packages/stage/src/const.ts index 68508921..df147d09 100644 --- a/packages/stage/src/const.ts +++ b/packages/stage/src/const.ts @@ -25,7 +25,7 @@ export const DRAG_EL_ID_PREFIX = 'drag_el_'; /** 高亮时需要在蒙层中创建一个占位节点,该节点的id前缀 */ export const HIGHLIGHT_EL_ID_PREFIX = 'highlight_el_'; -export const CONTAINER_HIGHLIGHT_CLASS = 'tmagic-stage-container-highlight'; +export const CONTAINER_HIGHLIGHT_CLASS_NAME = 'tmagic-stage-container-highlight'; export const PAGE_CLASS = 'magic-ui-page'; diff --git a/packages/stage/src/types.ts b/packages/stage/src/types.ts index 0301c636..4423afbe 100644 --- a/packages/stage/src/types.ts +++ b/packages/stage/src/types.ts @@ -21,20 +21,39 @@ import type { MoveableOptions } from 'moveable'; import Core from '@tmagic/core'; import type { Id, MApp, MContainer, MNode } from '@tmagic/schema'; -import { GuidesType } from './const'; +import { GuidesType, ZIndex } from './const'; import StageCore from './StageCore'; -import StageDragResize from './StageDragResize'; -import StageMask from './StageMask'; + +export type TargetElement = HTMLElement | SVGElement; export type CanSelect = (el: HTMLElement, event: MouseEvent, stop: () => boolean) => boolean | Promise; export type IsContainer = (el: HTMLElement) => boolean | Promise; +export type CustomizeRender = (renderer: StageCore) => Promise | HTMLElement; +/** 业务方自定义的moveableOptions,可以是配置,也可以是回调函数 */ +export type CustomizeMoveableOptions = + | ((config?: CustomizeMoveableOptionsCallbackConfig) => MoveableOptions) + | MoveableOptions + | undefined; +/** render提供给的接口,如果是id则转成el,如果是el则直接返回 */ +export type GetTargetElement = (idOrEl: Id | HTMLElement) => HTMLElement; +/** render提供的接口,通过坐标获得坐标下所有HTML元素数组 */ +export type GetElementsFromPoint = (point: Point) => HTMLElement[]; +export type GetRenderDocument = () => Document | undefined; +export type DelayedMarkContainer = (event: MouseEvent, exclude: Element[]) => NodeJS.Timeout | undefined; +export type MarkContainerEnd = () => HTMLElement | null; +export type GetRootContainer = () => HTMLDivElement | undefined; +/** 将组件添加到容器的方式 */ export enum ContainerHighlightType { + /** 默认方式:组件在容器上方悬停一段时间后加入 */ DEFAULT = 'default', + /** 按住alt键,并在容器上方悬停一段时间后加入 */ ALT = 'alt', } -export type StageCoreConfig = { +export type UpdateDragEl = (el: TargetElement, target: TargetElement) => void; + +export interface StageCoreConfig { /** 需要对齐的dom节点的CSS选择器字符串 */ snapElementQuerySelector?: string; /** 放大倍数,默认1倍 */ @@ -44,18 +63,45 @@ export type StageCoreConfig = { containerHighlightClassName?: string; containerHighlightDuration?: number; containerHighlightType?: ContainerHighlightType; - moveableOptions?: ((core?: StageCore) => MoveableOptions) | MoveableOptions; - multiMoveableOptions?: ((core?: StageCore) => MoveableOptions) | MoveableOptions; + moveableOptions?: CustomizeMoveableOptions; + multiMoveableOptions?: CustomizeMoveableOptions; /** runtime 的HTML地址,可以是一个HTTP地址,如果和编辑器不同域,需要设置跨域,也可以是一个相对或绝对路径 */ runtimeUrl?: string; render?: (renderer: StageCore) => Promise | HTMLElement; autoScrollIntoView?: boolean; - updateDragEl?: (el: HTMLDivElement, target: HTMLElement) => void; -}; + updateDragEl?: UpdateDragEl; +} + +export interface ActionManagerConfig { + container: HTMLElement; + containerHighlightClassName?: string; + containerHighlightDuration?: number; + containerHighlightType?: ContainerHighlightType; + moveableOptions?: CustomizeMoveableOptions; + multiMoveableOptions?: CustomizeMoveableOptions; + canSelect?: CanSelect; + isContainer: IsContainer; + getRootContainer: GetRootContainer; + getRenderDocument: GetRenderDocument; + updateDragEl?: UpdateDragEl; + getTargetElement: GetTargetElement; + getElementsFromPoint: GetElementsFromPoint; +} + +export interface MoveableOptionsManagerConfig { + container: HTMLElement; + moveableOptions?: CustomizeMoveableOptions; + getRootContainer: GetRootContainer; +} + +export interface CustomizeMoveableOptionsCallbackConfig { + targetElId?: string; +} export interface StageRenderConfig { runtimeUrl?: string; - render?: () => Promise; + zoom: number | undefined; + customizedRender?: () => Promise; } export interface StageMaskConfig { @@ -63,9 +109,29 @@ export interface StageMaskConfig { } export interface StageDragResizeConfig { - core: StageCore; container: HTMLElement; - mask: StageMask; + moveableOptions?: CustomizeMoveableOptions; + getRootContainer: GetRootContainer; + getRenderDocument: GetRenderDocument; + markContainerEnd: MarkContainerEnd; + delayedMarkContainer: DelayedMarkContainer; + updateDragEl?: UpdateDragEl; +} + +export interface StageMultiDragResizeConfig { + container: HTMLElement; + multiMoveableOptions?: CustomizeMoveableOptions; + getRootContainer: GetRootContainer; + getRenderDocument: GetRenderDocument; + updateDragEl?: UpdateDragEl; +} + +/** 选择状态 */ +export enum SelectStatus { + /** 单选 */ + SELECT = 'select', + /** 多选 */ + MULTI_SELECT = 'multiSelect', } /** 拖动状态 */ @@ -88,6 +154,11 @@ export interface Offset { top: number; } +export interface Point { + clientX: number; + clientY: number; +} + export interface GuidesEventData { type: GuidesType; guides: number[]; @@ -154,13 +225,14 @@ export interface RuntimeWindow extends Window { } export interface StageHighlightConfig { - core: StageCore; container: HTMLElement; + updateDragEl?: UpdateDragEl; + getRootContainer: GetRootContainer; } -export interface TargetCalibrateConfig { - parent: HTMLElement; - mask: StageMask; - dr: StageDragResize; - core: StageCore; +export interface TargetShadowConfig { + container: HTMLElement; + zIndex?: ZIndex; + updateDragEl?: UpdateDragEl; + idPrefix?: string; } diff --git a/packages/stage/src/util.ts b/packages/stage/src/util.ts index 4c8a5aa5..929f32da 100644 --- a/packages/stage/src/util.ts +++ b/packages/stage/src/util.ts @@ -18,7 +18,7 @@ import { removeClassName } from '@tmagic/utils'; import { GHOST_EL_ID_PREFIX, Mode, SELECTED_CLASS, ZIndex } from './const'; -import type { Offset, SortEventData } from './types'; +import type { Offset, SortEventData, TargetElement } from './types'; const getParents = (el: Element, relative: Element) => { let cur: Element | null = el.parentElement; @@ -30,12 +30,16 @@ const getParents = (el: Element, relative: Element) => { return parents; }; -export const getOffset = (el: HTMLElement): Offset => { - const { offsetParent } = el; +export const getOffset = (el: TargetElement): Offset => { + const htmlEl = el as HTMLElement; + const { offsetParent } = htmlEl; - const left = el.offsetLeft; - const top = el.offsetTop; + const left = htmlEl.offsetLeft || 0; + const top = htmlEl.offsetTop || 0; + // 在 Webkit 中,如果元素为隐藏的(该元素或其祖先元素的 style.display 为 "none"),或者该元素的 style.position 被设为 "fixed",则该属性返回 null。 + // 在 IE 9 中,如果该元素的 style.position 被设置为 "fixed",则该属性返回 null。(display:none 无影响。) + // body offsetParent 为 null if (offsetParent) { const parentOffset = getOffset(offsetParent as HTMLElement); return { @@ -51,7 +55,7 @@ export const getOffset = (el: HTMLElement): Offset => { }; // 将蒙层占位节点覆盖在原节点上方 -export const getTargetElStyle = (el: HTMLElement) => { +export const getTargetElStyle = (el: TargetElement, zIndex?: ZIndex) => { const offset = getOffset(el); const { transform } = getComputedStyle(el); return ` @@ -61,13 +65,16 @@ export const getTargetElStyle = (el: HTMLElement) => { top: ${offset.top}px; width: ${el.clientWidth}px; height: ${el.clientHeight}px; - z-index: ${ZIndex.DRAG_EL}; + ${typeof zIndex !== 'undefined' ? `z-index: ${zIndex};` : ''} `; }; export const getAbsolutePosition = (el: HTMLElement, { top, left }: Offset) => { const { offsetParent } = el; + // 在 Webkit 中,如果元素为隐藏的(该元素或其祖先元素的 style.display 为 "none"),或者该元素的 style.position 被设为 "fixed",则该属性返回 null。 + // 在 IE 9 中,如果该元素的 style.position 被设置为 "fixed",则该属性返回 null。(display:none 无影响。) + // body offsetParent 为 null if (offsetParent) { const parentOffset = getOffset(offsetParent as HTMLElement); return { @@ -87,7 +94,7 @@ export const isStatic = (style: CSSStyleDeclaration): boolean => style.position export const isFixed = (style: CSSStyleDeclaration): boolean => style.position === 'fixed'; -export const isFixedParent = (el: HTMLElement) => { +export const isFixedParent = (el: Element) => { let fixed = false; let dom = el; while (dom) { @@ -104,7 +111,7 @@ export const isFixedParent = (el: HTMLElement) => { return fixed; }; -export const getMode = (el: HTMLElement): Mode => { +export const getMode = (el: Element): Mode => { if (isFixedParent(el)) return Mode.FIXED; const style = getComputedStyle(el); if (isStatic(style) || isRelative(style)) return Mode.SORTABLE; @@ -168,7 +175,7 @@ export const calcValueByFontsize = (doc: Document, value: number) => { * @param {number} deltaTop 偏移量 * @param {Object} detail 当前选中的组件配置 */ -export const down = (deltaTop: number, target: HTMLElement | SVGElement): SortEventData | void => { +export const down = (deltaTop: number, target: TargetElement): SortEventData | void => { let swapIndex = 0; let addUpH = target.clientHeight; const brothers = Array.from(target.parentNode?.children || []).filter( @@ -204,7 +211,7 @@ export const down = (deltaTop: number, target: HTMLElement | SVGElement): SortEv * @param {number} deltaTop 偏移量 * @param {Object} detail 当前选中的组件配置 */ -export const up = (deltaTop: number, target: HTMLElement | SVGElement): SortEventData | void => { +export const up = (deltaTop: number, target: TargetElement): SortEventData | void => { const brothers = Array.from(target.parentNode?.children || []).filter( (node) => !node.id.startsWith(GHOST_EL_ID_PREFIX), ); diff --git a/packages/utils/src/dom.ts b/packages/utils/src/dom.ts index f14f4e0d..8af62d4d 100644 --- a/packages/utils/src/dom.ts +++ b/packages/utils/src/dom.ts @@ -108,3 +108,5 @@ export const createDiv = ({ className, cssText }: { className: string; cssText: el.style.cssText = cssText; return el; }; + +export const getDocument = () => globalThis.document; diff --git a/playground/src/pages/Editor.vue b/playground/src/pages/Editor.vue index 634778eb..8e7dc90a 100644 --- a/playground/src/pages/Editor.vue +++ b/playground/src/pages/Editor.vue @@ -41,7 +41,7 @@ import serialize from 'serialize-javascript'; import { editorService, MenuBarData, MoveableOptions, TMagicEditor } from '@tmagic/editor'; import type { Id, MContainer, MNode } from '@tmagic/schema'; import { NodeType } from '@tmagic/schema'; -import StageCore from '@tmagic/stage'; +import { CustomizeMoveableOptionsCallbackConfig } from '@tmagic/stage'; import { asyncLoadJs } from '@tmagic/utils'; import DeviceGroup from '../components/DeviceGroup.vue'; @@ -128,10 +128,10 @@ const menu: MenuBarData = { ], }; -const moveableOptions = (core?: StageCore): MoveableOptions => { +const moveableOptions = (config?: CustomizeMoveableOptionsCallbackConfig): MoveableOptions => { const options: MoveableOptions = {}; - const id = core?.dr?.target?.id; + const id = config?.targetElId; if (!id || !editor.value) return options; const node = editor.value.editorService.getNodeById(id);