index.vue 37 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303
  1. <template>
  2. <VueFlow ref="vueFlowRef" v-model:nodes="nodes" v-model:edges="edges" :class="{ dark }"
  3. class="basic-flow"
  4. style="background-image: url('src/assets/flowimg/surromdao.png'); background-position: center; background-repeat: no-repeat;"
  5. :default-viewport="{ zoom: 1.5 }" :min-zoom="0.2" :max-zoom="2.5" @drop="onDrop1"
  6. @node-contextmenu="logEvent('contextmenu', $event)"
  7. @dragover="onDragOver" @dragleave="onDragLeave" @edge-click="onEdgeClick" @node-double-click="onNodeDoubleClick"
  8. @node-click="onNodeClick" @edge-double-click="onEdgeDoubleClick">
  9. <template #node-default="props">
  10. <eltree :node="props" />
  11. </template>
  12. <Background pattern-color="#aaa" :gap="16"
  13. />
  14. <!-- <MiniMap /> @input="handleUpdate"-->
  15. <Panel :position="'right'" style="display: none;">
  16. <div class="field">
  17. <div>
  18. <label for="label">Label:</label>
  19. <input id="label" v-model.lazy="labelname" /></div>
  20. <button @click="handleUpdate" class="lableaniu">确定</button>
  21. </div>
  22. <!-- <div> -->
  23. <!-- <button class="remove" @click="removeNode(event)">N</button>
  24. <button class="remove" @click="removeEdge(event)">E</button>
  25. <button class="remove" @click="removeall(event)">all</button>
  26. </div> -->
  27. <!-- <button @click="onSelection()" class="lableaniu">获取</button> -->
  28. </Panel>
  29. <Controls position="top-left">
  30. <ControlButton title="重置" @click="resetTransform">
  31. <Icon name="reset" />
  32. </ControlButton>
  33. <ControlButton title="背景切换" @click="toggleDarkMode">
  34. <Icon v-if="dark" name="sun" />
  35. <Icon v-else name="moon" />
  36. </ControlButton>
  37. <ControlButton title="保存" @click="logToObject1">
  38. <!-- <Icon name="log" /> -->
  39. <el-icon :color="iconcolor"><UploadFilled /></el-icon>
  40. </ControlButton>
  41. <ControlButton title="删除节点" @click="removeNode()">
  42. <el-icon :color="iconcolor"><DocumentDelete /></el-icon>
  43. </ControlButton>
  44. <ControlButton title="删除线" @click="removeEdge()">
  45. <el-icon :color="iconcolor"><Crop /></el-icon>
  46. </ControlButton>
  47. <ControlButton title="清空全部" @click="confirmDelete()">
  48. <el-icon :color="iconcolor"><DeleteFilled /></el-icon>
  49. </ControlButton>
  50. </Controls>
  51. <el-dialog v-model="changeNameshow" align-center :modal="false" :close-on-click-modal="false"
  52. :append-to-body="true" draggable :fullscreen="false" :modal-append-to-body="false" modal-class="summary-dlg"
  53. width="400" class="dialog_class bgcolor colortext tianjia sel">
  54. <template #header="{ titleId, titleClass }">
  55. <div class="my-header ">
  56. <h4 :id="titleId" :class="titleClass">修改名称</h4>
  57. </div>
  58. </template>
  59. <div style="margin-top: 20px;padding: 20px;">
  60. <el-form :model="changeName" label-width="100px" class="demo-ruleForm">
  61. <el-form-item label="新名称:" prop="name">
  62. <el-input v-model="changeName.name" placeholder="请输入名称"></el-input>
  63. </el-form-item>
  64. </el-form>
  65. </div>
  66. <template #footer>
  67. <div class="dialog-footer">
  68. <el-button @click="changeNameshow = false">取 消</el-button>
  69. <el-button type="primary" @click="handleUpdate">确 定</el-button>
  70. </div>
  71. </template>
  72. </el-dialog>
  73. <!-- 双击线段 -->
  74. <el-dialog v-model="dataflowshow" align-center :modal="false" :close-on-click-modal="false"
  75. :append-to-body="true" draggable :fullscreen="false" :modal-append-to-body="false" modal-class="summary-dlg"
  76. width="500" class="dialog_style bgcolor colortext tianjia sel">
  77. <template #header="{ titleId, titleClass }">
  78. <div class="my-header ">
  79. <h4 :id="titleId" :class="titleClass">数据流</h4>
  80. </div>
  81. </template>
  82. <div style="margin-top: 5px;padding: 10px 35px;">
  83. <div v-if="datacontent === '1'">
  84. <el-card :shadow="none" style="min-height: 240px; max-height: 480px;">
  85. <el-checkbox v-model="checkAll"
  86. :indeterminate="isIndeterminate"
  87. @change="handlecheckAllchange">全选</el-checkbox>
  88. <div class="eldesign classtable">
  89. <el-table :data="youhuaFFDtable" border :show-header="false">
  90. <el-table-column
  91. type="index"
  92. label=""
  93. ></el-table-column>
  94. <el-table-column prop="steamflag" width="55">
  95. <template #default="{ row }">
  96. <el-checkbox
  97. :false-label="0"
  98. :true-label="1"
  99. v-model="row.steamflag"
  100. @change="handlecheckFlagchange"></el-checkbox>
  101. </template>
  102. </el-table-column>
  103. <el-table-column prop="name" label="" ></el-table-column>
  104. </el-table>
  105. </div>
  106. </el-card>
  107. </div>
  108. <div v-if="datacontent === '2'">
  109. <el-card :shadow="none" style="min-height: 240px; max-height: 480px;">
  110. <el-checkbox v-model="checkAll"
  111. :indeterminate="isIndeterminate"
  112. @change="handlecheckAllchange">全选</el-checkbox>
  113. <div class="eldesign classtable">
  114. <el-table :data="youhuaCSTtable" border :show-header="false">
  115. <el-table-column
  116. type="index"
  117. label=""
  118. ></el-table-column>
  119. <el-table-column prop="steamflag" width="55">
  120. <template #default="{ row }">
  121. <el-checkbox
  122. :false-label="0"
  123. :true-label="1"
  124. v-model="row.steamflag"
  125. @change="handlecheckFlagchange"></el-checkbox>
  126. </template>
  127. </el-table-column>
  128. <el-table-column prop="name" label="" ></el-table-column>
  129. </el-table>
  130. </div>
  131. </el-card>
  132. </div>
  133. <div v-if="datacontent === '3'">
  134. <el-card :shadow="none" style="min-height: 240px; max-height: 480px;">
  135. <el-checkbox v-model="checkAll"
  136. :indeterminate="isIndeterminate"
  137. @change="handlecheckAllchange">全选</el-checkbox>
  138. <div class="eldesign classtable">
  139. <el-table :data="ADflowyouhuatable" border :show-header="false">
  140. <el-table-column
  141. type="index"
  142. label=""
  143. ></el-table-column>
  144. <el-table-column prop="steamflag" width="55">
  145. <template #default="{ row }">
  146. <el-checkbox
  147. :false-label="0"
  148. :true-label="1"
  149. v-model="row.steamflag"
  150. @change="handlecheckFlagchange"></el-checkbox>
  151. </template>
  152. </el-table-column>
  153. <el-table-column prop="name" label="" ></el-table-column>
  154. <el-table-column prop="steamtype">
  155. <template #default="{ row }">
  156. <el-select v-model="row.steamtype">
  157. <el-option
  158. v-for="item in steamtypeoptions"
  159. :key="item.value"
  160. :label="item.label"
  161. :value="item.value"
  162. />
  163. </el-select>
  164. </template>
  165. </el-table-column>
  166. </el-table>
  167. </div>
  168. </el-card>
  169. </div>
  170. <div v-if="datacontent === '4'">
  171. <el-card :shadow="none" style="min-height: 240px; max-height: 480px;">
  172. <el-checkbox v-model="checkAll"
  173. :indeterminate="isIndeterminate"
  174. @change="handlecheckAllchange">全选</el-checkbox>
  175. <div class="eldesign classtable">
  176. <el-table :data="Xfoilyouhuatable" border :show-header="false">
  177. <el-table-column
  178. type="index"
  179. label=""
  180. ></el-table-column>
  181. <el-table-column prop="steamflag" width="55">
  182. <template #default="{ row }">
  183. <el-checkbox
  184. :false-label="0"
  185. :true-label="1"
  186. v-model="row.steamflag"
  187. @change="handlecheckFlagchange"></el-checkbox>
  188. </template>
  189. </el-table-column>
  190. <el-table-column prop="name" label="" show-overflow-tooltip></el-table-column>
  191. <el-table-column prop="steamtype">
  192. <template #default="{ row }">
  193. <el-select v-model="row.steamtype">
  194. <el-option
  195. v-for="item in steamtypeoptions"
  196. :key="item.value"
  197. :label="item.label"
  198. :value="item.value"
  199. />
  200. </el-select>
  201. </template>
  202. </el-table-column>
  203. </el-table>
  204. </div>
  205. </el-card>
  206. </div>
  207. </div>
  208. <template #footer>
  209. <div class="dialog-footer">
  210. <el-button @click="dataflowshow = false">取 消</el-button>
  211. <el-button type="primary" @click="confirmselection();dataflowshow = false ">确 定</el-button>
  212. </div>
  213. </template>
  214. </el-dialog>
  215. </VueFlow>
  216. <div class="dnd-flow">
  217. <!-- <div id="contextMenu" @click="deleteItemConfirm">删除</div> -->
  218. </div>
  219. </template>
  220. <script setup>
  221. import { ref, markRaw,inject } from 'vue'
  222. import { VueFlow,Panel, useVueFlow, MarkerType} from '@vue-flow/core'
  223. import { ElMessage, ElButton, ElDialog, ElSelect, ElMessageBox} from 'element-plus'
  224. import {
  225. DocumentDelete,
  226. Delete,
  227. UploadFilled,
  228. Histogram,
  229. DeleteFilled,
  230. Crop,
  231. } from '@element-plus/icons-vue'
  232. import { useRoute } from 'vue-router';
  233. import { request, uploadFile } from "@/utils/request";
  234. import { Background } from '@vue-flow/background'
  235. import { ControlButton, Controls } from '@vue-flow/controls'
  236. import { initialEdges, initialNodes } from './modeljs.js'
  237. import { MiniMap } from '@vue-flow/minimap'
  238. import "./main.css";//重置样式
  239. import DropzoneBackground from './DropzoneBackground.vue'
  240. import eltree from './eltree.vue'
  241. import CustomEdge from './CustomEdge.vue';
  242. import useDragAndDrop from './useDnD';
  243. import f11 from '@/assets/img/f11.png'
  244. import r2 from '@/assets/img/r2.png'
  245. import html2canvas from 'html2canvas';
  246. import Icon from './Icon.vue'
  247. import { formatTime } from '@/js/lindex.js';
  248. import emitter from "@/utils/emitter";
  249. // import func from 'vue-temp/vue-editor-bridge.js';
  250. const dark = ref(false)
  251. let datatree=ref();
  252. const route = useRoute();
  253. const { onInit, onNodeDragStop, onNodeContextMenu, onConnect, addEdges, setViewport, toObject,addNodes,updateEdgeData } = useVueFlow()
  254. let vueFlowRef = ref();
  255. let emit = defineEmits(['optimizerfalse']);
  256. let mergedObj=ref();
  257. let labelname=ref();
  258. let iconcolor=ref('#000')
  259. const props = defineProps({
  260. optimizer: {
  261. type: Boolean,
  262. },
  263. jboptimizer: {
  264. type: Boolean,
  265. },
  266. Xfoil: {
  267. type: Boolean,
  268. },
  269. // color1: {
  270. // type: String,
  271. // },
  272. })
  273. let newobj=ref({
  274. name:'',
  275. description:'',
  276. })
  277. let nodesitem=ref([]);
  278. let linenum=ref(1);
  279. let bgcolor=ref();
  280. let linecolor=ref('#b1b1b7')
  281. let pid=ref('');
  282. let newroter=ref();
  283. let vueflowimg=ref('');
  284. const shopShow = ref(false);
  285. let noid = ref([]);
  286. let Edgeid = ref();
  287. let seledge=ref(null);
  288. let djshow=ref(false);
  289. let node = ref();
  290. let contextMenu = ref({
  291. position: { x: 0, y: 0 },
  292. target: 'kong',
  293. })
  294. let Nested=ref([]);
  295. let Nested2=ref([]);
  296. let nnum=ref(0);
  297. const { onDragOver, onDrop, onDragLeave, isDragOver } = useDragAndDrop();
  298. const edges = ref([]);
  299. const nodes = ref([]);
  300. const changeNameshow = ref(false);
  301. const changeName = ref({
  302. name: '',
  303. })
  304. onNodeContextMenu((e) => {
  305. noid.value = e.node;
  306. changeName.value.name = e.node.data.label;
  307. changeNameshow.value = true;
  308. })
  309. // 线的类型 process-逻辑线 data-数据线
  310. let lineType = ref('process');
  311. // 线序号
  312. let linecount = ref(0);
  313. onConnect((connection) => {
  314. console.log('线连接',connection);
  315. // 创建唯一的边ID(由源节点和目标节点组成)
  316. const edgeId = `${connection.source}-${connection.target}`;
  317. connection.id = edgeId; // 自定义 ID
  318. connection.type = 'smoothstep';// smoothstep straight
  319. connection.markerEnd = MarkerType.ArrowClosed;
  320. connection.color=linecolor.value;
  321. // connection.label = '这是一条注释';
  322. connection.style = { strokeWidth:linenum.value ,stroke:linecolor.value};
  323. addEdges(connection)
  324. seledge.value=null;
  325. linecount.value++;
  326. const newName = `Seg${linecount.value}`;
  327. const sourceNode = vueFlowRef.value.getNode(connection.source);
  328. const targetNode = vueFlowRef.value.getNode(connection.target);
  329. // 调用 saveflow 保存数据并获取 wid
  330. saveflow(pid.value, '', newName, lineType.value, sourceNode.data.uid, targetNode.data.uid)
  331. .then((wid) => {
  332. // 更新连接线的数据
  333. const updatedData = {
  334. wid: wid,
  335. uid: newName,
  336. type: lineType.value,
  337. fromuid: sourceNode.data.uid,
  338. touid: targetNode.data.uid,
  339. };
  340. updateEdgeData(connection.id, updatedData);
  341. })
  342. .catch((error) => {
  343. console.error('保存流程失败:', error);
  344. });
  345. })
  346. //修改名称
  347. const handleUpdate = () => {
  348. if (noid.value && noid.value.data) {
  349. noid.value.data.label = changeName.value.name; // 更新名称
  350. console.log(noid.value.data.label); // 打印更新后的值
  351. } else {
  352. console.error("noid.value 或 noid.value.data 未定义"); // 错误日志
  353. }
  354. changeNameshow.value = false; // 更新后关闭对话框
  355. // noid.value.data.label = labelname;
  356. };
  357. emitter.on('child2Data', data => {
  358. datatree.value = data;
  359. console.log("datatree的值:",datatree.value);
  360. })
  361. function onNodeClick(e) {
  362. noid.value = e.node;
  363. changeName.value.name=e.node.data.label;
  364. console.log("shuju:",e.node.data);
  365. console.log("noid.value:",noid.value);
  366. console.log("changeName.value:",changeName.value.name);
  367. djshow.value = !djshow.value;
  368. // if(djshow.value){
  369. // console.log('选中');
  370. // }else{
  371. // console.log('取消选中');
  372. // }
  373. }
  374. //模块化
  375. const bgcolorfunc= (color)=>{
  376. bgcolor.value=color;
  377. if(noid.value.style!=undefined){
  378. console.log(noid.value.style);
  379. noid.value.style.backgroundColor=color;
  380. }
  381. }
  382. function onSelection(){
  383. let positionX=[];
  384. let positiony=[];
  385. for(let i=0;i<Nested.value.length;i++){
  386. // Nested2.value.push(Nested.value[i]);
  387. positionX.push(Nested.value[i].position.x)
  388. positiony.push(Nested.value[i].position.y)
  389. }
  390. let xmin=Math.min(...positionX);
  391. let ymax=Math.max(...positiony);
  392. console.log(positionX);
  393. // console.log( positionX)
  394. // console.log(44444)
  395. // console.log( positiony)
  396. nnum.value++;
  397. // let x=Math.floor(Math.random() * 91) + 10;
  398. // let y=Math.floor(Math.random() * 91) + 10;
  399. let name='模块'+nnum.value
  400. let item= {
  401. id:nnum.value.toString(),
  402. data: { label:name },
  403. position: { x:xmin , y:ymax},
  404. style: { backgroundColor: bgcolor.value, width: '200px',height: '200px' },
  405. //children: [],
  406. }
  407. if(Nested.value.length!=0){
  408. Nested2.value=[];
  409. for(let i=0;i<Nested.value.length;i++){
  410. Nested2.value.push(Nested.value[i]);
  411. }
  412. console.log(Nested2.value);
  413. nodesitem.value= Nested2.value.map(node => {
  414. if(node.parentNode==undefined){
  415. node.isParent=false;
  416. node.parentNode=item.id;
  417. console.log( node.parentNode)
  418. node.position.x= node.position.x /2;
  419. node.position.y= node.position.y/2;
  420. node.expandParent=true;
  421. //positionxy=node.position;
  422. // node.extent='parent';
  423. // return node;
  424. }
  425. return node;
  426. })
  427. }
  428. nodes.value.push(item)
  429. for(let i=0;i<nodesitem.value.length;i++){
  430. console.log(nodesitem.value[i]);
  431. nodes.value.push(nodesitem.value[i])
  432. }
  433. console.log( nodes.value);
  434. }
  435. function onNodeDoubleClick(e) {
  436. noid.value = e.node;
  437. const nowid = e.node.data.wid;
  438. console.log( e.node.data.name);
  439. if (e.node.data.name == 'optimizer') {
  440. emit('optimizerfalse',{name:'优化器',wid:nowid});
  441. }else if(e.node.data.name=="optimizer1"){
  442. emit('optimizerfalse',{name:'进化优化器',wid:nowid});
  443. }else if(e.node.data.name=="optimizer3"){
  444. emit('optimizerfalse',{name:'代理优化器',wid:nowid});
  445. }else if(e.node.data.name=="Xfoil"){
  446. emit('optimizerfalse',{name:'Xfoil',wid:nowid});
  447. }else if(e.node.data.name=="optimizer2"){
  448. emit('optimizerfalse',{name:'梯度优化器',wid:nowid});
  449. }else if(e.node.data.name=="CST"){
  450. emit('optimizerfalse',{name:'CST',wid:nowid});
  451. }else if(e.node.data.name=="ADflow"|| e.node.data.name=="RAE2822"){
  452. emit('optimizerfalse',{name:'ADflow',wid:nowid});
  453. }else if(e.node.data.name=="FFD"){
  454. emit('optimizerfalse',{name:'FFD',wid:nowid});
  455. }else if(e.node.data.name=="TACS"){
  456. emit('optimizerfalse',{name:'TACS',wid:nowid});
  457. }else if(e.node.data.name=="参数化"){
  458. emit('optimizerfalse',{name:'参数化',wid:nowid});
  459. }else if(e.node.data.name=="气动分析"){
  460. emit('optimizerfalse',{name:'气动分析',wid:nowid});
  461. }else if(onPythonlist.value.some(item => e.node.data.name.includes(item))){
  462. emit('optimizerfalse',{name:e.node.data.name,wid:nowid});
  463. }else if(e.node.data.name=="CATIA"){
  464. emit('optimizerfalse',{name:'CATIA',wid:nowid});
  465. }else if(e.node.data.name=="FSI"){
  466. emit('optimizerfalse',{name:'FSI',wid:nowid});
  467. }else if(e.node.data.name=="Flight"){
  468. emit('optimizerfalse',{name:'Flight',wid:nowid});
  469. }
  470. }
  471. let onPythonlist=ref(['Python','Branin','Rosenbrock','Rastrigin','G9','Forrester']);
  472. let previousEdge = null; // 用于保存上一个选中的边缘
  473. // 监听线
  474. function onEdgeClick(e) {
  475. console.log('Edge Click', e.edge);
  476. // 如果已经有选中的边缘
  477. if (seledge.value) {
  478. // 恢复上一个选中边缘的样式
  479. if (previousEdge) {
  480. previousEdge.style = {
  481. ...previousEdge.style,
  482. stroke: previousEdge.originalColor, // 恢复原始颜色
  483. strokeWidth: previousEdge.originalWidth,// 恢复原始宽度
  484. };
  485. }
  486. }
  487. // 保存当前点击的边缘为选中边缘
  488. Edgeid.value = e.edge.id;
  489. seledge.value = e.edge;
  490. // 暂时更改当前选中边缘的样式
  491. seledge.value.originalColor = seledge.value.style.stroke; // 保存当前边缘的原始颜色
  492. seledge.value.originalWidth = seledge.value.style.strokeWidth; // 保存当前边缘的原始宽度
  493. seledge.value.style = {
  494. ...seledge.value.style,
  495. stroke: '#2267B1', // 设置选中边缘的颜色
  496. strokeWidth: 2, // 设置选中边缘的宽度
  497. };
  498. // 保存当前选中的边缘作为上一个选中边缘
  499. previousEdge = seledge.value;
  500. }
  501. let dataflowshow=ref(false);
  502. let youhuaFFDtable = ref([
  503. { name:'FFD参数(sample)', steamflag:1 }
  504. ])
  505. let youhuaCSTtable = ref([
  506. { name:"上表面CST参数(upper)", steamflag:1 },
  507. { name:'下表面CST参数(lower)', steamflag:1 }
  508. ])
  509. let steamtypeoptions = ref([
  510. { label:'约束条件', value:1 },
  511. { label:'优化目标', value:2 }
  512. ])
  513. let ADflowyouhuatable = ref([
  514. { code: "cl", name:'升力系数Cl', comtype:2 , steamflag:1, steamtype:1 },
  515. { code: "cd", name:'阻力系数Cd', comtype:2 , steamflag:1, steamtype:1 },
  516. { code: "cm", name:'力矩系数Cm', comtype:2 , steamflag:1, steamtype:1 },
  517. ])
  518. let Xfoilyouhuatable = ref([
  519. // { code:'', name:'升力系数Cl', comtype:2 , steamflag:1, steamtype:1 },
  520. // { code:'', name:'阻力系数Cd', comtype:2 , steamflag:1, steamtype:1 },
  521. // { code:'', name:'压阻力系数Cdp', comtype:2 , steamflag:1, steamtype:1 },
  522. // { code:'', name:'力矩系数Cm', comtype:2 , steamflag:1, steamtype:1 },
  523. // { code:'', name:'上表面转换点位置xtr-upper', comtype:2 , steamflag:1, steamtype:1 },
  524. // { code:'', name:'下表面转换点位置xtr-lower', comtype:2 , steamflag:1, steamtype:1 },
  525. ])
  526. let checkAll = ref(false);
  527. let isIndeterminate = ref(false);
  528. const dataTables = {
  529. '1': youhuaFFDtable,
  530. '2': youhuaCSTtable,
  531. '3': ADflowyouhuatable,
  532. '4': Xfoilyouhuatable
  533. };
  534. const handlecheckFlagchange = () => {
  535. const table = dataTables[datacontent.value];
  536. if (!table) return;
  537. let checkedCount = table.value.filter(item => Boolean(item.steamflag)).length;
  538. checkAll.value = checkedCount === table.value.length;
  539. isIndeterminate.value = checkedCount > 0 && checkedCount < table.value.length;
  540. };
  541. const handlecheckAllchange = (val) => {
  542. const table = dataTables[datacontent.value];
  543. if (!table) return;
  544. table.value.forEach(item => item.steamflag = val ? 1 : 0);
  545. isIndeterminate.value = false;
  546. };
  547. const confirmselection=()=>{
  548. const table = dataTables[datacontent.value];
  549. let checkedData = table.value.filter((item) => item.steamflag).map((item) => item.name);
  550. console.log('xuanzhongshuju:',checkedData);
  551. console.log('seledge:',seledge.value);
  552. seledge.value.label=checkedData.join('\n');
  553. if(datacontent.value === '4') {
  554. dataFlowsave();
  555. }else if(datacontent.value === '3') {
  556. dataFlowsave2();
  557. }
  558. dataflowshow.value = false;
  559. }
  560. const xfid = ref('')
  561. const adid = ref('')
  562. // 监听组件xfoil返回的xfid
  563. const handleXfid = (xfidFromB) => {
  564. xfid.value = xfidFromB.value;
  565. };
  566. const handleAdid = (adidFromB) => {
  567. adid.value = adidFromB.value;
  568. }
  569. let datacontent = ref('')
  570. function onEdgeDoubleClick(e) {
  571. console.log('Edge Double Click', e)
  572. seledge.value = e.edge;
  573. // dataflowshow.value = true
  574. console.log('qidian:',e.edge.sourceNode.data.name);
  575. console.log('zhongdian:',e.edge.targetNode.data.name);
  576. let qidian = e.edge.sourceNode.data.name;
  577. let zhongdian = e.edge.targetNode.data.name;
  578. let youhualist = ['optimizer','optimizer3','optimizer1','optimizer2'];
  579. if( youhualist.includes(qidian) && zhongdian ==='FFD' ){
  580. datacontent.value = '1';
  581. dataflowshow.value = true;
  582. }else if( youhualist.includes(qidian) && zhongdian ==='CST') {
  583. datacontent.value = '2';
  584. dataflowshow.value = true;
  585. }else if( qidian === 'ADflow' && youhualist.includes(zhongdian) ) {
  586. datacontent.value = '3';
  587. emitter.emit('requestGetadid',pid.value);
  588. console.log('adid:',adid.value)
  589. if(adid.value) {
  590. querydataFlow(adid);
  591. dataflowshow.value = true;
  592. }else{
  593. ElMessage.error('ADflow未初始化!')
  594. }
  595. }else if ( qidian === 'Xfoil' && youhualist.includes(zhongdian) ) {
  596. datacontent.value = '4';
  597. emitter.emit('requestGetxfid',pid.value);
  598. console.log('xfid:',xfid.value)
  599. if(xfid.value) {
  600. querydataFlow(xfid);
  601. dataflowshow.value = true;
  602. }else{
  603. ElMessage.error('Xfoil未初始化!')
  604. }
  605. }
  606. console.log('leixing:',datacontent.value);
  607. nextTick(() => {
  608. handlecheckFlagchange();
  609. });
  610. }
  611. const querydataFlow = (comid) => {
  612. const params = {
  613. transCode: "MDO0052",
  614. pid: pid.value,
  615. comid: comid.value
  616. }
  617. request(params).then((res) => {
  618. if(datacontent.value === '3'){
  619. ADflowyouhuatable.value = res.params;
  620. handlecheckFlagchange();
  621. }
  622. if(datacontent.value === '4'){
  623. Xfoilyouhuatable.value = res.params;
  624. handlecheckFlagchange();
  625. }
  626. })
  627. .catch((err) => {
  628. ElMessage.error(err.returnMsg)
  629. })
  630. }
  631. const dataFlowsave = () => {
  632. const params = {
  633. transCode: "MDO0053",
  634. paramstr: convertToStringArray([],Xfoilyouhuatable.value),
  635. };
  636. request(params).then((res) => {
  637. ElMessage({
  638. message: res.returnMsg,
  639. type: 'success',
  640. })
  641. })
  642. .catch((err) => {
  643. ElMessage.error(err.returnMsg)
  644. })
  645. }
  646. const dataFlowsave2 = () => {
  647. console.log('ADflowyouhuatable:',ADflowyouhuatable.value);
  648. const params = {
  649. transCode: "MDO0053",
  650. paramstr: convertToStringArray([],ADflowyouhuatable.value),
  651. };
  652. request(params).then((res) => {
  653. ElMessage({
  654. message: res.returnMsg,
  655. type: 'success',
  656. })
  657. })
  658. .catch((err) => {
  659. ElMessage.error(err.returnMsg)
  660. })
  661. }
  662. const convertToStringArray = (result, Data) => {
  663. console.log('Data:', Data);
  664. // 安全检查 Data,确保它是一个数组
  665. if (!Array.isArray(Data)) {
  666. console.error('Data should be an array');
  667. return result; // 返回原 result 或者根据需要返回其他默认值
  668. }
  669. result = Data.map(row => {
  670. // 获取每一行的 `code`, `name`, `value` 和 `flag`
  671. const paramid = row.paramid ?? ' ';
  672. const steamflag = row.steamflag ?? '';
  673. const steamtype = row.steamtype ?? ' ';
  674. // 将字段连接为一个以逗号分隔的字符串
  675. return `${paramid},${steamflag},${steamtype}`;
  676. }).join(';'); // 每行之间用分号分隔
  677. return result;
  678. }
  679. function logEvent(name, event) {
  680. console.log(2222)
  681. }
  682. // 右键更改名字
  683. // const onContextMenu = (e) => {
  684. // e.preventDefault(); // 阻止浏览器默认的右键菜单
  685. // const node = e.target.closest(".vue-flow__node");
  686. // console.log("nodes:",nodes);
  687. // console.log("node:",node);
  688. // if (node) {
  689. // // 获取当前右键点击的节点
  690. // const nodeId = node.getAttribute("data-id");
  691. // const clickedNode = nodes.value.find((n) => n.id === nodeId);
  692. // console.log("clickedNode:",clickedNode);
  693. // if (clickedNode) {
  694. // noid.value = clickedNode;
  695. // changeName.value.name = clickedNode.label; // 将当前节点的label放入弹窗中
  696. // changeNameshow.value = true; // 显示弹窗
  697. // }
  698. // }
  699. // };
  700. onInit((vueFlowInstance) => {
  701. vueFlowInstance.fitView()
  702. })
  703. onNodeDragStop(({ event, nodes, node }) => {
  704. console.log(nodes)
  705. Nested.value=nodes;
  706. console.log('Node Drag Stop', { event, nodes, node })
  707. })
  708. // onConnect((connection) => {
  709. // addEdges(connection)
  710. // console.log('Connection', connection)
  711. // })
  712. function updatePos() {
  713. nodes.value = nodes.value.map((node) => {
  714. return {
  715. ...node,
  716. position: {
  717. x: Math.random() * 400,
  718. y: Math.random() * 400,
  719. },
  720. }
  721. })
  722. }
  723. function removeEdge(id) {
  724. id = Edgeid.value;
  725. const wid = seledge.value.data.wid;
  726. vueFlowRef.value.removeEdges(id);
  727. deleteflow(wid);
  728. seledge.value=null;
  729. console.log('msg:',datatree.value);
  730. }
  731. // 触摸
  732. const onDrop1=(event)=>{
  733. onDrop(event);
  734. emitter.emit('doSomethingEvent');
  735. }
  736. function removeNode(id) {
  737. id = noid.value.id;
  738. const wid = noid.value.data.wid;
  739. console.log('removeNodewid:',wid);
  740. if(datatree.value==undefined){
  741. if(nodes.value.length>0){
  742. for (let i = 0; i <nodes.value.length; i++) {
  743. if(id==nodes.value[i].id){
  744. console.log(44444)
  745. console.log( nodes.value[i]);
  746. nodes.value.splice(i, 1)
  747. deleteflow(wid);
  748. }
  749. }
  750. console.log( nodes.value);
  751. }
  752. }else{
  753. if(nodes.value.length>0){
  754. for (let i = 0; i <nodes.value.length; i++) {
  755. if(id==nodes.value[i].id){
  756. console.log( nodes.value[i]);
  757. nodes.value.splice(i, 1)
  758. }
  759. }
  760. }
  761. for (let i = 0; i <datatree.value[0].children.length; i++) {
  762. if(id.includes(datatree.value[0].children[i].Text)){
  763. for (let j = 0; j <datatree.value[0].children[i].children.length; j++) {
  764. if(id==datatree.value[0].children[i].children[j].id){
  765. //datatree.value[0].children.splice(datatree.value[0].children[i].children[j], 1);
  766. datatree.value[0].children[i].children.splice(j, 1);
  767. deleteflow(wid);
  768. vueFlowRef.value.removeNodes(id);
  769. removeRelatedEdges(id);
  770. }
  771. }
  772. }
  773. }
  774. }
  775. }
  776. function removeRelatedEdges(nodeId) {
  777. //过滤出与该节点相关的连线
  778. const relatedEdges = edges.value.filter(
  779. (edge) => edge.source === nodeId || edge.target === nodeId
  780. );
  781. // 遍历删除关联的连线
  782. relatedEdges.forEach((edge) => {
  783. if (edge.data && edge.data.wid) {
  784. deleteflow(edge.data.wid);
  785. }
  786. });
  787. }
  788. // 删除提示
  789. const confirmDelete = () => {
  790. ElMessageBox.confirm(
  791. '确定要删除全部吗?删除后不可恢复!',
  792. '删除确认',
  793. {
  794. confirmButtonText: '确定',
  795. cancelButtonText: '取消',
  796. type: 'warning',
  797. }
  798. )
  799. .then(() => {
  800. removeall();
  801. })
  802. .catch(() => {
  803. ElMessage({
  804. type: 'info',
  805. message: '已取消删除',
  806. })
  807. })
  808. }
  809. function removeall() {
  810. try {
  811. const allnodes = nodes.value;
  812. const allEdges = edges.value;
  813. for (let i = 0; i < allnodes.length; i++) {
  814. if (allnodes[i].data.wid) {
  815. deleteflow(allnodes[i].data.wid);
  816. }
  817. }
  818. for (let i = 0; i < allEdges.length; i++) {
  819. if (allEdges[i].data.wid) {
  820. deleteflow(allEdges[i].data.wid);
  821. }
  822. }
  823. nodes.value = []
  824. edges.value = []
  825. Nested2.value = []
  826. Nested.value = []
  827. // 判断 datatree 是否为空或未定义
  828. if (!datatree.value || datatree.value.length === 0 || !datatree.value[0]?.children) {
  829. console.warn('datatree 数据为空或未定义')
  830. ElMessage({
  831. type: 'warning',
  832. message: '没有数据可以删除'
  833. })
  834. return
  835. }
  836. // 清空 datatree 的 children
  837. for (let i = 0; i < datatree.value[0].children.length; i++) {
  838. if (datatree.value[0]?.children[i]?.children) {
  839. datatree.value[0].children[i].children = []
  840. }
  841. }
  842. } catch (error) {
  843. console.error('删除失败:', error)
  844. ElMessage({
  845. type: 'error',
  846. message: '删除过程中出错'
  847. })
  848. }
  849. }
  850. // 流查询
  851. const queryflow = () => {
  852. const params = {
  853. transCode: 'MDO0057',
  854. pid: pid.value,
  855. }
  856. request(params)
  857. .then((res) => {
  858. console.log(res);
  859. })
  860. .catch((err) => {
  861. ElMessage.error(err.returnMsg)
  862. })
  863. }
  864. // 保存流
  865. const saveflow = async (pid,wid, uid, type, fromuid, touid) => {
  866. const params = {
  867. transCode: 'MDO0058',
  868. pid: pid || '',
  869. wid: wid || '', // 流ID
  870. uid: uid || '',
  871. type: type || '',
  872. fromuid: fromuid || '',
  873. touid: touid || '',
  874. };
  875. try {
  876. // 直接使用 await 等待 request 返回
  877. const res = await request(params);
  878. return res.wid; // 返回 wid
  879. } catch (err) {
  880. // 处理错误
  881. ElMessage.error(err.returnMsg || '保存流程失败');
  882. }
  883. };
  884. // async function saveFlowExample() {
  885. // try {
  886. // const wid = await saveflow('flow123', 'user001', 'process', 'fromA', 'toB');
  887. // console.log('返回的 wid:', wid);
  888. // } catch (err) {
  889. // console.error('保存流程失败:', err.message);
  890. // }
  891. // }
  892. // saveFlowExample();
  893. // 删除流
  894. const deleteflow = (nowid) => {
  895. const params = {
  896. transCode: 'MDO0059',
  897. wid: nowid,
  898. }
  899. request(params)
  900. .then((res) => {
  901. console.log(res);
  902. })
  903. .catch((err) => {
  904. ElMessage.error('删除流程失败')
  905. })
  906. }
  907. async function logToObject1() {
  908. let obj = { nodes: toObject().nodes,edges:toObject().edges };
  909. mergedObj.value=JSON.stringify(obj);
  910. try {
  911. const container = vueFlowRef.value.$el;
  912. const canvas = await html2canvas(container);
  913. const img = canvas.toDataURL('image/png');
  914. // 创建一个图片元素并设置src属性为转换后的图片数据
  915. vueflowimg.value=img
  916. if(vueflowimg.value!=''){
  917. console.log("进入了")
  918. addflow();
  919. }
  920. // 添加到DOM中或者做其他操作
  921. } catch (error) {
  922. console.error('转换出错:', error);
  923. }
  924. }
  925. //添加接口
  926. const addflow = () => {
  927. const savedObj = JSON.parse(sessionStorage.getItem("objlist"));
  928. const stypeValue = savedObj ? savedObj.stype : '';
  929. // 用于首页主界面切换数据一致
  930. const updateobjlist = {
  931. pid: pid.value,
  932. name: newobj.value.name,
  933. remark: newobj.value.description,
  934. image: vueflowimg.value,
  935. isshare: '1',
  936. flow: mergedObj.value,
  937. stype: stypeValue,
  938. }
  939. sessionStorage.setItem("objlist",JSON.stringify(updateobjlist));
  940. console.log("打印stypeValue:",stypeValue);
  941. const params = {
  942. transCode: 'MDO0002',
  943. pid: pid.value,
  944. name: newobj.value.name,
  945. remark:newobj.value.description,
  946. image:vueflowimg.value,
  947. isshare:'1',
  948. flow:mergedObj.value ,
  949. stype:stypeValue,
  950. }
  951. console.log(params);
  952. request(params)
  953. .then((res) => {
  954. console.log(res);
  955. ElMessage({
  956. message: '工程保存成功',
  957. type: 'success',
  958. })
  959. })
  960. .catch((err) => {
  961. ElMessage.error(err.returnMsg)
  962. })
  963. }
  964. /**
  965. * Resets the current viewport transformation (zoom & pan)
  966. */
  967. function resetTransform() {
  968. setViewport({ x: 0, y: 0, zoom: 1 })
  969. }
  970. function toggleDarkMode() {
  971. dark.value = !dark.value;
  972. if(dark.value){
  973. iconcolor.value='#fff'
  974. }else{
  975. iconcolor.value='#000'
  976. }
  977. }
  978. onMounted(() => {
  979. setTimeout(function() {
  980. getroter();
  981. }, 1500);
  982. // childfun();
  983. if (vueFlowRef.value) {
  984. vueFlowRef.value.$el.addEventListener('click', (event) => {
  985. // 确保点击的不是边缘
  986. if (seledge.value && !event.target.closest('.vue-flow__edge')) {
  987. // 恢复选中边缘的原始样式
  988. seledge.value.style = {
  989. ...seledge.value.style,
  990. stroke: seledge.value.originalColor,
  991. strokeWidth: previousEdge?.originalWidth || 1, // 恢复原始宽度
  992. };
  993. // 清空选中的边缘
  994. seledge.value = null;
  995. Edgeid.value = null;
  996. previousEdge = null;
  997. }
  998. });
  999. }
  1000. emitter.on("xfidFromxfoil", handleXfid);
  1001. emitter.on("adidFromadflow", handleAdid);
  1002. });
  1003. onUnmounted(() => {
  1004. emitter.off('child2Data');
  1005. emitter.off("xfidFromxfoil", handleXfid);
  1006. emitter.off("adidFromadflow", handleAdid);
  1007. });
  1008. // 获取链接
  1009. const getroter=()=>{
  1010. //datatree.value[0].children=[];
  1011. let objlist=JSON.parse(sessionStorage.getItem("objlist"));
  1012. if(objlist.flow!=''){
  1013. let nodesflow=JSON.parse(objlist.flow)
  1014. nodes.value=nodesflow.nodes;
  1015. edges.value=nodesflow.edges;
  1016. }
  1017. newobj.value.name=objlist.name;
  1018. newobj.value.description=objlist.remark;
  1019. pid.value=objlist.pid;
  1020. // let item={
  1021. // id:'1-0',
  1022. // label: objlist.name,
  1023. // img:r2,
  1024. // }
  1025. // console.log( datatree.value[0].children);
  1026. // datatree.value[0].children.push(item);
  1027. }
  1028. //改变线的粗
  1029. const linestrokeWidth=(num)=>{
  1030. if(seledge.value){
  1031. seledge.value.style.strokeWidth = num;
  1032. }
  1033. linenum.value=num;
  1034. // console.log(seledge.value);
  1035. // let edgearr= toObject().edges;
  1036. // for (let i = 0; i <= edgearr.length-1; i++) {
  1037. // edgearr[i].style.strokeWidth = num;
  1038. // }
  1039. console.log(num)
  1040. //addEdges(edgearr);
  1041. }
  1042. //改变线的颜色
  1043. const changeAllEdgesColor = (color1) => {
  1044. console.log('yanse:', color1);
  1045. linecolor.value = color1;
  1046. // 找到当前选中的边缘
  1047. if (seledge.value) {
  1048. // 更新该边缘的颜色,不修改宽度
  1049. const updatedEdge = {
  1050. ...seledge.value,
  1051. style: {
  1052. ...seledge.value.style,
  1053. stroke: linecolor.value, // 只修改颜色
  1054. strokeWidth: 1, // 保持选中样式的宽度(如果想要恢复原始宽度,设置为 1)
  1055. }
  1056. };
  1057. // 更新 edges 数组
  1058. const updatedEdges = edges.value.map(edge =>
  1059. edge.id === seledge.value.id ? updatedEdge : edge
  1060. );
  1061. edges.value = updatedEdges; // 持久化修改后的边缘
  1062. }
  1063. };
  1064. watch(() => seledge.value, (newItems, oldItems) => {
  1065. if(seledge.value!=null){
  1066. // seledge.value.style.stroke = linecolor.value;
  1067. }
  1068. });
  1069. defineExpose({changeAllEdgesColor,linestrokeWidth,getroter,onSelection,bgcolorfunc,logToObject1});
  1070. </script>
  1071. <style>
  1072. /* .vue-flow__edge.selected .vue-flow__edge-path, .vue-flow__edge:focus .vue-flow__edge-path, .vue-flow__edge:focus-visible .vue-flow__edge-path {
  1073. stroke: #555 !important;
  1074. } */
  1075. .vue-flow__edge:focus .vue-flow__edge-path, .vue-flow__edge:focus-visible .vue-flow__edge-path {
  1076. stroke: #555 !important;
  1077. }
  1078. .vue-flow__edge {
  1079. text-align: left;
  1080. /* 设置edges的左对齐 */
  1081. }
  1082. .vue-flow__edge-text {
  1083. transform: translateY(-10px); /* 将 label 向上偏移 */
  1084. background: transparent !important;
  1085. font-size: 8px;
  1086. font-family: 'Microsoft YaHei';
  1087. color: #333333;
  1088. }
  1089. .vue-flow__edge-textbg {
  1090. fill: transparent !important; /* 将背景设置为透明 */
  1091. }
  1092. #contextMenu {
  1093. display: none;
  1094. position: absolute;
  1095. background-color: #fff;
  1096. border-radius: 5px;
  1097. padding: 10px;
  1098. text-align: center;
  1099. color: black;
  1100. font-size: 14px;
  1101. font-weight: 400;
  1102. cursor: pointer;
  1103. z-index: 99999;
  1104. }
  1105. .vue-flow__node-default.selectable:hover,
  1106. .vue-flow__node-input.selectable:hover,
  1107. .vue-flow__node-output.selectable:hover {
  1108. box-shadow: none;
  1109. }
  1110. panel {
  1111. cursor: pointer;
  1112. position: absolute;
  1113. z-index: 100000;
  1114. }
  1115. .remove {
  1116. background: #fff;
  1117. color: #666;
  1118. margin: 0 10px;
  1119. font-size: 12px;
  1120. }
  1121. .vue-flow__node-default, .vue-flow__node-input, .vue-flow__node-output{
  1122. /* width: auto !important; */
  1123. border: none;
  1124. background-color: rgba(0,0,0,0);
  1125. }
  1126. .node-content {
  1127. cursor: move; /* 更改鼠标光标表示可拖动 */
  1128. }
  1129. .vue-flow__node {
  1130. cursor: move;
  1131. }
  1132. /* 禁用文本选中效果 */
  1133. .left_main * {
  1134. -webkit-user-select: none; /* Safari */
  1135. -moz-user-select: none; /* Firefox */
  1136. -ms-user-select: none; /* IE10+/Edge */
  1137. user-select: none; /* Standard syntax */
  1138. }
  1139. .lableaniu{
  1140. font-size: 12px;
  1141. background-color: #ddd;
  1142. padding: 4px 16px;
  1143. /* margin-top: -17px; */
  1144. margin-left: 5px;
  1145. margin-top: 0px;
  1146. border-radius: 1px;
  1147. }
  1148. .vue-flow__controls-button svg{
  1149. max-width: 16px;
  1150. max-height: 16px;
  1151. }
  1152. .field{
  1153. display: flex;
  1154. }
  1155. </style>