From f52d2e475a5e3ef6dc9512f971c626852e536589 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 10:55:38 +0800 Subject: [PATCH 01/26] =?UTF-8?q?feat:windows=E4=B8=8B=E7=BC=96=E8=AF=91?= =?UTF-8?q?=E4=BF=AE=E6=94=B9?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- examples/x6-example-features/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/x6-example-features/package.json b/examples/x6-example-features/package.json index da862fa857..2bdac9c80b 100644 --- a/examples/x6-example-features/package.json +++ b/examples/x6-example-features/package.json @@ -3,7 +3,7 @@ "name": "@antv/x6-example-features", "version": "2.1.1", "scripts": { - "start": "export NODE_OPTIONS=--openssl-legacy-provider && umi dev", + "start": "set NODE_OPTIONS=--openssl-legacy-provider && umi dev", "build": "umi build", "lint": "eslint 'src/**/*.{js,ts}?(x)' --fix" }, From e4ed2d48f84520f06102cdc383e444f2e7259e9b Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 10:57:15 +0800 Subject: [PATCH 02/26] =?UTF-8?q?bug:x6=E5=B0=8F=E5=9C=B0=E5=9B=BE?= =?UTF-8?q?=E5=AE=9A=E4=BD=8D=E4=B8=8D=E5=87=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-plugin-minimap/src/index.ts | 97 ++++++++++++++++++++++--- 1 file changed, 86 insertions(+), 11 deletions(-) diff --git a/packages/x6-plugin-minimap/src/index.ts b/packages/x6-plugin-minimap/src/index.ts index 42a4167a80..e7cb12c79f 100644 --- a/packages/x6-plugin-minimap/src/index.ts +++ b/packages/x6-plugin-minimap/src/index.ts @@ -1,4 +1,4 @@ -import { FunctionExt, CssLoader, Dom, View, Graph, EventArgs } from '@antv/x6' +import { FunctionExt, CssLoader, Dom, View, Graph, EventArgs } from '../../x6/src/index' import { content } from './style/raw' export class MiniMap extends View implements Graph.Plugin { @@ -108,6 +108,7 @@ export class MiniMap extends View implements Graph.Plugin { this.sourceGraph.on('model:updated', this.onModelUpdated, this) } this.sourceGraph.on('resize', this.updatePaper, this) + // 当前container委托mousedown、touchstart事件到startAction事件;只有小地图容器被点击,才开始在startAction中监听mousemove等拖拽事件 this.delegateEvents({ mousedown: 'startAction', touchstart: 'startAction', @@ -140,12 +141,17 @@ export class MiniMap extends View implements Graph.Plugin { } protected onModelUpdated() { + // 模型每次更新,都调用小地图的targetGraph 缩放到合适位置 this.targetGraph.zoomToFit() } protected updatePaper(width: number, height: number): this protected updatePaper({ width, height }: EventArgs['resize']): this protected updatePaper(w: number | EventArgs['resize'], h?: number) { + // Graph 中指定的是viewPort 视口宽高,contentArea会根据内容撑开 + // MiniMap option中的宽高也是指定的 viewPort 视口宽高 + // 置于为什么contentArea 比viewPortArea小,并且全部显示了;还能拖拽移动,是svg的transform: matrix的作用 + let width: number let height: number if (typeof w === 'object') { @@ -156,14 +162,18 @@ export class MiniMap extends View implements Graph.Plugin { height = h as number } + // ratio:宽高比、scale:缩放比 + const origin = this.sourceGraph.options const scale = this.sourceGraph.transform.getScale() + // 小地图的最大宽高 = 宽高 - 2 * padding const maxWidth = this.options.width - 2 * this.options.padding const maxHeight = this.options.height - 2 * this.options.padding - + // 主视图的实际宽高 = 宽高 / 缩放比 width /= scale.sx // eslint-disable-line height /= scale.sy // eslint-disable-line + // 小地图宽|高比 = min(小地图viewArea宽/主视图viewArea宽, 小地图viewArea高/主地图viewArea高) this.ratio = Math.min(maxWidth / width, maxHeight / height) const ratio = this.ratio @@ -172,20 +182,26 @@ export class MiniMap extends View implements Graph.Plugin { width *= ratio // eslint-disable-line height *= ratio // eslint-disable-line + // 小地图图形实际宽高 = 主视图宽高 * 小地图宽|高比 this.targetGraph.resize(width, height) this.targetGraph.translate(x, y) if (this.scroller) { + // 有Scroller插件,直接使用宽高比作为缩放比 this.targetGraph.scale(ratio, ratio) } else { + // 没有使用Scroller插件,自适应缩放;并且计算小地图的缩放比 this.targetGraph.zoomToFit() } this.updateViewport() return this } - + // 更新小地图视口的框选区域 protected updateViewport() { + + + // 缩放比例 const sourceGraphScale = this.sourceGraph.transform.getScale() const targetGraphScale = this.targetGraph.transform.getScale() @@ -197,12 +213,21 @@ export class MiniMap extends View implements Graph.Plugin { } const position = Dom.position(this.targetGraph.container) + const translation = this.targetGraph.translate() translation.ty = translation.ty || 0 + + // 小地图框选div(整体区域)位置 this.geometry = { + // viewPort元素本身的top/left(不变) + 大窗口x.y位置 * 小地图缩放比例 + 小地图matrix矩阵偏移 x.y(不变) top: position.top + origin.y * targetGraphScale.sy + translation.ty, left: position.left + origin.x * targetGraphScale.sx + translation.tx, + // 小地图区域宽度 = 大窗口宽度 * 小地图缩放比例 / 大窗口缩放比例 + // 依据公式:小地图区域宽度 / 小地图缩放比例 = 大窗口宽度 / 大窗口缩放比例 + // 小地图缩放比例越小,小地图区域宽度也就越小 + + // 缩放比 = 视口宽高 / 画布内容区域宽高 width: (this.graphContainer.clientWidth! * targetGraphScale.sx) / sourceGraphScale.sx, @@ -215,8 +240,10 @@ export class MiniMap extends View implements Graph.Plugin { protected startAction(evt: Dom.MouseDownEvent) { const e = this.normalizeEvent(evt) + // 缩放还是平移 const action = e.target === this.zoomHandle ? 'zooming' : 'panning' const { tx, ty } = this.sourceGraph.translate() + const eventData: Util.EventData = { action, clientX: e.clientX, @@ -232,29 +259,44 @@ export class MiniMap extends View implements Graph.Plugin { this.targetGraphTransforming = true this.delegateDocumentEvents(Util.documentEvents, eventData) } - + // 拖动 protected doAction(evt: Dom.MouseMoveEvent) { const e = this.normalizeEvent(evt) const clientX = e.clientX const clientY = e.clientY const data = e.data as Util.EventData switch (data.action) { + // 平移 case 'panning': { const scale = this.sourceGraph.transform.getScale() + const targetScale = this.targetGraph.transform.getScale() + + // 相对于起始位置偏移了多少px const rx = (clientX - data.clientX) * scale.sx const ry = (clientY - data.clientY) * scale.sy + if (this.scroller) { this.graphContainer.scrollLeft = data.scrollLeft + rx / this.ratio this.graphContainer.scrollTop = data.scrollTop + ry / this.ratio } else { + // this.sourceGraph.translate( + // data.translateX - rx / this.ratio, + // data.translateY - ry / this.ratio, + // ) + + // 计算每次的偏移距离 + const x = (rx ) / targetScale.sx + const y = (ry ) / targetScale.sy + + // 现有偏移位置 - 每次偏移距离 this.sourceGraph.translate( - data.translateX - rx / this.ratio, - data.translateY - ry / this.ratio, + data.translateX - x, + data.translateY - y, ) } break } - + // 缩放 case 'zooming': { const startScale = data.scale const startGeometry = data.geometry @@ -284,7 +326,7 @@ export class MiniMap extends View implements Graph.Plugin { this.undelegateDocumentEvents() this.targetGraphTransforming = false } - + // 点击滚动 protected scrollTo(evt: Dom.MouseDownEvent) { const e = this.normalizeEvent(evt) @@ -294,6 +336,8 @@ export class MiniMap extends View implements Graph.Plugin { const ts = this.targetGraph.translate() ts.ty = ts.ty || 0 + console.log('scrollTo ts', ts) + if (e.offsetX == null) { const offset = Dom.offset(this.targetGraph.container) x = e.pageX - offset.left @@ -303,9 +347,40 @@ export class MiniMap extends View implements Graph.Plugin { y = e.offsetY } - const cx = (x - ts.tx) / this.ratio - const cy = (y - ts.ty) / this.ratio - this.sourceGraph.centerPoint(cx, cy) + const sourceScale = this.sourceGraph.transform.getScale() + const targetScale = this.targetGraph.transform.getScale() + + // // const cx = (x - ts.tx) / this.ratio + // // const cy = (y - ts.ty) / this.ratio + + // const originX = (x - ts.tx) + // const originY = (y - ts.ty) + // console.log('scrollTo', this.ratio, originX, originY) + // // 小地图scale缩小,大地图scale被放大, cx就越大;偏移差距也就越大 + // // 小地图scale放大,大地图scale被缩小, cx就越小;偏移差距也就越小 + // const cx = (originX * sourceScale.sx)/ targetScale.sx + // const cy = (originY * sourceScale.sy)/ targetScale.sy + + // console.log('big position', cx, cy) + + const position = Dom.position(this.viewport) + const containerPosition = Dom.position(this.targetGraph.container) + const {width, height} = this.viewport.getBoundingClientRect() + const start = { + x: position.left + width/2, + y: position.top - containerPosition.top + height/2, + } + const cx = (x - start.x) * sourceScale.sx / targetScale.sx + const cy = (y - start.y) * sourceScale.sy / targetScale.sy + + const { tx, ty } = this.sourceGraph.translate() + + this.sourceGraph.translate(tx - cx, ty - cy) + + + + + // this.sourceGraph.centerPoint(cx, cy) } @View.dispose() From df812a72527645cddab6ef4b26c2d13e9766d453 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 11:02:41 +0800 Subject: [PATCH 03/26] =?UTF-8?q?feat:=E9=83=A8=E5=88=86=E6=BA=90=E7=A0=81?= =?UTF-8?q?=E5=A4=87=E6=B3=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../x6-example-features/src/pages/index.tsx | 4 ++ .../src/pages/minimap/index.less | 8 +-- .../src/pages/minimap/index.tsx | 59 ++++++++++++------- packages/x6-common/src/dom/event/core.ts | 1 + packages/x6-geometry/src/rectangle.ts | 1 + packages/x6/src/graph/graph.ts | 2 +- packages/x6/src/graph/transform.ts | 12 +++- packages/x6/src/index.ts | 2 +- packages/x6/src/model/cell.ts | 4 +- 9 files changed, 62 insertions(+), 31 deletions(-) diff --git a/examples/x6-example-features/src/pages/index.tsx b/examples/x6-example-features/src/pages/index.tsx index d218d81ced..c676fb0d2a 100755 --- a/examples/x6-example-features/src/pages/index.tsx +++ b/examples/x6-example-features/src/pages/index.tsx @@ -183,6 +183,10 @@ const dataSource = [ example: 'history', description: '时光回溯', }, + { + example: 'minimap', + description: '小地图', + }, ].map((item, index) => ({ key: index, ...item })) const columns = [ diff --git a/examples/x6-example-features/src/pages/minimap/index.less b/examples/x6-example-features/src/pages/minimap/index.less index d79bb95346..166801ed6e 100644 --- a/examples/x6-example-features/src/pages/minimap/index.less +++ b/examples/x6-example-features/src/pages/minimap/index.less @@ -9,10 +9,10 @@ .app-minimap { position: absolute; - bottom: 0; - left: 620px; - width: 200px; - height: 150px; + bottom: 16px; + right: 167px; + width: 400px; + height: 400px; } .x6-widget-minimap-viewport { diff --git a/examples/x6-example-features/src/pages/minimap/index.tsx b/examples/x6-example-features/src/pages/minimap/index.tsx index 4b6add5436..cb141f820e 100644 --- a/examples/x6-example-features/src/pages/minimap/index.tsx +++ b/examples/x6-example-features/src/pages/minimap/index.tsx @@ -1,6 +1,6 @@ import * as React from 'react' import { Graph } from '@antv/x6' -import { MiniMap } from '@antv/x6-plugin-minimap' +import { MiniMap } from '../../../../../packages/x6-plugin-minimap/src/index' import { Scroller } from '@antv/x6-plugin-scroller' import { Radio } from 'antd' import { SimpleNodeView } from './simple-view' @@ -18,29 +18,28 @@ export default class Example extends React.Component { componentDidMount() { this.graph = new Graph({ + panning: { + enabled: true, + modifiers: [], + eventTypes: ['leftMouseDown'], + }, + mousewheel: { + enabled: true, + }, container: this.container, - width: 600, - height: 320, + width: 1600, + height: 800, background: { color: '#F2F7FA', }, }) - this.graph.use( - new Scroller({ - pageVisible: true, - pageBreak: false, - pannable: true, - }), - ) - this.graph.use( - new MiniMap({ - container: this.minimapContainer, - width: 200, - height: 160, - padding: 10, - }), - ) + // this.graph.use( + // new Scroller({ + // pannable: false, + // }), + // ) + this.graph.addNode({ x: 200, @@ -78,8 +77,8 @@ export default class Example extends React.Component { const target = this.graph.addNode({ shape: 'circle', - x: 160, - y: 180, + x: 1160, + y: 2180, width: 60, height: 60, label: 'World', @@ -102,6 +101,22 @@ export default class Example extends React.Component { }, }, }) + + + this.graph.use( + new MiniMap({ + container: this.minimapContainer, + width: 400, + height: 400, + padding: 1, + preserveAspectRatio:false, + graphOptions:{ + width:400, + height:400 + } + }), + ) + } onMinimapViewChange = (val: string) => { @@ -148,6 +163,9 @@ export default class Example extends React.Component { refMiniMapContainer = (container: HTMLDivElement) => { this.minimapContainer = container } + changeSize = () => { + this.graph.resize(1800, 800) + } render() { return ( @@ -159,6 +177,7 @@ export default class Example extends React.Component { defaultValue={'detailed'} optionType="button" /> +
diff --git a/packages/x6-common/src/dom/event/core.ts b/packages/x6-common/src/dom/event/core.ts index e37a917195..ee076737e3 100644 --- a/packages/x6-common/src/dom/event/core.ts +++ b/packages/x6-common/src/dom/event/core.ts @@ -244,6 +244,7 @@ export namespace Core { event.data = handleObj.data const hookHandle = EventHook.get(handleObj.originType).handle + console.log('dispatch 事件触发', handleObj.originType, hookHandle, handleObj.handler) const result = hookHandle ? hookHandle(matched.elem as Store.EventTarget, event, ...args) diff --git a/packages/x6-geometry/src/rectangle.ts b/packages/x6-geometry/src/rectangle.ts index f728e2dbba..7d1d785f06 100644 --- a/packages/x6-geometry/src/rectangle.ts +++ b/packages/x6-geometry/src/rectangle.ts @@ -624,6 +624,7 @@ export class Rectangle extends Geometry implements Rectangle.RectangleLike { /** * Returns a rectangle that is a union of this rectangle and rectangle `rect`. + * 返回一个矩形区域,该矩形是两个矩形区域的并集(最大区域)。 */ union(rect: Rectangle.RectangleLike | Rectangle.RectangleData) { const ref = Rectangle.clone(rect) diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index ea57fdf198..ed192acafc 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -1,5 +1,5 @@ import { Basecoat, NumberExt, Dom, KeyValue } from '@antv/x6-common' -import { Point, Rectangle } from '@antv/x6-geometry' +import { Point, Rectangle } from '../../../x6-geometry/src/index' import { Model, Collection, Cell, Node, Edge } from '../model' import { CellView } from '../view' import * as Registry from '../registry' diff --git a/packages/x6/src/graph/transform.ts b/packages/x6/src/graph/transform.ts index b4178d2c47..1e6d4bc5ff 100644 --- a/packages/x6/src/graph/transform.ts +++ b/packages/x6/src/graph/transform.ts @@ -1,5 +1,5 @@ -import { Dom, NumberExt } from '@antv/x6-common' -import { Point, Rectangle } from '@antv/x6-geometry' +import { Dom, NumberExt } from '../../../x6-common/src/index' +import { Point, Rectangle } from '../../../x6-geometry/src/index' import { Base } from './base' import { Util } from '../util' import { Cell } from '../model' @@ -322,6 +322,7 @@ export class TransformManager extends Base { options: TransformManager.ScaleContentToFitOptions = {}, translate = true, ) { + console.log('scaleContentToFitImpl', options) let contentBBox let contentLocalOrigin if (options.contentArea) { @@ -367,8 +368,10 @@ export class TransformManager extends Base { }) const currentScale = this.getScale() - + // 新的X轴缩放比(scaleX) = (视口宽 / 内容区域宽) * 当前X轴缩放比 + // 保证区域内容都在视口中展示 let newSX = (fittingBox.width / contentBBox.width) * currentScale.sx + // 新的y轴缩放比(scaleY) = (视口高 / 内容区域高) * 当前Y轴缩放比 let newSY = (fittingBox.height / contentBBox.height) * currentScale.sy if (options.preserveAspectRatio !== false) { @@ -421,8 +424,10 @@ export class TransformManager extends Base { const area = Rectangle.create(rect) const graph = this.graph + // 内容区域位置信息 options.contentArea = area if (options.viewportArea == null) { + // 视口位置信息(视口一般会小于内容区域位置,移动视口可以看内容区域不同位置) options.viewportArea = { x: graph.options.x, y: graph.options.y, @@ -459,6 +464,7 @@ export class TransformManager extends Base { y = cy - y * scale.sy // eslint-disable-line if (ts.tx !== x || ts.ty !== y) { + console.log('centerPoint', x, y) this.translate(x, y) } } diff --git a/packages/x6/src/index.ts b/packages/x6/src/index.ts index 7b9f015750..51f31725ac 100644 --- a/packages/x6/src/index.ts +++ b/packages/x6/src/index.ts @@ -7,7 +7,7 @@ export * from './graph' export * from './config' export * from './util' -export * from '@antv/x6-common' +export * from '../../x6-common/src/index' export * from '@antv/x6-geometry' export { Shape, Registry } diff --git a/packages/x6/src/model/cell.ts b/packages/x6/src/model/cell.ts index 87e94587a7..a0b66c2cd5 100644 --- a/packages/x6/src/model/cell.ts +++ b/packages/x6/src/model/cell.ts @@ -8,8 +8,8 @@ import { KeyValue, Size, Basecoat, -} from '@antv/x6-common' -import { Rectangle, Point } from '@antv/x6-geometry' +} from '../../../x6-common/src/index' +import { Rectangle, Point } from '../../../x6-geometry/src/index' import { NonUndefined } from 'utility-types' import { Attr } from '../registry' import { Model } from './model' From 494716f431d141618f1ea0f191b26f8680f18178 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 13:11:11 +0800 Subject: [PATCH 04/26] =?UTF-8?q?feat:x6-plugin-history=EF=BC=8C=E6=92=A4?= =?UTF-8?q?=E9=94=80=E5=9B=9E=E9=80=80=E6=8F=92=E4=BB=B6=E5=8E=9F=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-plugin-history/src/index.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packages/x6-plugin-history/src/index.ts b/packages/x6-plugin-history/src/index.ts index dd685eb411..004c17cb4d 100644 --- a/packages/x6-plugin-history/src/index.ts +++ b/packages/x6-plugin-history/src/index.ts @@ -176,6 +176,7 @@ export class History this.model.on('batch:start', this.initBatchCommand, this) this.model.on('batch:stop', this.storeBatchCommand, this) if (this.options.eventNames) { + // 监听哪些事件操作需要进入撤销回退队列,统一进入addCommand中处理 this.options.eventNames.forEach((name, index) => { this.handlers[index] = this.addCommand.bind(this, name) this.model.on(name, this.handlers[index]) @@ -525,11 +526,13 @@ export class History } protected push(cmd: History.Command, options: KeyValue) { + // 有新的操作进入撤销队列undoStack时,重做队列redoStack清空 this.redoStack = [] if (cmd.batch) { this.lastBatchIndex = Math.max(this.lastBatchIndex, 0) this.emit('batch', { cmd, options }) } else { + // 操作进入撤销队列 this.undoStackPush(cmd) this.consolidateCommands() this.notify('add', cmd, options) From ff4bf7121b51f8f6220897442bc9a77ccace8839 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 13:38:39 +0800 Subject: [PATCH 05/26] =?UTF-8?q?feat:=20Dom=E5=85=AC=E5=85=B1=E6=96=B9?= =?UTF-8?q?=E6=B3=95=E5=B0=81=E8=A3=85=EF=BC=88=E5=8F=AF=E5=80=9F=E9=89=B4?= =?UTF-8?q?=EF=BC=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/dom/index.ts | 2 +- packages/x6-common/src/dom/position.ts | 7 ++++++- 2 files changed, 7 insertions(+), 2 deletions(-) diff --git a/packages/x6-common/src/dom/index.ts b/packages/x6-common/src/dom/index.ts index 0a5c952325..4432a36d39 100644 --- a/packages/x6-common/src/dom/index.ts +++ b/packages/x6-common/src/dom/index.ts @@ -1,3 +1,3 @@ import * as Dom from './main' - +// 封装Dom相关的操作方法,可以借鉴 export { Dom } diff --git a/packages/x6-common/src/dom/position.ts b/packages/x6-common/src/dom/position.ts index 0bd5804306..10dfebe11c 100644 --- a/packages/x6-common/src/dom/position.ts +++ b/packages/x6-common/src/dom/position.ts @@ -20,7 +20,11 @@ export function height(elem: Element) { const rect = elem.getBoundingClientRect() return rect.height } - +/** + * 计算元素相对于父元素的绝对定位 + * @param elem + * @returns + */ export function position(elem: Element) { const isFixed = computeStyle(elem, 'position') === 'fixed' let offsetValue: ReturnType @@ -38,6 +42,7 @@ export function position(elem: Element) { (offsetParent === doc.body || offsetParent === doc.documentElement) && computeStyle(offsetParent, 'position') === 'static' ) { + // 如果 offsetParent 是 body 或 html 并且其定位属性为 static,则继续寻找更高层的定位父元素 offsetParent = offsetParent.parentNode } if (offsetParent !== elem && isElement(offsetParent)) { From db423672c99dad94e3a665470c3255f7fb2ef78f Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 14:59:53 +0800 Subject: [PATCH 06/26] =?UTF-8?q?feat:=E5=AF=B9=E8=B1=A1=E5=B1=9E=E6=80=A7?= =?UTF-8?q?mixins?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/object/mixins.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/x6-common/src/object/mixins.ts b/packages/x6-common/src/object/mixins.ts index ff848ad125..661493fe54 100644 --- a/packages/x6-common/src/object/mixins.ts +++ b/packages/x6-common/src/object/mixins.ts @@ -1,4 +1,5 @@ /** + * 对象属性混入mixins * @see https://www.typescriptlang.org/docs/handbook/mixins.html */ export function applyMixins(derivedCtor: any, ...baseCtors: any[]) { From 70702af01b1ef84c69fb33ee6800455a246fd1fb Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 15:23:32 +0800 Subject: [PATCH 07/26] =?UTF-8?q?feat:Dom=E4=BA=8B=E4=BB=B6=E6=B3=A8?= =?UTF-8?q?=E5=86=8C=E4=B8=8EGraph=E6=9C=AC=E8=BA=AB=E7=9A=84=E4=BA=8B?= =?UTF-8?q?=E4=BB=B6=E6=B3=A8=E5=86=8C=E5=8C=BA=E5=88=86=E5=BC=80?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/dom/event/core.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packages/x6-common/src/dom/event/core.ts b/packages/x6-common/src/dom/event/core.ts index ee076737e3..66296c198c 100644 --- a/packages/x6-common/src/dom/event/core.ts +++ b/packages/x6-common/src/dom/event/core.ts @@ -8,6 +8,8 @@ import './special' export namespace Core { let triggered: string | undefined + + // Dom的事件注册,与Graph本身的事件注册机制区分开;使用Dom.Event.on进行Dom事件注册 export function on( elem: Store.EventTarget, types: string, @@ -44,6 +46,7 @@ export namespace Core { let mainHandler = store.handler if (mainHandler == null) { mainHandler = store.handler = function (e, ...args: any[]) { + // 事件处理器 return triggered !== e.type ? dispatch(elem, e, ...args) : undefined } } From 65ddd454203601fa793b012c8dee67dd2657d74d Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 15:46:09 +0800 Subject: [PATCH 08/26] =?UTF-8?q?feat:=20Renderer=E6=B8=B2=E6=9F=93?= =?UTF-8?q?=E5=99=A8=E6=B8=B2=E6=9F=93=E5=8E=9F=E7=90=86?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/renderer/renderer.ts | 1 + 1 file changed, 1 insertion(+) diff --git a/packages/x6/src/renderer/renderer.ts b/packages/x6/src/renderer/renderer.ts index 72225ccd22..6f023df7c3 100644 --- a/packages/x6/src/renderer/renderer.ts +++ b/packages/x6/src/renderer/renderer.ts @@ -7,6 +7,7 @@ import { CellView, EdgeView } from '../view' import { Util } from '../util' export class Renderer extends Base { + // 初始化Renderer时,传入graph;再Renderer中的Scheduler调度器中,监听graph的相关事件(例如添加节点cell:added),然后进行渲染更新 private readonly schedule: Scheduler = new Scheduler(this.graph) requestViewUpdate(view: CellView, flag: number, options: any = {}) { From 5ebb94a420509824458f947ff7bef250e9589c58 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 16:45:33 +0800 Subject: [PATCH 09/26] =?UTF-8?q?feat:=E8=B0=83=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- examples/x6-example-features/src/pages/minimap/index.tsx | 2 +- packages/x6/src/graph/graph.ts | 4 +++- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/examples/x6-example-features/src/pages/minimap/index.tsx b/examples/x6-example-features/src/pages/minimap/index.tsx index cb141f820e..f423fc471e 100644 --- a/examples/x6-example-features/src/pages/minimap/index.tsx +++ b/examples/x6-example-features/src/pages/minimap/index.tsx @@ -1,5 +1,5 @@ import * as React from 'react' -import { Graph } from '@antv/x6' +import { Graph } from '../../../../../packages/x6' import { MiniMap } from '../../../../../packages/x6-plugin-minimap/src/index' import { Scroller } from '@antv/x6-plugin-scroller' import { Radio } from 'antd' diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index ed192acafc..6df21af833 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -1,4 +1,4 @@ -import { Basecoat, NumberExt, Dom, KeyValue } from '@antv/x6-common' +import { Basecoat, NumberExt, Dom, KeyValue } from '../../../x6-common' import { Point, Rectangle } from '../../../x6-geometry/src/index' import { Model, Collection, Cell, Node, Edge } from '../model' import { CellView } from '../view' @@ -52,6 +52,7 @@ export class Graph extends Basecoat { this.css = new Css(this) this.view = new GraphView(this) this.defs = new Defs(this) + // 坐标 this.coord = new Coord(this) this.transform = new Transform(this) this.highlight = new Highlight(this) @@ -68,6 +69,7 @@ export class Graph extends Basecoat { this.renderer = new ViewRenderer(this) this.panning = new Panning(this) this.mousewheel = new Wheel(this) + // 虚拟渲染 this.virtualRender = new VirtualRender(this) this.size = new Size(this) } From 1f88d3da6ae640b7ea8a2408c44e57eec1eddd68 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Mon, 19 Aug 2024 18:11:52 +0800 Subject: [PATCH 10/26] =?UTF-8?q?feat:=E9=BC=A0=E6=A0=87=E4=BA=A4=E4=BA=92?= =?UTF-8?q?=EF=BC=8C=E5=8F=8A=E5=85=83=E7=B4=A0=E6=8D=95=E6=8D=89+?= =?UTF-8?q?=E4=BA=8B=E4=BB=B6=E8=A7=A6=E5=8F=91?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/graph/graph.ts | 8 +++++++- packages/x6/src/graph/view.ts | 3 +++ 2 files changed, 10 insertions(+), 1 deletion(-) diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index 6df21af833..ec9a0ab305 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -52,7 +52,7 @@ export class Graph extends Basecoat { this.css = new Css(this) this.view = new GraphView(this) this.defs = new Defs(this) - // 坐标 + // 坐标?(用于捕捉交互的元素?) this.coord = new Coord(this) this.transform = new Transform(this) this.highlight = new Highlight(this) @@ -826,6 +826,12 @@ export class Graph extends Basecoat { // #region coord + /** + * 捕捉到栅格?(是否捕捉到栅格上的元素) + * @param x + * @param y + * @returns + */ snapToGrid(p: Point.PointLike): Point snapToGrid(x: number, y: number): Point snapToGrid(x: number | Point.PointLike, y?: number) { diff --git a/packages/x6/src/graph/view.ts b/packages/x6/src/graph/view.ts index 1923f1942a..83d5b2ae8b 100644 --- a/packages/x6/src/graph/view.ts +++ b/packages/x6/src/graph/view.ts @@ -50,6 +50,7 @@ export class GraphView extends View { } delegateEvents() { + // 委托监听this.container dom事件上的交互事件;在对应的handler上再去触发graph上的事件 const ctor = this.constructor as typeof GraphView super.delegateEvents(ctor.events) return this @@ -126,9 +127,11 @@ export class GraphView extends View { } const localPoint = this.graph.snapToGrid(e.clientX, e.clientY) + // 捕捉到元素,触发事件 if (view) { view.onClick(e, localPoint.x, localPoint.y) } else { + // 没有找到,触发空白点击事件 this.graph.trigger('blank:click', { e, x: localPoint.x, From c3b844e421a705977f206d807771835356eb0a40 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 09:02:48 +0800 Subject: [PATCH 11/26] =?UTF-8?q?feat:=E8=B0=83=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../x6-example-features/src/pages/minimap/index.tsx | 2 +- packages/x6/src/graph/graph.ts | 10 +++++----- 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/examples/x6-example-features/src/pages/minimap/index.tsx b/examples/x6-example-features/src/pages/minimap/index.tsx index f423fc471e..8d65106023 100644 --- a/examples/x6-example-features/src/pages/minimap/index.tsx +++ b/examples/x6-example-features/src/pages/minimap/index.tsx @@ -1,5 +1,5 @@ import * as React from 'react' -import { Graph } from '../../../../../packages/x6' +import { Graph } from '../../../../../packages/x6/src/index' import { MiniMap } from '../../../../../packages/x6-plugin-minimap/src/index' import { Scroller } from '@antv/x6-plugin-scroller' import { Radio } from 'antd' diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index ec9a0ab305..e170d43c04 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -1,4 +1,4 @@ -import { Basecoat, NumberExt, Dom, KeyValue } from '../../../x6-common' +import { Basecoat, NumberExt, Dom, KeyValue } from '@antv/x6-common/es' import { Point, Rectangle } from '../../../x6-geometry/src/index' import { Model, Collection, Cell, Node, Edge } from '../model' import { CellView } from '../view' @@ -826,11 +826,11 @@ export class Graph extends Basecoat { // #region coord - /** + /** * 捕捉到栅格?(是否捕捉到栅格上的元素) - * @param x - * @param y - * @returns + * @param x + * @param y + * @returns */ snapToGrid(p: Point.PointLike): Point snapToGrid(x: number, y: number): Point From 624da976b917b55dd1f19e49c3cd097a8fc01bf9 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 14:36:30 +0800 Subject: [PATCH 12/26] =?UTF-8?q?feat:=E4=BA=8B=E4=BB=B6=E5=A4=84=E7=90=86?= =?UTF-8?q?=EF=BC=8C=E6=9E=84=E5=BB=BA=E8=87=AA=E5=AE=9A=E4=B9=89event?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/dom/event/core.ts | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/packages/x6-common/src/dom/event/core.ts b/packages/x6-common/src/dom/event/core.ts index 66296c198c..d36adf52d0 100644 --- a/packages/x6-common/src/dom/event/core.ts +++ b/packages/x6-common/src/dom/event/core.ts @@ -212,6 +212,7 @@ export namespace Core { evt: Event | EventObject | string, ...args: any[] ) { + // 构建自定义的event数据 const event = EventObject.create(evt) event.delegateTarget = elem as Element @@ -243,11 +244,14 @@ export namespace Core { event.rnamespace == null || (handleObj.namespace && event.rnamespace.test(handleObj.namespace)) ) { + if(handleObj.originType === 'mousedown'){ + console.log('handleObj', handleObj.originType, handleObj) + } event.handleObj = handleObj event.data = handleObj.data const hookHandle = EventHook.get(handleObj.originType).handle - console.log('dispatch 事件触发', handleObj.originType, hookHandle, handleObj.handler) + // console.log('dispatch 事件触发', handleObj.originType) const result = hookHandle ? hookHandle(matched.elem as Store.EventTarget, event, ...args) From 48f6c0d08fc874f640c65aa30f34a30e3d5437b7 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 14:37:42 +0800 Subject: [PATCH 13/26] =?UTF-8?q?feat:=20render=E5=8F=8AqueueJob=20?= =?UTF-8?q?=E4=BB=BB=E5=8A=A1=E9=98=9F=E5=88=97?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/renderer/queueJob.ts | 13 +++++++++++-- packages/x6/src/renderer/renderer.ts | 2 +- packages/x6/src/renderer/scheduler.ts | 1 + 3 files changed, 13 insertions(+), 3 deletions(-) diff --git a/packages/x6/src/renderer/queueJob.ts b/packages/x6/src/renderer/queueJob.ts index 0a6690f00a..1a9128fcdc 100644 --- a/packages/x6/src/renderer/queueJob.ts +++ b/packages/x6/src/renderer/queueJob.ts @@ -58,7 +58,9 @@ export class JobQueue { this.queueFlush() } } - + /** + * 同步刷新执行任务队列 + */ flushJobsSync() { this.isFlushPending = false this.isFlushing = true @@ -76,6 +78,11 @@ export class JobQueue { this.isFlushing = false } + /** + * 二分法查找任务插入的位置,优先级高的排列在前 + * @param job + * @returns + */ private findInsertionIndex(job: Job) { let left = 0 let ins = this.queue.length @@ -92,7 +99,9 @@ export class JobQueue { } return ins } - + /** + * 异步任务调度,使用requestIdleCallback/setTimeout实现 + */ private scheduleJob() { if ('requestIdleCallback' in window) { if (this.scheduleId) { diff --git a/packages/x6/src/renderer/renderer.ts b/packages/x6/src/renderer/renderer.ts index 6f023df7c3..67c3a990ef 100644 --- a/packages/x6/src/renderer/renderer.ts +++ b/packages/x6/src/renderer/renderer.ts @@ -7,7 +7,7 @@ import { CellView, EdgeView } from '../view' import { Util } from '../util' export class Renderer extends Base { - // 初始化Renderer时,传入graph;再Renderer中的Scheduler调度器中,监听graph的相关事件(例如添加节点cell:added),然后进行渲染更新 + // 初始化Renderer时,传入graph;再Renderer中的Scheduler调度器中,监听graph的相关事件,注意这里监听的是model的事件(例如添加节点cell:added),然后进行渲染更新 private readonly schedule: Scheduler = new Scheduler(this.graph) requestViewUpdate(view: CellView, flag: number, options: any = {}) { diff --git a/packages/x6/src/renderer/scheduler.ts b/packages/x6/src/renderer/scheduler.ts index 7abddb9269..e2f8dd714c 100644 --- a/packages/x6/src/renderer/scheduler.ts +++ b/packages/x6/src/renderer/scheduler.ts @@ -316,6 +316,7 @@ export class Scheduler extends Disposable { if (viewItem) { const zIndex = view.cell.getZIndex() const pivot = this.addZPivot(zIndex) + // 先将view(NodeView/EdgeView)的Dom框架内容插入到container this.container.insertBefore(view.container, pivot) if (!view.cell.isVisible()) { From 17bef501c6b573c1830fd8dbc5eab56f203c99fb Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 14:44:28 +0800 Subject: [PATCH 14/26] =?UTF-8?q?feat:model=E4=B8=AD=E5=88=9B=E5=BB=BANode?= =?UTF-8?q?=E3=80=81Edge=E6=A8=A1=E5=9E=8B?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/model/edge.ts | 1 + packages/x6/src/model/model.ts | 2 ++ packages/x6/src/model/node.ts | 1 + 3 files changed, 4 insertions(+) diff --git a/packages/x6/src/model/edge.ts b/packages/x6/src/model/edge.ts index 1f7ce8b856..c8427b943c 100644 --- a/packages/x6/src/model/edge.ts +++ b/packages/x6/src/model/edge.ts @@ -1183,6 +1183,7 @@ export namespace Edge { export function create(options: Metadata) { const shape = options.shape || 'edge' + // 从registry 注册中心获取对应边的构造函数 const Ctor = registry.get(shape) if (Ctor) { return new Ctor(options) diff --git a/packages/x6/src/model/model.ts b/packages/x6/src/model/model.ts index a40912ccd5..92d2474ccb 100644 --- a/packages/x6/src/model/model.ts +++ b/packages/x6/src/model/model.ts @@ -252,6 +252,7 @@ export class Model extends Basecoat { } createNode(metadata: Node.Metadata) { + // 构建Node模型 return Node.create(metadata) } @@ -262,6 +263,7 @@ export class Model extends Basecoat { } createEdge(metadata: Edge.Metadata) { + // 构建Edge模型 return Edge.create(metadata) } diff --git a/packages/x6/src/model/node.ts b/packages/x6/src/model/node.ts index 0241e93093..a42e216c4b 100644 --- a/packages/x6/src/model/node.ts +++ b/packages/x6/src/model/node.ts @@ -1188,6 +1188,7 @@ export namespace Node { export function create(options: Metadata) { const shape = options.shape || 'rect' + // 从registry 注册中心获取对应图形的构造函数(自定义的图形也会在registry中注册) const Ctor = registry.get(shape) if (Ctor) { return new Ctor(options) From 86376baf426718f9014f2ad751e07433f71d9690 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 15:06:09 +0800 Subject: [PATCH 15/26] =?UTF-8?q?feat:=E5=85=83=E7=B4=A0=E6=8D=95=E6=8D=89?= =?UTF-8?q?=E9=80=BB=E8=BE=91(=E5=BD=93=E5=89=8D=E5=85=83=E7=B4=A0?= =?UTF-8?q?=E5=90=91=E4=B8=8A=E9=80=92=E5=BD=92=E6=8B=A5=E6=9C=89data-cell?= =?UTF-8?q?-id=E7=9A=84=E7=88=B6=E5=85=83=E7=B4=A0=E7=9A=84id=E5=80=BC)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/graph/view.ts | 8 ++++++-- packages/x6/src/renderer/renderer.ts | 2 ++ packages/x6/src/renderer/scheduler.ts | 2 ++ packages/x6/src/view/view.ts | 4 ++-- 4 files changed, 12 insertions(+), 4 deletions(-) diff --git a/packages/x6/src/graph/view.ts b/packages/x6/src/graph/view.ts index 83d5b2ae8b..423b6b18ef 100644 --- a/packages/x6/src/graph/view.ts +++ b/packages/x6/src/graph/view.ts @@ -88,7 +88,11 @@ export class GraphView extends View { return true } - + /** + * 根据element查找对应的view (NodeView/EdgeView) + * @param elem + * @returns + */ protected findView(elem: Element) { return this.graph.findViewByElem(elem) } @@ -127,7 +131,7 @@ export class GraphView extends View { } const localPoint = this.graph.snapToGrid(e.clientX, e.clientY) - // 捕捉到元素,触发事件 + // 捕捉到元素(NodeView、EdgeView),触发对应view中的onclick事件 if (view) { view.onClick(e, localPoint.x, localPoint.y) } else { diff --git a/packages/x6/src/renderer/renderer.ts b/packages/x6/src/renderer/renderer.ts index 67c3a990ef..e733563d15 100644 --- a/packages/x6/src/renderer/renderer.ts +++ b/packages/x6/src/renderer/renderer.ts @@ -35,10 +35,12 @@ export class Renderer extends Base { : elem[0] if (target) { + // 通过当前target元素,一直向上查找元素,直到找到拥有data-cell-id属性的元素,获取id值,在schedule中的views中,通过id找到对应的view const id = this.graph.view.findAttr('data-cell-id', target) if (id) { const views = this.schedule.views if (views[id]) { + // 返回view return views[id].view } } diff --git a/packages/x6/src/renderer/scheduler.ts b/packages/x6/src/renderer/scheduler.ts index e2f8dd714c..d5f1fc0d71 100644 --- a/packages/x6/src/renderer/scheduler.ts +++ b/packages/x6/src/renderer/scheduler.ts @@ -7,6 +7,7 @@ import { FlagManager } from '../view/flag' import { Graph } from '../graph' export class Scheduler extends Disposable { + // 存储所有的NodeView、EdgeView public views: KeyValue = {} public willRemoveViews: KeyValue = {} protected zPivots: KeyValue @@ -186,6 +187,7 @@ export class Scheduler extends Disposable { options, state: Scheduler.ViewState.CREATED, } + // scheduler中的views,存储所有的CellView(NodeView、EdgeView) this.views[id] = viewItem } } diff --git a/packages/x6/src/view/view.ts b/packages/x6/src/view/view.ts index d6d255884d..307526da96 100644 --- a/packages/x6/src/view/view.ts +++ b/packages/x6/src/view/view.ts @@ -1,4 +1,4 @@ -import { Dom, KeyValue, Basecoat } from '@antv/x6-common' +import { Dom, KeyValue, Basecoat } from '../../../x6-common/src/index.ts' import { EventArgs } from '@antv/x6-common/lib/event/types' import { Config } from '../config' import { Markup } from './markup' @@ -111,7 +111,7 @@ export abstract class View extends Basecoat { if (current === this.container) { return null } - + // 向上递归寻找拥有指定属性attrName的元素 current = current.parentNode as Element } From ea2b31bd6e5715518071712480b09e5d85c625c5 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 15:23:17 +0800 Subject: [PATCH 16/26] =?UTF-8?q?feat:=20view=E5=B1=82=EF=BC=8C=E4=BA=8B?= =?UTF-8?q?=E4=BB=B6handler=EF=BC=8C=E8=A7=A6=E5=8F=91=E5=AF=B9=E5=BA=94?= =?UTF-8?q?=E7=9A=84CellView(NodeView/EdgeView)=E7=9A=84=E4=BA=8B=E4=BB=B6?= =?UTF-8?q?=E7=9B=91=E5=90=AC?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/graph/view.ts | 1 + packages/x6/src/view/edge.ts | 2 ++ packages/x6/src/view/node.ts | 3 +++ 3 files changed, 6 insertions(+) diff --git a/packages/x6/src/graph/view.ts b/packages/x6/src/graph/view.ts index 423b6b18ef..60ab72c4d0 100644 --- a/packages/x6/src/graph/view.ts +++ b/packages/x6/src/graph/view.ts @@ -133,6 +133,7 @@ export class GraphView extends View { const localPoint = this.graph.snapToGrid(e.clientX, e.clientY) // 捕捉到元素(NodeView、EdgeView),触发对应view中的onclick事件 if (view) { + // view事件触发 view.onClick(e, localPoint.x, localPoint.y) } else { // 没有找到,触发空白点击事件 diff --git a/packages/x6/src/view/edge.ts b/packages/x6/src/view/edge.ts index 98502e2f17..864ac81e44 100644 --- a/packages/x6/src/view/edge.ts +++ b/packages/x6/src/view/edge.ts @@ -1403,7 +1403,9 @@ export class EdgeView< } onClick(e: Dom.ClickEvent, x: number, y: number) { + // 触发CellView的click事件 graph.on('cell:click') super.onClick(e, x, y) + // 触发EdgeView的click事件 graph.on('edge:click') this.notify('edge:click', this.getEventArgs(e, x, y)) } diff --git a/packages/x6/src/view/node.ts b/packages/x6/src/view/node.ts index 07ddcc5760..3d9fb80b86 100644 --- a/packages/x6/src/view/node.ts +++ b/packages/x6/src/view/node.ts @@ -518,8 +518,11 @@ export class NodeView< } onClick(e: Dom.ClickEvent, x: number, y: number) { + // 触发CellView的click事件 graph.on('cell:click') super.onClick(e, x, y) + // 触发NodeView的click事件 graph.on('node:click'),这个监听会更具体 this.notify('node:click', this.getEventArgs(e, x, y)) + // 触发连接桩点击事件 this.notifyPortEvent('node:port:click', e, { x, y }) } From 8b9675efd6386e448b2eb51eeba902502a4a6c38 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 17:38:54 +0800 Subject: [PATCH 17/26] =?UTF-8?q?feat:=E4=BA=8B=E4=BB=B6=E8=A7=A6=E5=8F=91?= =?UTF-8?q?=EF=BC=8C=E9=80=9A=E8=BF=87store=E5=AD=98=E5=82=A8=E4=B9=8B?= =?UTF-8?q?=E5=89=8Devent=E4=B8=AD=E7=9A=84data=E6=95=B0=E6=8D=AE=EF=BC=8C?= =?UTF-8?q?=E4=B8=8B=E6=AC=A1=E6=93=8D=E4=BD=9C=E5=90=8C=E7=B1=BB=E5=9E=8B?= =?UTF-8?q?=E4=BA=8B=E4=BB=B6=E8=83=BD=E5=8F=96=E5=88=B0=E7=9B=B8=E5=BA=94?= =?UTF-8?q?=E6=95=B0=E6=8D=AE?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/dom/event/core.ts | 10 ++++++---- packages/x6-common/src/dom/event/store.ts | 1 + packages/x6-common/src/dom/event/util.ts | 2 ++ 3 files changed, 9 insertions(+), 4 deletions(-) diff --git a/packages/x6-common/src/dom/event/core.ts b/packages/x6-common/src/dom/event/core.ts index d36adf52d0..41bb2ceda8 100644 --- a/packages/x6-common/src/dom/event/core.ts +++ b/packages/x6-common/src/dom/event/core.ts @@ -39,7 +39,7 @@ export namespace Core { // if (!Util.isValidSelector(elem, selector)) { // throw new Error('Delegate event with invalid selector.') // } - + // 确保elem进入store,后续dispatch 中Util.getHandlerQueue(elem, event)会用到 const store = Store.ensure(elem) // Ensure the main handle @@ -64,6 +64,8 @@ export namespace Core { } let type = originType + // 事件名称hook,通过事件名转换成其它事件名称;EventHook.register方法注册转换逻辑 + // 例如mouseenter 会被转换为mouseover 事件 let hook = EventHook.get(type) // If selector defined, determine special event type, otherwise given type @@ -109,8 +111,10 @@ export namespace Core { Util.setHandlerId(handleObj.handler, guid) } + // Add to the element's handler list, delegates in front if (selector) { + // 有选择器的情况下(理解为具体的cell选择器),将delegateCount+1,代表有多少个具体cell选择器 bag.handlers.splice(bag.delegateCount, 0, handleObj) bag.delegateCount += 1 } else { @@ -244,9 +248,7 @@ export namespace Core { event.rnamespace == null || (handleObj.namespace && event.rnamespace.test(handleObj.namespace)) ) { - if(handleObj.originType === 'mousedown'){ - console.log('handleObj', handleObj.originType, handleObj) - } + // 恢复之前的操作数据 event.handleObj = handleObj event.data = handleObj.data diff --git a/packages/x6-common/src/dom/event/store.ts b/packages/x6-common/src/dom/event/store.ts index 515ead168a..202f075937 100644 --- a/packages/x6-common/src/dom/event/store.ts +++ b/packages/x6-common/src/dom/event/store.ts @@ -27,6 +27,7 @@ export namespace Store { export function ensure(target: EventTarget) { if (!cache.has(target)) { + // 构建以element为key 的events对象 cache.set(target, { events: Object.create(null) }) } return cache.get(target)! diff --git a/packages/x6-common/src/dom/event/util.ts b/packages/x6-common/src/dom/event/util.ts index 9f1e788fc4..3ef3eb513e 100644 --- a/packages/x6-common/src/dom/event/util.ts +++ b/packages/x6-common/src/dom/event/util.ts @@ -100,6 +100,7 @@ export namespace Util { export namespace Util { export function getHandlerQueue(elem: Store.EventTarget, event: EventObject) { const queue = [] + // Store中使用WeakMap进行cache缓存,缓存相关数据 const store = Store.get(elem) const bag = store && store.events && store.events[event.type] const handlers = (bag && bag.handlers) || [] @@ -161,6 +162,7 @@ export namespace Util { // Add the remaining (directly-bound) handlers if (delegateCount < handlers.length) { + // 截取delegateCount之后的handler,也就是用于具体cell选择器的handler queue.push({ elem, handlers: handlers.slice(delegateCount) }) } From 630895074e575459620adbc9d65db6f6ca56218e Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 17:40:07 +0800 Subject: [PATCH 18/26] =?UTF-8?q?feat:=E5=85=B3=E4=BA=8E=E8=87=AA=E5=AE=9A?= =?UTF-8?q?=E4=B9=89=E4=BA=8B=E4=BB=B6event=EF=BC=8C=E5=AD=98=E5=82=A8?= =?UTF-8?q?=E8=87=AA=E5=AE=9A=E4=B9=89data=E7=9A=84=E9=83=A8=E5=88=86?= =?UTF-8?q?=E5=9C=BA=E6=99=AF?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/view/node.ts | 20 ++++++++++++++++---- 1 file changed, 16 insertions(+), 4 deletions(-) diff --git a/packages/x6/src/view/node.ts b/packages/x6/src/view/node.ts index 3d9fb80b86..17b8621a66 100644 --- a/packages/x6/src/view/node.ts +++ b/packages/x6/src/view/node.ts @@ -1055,21 +1055,28 @@ export class NodeView< return area || null } - + /** + * 开始节点拖拽 + * @param e + * @param x + * @param y + * @returns + */ protected startNodeDragging(e: Dom.MouseDownEvent, x: number, y: number) { const targetView = this.getDelegatedView() if (targetView == null || !targetView.can('nodeMovable')) { return this.notifyUnhandledMouseDown(e, x, y) } - + // 往自定义event中传数据 this.setEventData(e, { targetView, action: 'move', }) - + // 当前cell的位置 const position = Point.create(targetView.cell.getPosition()) targetView.setEventData(e, { moving: false, + // 当前点击位置相对于cell的位置的偏移量 offset: position.diff(x, y), restrict: this.getRestrictArea(targetView), }) @@ -1107,7 +1114,12 @@ export class NodeView< this.processEmbedding(e, data) } } - + /** + * 结束时间拖拽 + * @param e + * @param x + * @param y + */ protected stopNodeDragging(e: Dom.MouseUpEvent, x: number, y: number) { const data = this.getEventData(e) if (data.embedding) { From 0e97ca47b76cf49f4db03dd9b208c05a570a92e5 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 17:41:39 +0800 Subject: [PATCH 19/26] =?UTF-8?q?feat:=20CellView=EF=BC=88NodeView/EdgeVie?= =?UTF-8?q?w=EF=BC=89=E4=B8=AD=E5=8C=85=E5=90=ABCell=EF=BC=88Node/Edge?= =?UTF-8?q?=EF=BC=89?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/view/cell.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/x6/src/view/cell.ts b/packages/x6/src/view/cell.ts index 970c1135f7..9101389b84 100644 --- a/packages/x6/src/view/cell.ts +++ b/packages/x6/src/view/cell.ts @@ -102,7 +102,7 @@ export class CellView< constructor(cell: Entity, options: Partial = {}) { super() - + // CellView中包含 Model中的Node、Edge this.cell = cell this.options = this.ensureOptions(options) this.graph = this.options.graph From 7478b0b660bca873f109754588e982ed3598c55c Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 17:46:08 +0800 Subject: [PATCH 20/26] =?UTF-8?q?feat:=20Model=E4=B8=AD=E7=9A=84Collection?= =?UTF-8?q?=E6=B7=BB=E5=8A=A0Cell=EF=BC=8C=E9=80=9A=E8=BF=87=E4=BA=8B?= =?UTF-8?q?=E4=BB=B6=E5=91=8A=E7=9F=A5Scheduler=E8=B0=83=E5=BA=A6=E5=99=A8?= =?UTF-8?q?=EF=BC=8C=E8=BF=9B=E8=A1=8C=E7=9B=B8=E5=BA=94=E7=9A=84CellView?= =?UTF-8?q?=E7=9A=84=E5=88=9B=E5=BB=BA=EF=BC=8C=E5=B9=B6=E9=80=9A=E8=BF=87?= =?UTF-8?q?queueJob=E4=BB=BB=E5=8A=A1=E9=98=9F=E5=88=97=E8=BF=9B=E8=A1=8C?= =?UTF-8?q?=E8=A7=86=E5=9B=BE=E6=B8=B2=E6=9F=93?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/model/collection.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/x6/src/model/collection.ts b/packages/x6/src/model/collection.ts index 38c38918cd..e612b0b041 100644 --- a/packages/x6/src/model/collection.ts +++ b/packages/x6/src/model/collection.ts @@ -105,6 +105,8 @@ export class Collection extends Basecoat { } this.trigger('added', args) if (!localOptions.dryrun) { + // 元素添加后触发added事件,从而创建对应的CellView对象(NodeView、EdgeView ) + // 会触发model.notify('cell:added')事件,Scheduler 监听了model.on('cell:added')事件,从而在Scheduler中创建对应的CellView对象(NodeView、EdgeView ) cell.notify('added', { ...args }) } }) From b85ca7fb2848bbb7fbca58e57b5862724d6de797 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 17:46:22 +0800 Subject: [PATCH 21/26] =?UTF-8?q?feat:=E8=B0=83=E8=AF=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../src/pages/minimap/index.tsx | 35 +++++++++++-------- packages/x6/src/graph/graph.ts | 2 +- 2 files changed, 22 insertions(+), 15 deletions(-) diff --git a/examples/x6-example-features/src/pages/minimap/index.tsx b/examples/x6-example-features/src/pages/minimap/index.tsx index 8d65106023..c3f528141e 100644 --- a/examples/x6-example-features/src/pages/minimap/index.tsx +++ b/examples/x6-example-features/src/pages/minimap/index.tsx @@ -39,7 +39,6 @@ export default class Example extends React.Component { // pannable: false, // }), // ) - this.graph.addNode({ x: 200, @@ -102,20 +101,28 @@ export default class Example extends React.Component { }, }) + // this.graph.use( + // new MiniMap({ + // container: this.minimapContainer, + // width: 400, + // height: 400, + // padding: 1, + // preserveAspectRatio: false, + // graphOptions: { + // width: 400, + // height: 400, + // }, + // }), + // ) - this.graph.use( - new MiniMap({ - container: this.minimapContainer, - width: 400, - height: 400, - padding: 1, - preserveAspectRatio:false, - graphOptions:{ - width:400, - height:400 - } - }), - ) + + this.graph.on('cell:click', ({ e, x, y, cell, view }) => { + console.log('graph cell:click') + }) + + this.graph.on('node:click',()=>{ + console.log('graph node:click') + }) } diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index e170d43c04..1f2b45161b 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -827,7 +827,7 @@ export class Graph extends Basecoat { // #region coord /** - * 捕捉到栅格?(是否捕捉到栅格上的元素) + * 当前事件在栅格中的坐标 * @param x * @param y * @returns From dc6ca1a089f16f3ba347dccc05e60cbfeb988c8f Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Tue, 20 Aug 2024 18:31:17 +0800 Subject: [PATCH 22/26] =?UTF-8?q?feat=EF=BC=9A=E4=BA=8B=E4=BB=B6=E5=A7=94?= =?UTF-8?q?=E6=89=98=EF=BC=8C=E8=A7=A6=E5=8F=91=E6=B5=81=E7=A8=8B=E5=A4=87?= =?UTF-8?q?=E6=B3=A8?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/graph/view.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/packages/x6/src/graph/view.ts b/packages/x6/src/graph/view.ts index 60ab72c4d0..a59fcaa261 100644 --- a/packages/x6/src/graph/view.ts +++ b/packages/x6/src/graph/view.ts @@ -51,6 +51,9 @@ export class GraphView extends View { delegateEvents() { // 委托监听this.container dom事件上的交互事件;在对应的handler上再去触发graph上的事件 + // 1. GraphView中将ctor.events事件都绑定在this.container上,回调事件根据配置,指向GraphView上具体的函数 + // 2.this.container中的DOM事件触发之后,根据事件信息,找到对应的CellView子类,然后触发CellView上的事件 + // 3.CellView中的notify,会触发this.graph.trigger事件,从而将事件传递给graph;业务层如果监听了graph上的事件(例如 this.graph.on('cell:added'))),也会收到事件 const ctor = this.constructor as typeof GraphView super.delegateEvents(ctor.events) return this From a14c3c58ac8968596fe4ddeed5e863aba9d905f9 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Wed, 21 Aug 2024 16:43:49 +0800 Subject: [PATCH 23/26] =?UTF-8?q?feat:=20coord=20=E5=9D=90=E6=A0=87?= =?UTF-8?q?=E8=BD=AC=E6=8D=A2?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6/src/graph/graph.ts | 12 +++++++++--- 1 file changed, 9 insertions(+), 3 deletions(-) diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index 1f2b45161b..fe558303e4 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -52,7 +52,7 @@ export class Graph extends Basecoat { this.css = new Css(this) this.view = new GraphView(this) this.defs = new Defs(this) - // 坐标?(用于捕捉交互的元素?) + // 坐标(用于画布与客户端/浏览器的坐标转换) this.coord = new Coord(this) this.transform = new Transform(this) this.highlight = new Highlight(this) @@ -889,7 +889,10 @@ export class Graph extends Basecoat { return this.coord.localToPagePoint(x, y) } - + /** + * 画布坐标转换为客户端坐标(浏览器坐标) + * @param rect + */ clientToLocal(rect: Rectangle.RectangleLike): Rectangle clientToLocal(x: number, y: number, width: number, height: number): Rectangle clientToLocal(p: Point.PointLike): Point @@ -915,7 +918,10 @@ export class Graph extends Basecoat { return this.coord.clientToLocalPoint(x, y) } - + /** + * 浏览器坐标转换为画布端坐标 + * @param rect + */ localToClient(rect: Rectangle.RectangleLike): Rectangle localToClient(x: number, y: number, width: number, height: number): Rectangle localToClient(p: Point.PointLike): Point From f2c8d745beb33bc29534ccc2e4e0f870e51d89e0 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Thu, 22 Aug 2024 09:47:47 +0800 Subject: [PATCH 24/26] =?UTF-8?q?feat:=20dnd=EF=BC=88drag=20and=20drop?= =?UTF-8?q?=EF=BC=89=20=E6=8B=96=E6=8B=BD=E8=BF=9B=E5=85=A5=E6=8F=92?= =?UTF-8?q?=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-plugin-dnd/src/index.ts | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/packages/x6-plugin-dnd/src/index.ts b/packages/x6-plugin-dnd/src/index.ts index 67c768ab33..9d863e3b8a 100644 --- a/packages/x6-plugin-dnd/src/index.ts +++ b/packages/x6-plugin-dnd/src/index.ts @@ -15,6 +15,10 @@ import { import { alignPoint } from 'dom-align' import { content } from './style/raw' +/** + * 拖拽键入画布插件 + * drag and drop => dnd + */ export class Dnd extends View implements Graph.Plugin { public name = 'dnd' From bc11c59440e27f85fc06a1c19801e761d3e45624 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Thu, 22 Aug 2024 20:03:24 +0800 Subject: [PATCH 25/26] =?UTF-8?q?feat:=20coord=E5=9D=90=E6=A0=87=E7=B3=BB?= =?UTF-8?q?=E7=BB=9F?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-common/src/dom/matrix.ts | 1 + packages/x6/src/graph/coord.ts | 126 +++++++++++++++++++++++++-- packages/x6/src/graph/graph.ts | 22 +++-- packages/x6/src/graph/view.ts | 1 + packages/x6/src/util/index.ts | 5 +- 5 files changed, 140 insertions(+), 15 deletions(-) diff --git a/packages/x6-common/src/dom/matrix.ts b/packages/x6-common/src/dom/matrix.ts index c9922a9d7a..e62400550a 100644 --- a/packages/x6-common/src/dom/matrix.ts +++ b/packages/x6-common/src/dom/matrix.ts @@ -31,6 +31,7 @@ export interface Scale { } /** + * 构建svg中的Point点对象 * Returns a SVG point object initialized with the `x` and `y` coordinates. * @see https://developer.mozilla.org/en/docs/Web/API/SVGPoint */ diff --git a/packages/x6/src/graph/coord.ts b/packages/x6/src/graph/coord.ts index 0c951ca37a..8c0f7ab97c 100644 --- a/packages/x6/src/graph/coord.ts +++ b/packages/x6/src/graph/coord.ts @@ -1,10 +1,24 @@ -import { Dom } from '@antv/x6-common' -import { Point, Rectangle } from '@antv/x6-geometry' +import { Dom } from '../../../x6-common/src/index' +import { Point, Rectangle } from '../../../x6-geometry/src/index' import { Base } from './base' import { Util } from '../util' +/** + * X6坐标系概念 + * local:画布本地坐标系,默认情况下和 graph 坐标系一致,但是会随着画布的缩放和平移发生改变。画布中所有节点的坐标都是以 local 坐标系为准。初始(0,0)坐标与graph视口位置对齐,左偏移会变为负值。 (画布内容坐标系) + * graph:画布坐标系,也就是我们看到的画布视口,它不会随着画布缩放和平移而改变。(画布视口坐标系) + * client:浏览器坐标系,鼠标事件中的 e.clinetX、e.clientY 就是相对于浏览器坐标系。(浏览器视口坐标系) + * page:页面坐标系,与 client 相比,page 会考虑页面水平和垂直方向滚动。鼠标事件中的 e.pageX、e.pageY 就是相对于页面坐标系。(浏览器内容坐标系) + */ + + export class CoordManager extends Base { + /** + * 获取state元素的"屏幕坐标转换矩阵"(Screen Coordinate Transformation Matrix) + * @returns + */ getClientMatrix() { + // getScreenCTM: 获取state元素的"屏幕坐标转换矩阵"(Screen Coordinate Transformation Matrix) return Dom.createSVGMatrix(this.view.stage.getScreenCTM()) } @@ -24,7 +38,12 @@ export class CoordManager extends Base { // see: https://developer.mozilla.org/en-US/docs/Web/API/Element/getBoundingClientRect return this.getClientOffset().translate(window.scrollX, window.scrollY) } - + /** + * 将浏览器坐标转换为画布本地坐标并对齐到画布网格 + * @param x + * @param y + * @returns + */ snapToGrid(x: number | Point | Point.PointLike, y?: number) { const p = typeof x === 'number' @@ -88,9 +107,83 @@ export class CoordManager extends Base { const graphPoint = Point.create(x, y) return Util.transformPoint(graphPoint, this.graph.matrix().inverse()) } - + /** + * 将页面的浏览器坐标转换画布本地坐标 + * @param x + * @param y + * @returns + */ clientToLocalPoint(x: number | Point | Point.PointLike, y?: number) { const clientPoint = Point.create(x, y) + // 先获取state元素的"屏幕坐标转换矩阵"(Screen Coordinate Transformation Matrix) + // 再取得逆矩阵 (local to client使用正矩阵,client to local使用逆矩阵inverse) + // 使用client.x、client.y 构建一个svgPoint,然后使用svgPoint.matrixTransform(matrix) + + /** + * client to local 为什么用逆矩 + * 前提: ScreenCTM是svg中元素相对于屏幕的缩放、倾斜、偏移位置 + * 1. 在svg不进行缩放、倾斜、偏移的情况下,CTM偏移位置就是svg左上角位置;点击svg区域屏幕,计算点击位置在svg的坐标就是 clien.x - ctm.x + * 2. 当svg中的元素进行了偏移(比如向右偏移100px),CTM偏移位置就是svg左上角位置 + 100px;那点击svg区域屏幕(例如svg左上角定点),点击位置在svg的坐标就是 client.x - ctm.x = -100px + * 3. 所以在不缩放,只偏移的情况下;local的计算位置就是 CTX的偏移位置 与屏幕点击client位置的简单加减 + * 4. 在有缩放的情况下,再进行缩放的比例计算 + * 5. 那至于为什么要用逆矩阵;元素向右偏移100px,正常情况下我们的计算逻辑是 client.x - ctm.x;(一般不这么计算) + * 6. 但是按正常的理解逻辑,[2,0,0,2, 10, 10]的逆矩阵是 [-2,0,0,-2, -10, -10];然后根据这些缩放、偏移进行计算 + * 7. 但这样的计算,给定一个client位置信息,在clientToLocal, 再由localToClient时,位置信息是对不上的;实际上[2,0,0,2, 10,10]的逆矩阵为[0.5, 0, 0, 0.5, -5, -5] + */ + + /** + * demo: svg 右平移100px、放大2倍(放大到2倍) + * svg: screenCTM: [2,0,0,2, 100, 0] + * local: [50, 50] + * + * 正常计算: + * clientToLocal: + * 缩放后local位置: [50 * 2, 50 *2] = [100, 100] + * 平移移后local位置: [100 + 100, 100] + * 所以local点在client屏幕坐标位置: [200, 100] + * + * 使用反推,计算local位置 + * 撤销平移: [200 -100, 100 - 0] = [100, 100] + * 撤销缩放: [100 / 2, 100 / 2] = [50, 50] + * + */ + + /** + * + * 手动反推在简单变换时是可行的,但在复杂变换场景中变得不切实际 + * [2,0,0,2, 100, 0] 对应的矩阵为 + * 2 0 100 + * 0 2 0 + * 0 0 1 + * 固定格式: + * a c e + * b d f + * 0 0 1 + * + * 逆矩阵为(逆矩阵计算公式参见印象笔记中) + * 0.5 0 -50 + * 0 0.5 0 + * 0 0 1 + * + * 0.5 0 -50 + * [200, 100] * 0 0.5 0 = [50, 50] + * 0 0 1 + * + * 200 0.5 0 -50 + * 100 * 0 0.5 0 = [50, 50] + * 1 0 0 1 + * + * 逐行的每个位置元素与 200、100、1 相乘后相加 + * 第一行: + * 0.5 * 200 + 0 * 100 + -50 = 50 + * 第二行 + * 0 * 200 + 0.5 * 100 + 0 = 50 + * 第三行 + * 0 * 200 + 0 * 100 + 1 = 1 + * [50, 50, 1] => [50, 50] + * + */ + return Util.transformPoint(clientPoint, this.getClientMatrix().inverse()) } @@ -101,7 +194,12 @@ export class CoordManager extends Base { this.graph.matrix().multiply(this.getClientMatrix().inverse()), ) } - + /** + * 将页面点坐标转换为画布本地点坐标 + * @param x + * @param y + * @returns + */ pageToLocalPoint(x: number | Point | Point.PointLike, y?: number) { const pagePoint = Point.create(x, y) const graphPoint = pagePoint.diff(this.getPageOffset()) @@ -117,7 +215,14 @@ export class CoordManager extends Base { const graphRect = Rectangle.create(x, y, width, height) return Util.transformRectangle(graphRect, this.graph.matrix().inverse()) } - + /** + * 画布矩形转换为浏览器矩形 + * @param x + * @param y + * @param width + * @param height + * @returns + */ clientToLocalRect( x: number | Rectangle | Rectangle.RectangleLike, y?: number, @@ -140,7 +245,14 @@ export class CoordManager extends Base { this.graph.matrix().multiply(this.getClientMatrix().inverse()), ) } - + /** + * 将页面矩形坐标转换为画布本地矩形坐标 + * @param x + * @param y + * @param width + * @param height + * @returns + */ pageToLocalRect( x: number | Rectangle | Rectangle.RectangleLike, y?: number, diff --git a/packages/x6/src/graph/graph.ts b/packages/x6/src/graph/graph.ts index fe558303e4..a7d3f9b7f5 100644 --- a/packages/x6/src/graph/graph.ts +++ b/packages/x6/src/graph/graph.ts @@ -827,7 +827,7 @@ export class Graph extends Basecoat { // #region coord /** - * 当前事件在栅格中的坐标 + * 将浏览器坐标转换为画布本地坐标并对齐到画布网格 * @param x * @param y * @returns @@ -837,7 +837,10 @@ export class Graph extends Basecoat { snapToGrid(x: number | Point.PointLike, y?: number) { return this.coord.snapToGrid(x, y) } - + /** + * 将页面坐标转换为画布本地坐标 + * @param rect + */ pageToLocal(rect: Rectangle.RectangleLike): Rectangle pageToLocal(x: number, y: number, width: number, height: number): Rectangle pageToLocal(p: Point.PointLike): Point @@ -863,7 +866,10 @@ export class Graph extends Basecoat { return this.coord.pageToLocalPoint(x, y) } - + /** + * 将画布本地坐标转换为页面坐标 + * @param rect + */ localToPage(rect: Rectangle.RectangleLike): Rectangle localToPage(x: number, y: number, width: number, height: number): Rectangle localToPage(p: Point.PointLike): Point @@ -890,7 +896,7 @@ export class Graph extends Basecoat { return this.coord.localToPagePoint(x, y) } /** - * 画布坐标转换为客户端坐标(浏览器坐标) + * 将页面的浏览器坐标转换画布本地坐标 * @param rect */ clientToLocal(rect: Rectangle.RectangleLike): Rectangle @@ -919,7 +925,7 @@ export class Graph extends Basecoat { return this.coord.clientToLocalPoint(x, y) } /** - * 浏览器坐标转换为画布端坐标 + * 将页面的浏览器坐标转换画布本地坐标 * @param rect */ localToClient(rect: Rectangle.RectangleLike): Rectangle @@ -949,6 +955,7 @@ export class Graph extends Basecoat { } /** + * 将画布本地坐标转换为画布坐标 * Transform the rectangle `rect` defined in the local coordinate system to * the graph coordinate system. */ @@ -989,7 +996,10 @@ export class Graph extends Basecoat { return this.coord.localToGraphPoint(x, y) } - + /** + * 将画布坐标转换为画布本地坐标 + * @param rect + */ graphToLocal(rect: Rectangle.RectangleLike): Rectangle graphToLocal(x: number, y: number, width: number, height: number): Rectangle graphToLocal(p: Point.PointLike): Point diff --git a/packages/x6/src/graph/view.ts b/packages/x6/src/graph/view.ts index a59fcaa261..c5d7d23e32 100644 --- a/packages/x6/src/graph/view.ts +++ b/packages/x6/src/graph/view.ts @@ -37,6 +37,7 @@ export class GraphView extends View { this.defs = selectors.defs as SVGDefsElement this.viewport = selectors.viewport as SVGGElement this.primer = selectors.primer as SVGGElement + // svg stage舞台元素 this.stage = selectors.stage as SVGGElement this.decorator = selectors.decorator as SVGGElement this.overlay = selectors.overlay as SVGGElement diff --git a/packages/x6/src/util/index.ts b/packages/x6/src/util/index.ts index bade8cea8c..78863943bc 100644 --- a/packages/x6/src/util/index.ts +++ b/packages/x6/src/util/index.ts @@ -5,8 +5,8 @@ import { Polyline, Ellipse, Path, -} from '@antv/x6-geometry' -import { Dom, PointData, PointLike } from '@antv/x6-common' +} from '../../../x6-geometry/src/index' +import { Dom, PointData, PointLike } from '../../../x6-common/src/index' import { normalize } from '../registry/marker/util' export namespace Util { @@ -15,6 +15,7 @@ export namespace Util { * Transforms point by an SVG transformation represented by `matrix`. */ export function transformPoint(point: Point.PointLike, matrix: DOMMatrix) { + // 构建一个svg point对象之后,并基于matrix矩阵进行矩阵转换 const ret = Dom.createSVGPoint(point.x, point.y).matrixTransform(matrix) return new Point(ret.x, ret.y) } From cab6e2c56a721973ac3db07f1041327a4a61a9a7 Mon Sep 17 00:00:00 2001 From: yelikang <530331059@qq.com> Date: Fri, 24 Jan 2025 13:57:22 +0800 Subject: [PATCH 26/26] =?UTF-8?q?x6-vue-shape=E6=B8=B2=E6=9F=93=E8=87=AA?= =?UTF-8?q?=E5=AE=9A=E4=B9=89vue=E7=BB=84=E4=BB=B6?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- packages/x6-vue-shape/src/view.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/x6-vue-shape/src/view.ts b/packages/x6-vue-shape/src/view.ts index b839df10ab..07c71cf0e1 100644 --- a/packages/x6-vue-shape/src/view.ts +++ b/packages/x6-vue-shape/src/view.ts @@ -29,8 +29,10 @@ export class VueShapeView extends NodeView { const graph = this.graph if (root) { + // 从registry注册表中获取组件自定义vue组件 const { component } = shapeMaps[node.shape] if (component) { + // 渲染自定义vue组件 if (isVue2) { const Vue = Vue2 as any this.vm = new Vue({