bloombergSource.vue 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574
  1. <template>
  2. <div class="bloomberg-source-wrap">
  3. <div class="page-header">
  4. <div class="header-select-box">
  5. <!-- 频度 -->
  6. <el-select v-model="selectOption.frequenceSelect"
  7. :placeholder="$t('Edb.InputHolderAll.input_fre')"
  8. @change="selectOptionChange('frequenceSelect')"
  9. clearable
  10. style="width:240px">
  11. <el-option v-for="i in frequencyList" :key="i.label"
  12. :label="i.label" :value="i.value"
  13. />
  14. </el-select>
  15. <!-- 指标ID/名称 -->
  16. <el-input
  17. v-model="selectOption.keywordSelect"
  18. prefix-icon="el-icon-search"
  19. :placeholder="$t('Edb.InputHolderAll.input_name_orid')"
  20. @input="selectOptionChange('keywordSelect')"
  21. clearable
  22. style="width:360px"></el-input>
  23. <!-- 列表全选 -->
  24. <el-checkbox
  25. :indeterminate="isIndeterminate"
  26. v-model="isCheckAll"
  27. @change="listCheckAllChange">{{ $t('YsDataPage.label_all_check') }}</el-checkbox>
  28. </div>
  29. <!-- 添加到指标库 -->
  30. <el-button v-permission="permissionBtn.dataSourcePermission.Bloomberg_add2edb" :disabled="total===0"
  31. type="primary" @click="handleCheckDialogShow(false,{})">{{ $t('YsDataPage.add_base_btn') }}</el-button>
  32. </div>
  33. <div class="page-table">
  34. <el-table :data="tableData" border
  35. ref="edbDataRef"
  36. @selection-change="selectionChange"
  37. @select="selectHandle"
  38. @select-all="selectAllHandle"
  39. @sort-change="handleSortChange"
  40. >
  41. <!-- 多选 -->
  42. <el-table-column
  43. align="center"
  44. type="selection"
  45. width="55">
  46. </el-table-column>
  47. <el-table-column
  48. v-for="column in tableColumns" :key="column.key"
  49. :prop="column.key"
  50. :label="column.label"
  51. :sortable="column.sortable?column.sortable:false"
  52. align="center">
  53. <template slot-scope="{row}">
  54. <span v-if="column.key==='Frequency'">
  55. {{ getFrequencyTrans(row.Frequency) }}
  56. </span>
  57. <span v-else>{{ row[column.key] }}</span>
  58. </template>
  59. </el-table-column>
  60. <el-table-column :label="$t('Table.column_operations')" align="center">
  61. <template slot-scope="{row}">
  62. <el-button type="text" @click="handleCheckDialogShow(true,row)"
  63. v-if="!row.EdbExist&&permissionBtn.isShowBtn('dataSourcePermission','Bloomberg_add2edb')">{{ $t('YsDataPage.add_base_btn') }}</el-button>
  64. </template>
  65. </el-table-column>
  66. </el-table>
  67. <el-pagination
  68. :current-page="currentPage"
  69. :page-size="pageSize"
  70. :total="total"
  71. @current-change="handleCurrentChange"
  72. />
  73. </div>
  74. <!-- 操作提示弹窗 -->
  75. <el-dialog
  76. :title="$t('BloombergPage.operation_prompt')"
  77. :visible.sync="isHintDialogShow"
  78. :close-on-click-modal="false"
  79. :modal-append-to-body="false"
  80. @close="isHintDialogShow=false"
  81. width="578px"
  82. v-dialogDrag
  83. center>
  84. <div class="hint-dialog-wrap">
  85. <p style="margin-bottom: 20px;">{{ hintText }}</p>
  86. <ul>
  87. <li v-for="(item,index) in afterAddList" :key="index" class="hint-item" @click="gotoEdbDetail(item)">
  88. {{ index+1 }}、{{ item.IndexName }}({{ item.IndexCode }})
  89. </li>
  90. </ul>
  91. </div>
  92. <div style="text-align: center;margin-bottom: 30px;">
  93. <el-button type="primary" @click="handleCloseDialog">{{$t('Dialog.known')}}</el-button>
  94. </div>
  95. </el-dialog>
  96. <!-- 添加到指标库弹窗 -->
  97. <el-dialog
  98. :title="$t('YsDataPage.add_base_btn')"
  99. :visible.sync="isAddDialogShow"
  100. :close-on-click-modal="false"
  101. :modal-append-to-body="false"
  102. @close="isAddDialogShow=false"
  103. width="900px"
  104. top="5vh"
  105. v-dialogDrag
  106. center>
  107. <div class="add-dialog-wrap">
  108. <el-form :model="addForm" ref="addFormRef">
  109. <div class="group" v-for="(item,index) in addForm.list" :key="index">
  110. <div class="item-header">
  111. <span @click="item.expand=!item.expand">
  112. {{ item.IndexCode }}
  113. <i :class="item.expand?'el-icon-arrow-up':'el-icon-arrow-down'"></i>
  114. </span>
  115. </div>
  116. <div class="form-wrap" v-show="item.expand">
  117. <el-form-item
  118. :prop="`list[${index}].IndexName`"
  119. :rules="[
  120. {required:true,message:$t('Edb.InputHolderAll.input_name'),trigger: 'blur'},
  121. {validator:edbNameCheck}
  122. ]">
  123. <el-input v-model="item.IndexName"></el-input>
  124. </el-form-item>
  125. <el-form-item :prop="`list[${index}].Unit`" :rules="{required:true,message:$t('Edb.InputHolderAll.input_unit'),trigger: 'blur'}">
  126. <el-input v-model="item.Unit"
  127. :placeholder="$t('Edb.InputHolderAll.input_unit')">
  128. </el-input>
  129. </el-form-item>
  130. <el-form-item :prop="`list[${index}].Frequency`" :rules="{required:true,message:$t('Edb.InputHolderAll.input_fre'),trigger: 'blur'}">
  131. <el-select v-model="item.Frequency" :placeholder="$t('Edb.InputHolderAll.input_fre')" >
  132. <el-option v-for="i in frequencyList" :key="i.label"
  133. :label="i.label" :value="i.value"
  134. />
  135. </el-select>
  136. </el-form-item>
  137. <el-form-item :prop="`list[${index}].ClassifyId`" :rules="{required:true,message:$t('Edb.InputHolderAll.input_menu'),trigger: 'blur'}">
  138. <cascader
  139. v-model="item.ClassifyId"
  140. :placeholder="$t('Edb.InputHolderAll.input_menu')"
  141. :options="edbCatalogList"
  142. :config="{ checkStrictly: true, emitPath: false }"
  143. />
  144. </el-form-item>
  145. </div>
  146. </div>
  147. </el-form>
  148. </div>
  149. <div style="text-align: center;margin-bottom: 30px;">
  150. <el-button @click="isAddDialogShow=false">{{$t('Dialog.cancel_btn')}}</el-button>
  151. <el-button type="primary" @click="handleSaveCheck" :loading="btnLoading">{{$t('Dialog.confirm_save_btn')}}</el-button>
  152. </div>
  153. </el-dialog>
  154. </div>
  155. </template>
  156. <script>
  157. import {bloombergInterface,dataBaseInterface} from '@/api/api.js'
  158. import { frequencySelectList } from '@/utils/defaultOptions';
  159. export default {
  160. data() {
  161. return {
  162. selectOption:{
  163. frequenceSelect:'',
  164. keywordSelect:'',
  165. sortParam:'',
  166. sortType:''
  167. },
  168. //frequencyList:['日度','周度','旬度','月度','季度','半年度','年度'],
  169. tableData:[],
  170. tableDataIds:[],
  171. currentPage:1,
  172. pageSize:10,
  173. total:2,
  174. isIndeterminate:false,
  175. isCheckAll:false,//与isIndeterminate一起表示列表全选的状态
  176. isSelectAll:false,//是否勾选了列表全选:为true时,selectList是剔除的指标,为false时selectList是已选择的指标
  177. selectList:[],//已选择/已剔除的指标id
  178. selectionReactCancel:false,//select-事件触发时,为true;该事件触发完成后,为false;
  179. isHintDialogShow:false,
  180. afterAddList:[],//已添加进指标库的指标
  181. beforeAddList:[],//需要添加进指标库的指标
  182. edbCatalogList:[],//指标库目录数据
  183. isAddDialogShow:false,
  184. addForm:{
  185. list:[
  186. /* {
  187. IndexName:'',
  188. Unit:'',
  189. Frequency:'',
  190. ClassifyId:''
  191. } */
  192. ]
  193. },
  194. checkedList:[],//重名的指标code
  195. btnLoading:false,
  196. };
  197. },
  198. computed:{
  199. tableColumns(){
  200. return [
  201. {key:'IndexCode',label:this.$t('Edb.Detail.e_id')}, //指标ID
  202. {key:'IndexName',label:this.$t('Edb.Detail.e_name')},//指标名称
  203. {key:'Frequency',label:this.$t('Edb.Detail.e_fre')},//频度
  204. {key:'StartDate',label:this.$t('DataBankPage.table_col_start_time'),sortable:true},//指标开始时间
  205. {key:'EndDate',label:this.$t('DataBankPage.table_col_newest_time'),sortable:true},//指标最新时间
  206. {key:'ModifyTime',label:this.$t('Edb.Detail.e_update_time'),sortable:true} //更新时间
  207. ]
  208. },
  209. frequencyList(){
  210. return frequencySelectList()
  211. },
  212. hintText(){
  213. let key = 0
  214. if(this.afterAddList.length) key++
  215. if(!this.beforeAddList.length) key++
  216. return {
  217. 0:'',
  218. 1:this.$t('YsDataPage.haved_all_msg'),//本次添加的指标均已在指标库中,请勿重复添加!
  219. 2:this.$t('YsDataPage.haved_some_msg'),//指标库中已存在以下指标,会自动过滤!
  220. }[key]
  221. }
  222. },
  223. methods: {
  224. getEdbCatalog() {
  225. dataBaseInterface.menuListV3().then(res => {
  226. if(res.Ret === 200) {
  227. this.edbCatalogList = res.Data.AllNodes || [];
  228. this.filterNodes(this.edbCatalogList);
  229. }
  230. })
  231. },
  232. // 递归改变目录结构
  233. filterNodes(arr) {
  234. arr.length && arr.forEach(item => {
  235. item.Children.length && this.filterNodes(item.Children)
  236. if(!item.Children.length) {
  237. item.Children=null
  238. }
  239. })
  240. },
  241. getTableData(type){
  242. const {frequenceSelect,keywordSelect,sortParam,sortType} = this.selectOption
  243. bloombergInterface.getTableList({
  244. PageSize:this.pageSize,
  245. CurrentIndex:this.currentPage,
  246. Frequency:frequenceSelect,
  247. Keywords:keywordSelect,
  248. SortField:sortParam,
  249. SortRule:sortType
  250. }).then(res=>{
  251. if(res.Ret!==200) return
  252. this.tableData = res.Data.List||[]
  253. this.total = res.Data.Paging.Totals
  254. if(this.tableData.length>0){
  255. this.tableDataIds = this.tableData.map(it => it.IndexCode)
  256. }else{
  257. this.tableDataIds = []
  258. }
  259. if(type==='optionChange'){
  260. //如果是表格筛选项改变导致重新请求数据
  261. //数据获取完成后,列表全选,表格全选
  262. this.selectList = []
  263. this.listCheckAllChange(true)
  264. }else{
  265. //若不是,数据获取完成后,查询列表全选的值
  266. //若当页有数据在selectList内,则勾选/剔除
  267. this.adjustSelection()
  268. }
  269. })
  270. },
  271. handleCurrentChange(page){
  272. this.currentPage = page
  273. this.getTableData()
  274. },
  275. handleSortChange({prop,order}){
  276. console.log('prop',prop,'order',order)
  277. const propMap={
  278. 'StartDate':1,
  279. 'EndDate':2,
  280. 'ModifyTime':3
  281. }
  282. this.selectOption.sortParam = order?propMap[prop]||'':''
  283. this.selectOption.sortType = order?order==='ascending'?1:2:''
  284. this.selectOptionChange()
  285. },
  286. //表格筛选项改变时触发
  287. selectOptionChange(type){
  288. //若是取消筛选,则不进行全选
  289. const optionValue = this.selectOption[type]
  290. if(!optionValue){
  291. this.isSelectAll = false
  292. this.selectList = []
  293. this.isIndeterminate = false
  294. this.isCheckAll = false
  295. }
  296. this.currentPage = 1
  297. this.getTableData(optionValue?'optionChange':'')
  298. },
  299. //列表全选改变
  300. listCheckAllChange(value){
  301. this.selectList = []
  302. this.isSelectAll = value
  303. this.$refs.edbDataRef && this.$refs.edbDataRef.clearSelection()
  304. if(value){
  305. this.$refs.edbDataRef && this.$refs.edbDataRef.toggleAllSelection()
  306. }
  307. },
  308. selectionChange(){
  309. if(this.selectionReactCancel) return
  310. // selectAllHandle的触发在selectionChange后面,将selectionChange的逻辑延迟一下
  311. setTimeout(()=>{
  312. // 去重
  313. let duplicateArr = Array.from(new Set(this.selectList))
  314. //isSelectAll为true时,selectList表示需要剔除的项
  315. //isSelectAll为false时,selectList表示需要勾选的项
  316. //全选
  317. if((duplicateArr.length == this.total && (!this.isSelectAll))|| (duplicateArr.length == 0 && this.isSelectAll)){
  318. this.isCheckAll = true
  319. this.isIndeterminate=false
  320. //全不选
  321. }else if((duplicateArr.length == 0 && (!this.isSelectAll))|| (duplicateArr.length == this.total && this.isSelectAll)){
  322. this.isCheckAll = false
  323. this.isIndeterminate=false
  324. //半选
  325. }else{
  326. this.isCheckAll = false
  327. this.isIndeterminate=true
  328. }
  329. },1)
  330. },
  331. selectHandle(selection,row){
  332. if(this.selectionReactCancel) return
  333. let check = false;
  334. if(selection.some(it => it.IndexCode == row.IndexCode)){
  335. // 勾选
  336. if(this.isSelectAll){
  337. check=false
  338. }else{
  339. check=true
  340. }
  341. }else{
  342. // 取消勾选
  343. if(this.isSelectAll){
  344. check=true
  345. }else{
  346. check=false
  347. }
  348. }
  349. if(check){
  350. this.selectList.push(row.IndexCode)
  351. }else{
  352. this.selectList=this.selectList.filter(it => it!=row.IndexCode)
  353. }
  354. },
  355. selectAllHandle(selection){
  356. if(this.selectionReactCancel) return
  357. let check = false;
  358. if(selection && selection.length>0){
  359. // 全选
  360. if(this.isSelectAll){
  361. check=false
  362. }else{
  363. check=true
  364. }
  365. }else{
  366. // 全不选
  367. if(this.isSelectAll){
  368. check=true
  369. }else{
  370. check=false
  371. }
  372. }
  373. if(check){
  374. this.selectList = [...this.selectList,...this.tableDataIds]
  375. }else{
  376. this.selectList = this.selectList.filter(it => !this.tableDataIds.includes(it))
  377. }
  378. },
  379. adjustSelection(){
  380. this.selectionReactCancel=true
  381. if(!this.isSelectAll){
  382. this.selectList.map(it =>{
  383. let row = this.tableData.find(da => da.IndexCode==it)
  384. if(row){
  385. setTimeout(()=>{
  386. this.$refs.edbDataRef.toggleRowSelection(row,true)
  387. },10)
  388. }
  389. })
  390. }else{
  391. this.$refs.edbDataRef && this.$refs.edbDataRef.clearSelection()
  392. this.$refs.edbDataRef &&this.$refs.edbDataRef.toggleAllSelection()
  393. this.selectList.map(it =>{
  394. let row = this.tableData.find(da => da.IndexCode==it)
  395. if(row){
  396. setTimeout(()=>{
  397. this.$refs.edbDataRef.toggleRowSelection(row,false)
  398. },50)
  399. }
  400. })
  401. }
  402. setTimeout(()=>{
  403. this.selectionReactCancel=false
  404. },50)
  405. },
  406. handleCheckDialogShow(isSingle=false,data={}){
  407. //check
  408. const {frequenceSelect,keywordSelect,sortParam,sortType} = this.selectOption
  409. bloombergInterface.edbAddCheck({
  410. Frequency:frequenceSelect,
  411. Keywords:keywordSelect,
  412. SortField:sortParam||0,
  413. SortRule:sortType||0,
  414. ListAll:isSingle?false:this.isSelectAll,
  415. IndexCodes:isSingle?[data.IndexCode]:this.selectList
  416. }).then(res=>{
  417. if(res.Ret!==200) return
  418. const edbList = res.Data||[]
  419. this.afterAddList = edbList.filter(i=>i.EdbExist===1)
  420. this.beforeAddList = edbList.filter(i=>i.EdbExist===0).map(i=>{return{...i,expand:true}})
  421. this.afterAddList.length&&(this.isHintDialogShow=true)
  422. !this.afterAddList.length&&this.handleCloseDialog()
  423. })
  424. },
  425. //操作提示弹窗关闭
  426. handleCloseDialog(){
  427. this.isHintDialogShow = false
  428. //判断是否要展示下一个弹窗
  429. if(this.beforeAddList.length){
  430. this.addForm.list = this.beforeAddList
  431. this.isAddDialogShow = true
  432. }
  433. },
  434. //重名校验
  435. edbNameCheck(rule,value,callback){
  436. if(this.checkedList.includes(value)){
  437. callback(this.$t('BloombergPage.add_edb_check_hint2'))
  438. }else{
  439. callback()
  440. }
  441. },
  442. //保存前校验
  443. async handleSaveCheck(){
  444. //表单校验
  445. try{
  446. await this.$refs.addFormRef.validate()
  447. }catch(e){
  448. return
  449. }
  450. //重名校验
  451. this.checkedList = []
  452. bloombergInterface.edbNameCheck(
  453. this.addForm.list.map(i=>{return{EdbCode:i.IndexCode,EdbName:i.IndexName}})
  454. ).then(res=>{
  455. if(res.Ret!==200) return
  456. this.checkedList = res.Data.filter(i=>i.Exist).map(i=>i.EdbName)
  457. if(this.checkedList.length){
  458. this.$refs.addFormRef.validate()
  459. }else{
  460. this.saveEdb()
  461. }
  462. })
  463. },
  464. //批量新增
  465. saveEdb(){
  466. this.btnLoading = true
  467. bloombergInterface.edbBatchAdd(
  468. this.addForm.list.map(i=>{
  469. return {
  470. EdbCode:i.IndexCode,
  471. EdbName:i.IndexName,
  472. Frequency:i.Frequency,
  473. Unit:i.Unit,
  474. ClassifyId:i.ClassifyId
  475. }
  476. })
  477. ).then(res=>{
  478. this.btnLoading = false
  479. if(res.Ret!==200) return
  480. this.$message.success(this.$t('MsgPrompt.add_msg2'))
  481. this.isAddDialogShow = false
  482. this.handleCurrentChange(1)
  483. })
  484. },
  485. //跳转至指标详情
  486. gotoEdbDetail(data){
  487. const {EdbClassifyId,EdbInfoId,EdbUniqueCode} = data||{}
  488. const href = this.$router.resolve({
  489. path:'/database',
  490. query:{
  491. code:EdbUniqueCode,
  492. id:EdbInfoId,
  493. classifyId:EdbClassifyId,
  494. }
  495. }).href
  496. window.open(href,"_blank")
  497. },
  498. },
  499. mounted(){
  500. this.getTableData()
  501. this.getEdbCatalog()
  502. },
  503. };
  504. </script>
  505. <style lang="scss">
  506. .bloomberg-source-wrap{
  507. .add-dialog-wrap{
  508. .el-form{
  509. .group{
  510. .item-header{
  511. margin-bottom: 20px;
  512. }
  513. .form-wrap{
  514. display: flex;
  515. gap:10px;
  516. .el-input{
  517. width: 100%;
  518. }
  519. }
  520. }
  521. }
  522. }
  523. }
  524. </style>
  525. <style scoped lang="scss">
  526. .bloomberg-source-wrap{
  527. min-height: calc(100vh - 120px);
  528. display: flex;
  529. flex-direction: column;
  530. background-color: #fff;
  531. border:1px solid #C8CDD9;
  532. border-radius: 4px;
  533. padding:20px;
  534. box-sizing: border-box;
  535. .page-header{
  536. display: flex;
  537. justify-content: space-between;
  538. .header-select-box{
  539. display: flex;
  540. gap:20px;
  541. align-items: center;
  542. }
  543. }
  544. .page-table{
  545. .el-table{
  546. margin:20px 0;
  547. .el-button--text{
  548. padding:0;
  549. }
  550. }
  551. .el-pagination{
  552. text-align: right;
  553. }
  554. }
  555. .hint-dialog-wrap{
  556. padding-bottom:30px;
  557. .hint-item{
  558. cursor: pointer;
  559. margin-bottom: 10px;
  560. &:hover{
  561. color:#409EFF;
  562. text-decoration: underline;
  563. }
  564. }
  565. }
  566. }
  567. </style>