feat(editor, stage): 新增禁用多选的props

This commit is contained in:
roymondchen 2023-12-18 15:00:26 +08:00
parent 09fe6d29e2
commit 2a5b9ec6bd
12 changed files with 184 additions and 85 deletions

View File

@ -923,4 +923,22 @@ const updateDragEl = (el, target) => {
<m-editor :disabled-drag-start="true"></m-editor> <m-editor :disabled-drag-start="true"></m-editor>
</template> </template>
``` ```
## disabledMultiSelect
- **详情:**
禁止多选
- **类型:** `boolean`
- **默认值:** `false`
- **示例:**
```html
<template>
<m-editor :disabled-multi-select="true"></m-editor>
</template>
```

View File

@ -177,6 +177,7 @@ provide(
disabledDragStart: props.disabledDragStart, disabledDragStart: props.disabledDragStart,
renderType: props.renderType, renderType: props.renderType,
guidesOptions: props.guidesOptions, guidesOptions: props.guidesOptions,
disabledMultiSelect: props.disabledMultiSelect,
}), }),
); );

View File

@ -18,7 +18,7 @@
> >
<MIcon <MIcon
class="expand-icon" class="expand-icon"
:style="hasChilren ? '' : 'color: transparent; cursor: default'" :style="hasChildren ? '' : 'color: transparent; cursor: default'"
:icon="expanded ? ArrowDown : ArrowRight" :icon="expanded ? ArrowDown : ArrowRight"
@click="expandHandler" @click="expandHandler"
></MIcon> ></MIcon>
@ -35,7 +35,7 @@
</div> </div>
</div> </div>
<div v-if="hasChilren && expanded" class="m-editor-tree-node-children"> <div v-if="hasChildren && expanded" class="m-editor-tree-node-children">
<TreeNode <TreeNode
v-for="item in data.items" v-for="item in data.items"
:key="item.id" :key="item.id"
@ -114,7 +114,7 @@ const selected = computed(() => nodeStatus.value.selected);
const visible = computed(() => nodeStatus.value.visible); const visible = computed(() => nodeStatus.value.visible);
const draggable = computed(() => nodeStatus.value.draggable); const draggable = computed(() => nodeStatus.value.draggable);
const hasChilren = computed(() => props.data.items?.some((item) => props.nodeStatusMap.get(item.id)?.visible)); const hasChildren = computed(() => props.data.items?.some((item) => props.nodeStatusMap.get(item.id)?.visible));
const handleDragStart = (event: DragEvent) => { const handleDragStart = (event: DragEvent) => {
treeEmit?.('node-dragstart', event, props.data); treeEmit?.('node-dragstart', event, props.data);

View File

@ -73,6 +73,7 @@ export interface EditorProps {
/** 自定义依赖收集器,复制组件时会将关联依赖一并复制 */ /** 自定义依赖收集器,复制组件时会将关联依赖一并复制 */
collectorOptions?: CustomTargetOptions; collectorOptions?: CustomTargetOptions;
guidesOptions?: Partial<GuidesOptions>; guidesOptions?: Partial<GuidesOptions>;
disabledMultiSelect?: boolean;
} }
export const defaultEditorProps = { export const defaultEditorProps = {
@ -93,4 +94,5 @@ export const defaultEditorProps = {
containerHighlightType: ContainerHighlightType.DEFAULT, containerHighlightType: ContainerHighlightType.DEFAULT,
codeOptions: () => ({}), codeOptions: () => ({}),
renderType: RenderType.IFRAME, renderType: RenderType.IFRAME,
disabledMultiSelect: false,
}; };

View File

@ -1,4 +1,4 @@
import { computed } from 'vue'; import { computed, watch } from 'vue';
import type { MNode } from '@tmagic/schema'; import type { MNode } from '@tmagic/schema';
import StageCore, { GuidesType, RemoveEventData, SortEventData, UpdateEventData } from '@tmagic/stage'; import StageCore, { GuidesType, RemoveEventData, SortEventData, UpdateEventData } from '@tmagic/stage';
@ -45,8 +45,20 @@ export const useStage = (stageOptions: StageOptions) => {
moveableOptions: stageOptions.moveableOptions, moveableOptions: stageOptions.moveableOptions,
updateDragEl: stageOptions.updateDragEl, updateDragEl: stageOptions.updateDragEl,
guidesOptions: stageOptions.guidesOptions, guidesOptions: stageOptions.guidesOptions,
disabledMultiSelect: stageOptions.disabledMultiSelect,
}); });
watch(
() => editorService.get('disabledMultiSelect'),
(disabledMultiSelect) => {
if (disabledMultiSelect) {
stage.disableMultiSelect();
} else {
stage.enableMultiSelect();
}
},
);
stage.mask.setGuides([ stage.mask.setGuides([
getGuideLineFromCache(getGuideLineKey(H_GUIDE_LINE_STORAGE_KEY)), getGuideLineFromCache(getGuideLineKey(H_GUIDE_LINE_STORAGE_KEY)),
getGuideLineFromCache(getGuideLineKey(V_GUIDE_LINE_STORAGE_KEY)), getGuideLineFromCache(getGuideLineKey(V_GUIDE_LINE_STORAGE_KEY)),

View File

@ -47,6 +47,16 @@ export const initServiceState = (
}, },
); );
watch(
() => props.disabledMultiSelect,
(disabledMultiSelect) => {
editorService.set('disabledMultiSelect', disabledMultiSelect || false);
},
{
immediate: true,
},
);
watch( watch(
() => props.componentGroupList, () => props.componentGroupList,
(componentGroupList) => componentGroupList && componentListService.setList(componentGroupList), (componentGroupList) => componentGroupList && componentListService.setList(componentGroupList),

View File

@ -1,4 +1,4 @@
import { type ComputedRef, nextTick, type Ref, ref } from 'vue'; import { computed, type ComputedRef, nextTick, type Ref, ref } from 'vue';
import { throttle } from 'lodash-es'; import { throttle } from 'lodash-es';
import { Id, MNode } from '@tmagic/schema'; import { Id, MNode } from '@tmagic/schema';
@ -13,13 +13,15 @@ export const useClick = (
isCtrlKeyDown: Ref<boolean>, isCtrlKeyDown: Ref<boolean>,
nodeStatusMap: ComputedRef<Map<Id, LayerNodeStatus> | undefined>, nodeStatusMap: ComputedRef<Map<Id, LayerNodeStatus> | undefined>,
) => { ) => {
const isMultiSelect = computed(() => isCtrlKeyDown.value && !services?.editorService.get('disabledMultiSelect'));
// 触发画布选中 // 触发画布选中
const select = async (data: MNode) => { const select = async (data: MNode) => {
if (!data.id) { if (!data.id) {
throw new Error('没有id'); throw new Error('没有id');
} }
if (isCtrlKeyDown.value) { if (isMultiSelect.value) {
multiSelect(data); multiSelect(data);
} else { } else {
await services?.editorService.select(data); await services?.editorService.select(data);
@ -70,7 +72,7 @@ export const useClick = (
return; return;
} }
if (data.items && data.items.length > 0 && !isCtrlKeyDown.value) { if (data.items && data.items.length > 0 && !isMultiSelect.value) {
updateStatus(nodeStatusMap.value, data.id, { updateStatus(nodeStatusMap.value, data.id, {
expand: true, expand: true,
}); });

View File

@ -58,6 +58,7 @@ class Editor extends BaseService {
highlightNode: null, highlightNode: null,
modifiedNodeIds: new Map(), modifiedNodeIds: new Map(),
pageLength: 0, pageLength: 0,
disabledMultiSelect: false,
}); });
private isHistoryStateChange = false; private isHistoryStateChange = false;

View File

@ -133,6 +133,7 @@ export interface StageOptions {
updateDragEl: UpdateDragEl; updateDragEl: UpdateDragEl;
renderType: RenderType; renderType: RenderType;
guidesOptions: Partial<GuidesOptions>; guidesOptions: Partial<GuidesOptions>;
disabledMultiSelect?: boolean;
} }
export interface StoreState { export interface StoreState {
@ -146,6 +147,7 @@ export interface StoreState {
stageLoading: boolean; stageLoading: boolean;
modifiedNodeIds: Map<Id, Id>; modifiedNodeIds: Map<Id, Id>;
pageLength: number; pageLength: number;
disabledMultiSelect: boolean;
} }
export type StoreStateKey = keyof StoreState; export type StoreStateKey = keyof StoreState;

View File

@ -57,7 +57,7 @@ const defaultContainerHighlightDuration = 800;
*/ */
export default class ActionManager extends EventEmitter { export default class ActionManager extends EventEmitter {
private dr: StageDragResize; private dr: StageDragResize;
private multiDr: StageMultiDragResize; private multiDr?: StageMultiDragResize;
private highlightLayer: StageHighlight; private highlightLayer: StageHighlight;
/** 单选、多选、高亮的容器蒙层的content */ /** 单选、多选、高亮的容器蒙层的content */
private container: HTMLElement; private container: HTMLElement;
@ -81,6 +81,8 @@ export default class ActionManager extends EventEmitter {
private canSelect: CanSelect; private canSelect: CanSelect;
private isContainer: IsContainer; private isContainer: IsContainer;
private getRenderDocument: GetRenderDocument; private getRenderDocument: GetRenderDocument;
private disabledMultiSelect = false;
private config: ActionManagerConfig;
private mouseMoveHandler = throttle(async (event: MouseEvent): Promise<void> => { private mouseMoveHandler = throttle(async (event: MouseEvent): Promise<void> => {
if ((event.target as HTMLDivElement)?.classList?.contains('moveable-direction')) { if ((event.target as HTMLDivElement)?.classList?.contains('moveable-direction')) {
@ -99,41 +101,24 @@ export default class ActionManager extends EventEmitter {
constructor(config: ActionManagerConfig) { constructor(config: ActionManagerConfig) {
super(); super();
this.config = config;
this.container = config.container; this.container = config.container;
this.containerHighlightClassName = config.containerHighlightClassName || CONTAINER_HIGHLIGHT_CLASS_NAME; this.containerHighlightClassName = config.containerHighlightClassName || CONTAINER_HIGHLIGHT_CLASS_NAME;
this.containerHighlightDuration = config.containerHighlightDuration || defaultContainerHighlightDuration; this.containerHighlightDuration = config.containerHighlightDuration || defaultContainerHighlightDuration;
this.containerHighlightType = config.containerHighlightType; this.containerHighlightType = config.containerHighlightType;
this.disabledMultiSelect = config.disabledMultiSelect ?? false;
this.getTargetElement = config.getTargetElement; this.getTargetElement = config.getTargetElement;
this.getElementsFromPoint = config.getElementsFromPoint; this.getElementsFromPoint = config.getElementsFromPoint;
this.canSelect = config.canSelect || ((el: HTMLElement) => !!el.id); this.canSelect = config.canSelect || ((el: HTMLElement) => !!el.id);
this.getRenderDocument = config.getRenderDocument; this.getRenderDocument = config.getRenderDocument;
this.isContainer = config.isContainer; this.isContainer = config.isContainer;
const createDrHelper = () => this.dr = this.createDr(config);
new DragResizeHelper({
container: config.container, if (!this.disabledMultiSelect) {
updateDragEl: config.updateDragEl, this.multiDr = this.createMultiDr(config);
}); }
this.dr = new StageDragResize({
container: config.container,
disabledDragStart: config.disabledDragStart,
moveableOptions: this.changeCallback(config.moveableOptions, false),
dragResizeHelper: createDrHelper(),
getRootContainer: config.getRootContainer,
getRenderDocument: config.getRenderDocument,
markContainerEnd: this.markContainerEnd.bind(this),
delayedMarkContainer: this.delayedMarkContainer.bind(this),
});
this.multiDr = new StageMultiDragResize({
container: config.container,
moveableOptions: this.changeCallback(config.moveableOptions, true),
dragResizeHelper: createDrHelper(),
getRootContainer: config.getRootContainer,
getRenderDocument: config.getRenderDocument,
markContainerEnd: this.markContainerEnd.bind(this),
delayedMarkContainer: this.delayedMarkContainer.bind(this),
});
this.highlightLayer = new StageHighlight({ this.highlightLayer = new StageHighlight({
container: config.container, container: config.container,
updateDragEl: config.updateDragEl, updateDragEl: config.updateDragEl,
@ -142,7 +127,22 @@ export default class ActionManager extends EventEmitter {
this.initMouseEvent(); this.initMouseEvent();
this.initKeyEvent(); this.initKeyEvent();
this.initActionEvent(); }
public disableMultiSelect() {
this.disabledMultiSelect = true;
if (this.multiDr) {
this.multiDr.destroy();
this.multiDr = undefined;
}
}
public enableMultiSelect() {
this.disabledMultiSelect = false;
if (!this.multiDr) {
this.multiDr = this.createMultiDr(this.config);
}
} }
/** /**
@ -152,7 +152,7 @@ export default class ActionManager extends EventEmitter {
*/ */
public setGuidelines(type: GuidesType, guidelines: number[]): void { public setGuidelines(type: GuidesType, guidelines: number[]): void {
this.dr.setGuidelines(type, guidelines); this.dr.setGuidelines(type, guidelines);
this.multiDr.setGuidelines(type, guidelines); this.multiDr?.setGuidelines(type, guidelines);
} }
/** /**
@ -160,7 +160,7 @@ export default class ActionManager extends EventEmitter {
*/ */
public clearGuides(): void { public clearGuides(): void {
this.dr.clearGuides(); this.dr.clearGuides();
this.multiDr.clearGuides(); this.multiDr?.clearGuides();
} }
/** /**
@ -170,7 +170,7 @@ export default class ActionManager extends EventEmitter {
public updateMoveable(el?: HTMLElement): void { public updateMoveable(el?: HTMLElement): void {
this.dr.updateMoveable(el); this.dr.updateMoveable(el);
// 多选时不可配置元素因此不存在多选元素变更不需要传el // 多选时不可配置元素因此不存在多选元素变更不需要传el
this.multiDr.updateMoveable(); this.multiDr?.updateMoveable();
} }
/** /**
@ -197,7 +197,7 @@ export default class ActionManager extends EventEmitter {
if (this.dr.getTarget()) { if (this.dr.getTarget()) {
return this.dr.getOption(key); return this.dr.getOption(key);
} }
if (this.multiDr.targetList.length) { if (this.multiDr?.targetList.length) {
return this.multiDr.getOption(key); return this.multiDr.getOption(key);
} }
} }
@ -254,7 +254,7 @@ export default class ActionManager extends EventEmitter {
if (selectedEl?.className.includes(PAGE_CLASS)) { if (selectedEl?.className.includes(PAGE_CLASS)) {
return true; return true;
} }
return this.multiDr.canSelect(el, selectedEl); return this.multiDr?.canSelect(el, selectedEl) || false;
} }
public select(el: HTMLElement, event: MouseEvent | undefined): void { public select(el: HTMLElement, event: MouseEvent | undefined): void {
@ -266,7 +266,7 @@ export default class ActionManager extends EventEmitter {
public multiSelect(idOrElList: HTMLElement[] | Id[]): void { public multiSelect(idOrElList: HTMLElement[] | Id[]): void {
this.selectedElList = idOrElList.map((idOrEl) => this.getTargetElement(idOrEl)); this.selectedElList = idOrElList.map((idOrEl) => this.getTargetElement(idOrEl));
this.clearSelectStatus(SelectStatus.SELECT); this.clearSelectStatus(SelectStatus.SELECT);
this.multiDr.multiSelect(this.selectedElList); this.multiDr?.multiSelect(this.selectedElList);
} }
public getHighlightEl(): HTMLElement | undefined { public getHighlightEl(): HTMLElement | undefined {
@ -287,7 +287,7 @@ export default class ActionManager extends EventEmitter {
} }
// 选中组件不高亮、多选拖拽状态不高亮 // 选中组件不高亮、多选拖拽状态不高亮
if (el === this.getSelectedEl() || this.multiDr.dragStatus === StageDragStatus.ING) { if (el === this.getSelectedEl() || this.multiDr?.dragStatus === StageDragStatus.ING) {
this.clearHighlight(); this.clearHighlight();
return; return;
} }
@ -309,7 +309,7 @@ export default class ActionManager extends EventEmitter {
*/ */
public clearSelectStatus(selectType: SelectStatus): void { public clearSelectStatus(selectType: SelectStatus): void {
if (selectType === SelectStatus.MULTI_SELECT) { if (selectType === SelectStatus.MULTI_SELECT) {
this.multiDr.clearSelectStatus(); this.multiDr?.clearSelectStatus();
this.selectedElList = []; this.selectedElList = [];
} else { } else {
this.dr.clearSelectStatus(); this.dr.clearSelectStatus();
@ -361,10 +361,84 @@ export default class ActionManager extends EventEmitter {
this.container.removeEventListener('mouseleave', this.mouseLeaveHandler); this.container.removeEventListener('mouseleave', this.mouseLeaveHandler);
this.container.removeEventListener('wheel', this.mouseWheelHandler); this.container.removeEventListener('wheel', this.mouseWheelHandler);
this.dr.destroy(); this.dr.destroy();
this.multiDr.destroy(); this.multiDr?.destroy();
this.highlightLayer.destroy(); this.highlightLayer.destroy();
} }
private createDr(config: ActionManagerConfig) {
const createDrHelper = () =>
new DragResizeHelper({
container: config.container,
updateDragEl: config.updateDragEl,
});
const dr = new StageDragResize({
container: config.container,
disabledDragStart: config.disabledDragStart,
moveableOptions: this.changeCallback(config.moveableOptions, false),
dragResizeHelper: createDrHelper(),
getRootContainer: config.getRootContainer,
getRenderDocument: config.getRenderDocument,
markContainerEnd: this.markContainerEnd.bind(this),
delayedMarkContainer: this.delayedMarkContainer.bind(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');
})
.on('remove', () => {
const drTarget = this.dr.getTarget();
if (!drTarget) return;
const data: RemoveEventData = {
data: [{ el: drTarget }],
};
this.emit('remove', data);
})
.on('drag-start', (e: OnDragStart) => {
this.emit('drag-start', e);
});
return dr;
}
private createMultiDr(config: ActionManagerConfig) {
const createDrHelper = () =>
new DragResizeHelper({
container: config.container,
updateDragEl: config.updateDragEl,
});
const multiDr = new StageMultiDragResize({
container: config.container,
moveableOptions: this.changeCallback(config.moveableOptions, true),
dragResizeHelper: createDrHelper(),
getRootContainer: config.getRootContainer,
getRenderDocument: config.getRenderDocument,
markContainerEnd: this.markContainerEnd.bind(this),
delayedMarkContainer: this.delayedMarkContainer.bind(this),
});
multiDr
?.on('update', (data: UpdateEventData) => {
this.emit('multi-update', data);
})
.on('change-to-select', async (id: Id) => {
// 如果还在多选状态,不触发切换到单选
if (this.isMultiSelectStatus) return false;
const el = this.getTargetElement(id);
this.emit('change-to-select', el);
});
return multiDr;
}
private changeCallback(options: CustomizeMoveableOptions, isMulti: boolean): CustomizeMoveableOptions { private changeCallback(options: CustomizeMoveableOptions, isMulti: boolean): CustomizeMoveableOptions {
// 在actionManager才能获取到各种参数在这里传好参数有比较好的扩展性 // 在actionManager才能获取到各种参数在这里传好参数有比较好的扩展性
if (typeof options === 'function') { if (typeof options === 'function') {
@ -450,15 +524,21 @@ export default class ActionManager extends EventEmitter {
// 多选启用状态监听 // 多选启用状态监听
KeyController.global.keydown(ctrl, (e) => { KeyController.global.keydown(ctrl, (e) => {
e.inputEvent.preventDefault(); e.inputEvent.preventDefault();
this.isMultiSelectStatus = true; if (!this.disabledMultiSelect) {
this.isMultiSelectStatus = true;
}
}); });
// ctrl+tab切到其他窗口需要将多选状态置为false // ctrl+tab切到其他窗口需要将多选状态置为false
KeyController.global.on('blur', () => { KeyController.global.on('blur', () => {
this.isMultiSelectStatus = false; if (!this.disabledMultiSelect) {
this.isMultiSelectStatus = false;
}
}); });
KeyController.global.keyup(ctrl, (e) => { KeyController.global.keyup(ctrl, (e) => {
e.inputEvent.preventDefault(); e.inputEvent.preventDefault();
this.isMultiSelectStatus = false; if (!this.disabledMultiSelect) {
this.isMultiSelectStatus = false;
}
}); });
// alt健监听用于启用拖拽组件加入容器状态 // alt健监听用于启用拖拽组件加入容器状态
@ -474,46 +554,6 @@ export default class ActionManager extends EventEmitter {
}); });
} }
/**
*
*/
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');
})
.on('remove', () => {
const drTarget = this.dr.getTarget();
if (!drTarget) return;
const data: RemoveEventData = {
data: [{ el: drTarget }],
};
this.emit('remove', data);
})
.on('drag-start', (e: OnDragStart) => {
this.emit('drag-start', e);
});
this.multiDr
.on('update', (data: UpdateEventData) => {
this.emit('multi-update', data);
})
.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事件中再通知外面的编辑器更新 * down事件中集中cpu处理画布中选中操作渲染up事件中再通知外面的编辑器更新
*/ */

View File

@ -224,6 +224,14 @@ export default class StageCore extends EventEmitter {
return this.actionManager.getDragStatus(); return this.actionManager.getDragStatus();
} }
public disableMultiSelect() {
this.actionManager.disableMultiSelect();
}
public enableMultiSelect() {
this.actionManager.enableMultiSelect();
}
/** /**
* *
*/ */
@ -262,6 +270,7 @@ export default class StageCore extends EventEmitter {
moveableOptions: config.moveableOptions, moveableOptions: config.moveableOptions,
container: this.mask.content, container: this.mask.content,
disabledDragStart: config.disabledDragStart, disabledDragStart: config.disabledDragStart,
disabledMultiSelect: config.disabledMultiSelect,
canSelect: config.canSelect, canSelect: config.canSelect,
isContainer: config.isContainer, isContainer: config.isContainer,
updateDragEl: config.updateDragEl, updateDragEl: config.updateDragEl,

View File

@ -79,6 +79,7 @@ export interface StageCoreConfig {
disabledDragStart?: boolean; disabledDragStart?: boolean;
renderType?: RenderType; renderType?: RenderType;
guidesOptions?: Partial<GuidesOptions>; guidesOptions?: Partial<GuidesOptions>;
disabledMultiSelect?: boolean;
} }
export interface ActionManagerConfig { export interface ActionManagerConfig {
@ -88,6 +89,7 @@ export interface ActionManagerConfig {
containerHighlightType?: ContainerHighlightType; containerHighlightType?: ContainerHighlightType;
moveableOptions?: CustomizeMoveableOptions; moveableOptions?: CustomizeMoveableOptions;
disabledDragStart?: boolean; disabledDragStart?: boolean;
disabledMultiSelect?: boolean;
canSelect?: CanSelect; canSelect?: CanSelect;
isContainer: IsContainer; isContainer: IsContainer;
getRootContainer: GetRootContainer; getRootContainer: GetRootContainer;