xdesigner.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653
  1. <template>
  2. <div :class="['mainpage', theme]" v-contextmenu:contextmenu>
  3. <el-container>
  4. <el-header>
  5. <Header v-if="canvas" :canvasdata="canvas.data" @onMessage="onMessage" @onMenu="onMenu" />
  6. </el-header>
  7. <el-main id="content" style="overflow: hidden">
  8. <el-collapse-transition>
  9. <div>
  10. <!--v-show="show3"-->
  11. <div id="left">
  12. <Pens />
  13. </div>
  14. </div>
  15. </el-collapse-transition>
  16. <div id="middle">
  17. <div id="rootElement" ref="rootElement" class="full"></div>
  18. <!-- <i
  19. style="position: fixed;top: 50%;color:blue"
  20. :class="show3?'el-icon-d-arrow-left':'el-icon-d-arrow-right'"
  21. @click="show3 = !show3"
  22. ></i>
  23. <svg
  24. v-show="canvasGrid=='true'"
  25. width="100%"
  26. height="100%"
  27. xmlns="http://www.w3.org/2000/svg"
  28. >
  29. <defs>
  30. <pattern id="grid" width="10" height="10" patternUnits="userSpaceOnUse">
  31. <path d="M 10 0 L 0 0 0 10" fill="none" stroke="#f3f3f3" stroke-width="1" />
  32. </pattern>
  33. </defs>
  34. <rect width="100%" height="100%" fill="url(#grid)" />
  35. </svg>
  36. -->
  37. </div>
  38. <div id="right">
  39. <Filepropertys v-if="!isSelected" @onMessage="onMessage" :canvas="canvas" :page="page" />
  40. <Nodepropertys v-if="isSelected" @onMessage="onMessage" :canvas="canvas" :nodes="nodes" />
  41. </div>
  42. </el-main>
  43. </el-container>
  44. <v-contextmenu ref="contextmenu" :theme="theme">
  45. <v-contextmenu-item @click="OnTop">置顶</v-contextmenu-item>
  46. <v-contextmenu-item @click="OnBottom">置底</v-contextmenu-item>
  47. <v-contextmenu-item divider></v-contextmenu-item>
  48. <v-contextmenu-item
  49. @click="OnCombine"
  50. v-if="nodes && nodes.pens && (nodes.pens.length > 1)"
  51. >组合</v-contextmenu-item>
  52. <v-contextmenu-item @click="OnUncombine" v-if="nodes && nodes.pen && nodes.pen.children">取消组合</v-contextmenu-item>
  53. <v-contextmenu-item @click="OnPenLock">{{ penLocked ? '解锁' : '锁定' }}</v-contextmenu-item>
  54. <v-contextmenu-item divider></v-contextmenu-item>
  55. <v-contextmenu-item @click="OnDelete">删除</v-contextmenu-item>
  56. <v-contextmenu-item divider></v-contextmenu-item>
  57. <v-contextmenu-item @click="OnUndo">撤销 Ctrl+Z</v-contextmenu-item>
  58. <v-contextmenu-item @click="OnRedo">重做 Ctrl+Shift+Z</v-contextmenu-item>
  59. <v-contextmenu-item divider></v-contextmenu-item>
  60. <v-contextmenu-item @click="OnCut">剪切 Ctrl+X</v-contextmenu-item>
  61. <v-contextmenu-item @click="OnCopy">复制 Ctrl+C</v-contextmenu-item>
  62. <v-contextmenu-item @click="OnPaste">粘贴 Ctrl+V</v-contextmenu-item>
  63. <v-contextmenu-item divider></v-contextmenu-item>
  64. </v-contextmenu>
  65. </div>
  66. </template>
  67. <style scoped>
  68. a:hover {
  69. color: #1bb5f5;
  70. }
  71. i {
  72. font-size: 35px;
  73. }
  74. .mainpage {
  75. width: 100vw;
  76. height: 100vh;
  77. position: fixed;
  78. }
  79. .el-container {
  80. height: 100%;
  81. }
  82. .el-header {
  83. background-color: #545c64;
  84. text-align: center;
  85. }
  86. .el-main {
  87. flex: 1;
  88. padding: 0px !important;
  89. }
  90. .full {
  91. height: 100%;
  92. flex: 1;
  93. border: 1px solid #000;
  94. }
  95. #content {
  96. display: flex;
  97. width: 100%;
  98. height: 200px;
  99. }
  100. #left {
  101. flex: 0 0 220px;
  102. height: 100%;
  103. width: 246px;
  104. overflow-y: auto;
  105. }
  106. #middle {
  107. flex: 1;
  108. }
  109. #right {
  110. height: 100%;
  111. width: 246px;
  112. overflow-y: auto;
  113. }
  114. </style>
  115. <script>
  116. import { Topology, Lock, Options, Pen } from '@topology/core'
  117. import * as FileSaver from 'file-saver'
  118. import { Store } from 'le5le-store'
  119. import Header from "@/components/head"//"/components/head.vue"
  120. import Nodepropertys from "@/components/nodepropertys"
  121. import Filepropertys from "@/components/filepropertys"
  122. import Pens from "@/components/pens.vue"
  123. import store from '@/store/index'
  124. import { monitor } from '@/network/network'
  125. import "@/assets/js/canvas2svg"
  126. export default {
  127. components: {
  128. Header, Pens, Nodepropertys, Filepropertys
  129. },
  130. props: {
  131. theme: String,
  132. },
  133. data()
  134. {
  135. return {
  136. //show3: true,
  137. canvas: null,
  138. canvasGrid: false,
  139. canvasOptions: {
  140. rotateCursor: '/img/rotate.cur',
  141. color: this.$store.state.designsetting.elementColor,
  142. font: {
  143. color: this.$store.state.designsetting.elementColor,
  144. }
  145. },
  146. contextmenu: {},
  147. editFilename: false,
  148. page: {
  149. belong: '',
  150. path: "",
  151. name: "",
  152. elementColor: this.$store.state.designsetting.elementColor,
  153. paintColor: this.$store.state.designsetting.paintColor,
  154. },
  155. nodes: {
  156. pen: {},
  157. pens: []
  158. },
  159. isSelected: false,
  160. penLocked: false,
  161. cpPresetColors: [
  162. '#1890ff',
  163. '#096dd9',
  164. '#bae7ff',
  165. '#52c41a',
  166. '#3fad09',
  167. '#c6ebb4',
  168. '#faad14',
  169. '#d9a116',
  170. '#fff6dd',
  171. '#f50000',
  172. '#ff0000',
  173. '#ffc2c5',
  174. '#fa541c',
  175. '#531dab',
  176. '#314659',
  177. '#777777'
  178. ],
  179. }
  180. },
  181. mounted()
  182. {
  183. this.init();
  184. },
  185. computed: {
  186. event()
  187. {
  188. return this.$store.state.event.event
  189. }
  190. },
  191. methods: {
  192. handleSubmenuShow(vm, placement)
  193. {
  194. //console.log(vm, placement)
  195. },
  196. init()
  197. {
  198. window.scrollTo(0, 0);
  199. this.canvasOptions.on = this.onMessage;
  200. this.canvas = new Topology('rootElement', this.canvasOptions);
  201. this.canvasHeight = this.canvas.canvas.height;
  202. this.$store.state.canvas = this.canvas;
  203. },
  204. async open()
  205. {
  206. if (!this.$route.query.id) {
  207. return
  208. }
  209. if (data && data.id) {
  210. canvas.open(data.data)
  211. }
  212. },
  213. onMenu(docmd, value)
  214. {
  215. switch (docmd) {
  216. case "new":
  217. { this.OnNew(); } break;
  218. case "open":
  219. {
  220. this.OnNew();
  221. this.OnOpen();
  222. /**
  223. 0 /预处理
  224. 1 /预处理/监视页面/蒸发塔.json
  225. 2 /加料/测试页面.json
  226. 3 /主页面.json
  227. */
  228. } break;
  229. case "save":
  230. {
  231. /*
  232. 此处页面保存到服务端
  233. 文件名称:(前后端做好防注入准备,只允许输入中英文数字和转义的反斜杠)
  234. 然后界面需要提供一个树形结构或者列表,方便用户操作(路劲添加或者路劲选择)
  235. */
  236. this.OnSaveToServer();
  237. } break;
  238. case "donwloadJson":
  239. { this.OnSave(); } break;
  240. case "donwloadPNG":
  241. { this.OnSavePng(); } break;
  242. case "donwloadSVG":
  243. { this.OnSaveSvg(); } break;
  244. case "undo":
  245. { this.OnUndo(); } break;
  246. case "redo":
  247. { this.OnRedo(); } break;
  248. case "cut":
  249. { this.OnCut(); } break;
  250. case "copy":
  251. { this.OnCopy(); } break;
  252. case "paste":
  253. { this.OnPaste(); } break;
  254. case "scale":
  255. { this.canvas.scaleTo(value); }
  256. break;
  257. case "lock":
  258. { this.OnLock(value); } break;
  259. }
  260. },
  261. // 右侧输入框编辑状态时点击编辑区域其他元素,onMessage执行后才执行onUpdateProps方法,通过setTimeout让onUpdateProps先执行
  262. onMessage(event, value)
  263. {
  264. setTimeout(() =>
  265. {
  266. switch (event) {
  267. case 'node':
  268. case 'addNode':
  269. case 'line':
  270. case 'addLine':
  271. {
  272. this.nodes = {
  273. pen: value
  274. };
  275. this.isSelected = true;
  276. this.locked = value.locked;
  277. }
  278. break;
  279. case 'multi':
  280. {
  281. this.locked = true;
  282. if (value && value.length) {
  283. this.nodes = {
  284. pens: value
  285. };
  286. this.isSelected = true;
  287. for (const item of value) {
  288. if (!item.locked) {
  289. this.locked = false;
  290. break;
  291. }
  292. }
  293. }
  294. }
  295. break;
  296. case 'space':
  297. {
  298. this.nodes = {
  299. pen: {},
  300. pens: []
  301. }
  302. this.isSelected = false;
  303. }
  304. break;
  305. case 'moveOut':
  306. {
  307. this.$refs.rootElement.scrollLeft += 10;
  308. this.$refs.rootElement.scrollTop += 10;
  309. }
  310. break;
  311. case 'resize':
  312. {
  313. if (value) {
  314. this.canvasHeight = value.height;
  315. }
  316. }
  317. break;
  318. case 'locked':
  319. {
  320. Store.set('locked', value);
  321. break;
  322. }
  323. case "refresh":
  324. {
  325. this.canvas.render();
  326. } break;
  327. case "setElementColor":
  328. {
  329. if (this.canvasOptions) {
  330. this.canvasOptions.color = value;
  331. this.canvasOptions.font.color = value;
  332. }
  333. if (this.canvas && this.canvas.options) {
  334. this.canvas.options.color = value;
  335. this.canvas.options.font.color = value;
  336. }
  337. } break;
  338. case "showGrid":
  339. {
  340. if (this.canvas.data) {
  341. this.canvas.data.grid = value;
  342. this.canvasGrid = value;
  343. }
  344. } break;
  345. default:
  346. {
  347. } break;
  348. }
  349. //console.log('onMessage:', event, value);
  350. }, 50)
  351. window.n = this.nodes;
  352. },
  353. /***
  354. @param lockNumber {number} 0,1,2(可选) 0:解除lock, 1:可选中不可拖动 2:不可选中 */
  355. OnLock(lockNumber)
  356. {
  357. this.canvas.lock(lockNumber);
  358. },
  359. onUpdateProps(node)
  360. {
  361. canvas.updateProps(node)
  362. },
  363. OnNew(data)
  364. {
  365. this.canvas.open();
  366. this.canvas.data.lineName = this.$store.state.designsetting.lineStyleName;
  367. this.canvas.data.fromArrowType = this.$store.state.designsetting.fromArrowType;
  368. this.canvas.data.toArrowType = this.$store.state.designsetting.toArrowType;
  369. this.canvas.data.bkColor = this.$store.state.designsetting.paintColor;
  370. this.canvas.render();
  371. this.canvas.scaleTo(1);
  372. },
  373. OnOpen(data)
  374. {
  375. this.OnReplace(data)
  376. },
  377. OnReplace()
  378. {
  379. const input = document.createElement('input')
  380. input.type = 'file'
  381. input.onchange = event =>
  382. {
  383. const elem = event.target;
  384. if (elem.files && elem.files[0]) {
  385. if (elem.files[0].name.indexOf('.json') > 0) {
  386. this.OpenJson(elem.files[0]);
  387. } else {
  388. //this.openZip(elem.files[0]);
  389. }
  390. }
  391. }
  392. input.click()
  393. },
  394. OpenJson(file)
  395. {
  396. const name = file.name.replace('.json', '');
  397. const reader = new FileReader();
  398. reader.onload = (e) =>
  399. {
  400. const text = e.target.result + '';
  401. try {
  402. const data = JSON.parse(text);
  403. data.lineName = this.$store.state.designsetting.lineStyleName;
  404. data.fromArrowType = this.$store.state.designsetting.fromArrowType;
  405. data.toArrowType = this.$store.state.designsetting.toArrowType;
  406. data.name = name;
  407. //data.scale = 1;
  408. if (data.bkColor) {
  409. this.$store.dispatch('setCanvasBackColor', data.bkColor);
  410. }
  411. this.canvas.open(data);
  412. this.canvas.scaleTo(1);
  413. this.canvas.data.bkColor = data.bkColor;
  414. this.$store.dispatch('setCanvas', this.canvas);
  415. } catch (e) {
  416. console.log(e)
  417. }
  418. };
  419. reader.readAsText(file);
  420. },
  421. OnSaveToServer()
  422. {
  423. //#region ---------------------------------------不可为空
  424. if (!this.page.belong || this.page.belong == "") {
  425. this.$message.error("无法保存,页面-所属环节不可为空");
  426. this.onMessage('space');
  427. return;
  428. }
  429. if (!this.page.path || this.page.path == "") {
  430. this.$message.error("无法保存页面,页面路径不可为空");
  431. this.onMessage('space');
  432. return;
  433. }
  434. if (!this.page.name || this.page.name == "") {
  435. this.$message.error("无法保存页面,页面名称不可为空");
  436. this.onMessage('space');
  437. return;
  438. }
  439. //#endregion ------------------------------------------
  440. /***
  441. * 第一步 保存页面
  442. * 第二部 保存路径结构
  443. */
  444. var newPathSummary = JSON.parse(JSON.stringify(this.$store.state.pathSummary));
  445. if (newPathSummary[this.page.belong].page) {
  446. var newPageNode = {
  447. "name": this.page.name,
  448. "path": this.page.path
  449. }
  450. newPathSummary[this.page.belong].page[this.page.name] = newPageNode;
  451. var MonitorJson = {
  452. id: 0,
  453. pagepath: "/",
  454. pagejson: JSON.stringify(newPathSummary),
  455. }
  456. monitor.setJsonFile(MonitorJson, res =>
  457. {
  458. this.$message.success("提交文件结构" + res.data.msg);
  459. });
  460. this.canvas.data.page = this.page;
  461. var MonitorPageJson = {
  462. id: 0,
  463. pagepath: newPathSummary[this.page.belong].path + this.page.path,
  464. pagejson: JSON.stringify(this.canvas.data),
  465. }
  466. monitor.setJsonFile(MonitorPageJson, res =>
  467. {
  468. this.$message.success("提交页面" + res.data.msg);
  469. });
  470. }
  471. },
  472. OnSave(data)
  473. {
  474. FileSaver.saveAs(
  475. new Blob([JSON.stringify(this.canvas.data)], {
  476. type: 'text/plain;charset=utf-8'
  477. }),
  478. `X.json`
  479. )
  480. },
  481. OnSavePng(data)
  482. {
  483. this.canvas.saveAsImage('GyeeX.png')
  484. },
  485. OnSaveSvg(data)
  486. {
  487. const ctx = new C2S(this.canvas.canvas.width + 200, this.canvas.canvas.height + 200)
  488. for (const item of this.canvas.data.pens) {
  489. item.render(ctx)
  490. }
  491. let mySerializedSVG = ctx.getSerializedSvg()
  492. mySerializedSVG = mySerializedSVG.replace(
  493. '<defs/>',
  494. `<defs>
  495. <style type="text/css">
  496. @font-face {
  497. font-family: 'topology';
  498. src: url('http://at.alicdn.com/t/font_1331132_h688rvffmbc.ttf?t=1569311680797') format('truetype');
  499. }
  500. </style>
  501. </defs>`
  502. )
  503. mySerializedSVG = mySerializedSVG.replace(/--le5le--/g, '&#x')
  504. const urlObject = window.URL || window
  505. const export_blob = new Blob([mySerializedSVG])
  506. const url = urlObject.createObjectURL(export_blob)
  507. const a = document.createElement('a')
  508. a.setAttribute('download', 'XGyee.svg')
  509. a.setAttribute('href', url)
  510. const evt = document.createEvent('MouseEvents')
  511. evt.initEvent('click', true, true)
  512. a.dispatchEvent(evt)
  513. },
  514. OnUndo()
  515. {
  516. this.canvas.undo()
  517. },
  518. OnRedo()
  519. {
  520. this.canvas.redo()
  521. },
  522. OnCopy()
  523. {
  524. this.canvas.copy()
  525. },
  526. OnCut()
  527. {
  528. this.canvas.cut()
  529. },
  530. OnPaste()
  531. {
  532. this.canvas.paste()
  533. },
  534. onContextMenu(event)
  535. {
  536. event.preventDefault()
  537. event.stopPropagation()
  538. if (event.clientY + 360 < document.body.clientHeight) {
  539. this.contextmenu = {
  540. left: event.clientX + 'px',
  541. top: event.clientY + 'px'
  542. }
  543. } else {
  544. this.contextmenu = {
  545. left: event.clientX + 'px',
  546. bottom: document.body.clientHeight - event.clientY + 'px'
  547. }
  548. }
  549. },
  550. OnTop()
  551. {
  552. if (!this.nodes) {
  553. return;
  554. }
  555. if (this.nodes.pen) {
  556. this.canvas.top(this.nodes.pen);
  557. }
  558. if (this.nodes.pens) {
  559. for (const item of this.nodes.pens) {
  560. this.canvas.top(item);
  561. }
  562. }
  563. this.canvas.render();
  564. },
  565. OnBottom()
  566. {
  567. if (!this.nodes) {
  568. return;
  569. }
  570. if (this.nodes.pen) {
  571. this.canvas.bottom(this.nodes.pen);
  572. }
  573. if (this.nodes.pens) {
  574. for (const item of this.nodes.pens) {
  575. this.canvas.bottom(item);
  576. }
  577. }
  578. this.canvas.render();
  579. },
  580. OnCombine(stand = false)
  581. {
  582. if (!this.nodes || !this.nodes.pens || this.nodes.pens.length < 2) {
  583. return;
  584. }
  585. this.canvas.combine(this.nodes.pens, stand);
  586. },
  587. OnUncombine()
  588. {
  589. if (!this.nodes || !this.nodes.pen || this.nodes.pen.type) {
  590. return;
  591. }
  592. this.canvas.uncombine(this.nodes.pen);
  593. this.canvas.render();
  594. }
  595. ,//penLocked
  596. OnPenLock()
  597. {
  598. this.penLocked = !this.penLocked;
  599. if (this.nodes.pen) {
  600. this.canvas.lockPens([this.nodes.pen], this.penLocked ? 1 : 0);
  601. }
  602. if (this.nodes.pens) {
  603. this.canvas.lockPens(this.nodes.pens, this.penLocked ? 1 : 0);
  604. }
  605. this.canvas.render(true);
  606. },
  607. OnDelete()
  608. {
  609. this.canvas.delete();
  610. },
  611. destroyed()
  612. {
  613. canvas.destroy()
  614. },
  615. },
  616. }
  617. </script>