mirror of
https://github.com/wanglin2/mind-map.git
synced 2026-03-10 18:57:53 +08:00
Feature:新增时间轴结构
This commit is contained in:
parent
21f487321a
commit
2de0334e3b
@ -3,6 +3,7 @@ import LogicalStructure from './layouts/LogicalStructure'
|
|||||||
import MindMap from './layouts/MindMap'
|
import MindMap from './layouts/MindMap'
|
||||||
import CatalogOrganization from './layouts/CatalogOrganization'
|
import CatalogOrganization from './layouts/CatalogOrganization'
|
||||||
import OrganizationStructure from './layouts/OrganizationStructure'
|
import OrganizationStructure from './layouts/OrganizationStructure'
|
||||||
|
import Timeline from './layouts/Timeline'
|
||||||
import TextEdit from './TextEdit'
|
import TextEdit from './TextEdit'
|
||||||
import { copyNodeTree, simpleDeepClone, walk } from './utils'
|
import { copyNodeTree, simpleDeepClone, walk } from './utils'
|
||||||
import { shapeList } from './Shape'
|
import { shapeList } from './Shape'
|
||||||
@ -18,7 +19,9 @@ const layouts = {
|
|||||||
// 目录组织图
|
// 目录组织图
|
||||||
[CONSTANTS.LAYOUT.CATALOG_ORGANIZATION]: CatalogOrganization,
|
[CONSTANTS.LAYOUT.CATALOG_ORGANIZATION]: CatalogOrganization,
|
||||||
// 组织结构图
|
// 组织结构图
|
||||||
[CONSTANTS.LAYOUT.ORGANIZATION_STRUCTURE]: OrganizationStructure
|
[CONSTANTS.LAYOUT.ORGANIZATION_STRUCTURE]: OrganizationStructure,
|
||||||
|
// 时间轴
|
||||||
|
[CONSTANTS.LAYOUT.TIMELINE]: Timeline
|
||||||
}
|
}
|
||||||
|
|
||||||
// 渲染
|
// 渲染
|
||||||
|
|||||||
318
simple-mind-map/src/layouts/Timeline.js
Normal file
318
simple-mind-map/src/layouts/Timeline.js
Normal file
@ -0,0 +1,318 @@
|
|||||||
|
import Base from './Base'
|
||||||
|
import { walk, asyncRun } from '../utils'
|
||||||
|
|
||||||
|
// 时间轴
|
||||||
|
class CatalogOrganization extends Base {
|
||||||
|
// 构造函数
|
||||||
|
constructor(opt = {}) {
|
||||||
|
super(opt)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 布局
|
||||||
|
doLayout(callback) {
|
||||||
|
let task = [
|
||||||
|
() => {
|
||||||
|
this.computedBaseValue()
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
this.computedLeftTopValue()
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
this.adjustLeftTopValue()
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
callback(this.root)
|
||||||
|
}
|
||||||
|
]
|
||||||
|
asyncRun(task)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 遍历数据创建节点、计算根节点的位置,计算根节点的子节点的top值
|
||||||
|
computedBaseValue() {
|
||||||
|
walk(
|
||||||
|
this.renderer.renderTree,
|
||||||
|
null,
|
||||||
|
(cur, parent, isRoot, layerIndex) => {
|
||||||
|
let newNode = this.createNode(cur, parent, isRoot, layerIndex)
|
||||||
|
// 根节点定位在画布中心位置
|
||||||
|
if (isRoot) {
|
||||||
|
this.setNodeCenter(newNode)
|
||||||
|
} else {
|
||||||
|
// 非根节点
|
||||||
|
if (parent._node.isRoot) {
|
||||||
|
newNode.top =
|
||||||
|
parent._node.top +
|
||||||
|
(cur._node.height > parent._node.height
|
||||||
|
? -(cur._node.height - parent._node.height) / 2
|
||||||
|
: (parent._node.height - cur._node.height) / 2)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!cur.data.expand) {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
(cur, parent, isRoot, layerIndex) => {},
|
||||||
|
true,
|
||||||
|
0
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 遍历节点树计算节点的left、top
|
||||||
|
computedLeftTopValue() {
|
||||||
|
walk(
|
||||||
|
this.root,
|
||||||
|
null,
|
||||||
|
(node, parent, isRoot, layerIndex) => {
|
||||||
|
if (
|
||||||
|
node.nodeData.data.expand &&
|
||||||
|
node.children &&
|
||||||
|
node.children.length
|
||||||
|
) {
|
||||||
|
let marginX = this.getMarginX(layerIndex + 1)
|
||||||
|
let marginY = this.getMarginY(layerIndex + 1)
|
||||||
|
if (isRoot) {
|
||||||
|
let left = node.left + node.width
|
||||||
|
let totalLeft = left + marginX
|
||||||
|
node.children.forEach(cur => {
|
||||||
|
cur.left = totalLeft
|
||||||
|
totalLeft += cur.width + marginX
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
let totalTop = node.top + node.height + marginY + node.expandBtnSize
|
||||||
|
node.children.forEach(cur => {
|
||||||
|
cur.left = node.left + node.width * 0.5
|
||||||
|
cur.top = totalTop
|
||||||
|
totalTop += cur.height + marginY + node.expandBtnSize
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
null,
|
||||||
|
true
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 调整节点left、top
|
||||||
|
adjustLeftTopValue() {
|
||||||
|
walk(
|
||||||
|
this.root,
|
||||||
|
null,
|
||||||
|
(node, parent, isRoot, layerIndex) => {
|
||||||
|
if (!node.nodeData.data.expand) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// 调整left
|
||||||
|
if (node.isRoot) {
|
||||||
|
this.updateBrothersLeft(node)
|
||||||
|
}
|
||||||
|
// 调整top
|
||||||
|
let len = node.children.length
|
||||||
|
if (parent && !parent.isRoot && len > 0) {
|
||||||
|
let marginY = this.getMarginY(layerIndex + 1)
|
||||||
|
let totalHeight =
|
||||||
|
node.children.reduce((h, item) => {
|
||||||
|
return h + item.height
|
||||||
|
}, 0) +
|
||||||
|
(len + 1) * marginY +
|
||||||
|
len * node.expandBtnSize
|
||||||
|
this.updateBrothersTop(node, totalHeight)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
null,
|
||||||
|
true
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 递归计算节点的宽度
|
||||||
|
getNodeAreaWidth(node) {
|
||||||
|
let widthArr = []
|
||||||
|
let loop = (node, width) => {
|
||||||
|
if (node.children.length) {
|
||||||
|
width += node.width / 2
|
||||||
|
node.children.forEach(item => {
|
||||||
|
loop(item, width)
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
width += node.width
|
||||||
|
widthArr.push(width)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
loop(node, 0)
|
||||||
|
return Math.max(...widthArr)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 调整兄弟节点的left
|
||||||
|
updateBrothersLeft(node) {
|
||||||
|
let childrenList = node.children
|
||||||
|
let totalAddWidth = 0
|
||||||
|
childrenList.forEach(item => {
|
||||||
|
item.left += totalAddWidth
|
||||||
|
if (item.children && item.children.length) {
|
||||||
|
this.updateChildren(item.children, 'left', totalAddWidth)
|
||||||
|
}
|
||||||
|
let areaWidth = this.getNodeAreaWidth(item)
|
||||||
|
let difference = areaWidth - item.width
|
||||||
|
if (difference > 0) {
|
||||||
|
totalAddWidth += difference
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 调整兄弟节点的top
|
||||||
|
updateBrothersTop(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) => {
|
||||||
|
if (item.hasCustomPosition()) {
|
||||||
|
// 适配自定义位置
|
||||||
|
return
|
||||||
|
}
|
||||||
|
let _offset = 0
|
||||||
|
// 下面的节点往下移
|
||||||
|
if (_index > index) {
|
||||||
|
_offset = addHeight
|
||||||
|
}
|
||||||
|
item.top += _offset
|
||||||
|
// 同步更新子节点的位置
|
||||||
|
if (item.children && item.children.length) {
|
||||||
|
this.updateChildren(item.children, 'top', _offset)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// 更新父节点的位置
|
||||||
|
this.updateBrothersTop(node.parent, addHeight)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 绘制连线,连接该节点到其子节点
|
||||||
|
renderLine(node, lines, style) {
|
||||||
|
if (node.children.length <= 0) {
|
||||||
|
return []
|
||||||
|
}
|
||||||
|
let { left, top, width, height, expandBtnSize } = node
|
||||||
|
let len = node.children.length
|
||||||
|
if (node.isRoot) {
|
||||||
|
// 根节点
|
||||||
|
let prevBother = node
|
||||||
|
node.children.forEach((item, index) => {
|
||||||
|
let y2 = item.top + item.height
|
||||||
|
// 节点使用横线风格,需要额外渲染横线
|
||||||
|
let nodeUseLineStylePath = this.mindMap.themeConfig.nodeUseLineStyle
|
||||||
|
? ` L ${item.left},${y2} L ${item.left + item.width},${y2}`
|
||||||
|
: ''
|
||||||
|
let path =
|
||||||
|
`M ${prevBother.left + prevBother.width},${
|
||||||
|
node.top + node.height / 2
|
||||||
|
} L ${item.left},${node.top + node.height / 2}` + nodeUseLineStylePath
|
||||||
|
// 竖线
|
||||||
|
lines[index].plot(path)
|
||||||
|
style && style(lines[index], item)
|
||||||
|
prevBother = item
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
// 非根节点
|
||||||
|
let y1 = top + height
|
||||||
|
let maxy = -Infinity
|
||||||
|
let x2 = node.left + node.width * 0.3
|
||||||
|
node.children.forEach((item, index) => {
|
||||||
|
// 为了适配自定义位置,下面做了各种位置的兼容
|
||||||
|
let y2 = item.top + item.height / 2
|
||||||
|
if (y2 > maxy) {
|
||||||
|
maxy = y2
|
||||||
|
}
|
||||||
|
// 水平线
|
||||||
|
let path = ''
|
||||||
|
let _left = item.left
|
||||||
|
let _isLeft = item.left + item.width < x2
|
||||||
|
let _isXCenter = false
|
||||||
|
if (_isLeft) {
|
||||||
|
// 水平位置在父节点左边
|
||||||
|
_left = item.left + item.width
|
||||||
|
} else if (item.left < x2 && item.left + item.width > x2) {
|
||||||
|
// 水平位置在父节点之间
|
||||||
|
_isXCenter = true
|
||||||
|
y2 = item.top
|
||||||
|
maxy = y2
|
||||||
|
}
|
||||||
|
if (y2 > top && y2 < y1) {
|
||||||
|
// 自定义位置的情况:垂直位置节点在父节点之间
|
||||||
|
path = `M ${
|
||||||
|
_isLeft ? node.left : node.left + node.width
|
||||||
|
},${y2} L ${_left},${y2}`
|
||||||
|
} else if (y2 < y1) {
|
||||||
|
// 自定义位置的情况:垂直位置节点在父节点上面
|
||||||
|
if (_isXCenter) {
|
||||||
|
y2 = item.top + item.height
|
||||||
|
_left = x2
|
||||||
|
}
|
||||||
|
path = `M ${x2},${top} L ${x2},${y2} L ${_left},${y2}`
|
||||||
|
} else {
|
||||||
|
if (_isXCenter) {
|
||||||
|
_left = x2
|
||||||
|
}
|
||||||
|
path = `M ${x2},${y2} L ${_left},${y2}`
|
||||||
|
}
|
||||||
|
// 节点使用横线风格,需要额外渲染横线
|
||||||
|
let nodeUseLineStylePath = this.mindMap.themeConfig.nodeUseLineStyle
|
||||||
|
? ` L ${_left},${y2 - item.height / 2} L ${_left},${
|
||||||
|
y2 + item.height / 2
|
||||||
|
}`
|
||||||
|
: ''
|
||||||
|
path += nodeUseLineStylePath
|
||||||
|
lines[index].plot(path)
|
||||||
|
style && style(lines[index], item)
|
||||||
|
})
|
||||||
|
// 竖线
|
||||||
|
if (len > 0) {
|
||||||
|
let lin2 = this.draw.path()
|
||||||
|
expandBtnSize = len > 0 ? expandBtnSize : 0
|
||||||
|
node.style.line(lin2)
|
||||||
|
if (maxy < y1 + expandBtnSize) {
|
||||||
|
lin2.hide()
|
||||||
|
} else {
|
||||||
|
lin2.plot(`M ${x2},${y1 + expandBtnSize} L ${x2},${maxy}`)
|
||||||
|
lin2.show()
|
||||||
|
}
|
||||||
|
node._lines.push(lin2)
|
||||||
|
style && style(lin2, node)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 渲染按钮
|
||||||
|
renderExpandBtn(node, btn) {
|
||||||
|
let { width, height, expandBtnSize, isRoot } = node
|
||||||
|
if (!isRoot) {
|
||||||
|
let { translateX, translateY } = btn.transform()
|
||||||
|
btn.translate(
|
||||||
|
width * 0.3 - expandBtnSize / 2 - translateX,
|
||||||
|
height + expandBtnSize / 2 - translateY
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 创建概要节点
|
||||||
|
renderGeneralization(node, gLine, gNode) {
|
||||||
|
let {
|
||||||
|
top,
|
||||||
|
bottom,
|
||||||
|
right,
|
||||||
|
generalizationLineMargin,
|
||||||
|
generalizationNodeMargin
|
||||||
|
} = this.getNodeBoundaries(node, 'h')
|
||||||
|
let x1 = right + generalizationLineMargin
|
||||||
|
let y1 = top
|
||||||
|
let x2 = right + generalizationLineMargin
|
||||||
|
let y2 = bottom
|
||||||
|
let cx = x1 + 20
|
||||||
|
let cy = y1 + (y2 - y1) / 2
|
||||||
|
let path = `M ${x1},${y1} Q ${cx},${cy} ${x2},${y2}`
|
||||||
|
gLine.plot(path)
|
||||||
|
gNode.left = right + generalizationNodeMargin
|
||||||
|
gNode.top = top + (bottom - top - gNode.height) / 2
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default CatalogOrganization
|
||||||
@ -154,7 +154,8 @@ export const CONSTANTS = {
|
|||||||
LOGICAL_STRUCTURE: 'logicalStructure',
|
LOGICAL_STRUCTURE: 'logicalStructure',
|
||||||
MIND_MAP: 'mindMap',
|
MIND_MAP: 'mindMap',
|
||||||
ORGANIZATION_STRUCTURE: 'organizationStructure',
|
ORGANIZATION_STRUCTURE: 'organizationStructure',
|
||||||
CATALOG_ORGANIZATION: 'catalogOrganization'
|
CATALOG_ORGANIZATION: 'catalogOrganization',
|
||||||
|
TIMELINE: 'timeline'
|
||||||
},
|
},
|
||||||
DIR: {
|
DIR: {
|
||||||
UP: 'up',
|
UP: 'up',
|
||||||
@ -217,11 +218,16 @@ export const layoutList = [
|
|||||||
{
|
{
|
||||||
name: '目录组织图',
|
name: '目录组织图',
|
||||||
value: CONSTANTS.LAYOUT.CATALOG_ORGANIZATION,
|
value: CONSTANTS.LAYOUT.CATALOG_ORGANIZATION,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '时间轴',
|
||||||
|
value: CONSTANTS.LAYOUT.TIMELINE,
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
export const layoutValueList = [
|
export const layoutValueList = [
|
||||||
CONSTANTS.LAYOUT.LOGICAL_STRUCTURE,
|
CONSTANTS.LAYOUT.LOGICAL_STRUCTURE,
|
||||||
CONSTANTS.LAYOUT.MIND_MAP,
|
CONSTANTS.LAYOUT.MIND_MAP,
|
||||||
CONSTANTS.LAYOUT.CATALOG_ORGANIZATION,
|
CONSTANTS.LAYOUT.CATALOG_ORGANIZATION,
|
||||||
CONSTANTS.LAYOUT.ORGANIZATION_STRUCTURE
|
CONSTANTS.LAYOUT.ORGANIZATION_STRUCTURE,
|
||||||
|
CONSTANTS.LAYOUT.TIMELINE
|
||||||
]
|
]
|
||||||
@ -272,7 +272,7 @@ export default {
|
|||||||
this.mindMap = new MindMap({
|
this.mindMap = new MindMap({
|
||||||
el: this.$refs.mindMapContainer,
|
el: this.$refs.mindMapContainer,
|
||||||
data: root,
|
data: root,
|
||||||
layout: layout,
|
layout: 'timeline',
|
||||||
theme: theme.template,
|
theme: theme.template,
|
||||||
themeConfig: theme.config,
|
themeConfig: theme.config,
|
||||||
viewData: view,
|
viewData: view,
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user