0%

开发某机床厂远程运维平台的前端技术总结(2)

在昆山团队退出后,我们还额外负责了原来昆山做的【工作台】、【统计报表】部分。本文的代码在他们的基础上开发,大部分我做了一定幅度的改动。

4 统计报表

【统计报表】模块用于展示该公司的设备销售数据或故障数据。从这里开始,其实已经是纯前端工作,之后用到了大量的图表,我都是使用 Apache ECharts 进行开发的。

我负责【统计报表】-【统计分析】的【设备分析】部分,即分析公司在每个省份、每种设备类型各卖了多少。这个页面有两个图,分别是【各地区设备统计】和【设备销售类型销量分析】。由于没有从后端拿数据,数据在客户提供的 Excel 里,所以我这里将数据写死在前端。

各地区设备统计

设备销售类型销量分析

4.1 各地区设备统计

这其实是一个图表,只是分成了左右两个图,左边是一个柱状堆积图,右边是一个饼图。

4.1.1 图表组件

为了更好地管理图表,我将所有的图表都写成了 Vue 组件(Component)的形式,而非写在当前页面。在当前页面导入组件,并注册组件使用它。

1
2
3
<div>
<region_echart ref="region"></region_echart>
</div>
1
2
3
4
5
6
7
import region_echart from "@/components/statisticalReport/echarts/region_echart";
...

export default {
components: {region_echart, ...},
...
}

当父组件(页面)要使用子组件(图表)的数据时,例如父组件要调用子组件的函数,可以在子组件添加 ref 属性寻找它。

1
2
3
4
/** 搜索按钮操作 */
handleQuery() {
this.$refs.region.getProvinceData();
},

当子组件(图表)要使用父组件(页面)的数据时,例如子组件想知道父组件的搜索框是什么条件,可以使用 this.$parent 向上寻找父组件。注意:由于 Vue 实际上可能套了好几层 div,所以要寻找好几层才能到目标父组件,可以在控制台 console.log 一下 this 及其父元素。

1
2
3
4
5
6
7
8
9
10
11
getProvinceData() {
if (this.$parent.$parent.queryParams.provinces.length > 0) {
this.provinceData = this.$parent.$parent.queryParams.provinces;
} else {
this.provinceData = [];
for (let province of regionData) {
this.provinceData.push(province.label);
}
}
...
},

4.1.2 数据添加

这个表需要每个省份每种机床卖出了多少,因此我写在前端里的数据是一个大对象,属性是每个省份,每个省份的值又是一个对象,对应机床类型和销量。

1
2
3
4
5
data: {
'北京市': {'YGS3A...型': 3, 'YGS3B...型': 1, 'YGS3C...型': 2},
'上海市': {'YGS3A...型': 3, 'YGS3B...型': 10, 'YGS3C...型': 2, ...},
...
},

首先我们拿到这些数据后,要处理成 Echarts 支持的数据形式。这里我们使用数组的形式。先添加数据:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
getProvinceData() {
if (this.$parent.$parent.queryParams.provinces.length > 0) {
this.provinceData = this.$parent.$parent.queryParams.provinces;
} else {
this.provinceData = [];
for (let province of regionData) {
this.provinceData.push(province.label);
}
}
...
},
getTypeData() {
listType().then(response => {
if (this.$parent.$parent.queryParams.deviceTypes.length > 0) {
this.typeData = this.$parent.$parent.queryParams.deviceTypes;
} else {
this.typeData = [];
for (let type of response.rows) {
this.typeData.push(type.devicetypeName);
}
}
...
});
},
getDeviceNumber() {
this.deviceNumber = {};
this.deviceNumber.total = [];
for (let type of this.typeData) {
this.deviceNumber[type] = [];
for (let province in this.provinceData) {
this.deviceNumber[type].push(0);
}
}
for (let province in this.provinceData) {
this.deviceNumber.total.push(0);
}
for (let province in this.data) {
for (let type in this.data[province]) {
if (this.provinceData.indexOf(province) > -1 && this.typeData.indexOf(type) > -1) {
this.deviceNumber[type][this.provinceData.indexOf(province)] = this.data[province][type];
this.deviceNumber.total[this.provinceData.indexOf(province)] += this.data[province][type];
}
}
}
console.log(this.deviceNumber)
...
},

由于排序和饼图需要计算每个省份的设备总数,所以我们给 deviceNumber 对象加一个属性 total,其值为数组。provinceData 的顺序是确定的,它的索引对应 total 数组里和每个 type 数组里的索引。再根据已有数据,给对应类型的省份索引位置加数据,总数也要加数据,最终得到:

console.log(this.deviceNumber)

每个数组 34 个元素对应 34 个省级行政区,其中每个位置是哪个省由 provinceData 数组确定。

4.1.3 数据处理

以上数据还不能被图表使用,需要做进一步处理。

首先对于柱形图,我们需要让它按总数数量排序。Echarts 是不能做到这个功能的,所以我们需要处理数据源。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
transformData() {
let flag = 0;
let flagData = [];
for (let x = 0; x < this.deviceNumber.total.length; x++) {
flag = 0;
for (let i = 0; i < this.deviceNumber.total.length; i++) {
if (this.deviceNumber.total[x] > this.deviceNumber.total[i]) {
flag++;
}
}
flagData.push(flag);
}
for (let x = 0; x < flagData.length; x++) {
flag = 0;
for (let i = x + 1; i < flagData.length; i++) {
if (flagData[x] === flagData[i]) {
flagData[i] += 1;
}
}
}
this.sortedDeviceNumber = {
total: []
};
for (let type of this.typeData) {
this.sortedDeviceNumber[type] = []
}
this.sortedProvinceData = [];
for (let i in flagData) {
for (let key in this.deviceNumber) {
this.sortedDeviceNumber[key][flagData[i]] = this.deviceNumber[key][i];
}
this.sortedProvinceData[flagData[i]] = this.provinceData[i];
}
...
},

以上代码 4-12 行,我们得到的 flagData 是由 34 个数字组成的数组,代表了每个省份的数量排序,即对于某个省份数量,有多少个省份数量比它小,那它就排多少位。13-20 行处理并列的省份,这样就得到了从小到大的省份新顺序。再根据这个新顺序,将数量与省份赋值到 sortedDeviceNumbersortedProvinceData 里。柱形图就可以用这个数据了。

对于饼图,我们只需要省份和总数,注意总数为 0 时,将总数设置为 null,这样它就会被视为无效数据,不会出现在饼图上。

1
2
3
4
5
6
7
8
9
10
getPieData() {
this.pieData = [];
for (let i = this.sortedProvinceData.length - 1; i >= 0; i--) {
this.pieData.push({
name: this.sortedProvinceData[i],
value: this.sortedDeviceNumber.total[i] === 0 ? null : this.sortedDeviceNumber.total[i],
})
}
...
},

4.1.4 柱形图和饼形图数据传入

这两个图属于一个图表,因此 Echarts 的 series 数量为 n+1,n 是设备类型数(表示柱形图),1 表示饼图。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
getSeries() {
this.series = [];
for (let type of this.typeData) {
this.series.push({
name: type,
type: 'bar', // 表示柱形图
stack: 'total', // 表示堆积图
label: {
show: true,
position: 'insideRight', // 标签数值放在柱形右方
formatter: function (params) {
if (params.value < 10) {
return '';
} else {
return params.value;
}
}
},
barCategoryGap: '30%', // 柱形之间间隔
data: this.sortedDeviceNumber[type], // 数据使用数组
});
}
this.series.push({
type: 'pie', // 表示饼图
radius: [30, '60%'], // 内半径30像素,外半径为容器的60%
center: ['75%', '45%'], // 圆心位置
data: this.pieData,
label: {
fontSize: 16,
formatter: function (params) {
if (params.percent >= 1) {
return params.name + ':\n' + params.value + '(' + Math.round(params.percent) + '%)';
}
else {
return params.name + ':\n' + params.value + '(<1%)';
}
}
},
labelLine: {
smooth: true, // 平滑标签线
lineStyle: {
width: 2
}
},
tooltip: { // 悬浮提示框
trigger: 'item', // 触发类型为饼图项目
formatter: function (params) {
return params.name + '<br>' + params.marker + params.value + '(' + params.percent + '%)';
}
},
itemStyle: {
shadowBlur: 30,
shadowColor: 'rgba(0, 0, 0, 0.4)'
},
animationType: 'scale',
animationEasing: 'elasticOut',
animationDelay: function (idx) {
return Math.random() * 200;
}
})
...
},

以上代码 11-16 行,当柱形图某个数据小于 10 时,则不显示标签数值,以免影响美观,或出现多个 0 重叠的情况。

柱形图标签用函数格式化

以上代码 30-36 行,当饼图某个数据项占比小于 1% 时,就显示 <1%,以免被圆整为 0%。47-49 行,处理了饼图悬浮提示框的显示形式,用到了 params 即每个数据项的几个参数。

饼图标签与悬浮提示框用函数格式化

4.1.5 生成图表

生成图表用典型的 init 方法。

1
2
3
4
<template>
<div style="width: 100%; height: 900px" ref="dev_chart">
</div>
</template>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
initEcharts() {
this.chart = echarts.init(this.$refs.dev_chart, 'macarons');
this.chart.setOption({
tooltip: { // 设置柱形图标签,见上上图
trigger: 'axis',
axisPointer: {
type: 'shadow'
}
},
legend: { // 图例,使用设备类型,可以同时操作柱形图和饼图
data: this.typeData,
textStyle: {
color: '#fff',
fontSize: 14
},
left: '0%',
top: '0%',
width: '60%'
},
grid: { // 柱形图范围
left: '5%',
right: '60%',
top: '8%',
bottom: '10%',
containLabel: true,
},
xAxis: {
name: '设备数量/台',
type: 'value',
position: 'top', // x轴位置
splitLine: {show: false},
boundaryGap: [0, 0.01],
axisTick: { // 坐标轴刻度
show: false
},
axisLabel: { // 坐标轴标签
textStyle: {
color: 'white',
fontSize: 14
}
},
axisLine: { // 坐标轴线
show: true,
lineStyle: {
color: '#fff'
}
},
},
yAxis: {
name: '省级行政区',
nameLocation: 'start', // 名称显示位置
type: 'category', // 类目轴
data: this.sortedProvinceData,
axisTick: { // 坐标轴分隔线
show: false
},
splitLine: {
show: false
},
axisLabel: {
textStyle: {
color: '#fff',
fontSize: 14
}
},
axisLine: {
show: true,
lineStyle: {
color: '#fff'
}
},
},
series: this.series
}, true);
}

注意以上代码第 2 行,如果要使用 macarons 主题,需要找到它的位置并导入。

1
2
import echarts from 'echarts'
import "../../../../node_modules/echarts/theme/macarons";

4.1.6 解决获取不到属性的报错

之后的代码前端经常会报获取不到属性的错误,这甚至可能导致图表无法显示,为此有以下解决办法:

  • 如果是在 Vue 部分直接使用对象的属性,可以先用 v-if 判断其是否存在。如以下代码,staticInfo 本来是个空对象,是在调用函数后才赋值,如果没有 v-if,函数调用延迟时,Vue 尝试填充此处的文本插值,就会报找不到 ncsId 属性的错。

    1
    <div class="cell" v-if="staticInfo.ncsId">{{ staticInfo.ncsId }}</div>
  • 使用函数内部调用下一个函数,以强行控制函数执行流程,确保获取数据,再操作数据,如:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    14
    15
    created() {
    this.getList;
    },
    methods: {
    getList() {
    ListInfo().then(response => {
    ...
    this.handleData();
    });
    },
    handleData() {
    ...
    },
    ...
    }
  • 将生成图表的函数放在 mountednextTick 里,或直接设置整个函数体位于 nextTick 里,如:

    1
    2
    3
    4
    5
    6
    7
    8
    9
    10
    11
    12
    13
    mounted() {
    this.$nextTick(() => {
    this.initEchart();
    });
    },
    methods: {
    initEchart() {
    this.$nextTick(() => {
    let chart = echarts.init(this.$refs.dev_chart);
    chart.setOption(this.options);
    });
    },
    }

4.2 设备销售类型销量分析

这个图表是左右两张图,分别是柱形图和饼图,基本操作方法与之前类似。

4.2.1 让柱形图同一系列显示不同的颜色

柱形图其实只有一个系列,即设备数量。但为了让每一个柱形显示不同颜色,就要加单独的 itemStyle

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
getDeviceNumber() {
let color= [
"#4C98FB",
"#83CCE7",
"#26C7C8",
"#73DEBD",
"#26C978",
"#8CDF6C",
"#FBD657",
"#F56679",
"#E07BCE",
"#9D50E0",
"#634FDA"
];
this.deviceNumber = [];
for (let index in this.typeData) {
this.deviceNumber.push({
value: this.data[index],
itemStyle: {
color: color[index]
}
})
}
...
},

4.2.2 让柱形图横坐标标签旋转 45 度

横坐标标签过长,需要斜向放置,才能完整显示且更美观。

1
2
3
4
5
6
7
8
9
xAxis: {
name: '设备类型',
type: "category",
axisLabel: {
color: "#96A4F4",
rotate: 45, // 旋转角度
},
...
}

柱状图坐标标签旋转

饼图不用这么做,直接将 color 写在 option 里就好了,它会自动把不同颜色赋给不同的数据项。

5 工作台

【工作台】是个类似于「大屏看板」的功能区,分为【产品全地图】和【设备状态看板】两个模块。

5.1 产品全地图

【产品全地图】其实就是把【统计报表】里的设备分析图换了个累加方式,按省份累加并且展示到全国地图里。这边的代码是昆山写的,我只是改了其中的数据,就不展开讲了。

产品全地图页面

5.2 设备状态看板

【设备状态看板】展示了所有添加在【设备信息】里的设备,并根据采集到的数据展示它们的开关机状态,还可以点击每台设备右上角的按钮查看详细的监控数据。开关机状态所用的颜色来自 ElementUI 的调色板颜色

设备状态看板页面

5.2.1 右上角的时间

看板右上角的时间可以通过 js 的定时器完成。先在 Vue 中为其占位,并设置好 CSS 样式。

1
2
3
4
<div class="header_timer">
{{ nowDate }} {{ nowWeek }}<br>
<span style="font-size: 26px; font-weight: bold">{{ nowTime }}</span>
</div>
1
2
3
4
5
6
7
8
.header_timer {
color: rgb(255, 172, 80);
font-size: 20px;
font-weight: normal;
position: absolute;
top: 0%;
right: 3%;
}

定义一个函数,将当前时间写入变量:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
setNowTimes() {
let myDate = new Date();
let wk = myDate.getDay();
let yy = String(myDate.getFullYear());
let mm = myDate.getMonth() + 1;
let dd = String(myDate.getDate());
let hou = String(
myDate.getHours() < 10 ? "0" + myDate.getHours() : myDate.getHours()
);
let min = String(
myDate.getMinutes() < 10
? "0" + myDate.getMinutes()
: myDate.getMinutes()
);
let sec = String(
myDate.getSeconds() < 10
? "0" + myDate.getSeconds()
: myDate.getSeconds()
);
let weeks = [
"星期日",
"星期一",
"星期二",
"星期三",
"星期四",
"星期五",
"星期六",
];
let week = weeks[wk];
this.nowDate = yy + "年" + mm + "月" + dd + "日"
this.nowTime = hou + ":" + min + ":" + sec;
this.nowWeek = week;
}

再在 mounted 里设置一个定时器,设置为 1000 毫秒(即 1 秒)刷新一次就好,这样就实现了一个不断变化的时间。

1
2
3
4
5
mounted() {
this.timer = setInterval(() => {
this.setNowTimes();
}, 1000);
},

时间

5.2.2 搜索提示信息

客户要求,在上方搜索时,页面会提示它搜索的是什么企业或设备类型。这看起来是个很简单的工作,但着实费了我一阵功夫。

搜索信息

这个搜索功能和其他单表的搜索没有什么区别,都是获取 queryParams 然后调用 getList 函数。首先在页面的相关位置加上这三个提示的内容:

1
2
3
4
5
6
7
8
9
10
11
<el-row :gutter="10" v-show="isSearched" class="white-text medium-text" style="margin-bottom: 15px">
<el-col :lg="5" :xs="24" :sm="8">
<span style="font-weight: bold" v-show="focusEnter !== null">当前企业:</span>{{ focusEnter }}
</el-col>
<el-col :lg="5" :xs="24" :sm="8">
<span style="font-weight: bold" v-show="focusOrg !== null">当前部门:</span>{{ focusOrg }}
</el-col>
<el-col :lg="5" :xs="24" :sm="8">
<span style="font-weight: bold" v-show="focusType !== null">当前类型:</span>{{ focusType }}
</el-col>
</el-row>

定义布尔变量 isSearched,判断是否在进行搜索,以及三个变量放搜索值。在搜索和重置搜索函数上,也要对这些变量进行处理:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
/** 搜索按钮操作 */
handleQuery() {
this.queryParams.pageNum = 1;
this.getList();
this.getFocus();
if (this.focusEnter !== null || this.focusOrg !== null || this.focusType !== null) {
this.isSearched = true;
} else {
this.isSearched = false;
}
},
/** 重置按钮操作 */
resetQuery() {
this.resetForm("queryForm");
this.handleQuery();
this.isSearched = false;
this.focusEnter = null;
this.focusOrg = null;
this.focusType = null;
},

定义一个 getFocus 函数,用来把搜索值传给三个变量:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
getFocus() {
if (this.queryParams.enterpriseId !== null) {
for (let enter of this.enterOptions) {
if (enter.value === this.queryParams.enterpriseId) {
this.focusEnter = enter.label;
break;
}
}
}
if (this.queryParams.orgId !== null) {
for (let org of this.orgOptionsList) {
if (org.value === this.queryParams.orgId) {
this.focusOrg = org.label;
break;
}
}
}
if (this.queryParams.devicetypeId !== null) {
for (let type of this.typeOptions) {
if (type.value === this.queryParams.devicetypeId) {
this.focusType = type.label;
break;
}
}
}
},

这个当时写的代码明显可以用 find 函数替代啊,大意了。

当设备总数为 0 时,其实也要设计一个整体的页面,返回按钮与重置按钮功能一致,调用同一个函数。

1
2
3
4
5
6
7
8
9
10
<div class="device_container" style="margin-top: 15px">
<div v-if="total === 0" style="margin: 50px auto; height: 200px; text-align: center">
<p style="font-size: 20px; color: white">暂无数据</p>
<el-button size="medium" icon="el-icon-back" @click="resetQuery">返回</el-button>
</div>

<el-row :gutter="15" v-else>
...
</el-row>
</div>

设备总数为 0

5.2.3 纯前端功底:图例和设备框的设计

这我是完全用 HTML+CSS 写的,真正考验前端功底。上代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
<el-row :gutter="20" class="white-text medium-text">
<el-col :sm="5" :xs="24">
<div class="legend-total">
<span style="font-weight: bold">设备总数:</span>{{ total }}
</div>
</el-col>
<el-col :lg="3" :xs="24" :sm="6" :md="4">
<div class="legend-box" style="border-bottom-color: #67C23A">
<div class="legend-icon el-icon-set-up" style="background-color: #67C23A">
</div>
<div class="legend-text">开机</div>
</div>
</el-col>
<el-col :lg="3" :xs="24" :sm="6" :md="4">
<div class="legend-box" style="border-bottom-color: #F56C6C">
<div class="legend-icon el-icon-switch-button" style="background-color: #F56C6C">
</div>
<div class="legend-text">关机</div>
</div>
</el-col>
</el-row>

<div class="device_container" style="margin-top: 15px">
<div v-if="total === 0" ...>
... <!-- 这一部分就是上面的设备总数为0的界面 -->
</div>

<el-row :gutter="15" v-else>
<el-col :xl="4" :xs="24" :sm="12" :md="8" :lg="6" v-for="item in deviceList" :key="item.id"
style="margin-bottom: 15px">

<div class="box-card" v-loading="loading">
<div class="card-header"
:style="{'background-color': dynamicCheckStatusList[item.ncsId] ? '#67C23A' : '#F56C6C'}">
<span class="title">{{ item.deviceName }}</span>
<router-link :to="`/status/device/` + item.id">
<el-link icon="el-icon-monitor" :underline="false">
详细数据
</el-link>
</router-link>
</div>
<div class="card-main"
:style="{'border-color': dynamicCheckStatusList[item.ncsId] ? '#67C23A' : '#F56C6C'}">
<div class="el-table el-table--enable-row-hover el-table--medium">
<table cellspacing="0" style="width: 100%">
<tbody>
<tr>
<td>
<div class="label">地区</div>
</td>
<td>
<div v-if="item.enterpriseId">{{
enterOptions.find(enter => enter.value === item.enterpriseId) ? enterOptions.find(enter => enter.value === item.enterpriseId).province : ''
}}
</div>
</td>
</tr>
... <!-- 省略表格中的企业、类型代码 -->
<tr>
<td>
<div class="label">状态</div>
</td>
<td>
<div v-if="dynamicCheckStatusList">
{{ dynamicCheckStatusList[item.ncsId] ? "开机" : "关机" }}
</div>
</td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
</el-col>
</el-row>

以上代码第 29 行,使用 v-for 要绑定 key

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
.white-text {
color: white
}

.medium-text {
font-size: 18px;
}

.legend-total {
border-left: 5px red solid; /* 设备总数左边的小红条 */
padding-left: 10px;
margin: 5px;
}

.legend-box {
border-bottom: 3px solid;
height: 35px;
width: 130px;
display: flex;
justify-content: space-between;
align-items: center;
border-radius: 0 0 0 10px;
margin: 5px;
}

.legend-icon {
height: 32px;
width: 40px;
font-size: 25px;
text-align: center;
line-height: 35px;
border-radius: 5px 5px 0 5px;
}

.legend-text {
height: 32px;
width: 100px;
text-align: center;
line-height: 35px;
}

.card-header {
display: flex;
justify-content: space-between;
align-items: center;
padding: 5px 15px;
border-radius: 10px 10px 0 0;
height: 50px;
}

.title {
position: relative;
font-size: 20px;
color: #303133;
font-weight: bold;
}

.card-main {
background-color: #F2F6FC;
border: 3px solid;
border-radius: 0 0 10px 10px;
height: 180px;
display: flex;
justify-content: center;
align-items: center;
}

td {
background-color: #F2F6FC;
}

td div {
text-align: center;
}

.label {
font-weight: bold;
}

虽然现有的框架、组件非常好用,但要做出自己的效果还是需要靠 HTML+CSS 完成,这也让我体会到几十年前前端工程师的艰辛。

5.2.4 路由跳转

我们想让每台机床展现详细的监控数据,就做了一个按钮入口,这个实现方式有很多,比如 ElementUI 的 对话框弹出框抽屉 等。最终我们决定做一个单独的页面展示机床数据。

每个机床应该共用一个页面模板,但数据需要有所不同,可以根据机床的 id 进行区分。这用的是若依的 路由跳转

routerindex.js 里面,新配置一个路由:

1
2
3
4
5
6
7
8
9
10
11
12
13
{
path: '/status',
component: Layout,
hidden: true,
children: [
{
path: 'device/:id(\\d+)',
component: (resolve) => require(['@/views/staging/status/device'], resolve),
name: 'Device',
meta: { title: '设备详细数据', icon: '' }
}
]
}

同时设置路由跳转:

1
2
3
4
5
<router-link :to="`/status/device/` + item.id">
<el-link icon="el-icon-monitor" :underline="false">
详细数据
</el-link>
</router-link>

就可以跳转到指定的这个链接,链接所展示的页面是路由里 require 的页面,接下来编辑这个页面即可。

5.3 设备详细数据

这里展示每个机床的详细数据,不同的机床路径不同,数据不同。下图路径后的数字就是这台机床的 id。

设备详细数据页面

5.3.1 机床 id

机床 id 是最核心的数据,不仅这个页面的路径需要机床 id,页面所有的数据也通过机床 id 拿到。所以在 Vue 的 data 里,就定义好机床 id。

1
deviceId: this.$route.params && this.$route.params.id,

利用逻辑与的短路特性,如果获取到了 this.$route.param,就返回 id。

5.3.2 表格信息与变量绑定

所有的信息都是根据拿到的机床 id,然后使用后端的函数与前端的 api 查到的,包含静态数据(如设备归属信息)和动态数据(如设备运行状态)。特别要注意拿到的动态数据要看是否绑定,绑定的话还要显示用户自定义的名称和单位。

例如以下是设备基本信息的 Vue 代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
<el-col :lg="8" :md="12" class="card-box">
<el-card shadow="hover" style="margin-bottom: 15px; height: 315px">
<div slot="header" class="card-header">
<span style="font-weight: bold">设备基本信息</span>
</div>
<div class="el-table el-table--enable-row-hover el-table--medium">
<table cellspacing="0" style="width: 100%">
<tbody>
<tr v-if="this.showParam.hwProductSerialNrL">
<td>
<div class="cell">{{
bindingParamInfo.find(item => item.mongoName === 'hwProductSerialNrL').paramName
}}
</div>
</td>
<td>
<div class="cell" v-if="staticInfo.ncsId">{{ staticInfo.ncsId }}</div>
</td>
</tr>
<tr>
<td>
<div class="cell">设备名称</div>
</td>
<td>
<div class="cell" v-if="staticInfo.deviceName">{{ staticInfo.deviceName }}</div>
</td>
</tr>
<tr>
<td>
<div class="cell">设备启用状态</div>
</td>
<td>
<div class="cell" v-if="staticInfo.isStart !== null">{{
staticInfo.isStart === 0 ? "正常" : "停用"
}}
</div>
</td>
</tr>
<tr v-if="this.showParam.nckType">
<td>
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'nckType').paramName }}</div>
</td>
<td>
<div class="cell" v-if="!isEmptyObject(dynamicTimeInfo)">{{
dynamicTimeInfo.timePayLoad.nckType
}}
</div>
<div class="cell" v-else>(未找到)</div>
</td>
</tr>
<tr v-if="this.showParam.nckVersion">
<td>
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'nckVersion').paramName }}</div>
</td>
<td>
<div class="cell" v-if="!isEmptyObject(dynamicTimeInfo)">
{{ dynamicTimeInfo.timePayLoad.nckVersion }}
</div>
<div class="cell" v-else>(未找到)</div>
</td>
</tr>
</tbody>
</table>
</div>
</el-card>
</el-col>

再例如以下是设备运行状态的 Vue 代码:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
<el-col :lg="8" :md="24" class="card-box">
<el-card shadow="hover" style="margin-bottom: 15px; height: 315px">
<div slot="header" class="card-header">
<span style="font-weight: bold">设备运行状态</span>
<el-link type="primary" :underline="false" @click="handleDynamicRefresh">
刷新
</el-link>
</div>
<div class="el-table el-table--enable-row-hover el-table--medium">
<table cellspacing="0" style="width: 100%" v-loading="tableLoading">
<tbody v-if="!isEmptyObject(dynamicTimeInfo) || !isEmptyObject(dynamicProgStatus)">
<tr>
<td>
<div class="cell">本次开机时刻</div>
</td>
<td>
<div class="cell" v-if="dynamicTimeInfo">{{ dynamicTimeInfo.todayStartTime }}</div>
</td>
</tr>
<tr v-if="this.showParam.poweronTime">
<td>
<div class="cell">{{
bindingParamInfo.find(item => item.mongoName === 'poweronTime').paramName
}}
</div>
</td>
<td>
<div class="cell" v-if="dynamicTimeInfo">
{{ dynamicTimeInfo.timePayLoad.poweronTime }}
{{ bindingParamInfo.find(item => item.mongoName === 'poweronTime').paramUnit }}
</div>
</td>
</tr>
<tr v-if="this.showParam.setupTime">
<td>
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'setupTime').paramName }}</div>
</td>
<td>
<div class="cell" v-if="dynamicTimeInfo">
{{ dynamicTimeInfo.timePayLoad.setupTime }}
{{ bindingParamInfo.find(item => item.mongoName === 'setupTime').paramUnit }}
</div>
</td>
</tr>
<tr v-if="this.showParam.progStatus">
<td>
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'progStatus').paramName }}</div>
</td>
<td>
<div class="cell" v-if="dynamicCheckStatus && dynamicProgStatus.progStatus">
<el-button circle type="danger" size="small" style="vertical-align: middle"
v-if="[1,2,5].indexOf(dynamicProgStatus.progStatus) > -1"></el-button>
<el-button circle type="success" size="small" style="vertical-align: middle"
v-else-if="dynamicProgStatus.progStatus === 3"></el-button>
<el-button circle type="warning" size="small" style="vertical-align: middle"
v-else-if="dynamicProgStatus.progStatus === 4"></el-button>
{{ dynamicProgStatus.string }}
</div>
<div class="cell" v-else-if="!dynamicCheckStatus" style="color: red">
设备关机
</div>
</td>
</tr>
<tr v-if="this.showParam.progStatus">
<td>
<div class="cell">更新时刻</div>
</td>
<td>
<div class="cell" v-if="dynamicProgStatus.time">{{ dynamicProgStatus.time }}</div>
</td>
</tr>
</tbody>
<tbody v-else>
<tr>
<td>
<div class="cell">未找到设备运行状态,请<router-link :to="`../../deviceCenter/device`">
<el-link type="primary" :underline="false">检查数控系统id</el-link>
</router-link>
或联系技术人员
</div>
</td>
</tr>
</tbody>
</table>
</div>
</el-card>
</el-col>

这里首先拿到对象时要判断其是否为空对象,因为 空对象在 JavaScript 中被视为 true。这里我用的是自定义函数:

1
2
3
4
5
6
7
// 空对象
isEmptyObject(obj) {
for (let key in obj) {
return false;
}
return true;
},

(JavaScript 中,undefinednull-00+0)、NaN''(空字符串)都被视为 false,其他都被视为 true,如空数组 [] 和空对象 {} 也被视为 true

然后,我们会拿到一些机床数据,例如 dynamicTimeInfodynamicProgStatus 对象。为了实现变量绑定,我们还需要拿到:用户自定义的所有变量(从【变量信息】里拿),用户绑定到这台机床类型的变量(从绑定表里拿)。我用以下方式获取与处理变量:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
// 变量绑定
getParams() {
listParam().then(response => {
this.params = response.rows;
this.getMongos();
})
},
getMongos() {
listMongoVariable().then(response => {
this.mongos = response.rows;
this.getBindings();
})
},
getBindings() {
listDevicetypeparam().then(response => {
this.bindings = response.rows;
this.getBindingParamInfo();
})
},
getBindingParamInfo() {
this.bindingParamInfo = [];
for (let item of this.bindings) {
if (this.devicetypeId === item.devicetypeId) {
this.bindingParamInfo.push({
paramId: item.paramId
})
}
}
for (let bindingParam of this.bindingParamInfo) {
bindingParam.paramName = this.params.find(param => param.id === bindingParam.paramId).paramName;
bindingParam.paramUnit = this.params.find(param => param.id === bindingParam.paramId).paramUnit;
bindingParam.mongoId = this.params.find(param => param.id === bindingParam.paramId).mongoId;
bindingParam.mongoName = this.mongos.find(mongo => mongo.id === bindingParam.mongoId).mongoName;
}
this.showParam = {};
for (let mongo of this.mongos) {
this.showParam[mongo.mongoName] = false;
if (this.bindingParamInfo.find(item => item.mongoName === mongo.mongoName)) {
this.showParam[mongo.mongoName] = true;
}
}
this.loading.close();
},

getParams 函数拿所有的自定义变量,getMongos 函数拿所有的 MongoDB 变量,getBindings 函数拿所有的绑定关系,getBindingParamInfo 函数先把为当前类型绑定的变量放进数组里,再通过之前拿的信息为这些变量一一补充信息(如名称、单位等),最后在一个 showParam 数组里,为每个 MongoDB 变量设置布尔值,方便 Vue 里的 v-if 使用。

5.3.3 主轴信息图表:日期信息通信

下面两个图表结构基本一致,所要实现的功能和要处理的数据也基本一致。这两个图表我也是写在组件里的。

首先是这个日期搜索功能,其实也属于父组件和子组件的通信。以下是父组件的前端部分:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
<el-form :model="speedDate" label-width="45px" :inline="true">
<el-form-item label="日期">
<el-date-picker
v-model="speedDate.date"
type="date"
placeholder="请选择日期"
format="yyyy 年 MM 月 dd 日"
value-format="timestamp"
:clearable="false"
:picker-options="pickerOptions">
</el-date-picker>
</el-form-item>
<el-form-item>
<el-button type="primary" icon="el-icon-search" size="mini" @click="handleSpeedQuery">
搜索
</el-button>
</el-form-item>
<el-form-item v-if="$refs.speed && isEmptyObject($refs.speed.dynamicSpindleInfoList)">
<!-- 如果子组件存在,但子组件的主轴信息列表为空对象 -->
<div>系统当日无数据</div>
</el-form-item>
</el-form>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
// 日期
speedDate: {date: new Date(new Date().toLocaleDateString()).getTime()}, // 以获取当前的时间
...
pickerOptions: {
disabledDate(time) {
return time.getTime() > Date.now() - 8.64e6 // 不允许选择今日之后的时间
}
},
...
// 选择日期处理
handleSpeedQuery() {
this.speedLoading = true;
this.getDynamicSpindleInfo();
this.$refs.speed.getNcsId(); // 调子组件函数
},

日期选择

子组件要拿这个日期去查询信息:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
// 获取数控系统id
getNcsId() {
getDevice(this.deviceId).then(response => {
this.deviceNcsId = response.data.ncsId;
this.devicetypeId = response.data.devicetypeId;
this.getDynamicSpindleInfoList();
})
},
// 动态主轴信息列表获取
getDynamicSpindleInfoList() {
listSpindleDocByTime(this.deviceNcsId, this.$parent.$parent.$parent.$parent.speedDate.date).then(response => {
this.dynamicSpindleInfoList = [];
for (let doc of response) {
this.dynamicSpindleInfoList.push(doc);
}
...
});
},

这样子组件就拿到了某个日期的主轴信息列表,并拿此去画图。

5.3.4 主轴信息图表:父组件的信息展示与变量绑定

图表的下方有一个表格,这个表格位于父组件,展示读取到的最新的主轴动态数据。这个表格的处理很复杂,不仅要考虑到变量绑定,也要考虑到适配。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
<el-card shadow="hover" style="margin-bottom: 15px; height: 630px">
<div slot="header" class="card-header">
<span style="font-weight: bold">设备主轴速度图</span>
<el-button class="button" type="text"
@click="handleSpeedRefresh">
刷新
</el-button>
</div>
<div>
<el-form :model="speedDate" label-width="45px" :inline="true">
... <!-- 省略的是日期搜索框部分 -->
</el-form>
<div v-loading="speedLoading">
<spindle_speed ref="speed"></spindle_speed> <!-- 图表主体 -->
</div>
<div class="el-table el-table--enable-row-hover el-table--medium" style="margin-top: 15px">
<table cellspacing="0" style="width: 100%">
<tbody v-if="!isEmptyObject(dynamicSpindleInfo) && screenWidth >= 992">
<!-- 这里的tbody是屏幕宽度为350-991的情况,表格为1行6列 -->
<tr v-if="this.showParam.cmdSpeed || this.showParam.actSpeed">
<td v-if="this.showParam.cmdSpeed">
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'cmdSpeed').paramName }}</div>
</td>
<td v-if="this.showParam.cmdSpeed">
<div class="cell" v-if="this.dynamicSpindleInfo.spindlePayLoad.cmdSpeed !== null">
{{ dynamicSpindleInfo.spindlePayLoad.cmdSpeed }}
{{ bindingParamInfo.find(item => item.mongoName === 'cmdSpeed').paramUnit }}
</div>
</td>
<td v-if="this.showParam.actSpeed">
<div class="cell">{{ bindingParamInfo.find(item => item.mongoName === 'actSpeed').paramName }}</div>
</td>
<td v-if="this.showParam.actSpeed">
<div class="cell" v-if="this.dynamicSpindleInfo.spindlePayLoad.actSpeed !== null">
{{ dynamicSpindleInfo.spindlePayLoad.actSpeed }}
{{ bindingParamInfo.find(item => item.mongoName === 'actSpeed').paramUnit }}
</div>
</td>
<td>
<div class="cell">更新时刻</div>
</td>
<td>
<div class="cell" v-if="this.dynamicSpindleInfo.time">{{ dynamicSpindleInfo.time }}</div>
</td>
</tr>
<tr v-else>
<div class="cell">您可能未绑定速度相关变量,
<router-link :to="`../../deviceCenter/type`">
<el-link type="primary" :underline="false">点击绑定</el-link>
</router-link>
</div>
</tr>
</tbody>
<tbody v-else-if="screenWidth >= 350">
... <!-- 这里的tbody是屏幕宽度为350-991的情况,表格调整为3行2列,代码与上面基本一致 -->
</tbody>
<tbody v-else-if="screenWidth < 350">
... <!-- 这里的tbody是屏幕宽度小于350的情况,表格调整为6行1列,代码与上面基本一致 -->
</tbody>
<tbody v-if="isEmptyObject(dynamicSpindleInfo)">
<tr>
<div class="cell">未找到最新主轴速度信息。
</div>
</tr>
</tbody>
</table>
</div>
</div>
</el-card>

屏幕的宽度提前写在 data 里:

1
screenWidth: document.body.clientWidth

5.3.5 主轴信息图表:子组件的图表展示与变量绑定

变量绑定关系到图表的数据源,绑定就给图表数据,不绑定就不给,所以子组件最好也提前判断好是否绑定要画的变量。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
// 是否绑定?
findMongoId() {
listMongoVariable().then(response => {
this.mongoId.cmdSpeed = response.rows.find(row => row.mongoName === 'cmdSpeed').id;
this.mongoId.actSpeed = response.rows.find(row => row.mongoName === 'actSpeed').id;
this.findParamId();
})
},
findParamId() {
listParam().then(response => {
this.paramId.cmdSpeed = response.rows.find(row => row.mongoId === this.mongoId.cmdSpeed).id;
this.paramId.actSpeed = response.rows.find(row => row.mongoId === this.mongoId.actSpeed).id;
this.paramName.cmdSpeed = response.rows.find(row => row.mongoId === this.mongoId.cmdSpeed).paramName;
this.paramName.actSpeed = response.rows.find(row => row.mongoId === this.mongoId.actSpeed).paramName;
this.findWhetherBound();
})
},
findWhetherBound() {
listDevicetypeparam().then(response => {
this.isBound.cmdSpeed = !!response.rows.find(row => row.devicetypeId === this.devicetypeId && row.paramId === this.paramId.cmdSpeed);
this.isBound.actSpeed = !!response.rows.find(row => row.devicetypeId === this.devicetypeId && row.paramId === this.paramId.actSpeed);
...
})
},

这样通过 isBound 这个对象,就知道 cmdSpeedactSpeed 绑定与否。

再将数据处理为 Echarts 接受的形式(拿了所有数据):

1
2
3
4
5
6
7
8
9
10
11
getSpeedDatasetSource() {
this.speedDatasetSource = [];
for (let info of this.dynamicSpindleInfoList) {
this.speedDatasetSource.push({
time: info.time,
cmdSpeed: info.spindlePayLoad.cmdSpeed,
actSpeed: info.spindlePayLoad.actSpeed
});
}
...
},

最后初始化图表。客户要求默认只显示主轴指定转速(cmdSpeed),不显示主轴实际转速(actSpeed),这在 Echarts 中可以用 selected 属性设置。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
// 初始化图表
getSpeedGraph() {
let selected = {};
if (this.isBound.cmdSpeed) { // 如果主轴指定转速被绑定了,就只显示它
selected[this.paramName.cmdSpeed] = true;
selected[this.paramName.actSpeed] = false;
} else {
selected[this.paramName.actSpeed] = true;
}
this.speedGraph = echarts.init(this.$refs.speed_chart, "macarons");
this.speedGraph.setOption({
tooltip: {
trigger: 'axis',
axisPointer: { // 坐标轴指示器
type: 'line', // 形式为一条线
snap: true // 自动吸附(时间轴的点比较小)
}
},
legend: { // 图例
data: [this.paramName.cmdSpeed, this.paramName.actSpeed],
selected: selected // 默认选择项
},
grid: {
left: '3%',
right: '12%',
bottom: '10%',
containLabel: true,
},
dataset: {
dimensions: [{name: 'time', type: 'time'}, 'cmdSpeed', 'actSpeed'],
source: this.speedDatasetSource
},
xAxis: {
name: '时间',
nameTextStyle: {
padding: [0, 0, 0, 10]
},
type: 'time'
},
yAxis: {
name: '转速/(r/min)',
type: 'value',
nameTextStyle: {
padding: [0, 0, 0, 50]
}
},
dataZoom: [{ // 图表区域缩放
type: 'slider', // 滑动条缩放(图表下方)
start: 0,
end: 100,
xAxisIndex: 0, // 缩放x轴
filterMode: 'none' // 不过滤数据
},
{
type: 'inside', // 内置缩放(滚轮)
start: 0,
end: 100,
xAxisIndex: 0, // 缩放x轴
filterMode: 'none' // 不过滤数据
}],
series: [
this.isBound.cmdSpeed ? { // 绑定cmdSpeed的情况下
name: this.paramName.cmdSpeed,
type: 'line',
encode: {
x: 0,
y: 1
},
smooth: false
} : null,
this.isBound.actSpeed ? { // 绑定actSpeed的情况下
name: this.paramName.actSpeed,
type: 'line',
encode: {
x: 0,
y: 2
},
smooth: false
} : null
]
}, true);
this.$parent.$parent.$parent.$parent.speedLoading = false; // 停止父组件的加载动画
}

最终效果如下:

坐标轴指示器的自动吸附

两个图例都点上并进行缩放的情况

5.3.6 普通刷新

整个设备详细数据一共有 4 个普通刷新,分别是最上面的白色按钮「刷新状态」,以及 3 个动态数据卡片右上角的蓝色「刷新」文字。

刷新操作只要将按钮绑定到一个刷新函数就好了,函数内再调用重置相关信息或获得相关信息的函数,以实现数据刷新。这里我还写了几个 loading 变量,并将它们 v-loading 绑定到相关标签上,并在刷新时设置为 true,获取完毕数据后设置为 false,这样就可以实现加载功能。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
// 刷新处理
handleGlobalRefresh() {
this.openLoading();
this.speedLoading = true;
this.tableLoading = true;
this.driveLoading = true;
this.getBasicInfo();
this.$refs.speed.getNcsId();
this.$refs.drive.getNcsId();
},
handleDynamicRefresh() {
this.tableLoading = true;
this.getDynamicTimeInfo();
},
handleSpeedRefresh() {
this.speedLoading = true;
this.speedDate.date = new Date(new Date().toLocaleDateString()).getTime();
this.getDynamicSpindleInfo();
this.$refs.speed.getNcsId();
},
handleDriveRefresh() {
this.driveLoading = true;
this.driveDate.date = new Date(new Date().toLocaleDateString()).getTime();
this.getDynamicSpindleInfo();
this.$refs.drive.getNcsId();
},

全局刷新加载(左下角)和卡片刷新加载(右边中部)

5.3.7 自动刷新

这是个比较复杂的功能,需要综合运用 Vue 和 JavaScript 知识,写出来和 debug 都花了我一段时间。

实际上这个自动刷新用的还是上面的 handleGlobalRefresh 函数,只是要定时执行,并且还能调整开关与刷新频率。

先在页面的合适位置放上这两个按钮:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
<el-form
class="searchForm"
:model="refreshParams"
ref="refreshForm"
:inline="true"
label-width="75px"
style="display: inline; margin: 5px 0 5px 15px;"
>
<el-form-item label="自动刷新" prop="autoRefresh" style="margin-top:5px;">
<el-switch v-model="refreshParams.autoRefresh">
</el-switch>
</el-form-item>
<el-form-item label="刷新频率" prop="refreshFrequency" style="margin-top:5px;">
<el-select v-model="refreshParams.refreshFrequency" placeholder="请选择" @change="handleRefreshSetting" style="width: 85px;">
<el-option
v-for="item in refreshOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
</el-form>

自动刷新功能

定义变量,默认情况下开启刷新开关,并设置频率为 10 s:

1
2
3
4
5
6
7
8
9
10
11
12
13
// 刷新频率
refreshParams: {
autoRefresh: true,
refreshFrequency: 10000
},
refreshOptions: [
{value: 5000, label: '5 s'},
{value: 10000, label: '10 s'},
{value: 20000, label: '20 s'},
{value: 30000, label: '30 s'},
{value: 60000, label: '60 s'},
],
refreshTimer: null,

当刷新频率发生变化时,即上面的 el-select 选中别的选项时,触发 handleRefreshSetting 函数,处理这个变化:

1
2
3
4
5
6
7
8
9
10
11
handleRefreshSetting() {
if (this.refreshTimer) {
clearInterval(this.refreshTimer); // 清除定时器
this.refreshTimer = null;
}
this.refreshTimer = setInterval(() => { // 设置新的定时器
if (this.refreshParams.autoRefresh) { // 如果自动刷新开关打开,执行刷新
this.handleGlobalRefresh();
}
}, this.refreshParams.refreshFrequency); // 频率设置为新的频率
},

重点来了,由于这是个路由页面,所以可以利用 Vue 路由的导航守卫(导航表示路由发生改变)和 Vue 的生命周期 destroyed(现已改名为 unmounted),来处理可能发生的变化,并按不同的变化执行刷新操作。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
export default {
...
// 以下的to表示进入的路由对象,from表示离开的路由对象,next表示进入路由的钩子函数
beforeRouteEnter(to, from, next) { // 进入路由前
next(vm => { // 由于此时获取this,因此需要通过vm访问
vm.refreshTimer = setInterval(() => { // 新建一个定时器
if (vm.refreshParams.autoRefresh) {
vm.handleGlobalRefresh();
}
}, vm.refreshParams.refreshFrequency);
});
},
beforeRouteUpdate(to, from, next) { // 路由更新前(从一个机床的详细数据跳到另一个的时候)
if (this.refreshTimer) {
clearInterval(this.refreshTimer); // 清除定时器
this.refreshTimer = null;
}
next();
},
beforeRouteLeave(to, from, next) { // 离开路由前
if (this.refreshTimer) {
clearInterval(this.refreshTimer); // 清除定时器
this.refreshTimer = null;
}
next();
},
destroyed() { // Vue对象销毁时
if (this.refreshTimer) {
clearInterval(this.refreshTimer); // 清除定时器
}
this.refreshTimer = null; // 无论如何要设置定时器为null
},
...
}

凡是在路由内设置定时器,一定要通过导航守卫去处理它,否则会出现离开了详细数据页面,别的页面还在每隔 10 s 自动刷新的 bug。

6 首页的合作伙伴

首页的其他部分是苏州团队做的,下面的合作伙伴是我做的。合作伙伴只要放进公司的 logo 和名称就行了。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
<el-row :gutter="20" style="margin-bottom:0.5rem">
<el-col :xs="24" :sm="24">
<el-card class="update-log">
<div slot="header" class="clearfix">
<span>合作伙伴</span>
</div>
<div class="block">
<el-row :gutter="20">
<el-col :xl="2" :lg="3" :md="6" :sm="8" :xs="12"
v-for="(i, index) in 20" :key="i"
style="margin-bottom: 0.5rem; height: 70px">
<div style="text-align: center">
<el-image
style="width: auto; height: 30px; margin: 0 auto"
:src="partnerLogos[index]"
fit="scale-down"
></el-image>
<div style="padding: 10px; text-align: center">
<span> {{ partnerNames[index].abbr }}</span>
</div>
</div>
</el-col>
</el-row>
</div>
</el-card>
</el-col>
</el-row>

每一个企业都是一个 el-col,分为 logo 的 div 和名称的 div。每一个 el-col 使用了 v-for,所以要绑定 key

企业的名称写死在 data 中:

1
2
3
4
5
6
partnerNames: [
{full: "德国西门子", abbr: "德国西门子"},
{full: "江阴塞特精密工具有限公司", abbr: "江阴塞特"},
{full: "杭州新剑机器人技术股份有限公司", abbr: "杭州新剑"},
...
]

企业的 logo 由于位置一致,只是名称不同,因此写个函数去获取它:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
data() {
return {
...
partnerLogos: [],
...
};
},
created() {
this.getLogos();
},
methods: {
...
getLogos() {
for (let i = 1; i <= 20; i++) {
this.partnerLogos.push(require("../assets/partners/" + (i < 10 ? '0' + i : i) + '.png'));
}
}
}

从 5 月到 7 月,我就在做这些模块的前端开发。与我们之前做的模块不同,这些模块大部分是纯前端工作,比如统计图、设备看板等,除此之外,前端还要构思从后端拿到数据后怎么处理数据的逻辑。拿到变量绑定数据,怎么让它以用户自定义的情况显示?拿到机床主轴数据,怎么让它画成既直观又美观、还让客户满意的图表?都需要我去想想。

因此,越是傻瓜式的页面与操作方式,肯定是程序在背后为你做了越多的事情,这些程序也是程序员的思想结晶。

这个项目已经验收完毕了,通过开发这个项目,我也学会了很多前端的知识,和一些后端、数据库知识。这些知识希望在将来都能用得上。接下来的任务,就是聚焦一个研究方向,好好做科研了。

统计报表、工作台等剩余模块的 commit 过程