openatc-components 0.1.139 → 0.1.142
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package/kisscomps/components/CommonKanban/CommonKanban.vue +8 -4
- package/package/kisscomps/components/DrawChannelization/drawsvg/overlapAssociatedComponent.vue +1 -1
- package/package/kisscomps/components/DrawChannelization/drawsvg/phaseAssociatedComponent.vue +1 -1
- package/package/kisscomps/components/ExpendConfig/ExpendConfig.vue +1 -1
- package/package/kisscomps/components/KanBan/kanban.vue +1 -1
- package/package/kisscomps/components/OverLap/OverLap.vue +2 -2
- package/package/kisscomps/components/PatternOptimize/PatternOptimize.vue +1 -1
- package/package/kisscomps/components/PatternStatus/PatternStatus.vue +21 -20
- package/package/kisscomps/components/PhaseMarker/phasemarker.vue +1 -1
- package/package/kisscomps/components/SchemeConfig/SchemeConfig.vue +1 -1
- package/package/kisscomps/components/SchemeConfig/azimuthlocking/index.vue +54 -13
- package/package/kisscomps/components/SchemeConfig/azimuthlocking/utils.js +8 -8
- package/package/kisscomps/components/SchemeConfig/manualControlModal/index.vue +2 -0
- package/package/kisscomps/components/SchemeConfig/priorityControl/index.vue +1 -1
- package/package/kisscomps/components/SchemeConfig/tentativeplancontrolmodal/index.vue +1 -1
- package/package/kisscomps/components/Stages/index.vue +32 -2
- package/package/kisscomps/components/XRDDirSelector/XRDDirSelector.vue +458 -626
- package/package/kisscomps/components/XiaoKanban/DirSelector.vue +356 -0
- package/package/kisscomps/components/XiaoKanban/index.vue +102 -0
- package/package/kisscomps/components/patternConfig/index.js +2 -0
- package/package/kisscomps/components/patternConfig/index.vue +1010 -0
- package/package/kisscomps/components/patternConfig/pattern/patternTable.vue +349 -0
- package/package/kisscomps/components/patternConfig/pattern/planChart/model/coordinationModel.js +665 -0
- package/package/kisscomps/components/patternConfig/planContent.vue +572 -0
- package/package/kisscomps/components/patternConfig/planMenu.vue +329 -0
- package/package/kisscomps/components/patternConfig/utils.js +152 -0
- package/package/kisscomps/index.js +4 -0
- package/package/kissui.min.js +1 -1
- package/package.json +14 -13
- package/src/api/control.js +19 -0
- package/src/api/device.js +135 -0
- package/src/api/route.js +171 -0
- package/src/i18n/language/en.js +149 -0
- package/src/i18n/language/zh.js +133 -0
- package/src/kisscomps/components/CommonKanban/CommonKanban.vue +8 -4
- package/src/kisscomps/components/DrawChannelization/drawsvg/overlapAssociatedComponent.vue +1 -1
- package/src/kisscomps/components/DrawChannelization/drawsvg/phaseAssociatedComponent.vue +1 -1
- package/src/kisscomps/components/ExpendConfig/ExpendConfig.vue +1 -1
- package/src/kisscomps/components/KanBan/kanban.vue +1 -1
- package/src/kisscomps/components/OverLap/OverLap.vue +2 -2
- package/src/kisscomps/components/PatternOptimize/PatternOptimize.vue +1 -1
- package/src/kisscomps/components/PatternStatus/PatternStatus.vue +21 -20
- package/src/kisscomps/components/PhaseMarker/phasemarker.vue +1 -1
- package/src/kisscomps/components/SchemeConfig/SchemeConfig.vue +1 -1
- package/src/kisscomps/components/SchemeConfig/azimuthlocking/index.vue +54 -13
- package/src/kisscomps/components/SchemeConfig/azimuthlocking/utils.js +8 -8
- package/src/kisscomps/components/SchemeConfig/manualControlModal/index.vue +2 -0
- package/src/kisscomps/components/SchemeConfig/priorityControl/index.vue +1 -1
- package/src/kisscomps/components/SchemeConfig/tentativeplancontrolmodal/index.vue +1 -1
- package/src/kisscomps/components/Stages/index.vue +32 -2
- package/src/kisscomps/components/XRDDirSelector/XRDDirSelector.vue +458 -626
- package/src/kisscomps/components/XiaoKanban/DirSelector.vue +356 -0
- package/src/kisscomps/components/XiaoKanban/index.vue +102 -0
- package/src/kisscomps/components/patternConfig/index.js +2 -0
- package/src/kisscomps/components/patternConfig/index.vue +1010 -0
- package/src/kisscomps/components/patternConfig/pattern/patternTable.vue +349 -0
- package/src/kisscomps/components/patternConfig/pattern/planChart/model/coordinationModel.js +665 -0
- package/src/kisscomps/components/patternConfig/planContent.vue +572 -0
- package/src/kisscomps/components/patternConfig/planMenu.vue +329 -0
- package/src/kisscomps/components/patternConfig/utils.js +152 -0
- package/src/kisscomps/index.js +4 -0
- package/src/lib/publicjs/KissApi.js +2 -2
- package/src/router/index.js +7 -0
- package/src/utils/auth.js +24 -3
- package/src/views/intersection.vue +1 -1
- package/src/views/patternConfig.vue +1468 -0
- package/static/apiconfig.json +85 -0
- package/static/styles/common.scss +2 -0
- package/static/styles/patternConfig.scss +57 -0
- package/static/styles/xiaokanban.scss +51 -0
|
@@ -0,0 +1,665 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2020 kedacom
|
|
3
|
+
* OpenATC is licensed under Mulan PSL v2.
|
|
4
|
+
* You can use this software according to the terms and conditions of the Mulan PSL v2.
|
|
5
|
+
* You may obtain a copy of Mulan PSL v2 at:
|
|
6
|
+
* http://license.coscl.org.cn/MulanPSL2
|
|
7
|
+
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
|
8
|
+
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
|
9
|
+
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
|
10
|
+
* See the Mulan PSL v2 for more details.
|
|
11
|
+
**/
|
|
12
|
+
/**
|
|
13
|
+
* @Description: Coordination Echarts表格图
|
|
14
|
+
* @Author: yangdongyang
|
|
15
|
+
* @Date: Create in 11:27 2019/11/26
|
|
16
|
+
* @Modified By:
|
|
17
|
+
*/
|
|
18
|
+
import echarts from 'echarts'
|
|
19
|
+
import { getTheme } from '../../../../utils/auth'
|
|
20
|
+
export default class CoordinationModel {
|
|
21
|
+
constructor () {
|
|
22
|
+
this.rourte = {}
|
|
23
|
+
this.greenwave = []
|
|
24
|
+
}
|
|
25
|
+
getMaxY () {
|
|
26
|
+
let rourte = this.rourte
|
|
27
|
+
let maxY = 0
|
|
28
|
+
if (Object.keys(rourte).length === 0) return maxY
|
|
29
|
+
for (let devs of rourte.devs) {
|
|
30
|
+
for (let pattern of devs.patternList) {
|
|
31
|
+
if (pattern.cycle > maxY) {
|
|
32
|
+
maxY = pattern.cycle
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
return maxY
|
|
37
|
+
}
|
|
38
|
+
getMaxX () {
|
|
39
|
+
let rourte = this.rourte
|
|
40
|
+
let maxX = 0
|
|
41
|
+
if (Object.keys(rourte).length === 0) return maxX
|
|
42
|
+
for (let devs of rourte.devs) {
|
|
43
|
+
maxX = maxX + devs.distance
|
|
44
|
+
}
|
|
45
|
+
return maxX
|
|
46
|
+
}
|
|
47
|
+
getInitBarData () {
|
|
48
|
+
let rourte = this.rourte
|
|
49
|
+
let data = []
|
|
50
|
+
let xDistance = 0 // 表示该路口到起点的距离
|
|
51
|
+
for (let devs of rourte.devs) {
|
|
52
|
+
xDistance = xDistance + devs.distance
|
|
53
|
+
let list = [xDistance, 0]
|
|
54
|
+
data.push(list)
|
|
55
|
+
}
|
|
56
|
+
return data
|
|
57
|
+
}
|
|
58
|
+
getXName (val) {
|
|
59
|
+
let rourte = this.rourte
|
|
60
|
+
let xDistance = 0 // 表示该路口到起点的距离
|
|
61
|
+
let xName = ''
|
|
62
|
+
for (let devs of rourte.devs) {
|
|
63
|
+
xDistance = xDistance + devs.distance
|
|
64
|
+
if (xDistance === val) {
|
|
65
|
+
xName = devs.agentid + '(' + val + ')'
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return xName
|
|
69
|
+
}
|
|
70
|
+
getCycleNum () {
|
|
71
|
+
let rourte = this.rourte
|
|
72
|
+
let minCycle = 99999
|
|
73
|
+
for (let devs of rourte.devs) {
|
|
74
|
+
let cycle = devs.patternList[0].cycle
|
|
75
|
+
if (cycle < minCycle) {
|
|
76
|
+
minCycle = cycle
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
let maxValue = this.getMaxY()
|
|
80
|
+
let cycleNum = Math.ceil(maxValue * 3 / minCycle)
|
|
81
|
+
return cycleNum
|
|
82
|
+
}
|
|
83
|
+
getSeries () {
|
|
84
|
+
let _vue = this
|
|
85
|
+
let seriesList = []
|
|
86
|
+
let dataTemplate = this.getInitBarData()
|
|
87
|
+
let obj = {
|
|
88
|
+
type: 'bar',
|
|
89
|
+
clickable: false,
|
|
90
|
+
stack: '信号灯0',
|
|
91
|
+
barWidth: 6,
|
|
92
|
+
silent: true,
|
|
93
|
+
itemStyle: {
|
|
94
|
+
color: '#009f3c'
|
|
95
|
+
},
|
|
96
|
+
data: dataTemplate,
|
|
97
|
+
label: {
|
|
98
|
+
show: true,
|
|
99
|
+
position: 'bottom',
|
|
100
|
+
offset: [0, 5],
|
|
101
|
+
color: getTheme() === 'light' ? '#666666' : '#B9BABF',
|
|
102
|
+
formatter: function (val) {
|
|
103
|
+
return _vue.getXName(val.data[0])
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
seriesList.push(obj)
|
|
108
|
+
let allInterList = []
|
|
109
|
+
let forwardCycleDta = this.getForwardIntersectionCycle()
|
|
110
|
+
let backCycleDta = this.getBackIntersectionCycle()
|
|
111
|
+
allInterList.push(backCycleDta)
|
|
112
|
+
allInterList.push(forwardCycleDta)
|
|
113
|
+
let cycleNum = this.getCycleNum() // 计算时距图上显示几个周期(以周期最小的路口计算)
|
|
114
|
+
for (let s = 0; s < allInterList.length; s++) {
|
|
115
|
+
for (let t = 0; t < cycleNum; t++) {
|
|
116
|
+
for (let i = 0; i < 4; i++) {
|
|
117
|
+
let tempObj = {}
|
|
118
|
+
tempObj.type = 'bar'
|
|
119
|
+
tempObj.clickable = false
|
|
120
|
+
tempObj.stack = '信号灯' + s
|
|
121
|
+
tempObj.barWidth = 6
|
|
122
|
+
tempObj.silent = true
|
|
123
|
+
tempObj.itemStyle = {}
|
|
124
|
+
tempObj.data = []
|
|
125
|
+
if (i % 2 !== 0) {
|
|
126
|
+
tempObj.itemStyle.color = '#009f3c'
|
|
127
|
+
} else {
|
|
128
|
+
tempObj.itemStyle.color = '#ca0d0d'
|
|
129
|
+
}
|
|
130
|
+
for (let cycle of allInterList[s]) {
|
|
131
|
+
let cycleList = []
|
|
132
|
+
cycleList[0] = cycle.distance
|
|
133
|
+
cycleList[1] = cycle.value[i]
|
|
134
|
+
tempObj.data.push(cycleList)
|
|
135
|
+
}
|
|
136
|
+
seriesList.push(tempObj)
|
|
137
|
+
}
|
|
138
|
+
for (let cyc of allInterList[s]) {
|
|
139
|
+
let addCycle = cyc.cycle
|
|
140
|
+
let val = cyc.value
|
|
141
|
+
cyc.value = this.calculationNewCycle(val, addCycle)
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
}
|
|
145
|
+
let newSeriesList = seriesList.reverse()
|
|
146
|
+
// 画时距图阴影部分
|
|
147
|
+
let customList = []
|
|
148
|
+
let dir = this.rourte.direction
|
|
149
|
+
if (dir === 'up') {
|
|
150
|
+
let forwardCustomData = this.getCustomData('forward') // 获取正向阴影面积的数据
|
|
151
|
+
customList.push(forwardCustomData)
|
|
152
|
+
} else if (dir === 'down') {
|
|
153
|
+
let backCustomData = this.getCustomData('back') // 获取反向阴影部分面积数据
|
|
154
|
+
customList.push(backCustomData)
|
|
155
|
+
} else if (dir === 'all') {
|
|
156
|
+
let forwardCustomData = this.getCustomData('forward') // 获取正向阴影面积的数据
|
|
157
|
+
let backCustomData = this.getCustomData('back') // 获取反向阴影部分面积数据
|
|
158
|
+
customList.push(forwardCustomData)
|
|
159
|
+
customList.push(backCustomData)
|
|
160
|
+
}
|
|
161
|
+
for (let customl of customList) {
|
|
162
|
+
for (let custom of customl) {
|
|
163
|
+
let data = []
|
|
164
|
+
data.push(custom)
|
|
165
|
+
let customObj = {}
|
|
166
|
+
customObj.type = 'custom'
|
|
167
|
+
customObj.renderItem = this.renderItem
|
|
168
|
+
customObj.silent = true
|
|
169
|
+
customObj.data = data // data数据格式为[n1,n2,n3,n4,n5,n6]n1到n4为阴影面的四个带你的y坐标,n5和n6分别表示x轴上的两点
|
|
170
|
+
newSeriesList.push(customObj)
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
// 画时距图虚线部分
|
|
174
|
+
if (this.greenwave.length > 0) {
|
|
175
|
+
let allLine = []
|
|
176
|
+
let dir = this.rourte.direction
|
|
177
|
+
let greenwave = this.greenwave
|
|
178
|
+
let upGreenware = greenwave.filter(ele => ele.type === 'up')
|
|
179
|
+
let downGreenware = greenwave.filter(ele => ele.type === 'down')
|
|
180
|
+
if (dir === 'down') {
|
|
181
|
+
let lineDataBack = this.getLineData('down', downGreenware)
|
|
182
|
+
allLine.push(lineDataBack)
|
|
183
|
+
} else if (dir === 'up') {
|
|
184
|
+
let lineDataFoward = this.getLineData('up', upGreenware)
|
|
185
|
+
allLine.push(lineDataFoward)
|
|
186
|
+
} else if (dir === 'all') {
|
|
187
|
+
let lineDataFoward = this.getLineData('down', downGreenware)
|
|
188
|
+
let lineDataBack = this.getLineData('up', upGreenware)
|
|
189
|
+
allLine.push(lineDataFoward)
|
|
190
|
+
allLine.push(lineDataBack)
|
|
191
|
+
}
|
|
192
|
+
// let lineData = this.getLineData()
|
|
193
|
+
for (let lineData of allLine) {
|
|
194
|
+
for (let line of lineData) {
|
|
195
|
+
let lineObj = {
|
|
196
|
+
lineStyle: {
|
|
197
|
+
normal: {
|
|
198
|
+
color: '#009f3c',
|
|
199
|
+
width: 1,
|
|
200
|
+
type: 'dashed'
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
lineObj.name = '最优通行方案'
|
|
205
|
+
lineObj.type = 'line'
|
|
206
|
+
lineObj.data = line
|
|
207
|
+
newSeriesList.push(lineObj)
|
|
208
|
+
}
|
|
209
|
+
}
|
|
210
|
+
// for (let line of lineData) {
|
|
211
|
+
// let lineObj = {
|
|
212
|
+
// lineStyle: {
|
|
213
|
+
// normal: {
|
|
214
|
+
// color: '#009f3c',
|
|
215
|
+
// width: 1,
|
|
216
|
+
// type: 'dashed'
|
|
217
|
+
// }
|
|
218
|
+
// }
|
|
219
|
+
// }
|
|
220
|
+
// lineObj.name = '最优通行方案'
|
|
221
|
+
// lineObj.type = 'line'
|
|
222
|
+
// lineObj.data = line
|
|
223
|
+
// newSeriesList.push(lineObj)
|
|
224
|
+
// }
|
|
225
|
+
}
|
|
226
|
+
return newSeriesList
|
|
227
|
+
}
|
|
228
|
+
getLineData (direction, greenwave) {
|
|
229
|
+
let data = []
|
|
230
|
+
let rourte = this.rourte
|
|
231
|
+
// let greenwave = this.greenwave
|
|
232
|
+
if (greenwave.length === 0) return data
|
|
233
|
+
let routeList = rourte.devs
|
|
234
|
+
let firstXLine = routeList[0].distance // 虚线起点的x坐标
|
|
235
|
+
let lastXLine = this.getMaxX() // 虚线终点的x坐标
|
|
236
|
+
let firstYLine = greenwave[0].start // 虚线起点的y坐标
|
|
237
|
+
let firstYLine2 = greenwave[0].width + firstYLine // 第二条虚线起点的y坐标
|
|
238
|
+
let speed = greenwave[0].speed
|
|
239
|
+
let lastYLine = 0
|
|
240
|
+
if (direction === 'down') {
|
|
241
|
+
lastYLine = Number(((lastXLine - firstXLine) / (speed * 1000 / 3600)).toFixed(2)) * (-1) + firstYLine // 下行虚线终点的y坐标
|
|
242
|
+
} else if (direction === 'up') {
|
|
243
|
+
lastYLine = Number(((lastXLine - firstXLine) / (speed * 1000 / 3600)).toFixed(2)) + firstYLine // 上行虚线终点的y坐标
|
|
244
|
+
}
|
|
245
|
+
let lastYLine2 = lastYLine + greenwave[0].width // 第二条虚线终点的y坐标
|
|
246
|
+
let maxY = this.getMaxY() * 3
|
|
247
|
+
let firstDevCycle = rourte.devs[0].patternList[0].cycle
|
|
248
|
+
let lineNum = Math.ceil(maxY / firstDevCycle)
|
|
249
|
+
for (let i = 0; i < lineNum; i++) { // 根据周期画出多段虚线
|
|
250
|
+
let dataList = []
|
|
251
|
+
let tempList = []
|
|
252
|
+
let addNum = firstDevCycle * i
|
|
253
|
+
tempList[0] = firstXLine
|
|
254
|
+
tempList[1] = firstYLine + addNum
|
|
255
|
+
dataList.push(tempList)
|
|
256
|
+
tempList = []
|
|
257
|
+
tempList[0] = lastXLine
|
|
258
|
+
tempList[1] = lastYLine + addNum
|
|
259
|
+
dataList.push(tempList)
|
|
260
|
+
data.push(dataList)
|
|
261
|
+
tempList = []
|
|
262
|
+
dataList = []
|
|
263
|
+
tempList[0] = firstXLine
|
|
264
|
+
tempList[1] = firstYLine2 + addNum
|
|
265
|
+
dataList.push(tempList)
|
|
266
|
+
tempList = []
|
|
267
|
+
tempList[0] = lastXLine
|
|
268
|
+
tempList[1] = lastYLine2 + addNum
|
|
269
|
+
dataList.push(tempList)
|
|
270
|
+
data.push(dataList)
|
|
271
|
+
}
|
|
272
|
+
return data
|
|
273
|
+
}
|
|
274
|
+
getCustomData (direction) {
|
|
275
|
+
let rourte = this.rourte
|
|
276
|
+
let customList = []
|
|
277
|
+
let routeList = rourte.devs
|
|
278
|
+
if (routeList.length === 1) return customList // 如果只有一个路口则不画阴影部分
|
|
279
|
+
let speed = 0
|
|
280
|
+
if (direction === 'forward') {
|
|
281
|
+
speed = rourte.upspeed
|
|
282
|
+
}
|
|
283
|
+
if (direction === 'back') {
|
|
284
|
+
speed = rourte.downspeed
|
|
285
|
+
}
|
|
286
|
+
let dist = 0 // 计算每个路口到原点的距离
|
|
287
|
+
for (let i = 0; i < routeList.length - 1; i++) {
|
|
288
|
+
dist = dist + routeList[i].distance
|
|
289
|
+
if (direction === 'forward') {
|
|
290
|
+
let customArea = this.getCustomArea(routeList[i], routeList[i + 1], speed, dist, 'forward')
|
|
291
|
+
let firstDistance = dist // 表示第一个路口距离原点的距离
|
|
292
|
+
let secondDistance = dist + routeList[i + 1].distance // 表示第二个路口到原点的距离
|
|
293
|
+
for (let area of customArea) {
|
|
294
|
+
area.push(firstDistance)
|
|
295
|
+
area.push(secondDistance)
|
|
296
|
+
customList.push(area)
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
if (direction === 'back') {
|
|
300
|
+
let firstDistance = dist // 表示第一个路口距离原点的距离
|
|
301
|
+
let secondDistance = dist + routeList[i + 1].distance // 表示第二个路口到原点的距离
|
|
302
|
+
let customArea = this.getCustomArea(routeList[i + 1], routeList[i], speed, secondDistance, 'back')
|
|
303
|
+
for (let area of customArea) {
|
|
304
|
+
area.push(secondDistance)
|
|
305
|
+
area.push(firstDistance)
|
|
306
|
+
customList.push(area)
|
|
307
|
+
}
|
|
308
|
+
}
|
|
309
|
+
}
|
|
310
|
+
return customList
|
|
311
|
+
}
|
|
312
|
+
getCustomArea (intersection1, intersection2, speed, dist, direction) {
|
|
313
|
+
speed = (speed * 1000 / 3600).toFixed(2)
|
|
314
|
+
let maxY = this.getMaxY()
|
|
315
|
+
let intersectionCycle1
|
|
316
|
+
let intersectionCycle2
|
|
317
|
+
let distance
|
|
318
|
+
if (direction === 'forward') {
|
|
319
|
+
let rings1 = this.getByRingById(intersection1.forwardphaseid, intersection1.patternList[0].rings)
|
|
320
|
+
let rings2 = this.getByRingById(intersection2.forwardphaseid, intersection2.patternList[0].rings)
|
|
321
|
+
intersectionCycle1 = this.getCycleData(intersection1.forwardphaseid, rings1, intersection1.patternList[0].cycle, intersection1.patternList[0].offset)
|
|
322
|
+
intersectionCycle2 = this.getCycleData(intersection2.forwardphaseid, rings2, intersection2.patternList[0].cycle, intersection2.patternList[0].offset)
|
|
323
|
+
distance = intersection2.distance
|
|
324
|
+
}
|
|
325
|
+
if (direction === 'back') {
|
|
326
|
+
let rings3 = this.getByRingById(intersection1.backphaseid, intersection1.patternList[0].rings)
|
|
327
|
+
let rings4 = this.getByRingById(intersection2.backphaseid, intersection2.patternList[0].rings)
|
|
328
|
+
intersectionCycle1 = this.getCycleData(intersection1.backphaseid, rings3, intersection1.patternList[0].cycle, intersection1.patternList[0].offset)
|
|
329
|
+
intersectionCycle2 = this.getCycleData(intersection2.backphaseid, rings4, intersection2.patternList[0].cycle, intersection2.patternList[0].offset)
|
|
330
|
+
distance = intersection1.distance
|
|
331
|
+
}
|
|
332
|
+
let intersectionList1 = this.getGreenAreaList(intersectionCycle1, maxY, intersection1.patternList[0].cycle) // 获取第一个路口绿色部分的集合
|
|
333
|
+
let intersectionList2 = this.getGreenAreaList(intersectionCycle2, maxY, intersection2.patternList[0].cycle) // 获取第二个路口绿色部分的集合
|
|
334
|
+
let time = parseInt(distance / speed) // 时间 = 距离 / 速度
|
|
335
|
+
let customAreaList
|
|
336
|
+
if (direction === 'forward') {
|
|
337
|
+
customAreaList = this.getCustomAreaList(intersectionList1, intersectionList2, time, maxY, speed, dist, 'forward')
|
|
338
|
+
}
|
|
339
|
+
if (direction === 'back') {
|
|
340
|
+
customAreaList = this.getCustomAreaList(intersectionList1, intersectionList2, time, maxY, speed, dist, 'back')
|
|
341
|
+
}
|
|
342
|
+
return customAreaList
|
|
343
|
+
}
|
|
344
|
+
getCustomAreaList (intlist1, intlist2, time, maxY, speed, dist, direction) {
|
|
345
|
+
let shadowList = []
|
|
346
|
+
for (let int1 of intlist1) {
|
|
347
|
+
let shadow = []
|
|
348
|
+
for (let int2 of intlist2) {
|
|
349
|
+
let min = int1[0] + time
|
|
350
|
+
let max = int1[1] + time
|
|
351
|
+
if (min === int2[0] && max === int2[1]) {
|
|
352
|
+
shadow = [int1[0], int1[1], int2[1], int2[0]]
|
|
353
|
+
} else if (min < int2[0] && max >= int2[1]) {
|
|
354
|
+
shadow = [int2[0] - time, int2[1] - time, int2[1], int2[0]]
|
|
355
|
+
} else if (min >= int2[0] && max < int2[1]) {
|
|
356
|
+
shadow = [int1[0], int1[1], int1[1] + time, int1[0] + time]
|
|
357
|
+
} else if (min >= int2[0] && min < int2[1]) {
|
|
358
|
+
shadow = [int1[0], int2[1] - time, int2[1], int1[0] + time]
|
|
359
|
+
} else if (max > int2[0] && max <= int2[1]) {
|
|
360
|
+
shadow = [int2[0] - time, int1[1], int1[1] + time, int2[0]]
|
|
361
|
+
}
|
|
362
|
+
}
|
|
363
|
+
shadow = this.checkIsOverY(shadow, maxY, speed, dist, direction) // 检验所取数值是否大于y轴最大值,即阴影部分是否溢出
|
|
364
|
+
if (shadow.length !== 0) shadowList.push(shadow)
|
|
365
|
+
}
|
|
366
|
+
return shadowList
|
|
367
|
+
}
|
|
368
|
+
checkIsOverY (shadow, maxY, speed, dist, direction) {
|
|
369
|
+
let maxValue = maxY * 3 // 计算y轴最大值
|
|
370
|
+
if (shadow.length === 0) {
|
|
371
|
+
return []
|
|
372
|
+
} else if (shadow[0] >= maxValue || shadow[3] >= maxValue) {
|
|
373
|
+
return []
|
|
374
|
+
} else if (shadow[1] > maxValue && shadow[3] < maxValue) {
|
|
375
|
+
return [shadow[0], maxValue, shadow[3] + (maxValue - shadow[0]), shadow[3]]
|
|
376
|
+
} else if (shadow[2] > maxValue && shadow[3] < maxValue) { // 处理阴影面积部分溢出y轴的情况, return[n1, n2, n3, n4, n5], n1到n4为阴影面积四点的纵坐标值,n5为溢出点的横坐标值,n5对应的y轴value值为y轴最大值
|
|
377
|
+
let xValue = (maxValue - shadow[1]) * speed
|
|
378
|
+
if (shadow[1] > maxValue) {
|
|
379
|
+
return [shadow[0], maxValue, maxValue, shadow[3]]
|
|
380
|
+
} else {
|
|
381
|
+
if (direction === 'forward') {
|
|
382
|
+
return [shadow[0], shadow[1], maxValue, shadow[3], xValue + dist]
|
|
383
|
+
} else if (direction === 'back') {
|
|
384
|
+
return [shadow[0], shadow[1], maxValue, shadow[3], dist - xValue]
|
|
385
|
+
}
|
|
386
|
+
// return [shadow[0], shadow[1], maxValue, shadow[3], xValue + dist]
|
|
387
|
+
}
|
|
388
|
+
} else {
|
|
389
|
+
return shadow
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
getGreenAreaList (list, max, cycle) {
|
|
393
|
+
let greenAreaList = []
|
|
394
|
+
let cycleNum = Math.ceil(max * 3 / cycle)
|
|
395
|
+
for (let i = 0; i < cycleNum; i++) {
|
|
396
|
+
let tempList = []
|
|
397
|
+
let tempList1 = []
|
|
398
|
+
if (list[3] === 0) {
|
|
399
|
+
let val1 = (cycle * i) + list[0]
|
|
400
|
+
let val2 = (cycle * i) + list[1]
|
|
401
|
+
tempList.push(val1)
|
|
402
|
+
tempList.push(val2)
|
|
403
|
+
greenAreaList.push(tempList)
|
|
404
|
+
} else {
|
|
405
|
+
if (i === 0) {
|
|
406
|
+
let val1 = list[0]
|
|
407
|
+
let val2 = list[1]
|
|
408
|
+
let val3 = list[2]
|
|
409
|
+
let val4 = list[3] + list[1]
|
|
410
|
+
tempList.push(val1)
|
|
411
|
+
tempList.push(val2)
|
|
412
|
+
tempList1.push(val3)
|
|
413
|
+
tempList1.push(val4)
|
|
414
|
+
greenAreaList.push(tempList)
|
|
415
|
+
greenAreaList.push(tempList1)
|
|
416
|
+
} else {
|
|
417
|
+
let val3 = (cycle * i) + list[2]
|
|
418
|
+
let val4 = (cycle * i) + list[3] + list[1]
|
|
419
|
+
tempList.push(val3)
|
|
420
|
+
tempList.push(val4)
|
|
421
|
+
greenAreaList.push(tempList)
|
|
422
|
+
}
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
return greenAreaList
|
|
426
|
+
}
|
|
427
|
+
getForwardIntersectionCycle () { // 获取正向每个路口的相位周期数组
|
|
428
|
+
let rourte = this.rourte
|
|
429
|
+
let list = []
|
|
430
|
+
let distance = 0
|
|
431
|
+
for (let devs of rourte.devs) {
|
|
432
|
+
let ob = {}
|
|
433
|
+
ob.name = devs.agentid
|
|
434
|
+
distance = distance + devs.distance
|
|
435
|
+
ob.distance = distance
|
|
436
|
+
let cycle = devs.patternList[0].cycle
|
|
437
|
+
ob.cycle = cycle
|
|
438
|
+
let forwardphaseid = devs.forwardphaseid
|
|
439
|
+
let rings = this.getByRingById(forwardphaseid, devs.patternList[0].rings)
|
|
440
|
+
let offset = devs.patternList[0].offset
|
|
441
|
+
ob.value = this.getCycleData(forwardphaseid, rings, cycle, offset)
|
|
442
|
+
list.push(ob)
|
|
443
|
+
}
|
|
444
|
+
return list
|
|
445
|
+
}
|
|
446
|
+
getBackIntersectionCycle () { // 获取反向每个路口的相位周期数组
|
|
447
|
+
let rourte = this.rourte
|
|
448
|
+
let list = []
|
|
449
|
+
let distance = 0
|
|
450
|
+
for (let devs of rourte.devs) {
|
|
451
|
+
let ob = {}
|
|
452
|
+
ob.name = devs.agentid
|
|
453
|
+
distance = distance + devs.distance
|
|
454
|
+
ob.distance = distance
|
|
455
|
+
let cycle = devs.patternList[0].cycle
|
|
456
|
+
ob.cycle = cycle
|
|
457
|
+
let backphaseid = devs.backphaseid
|
|
458
|
+
let rings = this.getByRingById(backphaseid, devs.patternList[0].rings)
|
|
459
|
+
let offset = devs.patternList[0].offset
|
|
460
|
+
// ob.value = this.getCycleData(backphaseid, devs.patternList[0].rings, cycle, offset)
|
|
461
|
+
ob.value = this.getCycleData(backphaseid, rings, cycle, offset)
|
|
462
|
+
list.push(ob)
|
|
463
|
+
}
|
|
464
|
+
return list
|
|
465
|
+
}
|
|
466
|
+
calculationNewCycle (list, addNum) { // 将数组里的每个值加addnum
|
|
467
|
+
let ll = []
|
|
468
|
+
for (let li of list) {
|
|
469
|
+
li = li + addNum
|
|
470
|
+
ll.push(li)
|
|
471
|
+
}
|
|
472
|
+
return ll
|
|
473
|
+
}
|
|
474
|
+
getByRingById (phaseId, ringsList) {
|
|
475
|
+
for (let rings of ringsList) {
|
|
476
|
+
let ring = rings.filter(item => item.id === phaseId)
|
|
477
|
+
if (ring.length > 0) {
|
|
478
|
+
return rings
|
|
479
|
+
}
|
|
480
|
+
}
|
|
481
|
+
return []
|
|
482
|
+
}
|
|
483
|
+
getCycleData (phaseid, rings, cycle, offset) {
|
|
484
|
+
let tempList = [0, 0, 0, 0]
|
|
485
|
+
let num = rings.length - 1
|
|
486
|
+
if (rings.length === 0) return tempList
|
|
487
|
+
let val = 0
|
|
488
|
+
let greenNum = 0 // 表示时距图中一个周期绿色部分的高度
|
|
489
|
+
for (let i = 0; i < rings.length; i++) {
|
|
490
|
+
val = val + rings[i].value
|
|
491
|
+
if (rings[i].id === phaseid) {
|
|
492
|
+
greenNum = rings[i].value
|
|
493
|
+
// tempList[1] = greenNum
|
|
494
|
+
// tempList[2] = cycle
|
|
495
|
+
if (i === 0) {
|
|
496
|
+
tempList[1] = greenNum
|
|
497
|
+
tempList[2] = cycle
|
|
498
|
+
} else if (i === num) {
|
|
499
|
+
tempList[0] = cycle - greenNum
|
|
500
|
+
tempList[1] = cycle
|
|
501
|
+
} else {
|
|
502
|
+
tempList[0] = val - greenNum
|
|
503
|
+
tempList[1] = val
|
|
504
|
+
tempList[2] = cycle
|
|
505
|
+
}
|
|
506
|
+
}
|
|
507
|
+
}
|
|
508
|
+
// 根据offset对tempList数组进行调整
|
|
509
|
+
if ((offset % cycle) !== 0) {
|
|
510
|
+
let relativeOffset = offset % cycle
|
|
511
|
+
let greenVal = tempList[1]
|
|
512
|
+
if ((relativeOffset + greenVal) < cycle) {
|
|
513
|
+
tempList[0] = relativeOffset + greenVal - greenNum
|
|
514
|
+
tempList[1] = relativeOffset + greenVal
|
|
515
|
+
tempList[2] = cycle
|
|
516
|
+
tempList[3] = 0
|
|
517
|
+
} else if ((relativeOffset + greenVal) === cycle) {
|
|
518
|
+
tempList[0] = cycle - greenNum
|
|
519
|
+
tempList[1] = cycle
|
|
520
|
+
tempList[2] = 0
|
|
521
|
+
tempList[3] = 0
|
|
522
|
+
} else {
|
|
523
|
+
let greenOffset = relativeOffset + greenVal - cycle
|
|
524
|
+
// tempList[0] = 0
|
|
525
|
+
// tempList[1] = greenOffset
|
|
526
|
+
// tempList[2] = relativeOffset
|
|
527
|
+
// tempList[3] = cycle
|
|
528
|
+
if (greenOffset < greenNum) {
|
|
529
|
+
tempList[0] = 0
|
|
530
|
+
tempList[1] = greenOffset
|
|
531
|
+
tempList[2] = relativeOffset + greenVal - greenNum
|
|
532
|
+
tempList[3] = cycle
|
|
533
|
+
} else if (greenOffset === greenNum) {
|
|
534
|
+
tempList[0] = 0
|
|
535
|
+
tempList[1] = greenNum
|
|
536
|
+
tempList[2] = cycle
|
|
537
|
+
tempList[3] = 0
|
|
538
|
+
} else {
|
|
539
|
+
tempList[0] = greenOffset - greenNum
|
|
540
|
+
tempList[1] = greenOffset
|
|
541
|
+
tempList[2] = cycle
|
|
542
|
+
tempList[3] = 0
|
|
543
|
+
}
|
|
544
|
+
}
|
|
545
|
+
}
|
|
546
|
+
return tempList
|
|
547
|
+
}
|
|
548
|
+
renderItem (params, api) {
|
|
549
|
+
if (params.context.rendered) {
|
|
550
|
+
return
|
|
551
|
+
}
|
|
552
|
+
params.context.rendered = true
|
|
553
|
+
var points = []
|
|
554
|
+
if (!Number.isNaN(api.value(6))) { // 处理阴影面积溢出y轴的情况
|
|
555
|
+
points.push(api.coord([api.value(5), api.value(0)]))
|
|
556
|
+
points.push(api.coord([api.value(5), api.value(1)]))
|
|
557
|
+
points.push(api.coord([api.value(4), api.value(2)]))
|
|
558
|
+
points.push(api.coord([api.value(6), api.value(2)]))
|
|
559
|
+
points.push(api.coord([api.value(6), api.value(3)]))
|
|
560
|
+
} else { // 处理正常情况
|
|
561
|
+
points.push(api.coord([api.value(4), api.value(0)]))
|
|
562
|
+
points.push(api.coord([api.value(4), api.value(1)]))
|
|
563
|
+
points.push(api.coord([api.value(5), api.value(2)]))
|
|
564
|
+
points.push(api.coord([api.value(5), api.value(3)]))
|
|
565
|
+
}
|
|
566
|
+
return {
|
|
567
|
+
type: 'polygon',
|
|
568
|
+
shape: {
|
|
569
|
+
// points: points
|
|
570
|
+
points: echarts.graphic.clipPointsByRect(points, {
|
|
571
|
+
x: params.coordSys.x,
|
|
572
|
+
y: params.coordSys.y,
|
|
573
|
+
width: params.coordSys.width,
|
|
574
|
+
height: params.coordSys.height
|
|
575
|
+
})
|
|
576
|
+
},
|
|
577
|
+
style: api.style({
|
|
578
|
+
fill: 'rgba(0, 159, 60, 0.2)'
|
|
579
|
+
// stroke: echarts.color.lift(color)
|
|
580
|
+
})
|
|
581
|
+
}
|
|
582
|
+
}
|
|
583
|
+
RenderOption () {
|
|
584
|
+
return {
|
|
585
|
+
// legend: {
|
|
586
|
+
// data: ['红灯1', '绿灯1']
|
|
587
|
+
// },
|
|
588
|
+
tooltip: {
|
|
589
|
+
// trigger: 'axis',
|
|
590
|
+
axisPointer: {
|
|
591
|
+
type: 'cross',
|
|
592
|
+
label: {
|
|
593
|
+
backgroundColor: '#6a7985'
|
|
594
|
+
}
|
|
595
|
+
},
|
|
596
|
+
formatter: function (params) {
|
|
597
|
+
return ''
|
|
598
|
+
}
|
|
599
|
+
},
|
|
600
|
+
grid: {
|
|
601
|
+
left: '3%',
|
|
602
|
+
right: '4%',
|
|
603
|
+
bottom: '3%',
|
|
604
|
+
containLabel: true
|
|
605
|
+
},
|
|
606
|
+
xAxis: [
|
|
607
|
+
{
|
|
608
|
+
type: 'value',
|
|
609
|
+
name: '距离(m)',
|
|
610
|
+
nameGap: 5,
|
|
611
|
+
axisTick: {
|
|
612
|
+
show: false,
|
|
613
|
+
alignWithLabel: false
|
|
614
|
+
},
|
|
615
|
+
splitLine: {
|
|
616
|
+
show: false
|
|
617
|
+
},
|
|
618
|
+
axisLabel: { // 可以设置x轴的value格式
|
|
619
|
+
formatter: function (val) {
|
|
620
|
+
return ''
|
|
621
|
+
},
|
|
622
|
+
interval: 0
|
|
623
|
+
// rotate: 30
|
|
624
|
+
},
|
|
625
|
+
nameTextStyle: {
|
|
626
|
+
color: getTheme() === 'light' ? '#666666' : '#B9BABF'
|
|
627
|
+
},
|
|
628
|
+
min: -50,
|
|
629
|
+
max: this.getMaxX() + 50
|
|
630
|
+
}
|
|
631
|
+
],
|
|
632
|
+
yAxis: [
|
|
633
|
+
{
|
|
634
|
+
type: 'value',
|
|
635
|
+
name: '时间(s)',
|
|
636
|
+
axisLine: {
|
|
637
|
+
show: true,
|
|
638
|
+
onZero: false
|
|
639
|
+
},
|
|
640
|
+
axisTick: {
|
|
641
|
+
show: false
|
|
642
|
+
},
|
|
643
|
+
splitLine: {
|
|
644
|
+
show: true,
|
|
645
|
+
lineStyle: {
|
|
646
|
+
color: getTheme() === 'light' ? '#DCDFE6' : '#30384d'
|
|
647
|
+
}
|
|
648
|
+
},
|
|
649
|
+
nameTextStyle: {
|
|
650
|
+
color: getTheme() === 'light' ? '#666666' : '#B9BABF'
|
|
651
|
+
},
|
|
652
|
+
axisLabel: {
|
|
653
|
+
textStyle: {
|
|
654
|
+
color: getTheme() === 'light' ? '#666666' : '#B9BABF'
|
|
655
|
+
}
|
|
656
|
+
},
|
|
657
|
+
min: 0,
|
|
658
|
+
max: this.getMaxY() * 3
|
|
659
|
+
// show: false
|
|
660
|
+
}
|
|
661
|
+
],
|
|
662
|
+
series: this.getSeries()
|
|
663
|
+
}
|
|
664
|
+
}
|
|
665
|
+
}
|