edit.vue 34 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093
  1. <template>
  2. <basic-container>
  3. <div class="box-dashed flex">
  4. <div class="retain-box">
  5. <el-checkbox v-model="isRetain"></el-checkbox>
  6. <span>保留</span>
  7. <el-input-number
  8. v-model="retainNum"
  9. :step="1"
  10. :disabled="!isRetain"
  11. size="small"
  12. ></el-input-number>
  13. <span class="retain">位</span>
  14. </div>
  15. <div>
  16. <el-menu
  17. :default-active="activeIndex"
  18. class="el-menu-demo"
  19. mode="horizontal"
  20. @select="handleSelect"
  21. >
  22. <el-submenu v-for="(value,key,index) in formulaList" :key="key" :index="key">
  23. <template slot="title">
  24. <span>{{key}}</span>
  25. </template>
  26. <el-menu-item v-for="(item2,index2) in value" :key="index2" :index="(index+1) +'-' + (index2+1)">{{item2.name}}</el-menu-item>
  27. </el-submenu>
  28. </el-menu>
  29. </div>
  30. </div>
  31. <div class="box-dashed">
  32. <div class="mg-b-20">函数公式</div>
  33. <div class="edit-text">
  34. <span>
  35. <formula-item
  36. v-for="(item,index) in resultFormula" :key="index"
  37. :item="item" @click="obj => equationClick(obj,index,'resultFormula')"
  38. >
  39. </formula-item>
  40. </span>
  41. <span>=</span>
  42. <span>
  43. <formula-item
  44. v-for="(item,index) in processFormula" :key="index"
  45. :item="item" @click="obj => equationClick(obj,index,'processFormula')"
  46. >
  47. </formula-item>
  48. </span>
  49. </div>
  50. <div class="flex jc-sb">
  51. <div></div>
  52. <div><el-button type="info" size="small" @click="operationEdit">重置函数</el-button></div>
  53. </div>
  54. </div>
  55. <div v-show="operationVisible" class="operation-box">
  56. <div>选择参数设置</div>
  57. <div>
  58. <el-row :gutter="20">
  59. <el-col :span="8">
  60. <el-card shadow="never">
  61. <el-scrollbar style="height: 460px">
  62. <!-- <el-tree
  63. class="filter-tree"
  64. lazy
  65. :load="loadNode"
  66. @node-click="getNodeDetail"
  67. :props="defaultProps"
  68. :expand-on-click-node="false"
  69. highlight-current
  70. node-key="id"
  71. ref="tree"
  72. >
  73. </el-tree> -->
  74. <el-tree
  75. class="filter-tree"
  76. :data="treeData"
  77. :default-expanded-keys="defaultExpanded"
  78. @node-click="getNodeDetail"
  79. :props="defaultProps"
  80. :expand-on-click-node="false"
  81. highlight-current
  82. node-key="id"
  83. ref="tree"
  84. >
  85. </el-tree>
  86. </el-scrollbar>
  87. </el-card>
  88. </el-col>
  89. <el-col :span="16">
  90. <el-select v-model="eleTableId" @change="getTableEle" placeholder="请选择元素表" style="width:300px">
  91. <el-option v-for="item in eleTableList" :key="item.id" :label="item.tableName" :value="item.id"></el-option>
  92. </el-select>
  93. <div class="mg-t-10 no-mb-col">
  94. <el-scrollbar style="height: 210px">
  95. <el-row>
  96. <el-col :span="6" v-for="item in eleList" :key="item.id">
  97. <div class="ele-box">
  98. <span>{{item.eName}}</span>
  99. <el-checkbox v-model="item.checked" @change="value => eleChang(value,item)"></el-checkbox>
  100. </div>
  101. </el-col>
  102. </el-row>
  103. </el-scrollbar>
  104. </div>
  105. <div class="flex jc-sb">
  106. <!-- <div>定位数据位置:</div> -->
  107. <div class="icon-box">
  108. <el-link :underline="false" icon="el-icon-delete" type="danger" @click="removeSelect"></el-link>
  109. <el-link :underline="false" type="primary" @click="addOperator('+')" icon="el-icon-circle-plus-outline"></el-link>
  110. <el-link :underline="false" type="primary" @click="addOperator('-')" icon="el-icon-remove-outline"></el-link>
  111. <el-link :underline="false" type="primary" @click="addOperator('*')" icon="el-icon-circle-close"></el-link>
  112. <el-link :underline="false" type="primary" @click="addOperator('%')">÷</el-link>
  113. </div>
  114. <div>
  115. <!-- <el-link :underline="false" type="primary" class="mg-r-20" @click="eleAddFormula">元素添加到公式</el-link> -->
  116. <el-link :underline="false" type="primary" class="mg-r-10" @click="addBrackets('(',false)">(</el-link>
  117. <el-link :underline="false" type="primary" class="mg-r-10" @click="addBrackets(')',true)">)</el-link>
  118. <!-- <el-link :underline="false" type="primary" class="mg-r-10" @click="addBrackets('[',false)">【</el-link>
  119. <el-link :underline="false" type="primary" class="mg-r-10" @click="addBrackets(']',true)">】</el-link> -->
  120. <el-link :underline="false" type="primary" @click="addText">输入值</el-link>
  121. </div>
  122. </div>
  123. <div class="border-grey sele-ele-box">
  124. <div>
  125. <formula-item
  126. v-for="(item,index) in selectEleFormula" :key="index"
  127. :item="item" @click="obj => eleFormulaClick(obj,index)"
  128. >
  129. </formula-item>
  130. </div>
  131. </div>
  132. </el-col>
  133. </el-row>
  134. <div class="text-align-c">
  135. <el-button size="small" @click="operationHandle" type="primary">保存</el-button>
  136. <el-button size="small" @click="operationVisible = false">取消</el-button>
  137. </div>
  138. </div>
  139. </div>
  140. <div v-show="!operationVisible && !showFunDetail">
  141. <div class="box-dashed">
  142. <div class="mg-b-20">函数公式运算执行溯源</div>
  143. <div>
  144. <el-select
  145. v-model="projectId"
  146. @change="projectChange"
  147. placeholder="请选择项目"
  148. style="width: 380px"
  149. >
  150. <el-option
  151. v-for="item in projectList"
  152. :key="item.id"
  153. :label="item.projectName"
  154. :value="item.id"
  155. ></el-option>
  156. </el-select>
  157. <el-select v-model="contractId" placeholder="请选择合同段">
  158. <el-option
  159. v-for="item in contractList"
  160. :key="item.id"
  161. :label="item.contractName"
  162. :value="item.id"
  163. ></el-option>
  164. </el-select>
  165. <el-button type="info">查询</el-button>
  166. </div>
  167. </div>
  168. <div class="text-align-c">
  169. <el-button type="warning" @click="$router.back()">取消</el-button>
  170. <el-button type="primary" @click="saveFormula">保存</el-button>
  171. </div>
  172. </div>
  173. <div v-if="!operationVisible && showFunDetail">
  174. <el-tabs v-model="actiFunIndex" closable @tab-remove="removeFun" :before-leave="funLeave">
  175. <el-tab-pane v-for="(item,index) in equationSelectEle.children" :key="index" :label="item.name" :name="index.toString()">
  176. <template v-if="!componentMap[item.name]">
  177. <formula-template ref="dynamiccomponent" :formulainfo="item" :curele="equationSelectEle" @sele-ele-handle="showChooseEle">
  178. </formula-template>
  179. </template>
  180. <template v-else>
  181. <div class="flex">
  182. <div class="flex flex-d-c">
  183. <component ref="dynamiccomponent" v-bind:is="componentMap[item.name]" :formulainfo="item" :curele="equationSelectEle" :formulamap="formulaMap" class="flex1"></component>
  184. <div v-show="item.showSelectEle">
  185. <el-select v-model="eleTableIdComp" @change="getTableEleComp" placeholder="请选择元素表" style="width:100%">
  186. <el-option v-for="item in eleTableListComp" :key="item.id" :label="item.tableName" :value="item.id"></el-option>
  187. </el-select>
  188. <div class="mg-t-10 mg-b-10 no-mb-col" style="width:900px">
  189. <el-scrollbar style="max-height: 210px;min-height:100px">
  190. <el-row>
  191. <el-col :span="6" v-for="item in eleListComp" :key="item.id">
  192. <div class="ele-box">
  193. <span>{{item.eName}}</span>
  194. <el-checkbox @change="value => setComponentEle(value,item,index)"></el-checkbox>
  195. </div>
  196. </el-col>
  197. </el-row>
  198. </el-scrollbar>
  199. </div>
  200. </div>
  201. </div>
  202. <div class="flex1" v-show="item.showSelectEle">
  203. <el-scrollbar style="height: 460px">
  204. <el-tree
  205. class="filter-tree"
  206. :data="treeData"
  207. :default-expanded-keys="defaultExpanded"
  208. @node-click="getNodeDetailComp"
  209. :props="defaultProps"
  210. :expand-on-click-node="false"
  211. highlight-current
  212. node-key="id"
  213. ref="tree"
  214. >
  215. </el-tree>
  216. </el-scrollbar>
  217. </div>
  218. </div>
  219. </template>
  220. </el-tab-pane>
  221. </el-tabs>
  222. </div>
  223. <el-dialog title="输入值" :visible.sync="inputVisible" width="300px" append-to-body :close-on-click-modal="false">
  224. <el-input v-model="inputText" placeholder="请输入内容"></el-input>
  225. <div class="text-align-c mg-t-10">
  226. <el-button size="small" @click="addTextHandle" type="primary">保存</el-button>
  227. <el-button size="small" @click="inputVisible = false">取消</el-button>
  228. </div>
  229. </el-dialog>
  230. <el-dialog title="选择元素" :visible.sync="chooseEleVisible" width="70%" append-to-body :close-on-click-modal="false">
  231. <div>
  232. <el-row :gutter="20">
  233. <el-col :span="8">
  234. <el-card shadow="never">
  235. <el-scrollbar style="height: 460px">
  236. <el-tree
  237. class="filter-tree"
  238. :data="treeData"
  239. :default-expanded-keys="defaultExpanded"
  240. @node-click="getNodeDetail"
  241. :props="defaultProps"
  242. :expand-on-click-node="false"
  243. highlight-current
  244. node-key="id"
  245. ref="tree"
  246. >
  247. </el-tree>
  248. </el-scrollbar>
  249. </el-card>
  250. </el-col>
  251. <el-col :span="16">
  252. <el-select v-model="eleTableId" @change="getTableEle" placeholder="请选择元素表">
  253. <el-option v-for="item in eleTableList" :key="item.id" :label="item.tableName" :value="item.id"></el-option>
  254. </el-select>
  255. <div class="mg-t-10 mg-b-10 no-mb-col">
  256. <el-scrollbar style="height: 210px">
  257. <el-row>
  258. <el-col :span="6" v-for="item in eleList" :key="item.id">
  259. <div class="ele-box">
  260. <span>{{item.eName}}</span>
  261. <el-checkbox v-model="item.checked" @change="value => eleCheckHandle(value,item)"></el-checkbox>
  262. </div>
  263. </el-col>
  264. </el-row>
  265. </el-scrollbar>
  266. </div>
  267. </el-col>
  268. </el-row>
  269. <div class="text-align-c">
  270. <el-button size="small" @click="chooseEleHandle" type="primary">保存</el-button>
  271. <el-button size="small" @click="chooseEleVisible = false">取消</el-button>
  272. </div>
  273. </div>
  274. </el-dialog>
  275. </basic-container>
  276. </template>
  277. <script>
  278. import { getLazytree,selectByNodeTable,selectFormElements,getAlltree} from "@/api/manager/wbstree";
  279. import { getProjectList,findProjectTree } from "@/api/manager/projectinfo";
  280. import { findContractByProjectId } from "@/api/manager/contractinfo";
  281. import { getDetail as getEleDeatil } from "@/api/manager/wbsformelement";
  282. import { getTypeMap,saveFormula,formulaDetail,updateFormula } from "@/api/formula/formula";
  283. import { findNodeTableByCondition } from "@/api/manager/wbsprivate";
  284. import {mapGetters} from "vuex";
  285. import formulaItem from "./component/formulaItem"
  286. import formulaTemplate from "./component/formulaTemplate"
  287. import dateDeviation from "./component/funComponent/dateDeviation"
  288. import dateFormat from "./component/funComponent/dateFormat"
  289. import datasRepeat from "./component/funComponent/datasRepeat"
  290. import datasReme from "./component/funComponent/datasReme"
  291. import datasGetlist from "./component/funComponent/datasGetlist"
  292. import datasJoin from "./component/funComponent/datasJoin"
  293. import ifelse from "./component/funComponent/ifelse"
  294. import {formulaArrayToString} from "./formulaArrayToString"
  295. import {formulaStringToArray} from "./formulaStringToArray"
  296. export default {
  297. components: {
  298. formulaItem,
  299. formulaTemplate,
  300. dateDeviation,
  301. dateFormat,
  302. datasRepeat,
  303. datasReme,
  304. datasGetlist,
  305. datasJoin,
  306. ifelse
  307. },
  308. data() {
  309. return {
  310. wbsid: "", //从哪个wbs树过来的
  311. eleid: "", //元素id
  312. nodeid:'',//所在树节点id
  313. pid:'',//项目id 私有树才有
  314. formulaid:'',
  315. treeData:[],//树节点
  316. defaultExpanded:[],//默认展开节点
  317. isRetain: false, //是否保留小数
  318. retainNum: 2, //保留几位小数
  319. formulaList:{},
  320. formulaMap:{},
  321. activeIndex: "1-1", //当前选择的公式
  322. projectList: [], //项目备选列表
  323. projectId: "", //溯源的项目ID
  324. curProjiect: {}, //当前项目对象
  325. contractList: [], //合同段备选列表
  326. contractId: "", //合同段id
  327. operationVisible: false, //基础运算弹窗
  328. defaultProps: {
  329. children: "children",
  330. label: "title",
  331. isLeaf: function (data) {
  332. return !data.hasChildren || (data.isExistForm==1);
  333. },
  334. },
  335. eleTableId:'',//选中的元素表id
  336. eleTableList:[],
  337. eleList:[],
  338. selectEleFormula:[],
  339. curSeleEleIndex:-1,//公式文字里面选中的元素索引
  340. inputVisible:false,//输入弹窗
  341. inputText:"",//输入值
  342. resultFormula:[],//=等号左边的数组
  343. processFormula:[],//=等号右边的数组
  344. processType:'',//选中的元素在等号哪边
  345. processSelectIndex:0,//选中的索引
  346. actiFunIndex:0,//元素下挂载的计算式的索引
  347. chooseEleVisible:false,//选择元素弹窗
  348. argumenObj:{},
  349. symbolReg:/(\+|-|\*|\/)(.+)/,
  350. operatorReg : /^\+|-|\*|%/,//加减乘除
  351. startFCRegExp : /^FC\.([a-zA-Z]+)\(/,// 匹配开始的FC.xxx(
  352. componentMap:{
  353. '日期偏移':'date-deviation',
  354. '日期格式化':'date-format',
  355. '去重':'datas-repeat',
  356. '去空':'datas-reme',
  357. '下标取数':'datas-getlist',
  358. '数组转字符串':"datas-join",
  359. '判断':'ifelse'
  360. },
  361. eleListComp:[],//方法下面元素列表
  362. eleTableListComp:[],//方法下面元素表列表
  363. eleTableIdComp:'',//方法下面元素表id
  364. };
  365. },
  366. computed: {
  367. ...mapGetters(["userInfo"]),
  368. // selectEleFormulaText:function(){
  369. // let text = '';
  370. // this.selectEleFormula.forEach((Element)=>{
  371. // text+=Element.name;
  372. // })
  373. // return text
  374. // }
  375. //等式中选中的元素
  376. equationSelectEle:function(){
  377. if(this.processType){
  378. return this[this.processType][this.processSelectIndex];
  379. }else{
  380. return null;
  381. }
  382. },
  383. //是否显示元素下挂载的计算式信息
  384. showFunDetail:function(){
  385. if(this.equationSelectEle && this.equationSelectEle.children && this.equationSelectEle.children.length>0){
  386. return true;
  387. }else{
  388. return false;
  389. }
  390. }
  391. },
  392. created() {
  393. this.wbsid = this.$route.query.wbsid;
  394. this.eleid = this.$route.query.eleid;
  395. this.nodeid = this.$route.query.nodeid;
  396. this.pid = this.$route.query.projectid;//项目id 私有树才有
  397. this.init();
  398. },
  399. methods: {
  400. async init() {
  401. this.getEleDeatil();
  402. this.getProjectList();
  403. this.geTreeData();
  404. await this.getTypeMap();
  405. this.formulaStringToArray();
  406. },
  407. geTreeData(){
  408. if(this.pid){
  409. findProjectTree(this.pid, this.wbsid).then((res) => {
  410. this.treeData = res.data.data;
  411. this.defaultExpanded = [this.nodeid];
  412. })
  413. }else{
  414. getAlltree(this.userInfo.tenant_id, 1, this.wbsid).then((res) => {
  415. this.treeData = res.data.data;
  416. this.defaultExpanded = [this.nodeid];
  417. })
  418. }
  419. },
  420. //懒加载树
  421. loadNode(node, resolve) {
  422. let pid = 0;
  423. if (node.level != 0) {
  424. pid = node.data.id;
  425. }
  426. getLazytree(this.wbsid, pid, this.userInfo.tenant_id).then((res) => {
  427. let arr = [];
  428. if (Array.isArray(res.data.data)) {
  429. arr = res.data.data;
  430. }
  431. return resolve(arr);
  432. });
  433. },
  434. //获取项目列表
  435. getProjectList() {
  436. getProjectList(1, 999).then((res) => {
  437. this.projectList = res.data.data.records;
  438. });
  439. },
  440. //选择公式处理
  441. handleSelect(index,indexPath) {
  442. //console.log(index,'index')
  443. //console.log(indexPath,'indexPath')
  444. if(this.operationVisible){
  445. this.openerationSelect(index,indexPath)
  446. }else{
  447. this.equationSelect(index,indexPath)
  448. }
  449. },
  450. //在选择元素模式下点选计算式
  451. openerationSelect(index,indexPath){
  452. if(indexPath[0]!='基础运算'){
  453. this.$message({
  454. type: "warning",
  455. message: "当前只能使用基础运算"
  456. });
  457. return;
  458. }
  459. let formulaindex = Number(indexPath[1].split('-')[1])-1;
  460. this.eleAddFormulaHandle(this.formulaList[indexPath[0]][formulaindex]);
  461. },
  462. //溯源项目id切换
  463. projectChange(id) {
  464. for (let i = 0; i < this.projectList.length; i++) {
  465. if (id == this.projectList[i].id) {
  466. this.curProjiect = this.projectList[i];
  467. //根据项目id获取合同段列表
  468. findContractByProjectId(this.curProjiect.id).then((res) => {
  469. this.contractList = res.data.data;
  470. this.contractId = "";
  471. });
  472. return;
  473. }
  474. }
  475. },
  476. operationEdit(){
  477. this.selectEleFormula= JSON.parse(JSON.stringify(this.processFormula));
  478. this.operationVisible = true;
  479. },
  480. eleAddFormula(){
  481. for (let i = 0; i < this.eleList.length; i++) {
  482. if (this.eleList[i].checked) {
  483. this.eleAddFormulaHandle(this.eleList[i]);
  484. break;
  485. }
  486. }
  487. },
  488. eleChang(value,item){
  489. //console.log(value)
  490. //console.log(item)
  491. if(value){
  492. //简单语法判断
  493. if(this.selectEleFormula.length != 0){
  494. let lastEle = this.selectEleFormula[this.selectEleFormula.length-1];
  495. if(lastEle.type == 'Element'){
  496. this.$message({
  497. type: "warning",
  498. message: "元素无法连续出现在元素后面"
  499. });
  500. item.checked = false;
  501. return;
  502. }
  503. if(lastEle.type == 'Text'){
  504. this.$message({
  505. type: "warning",
  506. message: "元素无法连续出现在输入值后面"
  507. });
  508. item.checked = false;
  509. return;
  510. }
  511. if(lastEle.type == 'Brackets' && lastEle.name == ')'){
  512. this.$message({
  513. type: "warning",
  514. message: "元素无法连续出现在右括号后面"
  515. });
  516. item.checked = false;
  517. return;
  518. }
  519. }
  520. this.eleAddFormulaHandle(item);
  521. }else{
  522. let index = -1;
  523. for (let i = 0; i < this.selectEleFormula.length; i++) {
  524. if(this.selectEleFormula[i].id == item.id){
  525. index = i;
  526. break;
  527. }
  528. }
  529. if(index>-1){
  530. this.selectEleFormula.splice(index,1);
  531. }
  532. }
  533. },
  534. //快捷添加运算符号
  535. addOperator(operator){
  536. this.eleAddFormulaHandle(this.formulaMap[operator]);
  537. },
  538. //把元素加到公式里
  539. eleAddFormulaHandle(ele){
  540. if(ele.tableElementKey){
  541. //元素
  542. this.selectEleFormula.push({
  543. type:'Element',
  544. name:ele.eName,
  545. id:ele.id,
  546. selected:false,
  547. tableElementKey:ele.tableElementKey,
  548. children:[],
  549. })
  550. }else if(ele.template && ele.example){
  551. //简单语法判断
  552. if(this.selectEleFormula.length == 0){
  553. this.$message({
  554. type: "warning",
  555. message: "公式开头不能是运算符号"
  556. });
  557. return;
  558. }else{
  559. let lastEle = this.selectEleFormula[this.selectEleFormula.length-1];
  560. if(lastEle.type == 'Operator'){
  561. this.$message({
  562. type: "warning",
  563. message: "运算符号无法连续出现在运算符号后面"
  564. });
  565. return;
  566. }
  567. if(lastEle.type == 'Brackets' && lastEle.name == '('){
  568. this.$message({
  569. type: "warning",
  570. message: "运算符号无法连续出现在左括号后面"
  571. });
  572. return;
  573. }
  574. }
  575. //运算符号
  576. this.selectEleFormula.push({
  577. type:'Operator',
  578. name:this.symbolReg.exec(ele.name)[1],
  579. selected:false,
  580. template:ele.template
  581. })
  582. }else if(ele.type == 'Brackets'){
  583. //括号
  584. this.selectEleFormula.splice(ele.selectIndex,0,{
  585. type:'Brackets',
  586. name:ele.name,
  587. selected:false,
  588. })
  589. }else if(ele.type == 'Text'){
  590. //输入值
  591. this.selectEleFormula.push({
  592. type:'Text',
  593. name:ele.name,
  594. selected:false,
  595. })
  596. }
  597. },
  598. //添加括号
  599. addBrackets(text,type){
  600. //type 是true 表示在元素右边插入
  601. if(this.curSeleEleIndex == Number(this.curSeleEleIndex)){
  602. this.eleAddFormulaHandle({
  603. type:'Brackets',
  604. name:text,
  605. selectIndex:type?Number(this.curSeleEleIndex)+1:this.curSeleEleIndex
  606. })
  607. //如果在左边插入index要增1
  608. if(!type){
  609. this.curSeleEleIndex = Number(this.curSeleEleIndex)+1;
  610. }
  611. }
  612. },
  613. addText(){
  614. this.inputVisible = true;
  615. },
  616. //添加输入值
  617. addTextHandle(){
  618. //简单语法判断
  619. if(this.selectEleFormula.length != 0){
  620. let lastEle = this.selectEleFormula[this.selectEleFormula.length-1];
  621. if(lastEle.type == 'Element'){
  622. this.$message({
  623. type: "warning",
  624. message: "输入值无法连续出现在元素后面"
  625. });
  626. return;
  627. }
  628. if(lastEle.type == 'Text'){
  629. this.$message({
  630. type: "warning",
  631. message: "输入值无法连续出现在输入值后面"
  632. });
  633. return;
  634. }
  635. if(lastEle.type == 'Brackets' && lastEle.name == ')'){
  636. this.$message({
  637. type: "warning",
  638. message: "输入值无法连续出现在右括号后面"
  639. });
  640. return;
  641. }
  642. }
  643. this.eleAddFormulaHandle({
  644. type:'Text',
  645. name:this.inputText
  646. })
  647. this.inputVisible = false;
  648. },
  649. //勾选元素
  650. eleCheckHandle(checked,item){
  651. if(checked){
  652. this.eleList.forEach((ele)=>{
  653. this.$set(ele,'checked',false);
  654. //ele.checked = false;
  655. })
  656. item.checked = true;
  657. }
  658. },
  659. //点选公式中的元素
  660. eleFormulaClick({selected,item},index){
  661. if(selected){
  662. this.selectEleFormula.forEach((ele)=>{
  663. ele.selected = false;
  664. })
  665. item.selected = true;
  666. this.curSeleEleIndex = index;
  667. }else{
  668. this.curSeleEleIndex = -1;
  669. }
  670. },
  671. //删除点选公式中的元素
  672. removeSelect(){
  673. if(this.curSeleEleIndex > -1){
  674. this.selectEleFormula.splice(this.curSeleEleIndex,1);
  675. this.curSeleEleIndex = -1;
  676. }
  677. },
  678. //赋值给等号右边的数组
  679. operationHandle(){
  680. //检测左右括号数量是否相等
  681. let lBracketNum = 0;
  682. let rBracketNum = 0;
  683. this.selectEleFormula.forEach((ele)=>{
  684. if(ele.type == 'Brackets'){
  685. if(ele.name == '('){
  686. lBracketNum++;
  687. }else if(ele.name == ')'){
  688. rBracketNum++;
  689. }
  690. }
  691. })
  692. if(lBracketNum != rBracketNum){
  693. this.$message({
  694. type: "warning",
  695. message: "左右括号数量不相等,请先检查是否正确"
  696. });
  697. return;
  698. }
  699. this.processFormula = JSON.parse(JSON.stringify(this.selectEleFormula));
  700. this.operationVisible = false;
  701. },
  702. //点选等式中的元素
  703. equationClick({selected,item},index,arrName){
  704. if(selected){
  705. this.resultFormula.forEach((ele)=>{
  706. ele.selected = false;
  707. })
  708. this.processFormula.forEach((ele)=>{
  709. ele.selected = false;
  710. })
  711. this.processSelectIndex = index;
  712. this.processType = arrName;
  713. item.selected = true;
  714. }else{
  715. this.processType = '';
  716. }
  717. },
  718. //在等式模式下点选计算式
  719. equationSelect(index,indexPath){
  720. if(!this.equationSelectEle ||(this.equationSelectEle && this.equationSelectEle.type != 'Element') ){
  721. this.$message({
  722. type: "warning",
  723. message: "请先选中元素"
  724. });
  725. return;
  726. }
  727. let formulaindex = Number(indexPath[1].split('-')[1])-1;
  728. let expression = this.formulaList[indexPath[0]][formulaindex];
  729. if(expression.type ==1){
  730. return;
  731. }
  732. //console.log(JSON.parse(expression.template));
  733. let obj = Object.assign({}, expression);
  734. //obj.template = JSON.parse(obj.template);
  735. obj.arguments = new Array(obj.template.args.length);
  736. obj.arguments[0] = {
  737. id:this.equationSelectEle.id,
  738. name:this.equationSelectEle.name,
  739. selected:false,
  740. tableElementKey:this.equationSelectEle.tableElementKey,
  741. type:"Element",
  742. };
  743. this.equationSelectEle.children.push(obj);
  744. },
  745. //选择元素
  746. chooseEleHandle(){
  747. for (let i = 0; i < this.eleList.length; i++) {
  748. if (this.eleList[i].checked) {
  749. let ele = this.eleList[i]
  750. let obj = {
  751. type:'Element',
  752. name:ele.eName,
  753. id:ele.id,
  754. selected:false,
  755. tableElementKey:ele.tableElementKey,
  756. children:[],
  757. }
  758. this.$set(this.argumenObj.arguments,this.argumenObj.index,obj);
  759. this.chooseEleVisible = false;
  760. break;
  761. }
  762. }
  763. },
  764. //显示选择元素弹窗
  765. showChooseEle(argumenObj){
  766. this.argumenObj = argumenObj;
  767. this.chooseEleVisible = true;
  768. },
  769. //移除挂载的函数
  770. removeFun(name){
  771. //console.log(name)
  772. this.equationSelectEle.children.splice(Number(name), 1);
  773. },
  774. //切换公式tab标签
  775. funLeave(activeName, oldActiveName){
  776. if(oldActiveName){
  777. let formula = this.equationSelectEle.children[Number(oldActiveName)];
  778. if(formula){
  779. return this.checkFormulaLegal(formula);
  780. }
  781. }
  782. },
  783. //检测公式合法
  784. checkFormulaLegal(formula){
  785. if(!formula.arguments){
  786. return false;
  787. }
  788. //当前选中的元素
  789. let curEle = this.equationSelectEle;
  790. let isIn = false;
  791. for (let i = 0; i < formula.arguments.length; i++) {
  792. if(Array.isArray(formula.arguments[i])){
  793. for(let j=0;j<formula.arguments[i].length;j++){
  794. if(formula.arguments[i][j] && formula.arguments[i][j].id ==curEle.id){
  795. isIn = true;
  796. break;
  797. }
  798. }
  799. if(isIn){
  800. break;
  801. }
  802. }else if(formula.arguments[i] && formula.arguments[i].id ==curEle.id){
  803. isIn = true;
  804. break;
  805. }
  806. }
  807. if(!isIn){
  808. this.$message({
  809. type: "warning",
  810. message: "参数必须有一个值是当前元素"
  811. });
  812. return false;
  813. }
  814. return true;
  815. },
  816. //保存公式
  817. saveFormula(){
  818. let obj = formulaArrayToString(this.processFormula,this.resultFormula);
  819. //return;
  820. //console.log(text);
  821. if(this.formulaid){
  822. updateFormula({
  823. id:this.formulaid,
  824. formula:obj.text,
  825. remark:'',
  826. wbsId:this.wbsid,
  827. elementId:this.eleid,
  828. map:JSON.stringify(obj.eleMap)
  829. }).then(()=>{
  830. this.$message({
  831. type: "success",
  832. message: "修改成功"
  833. });
  834. })
  835. }else{
  836. saveFormula({
  837. formula:obj.text,
  838. remark:'',
  839. wbsId:this.wbsid,
  840. elementId:this.eleid,
  841. map:JSON.stringify(obj.eleMap)
  842. }).then(()=>{
  843. this.$message({
  844. type: "success",
  845. message: "保存成功"
  846. });
  847. })
  848. }
  849. },
  850. //把公式文本还原数组
  851. async formulaStringToArray(){
  852. let detail = (await formulaDetail({elementId:this.eleid})).data.data;
  853. //console.log(detail);
  854. if(detail.id){
  855. this.formulaid = detail.id;
  856. //let formula = formulaStringToArray('FC.sum(FC.repeat(E[测试测试_222]))+FC.ifelse(3<E[测试测试_333]&&E[测试测试_333]<10,E[测试测试_222]+E[测试测试_333],E[测试测试_333])',detail.map,this.formulaMap);
  857. detail.formula = detail.formula.replace(/&lt;/g,"<").replace(/&amp;/g,"&");
  858. let formula = formulaStringToArray(detail.formula,detail.map,this.formulaMap);
  859. this.processFormula = formula.processFormula;
  860. formula.resultFormula[0].id = this.resultFormula[0].id;
  861. formula.resultFormula[0].name = this.resultFormula[0].name;
  862. formula.resultFormula[0].tableElementKey = this.resultFormula[0].tableElementKey;
  863. this.resultFormula[0].children = formula.resultFormula[0].children;
  864. }
  865. },
  866. //设置动态组件里面的元素
  867. setComponentEle(value,item,index){
  868. if(value){
  869. //console.log(this.$refs.dynamiccomponent[index])
  870. this.$refs.dynamiccomponent[index].setELe(item);
  871. }
  872. },
  873. getNodeDetail(data) {
  874. if(this.pid){
  875. findNodeTableByCondition(data.id, this.pid, this.wbsid).then((res) => {
  876. if(res.data.data.length){
  877. this.eleTableList = res.data.data;
  878. this.eleTableId = this.eleTableList[0].id;
  879. this.getTableEle(this.eleTableId);
  880. }else{
  881. this.eleTableList = [];
  882. this.eleTableId = '';
  883. this.eleList = [];
  884. }
  885. })
  886. }else{
  887. selectByNodeTable(data.id).then((res)=>{
  888. if(res.data.data.length){
  889. this.eleTableList = res.data.data;
  890. this.eleTableId = this.eleTableList[0].id;
  891. this.getTableEle(this.eleTableId);
  892. }else{
  893. this.eleTableList = [];
  894. this.eleTableId = '';
  895. this.eleList = [];
  896. }
  897. })
  898. }
  899. },
  900. getEleDeatil(){
  901. getEleDeatil(this.eleid).then((res)=>{
  902. let ele = res.data.data;
  903. this.resultFormula = [{
  904. type:'Element',
  905. name:ele.eName,
  906. id:ele.id,
  907. selected:false,
  908. tableElementKey:ele.tableElementKey,
  909. children:[],
  910. }]
  911. })
  912. },
  913. getTableEle(tableId){
  914. selectFormElements(tableId).then((res)=>{
  915. this.eleList = res.data.data;
  916. })
  917. },
  918. //方法下面的点击树节点
  919. getNodeDetailComp(data) {
  920. if(this.pid){
  921. findNodeTableByCondition(data.id, this.pid, this.wbsid).then((res) => {
  922. if(res.data.data.length){
  923. this.eleTableListComp = res.data.data;
  924. this.eleTableIdComp = this.eleTableListComp[0].id;
  925. this.getTableEleComp(this.eleTableIdComp);
  926. }else{
  927. this.eleTableListComp = [];
  928. this.eleTableIdComp = '';
  929. this.eleListComp = [];
  930. }
  931. })
  932. }else{
  933. selectByNodeTable(data.id).then((res)=>{
  934. if(res.data.data.length){
  935. this.eleTableListComp = res.data.data;
  936. this.eleTableIdComp = this.eleTableListComp[0].id;
  937. this.getTableEleComp(this.eleTableIdComp);
  938. }else{
  939. this.eleTableListComp = [];
  940. this.eleTableIdComp = '';
  941. this.eleListComp = [];
  942. }
  943. })
  944. }
  945. },
  946. //方法下面的查询元素
  947. getTableEleComp(tableId){
  948. selectFormElements(tableId).then((res)=>{
  949. this.eleListComp = res.data.data;
  950. })
  951. },
  952. getTypeMap(){
  953. return new Promise((resolve)=>{
  954. getTypeMap().then((res)=>{
  955. //console.log(res)
  956. this.formulaList = res.data.data;
  957. //生成map,方便查找
  958. for (let key in this.formulaList) {
  959. if(typeof(this.formulaList[key]) == 'object'){
  960. this.formulaList[key].forEach((formula)=>{
  961. formula.template = JSON.parse(formula.template);
  962. if(this.operatorReg.test(formula.template.ft)){
  963. this.formulaMap[formula.template.ft] = formula;
  964. }else if(this.startFCRegExp.test(formula.template.ft)){
  965. let regRes = formula.template.ft.match(this.startFCRegExp);
  966. this.formulaMap[regRes[0]] = formula;
  967. }
  968. })
  969. }
  970. }
  971. }).finally(() => {
  972. resolve();
  973. })
  974. })
  975. }
  976. },
  977. };
  978. </script>
  979. <style scoped lang="scss">
  980. .box-dashed {
  981. border: 1px dashed #bbbbbb;
  982. border-radius: 6px;
  983. padding: 10px;
  984. margin-bottom: 10px;
  985. }
  986. .retain-box {
  987. border-right: 1px dashed #bbbbbb;
  988. }
  989. .retain {
  990. line-height: 50px;
  991. margin-right: 10px;
  992. }
  993. .edit-text {
  994. font-size: 26px;
  995. margin-left: 20px;
  996. }
  997. .el-menu--popup .el-menu-item.is-active {
  998. background-color: #fff;
  999. }
  1000. .ele-box{
  1001. border: 1px solid #bbb;
  1002. height: 26px;
  1003. display: flex;
  1004. justify-content: space-between;
  1005. align-items: center;
  1006. padding: 6px;
  1007. }
  1008. .no-mb-col .el-col{
  1009. margin-bottom: 0px;
  1010. }
  1011. .sele-ele-box{
  1012. height: 160px;
  1013. padding: 20px;
  1014. // margin-top: 10px;
  1015. }
  1016. .icon-box .el-link{
  1017. font-size: 24px;
  1018. margin-right: 10px;
  1019. }
  1020. </style>