xdesigner.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  1. <template>
  2. <div class="mainpage">
  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. </div>
  45. </template>
  46. <style>
  47. a:hover {
  48. color: #1bb5f5;
  49. }
  50. i {
  51. font-size: 35px;
  52. }
  53. .mainpage {
  54. width: 100vw;
  55. height: 100vh;
  56. position: fixed;
  57. }
  58. .el-container {
  59. height: 100%;
  60. }
  61. .el-header {
  62. background-color: #545c64;
  63. text-align: center;
  64. }
  65. .el-main {
  66. flex: 1;
  67. padding: 0px !important;
  68. }
  69. .full {
  70. height: 100%;
  71. flex: 1;
  72. border: 1px solid #000;
  73. }
  74. #content {
  75. display: flex;
  76. width: 100%;
  77. height: 200px;
  78. }
  79. #left {
  80. flex: 0 0 220px;
  81. height: 100%;
  82. width: 246px;
  83. overflow-y: auto;
  84. }
  85. #middle {
  86. flex: 1;
  87. }
  88. #right {
  89. height: 100%;
  90. width: 246px;
  91. overflow-y: auto;
  92. }
  93. </style>
  94. <script>
  95. import { Topology, Lock, Options, Pen } from '@topology/core'
  96. import * as FileSaver from 'file-saver'
  97. import { Store } from 'le5le-store'
  98. import Header from "@/components/head"//"/components/head.vue"
  99. import Nodepropertys from "@/components/nodepropertys"
  100. import Filepropertys from "@/components/filepropertys"
  101. import Pens from "@/components/pens.vue"
  102. import store from '@/store/index'
  103. import "@/assets/js/canvas2svg"
  104. export default {
  105. components: {
  106. Header, Pens, Nodepropertys, Filepropertys
  107. },
  108. data()
  109. {
  110. return {
  111. //show3: true,
  112. canvas: null,
  113. canvasGrid: false,
  114. canvasOptions: {
  115. rotateCursor: '/img/rotate.cur',
  116. color: this.$store.state.designsetting.elementColor,
  117. font: {
  118. color: this.$store.state.designsetting.elementColor,
  119. }
  120. },
  121. contextmenu: {},
  122. editFilename: false,
  123. page: {
  124. path: "/",
  125. name: "",
  126. elementColor: this.$store.state.designsetting.elementColor,
  127. paintColor: this.$store.state.designsetting.paintColor,
  128. },
  129. nodes: {
  130. pen: {},
  131. pens: []
  132. },
  133. isSelected: false,
  134. cpPresetColors: [
  135. '#1890ff',
  136. '#096dd9',
  137. '#bae7ff',
  138. '#52c41a',
  139. '#3fad09',
  140. '#c6ebb4',
  141. '#faad14',
  142. '#d9a116',
  143. '#fff6dd',
  144. '#f50000',
  145. '#ff0000',
  146. '#ffc2c5',
  147. '#fa541c',
  148. '#531dab',
  149. '#314659',
  150. '#777777'
  151. ],
  152. }
  153. },
  154. created()
  155. {
  156. // if (process.client && window['echartsData']) {
  157. // for (let key in window['echartsData']) {
  158. // document.body.removeChild(window['echartsData'][key]).div
  159. // }
  160. // window['echartsData'] = {}
  161. // }
  162. //canvasRegister();
  163. // if (process.client) {
  164. // document.onclick = event =>
  165. // {
  166. // this.contextmenu = {
  167. // left: null,
  168. // top: null,
  169. // bottom: null
  170. // }
  171. // }
  172. // }
  173. },
  174. mounted()
  175. {
  176. this.init();
  177. },
  178. computed: {
  179. event()
  180. {
  181. return this.$store.state.event.event
  182. }
  183. },
  184. watch: {
  185. // event(curVal)
  186. // {
  187. // console.log("_________________________________________________!@!!!!!!!!!!!!!!!__________")
  188. // console.log(curVal);
  189. // if (this[curVal.name]) {
  190. // this[curVal.name](curVal.data)
  191. // }
  192. // },
  193. // $route(val)
  194. // {
  195. // this.open()
  196. // }
  197. },
  198. methods: {
  199. init()
  200. {
  201. window.scrollTo(0, 0);
  202. this.canvasOptions.on = this.onMessage;
  203. this.canvas = new Topology('rootElement', this.canvasOptions);
  204. this.canvasHeight = this.canvas.canvas.height;
  205. this.$store.state.canvas = this.canvas;
  206. window.v = this.canvas;
  207. window.s = this.$store;
  208. },
  209. async open()
  210. {
  211. if (!this.$route.query.id) {
  212. return
  213. }
  214. if (data && data.id) {
  215. canvas.open(data.data)
  216. }
  217. },
  218. onMenu(docmd, value)
  219. {
  220. console.log(docmd)
  221. console.log(value)
  222. switch (docmd) {
  223. case "new":
  224. { this.OnNew(); } break;
  225. case "open":
  226. {
  227. this.OnNew();
  228. this.OnOpen();
  229. /**
  230. 0 /预处理
  231. 1 /预处理/监视页面/蒸发塔.json
  232. 2 /加料/测试页面.json
  233. 3 /主页面.json
  234. */
  235. } break;
  236. case "save":
  237. {
  238. /*
  239. 此处页面保存到服务端
  240. 文件名称:(前后端做好防注入准备,只允许输入中英文数字和转义的反斜杠)
  241. 然后界面需要提供一个树形结构或者列表,方便用户操作(路劲添加或者路劲选择)
  242. */
  243. this.OnSave();
  244. } break;
  245. case "donwloadJson":
  246. { this.OnSave(); } break;
  247. case "donwloadPNG":
  248. { this.OnSavePng(); } break;
  249. case "donwloadSVG":
  250. { this.OnSaveSvg(); } break;
  251. case "undo":
  252. { this.OnUndo(); } break;
  253. case "redo":
  254. { this.OnRedo(); } break;
  255. case "cut":
  256. { this.OnCut(); } break;
  257. case "copy":
  258. { this.OnCopy(); } break;
  259. case "paste":
  260. { this.OnPaste(); } break;
  261. case "scale":
  262. { this.canvas.scaleTo(value); }
  263. break;
  264. case "lock":
  265. { this.OnLock(value); } break;
  266. }
  267. },
  268. // 右侧输入框编辑状态时点击编辑区域其他元素,onMessage执行后才执行onUpdateProps方法,通过setTimeout让onUpdateProps先执行
  269. onMessage(event, value)
  270. {
  271. setTimeout(() =>
  272. {
  273. switch (event) {
  274. case 'node':
  275. case 'addNode':
  276. case 'line':
  277. case 'addLine':
  278. {
  279. this.nodes = {
  280. pen: value
  281. };
  282. this.isSelected = true;
  283. this.locked = value.locked;
  284. }
  285. break;
  286. case 'multi':
  287. {
  288. this.locked = true;
  289. if (value && value.length) {
  290. this.nodes = {
  291. pens: value
  292. };
  293. this.isSelected = true;
  294. for (const item of value) {
  295. if (!item.locked) {
  296. this.locked = false;
  297. break;
  298. }
  299. }
  300. }
  301. }
  302. break;
  303. case 'space':
  304. {
  305. this.nodes = {
  306. pen: {},
  307. pens: []
  308. }
  309. this.isSelected = false;
  310. }
  311. break;
  312. case 'moveOut':
  313. {
  314. this.$refs.rootElement.scrollLeft += 10;
  315. this.$refs.rootElement.scrollTop += 10;
  316. }
  317. break;
  318. case 'resize':
  319. {
  320. if (value) {
  321. this.canvasHeight = value.height;
  322. }
  323. }
  324. break;
  325. case 'locked':
  326. {
  327. Store.set('locked', value);
  328. break;
  329. }
  330. case "refresh":
  331. {
  332. this.canvas.render();
  333. } break;
  334. case "setElementColor":
  335. {
  336. if (this.canvasOptions) {
  337. this.canvasOptions.color = value;
  338. this.canvasOptions.font.color = value;
  339. }
  340. if (this.canvas && this.canvas.options) {
  341. this.canvas.options.color = value;
  342. this.canvas.options.font.color = value;
  343. }
  344. } break;
  345. case "showGrid":
  346. {
  347. if (this.canvas.data) {
  348. this.canvas.data.grid = value;
  349. this.canvasGrid = value;
  350. }
  351. } break;
  352. default:
  353. {
  354. console.log('onMessage_____Default:', event, value);
  355. } break;
  356. }
  357. console.log('onMessage:', event, value);
  358. }, 50)
  359. window.n = this.nodes;
  360. },
  361. /***
  362. @param lockNumber {number} 0,1,2(可选) 0:解除lock, 1:可选中不可拖动 2:不可选中 */
  363. OnLock(lockNumber)
  364. {
  365. this.canvas.lock(lockNumber);
  366. // if (data.nodes && data.nodes.length) {
  367. // for (const item of data.nodes) {
  368. // if (!item.locked) {
  369. // locked = false
  370. // break
  371. // }
  372. // }
  373. // }
  374. },
  375. onUpdateProps(node)
  376. {
  377. console.log("canvas.updateProps(node)");
  378. // 如果是node属性改变,需要传入node,重新计算node相关属性值
  379. // 如果是line属性改变,无需传参
  380. canvas.updateProps(node)
  381. },
  382. OnNew(data)
  383. {
  384. this.canvas.open();
  385. this.canvas.data.lineName = this.$store.state.designsetting.lineStyleName;
  386. this.canvas.data.fromArrowType = this.$store.state.designsetting.fromArrowType;
  387. this.canvas.data.toArrowType = this.$store.state.designsetting.toArrowType;
  388. this.canvas.data.bkColor = this.$store.state.designsetting.paintColor;
  389. this.canvas.render();
  390. this.canvas.scaleTo(1);
  391. },
  392. OnOpen(data)
  393. {
  394. this.OnReplace(data)
  395. },
  396. OnReplace()
  397. {
  398. const input = document.createElement('input')
  399. input.type = 'file'
  400. input.onchange = event =>
  401. {
  402. const elem = event.target;
  403. if (elem.files && elem.files[0]) {
  404. if (elem.files[0].name.indexOf('.json') > 0) {
  405. this.OpenJson(elem.files[0]);
  406. } else {
  407. //this.openZip(elem.files[0]);
  408. }
  409. }
  410. }
  411. input.click()
  412. },
  413. OpenJson(file)
  414. {
  415. const name = file.name.replace('.json', '');
  416. const reader = new FileReader();
  417. reader.onload = (e) =>
  418. {
  419. const text = e.target.result + '';
  420. try {
  421. const data = JSON.parse(text);
  422. data.lineName = this.$store.state.designsetting.lineStyleName;
  423. data.fromArrowType = this.$store.state.designsetting.fromArrowType;
  424. data.toArrowType = this.$store.state.designsetting.toArrowType;
  425. data.name = name;
  426. //data.scale = 1;
  427. if (data.bkColor) {
  428. this.$store.dispatch('setCanvasBackColor', data.bkColor);
  429. }
  430. this.canvas.open(data);
  431. this.canvas.scaleTo(1);
  432. this.canvas.data.bkColor = data.bkColor;
  433. this.$store.dispatch('setCanvas', this.canvas);
  434. } catch (e) {
  435. //console.log(e)
  436. }
  437. };
  438. reader.readAsText(file);
  439. },
  440. OnSave(data)
  441. {
  442. FileSaver.saveAs(
  443. new Blob([JSON.stringify(this.canvas.data)], {
  444. type: 'text/plain;charset=utf-8'
  445. }),
  446. `X.json`
  447. )
  448. },
  449. OnSavePng(data)
  450. {
  451. this.canvas.saveAsImage('GyeeX.png')
  452. },
  453. OnSaveSvg(data)
  454. {
  455. const ctx = new C2S(this.canvas.canvas.width + 200, this.canvas.canvas.height + 200)
  456. for (const item of this.canvas.data.pens) {
  457. item.render(ctx)
  458. }
  459. let mySerializedSVG = ctx.getSerializedSvg()
  460. mySerializedSVG = mySerializedSVG.replace(
  461. '<defs/>',
  462. `<defs>
  463. <style type="text/css">
  464. @font-face {
  465. font-family: 'topology';
  466. src: url('http://at.alicdn.com/t/font_1331132_h688rvffmbc.ttf?t=1569311680797') format('truetype');
  467. }
  468. </style>
  469. </defs>`
  470. )
  471. mySerializedSVG = mySerializedSVG.replace(/--le5le--/g, '&#x')
  472. const urlObject = window.URL || window
  473. const export_blob = new Blob([mySerializedSVG])
  474. const url = urlObject.createObjectURL(export_blob)
  475. const a = document.createElement('a')
  476. a.setAttribute('download', 'XGyee.svg')
  477. a.setAttribute('href', url)
  478. const evt = document.createEvent('MouseEvents')
  479. evt.initEvent('click', true, true)
  480. a.dispatchEvent(evt)
  481. },
  482. OnUndo(data)
  483. {
  484. this.canvas.undo()
  485. },
  486. OnRedo(data)
  487. {
  488. this.canvas.redo()
  489. },
  490. OnCopy(data)
  491. {
  492. this.canvas.copy()
  493. },
  494. OnCut(data)
  495. {
  496. this.canvas.cut()
  497. },
  498. OnPaste(data)
  499. {
  500. this.canvas.paste()
  501. },
  502. onContextMenu(event)
  503. {
  504. event.preventDefault()
  505. event.stopPropagation()
  506. if (event.clientY + 360 < document.body.clientHeight) {
  507. this.contextmenu = {
  508. left: event.clientX + 'px',
  509. top: event.clientY + 'px'
  510. }
  511. } else {
  512. this.contextmenu = {
  513. left: event.clientX + 'px',
  514. bottom: document.body.clientHeight - event.clientY + 'px'
  515. }
  516. }
  517. },
  518. destroyed()
  519. {
  520. canvas.destroy()
  521. },
  522. },
  523. }
  524. </script>