addAnalysisSheet.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523
  1. <template>
  2. <div class="addSheet-wrap">
  3. <createTargetForm
  4. v-if="$route.path==='/createTaregtBySheet'"
  5. ref="createTargetRef"
  6. @save="handleCreateTarget"
  7. />
  8. <div class="main">
  9. <div class="left-section">
  10. <el-tabs
  11. v-model="selectIndex"
  12. type="card"
  13. >
  14. <el-tab-pane
  15. :key="item.name"
  16. v-for="(item,index) in uploadSheetsList"
  17. :label="item.name"
  18. :name="String(index)"
  19. />
  20. </el-tabs>
  21. <div class="sheet-wrapper">
  22. <Sheet ref="sheetRef" :option="sheetConfig" v-if="sheetConfig.data" :limit="{disabled:false}"/>
  23. <dataLoading :loading="isLoading"/>
  24. </div>
  25. </div>
  26. <!-- 指标列表 -->
  27. <rightSection
  28. ref="edbWrapRef"
  29. v-if="$route.path==='/createTaregtBySheet'"
  30. :list="edbList"
  31. :sheetDetailInfo="this.sheetDetailInfo"
  32. @choose="chooseEdbHandle"
  33. />
  34. </div>
  35. <!-- 上传文件 -->
  36. <bottomSection
  37. ref="bottomSecRef"
  38. v-if="$route.path==='/addAnalysisSheet'"
  39. :sheetList="sheetConfig.data||[]"
  40. :classifyArr="classifyArr"
  41. @save="handleAddSheet"
  42. />
  43. </div>
  44. </template>
  45. <script>
  46. import * as sheetInterface from '@/api/modules/sheetApi.js';
  47. import Sheet from '../components/SheetExcel.vue';
  48. import { getSheetImage } from '../common/option';
  49. import bottomSection from './components/bottomSection.vue'
  50. import rightSection from './components/rightSection.vue';
  51. import createTargetForm from './components/createTargetForm.vue';
  52. export default {
  53. components: { Sheet,bottomSection,rightSection,createTargetForm },
  54. computed: {
  55. files() {
  56. return this.$store.state.sheet.files;
  57. },
  58. sheetConfig() {
  59. return this.$route.path==='/createTaregtBySheet'
  60. ? {
  61. showsheetbar: true,
  62. // allowCopy:false,
  63. // allowEdit:false,
  64. // allowUpdate:false,
  65. // enableAddRow:false,
  66. data: null,
  67. hook: {
  68. //选区时
  69. rangeSelect: (sheet,range) => {
  70. if(this.$refs.createTargetRef && this.$refs.createTargetRef.selectArea && !this.$refs.createTargetRef.isLockUpdate) this.getRangeCell()
  71. },
  72. }
  73. }
  74. : {
  75. showsheetbar: true,
  76. data: null
  77. }
  78. },
  79. },
  80. data() {
  81. return {
  82. isLoading: false,
  83. uploadSheetsList: [],
  84. pageSize: 2000,
  85. currentPage: 0,
  86. maxPage:0,
  87. selectIndex: '0',
  88. classifyArr: [],
  89. sheetDetailInfo: {},
  90. sheetDataPage: 2,
  91. sheetAllcellData:[],//全部单元格数据 分页push
  92. dataToalPage: 0,
  93. edbList: [],//生成的指标列表
  94. }
  95. },
  96. methods: {
  97. backHandle(scence=null) {
  98. if(scence === 'into-detail') {
  99. const { ExcelInfoId, UniqueCode } = this.sheetDetailInfo;
  100. this.$router.replace({
  101. path: '/sheetAnalysisList',
  102. query: {
  103. code: UniqueCode,
  104. id: ExcelInfoId
  105. }
  106. })
  107. }else {
  108. this.$router.go(-1);
  109. }
  110. },
  111. /* 切换表格 */
  112. handleSwitchSheet() {
  113. this.sheetConfig.data = null;
  114. this.handelTranslateData();
  115. },
  116. /* 获取分类 */
  117. getClassify() {
  118. sheetInterface.sheetAnalysisInterface.getSheetTreeList({IsShare:false}).then((res) => {
  119. if(res.Ret !==200) return
  120. this.classifyArr = this.formatTreeData(_.cloneDeep(res.Data.AllNodes || []))
  121. })
  122. },
  123. formatTreeData(tree){
  124. function dfs(node){
  125. if (Array.isArray(node.Children)) {
  126. node.Children = node.Children.filter(dfs)
  127. if(node.Children.length===0) delete node.Children
  128. }
  129. if(node.ExcelInfoId) {
  130. return null
  131. }
  132. return node
  133. }
  134. return tree.filter(dfs)
  135. },
  136. loadDataSync() {
  137. let len = this.sheetConfig.data.length;
  138. for(let i =0;i<len;i++) {
  139. if(this.allSheetData[i].celldata[this.currentPage*this.pageSize]) {
  140. let concatData = this.allSheetData[i].celldata.slice(this.currentPage*this.pageSize, (this.currentPage+1)*this.pageSize);
  141. // this.sheetConfig.data[i].celldata = this.sheetConfig.data[i].celldata.concat(concatData)
  142. // this.$refs.sheetRef.init()
  143. }
  144. continue
  145. }
  146. if (this.currentPage<this.maxPage) {
  147. this.currentPage++
  148. requestAnimationFrame(this.loadDataSync());
  149. }
  150. },
  151. /* 分割数据 */
  152. splitSheetData(sheets) {
  153. // this.allSheetData = sheets.map(_ => ({
  154. // index: _.index, //工作表索引
  155. // order: _.order, //工作表的下标
  156. // name: _.name,
  157. // calcChain: _.calcChain,
  158. // celldata: _.celldata,
  159. // config: _.config
  160. // }));
  161. // this.maxPage = Math.max(...sheets.map(_ => Math.ceil(_.celldata.length / this.pageSize)))
  162. console.log(sheets)
  163. this.sheetConfig.data = sheets.map(_ => ({
  164. index: _.index, //工作表索引
  165. order: Number(_.order), //工作表的下标
  166. name: _.name,
  167. calcChain: _.calcChain,
  168. config: _.config,
  169. celldata: _.celldata,
  170. }));
  171. //辣鸡插件连更新数据的api都没有
  172. // this.loadDataSync();
  173. this.isLoading = false;
  174. },
  175. handelTranslateData() {
  176. this.isLoading = true;
  177. LuckyExcel.transformExcelToLucky(this.uploadSheetsList[Number(this.selectIndex)], (exportJson, luckysheetfile) =>{
  178. if(exportJson.sheets==null || exportJson.sheets.length==0){
  179. this.$message.warning(this.$t('OnlineExcelPage.parsing_file_error_msg') )
  180. return;
  181. }
  182. this.splitSheetData(exportJson.sheets);
  183. });
  184. },
  185. /* 移除表格 */
  186. handleRemoveSheet(index) {
  187. if(this.uploadSheetsList.length === 1) {
  188. this.backHandle()
  189. return
  190. }
  191. this.uploadSheetsList.splice(Number(index),1)
  192. if(this.selectIndex === index) {
  193. this.selectIndex = '0';
  194. this.handleSwitchSheet()
  195. }
  196. },
  197. /* 保存表格 */
  198. handleAddSheet: _.debounce(async function() {
  199. luckysheet.exitEditMode()
  200. let data = luckysheet.getAllSheets().filter(_ => this.$refs.bottomSecRef.sheetChecked.includes(_.name));
  201. console.log(data)
  202. this.loading = this.$loading({
  203. target:'.addSheet-wrap',
  204. lock: true,
  205. text: this.$t('OnlineExcelPage.saving_loading_text') ,
  206. spinner: 'el-icon-loading',
  207. background: 'rgba(255, 255, 255, 0.6)'
  208. });
  209. let img = getSheetImage(data[0]);
  210. const form = new FormData();
  211. form.append('Image', img);
  212. const { Data } = await sheetInterface.uploadImg(form)
  213. data.luckysheet_select_save = [];
  214. const res = await sheetInterface.sheetAnalysisInterface.excelSheetAdd({
  215. ExcelName: this.uploadSheetsList[0].name,
  216. ExcelClassifyId: this.$refs.bottomSecRef.select_classify,
  217. ExcelImage: Data.ResourceUrl,
  218. Content: JSON.stringify(data)
  219. })
  220. this.loading.close()
  221. if(res.Ret !== 200) return
  222. this.$message.success(res.Msg);
  223. // const { ExcelInfoId, UniqueCode } = res.Data;
  224. this.$router.replace({
  225. path: '/sheetAnalysisList',
  226. // query: {
  227. // code: UniqueCode,
  228. // id: ExcelInfoId
  229. // }
  230. })
  231. },300),
  232. /* 获取表格详情 */
  233. getDetailHandle() {
  234. this.isLoading = true;
  235. sheetInterface.sheetAnalysisInterface.getExcelDetail({
  236. UniqueCode: this.$route.query.code,
  237. }).then((res) => {
  238. if (res.Ret !== 200) return;
  239. this.sheetDetailInfo = res.Data.ExcelInfo;
  240. this.dataToalPage = Math.max(...res.Data.SheetList.map(_ => _.PageNum));
  241. this.sheetAllcellData = res.Data.SheetList.map(_ => _.Data ? JSON.parse(_.Data.Data): []);
  242. this.getCellData(res.Data.SheetList)
  243. });
  244. },
  245. //分页获取表格数据
  246. async getCellData(sheets) {
  247. let res = await sheetInterface.sheetAnalysisInterface.getExcelDataByPage({
  248. UniqueCode: this.$route.query.code,
  249. Page: this.sheetDataPage
  250. })
  251. if(res.Ret !== 200) return
  252. for(let i = 0;i<this.sheetAllcellData.length;i++) {
  253. if(res.Data[i].Data) {
  254. this.sheetAllcellData[i] = [...this.sheetAllcellData[i],...JSON.parse(res.Data[i].Data.Data)]
  255. }
  256. continue
  257. }
  258. //数据继续加载或渲染表格
  259. if(this.sheetDataPage < this.dataToalPage) {
  260. this.sheetDataPage++;
  261. this.getCellData(sheets)
  262. }else {
  263. this.sheetConfig.data = sheets.map((_,index) => ({
  264. index: _.Index, //工作表id
  265. order: _.Sort, //工作表的下标
  266. name: _.SheetName,
  267. calcChain: JSON.parse(_.CalcChain),
  268. config: JSON.parse(_.Config),
  269. celldata: this.sheetAllcellData[index],
  270. }))
  271. this.isLoading = false;
  272. //this.getTargetList()
  273. }
  274. },
  275. /* 获取选取对应单元格数组和拼接选取的公式
  276. Sheet1!$A$1:$A$25
  277. Sheet1!$E:$E
  278. */
  279. getRangeCell: _.debounce(function() {
  280. let sheet = luckysheet.getSheet();
  281. let rangeArr = luckysheet.getRangeAxis();
  282. if(rangeArr.length > 1) return this.$message.warning(this.$t('OnlineExcelPage.select_one_region_msg') );
  283. //检查选取是否满足同行/同列
  284. if(!this.checkRangeVaild(rangeArr[0])) return this.$message.warning(this.$t('OnlineExcelPage.only_allowed_select_msg') );
  285. let rangeCells = luckysheet.getRangeValue().flat().map(_=>_?_.m:'');
  286. let format = `${sheet.name}!${this.formatStr(rangeArr[0])}`;
  287. console.log(format)
  288. this.$refs.createTargetRef.setFormula(format,rangeCells)
  289. },300),
  290. //检验同行同列
  291. checkRangeVaild(range) {
  292. let reg = /^([A-Z]+)(\d+)$/;
  293. let arr = range.split(':').map(_ => [_.match(reg)[1],_.match(reg)[2]]);
  294. if(arr.length ===1) return true;
  295. if(arr[0][0] === arr[1][0] || arr[0][1] === arr[1][1]) {
  296. return true
  297. }else {
  298. return false
  299. }
  300. },
  301. formatStr(inputRange) {
  302. // 将字母和数字前面都拼接"$"
  303. const parts = inputRange.split(':');
  304. if (parts.length === 2) {
  305. const start = parts[0].replace(/(^[A-Z]+)(\d+)$/, '$$$1$$$2');
  306. const end = parts[1].replace(/(^[A-Z]+)(\d+)$/, '$$$1$$$2');
  307. return start + ':' + end;
  308. }
  309. return inputRange;
  310. },
  311. /* 生成指标 */
  312. async handleCreateTarget() {
  313. console.log(this.$refs.createTargetRef.formData)
  314. this.loading = this.$loading({
  315. target:'.addSheet-wrap',
  316. lock: true,
  317. text: this.$t('OnlineExcelPage.saving_loading_text') ,
  318. spinner: 'el-icon-loading',
  319. background: 'rgba(255, 255, 255, 0.6)'
  320. });
  321. let data = luckysheet.getAllSheets();
  322. data.luckysheet_select_save = [];
  323. const { ExcelInfoId, ExcelName, ExcelClassifyId } = this.sheetDetailInfo;
  324. await sheetInterface.sheetAnalysisInterface.sheetEdit({
  325. ExcelInfoId,
  326. ExcelName,
  327. ExcelClassifyId,
  328. // ExcelImage: Data.ResourceUrl,
  329. Content: JSON.stringify(data)
  330. });
  331. const { edbInfoId,
  332. dateSeries,
  333. valueSeries,
  334. dateArr,
  335. valueArr,
  336. edbName,
  337. classify,
  338. frequency,
  339. unit } = this.$refs.createTargetRef.formData;
  340. let params = {
  341. EdbName: edbName,
  342. ExcelInfoId: ExcelInfoId,
  343. ClassifyId: classify,
  344. Frequency: frequency,
  345. Unit: unit,
  346. DateSequenceVal: dateArr,
  347. DataSequenceVal: valueArr,
  348. DateSequenceStr: dateSeries,
  349. DataSequenceStr: valueSeries
  350. }
  351. const res = edbInfoId
  352. ? await sheetInterface.sheetAnalysisInterface.edbEditBysheet({...params,EdbInfoId: edbInfoId})
  353. : await sheetInterface.sheetAnalysisInterface.edbAddBysheet(params)
  354. this.loading.close();
  355. if(res.Ret !== 200) return
  356. this.$message.success(res.Msg)
  357. if(!edbInfoId) this.$refs.createTargetRef.initData();
  358. //this.getTargetList()
  359. if(this.$refs.edbWrapRef){
  360. this.$refs.edbWrapRef.getTargetList('init')
  361. }
  362. },
  363. /* 获取生成指标列表 */
  364. async getTargetList() {
  365. const res = await sheetInterface.sheetAnalysisInterface.edbListBySheet({
  366. ExcelInfoId: this.sheetDetailInfo.ExcelInfoId
  367. })
  368. if(res.Ret !== 200) return
  369. this.edbList = res.Data || [];
  370. },
  371. /* 选择指标列表更新表单信息和区域选中 */
  372. chooseEdbHandle() {
  373. if(this.$refs.edbWrapRef.selectEdb.EdbInfoId) {
  374. this.$refs.createTargetRef.initData(this.$refs.edbWrapRef.selectEdb)
  375. } else {
  376. this.$refs.createTargetRef.initData()
  377. }
  378. }
  379. },
  380. mounted() {
  381. this.getClassify();
  382. //上传文件的解析渲染
  383. if(this.files) {
  384. this.uploadSheetsList = Object.values(this.files);
  385. this.handelTranslateData()
  386. this.$store.commit('sheet/SET_UPLOADFIlES',null)
  387. }
  388. //详情的渲染
  389. if(this.$route.query.code) this.getDetailHandle();
  390. }
  391. }
  392. </script>
  393. <style scoped lang="scss">
  394. *{ box-sizing: border-box; }
  395. .addSheet-wrap {
  396. min-height: calc(100vh - 120px);
  397. .wrap-top {
  398. display: flex;
  399. justify-content: space-between;
  400. align-items: center;
  401. margin-bottom: 20px;
  402. padding: 20px;
  403. background: #fff;
  404. border: 1px solid #ececec;
  405. border-radius: 4px;
  406. box-shadow: 0 3px 6px rgba(0, 0, 0, 0.05);
  407. display: flex;
  408. z-index: 1;
  409. .form-ul {
  410. flex: 1;
  411. display: flex;
  412. li {
  413. margin-right: 30px;
  414. }
  415. }
  416. }
  417. .main {
  418. display: flex;
  419. position: relative;
  420. min-height: 700px;
  421. .left-section {
  422. flex:1;
  423. .sheet-wrapper {
  424. position: relative;
  425. background: #fff;
  426. min-height: 700px;
  427. }
  428. }
  429. &.full-height {
  430. min-height: calc(100vh - 120px);
  431. }
  432. }
  433. }
  434. </style>
  435. <style lang="scss">
  436. .addSheet-wrap {
  437. .el-tabs__nav {
  438. background: #fff;
  439. }
  440. .el-tabs__header {
  441. margin-bottom: 0;
  442. // border-bottom: none;
  443. }
  444. }
  445. </style>