DownAppQualityReport.vue 48 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475
  1. <template>
  2. <div class="quality-report">
  3. <nav style="" class="nav-content">
  4. <a class="nav-list" v-for="(item, index) in navList" :key="index" @click="jump(index)" :class="index==currentIndex?'current-item':''">{{item}}</a>
  5. </nav>
  6. <div class="report-container" id="report" ref="downReport">
  7. <div class="report-header">
  8. <div class="header-title">数据质量报告</div>
  9. <div class="header-subtitle">
  10. <span>数据范围:{{queryParams.appName}}</span>
  11. <span>报告类型:{{queryParams.reportClassName}}</span>
  12. <span v-if="queryParams.reportClassName == '日报' || queryParams.reportClassName == '年报'">时间范围:{{queryParams.beginDate}}</span>
  13. <span v-else>时间范围:{{queryParams.beginDate}}~{{queryParams.endDate}}</span>
  14. </div>
  15. </div>
  16. <div class="report-content">
  17. <div class="report-section">
  18. <div class="title-content">
  19. <div class="title-line">
  20. <span></span>综合数据质量分析
  21. </div>
  22. </div>
  23. <div class="report-echarts">
  24. <Row :gutter="16" class-name="report-echarts-row">
  25. <Col :span="12">
  26. <div class="report-echarts-main">
  27. <div class="echarts-title">1. 综合得分</div>
  28. <div id="appGuage" class="echart-detail"></div>
  29. </div>
  30. </Col>
  31. <Col :span="12">
  32. <div class="report-echarts-main">
  33. <div class="echarts-title">2. 评估维度</div>
  34. <div id="appRadar" class="echart-detail"></div>
  35. </div>
  36. </Col>
  37. </Row>
  38. <Row :gutter="16" class-name="report-echarts-row">
  39. <Col :span="24">
  40. <div class="report-echarts-main">
  41. <div class="echarts-title">3. 综合数据质量趋势</div>
  42. <div id="appLine3" class="echart-detail"></div>
  43. </div>
  44. </Col>
  45. </Row>
  46. <Row :gutter="16" class-name="report-echarts-row">
  47. <Col :span="24">
  48. <div class="report-echarts-main no-truncation">
  49. <div class="echarts-title">4. 综合数据质量分析-按规则类型</div>
  50. <div id="appLBar4" class="echart-detail"></div>
  51. <div class="report-echarts-table">
  52. <Table :columns="columns4" :data="tableData4" class="common-table report-table-detail" no-data-text="" :row-class-name="rowClassName">
  53. <template slot-scope="{row,index}" slot="paramDou01">
  54. <span :style="index==0 ? 'color:#5EB6FC': ''">{{row.paramDou01+'%'}}</span>
  55. </template>
  56. <template slot-scope="{row,index}" slot="paramDou02">
  57. <span :style="index==0 ? 'color:#5EB6FC': ''">{{row.paramDou02+'%'}}</span>
  58. </template>
  59. <template slot-scope="{row}" slot="paramDou04">
  60. <Icon type="md-arrow-up" style="color: #09CB09;font-size:20px" v-show="row.paramDou04>0"/>
  61. <Icon type="md-arrow-down" style="color: #F32F2F;font-size:20px" v-show="row.paramDou04<0" />
  62. {{Math.abs(row.paramDou04)+'%'}}
  63. </template>
  64. </Table>
  65. </div>
  66. </div>
  67. </Col>
  68. </Row>
  69. </div>
  70. </div>
  71. <div class="report-section">
  72. <div class="title-content no-truncation">
  73. <div class="title-line">
  74. <span></span>数据异常分析
  75. </div>
  76. </div>
  77. <div class="report-echarts">
  78. <Row :gutter="16" class-name="report-echarts-row">
  79. <Col :span="24">
  80. <div class="report-echarts-main no-truncation">
  81. <div class="echarts-title">5. 数据异常分析 - 按规则类型</div>
  82. <div class="echarts-sub-title">数据质量检测异常,按来源规则类型区分数据如下:</div>
  83. <Row :gutter="16" align="middle">
  84. <Col :span="8">
  85. <div id="appPie5" class="echart-detail"></div>
  86. </Col>
  87. <Col :span="16">
  88. <div class="report-echarts-table report-echarts-right-table">
  89. <Table :columns="columnsRule" :data="tableData5" class="common-table report-table-detail" no-data-text="" :row-class-name="rowClassName" ref="appTable5">
  90. <template slot-scope="{ index }" slot="xAxis">
  91. <span class="table-legend" :style="'background:'+tableColumnColor5[index]" v-show="index>0"></span>
  92. </template>
  93. <template slot-scope="{row,index}" slot="paramDou01">
  94. <span :style="index==0 ? 'color:#ECCF0C': ''">{{row.paramDou01}}%</span>
  95. </template>
  96. </Table>
  97. </div>
  98. </Col>
  99. </Row>
  100. </div>
  101. </Col>
  102. </Row>
  103. <Row :gutter="16" class-name="report-echarts-row">
  104. <Col :span="24">
  105. <div class="report-echarts-main no-truncation">
  106. <div class="echarts-title">6. 异常数来源 - 质量规则 TOP10</div>
  107. <div class="echarts-sub-title">数据质量检查错数按规则排名如下:</div>
  108. <Row :gutter="16">
  109. <Col :span="24">
  110. <div id="appBar6" class="echart-detail"></div>
  111. <div class="report-echarts-table">
  112. <Table :columns="columnsException" :data="tableData6" class="common-table report-table-detail" no-data-text="" :row-class-name="rowClassName">
  113. <template slot-scope="{row,index}" slot="paramDou01">
  114. <span :style="index==0 ? 'color:#5EB6FC': ''">{{row.paramDou01+'%'}}</span>
  115. </template>
  116. <template slot-scope="{row,index}" slot="paramDou02">
  117. <span :style="index==0 ? 'color:#5EB6FC': ''">{{row.paramDou02+'%'}}</span>
  118. </template>
  119. </Table>
  120. </div>
  121. </Col>
  122. </Row>
  123. </div>
  124. </Col>
  125. </Row>
  126. </div>
  127. <div class="report-section">
  128. <div class="title-content no-truncation">
  129. <div class="title-line">
  130. <div class="num-title">
  131. <span></span>规则配置状况
  132. </div>
  133. </div>
  134. </div>
  135. <div class="report-echarts">
  136. <Row :gutter="16" class-name="report-echarts-row">
  137. <Col :span="24">
  138. <div class="report-echarts-main no-truncation">
  139. <div class="echarts-title">7.规则配置状况</div>
  140. <div class="echarts-sub-title">规则配置数量级及占比</div>
  141. <Row :gutter="16" align="middle">
  142. <Col :span="8">
  143. <div id="appPie7" class="echart-detail"></div>
  144. </Col>
  145. <Col :span="16">
  146. <div class="report-echarts-table report-echarts-right-table">
  147. <Table :columns="columnsConfig" :data="tableData7" class="common-table report-table-detail" no-data-text="" :row-class-name="rowClassName">
  148. <template slot-scope="{ index }" slot="xAxis">
  149. <span class="table-legend" :style="'background:'+tableColumnColor7[index]" v-show="index>0"></span>
  150. </template>
  151. <template slot-scope="{row,index}" slot="paramDou05">
  152. <span :style="index==0 ? 'color:#5EB6FC': ''">{{row.paramDou05+'%'}}</span>
  153. </template>
  154. <template slot-scope="{row}" slot="paramDou04">
  155. <Icon type="md-arrow-up" style="color: #09CB09;font-size:20px" v-show="row.paramDou04>0"/>
  156. <Icon type="md-arrow-down" style="color: #F32F2F;font-size:20px" v-show="row.paramDou04<0" />
  157. {{Math.abs(row.paramDou04)+'%'}}
  158. </template>
  159. </Table>
  160. </div>
  161. </Col>
  162. </Row>
  163. </div>
  164. </Col>
  165. </Row>
  166. </div>
  167. </div>
  168. </div>
  169. </div>
  170. </div>
  171. </div>
  172. </template>
  173. <script>
  174. import * as echarts from "echarts"
  175. import html2canvas from 'html2canvas';
  176. import JsPDF from 'jspdf'
  177. export default {
  178. name: "DownAppQualityReport",
  179. data() {
  180. return {
  181. scroll: '',
  182. navList: ['综合数据质量分析', '数据异常分析','规则配置状况'],
  183. currentIndex:0,
  184. listBoxState: true,//点击导航栏时,暂时停止监听页面滚动
  185. queryParams: {reportClassName:'',beginDate:'',endDate: '',appId:0,appName: ''},
  186. columns4: [
  187. {
  188. title: '规则类型',
  189. key: 'xAxis',
  190. align: 'center',
  191. ellipsis: true,
  192. tooltip: true
  193. },
  194. {
  195. title: '记录总数',
  196. key: 'paramLong01',
  197. align: 'center',
  198. ellipsis: true,
  199. tooltip: true
  200. },
  201. {
  202. title: '本期质量',
  203. slot: 'paramDou01',
  204. align: 'center',
  205. },
  206. {
  207. title: '上期质量',
  208. slot: 'paramDou02',
  209. align: 'center',
  210. },
  211. {
  212. title: '趋势',
  213. slot: 'paramDou04',
  214. align: 'center',
  215. },
  216. ],
  217. tableData4: [],
  218. tableData5: [],
  219. tableData6: [],
  220. tableData7: [],
  221. pieColumnColor5: ['#646AD9','#FDDD60','#FF6E76','#7CFFB2','#4992FF','#58D9F9'],
  222. tableColumnColor5: ['','#646AD9','#FDDD60','#FF6E76','#7CFFB2','#4992FF'],
  223. pieColumnColor7: ['#FF6E76','#646AD9','#FDDD60','#7CFFB2','#4992FF','#58D9F9'],
  224. tableColumnColor7: ['','#FF6E76','#646AD9','#FDDD60','#7CFFB2','#4992FF'],
  225. columnsRule: [
  226. {
  227. title: '图例颜色',
  228. slot: 'xAxis',
  229. align: 'right',
  230. width: 70,
  231. className: 'table-legend-list',
  232. },
  233. {
  234. title: '规则类型',
  235. key: 'xAxis',
  236. align: 'center',
  237. ellipsis: true,
  238. tooltip: true
  239. },
  240. {
  241. title: '记录总数',
  242. key: 'paramNum01',
  243. align: 'center',
  244. ellipsis: true,
  245. tooltip: true
  246. },
  247. {
  248. title: '异常总数',
  249. key: 'paramNum02',
  250. align: 'center',
  251. ellipsis: true,
  252. tooltip: true
  253. },
  254. {
  255. title: '异常占比',
  256. slot: 'paramDou01',
  257. align: 'center',
  258. }],
  259. columnsException: [
  260. {
  261. title: '来源规则名称',
  262. key: 'xAxis',
  263. align: 'center',
  264. ellipsis: true,
  265. tooltip: true
  266. },
  267. {
  268. title: '记录总数',
  269. key: 'paramLong01',
  270. align: 'center',
  271. ellipsis: true,
  272. tooltip: true
  273. },
  274. {
  275. title: '异常总数',
  276. key: 'paramNum01',
  277. align: 'center',
  278. ellipsis: true,
  279. tooltip: true
  280. },
  281. {
  282. title: '本期质量',
  283. slot: 'paramDou01',
  284. align: 'center',
  285. },
  286. {
  287. title: '异常数\n总体占比', // iview的table表头文字换行需要\n和.ivu-table-cell {white-space: pre;}
  288. slot: 'paramDou02',
  289. align: 'center',
  290. },
  291. ],
  292. columnsConfig: [
  293. {
  294. title: '图例颜色',
  295. slot: 'xAxis',
  296. align: 'right',
  297. width: 70,
  298. className: 'table-legend-list',
  299. },
  300. {
  301. title: '规则类型',
  302. key: 'xAxis',
  303. align: 'center',
  304. ellipsis: true,
  305. tooltip: true
  306. },
  307. {
  308. title: '数量',
  309. key: 'paramNum01',
  310. align: 'center',
  311. ellipsis: true,
  312. tooltip: true
  313. },
  314. {
  315. title: '占比',
  316. slot: 'paramDou05',
  317. align: 'center',
  318. },
  319. {
  320. title: '趋势',
  321. slot: 'paramDou04',
  322. align: 'center',
  323. },
  324. ],
  325. downloadStatus: false,
  326. };
  327. },
  328. watch: {
  329. scroll: function () {
  330. this.loadSroll()
  331. }
  332. },
  333. created () {
  334. if (this.$route.query.reportClassName) {
  335. this.queryParams = this.$route.query
  336. let parmasObj = {
  337. beginDate: this.queryParams.beginDate,
  338. endDate: this.queryParams.endDate,
  339. appId: this.queryParams.appId
  340. }
  341. this.downloadStatus = false
  342. this.getData(parmasObj)
  343. }
  344. },
  345. mounted() {
  346. //vue通过window.addEventListener('scroll', XXXX)无法监听屏幕滚动事件,上一个页面可以,换了一个页面,却不行了,后来百度,往后面加了一个true
  347. window.addEventListener('scroll', this.dataScroll,true)
  348. },
  349. methods: {
  350. // js 节流浏览器性能优化
  351. dataScroll: function () {
  352. this.scroll = document.documentElement.scrollTop || document.body.scrollTop;
  353. },
  354. jump(index) {
  355. let jump = document.getElementsByClassName('report-section');
  356. // 获取需要滚动的距离
  357. let total = jump[index].offsetTop;
  358. // Chrome
  359. document.body.scrollTop = total;
  360. // Firefox
  361. document.documentElement.scrollTop = total;
  362. // Safari
  363. window.pageYOffset = total;
  364. this.currentIndex = index
  365. this.listBoxState = false
  366. let timeId;
  367. clearTimeout(timeId);
  368. timeId = setTimeout(() => {
  369. this.listBoxState = true
  370. },100)
  371. // $('html, body').animate({
  372. // 'scrollheader': total
  373. // }, 400);
  374. },
  375. loadSroll: function () {
  376. var self = this;
  377. var sections = document.getElementsByClassName('report-section');
  378. if (this.listBoxState) {
  379. for (var i = sections.length - 1; i >= 0; i--) {
  380. if (self.scroll >= sections[i].offsetTop-100) {
  381. this.currentIndex = i
  382. break;
  383. }
  384. }
  385. }
  386. },
  387. rowClassName(row, index) {
  388. if (index % 2 == 0) {
  389. return "ivu-table-stripe-even";
  390. } else {
  391. return "ivu-table-stripe-odd";
  392. }
  393. },
  394. creatGauge(gaugeData) {
  395. let myChart = echarts.init(document.getElementById(gaugeData.id))
  396. let option = {
  397. series: [
  398. {
  399. type: 'gauge',
  400. center: ['50%', '68%'],
  401. radius:'90%',
  402. startAngle: 200,
  403. endAngle: -20,
  404. min: 0,
  405. max: 100,
  406. splitNumber: 10,
  407. itemStyle: {
  408. color: '#FFAB91'
  409. },
  410. progress: {
  411. show: true,
  412. width: 15
  413. },
  414. pointer: {
  415. show: false
  416. },
  417. axisLine: {
  418. lineStyle: {
  419. color:[[1, '#284677']],
  420. width: 15
  421. }
  422. },
  423. axisTick: {
  424. distance: -35,
  425. splitNumber: 5,
  426. lineStyle: {
  427. width: 2,
  428. color: '#4D5F7D'
  429. }
  430. },
  431. splitLine: {
  432. distance: -52,
  433. length: 14,
  434. lineStyle: {
  435. width: 3,
  436. color: '#4D5F7D'
  437. }
  438. },
  439. axisLabel: {
  440. distance: -20,
  441. color: '#5D6B82',
  442. fontSize: 12
  443. },
  444. anchor: {
  445. show: false
  446. },
  447. title: {
  448. show: false
  449. },
  450. detail: {
  451. valueAnimation: true,
  452. width: '60%',
  453. lineHeight: 40,
  454. borderRadius: 8,
  455. offsetCenter: [0, '-15%'],
  456. fontSize: 30,
  457. fontWeight: 'bolder',
  458. formatter: '{value} %',
  459. color: 'auto'
  460. },
  461. data: [
  462. {
  463. value: gaugeData.scores
  464. }
  465. ]
  466. },
  467. {
  468. type: 'gauge',
  469. center: ['50%', '68%'],
  470. radius:'90%',
  471. startAngle: 200,
  472. endAngle: -20,
  473. min: 0,
  474. max: 100,
  475. itemStyle: {
  476. color: '#FD6636'
  477. },
  478. progress: {
  479. show: true,
  480. width: 5
  481. },
  482. pointer: {
  483. show: false
  484. },
  485. axisLine: {
  486. show: false
  487. },
  488. axisTick: {
  489. show: false
  490. },
  491. splitLine: {
  492. show: false
  493. },
  494. axisLabel: {
  495. show: false
  496. },
  497. detail: {
  498. show: false
  499. },
  500. data: [
  501. {
  502. value: gaugeData.scores
  503. }
  504. ]
  505. }
  506. ]
  507. }
  508. myChart.resize()
  509. myChart.clear()
  510. myChart.setOption(option)
  511. window.addEventListener("resize", function() {
  512. myChart.resize()
  513. })
  514. },
  515. //雷达图显示文字
  516. contains(arr, val) {
  517. var i = arr.length;
  518. while (i--) {
  519. if (arr[i].name === val) {
  520. return i;
  521. }
  522. }
  523. return false;
  524. },
  525. creatSolidRadar (SolidRadarData) {
  526. if (echarts.init(document.getElementById(SolidRadarData.id))) {
  527. let myChart = echarts.init(document.getElementById(SolidRadarData.id));
  528. myChart.clear()
  529. const createSvg = ({width, height, shadowColor, shadowBlur, points}) => {
  530. const ret = [`M${points[0][0]} ${points[0][1]}`];
  531. for (let i = 1; i < points.length; i++) {
  532. ret.push(`L${points[i][0]} ${points[i][1]}`);
  533. }
  534. ret.push('Z');
  535. const rectPath = ret.join(' ');
  536. return (`
  537. <svg version="1.1" xmlns="http://www.w3.org/2000/svg"
  538. x="0px" y="0px"
  539. width="${width}"
  540. height="${height}"
  541. >
  542. <style>
  543. .st1 {
  544. fill: transparent;
  545. stroke: ${shadowColor};
  546. stroke-width: ${shadowBlur}px;
  547. filter: url(#chart-inset-shadow);
  548. clip-path: url(#chart-clip);
  549. }
  550. </style>
  551. <defs>
  552. <clipPath id="chart-clip">
  553. <path d="${rectPath}" />
  554. </clipPath>
  555. <filter id="chart-inset-shadow" width="200%" height="200%" x="-50%" y="-50%">
  556. <feGaussianBlur in="SourceGraphic" result="gass" stdDeviation="${shadowBlur * 0.75}" />
  557. <feMerge>
  558. <feMergeNode in="gass" />
  559. </feMerge>
  560. </filter>
  561. </defs>
  562. <g>
  563. <path class="st1" d="${rectPath}" />
  564. </g>
  565. </svg>
  566. `);
  567. };
  568. const cretateSvgUrl = (svgOption) => {
  569. const svgString = createSvg(svgOption);
  570. const svg = new Blob([svgString], {type: "image/svg+xml;charset=utf-8"});
  571. const DOMURL = window.URL || window.webkitURL || window;
  572. const insetShadowUrl = DOMURL.createObjectURL(svg);
  573. return insetShadowUrl;
  574. };
  575. const dataSet = [], nameArr = []
  576. SolidRadarData.category[0].forEach((item,index)=> {
  577. nameArr.push(item.name)
  578. })
  579. dataSet[0] = nameArr
  580. dataSet[1] = [100,100,100,100,100]
  581. dataSet[2] = SolidRadarData.data[0].value
  582. const maxValue = [...dataSet[1], ...dataSet[2]].reduce((m, v) => Math.max(m, v), -Infinity);
  583. const radius = 0.8;
  584. const theta = Math.PI * 2 / dataSet[2].length
  585. const getPoints = (R, ps, max) => ps.map((v, i) => {
  586. const t = i * theta;
  587. const d = v / max;
  588. const x = R - Math.sin(t) * R * d;
  589. const y = R - Math.cos(t) * R * d;
  590. return [x, y];
  591. });
  592. // let i = -1;
  593. let option = {
  594. polar: {
  595. radius: radius * 100 + '%',
  596. // center: ['50%', '58%'],
  597. center: ['50%', '50%'],
  598. },
  599. angleAxis: {
  600. type: 'category',
  601. clockwise: false,
  602. boundaryGap: false,
  603. splitLine: {
  604. show: false,
  605. },
  606. axisLine: {
  607. show: false
  608. }
  609. },
  610. radiusAxis: {
  611. type: 'value',
  612. min: 0,
  613. max: 100,
  614. splitLine: {
  615. show: false,
  616. },
  617. axisTick: {
  618. show: false,
  619. },
  620. axisLabel: {
  621. show: false,
  622. color:'#fff',
  623. formatter: '{value} %'
  624. },
  625. },
  626. radar: {
  627. indicator: SolidRadarData.category[0],
  628. radius: radius * 100 + '%',
  629. center: ['50%', '50%'],
  630. shape: 'polygon',
  631. splitNumber: 5,
  632. // nameGap : 1,
  633. name: {
  634. // rich: {
  635. // a: {
  636. // color: '#fff',
  637. // lineHeight: 20,
  638. // },
  639. // b: {
  640. // color: '#fff',
  641. // align: 'center',
  642. // padding: 2,
  643. // borderRadius: 4
  644. // }
  645. // },
  646. // formatter: (value)=>{
  647. // let i = this.contains(SolidRadarData.category[0], value); // 处理对应要显示的样式
  648. // return `{a|${value}}{b|${dataSet[2][i]}%}`
  649. // },
  650. textStyle: {
  651. color: '#fff',
  652. fontSize: 12,
  653. padding: [-10, -10]
  654. }
  655. },
  656. splitArea: {
  657. areaStyle: {
  658. color: [
  659. 'rgba(8,20,58,0.1)',
  660. 'rgba(,20,58,0.03)',
  661. 'rgba(,20,58,0.1)',
  662. 'rgba(,20,58,0.03)',
  663. ]
  664. }
  665. },
  666. splitLine: {
  667. lineStyle: {
  668. color: [
  669. '#2A3A6E', '#2A3A6E',
  670. '#2A3A6E', '#2A3A6E',
  671. '#2A3A6E'
  672. ].reverse(),
  673. width: 3
  674. }
  675. },
  676. axisLine: {
  677. lineStyle: {
  678. color: '#264474'
  679. }
  680. }
  681. },
  682. animationDuration: 3000,
  683. series: [
  684. {
  685. type: 'custom',
  686. name: 's1-inset-shadow',
  687. silent: true,
  688. coordinateSystem : 'polar',
  689. // data: [0],
  690. renderItem: (params,api) => {
  691. const R = params.coordSys.r;
  692. const cx = params.coordSys.cx;
  693. const cy = params.coordSys.cy;
  694. const x = cx - R;
  695. const y = cy - R;
  696. const width = 2 * R;
  697. const height = 2 * R;
  698. return {
  699. type: 'image',
  700. style: {
  701. image: cretateSvgUrl({
  702. width, height,
  703. shadowColor: '#00A0E9',
  704. shadowBlur: 6,
  705. points: getPoints(R, dataSet[1], 100)
  706. }),
  707. x,
  708. y,
  709. width,
  710. height,
  711. },
  712. };
  713. },
  714. },
  715. {
  716. name: 's2',
  717. type: 'radar',
  718. data: [
  719. { value: dataSet[2] }
  720. ],
  721. // symbol: 'none',
  722. label: {
  723. show: true,
  724. formatter: function(params) {
  725. return params.value + '%';
  726. },
  727. color: '#fff',
  728. position: [-10, -10],
  729. // align: 'left',
  730. // distance: -25,
  731. },
  732. symbolSize: [6, 6],
  733. itemStyle: {
  734. shadowColor: 'rgba(0, 0, 0, 0.5)',
  735. shadowBlur: 20,
  736. normal: {
  737. color: 'rgba(87,201,255,0.8)',
  738. borderColor: 'rgba(87,201,255,0.2)',
  739. borderWidth: 12,
  740. }
  741. },
  742. areaStyle: {
  743. normal: {
  744. color: '#383F2F',
  745. }
  746. },
  747. lineStyle: {
  748. normal: {
  749. color: '#FAEC05',
  750. width: 2,
  751. }
  752. },
  753. z: 3,
  754. },
  755. ]
  756. };
  757. myChart.resize()
  758. myChart.setOption(option)
  759. window.addEventListener("resize", function () {
  760. myChart.resize()
  761. })
  762. }
  763. },
  764. creatLine (lineData) {
  765. let toplinechart = echarts.init(document.getElementById(lineData.id));
  766. let seriesData = _.map(lineData.data, (item,index)=>({
  767. name: lineData.legend[index], type: 'line', symbol: 'circle', symbolSize: 2,smooth: true, // 平滑曲线
  768. lineStyle: { normal: { color: lineData.lineColor[index], width: 2 } },
  769. itemStyle: { normal: { color: lineData.lineColor[index], borderColor: lineData.lineColor[index], borderWidth: 2 },
  770. emphasis: { color: '#fff' }
  771. },
  772. areaStyle: { opacity:0.4,color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [{ offset: 0.3, color: lineData.lineColor[index]}, { offset: 1, color: 'rgba(0,0,0,0)' }])},
  773. data: item
  774. }))
  775. let toplineoption = {
  776. legend: {
  777. top: '5%', left: 'center', itemWidth: 10, itemHeight: 10, itemGap: 25,
  778. textStyle: { color: '#fff', fontSize: 12 },
  779. data: lineData.legend
  780. },
  781. tooltip: { trigger: "axis",
  782. confine: true,
  783. backgroundColor: '#011235',
  784. borderWidth: 0,
  785. extraCssText: 'opacity:0.8',
  786. textStyle: {
  787. color: '#fff'
  788. },
  789. formatter: function (params) {
  790. let res = ''
  791. params.forEach((item,i) => {
  792. res += '<div style="display: flex;align-items:center;justify-content:space-between"><span>'+ item.marker + item.seriesName + ':</span><span>' + item.value + lineData.unit+'</span></div> '
  793. })
  794. return params[0].name + res
  795. }
  796. },
  797. grid: {
  798. top: '18%',
  799. left: "4%",
  800. right: "4%",
  801. bottom: "4%",
  802. containLabel: true
  803. },
  804. xAxis: [{
  805. type: 'category',
  806. data: lineData.category,
  807. boundaryGap: false,
  808. axisLine: {
  809. lineStyle: {
  810. color: '#00479D',
  811. width: 1
  812. }
  813. },
  814. axisLabel: {
  815. color: '#fff',
  816. },
  817. axisTick: {
  818. alignWithLabel: true
  819. }
  820. }],
  821. yAxis: [{
  822. type: 'value',
  823. axisLine: {
  824. show: true,
  825. lineStyle: {
  826. color: '#00479D',
  827. width: 1
  828. }
  829. },
  830. axisTick: {
  831. show: true
  832. },
  833. splitLine: {
  834. show: true,
  835. lineStyle: {
  836. type: "solid",
  837. width: 1,
  838. color: "#153068"
  839. }
  840. },
  841. axisLabel: {
  842. show: true,
  843. color: '#fff'
  844. }
  845. }],
  846. series: seriesData
  847. }
  848. toplinechart.resize()
  849. toplinechart.clear()
  850. toplinechart.setOption(toplineoption)
  851. console.log('line')
  852. window.addEventListener("resize", function () { toplinechart.resize() })
  853. },
  854. creatBar(botmLeftBarData) {
  855. let botmLeftChart = echarts.init(document.getElementById(botmLeftBarData.id))
  856. let barseries = []
  857. for (var i = 0; i < botmLeftBarData.legend.length; i++) {
  858. let itemStyle = {}
  859. if (botmLeftBarData.showStack) {
  860. itemStyle = {
  861. normal: {
  862. color:botmLeftBarData.color[i]
  863. }
  864. }
  865. } else {
  866. itemStyle = {
  867. normal: {
  868. color: {
  869. type: 'linear',
  870. x: 0,
  871. y: 0,
  872. x2: 0,
  873. y2: 1,
  874. colorStops: [{
  875. offset: 0,
  876. color: botmLeftBarData.color[i][0]
  877. }, {
  878. offset: 1,
  879. color: botmLeftBarData.color[i][1]
  880. }],
  881. globalCoord: false // 缺省为 false
  882. },
  883. }
  884. }
  885. }
  886. barseries.push({
  887. name: botmLeftBarData.legend[i],
  888. type: 'bar',
  889. stack: botmLeftBarData.showStack ? 'total' : false,
  890. barMaxWidth: 28,
  891. itemStyle: itemStyle,
  892. data: botmLeftBarData.data[i]
  893. })
  894. }
  895. let botmLeftOption = {
  896. tooltip: {
  897. trigger: botmLeftBarData.legend.length==1 ? 'item' : 'axis',
  898. axisPointer: { type: 'shadow' },
  899. confine: true,
  900. backgroundColor: '#011235',
  901. borderWidth: 0,
  902. extraCssText: 'opacity:0.8',
  903. textStyle: {
  904. color: '#fff'
  905. },
  906. formatter: function(params) {
  907. if (Array.isArray(params)) {
  908. let res = ''
  909. params.forEach((item,i) => {
  910. res += '<div style="display: flex;align-items:center;justify-content:space-between"><span>'+ item.marker + item.seriesName + ':</span><span>' + item.value + botmLeftBarData.unit+'</span></div> '
  911. })
  912. return params[0].name + res
  913. } else {
  914. return params.marker + params.name+':'+ params.value + botmLeftBarData.unit
  915. }
  916. }
  917. },
  918. legend: {
  919. show: botmLeftBarData.legend.length>1 ? true : false,
  920. orient: 'horizontal', left: 'center', top: 2, itemWidth: 10, itemHeight: 10,
  921. textStyle: { color: '#ffffff', fontSize: 12 },
  922. data: botmLeftBarData.legend
  923. },
  924. grid: {
  925. top: '14%',
  926. left: "4%",
  927. right: "8%",
  928. bottom: "2%",
  929. containLabel: true
  930. },
  931. xAxis: [{
  932. type: 'category',
  933. axisLine: {
  934. show: true,
  935. lineStyle: {
  936. color: '#00479D'
  937. }
  938. },
  939. axisLabel: {
  940. textStyle: {
  941. color: '#ffffff',
  942. fontSize: 12
  943. },
  944. },
  945. axisTick: {
  946. alignWithLabel: true
  947. },
  948. data: botmLeftBarData.category
  949. }],
  950. yAxis: [{
  951. type: 'value',
  952. splitLine: {
  953. show: true,
  954. lineStyle: {
  955. type: "solid",
  956. width: 1,
  957. color: "#153068"
  958. }
  959. },
  960. axisLine: {
  961. show: true,
  962. lineStyle: {
  963. color: '#00479D',
  964. width: 1
  965. }
  966. },
  967. axisLabel: {
  968. textStyle: {
  969. color: '#ffffff',
  970. fontSize: 12
  971. }
  972. },
  973. axisTick: {
  974. show: true
  975. },
  976. }],
  977. series: barseries
  978. }
  979. botmLeftChart.resize()
  980. botmLeftChart.clear()
  981. botmLeftChart.setOption(botmLeftOption)
  982. window.addEventListener("resize", function() {
  983. botmLeftChart.resize()
  984. })
  985. },
  986. creatPie(pieData) {
  987. let sum = pieData.data.reduce((prev, current) => prev + current.value, 0) // 数组求和
  988. pieData.data.forEach((item)=> {
  989. if (item.name == '其它') {
  990. item.labelLine = {
  991. lineStyle: {
  992. color: pieData.color[pieData.color.length-1]
  993. }
  994. }
  995. } else {
  996. item.labelLine = {
  997. lineStyle: {
  998. color: 'transparent'
  999. }
  1000. }
  1001. }
  1002. })
  1003. let myChart = echarts.init(document.getElementById(pieData.id))
  1004. let option = {
  1005. title: {
  1006. text: '{a|' + sum + '}{c|条}',
  1007. x: 'center',
  1008. y: 'center',
  1009. textStyle: {
  1010. rich: {
  1011. a: {
  1012. padding: [10, 0],
  1013. color: '#1EAFF5',
  1014. fontSize: 20,
  1015. align: 'center',
  1016. fontWeight: 'bold'
  1017. },
  1018. c: {
  1019. color: "#fff",
  1020. fontSize: 12,
  1021. }
  1022. }
  1023. },
  1024. },
  1025. tooltip: {
  1026. trigger: "item",
  1027. labelLine: false,
  1028. confine: true,
  1029. backgroundColor: '#011235',
  1030. borderWidth: 0,
  1031. extraCssText: 'opacity:0.8',
  1032. textStyle: {
  1033. color: '#fff'
  1034. },
  1035. formatter: function(params) {
  1036. return params.name + ':' + params.percent + '% (' + params.value + pieData.unit + ')'
  1037. }
  1038. },
  1039. series: [{
  1040. type: 'pie',
  1041. radius: ['50%', '70%'],
  1042. center: ['50%', '50%'],
  1043. label: {
  1044. position: 'outer',
  1045. alignTo: 'none',
  1046. bleedMargin: 0,
  1047. color: '#FFFFFF',
  1048. fontSize: 12,
  1049. formatter: function(params) {
  1050. if (params.data.name == '其它') {
  1051. return params.name + ':' + params.percent + '% (' + params.value + pieData.unit + ')'
  1052. } else {
  1053. return ''
  1054. }
  1055. }
  1056. },
  1057. itemStyle: {
  1058. normal: {
  1059. color: function (params) {
  1060. return pieData.color[params.dataIndex]
  1061. }
  1062. }
  1063. },
  1064. data: pieData.data,
  1065. }, ]
  1066. };
  1067. myChart.resize();
  1068. myChart.clear();
  1069. myChart.setOption(option);
  1070. window.addEventListener("resize", function() {
  1071. myChart.resize();
  1072. })
  1073. },
  1074. // getEchartData (params) {
  1075. // this.$get('metroapi/dataQuality/report/comScores',params).then(res => {
  1076. // if (res.httpCode == 1) {
  1077. // let obj = res.data
  1078. // obj.id = 'appGuage'
  1079. // this.$nextTick(() => this.creatGauge(obj))
  1080. // }
  1081. // })
  1082. // this.$get('metroapi/dataQuality/report/difDimScores',params).then(res => {
  1083. // if (res.httpCode == 1) {
  1084. // let obj2 = res.data
  1085. // obj2.id = 'appRadar'
  1086. // this.$nextTick(() => this.creatSolidRadar(obj2))
  1087. // }
  1088. // })
  1089. // },
  1090. async getEchartData1(url,params) {
  1091. let rtnData
  1092. await this.$get( url, params).then(res=>{
  1093. rtnData = res
  1094. })
  1095. return rtnData
  1096. },
  1097. downReport(row) {
  1098. this.downloadStatus = true
  1099. this.queryParams = row
  1100. let paramsObj = {
  1101. beginDate: row.beginDate,
  1102. endDate: row.endDate,
  1103. appId: row.appId
  1104. }
  1105. this.getData(paramsObj)
  1106. },
  1107. downPdf () {
  1108. this.$nextTick(()=> {
  1109. let title = this.queryParams.reportName+'-'+this.queryParams.appName
  1110. // 如果这个页面有左右移动,canvas 也要做响应的移动,不然会出现canvas 内容不全
  1111. const xOffset = window.pageXOffset
  1112. // 避免笔下误 灯下黑 统一写
  1113. const A4_WIDTH = 592.28
  1114. // const A4_HEIGHT = 841.89
  1115. const A4_HEIGHT = 841.89
  1116. let printDom = this.$refs.downReport
  1117. // 根据A4的宽高计算DOM页面一页应该对应的高度
  1118. let pageHeight = printDom.offsetWidth / A4_WIDTH * A4_HEIGHT
  1119. // 将所有不允许被截断的元素进行处理
  1120. let wholeNodes = document.querySelectorAll('.no-truncation') //pdf加分页防止内容截断
  1121. for (let i = 0; i < wholeNodes.length; i++) {
  1122. //1、 判断当前的不可分页元素是否在两页显示
  1123. const topPageNum = Math.ceil((wholeNodes[i].offsetTop) / pageHeight) // wholeNodes[i].offsetTop 它返回当前元素相对于其 offsetParent 元素的顶部内边距的距离 如果算距离对顶部的距离它的父级不能设置定位,,否则结果为0
  1124. const bottomPageNum = Math.ceil((wholeNodes[i].offsetTop + wholeNodes[i].offsetHeight) / pageHeight)
  1125. if (topPageNum !== bottomPageNum) {
  1126. //说明该dom会被截断
  1127. // 2、插入空白块使被截断元素下移
  1128. let divParent = wholeNodes[i].parentNode
  1129. let newBlock = document.createElement('div')
  1130. newBlock.className = 'emptyDiv'
  1131. newBlock.style.background = '#06214d'
  1132. // 3、计算插入空白块的高度 可以适当流出空间使得内容太靠边,根据自己需求而定
  1133. let _H = topPageNum * pageHeight - wholeNodes[i].offsetTop
  1134. newBlock.style.height = _H + 30 + 'px'
  1135. divParent.insertBefore(newBlock, wholeNodes[i])
  1136. }
  1137. }
  1138. // 以上完成dom层面的分页 可以转为图片进一步处理了
  1139. html2canvas(printDom, { height: printDom.offsetHeight, width: printDom.offsetWidth, scrollX: -xOffset,backgroundColor: "#fff",allowTaint: true ,}).then(canvas => {
  1140. //dom 已经转换为canvas 对象,可以将插入的空白块删除了
  1141. let emptyDivs = document.querySelectorAll('.emptyDiv')
  1142. for (let i = 0; i < emptyDivs.length; i++) {
  1143. emptyDivs[i].parentNode.removeChild(emptyDivs[i])
  1144. }
  1145. // 有一点重复的代码
  1146. let contentWidth = canvas.width
  1147. let contentHeight = canvas.height
  1148. let pageHeight = contentWidth / A4_WIDTH * A4_HEIGHT
  1149. let leftHeight = contentHeight
  1150. let position = 0
  1151. let imgWidth = A4_WIDTH
  1152. let imgHeight = A4_WIDTH / contentWidth * contentHeight
  1153. let pageData = canvas.toDataURL('image/jpeg', 1.0)
  1154. // if (isPrint) {
  1155. // //如果是打印,可以拿着分号页的数据 直接使用
  1156. // printJs({ printable: pageData, type: 'image', base64: true, documentTitle: '\u200E' })
  1157. // return
  1158. // }
  1159. //计算分页的pdf
  1160. let PDF = new JsPDF('', 'pt', 'a4')
  1161. if (leftHeight <= pageHeight) {
  1162. PDF.addImage(pageData, 'JPEG', 0, 0, imgWidth, imgHeight)
  1163. } else {
  1164. while (leftHeight > 0) {
  1165. PDF.addImage(pageData, 'JPEG', 0, position, imgWidth, imgHeight)
  1166. leftHeight -= pageHeight
  1167. position -= A4_HEIGHT
  1168. if (leftHeight > 0) {
  1169. PDF.addPage()
  1170. }
  1171. }
  1172. }
  1173. this.downloadStatus = false
  1174. PDF.save(title + '.pdf')
  1175. })
  1176. })
  1177. },
  1178. getData(params){
  1179. let interface1 = this.getEchartData1('metroapi/dataQuality/report/comScores',params)
  1180. let interface2 = this.getEchartData1('metroapi/dataQuality/report/difDimScores',params)
  1181. let interface3 = this.getEchartData1('metroapi/dataQuality/report/comScoresTrend',params)
  1182. let interface4 = this.getEchartData1('metroapi/dataQuality/report/comScores/model/bar',params)
  1183. let interfaceTable4 = this.getEchartData1('metroapi/dataQuality/report/comScores/model/list',params)
  1184. let interface5 = this.getEchartData1('metroapi/dataQuality/report/abnDataAnalysis/model/pie',params)
  1185. let interfaceTable5 = this.getEchartData1('metroapi/dataQuality/report/abnDataAnalysis/model/list',params)
  1186. let interface6 = this.getEchartData1('metroapi/dataQuality/report/abnSource/rule/bar',params)
  1187. let interfaceTable6 = this.getEchartData1('metroapi/dataQuality/report/abnSource/rule/list',params)
  1188. let interface7 = this.getEchartData1('metroapi/dataQuality/report/ruleConf/pie',params)
  1189. let interfaceTable7 = this.getEchartData1('metroapi/dataQuality/report/ruleConf/list',params)
  1190. this.$axiosAll([interface1,interface2,interface3,interface4,interfaceTable4,interface5,interfaceTable5,interface6,interfaceTable6,interface7,interfaceTable7]).then(res=> {
  1191. if (res[0].httpCode == 1) {
  1192. let obj = res[0].data
  1193. obj.id = 'appGuage'
  1194. this.$nextTick(() => this.creatGauge(obj))
  1195. }
  1196. if (res[1].httpCode == 1) {
  1197. let obj2 = res[1].data
  1198. obj2.id = 'appRadar'
  1199. this.$nextTick(() => this.creatSolidRadar(obj2))
  1200. }
  1201. if (res[2].httpCode == 1) {
  1202. let obj3 = res[2].data
  1203. obj3.id = 'appLine3'
  1204. obj3.unit = '%'
  1205. obj3.lineColor = ['#31A4FD', '#86BE73']
  1206. this.$nextTick(() => this.creatLine(obj3))
  1207. }
  1208. if (res[3].httpCode == 1) {
  1209. let obj4 = res[3].data
  1210. obj4.id = 'appLBar4'
  1211. obj4.unit = '%'
  1212. obj4.color = [['#0874E2','#62BAFD'], ['#86BE73', '#42A76D']]
  1213. this.$nextTick(() => this.creatBar(obj4))
  1214. }
  1215. if (res[4].httpCode == 1) {
  1216. this.tableData4 = res[4].data.data
  1217. this.tableData4.forEach((item,index)=> {
  1218. if (index == 0) {
  1219. item.cellClassName = {
  1220. paramLong01: 'total-records-table-cell',
  1221. paramDou01: 'quality-table-cell',
  1222. paramDou02: 'quality-table-cell',
  1223. }
  1224. }
  1225. })
  1226. }
  1227. if (res[5].httpCode == 1) {
  1228. let obj5 = res[5].data
  1229. obj5.id = 'appPie5'
  1230. obj5.unit = '条'
  1231. obj5.color = this.pieColumnColor5
  1232. this.$nextTick(() => this.creatPie(obj5))
  1233. }
  1234. if (res[6].httpCode == 1) {
  1235. this.tableData5 = res[6].data.data
  1236. this.tableData5.forEach((item,index)=> {
  1237. if (index == 0) {
  1238. item.cellClassName = {
  1239. paramNum01: 'total-records-table-cell',
  1240. paramNum02: 'total-exceptions-table-cell',
  1241. }
  1242. }
  1243. })
  1244. }
  1245. if (res[7].httpCode == 1) {
  1246. let obj6 = res[7].data
  1247. obj6.id = 'appBar6'
  1248. obj6.unit = '条'
  1249. obj6.legend = ['']
  1250. obj6.color = [['#0874E2','#62BAFD']]
  1251. this.$nextTick(() => this.creatBar(obj6))
  1252. }
  1253. if (res[8].httpCode == 1) {
  1254. this.tableData6 = res[8].data.data
  1255. this.tableData6.forEach((item,index)=> {
  1256. if (index == 0) {
  1257. item.cellClassName = {
  1258. paramLong01: 'total-records-table-cell',
  1259. paramNum01: 'total-exceptions-table-cell'
  1260. }
  1261. }
  1262. })
  1263. }
  1264. if (res[9].httpCode == 1) {
  1265. let obj7 = res[9].data
  1266. obj7.id = 'appPie7'
  1267. obj7.unit = '条'
  1268. obj7.color = this.pieColumnColor7
  1269. this.$nextTick(() => this.creatPie(obj7))
  1270. }
  1271. if (res[10].httpCode == 1) {
  1272. this.tableData7 = res[10].data.data
  1273. this.tableData7.forEach((item,index)=> {
  1274. if (index == 0) {
  1275. item.cellClassName = {
  1276. paramNum01: 'quality-table-cell',
  1277. }
  1278. }
  1279. })
  1280. }
  1281. this.$nextTick(()=> {
  1282. if (this.downloadStatus) {
  1283. setTimeout(() => {
  1284. this.downPdf()
  1285. }, 1000)
  1286. }
  1287. })
  1288. })
  1289. }
  1290. }
  1291. };
  1292. </script>
  1293. <style lang="stylus">
  1294. * {
  1295. margin: 0;
  1296. padding: 0;
  1297. }
  1298. .quality-report {
  1299. // z-index:9999;
  1300. width: 100%;
  1301. position: relative;
  1302. display: flex;
  1303. justify-content: center;
  1304. background: #06214D;
  1305. }
  1306. </style>
  1307. <style lang="stylus" scoped>
  1308. .nav-content {
  1309. position:fixed;
  1310. right: 10PX;
  1311. top: 164px;
  1312. display: flex;
  1313. flex-direction: column;
  1314. z-index: 99;
  1315. // width: 108px;
  1316. /* height: 44px; */
  1317. // background: #214388;
  1318. // header:300px;
  1319. }
  1320. .nav-list {
  1321. height: 48px;
  1322. line-height: 48px;
  1323. text-align: center;
  1324. color: #FFFFFF;
  1325. font-size: 16px;
  1326. font-weight: bold;
  1327. margin-bottom: 20px;
  1328. background: linear-gradient(-79deg, #062D6F, #0185EA);
  1329. opacity: 0.6;
  1330. padding: 0 10px;
  1331. }
  1332. .report-container {
  1333. width: 70%;
  1334. margin-bottom: 20px;
  1335. position: relative;
  1336. }
  1337. .report-header {
  1338. text-align: center;
  1339. background: url('../assets/images/reportBg.png') no-repeat center;
  1340. background-size: 100% 100%;
  1341. height: 132px;
  1342. }
  1343. .header-title {
  1344. font-size: 36px;
  1345. font-weight: bold;
  1346. color: #FFFFFF;
  1347. padding-top: 32px;
  1348. line-height: 38px;
  1349. }
  1350. .header-subtitle {
  1351. font-size: 14px;
  1352. font-weight: 400;
  1353. color: #FFFFFF;
  1354. margin: 28px 0 22px;
  1355. line-height: 14px;
  1356. display: flex;
  1357. justify-content: space-around;
  1358. }
  1359. .report-content {
  1360. background: #0F2D5E;
  1361. padding: 0 16px;
  1362. }
  1363. .report-section {
  1364. margin: 18px 0;
  1365. }
  1366. .report-section:first-child {
  1367. margin-top: 0;
  1368. padding-top: 18px;
  1369. }
  1370. .title-content{
  1371. width: 100%;
  1372. height: 33px;
  1373. padding-bottom: 20px;
  1374. border-bottom: 2px solid #E5E5E5;
  1375. margin-bottom: 24px;
  1376. }
  1377. .title-line {
  1378. background: #1E58B8;
  1379. border-radius: 4px;
  1380. height: 45px;
  1381. padding-left: 20px;
  1382. display: flex;
  1383. align-items: center;
  1384. font-size: 14px;
  1385. color: #FFFFFF;
  1386. span {
  1387. height: 18px;
  1388. border-left: 1PX solid #FFFFFF;
  1389. padding-left: 10px;
  1390. }
  1391. }
  1392. .report-echarts {
  1393. .report-echarts-row {
  1394. margin-top: 10px;
  1395. margin-bottom: 10px;
  1396. }
  1397. .report-echarts-main {
  1398. // height: 310px;
  1399. border: 1px solid #2259B3;
  1400. border-radius: 4px;
  1401. }
  1402. .echarts-title {
  1403. font-size: 16px;
  1404. font-weight: 400;
  1405. color: #31A4FD;
  1406. line-height: 16px;
  1407. padding: 18px 18px 0 18px;
  1408. }
  1409. .echarts-sub-title {
  1410. font-size: 12px;
  1411. color: #fff;
  1412. padding: 20px 18px 0 18px;
  1413. }
  1414. }
  1415. .echart-detail {
  1416. width: 100%;
  1417. height: 290px;
  1418. }
  1419. // .echart-pie {
  1420. // width: 100%;
  1421. // height: 350px;
  1422. // }
  1423. .current-item {
  1424. background: linear-gradient(-79deg, #0185EA, #9FD0F5);
  1425. }
  1426. >>> .table-legend-list .ivu-table-cell{
  1427. display: flex;
  1428. align-items: center;
  1429. justify-content: center;
  1430. }
  1431. .table-legend {
  1432. width: 12px;
  1433. height: 12px;
  1434. border-radius: 4px;
  1435. display:inline-block;
  1436. }
  1437. .report-echarts-table {
  1438. padding: 20px 40px;
  1439. }
  1440. .report-echarts-right-table {
  1441. padding-top: 0;
  1442. padding-left: 0;
  1443. }
  1444. >>> .report-table-detail .ivu-table-stripe-even td {
  1445. background-color: #0F2D5E;
  1446. height: 40px;
  1447. }
  1448. >>> .report-table-detail .ivu-table-stripe-odd td {
  1449. background-color: #15376F;
  1450. height: 40px;
  1451. }
  1452. >>> .report-echarts-table .ivu-table .ivu-table-header .ivu-table-cell {
  1453. white-space: pre;
  1454. }
  1455. >>> .ivu-row .ivu-col {
  1456. position: static;
  1457. }
  1458. >>> .quality-table-cell span{
  1459. color:#5EB6FC;
  1460. }
  1461. >>> .total-records-table-cell span {
  1462. color: #42C151;
  1463. }
  1464. >>> .total-exceptions-table-cell span {
  1465. color: #D70101;
  1466. }
  1467. </style>