diff --git a/examples/feature-examples/src/pages/graph/index.tsx b/examples/feature-examples/src/pages/graph/index.tsx index a4403b2ed8..52e0ad21a3 100644 --- a/examples/feature-examples/src/pages/graph/index.tsx +++ b/examples/feature-examples/src/pages/graph/index.tsx @@ -547,6 +547,20 @@ export default function BasicNode() { > 删除节点 + 节点面板 diff --git a/packages/core/src/algorithm/rotate.ts b/packages/core/src/algorithm/rotate.ts new file mode 100644 index 0000000000..158c9590d9 --- /dev/null +++ b/packages/core/src/algorithm/rotate.ts @@ -0,0 +1,55 @@ +export interface SimplePoint { + x: number + y: number +} + +/** + * 根据两个点获取中心点坐标 + */ +export function getNewCenter(startPoint: SimplePoint, endPoint: SimplePoint) { + const { x: x1, y: y1 } = startPoint + const { x: x2, y: y2 } = endPoint + const newCenter = { + x: x1 + (x2 - x1) / 2, + y: y1 + (y2 - y1) / 2, + } + return newCenter +} + +/** + * 旋转矩阵公式,可以获取某一个坐标旋转angle后的坐标 + * @param p 当前坐标 + * @param center 旋转中心 + * @param angle 旋转角度(不是弧度) + */ +export function calculatePointAfterRotateAngle( + p: SimplePoint, + center: SimplePoint, + angle: number, +) { + const radian = angleToRadian(angle) + const dx = p.x - center.x + const dy = p.y - center.y + const x = dx * Math.cos(radian) - dy * Math.sin(radian) + center.x + const y = dx * Math.sin(radian) + dy * Math.cos(radian) + center.y + return { + x, + y, + } +} + +/** + * 角度转弧度 + * @param angle 角度 + */ +export function angleToRadian(angle: number) { + return (angle * Math.PI) / 180 +} + +/** + * 弧度转角度 + * @param radian 弧度 + */ +export function radianToAngle(radian: number) { + return (radian / Math.PI) * 180 +} diff --git a/packages/core/src/model/node/BaseNodeModel.ts b/packages/core/src/model/node/BaseNodeModel.ts index 6489805b33..1a22e9df06 100644 --- a/packages/core/src/model/node/BaseNodeModel.ts +++ b/packages/core/src/model/node/BaseNodeModel.ts @@ -687,6 +687,10 @@ export class BaseNodeModel

this.y = this.y + deltaY this.text && this.moveText(0, deltaY) } + if (isAllowMoveX || isAllowMoveY) { + // 更新x和y的同时也要更新对应的transform旋转矩阵(依赖x、y) + this.rotate = this._rotate + } return isAllowMoveX || isAllowMoveY } diff --git a/packages/core/src/util/resize.ts b/packages/core/src/util/resize.ts index abc46b538f..732c65beb5 100644 --- a/packages/core/src/util/resize.ts +++ b/packages/core/src/util/resize.ts @@ -5,6 +5,70 @@ import { EventType } from '../constant' import ResizeInfo = ResizeControl.ResizeInfo import ResizeNodeData = ResizeControl.ResizeNodeData +import { + calculatePointAfterRotateAngle, + getNewCenter, + radianToAngle, +} from '../algorithm/rotate' +import type { SimplePoint } from '../algorithm/rotate' + +function recalcRotatedResizeInfo( + pct: number, + resizeInfo: ResizeInfo, + rotate: number, + anchorX: number, + anchorY: number, + oldCenterX: number, + oldCenterY: number, + freezeWidth = false, + freezeHeight = false, +) { + // 假设我们触摸的点是右下角的anchor + const { deltaX, deltaY, width: oldWidth, height: oldHeight } = resizeInfo + const angle = radianToAngle(rotate) + + // 右下角的anchor + const startZeroTouchAnchorPoint = { + x: anchorX, // control锚点的坐标x + y: anchorY, // control锚点的坐标y + } + const oldCenter = { x: oldCenterX, y: oldCenterY } + // 右下角的anchor坐标(transform后的-touchStartPoint) + const startRotatedTouchAnchorPoint = calculatePointAfterRotateAngle( + startZeroTouchAnchorPoint, + oldCenter, + angle, + ) + // 右下角的anchor坐标(transform后的-touchEndPoint) + const endRotatedTouchAnchorPoint = { + x: startRotatedTouchAnchorPoint.x + deltaX, + y: startRotatedTouchAnchorPoint.y + deltaY, + } + // 计算出新的宽度和高度以及新的中心点 + const { + width: newWidth, + height: newHeight, + center: newCenter, + } = calculateWidthAndHeight( + startRotatedTouchAnchorPoint, + endRotatedTouchAnchorPoint, + oldCenter, + angle, + freezeWidth, + freezeHeight, + oldWidth, + oldHeight, + ) + // calculateWidthAndHeight()得到的是整个宽度,比如圆pct=0.5,此时newWidth等于整个圆直径 + resizeInfo.width = newWidth * pct + resizeInfo.height = newHeight * pct + + // BaseNodeModel.resize(deltaX/2, deltaY/2),因此这里要*2 + resizeInfo.deltaX = (newCenter.x - oldCenter.x) * 2 + resizeInfo.deltaY = (newCenter.y - oldCenter.y) * 2 + + return resizeInfo +} /** * 计算 Control 拖动后,节点的高度信息 @@ -20,6 +84,11 @@ export const recalcResizeInfo = ( pct = 1, freezeWidth = false, freezeHeight = false, + rotate = 0, + anchorX: number, + anchorY: number, + oldCenterX: number, + oldCenterY: number, ): ResizeInfo => { const nextResizeInfo = cloneDeep(resizeInfo) let { deltaX, deltaY } = nextResizeInfo @@ -101,6 +170,23 @@ export const recalcResizeInfo = ( return nextResizeInfo } + if (rotate % (2 * Math.PI) !== 0) { + // 角度rotate不为0,则触发另外的计算修正resize的deltaX和deltaY + // 因为rotate不为0的时候,左上角的坐标一直在变化 + // 角度rotate不为0得到的resizeInfo.deltaX仅仅代表中心点的变化,而不是宽度的变化 + return recalcRotatedResizeInfo( + pct, + nextResizeInfo, + rotate, + anchorX, + anchorY, + oldCenterX, + oldCenterY, + freezeWidth, + freezeHeight, + ) + } + // 如果限制了宽/高不变,对应的 width/height 保持一致 switch (index) { case ResizeControlIndex.LEFT_TOP: @@ -213,6 +299,8 @@ export const triggerResizeEvent = ( * @param cancelCallback */ export const handleResize = ({ + x, + y, deltaX, deltaY, index, @@ -232,6 +320,9 @@ export const handleResize = ({ minHeight, maxWidth, maxHeight, + rotate, + x: oldCenterX, + y: oldCenterY, } = nodeModel const isFreezeWidth = minWidth === maxWidth const isFreezeHeight = minHeight === maxHeight @@ -245,12 +336,19 @@ export const handleResize = ({ } const pct = r || (rx && ry) ? 1 / 2 : 1 + const anchorX = x + const anchorY = y const nextSize = recalcResizeInfo( index, resizeInfo, pct, isFreezeWidth, isFreezeHeight, + rotate, + anchorX, + anchorY, + oldCenterX, + oldCenterY, ) // 限制放大缩小的最大最小范围 @@ -264,9 +362,14 @@ export const handleResize = ({ cancelCallback?.() return } - // 如果限制了宽高不变,对应的 x/y 不产生位移 - nextSize.deltaX = isFreezeWidth ? 0 : nextSize.deltaX - nextSize.deltaY = isFreezeWidth ? 0 : nextSize.deltaY + if (rotate % (2 * Math.PI) == 0 || PCTResizeInfo) { + // rotate!==0并且不是PCTResizeInfo时,即使是isFreezeWidth||isFreezeHeight + // recalcRotatedResizeInfo()计算出来的中心点会发生变化 + + // 如果限制了宽高不变,对应的 x/y 不产生位移 + nextSize.deltaX = isFreezeWidth ? 0 : nextSize.deltaX + nextSize.deltaY = isFreezeHeight ? 0 : nextSize.deltaY + } const preNodeData = nodeModel.getData() const curNodeData = nodeModel.resize(nextSize) @@ -284,3 +387,114 @@ export const handleResize = ({ graphModel, ) } + +export function calculateWidthAndHeight( + startRotatedTouchAnchorPoint: SimplePoint, + endRotatedTouchAnchorPoint: SimplePoint, + oldCenter: SimplePoint, + angle: number, + freezeWidth = false, + freezeHeight = false, + oldWidth: number, + oldHeight: number, +) { + // 假设目前触摸的是右下角的anchor + // 计算出来左上角的anchor坐标,resize过程左上角的anchor坐标保持不变 + const freezePoint: SimplePoint = { + x: oldCenter.x - (startRotatedTouchAnchorPoint.x - oldCenter.x), + y: oldCenter.y - (startRotatedTouchAnchorPoint.y - oldCenter.y), + } + // 【touchEndPoint】右下角 + freezePoint左上角 计算出新的中心点 + let newCenter = getNewCenter(freezePoint, endRotatedTouchAnchorPoint) + + // 得到【touchEndPoint】右下角-没有transform的坐标 + let endZeroTouchAnchorPoint: SimplePoint = calculatePointAfterRotateAngle( + endRotatedTouchAnchorPoint, + newCenter, + -angle, + ) + + // ---------- 使用transform之前的坐标计算出新的width和height ---------- + + // 得到左上角---没有transform的坐标 + let zeroFreezePoint: SimplePoint = calculatePointAfterRotateAngle( + freezePoint, + newCenter, + -angle, + ) + + if (freezeWidth) { + // 如果固定width,那么不能单纯使用endZeroTouchAnchorPoint.x=startZeroTouchAnchorPoint.x + // 因为去掉transform的左上角不一定是重合的,我们要保证的是transform后的左上角重合 + const newWidth = Math.abs(endZeroTouchAnchorPoint.x - zeroFreezePoint.x) + const widthDx = newWidth - oldWidth + + // 点击的是左边锚点,是+widthDx/2,点击是右边锚点,是-widthDx/2 + if (newCenter.x > endZeroTouchAnchorPoint.x) { + // 当前触摸的是左边锚点 + newCenter.x = newCenter.x + widthDx / 2 + } else { + // 当前触摸的是右边锚点 + newCenter.x = newCenter.x - widthDx / 2 + } + } + if (freezeHeight) { + const newHeight = Math.abs(endZeroTouchAnchorPoint.y - zeroFreezePoint.y) + const heightDy = newHeight - oldHeight + if (newCenter.y > endZeroTouchAnchorPoint.y) { + // 当前触摸的是上边锚点 + newCenter.y = newCenter.y + heightDy / 2 + } else { + newCenter.y = newCenter.y - heightDy / 2 + } + } + + if (freezeWidth || freezeHeight) { + // 如果调整过transform之前的坐标,那么transform后的坐标也会改变,那么算出来的newCenter也得调整 + // 由于无论如何rotate,中心点都是不变的,因此我们可以使用transform之前的坐标算出新的中心点 + const nowFreezePoint = calculatePointAfterRotateAngle( + zeroFreezePoint, + newCenter, + angle, + ) + + // 得到当前新rect的左上角与实际上transform后的左上角的偏移量 + const dx = nowFreezePoint.x - freezePoint.x + const dy = nowFreezePoint.y - freezePoint.y + + // 修正不使用transform的坐标: 左上角、右下角、center + newCenter.x = newCenter.x - dx + newCenter.y = newCenter.y - dy + zeroFreezePoint = calculatePointAfterRotateAngle( + freezePoint, + newCenter, + -angle, + ) + endZeroTouchAnchorPoint = { + x: newCenter.x - (zeroFreezePoint.x - newCenter.x), + y: newCenter.y - (zeroFreezePoint.y - newCenter.y), + } + } + + // transform之前的坐标的左上角+右下角计算出宽度和高度 + let width = Math.abs(endZeroTouchAnchorPoint.x - zeroFreezePoint.x) + let height = Math.abs(endZeroTouchAnchorPoint.y - zeroFreezePoint.y) + + // ---------- 使用transform之前的坐标计算出新的width和height ---------- + + if (freezeWidth) { + // 理论计算出来的width应该等于oldWidth + // 但是有误差,比如oldWidth = 100; newWidth=100.000000000001 + // 会在handleResize()限制放大缩小的最大最小范围中被阻止滑动 + width = oldWidth + } + if (freezeHeight) { + height = oldHeight + } + + return { + width, + height, + center: newCenter, + } +} diff --git a/packages/core/src/view/Control.tsx b/packages/core/src/view/Control.tsx index 4f96b694da..9fbce9c5b1 100644 --- a/packages/core/src/view/Control.tsx +++ b/packages/core/src/view/Control.tsx @@ -56,6 +56,10 @@ export class ResizeControl extends Component< }) } + componentWillUnmount() { + this.dragHandler.cancelDrag() + } + updateEdgePointByAnchors = () => { // https://github.com/didi/LogicFlow/issues/807 // https://github.com/didi/LogicFlow/issues/875 @@ -241,10 +245,12 @@ export class ResizeControl extends Component< resizeNode = ({ deltaX, deltaY }: VectorData) => { const { index } = this - const { model, graphModel } = this.props + const { model, graphModel, x, y } = this.props // DONE: 调用每个节点中更新缩放时的方法 updateNode 函数,用来各节点缩放的方法 handleResize({ + x, + y, deltaX, deltaY, index,