Commit 36f46c76fc7269d90a6dac549076caee33489983
1 parent
7b1f93b1
fix: report export trend charts not render multiple device charts
Showing
2 changed files
with
199 additions
and
69 deletions
@@ -7,3 +7,43 @@ export type ExportParam = { | @@ -7,3 +7,43 @@ export type ExportParam = { | ||
7 | orderFiled: string; | 7 | orderFiled: string; |
8 | orderType: string; | 8 | orderType: string; |
9 | }; | 9 | }; |
10 | +export interface ExecuteCondition2 { | ||
11 | + interval: number; | ||
12 | + limit: number; | ||
13 | + agg: string; | ||
14 | + orderBy: string; | ||
15 | + useStrictDataTypes: boolean; | ||
16 | + startTs: number; | ||
17 | + endTs: number; | ||
18 | + queryMode: number; | ||
19 | +} | ||
20 | + | ||
21 | +export interface ExecuteAttribute { | ||
22 | + device: string; | ||
23 | + name: string; | ||
24 | + attributes: string[]; | ||
25 | +} | ||
26 | + | ||
27 | +export interface ExecuteCondition { | ||
28 | + executeCondition: ExecuteCondition2; | ||
29 | + executeAttributes: ExecuteAttribute[]; | ||
30 | +} | ||
31 | + | ||
32 | +export interface ExecuteReportRecord { | ||
33 | + id: string; | ||
34 | + creator: string; | ||
35 | + createTime: string; | ||
36 | + updateTime: string; | ||
37 | + enabled: boolean; | ||
38 | + tenantId: string; | ||
39 | + reportConfigName: string; | ||
40 | + organizationName: string; | ||
41 | + organizationId: string; | ||
42 | + dataCompare: number; | ||
43 | + executeWay: number; | ||
44 | + executeStatus: number; | ||
45 | + executeTime: string; | ||
46 | + reportPath: string; | ||
47 | + executeCondition: ExecuteCondition; | ||
48 | + jobId: string; | ||
49 | +} |
@@ -7,38 +7,63 @@ | @@ -7,38 +7,63 @@ | ||
7 | :height="heightNum" | 7 | :height="heightNum" |
8 | @register="register" | 8 | @register="register" |
9 | title="报表趋势图" | 9 | title="报表趋势图" |
10 | + :minHeight="500" | ||
10 | :showOkBtn="false" | 11 | :showOkBtn="false" |
11 | > | 12 | > |
12 | - <div :class="[initChartData.length % 2 !== 0 ? '' : 'wrapper']"> | ||
13 | - <div | ||
14 | - v-if="initChartData.length > 0" | ||
15 | - :class="[initChartData.length % 2 !== 0 ? '' : 'chart-style']" | ||
16 | - > | 13 | + <div :class="[chartInstance.length % 2 !== 0 ? '' : 'wrapper']"> |
14 | + <Spin :spinning="loading"> | ||
17 | <div | 15 | <div |
18 | - :class="[ | ||
19 | - initChartData.length % 2 !== 0 ? '' : 'inner', | ||
20 | - initChartData.length % 2 !== 0 ? '' : 'item', | ||
21 | - ]" | ||
22 | - v-for="(item, index) in initChartData" | ||
23 | - :key="index" | 16 | + v-if="chartInstance.length > 0" |
17 | + :class="[chartInstance.length % 2 !== 0 ? '' : 'chart-style']" | ||
24 | > | 18 | > |
25 | - <p style="display: none">{{ item }}</p> | ||
26 | - <div :id="`chart${index}`" :style="{ height, width }"></div> | 19 | + <div |
20 | + :class="[ | ||
21 | + chartInstance.length % 2 !== 0 ? '' : 'inner', | ||
22 | + chartInstance.length % 2 !== 0 ? '' : 'item', | ||
23 | + ]" | ||
24 | + v-for="item in chartInstance" | ||
25 | + :key="item.device" | ||
26 | + > | ||
27 | + <p class="text-black text-lg">{{ item.name }}</p> | ||
28 | + <div class="flex text-black items-center"> | ||
29 | + <div class="mr-4 text-sm">属性:</div> | ||
30 | + <Select | ||
31 | + class="min-w-25" | ||
32 | + v-model:value="item.active" | ||
33 | + @change="(value) => handleChangeChars(value, item.device)" | ||
34 | + placeholder="请选择设备属性" | ||
35 | + > | ||
36 | + <Select.Option v-for="attr in item.attributes" :key="attr" :value="attr"> | ||
37 | + {{ attr }} | ||
38 | + </Select.Option> | ||
39 | + </Select> | ||
40 | + </div> | ||
41 | + <div> | ||
42 | + <div :id="`chart-${item.device}`" :style="{ height, width }"></div> | ||
43 | + </div> | ||
44 | + </div> | ||
27 | </div> | 45 | </div> |
28 | - </div> | ||
29 | - <div v-else style="display: flex; justify-content: center; align-items: center"> | ||
30 | - <div style="position: relative; left: 0rem; top: 3rem">暂无数据</div> | ||
31 | - </div> | 46 | + <div v-else style="display: flex; justify-content: center; align-items: center"> |
47 | + <div style="position: relative; left: 0rem; top: 3rem">暂无数据</div> | ||
48 | + </div> | ||
49 | + </Spin> | ||
32 | </div> | 50 | </div> |
33 | </BasicModal> | 51 | </BasicModal> |
34 | </div> | 52 | </div> |
35 | </template> | 53 | </template> |
36 | <script setup lang="ts"> | 54 | <script setup lang="ts"> |
37 | - import { ref, PropType, nextTick } from 'vue'; | 55 | + import { ref, PropType, nextTick, shallowReactive, onMounted, onUnmounted } from 'vue'; |
38 | import { BasicModal, useModalInner } from '/@/components/Modal'; | 56 | import { BasicModal, useModalInner } from '/@/components/Modal'; |
39 | import * as echarts from 'echarts'; | 57 | import * as echarts from 'echarts'; |
40 | import { exportViewChartApi } from '/@/api/export/exportManager'; | 58 | import { exportViewChartApi } from '/@/api/export/exportManager'; |
41 | import moment from 'moment'; | 59 | import moment from 'moment'; |
60 | + import { ExecuteReportRecord } from '/@/api/export/model/exportModel'; | ||
61 | + import { Select, Spin } from 'ant-design-vue'; | ||
62 | + | ||
63 | + interface ResponsData { | ||
64 | + attr: string; | ||
65 | + val: { ts: number; value: string }[]; | ||
66 | + } | ||
42 | 67 | ||
43 | defineProps({ | 68 | defineProps({ |
44 | width: { | 69 | width: { |
@@ -52,8 +77,14 @@ | @@ -52,8 +77,14 @@ | ||
52 | }); | 77 | }); |
53 | defineEmits(['register']); | 78 | defineEmits(['register']); |
54 | const heightNum = ref(800); | 79 | const heightNum = ref(800); |
55 | - const getItem: any = ref([]); | ||
56 | - const initChartData: any = ref([]); | 80 | + |
81 | + let currentRecord: ExecuteReportRecord = {}; | ||
82 | + | ||
83 | + const chartInstance = ref< | ||
84 | + { device: string; name: string; attributes: string[]; active?: string }[] | ||
85 | + >([]); | ||
86 | + | ||
87 | + const chartsInstance = shallowReactive<{ [key: string]: echarts.ECharts }>({}); | ||
57 | //生成随机颜色 | 88 | //生成随机颜色 |
58 | let getRandomColor = function () { | 89 | let getRandomColor = function () { |
59 | return ( | 90 | return ( |
@@ -66,55 +97,72 @@ | @@ -66,55 +97,72 @@ | ||
66 | ')' | 97 | ')' |
67 | ); | 98 | ); |
68 | }; | 99 | }; |
69 | - const [register, { setModalProps }] = useModalInner(async (data) => { | ||
70 | - setModalProps({ loading: true }); | ||
71 | - try { | ||
72 | - const entityId = data.record.executeCondition?.executeAttributes.map((m) => m?.device); | ||
73 | - let key = data.record.executeCondition?.executeAttributes.map((m) => m?.attributes); | ||
74 | - let params: any = {}; | ||
75 | - params = { | ||
76 | - ...data.record.executeCondition?.executeCondition, | ||
77 | - ...{ | ||
78 | - keys: key.length !== 0 ? key.join(',') : '', | ||
79 | - }, | ||
80 | - }; | ||
81 | - const result = await exportViewChartApi(entityId[0], params); | ||
82 | - getItem.value = Object.entries(result).map((item) => { | ||
83 | - const [attr, val] = item; | ||
84 | - return { attr, val }; | ||
85 | - }); | ||
86 | - //服务端返回值 | ||
87 | - initChartData.value = getItem.value.map((item): any => { | 100 | + |
101 | + const getChartsOption = (result: ResponsData) => { | ||
102 | + const generateInfo = Object.entries(result).map((item) => { | ||
103 | + const [attr, val] = item; | ||
104 | + return { attr, val } as { attr: string; val: { ts: number; value: string }[] }; | ||
105 | + }); | ||
106 | + const chartDataConfig = | ||
107 | + generateInfo.map((item) => { | ||
88 | const seriesData = item.val.map((m1) => Number(m1.value)); | 108 | const seriesData = item.val.map((m1) => Number(m1.value)); |
89 | return { | 109 | return { |
90 | - attr: item.attr, | ||
91 | - lengendData: [item.attr], | ||
92 | - xAxisData: item.val.map((m) => moment(m.ts).format('YYYY-MM-DD HH:mm:ss')), | ||
93 | - seriesData: [ | ||
94 | - { | ||
95 | - name: item.attr, | ||
96 | - type: 'line', | ||
97 | - data: seriesData, | ||
98 | - lineStyle: { | ||
99 | - color: getRandomColor(), | ||
100 | - }, | 110 | + xAxis: item.val.map((m) => moment(m.ts).format('YYYY-MM-DD HH:mm:ss')), |
111 | + series: { | ||
112 | + name: item.attr, | ||
113 | + type: 'line', | ||
114 | + data: seriesData, | ||
115 | + lineStyle: { | ||
116 | + color: getRandomColor(), | ||
101 | }, | 117 | }, |
102 | - ], | ||
103 | - }; | ||
104 | - }); | ||
105 | - nextTick(() => { | ||
106 | - initChartData.value.forEach((item, index) => { | ||
107 | - let myChart = echarts.init(document.getElementById(`chart${index}`) as HTMLElement); | ||
108 | - let myOption = { | 118 | + }, |
119 | + } as echarts.EChartsOption; | ||
120 | + }) || ([] as echarts.EChartsOption[]); | ||
121 | + const chartOption = { | ||
122 | + xAxisData: chartDataConfig.at(0)?.xAxis, | ||
123 | + series: [chartDataConfig.at(0)?.series], | ||
124 | + }; | ||
125 | + | ||
126 | + return chartOption; | ||
127 | + }; | ||
128 | + | ||
129 | + const [register, { setModalProps }] = useModalInner( | ||
130 | + async (data: { record: ExecuteReportRecord }) => { | ||
131 | + setModalProps({ loading: true }); | ||
132 | + try { | ||
133 | + currentRecord = data.record; | ||
134 | + const deviceInfo = data.record.executeCondition.executeAttributes || []; | ||
135 | + chartInstance.value = deviceInfo.map((item) => ({ | ||
136 | + ...item, | ||
137 | + active: item.attributes.at(0), | ||
138 | + })); | ||
139 | + for (const item of deviceInfo) { | ||
140 | + const { attributes, device } = item; | ||
141 | + const keys = attributes.length ? attributes.at(0) : ''; | ||
142 | + const sendParams = { | ||
143 | + ...data.record.executeCondition.executeCondition, | ||
144 | + ...{ | ||
145 | + keys, | ||
146 | + }, | ||
147 | + }; | ||
148 | + | ||
149 | + const result = await exportViewChartApi(device, sendParams); | ||
150 | + const { xAxisData, series } = getChartsOption(result as unknown as ResponsData); | ||
151 | + | ||
152 | + await nextTick(); | ||
153 | + chartsInstance[device] = echarts.init( | ||
154 | + document.getElementById(`chart-${device}`) as HTMLElement | ||
155 | + ); | ||
156 | + | ||
157 | + const chartOption = { | ||
109 | title: { | 158 | title: { |
110 | - text: `${item.attr}趋势图`, | ||
111 | left: 'center', | 159 | left: 'center', |
112 | }, | 160 | }, |
113 | tooltip: { | 161 | tooltip: { |
114 | trigger: 'axis', | 162 | trigger: 'axis', |
115 | }, | 163 | }, |
116 | legend: { | 164 | legend: { |
117 | - data: item.lengendData, | 165 | + data: attributes, |
118 | top: '20px', | 166 | top: '20px', |
119 | }, | 167 | }, |
120 | toolbox: {}, | 168 | toolbox: {}, |
@@ -136,11 +184,11 @@ | @@ -136,11 +184,11 @@ | ||
136 | ], | 184 | ], |
137 | xAxis: { | 185 | xAxis: { |
138 | type: 'category', | 186 | type: 'category', |
139 | - data: item.xAxisData, | 187 | + data: xAxisData, |
140 | boundaryGap: false, | 188 | boundaryGap: false, |
141 | axisPointer: { type: 'shadow' }, | 189 | axisPointer: { type: 'shadow' }, |
142 | axisLabel: { | 190 | axisLabel: { |
143 | - interval: 0, | 191 | + interval: 0, |
144 | rotate: 65, | 192 | rotate: 65, |
145 | textStyle: { | 193 | textStyle: { |
146 | color: '#000', | 194 | color: '#000', |
@@ -159,18 +207,60 @@ | @@ -159,18 +207,60 @@ | ||
159 | type: 'value', | 207 | type: 'value', |
160 | boundaryGap: false, | 208 | boundaryGap: false, |
161 | }, | 209 | }, |
162 | - series: item.seriesData, | 210 | + series, |
163 | }; | 211 | }; |
164 | - myChart.setOption(myOption); | 212 | + chartsInstance[device].setOption(chartOption); |
165 | //自适应 | 213 | //自适应 |
166 | - window.addEventListener('resize', () => { | ||
167 | - myChart.resize(); | ||
168 | - }); | ||
169 | - }); | ||
170 | - }); | 214 | + // window.addEventListener('resize', () => { |
215 | + // chartsInstance[device].resize(); | ||
216 | + // }); | ||
217 | + } | ||
218 | + } catch (error) { | ||
219 | + throw error; | ||
220 | + } finally { | ||
221 | + setModalProps({ loading: false }); | ||
222 | + } | ||
223 | + } | ||
224 | + ); | ||
225 | + const loading = ref(false); | ||
226 | + const renderCharts = async (device: string, keys: string) => { | ||
227 | + const sendParams = { | ||
228 | + ...currentRecord.executeCondition.executeCondition, | ||
229 | + ...{ | ||
230 | + keys, | ||
231 | + }, | ||
232 | + }; | ||
233 | + try { | ||
234 | + loading.value = true; | ||
235 | + const result = await exportViewChartApi(device, sendParams); | ||
236 | + const { xAxisData, series } = getChartsOption(result as unknown as ResponsData); | ||
237 | + | ||
238 | + chartsInstance[device].setOption({ | ||
239 | + series, | ||
240 | + xAxis: { data: xAxisData }, | ||
241 | + } as echarts.EChartsOption); | ||
242 | + } catch (error) { | ||
171 | } finally { | 243 | } finally { |
172 | - setModalProps({ loading: false }); | 244 | + loading.value = false; |
173 | } | 245 | } |
246 | + }; | ||
247 | + | ||
248 | + const handleChangeChars = (value: string, device: string) => { | ||
249 | + renderCharts(device, value); | ||
250 | + }; | ||
251 | + | ||
252 | + const resize = () => { | ||
253 | + Object.keys(chartsInstance).forEach((key) => { | ||
254 | + chartsInstance[key].resize(); | ||
255 | + }); | ||
256 | + }; | ||
257 | + | ||
258 | + onMounted(() => { | ||
259 | + window.addEventListener('resize', resize); | ||
260 | + }); | ||
261 | + | ||
262 | + onUnmounted(() => { | ||
263 | + window.removeEventListener('resize', resize); | ||
174 | }); | 264 | }); |
175 | </script> | 265 | </script> |
176 | <style lang="less" scoped> | 266 | <style lang="less" scoped> |