index.vue 35 KB

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