index.vue 36 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280
  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 == '优化器') {
  440. emit('optimizerfalse',{name:'优化器',wid:nowid});
  441. }else if(e.node.data.name=="进化优化器"){
  442. emit('optimizerfalse',{name:'进化优化器',wid:nowid});
  443. }else if(e.node.data.name=="代理优化器"){
  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=="梯度优化器"){
  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. { code: " ", name:'体积', comtype:2 , steamflag:1, steamtype:1 },
  518. ])
  519. let Xfoilyouhuatable = ref([
  520. // { code:'', name:'升力系数Cl', comtype:2 , steamflag:1, steamtype:1 },
  521. // { code:'', name:'阻力系数Cd', comtype:2 , steamflag:1, steamtype:1 },
  522. // { code:'', name:'压阻力系数Cdp', comtype:2 , steamflag:1, steamtype:1 },
  523. // { code:'', name:'力矩系数Cm', comtype:2 , steamflag:1, steamtype:1 },
  524. // { code:'', name:'上表面转换点位置xtr-upper', comtype:2 , steamflag:1, steamtype:1 },
  525. // { code:'', name:'下表面转换点位置xtr-lower', comtype:2 , steamflag:1, steamtype:1 },
  526. ])
  527. let checkAll = ref(false);
  528. let isIndeterminate = ref(false);
  529. const dataTables = {
  530. '1': youhuaFFDtable,
  531. '2': youhuaCSTtable,
  532. '3': ADflowyouhuatable,
  533. '4': Xfoilyouhuatable
  534. };
  535. const handlecheckFlagchange = () => {
  536. const table = dataTables[datacontent.value];
  537. if (!table) return;
  538. let checkedCount = table.value.filter(item => Boolean(item.steamflag)).length;
  539. checkAll.value = checkedCount === table.value.length;
  540. isIndeterminate.value = checkedCount > 0 && checkedCount < table.value.length;
  541. };
  542. const handlecheckAllchange = (val) => {
  543. const table = dataTables[datacontent.value];
  544. if (!table) return;
  545. table.value.forEach(item => item.steamflag = val ? 1 : 0);
  546. isIndeterminate.value = false;
  547. };
  548. const confirmselection=()=>{
  549. const table = dataTables[datacontent.value];
  550. let checkedData = table.value.filter((item) => item.steamflag).map((item) => item.name);
  551. console.log('xuanzhongshuju:',checkedData);
  552. console.log('seledge:',seledge.value);
  553. seledge.value.label=checkedData.join('\n');
  554. if(datacontent.value === '4') {
  555. dataFlowsave();
  556. }
  557. dataflowshow.value = false;
  558. }
  559. const xfid = ref('')
  560. const adid = ref('')
  561. // 监听组件xfoil返回的xfid
  562. const handleXfid = (xfidFromB) => {
  563. xfid.value = xfidFromB.value;
  564. };
  565. const handleAdid = (adidFromB) => {
  566. adid.value = adidFromB.value;
  567. }
  568. let datacontent = ref('')
  569. function onEdgeDoubleClick(e) {
  570. console.log('Edge Double Click', e)
  571. seledge.value = e.edge;
  572. // dataflowshow.value = true
  573. console.log('qidian:',e.edge.sourceNode.data.name);
  574. console.log('zhongdian:',e.edge.targetNode.data.name);
  575. let qidian = e.edge.sourceNode.data.name;
  576. let zhongdian = e.edge.targetNode.data.name;
  577. let youhualist = ['优化器','代理优化器','进化优化器','梯度优化器'];
  578. if( youhualist.includes(qidian) && zhongdian ==='FFD' ){
  579. datacontent.value = '1';
  580. dataflowshow.value = true;
  581. }else if( youhualist.includes(qidian) && zhongdian ==='CST') {
  582. datacontent.value = '2';
  583. dataflowshow.value = true;
  584. }else if( qidian === 'ADflow' && youhualist.includes(zhongdian) ) {
  585. datacontent.value = '3';
  586. emitter.emit('requestGetadid',pid.value);
  587. console.log('adid:',adid.value)
  588. if(adid.value) {
  589. // querydataFlow(adid);
  590. dataflowshow.value = true;
  591. }else{
  592. ElMessage.error('ADflow未初始化!')
  593. }
  594. }else if ( qidian === 'Xfoil' && youhualist.includes(zhongdian) ) {
  595. datacontent.value = '4';
  596. emitter.emit('requestGetxfid',pid.value);
  597. console.log('xfid:',xfid.value)
  598. if(xfid.value) {
  599. querydataFlow(xfid);
  600. dataflowshow.value = true;
  601. }else{
  602. ElMessage.error('Xfoil未初始化!')
  603. }
  604. }
  605. console.log('leixing:',datacontent.value);
  606. nextTick(() => {
  607. handlecheckFlagchange();
  608. });
  609. }
  610. const querydataFlow = (comid) => {
  611. const params = {
  612. transCode: "MDO0052",
  613. pid: pid.value,
  614. comid: comid.value
  615. }
  616. request(params).then((res) => {
  617. if(datacontent.value === '4'){
  618. Xfoilyouhuatable.value = res.params;
  619. handlecheckFlagchange();
  620. }
  621. })
  622. .catch((err) => {
  623. ElMessage.error(err.returnMsg)
  624. })
  625. }
  626. const dataFlowsave = () => {
  627. const params = {
  628. transCode: "MDO0053",
  629. paramstr: convertToStringArray([],Xfoilyouhuatable.value),
  630. };
  631. request(params).then((res) => {
  632. ElMessage({
  633. message: res.returnMsg,
  634. type: 'success',
  635. })
  636. })
  637. .catch((err) => {
  638. ElMessage.error(err.returnMsg)
  639. })
  640. }
  641. const convertToStringArray = (result, Data) => {
  642. console.log('Data:', Data);
  643. // 安全检查 Data,确保它是一个数组
  644. if (!Array.isArray(Data)) {
  645. console.error('Data should be an array');
  646. return result; // 返回原 result 或者根据需要返回其他默认值
  647. }
  648. result = Data.map(row => {
  649. // 获取每一行的 `code`, `name`, `value` 和 `flag`
  650. const paramid = row.paramid ?? ' ';
  651. const steamflag = row.steamflag ?? '';
  652. const steamtype = row.steamtype ?? ' ';
  653. // 将字段连接为一个以逗号分隔的字符串
  654. return `${paramid},${steamflag},${steamtype}`;
  655. }).join(';'); // 每行之间用分号分隔
  656. return result;
  657. }
  658. function logEvent(name, event) {
  659. console.log(2222)
  660. }
  661. // 右键更改名字
  662. // const onContextMenu = (e) => {
  663. // e.preventDefault(); // 阻止浏览器默认的右键菜单
  664. // const node = e.target.closest(".vue-flow__node");
  665. // console.log("nodes:",nodes);
  666. // console.log("node:",node);
  667. // if (node) {
  668. // // 获取当前右键点击的节点
  669. // const nodeId = node.getAttribute("data-id");
  670. // const clickedNode = nodes.value.find((n) => n.id === nodeId);
  671. // console.log("clickedNode:",clickedNode);
  672. // if (clickedNode) {
  673. // noid.value = clickedNode;
  674. // changeName.value.name = clickedNode.label; // 将当前节点的label放入弹窗中
  675. // changeNameshow.value = true; // 显示弹窗
  676. // }
  677. // }
  678. // };
  679. onInit((vueFlowInstance) => {
  680. vueFlowInstance.fitView()
  681. })
  682. onNodeDragStop(({ event, nodes, node }) => {
  683. console.log(nodes)
  684. Nested.value=nodes;
  685. console.log('Node Drag Stop', { event, nodes, node })
  686. })
  687. // onConnect((connection) => {
  688. // addEdges(connection)
  689. // console.log('Connection', connection)
  690. // })
  691. function updatePos() {
  692. nodes.value = nodes.value.map((node) => {
  693. return {
  694. ...node,
  695. position: {
  696. x: Math.random() * 400,
  697. y: Math.random() * 400,
  698. },
  699. }
  700. })
  701. }
  702. function removeEdge(id) {
  703. id = Edgeid.value;
  704. const wid = seledge.value.data.wid;
  705. vueFlowRef.value.removeEdges(id);
  706. deleteflow(wid);
  707. seledge.value=null;
  708. console.log('msg:',datatree.value);
  709. }
  710. // 触摸
  711. const onDrop1=(event)=>{
  712. onDrop(event);
  713. emitter.emit('doSomethingEvent');
  714. }
  715. function removeNode(id) {
  716. id = noid.value.id;
  717. const wid = noid.value.data.wid;
  718. console.log('removeNodewid:',wid);
  719. if(datatree.value==undefined){
  720. if(nodes.value.length>0){
  721. for (let i = 0; i <nodes.value.length; i++) {
  722. if(id==nodes.value[i].id){
  723. console.log(44444)
  724. console.log( nodes.value[i]);
  725. nodes.value.splice(i, 1)
  726. deleteflow(wid);
  727. }
  728. }
  729. console.log( nodes.value);
  730. }
  731. }else{
  732. if(nodes.value.length>0){
  733. for (let i = 0; i <nodes.value.length; i++) {
  734. if(id==nodes.value[i].id){
  735. console.log( nodes.value[i]);
  736. nodes.value.splice(i, 1)
  737. }
  738. }
  739. }
  740. for (let i = 0; i <datatree.value[0].children.length; i++) {
  741. if(id.includes(datatree.value[0].children[i].Text)){
  742. for (let j = 0; j <datatree.value[0].children[i].children.length; j++) {
  743. if(id==datatree.value[0].children[i].children[j].id){
  744. //datatree.value[0].children.splice(datatree.value[0].children[i].children[j], 1);
  745. datatree.value[0].children[i].children.splice(j, 1);
  746. deleteflow(wid);
  747. vueFlowRef.value.removeNodes(id);
  748. removeRelatedEdges(id);
  749. }
  750. }
  751. }
  752. }
  753. }
  754. }
  755. function removeRelatedEdges(nodeId) {
  756. //过滤出与该节点相关的连线
  757. const relatedEdges = edges.value.filter(
  758. (edge) => edge.source === nodeId || edge.target === nodeId
  759. );
  760. // 遍历删除关联的连线
  761. relatedEdges.forEach((edge) => {
  762. if (edge.data && edge.data.wid) {
  763. deleteflow(edge.data.wid);
  764. }
  765. });
  766. }
  767. // 删除提示
  768. const confirmDelete = () => {
  769. ElMessageBox.confirm(
  770. '确定要删除全部吗?删除后不可恢复!',
  771. '删除确认',
  772. {
  773. confirmButtonText: '确定',
  774. cancelButtonText: '取消',
  775. type: 'warning',
  776. }
  777. )
  778. .then(() => {
  779. removeall();
  780. })
  781. .catch(() => {
  782. ElMessage({
  783. type: 'info',
  784. message: '已取消删除',
  785. })
  786. })
  787. }
  788. function removeall() {
  789. try {
  790. const allnodes = nodes.value;
  791. const allEdges = edges.value;
  792. for (let i = 0; i < allnodes.length; i++) {
  793. if (allnodes[i].data.wid) {
  794. deleteflow(allnodes[i].data.wid);
  795. }
  796. }
  797. for (let i = 0; i < allEdges.length; i++) {
  798. if (allEdges[i].data.wid) {
  799. deleteflow(allEdges[i].data.wid);
  800. }
  801. }
  802. nodes.value = []
  803. edges.value = []
  804. Nested2.value = []
  805. Nested.value = []
  806. // 判断 datatree 是否为空或未定义
  807. if (!datatree.value || datatree.value.length === 0 || !datatree.value[0]?.children) {
  808. console.warn('datatree 数据为空或未定义')
  809. ElMessage({
  810. type: 'warning',
  811. message: '没有数据可以删除'
  812. })
  813. return
  814. }
  815. // 清空 datatree 的 children
  816. for (let i = 0; i < datatree.value[0].children.length; i++) {
  817. if (datatree.value[0]?.children[i]?.children) {
  818. datatree.value[0].children[i].children = []
  819. }
  820. }
  821. } catch (error) {
  822. console.error('删除失败:', error)
  823. ElMessage({
  824. type: 'error',
  825. message: '删除过程中出错'
  826. })
  827. }
  828. }
  829. // 流查询
  830. const queryflow = () => {
  831. const params = {
  832. transCode: 'MDO0057',
  833. pid: pid.value,
  834. }
  835. request(params)
  836. .then((res) => {
  837. console.log(res);
  838. })
  839. .catch((err) => {
  840. ElMessage.error(err.returnMsg)
  841. })
  842. }
  843. // 保存流
  844. const saveflow = async (pid,wid, uid, type, fromuid, touid) => {
  845. const params = {
  846. transCode: 'MDO0058',
  847. pid: pid || '',
  848. wid: wid || '', // 流ID
  849. uid: uid || '',
  850. type: type || '',
  851. fromuid: fromuid || '',
  852. touid: touid || '',
  853. };
  854. try {
  855. // 直接使用 await 等待 request 返回
  856. const res = await request(params);
  857. return res.wid; // 返回 wid
  858. } catch (err) {
  859. // 处理错误
  860. ElMessage.error(err.returnMsg || '保存流程失败');
  861. }
  862. };
  863. // async function saveFlowExample() {
  864. // try {
  865. // const wid = await saveflow('flow123', 'user001', 'process', 'fromA', 'toB');
  866. // console.log('返回的 wid:', wid);
  867. // } catch (err) {
  868. // console.error('保存流程失败:', err.message);
  869. // }
  870. // }
  871. // saveFlowExample();
  872. // 删除流
  873. const deleteflow = (nowid) => {
  874. const params = {
  875. transCode: 'MDO0059',
  876. wid: nowid,
  877. }
  878. request(params)
  879. .then((res) => {
  880. console.log(res);
  881. })
  882. .catch((err) => {
  883. ElMessage.error('删除流程失败')
  884. })
  885. }
  886. async function logToObject1() {
  887. let obj = { nodes: toObject().nodes,edges:toObject().edges };
  888. mergedObj.value=JSON.stringify(obj);
  889. try {
  890. const container = vueFlowRef.value.$el;
  891. const canvas = await html2canvas(container);
  892. const img = canvas.toDataURL('image/png');
  893. // 创建一个图片元素并设置src属性为转换后的图片数据
  894. vueflowimg.value=img
  895. if(vueflowimg.value!=''){
  896. console.log("进入了")
  897. addflow();
  898. }
  899. // 添加到DOM中或者做其他操作
  900. } catch (error) {
  901. console.error('转换出错:', error);
  902. }
  903. }
  904. //添加接口
  905. const addflow = () => {
  906. const savedObj = JSON.parse(sessionStorage.getItem("objlist"));
  907. const stypeValue = savedObj ? savedObj.stype : '';
  908. // 用于首页主界面切换数据一致
  909. const updateobjlist = {
  910. pid: pid.value,
  911. name: newobj.value.name,
  912. remark: newobj.value.description,
  913. image: vueflowimg.value,
  914. isshare: '1',
  915. flow: mergedObj.value,
  916. stype: stypeValue,
  917. }
  918. sessionStorage.setItem("objlist",JSON.stringify(updateobjlist));
  919. console.log("打印stypeValue:",stypeValue);
  920. const params = {
  921. transCode: 'MDO0002',
  922. pid: pid.value,
  923. name: newobj.value.name,
  924. remark:newobj.value.description,
  925. image:vueflowimg.value,
  926. isshare:'1',
  927. flow:mergedObj.value ,
  928. stype:stypeValue,
  929. }
  930. console.log(params);
  931. request(params)
  932. .then((res) => {
  933. console.log(res);
  934. ElMessage({
  935. message: '工程保存成功',
  936. type: 'success',
  937. })
  938. })
  939. .catch((err) => {
  940. ElMessage.error(err.returnMsg)
  941. })
  942. }
  943. /**
  944. * Resets the current viewport transformation (zoom & pan)
  945. */
  946. function resetTransform() {
  947. setViewport({ x: 0, y: 0, zoom: 1 })
  948. }
  949. function toggleDarkMode() {
  950. dark.value = !dark.value;
  951. if(dark.value){
  952. iconcolor.value='#fff'
  953. }else{
  954. iconcolor.value='#000'
  955. }
  956. }
  957. onMounted(() => {
  958. setTimeout(function() {
  959. getroter();
  960. }, 1500);
  961. // childfun();
  962. if (vueFlowRef.value) {
  963. vueFlowRef.value.$el.addEventListener('click', (event) => {
  964. // 确保点击的不是边缘
  965. if (seledge.value && !event.target.closest('.vue-flow__edge')) {
  966. // 恢复选中边缘的原始样式
  967. seledge.value.style = {
  968. ...seledge.value.style,
  969. stroke: seledge.value.originalColor,
  970. strokeWidth: previousEdge?.originalWidth || 1, // 恢复原始宽度
  971. };
  972. // 清空选中的边缘
  973. seledge.value = null;
  974. Edgeid.value = null;
  975. previousEdge = null;
  976. }
  977. });
  978. }
  979. emitter.on("xfidFromxfoil", handleXfid);
  980. emitter.on("adidFromadflow", handleAdid);
  981. });
  982. onUnmounted(() => {
  983. emitter.off('child2Data');
  984. emitter.off("xfidFromxfoil", handleXfid);
  985. emitter.off("adidFromadflow", handleAdid);
  986. });
  987. // 获取链接
  988. const getroter=()=>{
  989. //datatree.value[0].children=[];
  990. let objlist=JSON.parse(sessionStorage.getItem("objlist"));
  991. if(objlist.flow!=''){
  992. let nodesflow=JSON.parse(objlist.flow)
  993. nodes.value=nodesflow.nodes;
  994. edges.value=nodesflow.edges;
  995. }
  996. newobj.value.name=objlist.name;
  997. newobj.value.description=objlist.remark;
  998. pid.value=objlist.pid;
  999. // let item={
  1000. // id:'1-0',
  1001. // label: objlist.name,
  1002. // img:r2,
  1003. // }
  1004. // console.log( datatree.value[0].children);
  1005. // datatree.value[0].children.push(item);
  1006. }
  1007. //改变线的粗
  1008. const linestrokeWidth=(num)=>{
  1009. if(seledge.value){
  1010. seledge.value.style.strokeWidth = num;
  1011. }
  1012. linenum.value=num;
  1013. // console.log(seledge.value);
  1014. // let edgearr= toObject().edges;
  1015. // for (let i = 0; i <= edgearr.length-1; i++) {
  1016. // edgearr[i].style.strokeWidth = num;
  1017. // }
  1018. console.log(num)
  1019. //addEdges(edgearr);
  1020. }
  1021. //改变线的颜色
  1022. const changeAllEdgesColor = (color1) => {
  1023. console.log('yanse:', color1);
  1024. linecolor.value = color1;
  1025. // 找到当前选中的边缘
  1026. if (seledge.value) {
  1027. // 更新该边缘的颜色,不修改宽度
  1028. const updatedEdge = {
  1029. ...seledge.value,
  1030. style: {
  1031. ...seledge.value.style,
  1032. stroke: linecolor.value, // 只修改颜色
  1033. strokeWidth: 1, // 保持选中样式的宽度(如果想要恢复原始宽度,设置为 1)
  1034. }
  1035. };
  1036. // 更新 edges 数组
  1037. const updatedEdges = edges.value.map(edge =>
  1038. edge.id === seledge.value.id ? updatedEdge : edge
  1039. );
  1040. edges.value = updatedEdges; // 持久化修改后的边缘
  1041. }
  1042. };
  1043. watch(() => seledge.value, (newItems, oldItems) => {
  1044. if(seledge.value!=null){
  1045. // seledge.value.style.stroke = linecolor.value;
  1046. }
  1047. });
  1048. defineExpose({changeAllEdgesColor,linestrokeWidth,getroter,onSelection,bgcolorfunc,logToObject1});
  1049. </script>
  1050. <style>
  1051. /* .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 {
  1052. stroke: #555 !important;
  1053. } */
  1054. .vue-flow__edge:focus .vue-flow__edge-path, .vue-flow__edge:focus-visible .vue-flow__edge-path {
  1055. stroke: #555 !important;
  1056. }
  1057. .vue-flow__edge {
  1058. text-align: left;
  1059. /* 设置edges的左对齐 */
  1060. }
  1061. .vue-flow__edge-text {
  1062. transform: translateY(-10px); /* 将 label 向上偏移 */
  1063. background: transparent !important;
  1064. font-size: 8px;
  1065. font-family: 'Microsoft YaHei';
  1066. color: #333333;
  1067. }
  1068. .vue-flow__edge-textbg {
  1069. fill: transparent !important; /* 将背景设置为透明 */
  1070. }
  1071. #contextMenu {
  1072. display: none;
  1073. position: absolute;
  1074. background-color: #fff;
  1075. border-radius: 5px;
  1076. padding: 10px;
  1077. text-align: center;
  1078. color: black;
  1079. font-size: 14px;
  1080. font-weight: 400;
  1081. cursor: pointer;
  1082. z-index: 99999;
  1083. }
  1084. .vue-flow__node-default.selectable:hover,
  1085. .vue-flow__node-input.selectable:hover,
  1086. .vue-flow__node-output.selectable:hover {
  1087. box-shadow: none;
  1088. }
  1089. panel {
  1090. cursor: pointer;
  1091. position: absolute;
  1092. z-index: 100000;
  1093. }
  1094. .remove {
  1095. background: #fff;
  1096. color: #666;
  1097. margin: 0 10px;
  1098. font-size: 12px;
  1099. }
  1100. .vue-flow__node-default, .vue-flow__node-input, .vue-flow__node-output{
  1101. /* width: auto !important; */
  1102. border: none;
  1103. background-color: rgba(0,0,0,0);
  1104. }
  1105. .node-content {
  1106. cursor: move; /* 更改鼠标光标表示可拖动 */
  1107. }
  1108. .vue-flow__node {
  1109. cursor: move;
  1110. }
  1111. /* 禁用文本选中效果 */
  1112. .left_main * {
  1113. -webkit-user-select: none; /* Safari */
  1114. -moz-user-select: none; /* Firefox */
  1115. -ms-user-select: none; /* IE10+/Edge */
  1116. user-select: none; /* Standard syntax */
  1117. }
  1118. .lableaniu{
  1119. font-size: 12px;
  1120. background-color: #ddd;
  1121. padding: 4px 16px;
  1122. /* margin-top: -17px; */
  1123. margin-left: 5px;
  1124. margin-top: 0px;
  1125. border-radius: 1px;
  1126. }
  1127. .vue-flow__controls-button svg{
  1128. max-width: 16px;
  1129. max-height: 16px;
  1130. }
  1131. .field{
  1132. display: flex;
  1133. }
  1134. </style>