batchComputedV2.vue 36 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982
  1. <template>
  2. <el-dialog
  3. :visible.sync="isShow"
  4. :close-on-click-modal="false"
  5. :modal-append-to-body="false"
  6. @close="handleClose"
  7. custom-class="batch-computed-dialog fit-screen-dialog"
  8. center
  9. top="5vh"
  10. v-dialogDrag
  11. :title="showSave?'批量计算-结果保存':'批量计算-选择指标和计算公式'"
  12. >
  13. <div class="batch-computed-wrap" v-if="!showSave">
  14. <el-form inline :disabled="operationForm.view">
  15. <div class="type-wrap">
  16. <div style="margin-right:50px">
  17. <span>计算公式</span>
  18. <el-select :disabled="isEdit" v-model="computedType" @change="handleComputedTypeChange" placeholder="请选择">
  19. <el-option
  20. v-for="opt in computedBatchTypes"
  21. :key="opt.type"
  22. :label="opt.name"
  23. :value="opt.type"
  24. />
  25. </el-select>
  26. </div>
  27. <div style="flex:1">
  28. <!-- 累计值转月/季值 -->
  29. <el-form-item v-if="computedType=='toMonthSeason'">
  30. <el-radio-group v-model="subComputedType" @change="handleComputedSubTypeChange">
  31. <el-radio :label="5">转月值</el-radio>
  32. <el-radio :label="61">转季值</el-radio>
  33. </el-radio-group>
  34. </el-form-item>
  35. <!-- N期移动均值、N期环比值、N期环差值 -->
  36. <el-form-item required v-if="[8,12,13].includes(computedType)" label="N等于" style="width:280px">
  37. <el-input v-model="formData.nNum" placeholder="请输入N数值" type="number" style="width:200px"></el-input>
  38. </el-form-item>
  39. <!-- 超级季节性 -->
  40. <template v-if="computedType==35">
  41. <el-form-item required label="N等于" style="width:280px">
  42. <el-input v-model="formData.nNum" placeholder="请输入N数值" type="number" style="width:200px"></el-input>
  43. </el-form-item>
  44. <el-form-item label="日历" style="width:180px">
  45. <el-select v-model="formData.calendarType" style="width:120px">
  46. <el-option label="公历" value="公历"></el-option>
  47. <el-option label="农历" value="农历"></el-option>
  48. </el-select>
  49. </el-form-item>
  50. </template>
  51. <!-- 降频 -->
  52. <template v-if="computedType==51">
  53. <el-form-item required label="频度" style="width:180px">
  54. <el-select v-model="formData.frequency" style="width:120px">
  55. <el-option v-for="opt in frequencyArr" :key="opt" :label="opt" :value="opt"></el-option>
  56. </el-select>
  57. </el-form-item>
  58. <el-form-item label="数据取值" style="width:200px">
  59. <el-select v-model="formData.valueType" style="width:120px">
  60. <el-option key="期末值" label="期末值" value="期末值"/>
  61. <el-option key="平均值" label="平均值" value="平均值"/>
  62. </el-select>
  63. </el-form-item>
  64. </template>
  65. <!-- 累计值 -->
  66. <template v-if="computedType=='accumulate'">
  67. <el-form-item>
  68. <el-radio v-model="subComputedType" :label="62">累计值</el-radio>
  69. </el-form-item>
  70. <el-form-item required label="频度" style="width:180px" v-if="subComputedType==62">
  71. <el-select v-model="formData.frequency" style="width:120px">
  72. <el-option v-for="opt in frequencyArr" :key="opt" :label="opt" :value="opt"></el-option>
  73. </el-select>
  74. </el-form-item>
  75. <el-form-item label="最新值处理" style="width:210px" v-if="subComputedType==62">
  76. <el-select v-model="formData.newValue" style="width:120px">
  77. <el-option label="默认" :value="0"/>
  78. <el-option label="均值填充" :value="1"/>
  79. </el-select>
  80. </el-form-item>
  81. <el-form-item>
  82. <el-radio v-model="subComputedType" :label="63">年初至今累计值</el-radio>
  83. </el-form-item>
  84. </template>
  85. <!-- 指数修匀 -->
  86. <template v-if="computedType==72">
  87. <el-form-item required label="alpha值" style="width:220px">
  88. <el-input v-model.trim="formData.alphaValue" style="width:140px" placeholder="请输入alpha值"></el-input>
  89. </el-form-item>
  90. </template>
  91. <!-- 与常数计算 -->
  92. <template v-if="computedType=='withNum'">
  93. <el-form-item required label="公式">
  94. <el-input v-model="formData.formula" placeholder="请输入公式" clearable style="width: 200px"/>
  95. <span>公式示例:A+100,或A*2</span>
  96. </el-form-item>
  97. </template>
  98. <!-- 与单指标计算 -->
  99. <template v-if="computedType=='withEDB'">
  100. <el-form-item required label="指标B">
  101. <el-select
  102. v-model="select_target"
  103. v-loadMore="searchLoad"
  104. :filterable="!select_target"
  105. clearable
  106. placeholder="请输入指标名称"
  107. style="width: 250px"
  108. remote
  109. :remote-method="getTarget"
  110. @click.native="inputFocusHandle"
  111. @change="chooseTarget"
  112. @blur="search_have_more = false"
  113. >
  114. <i slot="prefix" class="el-input__icon el-icon-search"></i>
  115. <el-option
  116. v-for="item in searchOptions"
  117. :key="item.EdbInfoId"
  118. :label="$parent.currentLang==='en'?(item.EdbNameEn||item.EdbName):item.EdbName"
  119. :value="item.EdbInfoId"
  120. />
  121. </el-select>
  122. </el-form-item>
  123. <el-form-item required label="公式">
  124. <el-input v-model="formData.formula" placeholder="请输入公式" clearable style="width: 200px"/>
  125. <span>公式示例:A+B,或A/B,A/(A+B)</span>
  126. </el-form-item>
  127. <el-form-item label="生成指标时间序列">
  128. <el-cascader
  129. v-model="formData.timeSeriesVal"
  130. style="width:180px"
  131. :options="timeSeriesOpt"
  132. :props="{emitPath:false}"
  133. :show-all-levels="false"
  134. placeholder="请选择"
  135. ></el-cascader>
  136. </el-form-item>
  137. <el-form-item label="空值处理">
  138. <el-select
  139. v-model="formData.nullValueWay"
  140. placeholder="请选择"
  141. style="width:180px"
  142. >
  143. <el-option
  144. v-for="item in nullWayOptions"
  145. :key="item.value"
  146. :label="item.label"
  147. :value="item.value"
  148. >
  149. </el-option>
  150. </el-select>
  151. </el-form-item>
  152. <el-form-item label="MAX、MIN空值处理" v-if="showMaxNullDeal">
  153. <el-select
  154. v-model="formData.maxNullWay"
  155. placeholder="请选择"
  156. >
  157. <el-option label="等于0" :value="1" />
  158. <el-option label="跳过空值" :value="2" />
  159. </el-select>
  160. </el-form-item>
  161. </template>
  162. <!-- 多指标求和/平均 -->
  163. <template v-if="computedType=='multipleEDB'">
  164. <el-form-item style="margin-right:30px">
  165. <el-radio-group :disabled="isEdit" v-model="subComputedType">
  166. <el-radio :label="81">求和</el-radio>
  167. <el-radio :label="82">求平均</el-radio>
  168. </el-radio-group>
  169. </el-form-item>
  170. <el-form-item label="生成指标时间序列">
  171. <el-cascader
  172. v-model="formData.timeSeriesVal"
  173. style="width:180px"
  174. :options="timeSeriesOpt"
  175. :props="{emitPath:false}"
  176. :show-all-levels="false"
  177. placeholder="请选择"
  178. ></el-cascader>
  179. </el-form-item>
  180. <el-form-item label="空值处理">
  181. <el-select
  182. v-model="formData.nullValueWay"
  183. placeholder="请选择"
  184. style="width:180px"
  185. >
  186. <el-option
  187. v-for="item in nullWayOptions"
  188. :key="item.value"
  189. :label="item.label"
  190. :value="item.value"
  191. >
  192. </el-option>
  193. </el-select>
  194. </el-form-item>
  195. </template>
  196. </div>
  197. </div>
  198. </el-form>
  199. <div class="filter-wrap" style="margin-top:20px">
  200. <p style="margin-bottom:10px">待选指标(选择指标数量不超过{{MAXAddNUM}})</p>
  201. <div>
  202. <el-cascader
  203. v-model="filter.classify"
  204. :options="classifyOpt"
  205. :props="classifyProps"
  206. clearable
  207. collapse-tags
  208. placeholder="指标分类"
  209. style="width: 240px"
  210. @change="handleFilter"
  211. />
  212. <el-select
  213. v-model="filter.frequency"
  214. placeholder="请选择频率"
  215. style="width: 140px"
  216. clearable
  217. multiple
  218. collapse-tags
  219. @change="handleFilter"
  220. :disabled="[5,61].includes(subComputedType)"
  221. >
  222. <el-option
  223. v-for="item in frequencyArr"
  224. :key="item"
  225. :label="item"
  226. :value="item"
  227. >
  228. </el-option>
  229. </el-select>
  230. <el-cascader
  231. v-model="filter.user"
  232. placeholder="创建人"
  233. :options="sysUserOpts"
  234. :props="sysUserProps"
  235. collapse-tags
  236. :show-all-levels="false"
  237. clearable
  238. filterable
  239. style="width:240px"
  240. @change="handleFilter"
  241. />
  242. <el-input
  243. placeholder="指标ID/指标名称"
  244. v-model="filter.keyword"
  245. style="width: 200px"
  246. @keydown.enter.native="handleFilter"
  247. >
  248. <i slot="prefix" class="el-input__icon el-icon-search"></i>
  249. </el-input>
  250. <el-checkbox
  251. label="列表全选"
  252. v-model="isCheckAll"
  253. :indeterminate="isCheckIndeterminate"
  254. style="margin-left:10px"
  255. @change="listCheckAllChange"
  256. :disabled="operationForm.view"
  257. />
  258. </div>
  259. </div>
  260. <div class="table-wrap">
  261. <div class="left-box">
  262. <el-table
  263. :data="list"
  264. border
  265. @sort-change="sortChange"
  266. @selection-change="selectionChange"
  267. ref="edbDataRef"
  268. @select="selectHandle"
  269. @select-all="selectAllHandle"
  270. height="500px"
  271. v-loading="listLoading"
  272. >
  273. <el-table-column type="selection" min-width="50" align="center"/>
  274. <el-table-column label="指标全称" show-overflow-tooltip align="center" prop="EdbName"/>
  275. <el-table-column label="最新日期" align="center" prop="EndDate" width="120px" />
  276. <el-table-column label="最新值" align="center" show-overflow-tooltip prop="EndValue" width="80px"/>
  277. <el-table-column label="创建人" align="center" show-overflow-tooltip prop="SysUserRealName" width="80px"/>
  278. <el-table-column label="频度" align="center" prop="Frequency" width="50px"/>
  279. <el-table-column label="单位" show-overflow-tooltip align="center" prop="Unit" width="50px"/>
  280. </el-table>
  281. <m-page
  282. style="margin-top:10px"
  283. class="table-page"
  284. v-show="total"
  285. :total="total"
  286. :pageSize="pageSize"
  287. :page_no="page"
  288. :pagercount="5"
  289. @handleCurrentChange="pageNumberChange"
  290. />
  291. </div>
  292. <div style="padding-top:200px;margin:0 30px">
  293. <el-button type="primary" :disabled="operationForm.view" :loading="addLoading" @click="getAddEdbListData">加入已选指标</el-button>
  294. </div>
  295. <div class="right-box">
  296. <el-table
  297. :data="selectList"
  298. border
  299. height="500px"
  300. >
  301. <el-table-column label="序号" width="50px" align="center" prop="No" v-if="computedType=='multipleEDB'">
  302. <template slot-scope="scope">
  303. {{scope.$index|getNoText}}
  304. </template>
  305. </el-table-column>
  306. <el-table-column label="指标全称" show-overflow-tooltip align="center" prop="EdbName"/>
  307. <el-table-column width="50px" align="center" v-if="!operationForm.view">
  308. <template slot="header" slot-scope="scope">
  309. <img @click="handleDelSelect('all')" style="width:15px;height:15px;cursor: pointer;" src="~@/assets/img/ai_m/delete.png" alt="">
  310. </template>
  311. <template slot-scope="scope">
  312. <img @click="handleDelSelect(scope)" style="width:15px;height:15px;cursor: pointer;" src="~@/assets/img/ai_m/delete.png" alt="">
  313. </template>
  314. </el-table-column>
  315. </el-table>
  316. </div>
  317. </div>
  318. <div style="text-align:center;margin:60px 0 40px 0">
  319. <el-button type="primary" plain @click="handleCloseSelf">取消</el-button>
  320. <el-button type="primary" @click="handleNextStep">下一步</el-button>
  321. </div>
  322. </div>
  323. <!-- 结果保存 -->
  324. <batchComputedSave
  325. :select_target="select_target"
  326. :computedType="computedType"
  327. :subComputedType="subComputedType"
  328. :selectList="selectList"
  329. :formData="formData"
  330. :isEdit="isEdit"
  331. :operationForm="operationForm"
  332. @close="showSave=false"
  333. @addCallBack="handleAddSuccess"
  334. v-else
  335. />
  336. </el-dialog>
  337. </template>
  338. <script>
  339. import mPage from '@/components/mPage.vue'
  340. import {computedBatchTypesV2} from './util'
  341. import { dataBaseInterface,departInterence } from '@/api/api.js';
  342. import batchComputedSave from './batchComputedSave.vue';
  343. import {generateSeriesArray} from './util'
  344. const tag_arr = generateSeriesArray();
  345. export default {
  346. components:{mPage,batchComputedSave},
  347. props:{
  348. isShow:{
  349. type: Boolean
  350. },
  351. operationForm: {
  352. type: Object,
  353. },
  354. type: {
  355. type: Number
  356. },
  357. },
  358. filters:{
  359. getNoText(e){
  360. return tag_arr[e]
  361. }
  362. },
  363. computed:{
  364. // 最大添加指标的上限
  365. MAXAddNUM(){
  366. let num=50
  367. if(this.computedType=='multipleEDB'){
  368. num=100
  369. }
  370. return num
  371. },
  372. /* max空值处理显示 当输入的公式包含MAX、MIN且空值处理为0时,输入公式失焦后出现右侧选项; */
  373. showMaxNullDeal() {
  374. let haveMaxOrMin = this.formData.formula.toUpperCase().includes('MAX') || this.formData.formula.toUpperCase().includes('MIN')
  375. return haveMaxOrMin && this.formData.nullValueWay===4
  376. },
  377. timeSeriesOpt(){
  378. let arr=[
  379. {
  380. label:`指标A`,
  381. value:'A'
  382. },
  383. {
  384. label:`指标B`,
  385. value:'B'
  386. }
  387. ]
  388. if(this.computedType=='multipleEDB'){
  389. arr=this.selectList.map((item,index)=>{
  390. return{
  391. label:`指标${tag_arr[index]}`,
  392. value:tag_arr[index]
  393. }
  394. })
  395. }
  396. return [
  397. {
  398. label:'指定指标时间序列',
  399. value:'0',
  400. children:arr
  401. },
  402. {
  403. label:'所有指标时间序列并集',
  404. value:'all',
  405. }
  406. ]
  407. },
  408. frequencyArr(){
  409. if(this.computedType==75){
  410. return ['周度','旬度','月度', '季度', '年度']
  411. }
  412. return ['日度', '周度','旬度','月度', '季度', '年度']
  413. }
  414. },
  415. watch: {
  416. isShow(n){
  417. if(!n){
  418. this.subComputedType=''
  419. this.select_target=''
  420. this.selectList=[]
  421. this.tableDataCheckedList=[]
  422. this.tableDataIds=[]
  423. this.isCheckAll=false
  424. this.isCheckIndeterminate=false
  425. this.checkAllStatus=false
  426. this.selectionReactCancel=false
  427. this.isEdit=false
  428. this.showSave=false
  429. this.computedType=computedBatchTypesV2[0].type
  430. this.filter={
  431. classify:'',
  432. frequency:'',
  433. user:'',
  434. keyword:''
  435. }
  436. this.formData={
  437. nNum:1,
  438. calendarType:'公历',
  439. frequency:'',
  440. valueType:'期末值',
  441. newValue:0,
  442. alphaValue:'',
  443. formula:'',
  444. timeSeriesVal:'A',
  445. nullValueWay:0,
  446. maxNullWay:1,
  447. }
  448. }else{
  449. // 编辑多指标求和\平均
  450. if([81,82].includes(this.type)){
  451. this.initData()
  452. }
  453. this.handleFilter()
  454. }
  455. }
  456. },
  457. data() {
  458. return {
  459. showSave:false,//显示结果保存模块
  460. computedBatchTypes:computedBatchTypesV2,//计算类型筛选项
  461. computedType:computedBatchTypesV2[0].type,//当前选中的计算类型
  462. subComputedType:'',//二级计算类型
  463. formData:{
  464. nNum:1,
  465. calendarType:'公历',
  466. frequency:'',
  467. valueType:'期末值',
  468. newValue:0,
  469. alphaValue:'',
  470. formula:'',
  471. timeSeriesVal:'A',
  472. nullValueWay:0,
  473. maxNullWay:1,
  474. },
  475. select_target:'',
  476. searchOptions:[],//指标列表
  477. search_have_more:false,
  478. search_page: 1,
  479. current_search: '',
  480. filter:{
  481. classify:'',
  482. frequency:'',
  483. user:'',
  484. keyword:''
  485. },
  486. // frequencyArr: ['日度', '周度','旬度','月度', '季度', '年度'],
  487. classifyOpt: [],
  488. classifyProps: {
  489. label: 'ClassifyName',
  490. value: 'ClassifyId',
  491. children: 'Children',
  492. multiple: true,
  493. emitPath:false
  494. },
  495. sysUserOpts:[],
  496. sysUserProps:{
  497. value: "AdminId",
  498. label: "RealName",
  499. children: "ChildrenList",
  500. multiple: true,
  501. emitPath:false
  502. },
  503. nullWayOptions: [
  504. { label: '查找前后35天最近值',value: 0 },
  505. { label: '不计算',value: 1 },
  506. { label: '前值填充',value: 2 },
  507. { label: '后值填充',value: 3 },
  508. { label: '等于0',value: 4 },
  509. ],
  510. isCheckAll:false,//是否全选
  511. isCheckIndeterminate:false,// 标志列表当前是全选状态还是不是全选状态和 isCheckAll不一样
  512. checkAllStatus:false,
  513. selectionReactCancel:false,// 是否不触发 selection的逻辑
  514. listLoading:false,
  515. page:1,
  516. pageSize:20,
  517. total:0,
  518. list:[],
  519. tableDataCheckedList:[],
  520. tableDataIds:[],
  521. selectList:[],//添加到右侧的数据
  522. addLoading:false,
  523. isEdit:false,//是否为多指标求和等编辑
  524. }
  525. },
  526. mounted() {
  527. this.getClassifyOpt()
  528. this.getEDBList()
  529. this.getSysUserList()
  530. },
  531. methods: {
  532. // 编辑时初始化数据
  533. initData(){
  534. this.isEdit=true
  535. const arr=this.operationForm.CalculateList||[]
  536. this.selectList=arr.map(item=>{
  537. return {
  538. ClassifyId:item.ClassifyId,
  539. EdbInfoId:item.FromEdbInfoId,
  540. EdbName:item.FromEdbName,
  541. Frequency:item.Frequency,
  542. Unit:item.Unit
  543. }
  544. })
  545. this.computedType='multipleEDB'
  546. this.subComputedType=this.type
  547. this.formData.timeSeriesVal=this.operationForm.EdbInfoDetail.Extra?JSON.parse(this.operationForm.EdbInfoDetail.Extra).DateTag:'A'
  548. this.formData.nullValueWay=this.operationForm.EdbInfoDetail.EmptyType||0
  549. },
  550. //批量计算成功回调
  551. handleAddSuccess(type,params){
  552. this.$emit('addCallBack',type,params)
  553. },
  554. // 跳转下一步
  555. handleNextStep(){
  556. if(this.selectList.length===0){
  557. this.$message.warning('请选择指标')
  558. return
  559. }
  560. if([8,12,13,35].includes(this.computedType) && !this.formData.nNum){
  561. this.$message.warning('请输入N数值')
  562. return
  563. }
  564. if([51].includes(this.computedType)&& !this.formData.frequency){
  565. this.$message.warning('请选择频度')
  566. return
  567. }
  568. if(['accumulate'].includes(this.computedType)&&this.subComputedType==62&& !this.formData.frequency){
  569. this.$message.warning('请选择频度')
  570. return
  571. }
  572. if(this.computedType==72&&!this.formData.alphaValue){
  573. this.$message.warning('请输入alpha值')
  574. return
  575. }
  576. if(this.computedType==72&&(Number(this.formData.alphaValue)<=0||Number(this.formData.alphaValue)>=1)){
  577. this.$message.warning('请输入>0,<1的数值的alpha值')
  578. return
  579. }
  580. if(this.computedType=='withEDB'&&!this.select_target){
  581. this.$message.warning('请选择指标B')
  582. return
  583. }
  584. if(['withNum','withEDB'].includes(this.computedType)&&!this.formData.formula){
  585. this.$message.warning('请输入公式')
  586. return
  587. }
  588. this.isCheckAll=false
  589. this.isCheckIndeterminate=false
  590. this.checkAllStatus=false
  591. this.tableDataCheckedList=[]
  592. this.tableDataIds=[]
  593. this.showSave=true
  594. },
  595. // 点击加入已选指标库
  596. async getAddEdbListData(){
  597. if(!(this.isCheckAll || this.isCheckIndeterminate ) || (!(this.list && this.list.length>0))){
  598. this.$message.warning('请选择指标')
  599. return
  600. }
  601. if(this.selectList.length>=this.MAXAddNUM){
  602. this.$message.warning('已达批量添加指标数量上限')
  603. return
  604. }
  605. const params={
  606. SysUserIds:this.filter.user?this.filter.user.join(','):'',
  607. ClassifyIds:this.filter.classify?this.filter.classify.join(','):'',
  608. Keyword:this.filter.keyword,
  609. Frequency:this.filter.frequency?this.filter.frequency.join(','):'',
  610. SelectAll:this.checkAllStatus,
  611. EdbInfoIds:this.tableDataCheckedList.join(',')
  612. }
  613. this.addLoading=true
  614. const res=await dataBaseInterface.getBatchFilterAddEdbList(params)
  615. this.addLoading=false
  616. if(res.Ret!=200) return
  617. // 加入到已选指标中 要去重
  618. const arr=res.Data.SearchItem||[]
  619. const temArr=this.mergeAndDistinct(this.selectList,arr)
  620. // 截取数组 防止数量溢出
  621. this.selectList=temArr.slice(0,this.MAXAddNUM)
  622. },
  623. mergeAndDistinct(arr1, arr2) {
  624. // 合并两个数组
  625. const mergedArray = arr1.concat(arr2);
  626. // 根据 EdbInfoId 字段进行去重
  627. const distinctArray = mergedArray.filter((item, index, self) => {
  628. return index === self.findIndex(t => t.EdbInfoId === item.EdbInfoId);
  629. });
  630. return distinctArray;
  631. },
  632. // 删除已选指标库指标
  633. handleDelSelect(e){
  634. if(this.operationForm.view) return
  635. if(e==='all'){
  636. this.selectList=[]
  637. return
  638. }
  639. const index=e.$index
  640. this.selectList.splice(index,1)
  641. },
  642. // 切换计算类型
  643. handleComputedTypeChange(){
  644. this.subComputedType=''
  645. this.select_target=''
  646. this.formData={
  647. nNum:1,
  648. calendarType:'公历',
  649. frequency:'',
  650. valueType:'期末值',
  651. newValue:0,
  652. alphaValue:'',
  653. formula:'',
  654. timeSeriesVal:'A',
  655. nullValueWay:0,
  656. maxNullWay:1,
  657. }
  658. this.filter={
  659. classify:'',
  660. frequency:'',
  661. user:'',
  662. keyword:''
  663. }
  664. this.selectList=[]
  665. if(this.computedType=='toMonthSeason'){
  666. this.subComputedType=5
  667. this.filter.frequency=['月度']
  668. }else if(this.computedType=='accumulate'){
  669. this.subComputedType=62
  670. }else if(this.computedType=='multipleEDB'){
  671. this.subComputedType=81
  672. }
  673. this.handleFilter()
  674. },
  675. handleComputedSubTypeChange(){
  676. if(this.subComputedType==61){
  677. this.filter.frequency=['季度']
  678. }
  679. if(this.subComputedType==5){
  680. this.filter.frequency=['月度']
  681. }
  682. this.selectList=[]
  683. this.handleFilter()
  684. },
  685. /* 指标列表 */
  686. getTarget(query) {
  687. this.search_page = 1;
  688. this.current_search = query;
  689. this.searchApi(this.current_search);
  690. },
  691. /* 聚焦获取当前检索 */
  692. inputFocusHandle(e) {
  693. this.search_page = 1;
  694. this.current_search = e.target.value;
  695. this.searchApi(this.current_search);
  696. },
  697. searchApi(query,page=1) {
  698. dataBaseInterface.targetSearchByPage({
  699. KeyWord:query,
  700. CurrentIndex: page,
  701. FilterSource: 1,
  702. Frequency: ''
  703. }).then(res => {
  704. if(res.Ret !== 200) return
  705. const { List,Paging } = res.Data;
  706. this.search_have_more = page < Paging.Pages;
  707. let arr = page === 1 ? List : this.searchOptions.concat(List);
  708. this.searchOptions = arr;
  709. })
  710. },
  711. searchLoad() {
  712. if(!this.search_have_more) return;
  713. this.searchApi(this.current_search,++this.search_page)
  714. },
  715. handleClose(){
  716. this.showSave=false
  717. this.$emit('close')
  718. },
  719. handleCloseSelf(){//仅仅关闭批量计算弹窗
  720. this.$emit('closeSelf')
  721. },
  722. filterNodes(arr) {
  723. arr.length &&
  724. arr.forEach((item) => {
  725. item.Children.length && this.filterNodes(item.Children);
  726. if (!item.Children.length) {
  727. delete item.Children;
  728. }
  729. });
  730. },
  731. // 获取指标分类
  732. async getClassifyOpt(){
  733. const res=await dataBaseInterface.menuListV3()
  734. if (res.Ret !== 200) return
  735. this.filterNodes(res.Data.AllNodes||[]);
  736. this.classifyOpt = res.Data.AllNodes || [];
  737. },
  738. // 获取所有系统用户
  739. async getSysUserList() {
  740. const res = await departInterence.getQuestionAdminList();
  741. if (res.Ret === 200) {
  742. this.sysUserOpts = res.Data.List||[];
  743. }
  744. },
  745. // 获取指标列表
  746. async getEDBList(type){
  747. this.listLoading=true
  748. const res=await dataBaseInterface.getBatchAddEdbSearchList({
  749. CurrentIndex:this.page,
  750. PageSize: this.pageSize,
  751. SysUserIds:this.filter.user?this.filter.user.join(','):'',
  752. ClassifyIds:this.filter.classify?this.filter.classify.join(','):'',
  753. Keyword:this.filter.keyword,
  754. Frequency:this.filter.frequency?this.filter.frequency.join(','):'',
  755. NotFrequency:this.computedType==75?'日度':''
  756. })
  757. this.listLoading=false
  758. if(res.Ret===200){
  759. this.list=res.Data.SearchItem||[]
  760. this.total=res.Data.Paging.Totals||0
  761. if(this.list.length>0){
  762. this.tableDataIds = this.list.map(it => it.EdbInfoId)
  763. }else{
  764. this.tableDataIds = []
  765. }
  766. if(type == 'adjustSelection'){
  767. this.adjustSelection()
  768. }else{
  769. this.isCheckAll=false
  770. this.checkAllStatus=false
  771. this.isCheckIndeterminate=false
  772. this.listCheckAllChange(this.isCheckAll)
  773. }
  774. }
  775. },
  776. pageNumberChange(e){
  777. this.page=e
  778. this.getEDBList('adjustSelection')
  779. },
  780. handleFilter(e){
  781. this.page=1
  782. this.list=[]
  783. this.getEDBList()
  784. },
  785. // 切换列表全选按钮状态
  786. listCheckAllChange(check){
  787. this.tableDataCheckedList=[]
  788. this.checkAllStatus=check
  789. if(check){
  790. // 全选
  791. this.$refs.edbDataRef && this.$refs.edbDataRef.clearSelection()
  792. this.$refs.edbDataRef && this.$refs.edbDataRef.toggleAllSelection()
  793. }else{
  794. //全不选
  795. this.$refs.edbDataRef && this.$refs.edbDataRef.clearSelection()
  796. }
  797. },
  798. selectionChange(val){
  799. if(this.selectionReactCancel) return
  800. // selectAllHandle的触发在selectionChange后面,将selectionChange的逻辑延迟一下
  801. setTimeout(()=>{
  802. // 去重
  803. let duplicateArr = Array.from(new Set(this.tableDataCheckedList))
  804. if((duplicateArr.length == this.total && (!this.checkAllStatus))
  805. || (duplicateArr.length == 0 && this.checkAllStatus)){
  806. this.isCheckAll = true
  807. this.isCheckIndeterminate=false
  808. }else if((duplicateArr.length == 0 && (!this.checkAllStatus))
  809. || (duplicateArr.length == this.total && this.checkAllStatus)){
  810. this.isCheckAll = false
  811. this.isCheckIndeterminate=false
  812. }else{
  813. this.isCheckAll = false
  814. this.isCheckIndeterminate=true
  815. }
  816. },1)
  817. },
  818. //用户手动勾选数据行的 Checkbox 时触发的事件
  819. selectHandle(selection, row){
  820. if(this.selectionReactCancel) return
  821. let check=false
  822. if(selection.some(it => it.EdbInfoId == row.EdbInfoId)){
  823. // 勾选
  824. // 勾选
  825. if(this.checkAllStatus){
  826. check=false
  827. }else{
  828. check=true
  829. }
  830. }else{
  831. // 取消勾选
  832. if(this.checkAllStatus){
  833. check=true
  834. }else{
  835. check=false
  836. }
  837. }
  838. if(check){
  839. this.tableDataCheckedList.push(row.EdbInfoId)
  840. }else{
  841. this.tableDataCheckedList=this.tableDataCheckedList.filter(it => it!=row.EdbInfoId)
  842. }
  843. },
  844. // 用户手动勾选全选 Checkbox 时触发的事件
  845. selectAllHandle(selection){
  846. if(this.selectionReactCancel) return
  847. let check = false; // 从tableDataCheckedList 添加还是删除
  848. if(selection && selection.length>0){
  849. // 全选
  850. if(this.checkAllStatus){
  851. check=false
  852. }else{
  853. check=true
  854. }
  855. }else{
  856. // 全不选
  857. if(this.checkAllStatus){
  858. check=true
  859. }else{
  860. check=false
  861. }
  862. }
  863. if(check){
  864. this.tableDataCheckedList = [...this.tableDataCheckedList,...this.tableDataIds]
  865. }else{
  866. this.tableDataCheckedList = this.tableDataCheckedList.filter(it => !this.tableDataIds.includes(it))
  867. }
  868. },
  869. adjustSelection(){
  870. this.selectionReactCancel=true
  871. if(!this.checkAllStatus){
  872. this.tableDataCheckedList.map(it =>{
  873. let row = this.list.find(da => da.EdbInfoId==it)
  874. if(row){
  875. setTimeout(()=>{
  876. this.$refs.edbDataRef.toggleRowSelection(row,true)
  877. },10)
  878. }
  879. })
  880. }else{
  881. this.$refs.edbDataRef.toggleAllSelection()
  882. this.tableDataCheckedList.map(it =>{
  883. let row = this.list.find(da => da.EdbInfoId==it)
  884. if(row){
  885. setTimeout(()=>{
  886. this.$refs.edbDataRef.toggleRowSelection(row,false)
  887. },50)
  888. }
  889. })
  890. }
  891. setTimeout(()=>{
  892. this.selectionReactCancel=false
  893. },50)
  894. },
  895. },
  896. }
  897. </script>
  898. <style lang="scss">
  899. .batch-computed-dialog{
  900. max-width: 1200px;
  901. width:90vw;
  902. overflow: hidden;
  903. }
  904. .batch-computed-wrap{
  905. .type-wrap{
  906. display: flex;
  907. }
  908. .table-wrap{
  909. margin-top: 20px;
  910. display: flex;
  911. justify-content: space-between;
  912. .left-box{
  913. width: 50%;
  914. }
  915. .right-box{
  916. flex: 1;
  917. flex-shrink: 0;
  918. }
  919. }
  920. }
  921. </style>