Skip to content

Commit

Permalink
fix(core): 修复issue反馈的问题 (didi#1972)
Browse files Browse the repository at this point in the history
* 修复初始化画布时因为节点吸附网格导致的节点与边错位问题 close didi#1954
* line 和 polyline 增加getPath方法 close didi#1929
* 增加节点和边的focus和blur事件上报 close didi#1917

* 节点和边focus和blur事件逻辑优化:处理浏览器默认focus样式问题&增加类型声明
* 修复polyline points内容变化导致的无法解析问题 & 优化初始化边对齐相关逻辑
* line getPath返回数据格式优化

---------

Co-authored-by: liuziqi <[email protected]>
Co-authored-by: 你说呢? <[email protected]>
  • Loading branch information
3 people authored and menghany committed Nov 28, 2024
1 parent 196bfee commit 0198569
Show file tree
Hide file tree
Showing 11 changed files with 129 additions and 6 deletions.
2 changes: 1 addition & 1 deletion packages/core/src/algorithm/outline.ts
Original file line number Diff line number Diff line change
Expand Up @@ -66,7 +66,7 @@ export const getEdgeOutline = (
edge: BaseEdgeModel,
): OutlineInfo | undefined => {
if (edge.modelType === ModelType.LINE_EDGE) {
return getLineOutline(edge)
return getLineOutline(edge as LineEdgeModel)
}
if (edge.modelType === ModelType.POLYLINE_EDGE) {
return getPolylineOutline(edge as PolylineEdgeModel)
Expand Down
4 changes: 4 additions & 0 deletions packages/core/src/constant/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,8 @@ export enum EventType {
NODE_CONTEXTMENU = 'node:contextmenu',
NODE_ROTATE = 'node:rotate',
NODE_RESIZE = 'node:resize',
NODE_FOCUS = 'node:focus',
NODE_BLUR = 'node:blur',

// 节点 properties 变化事件
NODE_PROPERTIES_CHANGE = 'node:properties-change',
Expand All @@ -67,6 +69,8 @@ export enum EventType {
EDGE_DELETE = 'edge:delete',
EDGE_CLICK = 'edge:click',
EDGE_DBCLICK = 'edge:dbclick',
EDGE_FOCUS = 'edge:focus',
EDGE_BLUR = 'edge:blur',

EDGE_MOUSEENTER = 'edge:mouseenter',
EDGE_MOUSELEAVE = 'edge:mouseleave',
Expand Down
16 changes: 16 additions & 0 deletions packages/core/src/event/eventArgs.ts
Original file line number Diff line number Diff line change
Expand Up @@ -188,6 +188,14 @@ interface NodeEventArgs {
*/
properties: Record<string, any>
}
/**
* 节点获焦
*/
'node:focus': NodeEventArgsPick<'data'>
/**
* 节点失焦
*/
'node:blur': NodeEventArgsPick<'data'>
}

type EdgeEventArgsPick<T extends 'data' | 'e' | 'position'> = Pick<
Expand Down Expand Up @@ -259,6 +267,14 @@ interface EdgeEventArgs {
oldEdge: EdgeData
}
}
/**
* 边获焦
*/
'edge:focus': EdgeEventArgsPick<'data'>
/**
* 边失焦
*/
'edge:blur': EdgeEventArgsPick<'data'>
}

/**
Expand Down
55 changes: 54 additions & 1 deletion packages/core/src/model/GraphModel.ts
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import { find, forEach, map, merge, isBoolean, debounce,isNil } from 'lodash-es'
import { find, forEach, map, merge, isBoolean, debounce, isEqual,isNil } from 'lodash-es'
import { action, computed, observable } from 'mobx'
import {
BaseEdgeModel,
Expand Down Expand Up @@ -41,6 +41,7 @@ import Position = LogicFlow.Position
import PointTuple = LogicFlow.PointTuple
import GraphData = LogicFlow.GraphData
import NodeConfig = LogicFlow.NodeConfig
import AnchorConfig = Model.AnchorConfig
import BaseNodeModelCtor = LogicFlow.BaseNodeModelCtor
import BaseEdgeModelCtor = LogicFlow.BaseEdgeModelCtor

Expand Down Expand Up @@ -474,6 +475,58 @@ export class GraphModel {
throw new Error(`找不到${edge.type}对应的边。`)
}
const edgeModel = new Model(edge, this)
// 根据edgeModel中存储的数据找到当前画布上的起终锚点坐标
// 判断当前起终锚点数据和Model中存储的起终点数据是否一致,不一致更新起终点信息
const {
sourceNodeId,
targetNodeId,
sourceAnchorId = '',
targetAnchorId = '',
startPoint,
endPoint,
text,
textPosition,
} = edgeModel
const updateAnchorPoint = (
node: BaseNodeModel | undefined,
anchorId: string,
point: Position,
updatePoint: (anchor: AnchorConfig) => void,
) => {
const anchor = node?.anchors.find((anchor) => anchor.id === anchorId)
if (anchor && !isEqual(anchor, point)) {
updatePoint(anchor)
}
}

const sourceNode = this.getNodeModelById(sourceNodeId)
const targetNode = this.getNodeModelById(targetNodeId)

updateAnchorPoint(
sourceNode,
sourceAnchorId,
startPoint,
edgeModel.updateStartPoint.bind(edgeModel),
)
updateAnchorPoint(
targetNode,
targetAnchorId,
endPoint,
edgeModel.updateEndPoint.bind(edgeModel),
)

// 而文本需要先算一下文本与默认文本位置之间的相对位置差
// 再计算新路径的文本默认位置,加上相对位置差,得到调整后边的文本的位置
if (text) {
const { x, y } = text
const { x: defaultX, y: defaultY } = textPosition
if (x && y && defaultX && defaultY) {
const deltaX = x - defaultX
const deltaY = y - defaultY
edgeModel.resetTextPosition()
edgeModel.moveText(deltaX, deltaY)
}
}
this.edgeModelMap.set(edgeModel.id, edgeModel)
this.elementsModelMap.set(edgeModel.id, edgeModel)

Expand Down
8 changes: 8 additions & 0 deletions packages/core/src/model/edge/LineEdgeModel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,14 @@ export class LineEdgeModel extends BaseEdgeModel {
...cloneDeep(customStyle),
}
}
initEdgeData(data: LogicFlow.EdgeConfig): void {
super.initEdgeData(data)
this.points = this.getPath([this.startPoint, this.endPoint])
}
getPath(points: Point[]): string {
const [start, end] = points
return `${start.x},${start.y} ${end.x},${end.y}`
}
getTextPosition(): Point {
return {
x: (this.startPoint.x + this.endPoint.x) / 2,
Expand Down
8 changes: 5 additions & 3 deletions packages/core/src/model/edge/PolylineEdgeModel.ts
Original file line number Diff line number Diff line change
Expand Up @@ -326,12 +326,14 @@ export class PolylineEdgeModel extends BaseEdgeModel {
})
}

getPath(points: Point[]): string {
return points.map((point) => `${point.x},${point.y}`).join(' ')
}

@action
initPoints() {
if (this.pointsList.length > 0) {
this.points = this.pointsList
.map((point) => `${point.x},${point.y}`)
.join(' ')
this.points = this.getPath(this.pointsList)
} else {
this.updatePoints()
}
Expand Down
5 changes: 5 additions & 0 deletions packages/core/src/style/index.less
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,11 @@
cursor: move;
}

// 在元素focus时浏览器会给元素outline设置一个5像素宽的默认样式,这里先全局禁用一下,后续再根据需要再做调整
*:focus {
outline: none;
}

.lf-node-anchor {
cursor: crosshair;
}
Expand Down
3 changes: 3 additions & 0 deletions packages/core/src/style/raw.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,9 @@ export const content = `.lf-graph {
.lf-text-draggable {
cursor: move;
}
*:focus {
outline: none;
}
.lf-node-anchor {
cursor: crosshair;
}
Expand Down
16 changes: 16 additions & 0 deletions packages/core/src/view/edge/BaseEdge.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -459,6 +459,20 @@ export abstract class BaseEdge<P extends IProps> extends Component<
this.toFront()
}

handleFocus = () => {
const { model, graphModel } = this.props
graphModel.eventCenter.emit(EventType.EDGE_FOCUS, {
data: model.getData(),
})
}

handleBlur = () => {
const { model, graphModel } = this.props
graphModel.eventCenter.emit(EventType.EDGE_BLUR, {
data: model.getData(),
})
}

/**
* @overridable 支持重写, 此方法为获取边的形状,如果需要自定义边的形状,请重写此方法。
* @example https://docs.logic-flow.cn/docs/#/zh/guide/basic/edge?id=%e5%9f%ba%e4%ba%8e-react-%e7%bb%84%e4%bb%b6%e8%87%aa%e5%ae%9a%e4%b9%89%e8%be%b9
Expand Down Expand Up @@ -499,6 +513,8 @@ export abstract class BaseEdge<P extends IProps> extends Component<
onMouseOver={this.setHoverOn}
onMouseEnter={this.setHoverOn}
onMouseLeave={this.setHoverOff}
onFocus={this.handleFocus}
onBlur={this.handleBlur}
>
{this.getShape()}
{this.getAppend()}
Expand Down
16 changes: 16 additions & 0 deletions packages/core/src/view/node/BaseNode.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -424,6 +424,20 @@ export abstract class BaseNode<P extends IProps = IProps> extends Component<
}
}

handleFocus = () => {
const { model, graphModel } = this.props
graphModel.eventCenter.emit(EventType.NODE_FOCUS, {
data: model.getData(),
})
}

handleBlur = () => {
const { model, graphModel } = this.props
graphModel.eventCenter.emit(EventType.NODE_BLUR, {
data: model.getData(),
})
}

// 因为自定义节点的时候,可能会基于hover状态自定义不同的样式。
setHoverOn = (ev: MouseEvent) => {
const { model, graphModel } = this.props
Expand Down Expand Up @@ -508,6 +522,8 @@ export abstract class BaseNode<P extends IProps = IProps> extends Component<
onMouseLeave={this.setHoverOff}
onMouseOut={this.onMouseOut}
onContextMenu={this.handleContextMenu}
onFocus={this.handleFocus}
onBlur={this.handleBlur}
{...restAttributes}
>
{nodeShapeInner}
Expand Down
2 changes: 1 addition & 1 deletion packages/core/src/view/overlay/OutlineOverlay.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -82,7 +82,7 @@ export class OutlineOverlay extends Component<IProps> {
(hoverOutline && edge.isHovered)
) {
if (edge.modelType === ModelType.LINE_EDGE) {
edgeOutline.push(this.getLineOutline(edge))
edgeOutline.push(this.getLineOutline(edge as LineEdgeModel))
} else if (edge.modelType === ModelType.POLYLINE_EDGE) {
edgeOutline.push(this.getPolylineOutline(edge as PolylineEdgeModel))
} else if (edge.modelType === ModelType.BEZIER_EDGE) {
Expand Down

0 comments on commit 0198569

Please sign in to comment.