mind-map/simple-mind-map/src/layouts/_CatalogOrganization.js
2021-07-10 22:06:45 +08:00

270 lines
8.7 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import {
walk
} from '../Utils'
import Node from '../Node'
import merge from 'deepmerge'
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 16:25:07
* @Desc: 目录组织图
* 思路第一轮只计算节点的宽高以及某个节点的所有子节点所占的高度之和以及该节点里所有子节点中宽度最宽是多少、第二轮计算节点的left和top需要区分二级节点和其他节点二级节点top相同一行依次从做向右排开其他节点的left相同一列从上往下依次排开
*/
class Render {
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 16:25:32
* @Desc: 构造函数
*/
constructor(opt = {}) {
this.opt = opt
this.mindMap = opt.mindMap
this.draw = this.mindMap.draw
// 渲染树
this.renderTree = merge({}, this.mindMap.opt.data || {})
// 根节点
this.root = null
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 16:27:55
* @Desc: 渲染
*/
render() {
this.computed()
this.root.render()
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-06 14:04:20
* @Desc: 计算位置数据
*/
computed() {
// 计算节点的width、height
this.computedBaseValue()
// 计算节点的left、top
this.computedLeftTopValue()
// 调整节点top
this.adjustTopValue()
// 调整节点left
this.adjustLeftValue()
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 09:49:32
* @Desc: 计算节点的width、height
*/
computedBaseValue() {
walk(this.renderTree, null, (node, parent, isRoot) => {
// 设置width、height
let {
children,
...props
} = node
let newNode = new Node({
...props,
mindMap: this.mindMap,
draw: this.draw
})
// 计算节点的宽高
newNode.getSize()
// 计算节点的top
if (isRoot) {
newNode.isRoot = true
newNode.left = (this.mindMap.width - newNode.width) / 2
newNode.top = (this.mindMap.height - newNode.height) / 2
this.root = newNode
} else {
newNode.parent = parent._node
parent._node.addChildren(newNode)
}
node._node = newNode
}, (node) => {
// 遍历完子节点返回时
// 计算节点的areaHeight也就是子节点所占的高度之和包括外边距
let len = node._node.children.length
if (node._node.isRoot) {
node._node.childrenAreaWidth = len ? node._node.children.reduce((h, cur) => {
return h + cur.width
}, 0) + (len + 1) * this.mindMap.opt.marginX : 0
}
node._node.childrenAreaHeight = len ? node._node.children.reduce((h, cur) => {
return h + cur.height
}, 0) + (len + 1) * this.mindMap.opt.marginY : 0
}, true)
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 09:59:25
* @Desc: 计算节点的left、top
*/
computedLeftTopValue() {
walk(this.root, null, (node) => {
if (node.children && node.children.length) {
if (node.isRoot) {
let left = node.left + node.width / 2 - node.childrenAreaWidth / 2
let totalLeft = left + this.mindMap.opt.marginX
node.children.forEach((cur) => {
// left
cur.left = totalLeft
totalLeft += cur.width + this.mindMap.opt.marginX
// top
cur.top = node.top + node.height + this.mindMap.opt.marginY
})
} else {
let totalTop = node.top + node.height + this.mindMap.opt.marginY
node.children.forEach((cur) => {
cur.left = node.left + node.width / 5 + this.mindMap.opt.marginX
cur.top = totalTop
totalTop += cur.height + this.mindMap.opt.marginY
})
}
}
}, null, true)
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-12 17:07:29
* @Desc: 调整节点left
*/
adjustLeftValue() {
walk(this.root, null, (node) => {
if (node.parent && node.parent.isRoot) {
let childrenAreaWidth = this.getNodeWidth(node)
let difference = childrenAreaWidth - node.width
if (difference > 0) {
this.updateBrothersLeftValue(node, difference / 2)
}
}
}, null, true)
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-12 18:55:03
* @Desc: 计算节点的宽度,包括子节点
*/
getNodeWidth(node) {
let widthArr = []
let loop = (node, width) => {
if (node.children.length) {
width += node.width / 5 + this.mindMap.opt.marginX
node.children.forEach((item) => {
loop(item, width)
})
} else {
width += node.width
widthArr.push(width)
}
}
loop(node, 0)
return Math.max(...widthArr)
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-12 18:21:46
* @Desc: 调整兄弟节点的left
*/
updateBrothersLeftValue(node, addWidth) {
if (node.parent) {
let childrenList = node.parent.children
let index = childrenList.findIndex((item) => {
return item === node
})
childrenList.forEach((item, _index) => {
let _offset = 0
if (_index > index) {
_offset = addWidth
} else {
_offset = -addWidth
}
item.left += _offset
// 同步更新子节点的位置
if (item.children && item.children.length) {
this.updateChildren(item.children, 'left', _offset)
}
})
// 更新父节点的位置
this.updateBrothersLeftValue(node.parent, addWidth)
}
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-08 10:04:05
* @Desc: 调整节点top该节点之后的节点都往下进行偏移
*/
adjustTopValue() {
let marginY = this.mindMap.opt.marginY
walk(this.root, null, (node) => {
if (!node.isRoot && !node.parent.isRoot) {
// 判断子节点的areaHeight是否大于该节点自身大于则需要调整位置
if (node.children && node.children.length > 0) {
let difference = node.childrenAreaHeight - marginY
this.updateBrothersTopValue(node, difference)
}
}
}, null, true)
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-07 14:26:03
* @Desc: 更新兄弟节点的top
*/
updateBrothersTopValue(node, addHeight) {
if (node.parent && !node.parent.isRoot) {
let childrenList = node.parent.children
let index = childrenList.findIndex((item) => {
return item === node
})
childrenList.forEach((item, _index) => {
let _offset = 0
if (_index > index) {
_offset = addHeight
}
item.top += _offset
// 同步更新子节点的位置
if (item.children && item.children.length) {
this.updateChildren(item.children, 'top', _offset)
}
})
// 更新父节点的位置
this.updateBrothersTopValue(node.parent, addHeight)
}
}
/**
* javascript comment
* @Author: 王林25
* @Date: 2021-04-07 11:25:52
* @Desc: 更新子节点属性
*/
updateChildren(children, prop, offset) {
children.forEach((item) => {
item[prop] += offset
if (item.children && item.children.length) {
this.updateChildren(item.children, prop, offset)
}
})
}
}
export default Render