fittingEquationChartEditor.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. <template>
  2. <div class="fittingEquation-chart-editor-container" id="box">
  3. <span
  4. class="slide-icon slide-right"
  5. @click="isSlideLeft = !isSlideLeft"
  6. v-show="isSlideLeft"
  7. >
  8. <i class="el-icon-d-arrow-right"></i>
  9. </span>
  10. <div class="left-cont" v-show="!isSlideLeft" id="left">
  11. <div class="left-top">
  12. <el-button type="primary" @click="saveHandle">{{$t('Dialog.confirm_save_btn')}}</el-button>
  13. <el-button style="margin-right:20px" type="primary" plain @click="$router.replace({
  14. path: '/fittingEquationList',
  15. query: $route.query
  16. })">{{$t('Dialog.cancel_btn')}}</el-button>
  17. <div style="color:#409EFF;font-size: 16px;cursor: pointer;" @click="showExplain = true">
  18. <i class="el-icon-document" style="font-size:22px;"></i>
  19. {{$t('StatisticAnalysis.ChartRelevance.opt_tip_btn')}}
  20. </div>
  21. </div>
  22. <div class="left-min">
  23. <div class="section-item">
  24. <label style="display: block">{{$t('StatisticAnalysis.ChartRelevance.time')}}:</label>
  25. <el-select
  26. style="width: 120px;margin: 10px 0"
  27. v-model="leftOption.DateType"
  28. placeholder="请选择"
  29. @change="leftOption.date=[]"
  30. >
  31. <el-option
  32. v-for="item in yearSelector"
  33. :key="item.value"
  34. :label="item.name"
  35. :value="item.value"
  36. />
  37. </el-select>
  38. <date-picker
  39. v-model="leftOption.date"
  40. v-show="leftOption.DateType===5"
  41. style="width:120px;"
  42. type="date"
  43. range
  44. value-type="format"
  45. :placeholder="$t('Chart.choose_time')"
  46. />
  47. </div>
  48. <el-collapse v-model="activeNames" class="targetset-cont">
  49. <el-collapse-item
  50. class="series-li"
  51. :name="0"
  52. >
  53. <template slot="title">
  54. <el-button type="text" size="small" style="font-size: 15px;">{{$t('StatisticAnalysis.FittingEquationChart.x_edb')}}</el-button>
  55. </template>
  56. <div class="setting-wrapper">
  57. <selectTarget
  58. ref="xSelectRef"
  59. @select="chooseXEdbs"
  60. />
  61. <ul class="edb-list">
  62. <li v-for="(item,index) in leftOption.xEdbs" :key="item.id">
  63. <div class="info">
  64. <el-popover
  65. placement="top-start"
  66. width="200"
  67. trigger="hover"
  68. :content="item.name">
  69. <div class="text_oneLine" slot="reference" style="max-width:200px;">
  70. <span>{{index+1}}:</span>
  71. <span>{{item.name}}</span>
  72. </div>
  73. </el-popover>
  74. <i class="el-icon-delete deletesty" @click="removeTarget('x',index)"></i>
  75. </div>
  76. </li>
  77. </ul>
  78. </div>
  79. </el-collapse-item>
  80. <el-collapse-item
  81. class="series-li"
  82. :name="1"
  83. >
  84. <template slot="title">
  85. <el-button type="text" size="small" style="font-size: 15px;">{{$t('StatisticAnalysis.FittingEquationChart.y_edb')}}</el-button>
  86. </template>
  87. <div class="setting-wrapper">
  88. <selectTarget
  89. ref="ySelectRef"
  90. @select="chooseYEdbs"
  91. />
  92. <ul class="edb-list">
  93. <li v-for="(item,index) in leftOption.yEdbs" :key="item.id">
  94. <div class="info">
  95. <el-popover
  96. placement="top-start"
  97. width="200"
  98. trigger="hover"
  99. :content="item.name">
  100. <div class="text_oneLine" slot="reference" style="max-width:200px;">
  101. <span>{{index+1}}:</span>
  102. <span>{{item.name}}</span>
  103. </div>
  104. </el-popover>
  105. <i class="el-icon-delete deletesty" @click="removeTarget('y',index)"></i>
  106. </div>
  107. </li>
  108. </ul>
  109. </div>
  110. </el-collapse-item>
  111. </el-collapse>
  112. <el-button type="primary" @click="getPreviewData">{{$t('Dialog.confirm_btn')}}</el-button>
  113. </div>
  114. <span
  115. class="move-btn resize"
  116. v-drag
  117. id="resize"
  118. ></span>
  119. <span class="slide-icon slide-left" @click="isSlideLeft = !isSlideLeft">
  120. <i class="el-icon-d-arrow-left"></i>
  121. </span>
  122. </div>
  123. <div class="right-cont" id="right" :style="isSlideLeft ? 'width:100%' : `width:80%`">
  124. <div class="chart-min-cont" v-if="chartBatchData.length">
  125. <div class="card-wrapper" v-for="(item,index) in chartBatchData" :key="index">
  126. <chartCard
  127. :entryType="index+1"
  128. :ref="`chartCard${index+1}`"
  129. :data="item"
  130. />
  131. </div>
  132. </div>
  133. <div class="nodata" v-else>
  134. <tableNoData text="暂无信息"/>
  135. </div>
  136. </div>
  137. <!-- 保存弹窗 -->
  138. <fittingEquationSaveDia
  139. :isShow.sync="isSaveDialog"
  140. :initData="leftOption.ChartMappingList"
  141. @saveBack="editChartBackHandle"
  142. />
  143. <!-- 操作说明 -->
  144. <ExplainDialog
  145. textArrName="fittingEquationListTextArr"
  146. :show-explain="showExplain"
  147. @close="showExplain = false"
  148. />
  149. </div>
  150. </template>
  151. <script>
  152. import { fittingEquationInterface } from '@/api/modules/chartRelevanceApi';
  153. import { yearSelector } from '@/utils/defaultOptions';
  154. import Chart from '@/views/dataEntry_manage/components/chart';
  155. import selectTarget from '../components/selectTarget.vue';
  156. import fittingEquationSaveDia from '../components/fittingEquationSaveDia.vue';
  157. import chartCard from '../components/chartCard.vue';
  158. import ExplainDialog from '../components/explainDialog.vue';
  159. export default {
  160. components: { Chart,selectTarget,fittingEquationSaveDia,chartCard,ExplainDialog },
  161. directives: {
  162. drag(el, bindings) {
  163. el.onmousedown = function (e) {
  164. var init = e.clientX;
  165. // console.log(init);
  166. var box = $('#box')[0];
  167. // console.log(box.clientWidth)
  168. let total_wid = box.offsetWidth;
  169. var left = $('#left')[0];
  170. var right = $('#right')[0];
  171. var initWidth = left.offsetWidth;
  172. document.onmousemove = function (e) {
  173. var end = e.clientX;
  174. var newWidth = end - init + initWidth;
  175. left.style.width = newWidth + 'px';
  176. right.style.width = newWidth > 300 ? total_wid - newWidth + 'px' : total_wid - 320 + 'px';
  177. };
  178. document.onmouseup = function () {
  179. document.onmousemove = document.onmouseup = null;
  180. e.releaseCapture && e.releaseCapture();
  181. };
  182. e.setCapture && e.setCapture();
  183. return false;
  184. };
  185. },
  186. },
  187. computed: {
  188. roleName() {
  189. return localStorage.getItem('userName');
  190. },
  191. yearSelector(){
  192. return [
  193. ...yearSelector,
  194. { name: this.$i18n.locale == 'zh'?'自定义':'custom',value: 5 }
  195. ]
  196. }
  197. },
  198. data() {
  199. return {
  200. chart_code: this.$route.query.code || '',
  201. isSlideLeft: false,
  202. activeNames:[0],
  203. chartBatchData: [],
  204. dateSelector: [
  205. { name: '过去1年',value: 1 },
  206. { name: '过去2年',value: 2 },
  207. { name: '过去3年',value: 3 },
  208. { name: '过去5年',value: 4 },
  209. { name: '过去10年',value: 5 },
  210. { name: '自定义',value: 6 },
  211. ],
  212. leftOption: {
  213. MultipleGraphConfigId: 0,
  214. ChartMappingList:[],
  215. DateType: yearSelector[0].value,
  216. date: [],
  217. xEdbs: [],
  218. yEdbs: []
  219. },
  220. classifyOptions: [],//分类option
  221. isSaveDialog: false,
  222. useTip:`1、选择两组有相关性的指标<br>
  223. 2、对两组在选定时间范围内的每一个时间节点,生成拟合方程Y=aX+b以及相关系数R²<br>
  224. 3、分别画出弹性系数a,截距b,相关系数R²,在选定时间范围内的曲线图`,
  225. //操作说明弹窗
  226. showExplain:false
  227. };
  228. },
  229. methods: {
  230. /* 获取图表详情 */
  231. getChartInfo() {
  232. fittingEquationInterface.getConfigSetting({
  233. UniqueCode: this.chart_code,
  234. }).then((res) => {
  235. if (res.Ret !== 200) return;
  236. const { ChartMappingList,EdbInfoList,LineChartInfoConfig,MultipleGraphConfigId } = res.Data;
  237. this.leftOption = {
  238. MultipleGraphConfigId,
  239. ChartMappingList,
  240. DateType: LineChartInfoConfig.DateType,
  241. date: [LineChartInfoConfig.StartDate,LineChartInfoConfig.EndDate],
  242. xEdbs: LineChartInfoConfig.XEdbInfoIdList.map(_ => ({ id: _,name:EdbInfoList.find(_edb =>_edb.EdbInfoId===_).EdbName })),
  243. yEdbs: LineChartInfoConfig.YEdbInfoIdList.map(_ => ({ id: _,name:EdbInfoList.find(_edb =>_edb.EdbInfoId===_).EdbName })),
  244. }
  245. this.getPreviewData();
  246. });
  247. },
  248. /* 获取数据 */
  249. getPreviewData() {
  250. if(!this.leftOption.xEdbs.length || !this.leftOption.yEdbs.length) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg01'));
  251. if(this.leftOption.xEdbs.length < 2 || this.leftOption.yEdbs.length < 2) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg02'))
  252. if(this.leftOption.yEdbs.length !== this.leftOption.xEdbs.length) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg03'));
  253. const { DateType,date,xEdbs,yEdbs } = this.leftOption;
  254. let params = {
  255. DateType,
  256. StartDate: date[0]||'',
  257. EndDate: date[1]||'',
  258. XEdbInfoIdList: xEdbs.map(_ => _.id),
  259. YEdbInfoIdList: yEdbs.map(_ => _.id)
  260. }
  261. const chartNameMap = {
  262. 1: this.leftOption.ChartMappingList.find(_ => _.Source===5) ? this.leftOption.ChartMappingList.find(_ => _.Source===5).ChartName : '',
  263. 2: this.leftOption.ChartMappingList.find(_ => _.Source===6) ? this.leftOption.ChartMappingList.find(_ => _.Source===6).ChartName : '',
  264. 3: this.leftOption.ChartMappingList.find(_ => _.Source===7) ? this.leftOption.ChartMappingList.find(_ => _.Source===7).ChartName : '',
  265. }
  266. fittingEquationInterface.previewData(params).then(res => {
  267. if(res.Ret !== 200) return
  268. this.chartBatchData = res.Data.List ? res.Data.List.map((_,index) =>({
  269. ChartInfo: { Source: 6,ChartName: chartNameMap[index+1] },
  270. EdbInfo: {
  271. ..._,
  272. ChartColor:'#00f',
  273. ChartStyle:'spline',
  274. PredictChartColor: '#00f',
  275. ChartType:0,
  276. ChartWidth:3,
  277. EdbName:_.Name,
  278. EdbNameEn: _.NameEn,
  279. IsAxis:1
  280. }
  281. })) : [];
  282. })
  283. },
  284. // 选择指标
  285. chooseXEdbs(edb) {
  286. if(edb){
  287. this.leftOption.xEdbs.push({
  288. id: edb.EdbInfoId,
  289. name: edb.EdbName,
  290. })
  291. this.$refs.xSelectRef.search_txt = '';
  292. }
  293. },
  294. chooseYEdbs(edb) {
  295. if(edb) {
  296. if(this.leftOption.yEdbs)
  297. this.leftOption.yEdbs.push({
  298. id: edb.EdbInfoId,
  299. name: edb.EdbName,
  300. })
  301. this.$refs.ySelectRef.search_txt = '';
  302. }
  303. },
  304. /* 删除指标 */
  305. removeTarget(type,index) {
  306. type === 'x' ? this.leftOption.xEdbs.splice(index,1) : this.leftOption.yEdbs.splice(index,1);
  307. },
  308. reloadRightWid() {
  309. let total_wid = $('#box')[0].offsetWidth;
  310. let left = $('#left')[0].offsetWidth;
  311. let rigtWid = total_wid - left - 20 + 'px';
  312. $('#right')[0].style.width = rigtWid;
  313. },
  314. /* 保存 */
  315. async saveHandle() {
  316. if(!this.leftOption.xEdbs.length || !this.leftOption.yEdbs.length) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg01'));
  317. if(this.leftOption.xEdbs.length < 2 || this.leftOption.yEdbs.length < 2) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg02'))
  318. if(this.leftOption.yEdbs.length !== this.leftOption.xEdbs.length) return this.$message.warning(this.$t('StatisticAnalysis.FittingEquationChart.msg03'));
  319. this.isSaveDialog = true;
  320. },
  321. /* 保存成功跳列表 */
  322. editChartBackHandle(query) {
  323. this.$router.replace({
  324. path: '/fittingEquationList',
  325. query
  326. })
  327. }
  328. },
  329. mounted() {
  330. this.chart_code && this.getChartInfo();
  331. window.addEventListener('resize', this.reloadRightWid);
  332. },
  333. destroyed() {
  334. window.removeEventListener('resize', this.reloadRightWid);
  335. },
  336. beforeRouteEnter(to, from, next) {
  337. if(to.query.code){
  338. to.matched[1].name='编辑图表'
  339. }else{
  340. to.matched[1].name='添加图表'
  341. }
  342. next()
  343. }
  344. };
  345. </script>
  346. <style lang="scss">
  347. .fittingEquation-chart-editor-container {
  348. $font-small: 12px; $font-normal: 14px;
  349. display: flex;
  350. *{ box-sizing: border-box; }
  351. .el-date-editor .el-range-separator {
  352. width: 10%;
  353. }
  354. .slide-icon {
  355. padding: 20px 0;
  356. /* display: block; */
  357. box-shadow: 0px 3px 6px rgba(0, 0, 0, 0.3);
  358. border-radius: 5px;
  359. cursor: pointer;
  360. position: absolute;
  361. top: 50%;
  362. transform: translateY(-50%);
  363. z-index: 99;
  364. &:hover {
  365. background-color: rgba(0, 0, 0, 0.05);
  366. }
  367. &.slide-left {
  368. right: 0;
  369. }
  370. &.slide-right {
  371. left: 0;
  372. }
  373. }
  374. .left-cont {
  375. width: 400px;
  376. min-width: 310px;
  377. background: #fff;
  378. margin-right: 20px;
  379. border: 1px solid #ececec;
  380. border-radius: 4px;
  381. box-shadow: 0px 3px 6px rgba(0, 0, 0, 0.05);
  382. height: calc(100vh - 113px);
  383. overflow: hidden;
  384. position: relative;
  385. box-sizing: border-box;
  386. .left-top {
  387. padding: 15px 20px;
  388. border: 1px solid #ececec;
  389. box-shadow: 0px 3px 6px rgba(167, 167, 167, 0.09);
  390. display: flex;
  391. align-items: center;
  392. justify-content: space-between;
  393. }
  394. .left-min {
  395. padding: 30px 20px;
  396. max-height: calc(100vh - 194px);
  397. overflow-y: auto;
  398. .targetset-cont {
  399. border: 1px solid #DCDFE6;
  400. margin: 20px 0;
  401. .el-collapse-item.is-disabled .el-collapse-item__header {
  402. color: #333;
  403. }
  404. .setting-wrapper {
  405. padding: 15px;
  406. }
  407. .el-collapse-item__header {
  408. background-color: #F0F2F5;
  409. margin-bottom: 0;
  410. border-bottom: 1px solid #DCDFE6;
  411. padding: 0 30px;
  412. .el-collapse-item__arrow {
  413. position: absolute;
  414. left: 8px;
  415. }
  416. }
  417. .edb-list {
  418. margin: 10px 0;
  419. .info {
  420. display: flex;
  421. justify-content: space-between;
  422. align-items: center;
  423. margin-bottom: 10px;
  424. color: #666;
  425. }
  426. }
  427. }
  428. }
  429. .move-btn {
  430. height: 100%;
  431. width: 4px;
  432. position: absolute;
  433. right: 0px;
  434. top: 0;
  435. &:hover {
  436. cursor: col-resize;
  437. }
  438. }
  439. }
  440. .right-cont {
  441. width: 80%;
  442. .mx-datepicker {
  443. width: 220px !important;
  444. }
  445. .chart-min-cont {
  446. background: #fff;
  447. height: calc(100vh - 118px);
  448. border: 1px solid #ececec;
  449. overflow-y: auto;
  450. display: flex;
  451. flex-wrap: wrap;
  452. .card-wrapper {
  453. width: 48%;
  454. margin-bottom: 20px;
  455. min-height: 350px;
  456. min-width: 410px;
  457. margin-right: 15px;
  458. .card-item {
  459. padding: 20px;
  460. .top {
  461. display: flex;
  462. justify-content: space-between;
  463. align-items: center;
  464. }
  465. .chartWrapper {
  466. .title {
  467. font-size: 15px;
  468. text-align: center;
  469. margin: 10px 0;
  470. }
  471. }
  472. }
  473. }
  474. }
  475. .nodata {
  476. height: calc(100vh - 120px);
  477. background-color: #fff;
  478. text-align: center;
  479. font-size: 16px;
  480. color: #666;
  481. padding: 100px 0;
  482. }
  483. }
  484. }
  485. </style>
  486. <style lang="scss">
  487. .edb-item-style .el-input__icon {
  488. line-height: 27px;
  489. }
  490. </style>