123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653 |
- <template>
- <div :class="['mainpage', theme]" v-contextmenu:contextmenu>
- <el-container>
- <el-header>
- <Header v-if="canvas" :canvasdata="canvas.data" @onMessage="onMessage" @onMenu="onMenu" />
- </el-header>
- <el-main id="content" style="overflow: hidden">
- <el-collapse-transition>
- <div>
- <!--v-show="show3"-->
- <div id="left">
- <Pens />
- </div>
- </div>
- </el-collapse-transition>
- <div id="middle">
- <div id="rootElement" ref="rootElement" class="full"></div>
- <!-- <i
- style="position: fixed;top: 50%;color:blue"
- :class="show3?'el-icon-d-arrow-left':'el-icon-d-arrow-right'"
- @click="show3 = !show3"
- ></i>
-
- <svg
- v-show="canvasGrid=='true'"
- width="100%"
- height="100%"
- xmlns="http://www.w3.org/2000/svg"
- >
- <defs>
- <pattern id="grid" width="10" height="10" patternUnits="userSpaceOnUse">
- <path d="M 10 0 L 0 0 0 10" fill="none" stroke="#f3f3f3" stroke-width="1" />
- </pattern>
- </defs>
- <rect width="100%" height="100%" fill="url(#grid)" />
- </svg>
- -->
- </div>
- <div id="right">
- <Filepropertys v-if="!isSelected" @onMessage="onMessage" :canvas="canvas" :page="page" />
- <Nodepropertys v-if="isSelected" @onMessage="onMessage" :canvas="canvas" :nodes="nodes" />
- </div>
- </el-main>
- </el-container>
- <v-contextmenu ref="contextmenu" :theme="theme">
- <v-contextmenu-item @click="OnTop">置顶</v-contextmenu-item>
- <v-contextmenu-item @click="OnBottom">置底</v-contextmenu-item>
- <v-contextmenu-item divider></v-contextmenu-item>
- <v-contextmenu-item
- @click="OnCombine"
- v-if="nodes && nodes.pens && (nodes.pens.length > 1)"
- >组合</v-contextmenu-item>
- <v-contextmenu-item @click="OnUncombine" v-if="nodes && nodes.pen && nodes.pen.children">取消组合</v-contextmenu-item>
- <v-contextmenu-item @click="OnPenLock">{{ penLocked ? '解锁' : '锁定' }}</v-contextmenu-item>
- <v-contextmenu-item divider></v-contextmenu-item>
- <v-contextmenu-item @click="OnDelete">删除</v-contextmenu-item>
- <v-contextmenu-item divider></v-contextmenu-item>
- <v-contextmenu-item @click="OnUndo">撤销 Ctrl+Z</v-contextmenu-item>
- <v-contextmenu-item @click="OnRedo">重做 Ctrl+Shift+Z</v-contextmenu-item>
- <v-contextmenu-item divider></v-contextmenu-item>
- <v-contextmenu-item @click="OnCut">剪切 Ctrl+X</v-contextmenu-item>
- <v-contextmenu-item @click="OnCopy">复制 Ctrl+C</v-contextmenu-item>
- <v-contextmenu-item @click="OnPaste">粘贴 Ctrl+V</v-contextmenu-item>
- <v-contextmenu-item divider></v-contextmenu-item>
- </v-contextmenu>
- </div>
- </template>
- <style scoped>
- a:hover {
- color: #1bb5f5;
- }
- i {
- font-size: 35px;
- }
- .mainpage {
- width: 100vw;
- height: 100vh;
- position: fixed;
- }
- .el-container {
- height: 100%;
- }
- .el-header {
- background-color: #545c64;
- text-align: center;
- }
- .el-main {
- flex: 1;
- padding: 0px !important;
- }
- .full {
- height: 100%;
- flex: 1;
- border: 1px solid #000;
- }
- #content {
- display: flex;
- width: 100%;
- height: 200px;
- }
- #left {
- flex: 0 0 220px;
- height: 100%;
- width: 246px;
- overflow-y: auto;
- }
- #middle {
- flex: 1;
- }
- #right {
- height: 100%;
- width: 246px;
- overflow-y: auto;
- }
- </style>
- <script>
- import { Topology, Lock, Options, Pen } from '@topology/core'
- import * as FileSaver from 'file-saver'
- import { Store } from 'le5le-store'
- import Header from "@/components/head"//"/components/head.vue"
- import Nodepropertys from "@/components/nodepropertys"
- import Filepropertys from "@/components/filepropertys"
- import Pens from "@/components/pens.vue"
- import store from '@/store/index'
- import { monitor } from '@/network/network'
- import "@/assets/js/canvas2svg"
- export default {
- components: {
- Header, Pens, Nodepropertys, Filepropertys
- },
- props: {
- theme: String,
- },
- data()
- {
- return {
- //show3: true,
- canvas: null,
- canvasGrid: false,
- canvasOptions: {
- rotateCursor: '/img/rotate.cur',
- color: this.$store.state.designsetting.elementColor,
- font: {
- color: this.$store.state.designsetting.elementColor,
- }
- },
- contextmenu: {},
- editFilename: false,
- page: {
- belong: '',
- path: "",
- name: "",
- elementColor: this.$store.state.designsetting.elementColor,
- paintColor: this.$store.state.designsetting.paintColor,
- },
- nodes: {
- pen: {},
- pens: []
- },
- isSelected: false,
- penLocked: false,
- cpPresetColors: [
- '#1890ff',
- '#096dd9',
- '#bae7ff',
- '#52c41a',
- '#3fad09',
- '#c6ebb4',
- '#faad14',
- '#d9a116',
- '#fff6dd',
- '#f50000',
- '#ff0000',
- '#ffc2c5',
- '#fa541c',
- '#531dab',
- '#314659',
- '#777777'
- ],
- }
- },
- mounted()
- {
- this.init();
- },
- computed: {
- event()
- {
- return this.$store.state.event.event
- }
- },
- methods: {
- handleSubmenuShow(vm, placement)
- {
- //console.log(vm, placement)
- },
- init()
- {
- window.scrollTo(0, 0);
- this.canvasOptions.on = this.onMessage;
- this.canvas = new Topology('rootElement', this.canvasOptions);
- this.canvasHeight = this.canvas.canvas.height;
- this.$store.state.canvas = this.canvas;
- },
- async open()
- {
- if (!this.$route.query.id) {
- return
- }
- if (data && data.id) {
- canvas.open(data.data)
- }
- },
- onMenu(docmd, value)
- {
- switch (docmd) {
- case "new":
- { this.OnNew(); } break;
- case "open":
- {
- this.OnNew();
- this.OnOpen();
- /**
- 0 /预处理
- 1 /预处理/监视页面/蒸发塔.json
- 2 /加料/测试页面.json
- 3 /主页面.json
- */
- } break;
- case "save":
- {
- /*
- 此处页面保存到服务端
- 文件名称:(前后端做好防注入准备,只允许输入中英文数字和转义的反斜杠)
- 然后界面需要提供一个树形结构或者列表,方便用户操作(路劲添加或者路劲选择)
- */
- this.OnSaveToServer();
- } break;
- case "donwloadJson":
- { this.OnSave(); } break;
- case "donwloadPNG":
- { this.OnSavePng(); } break;
- case "donwloadSVG":
- { this.OnSaveSvg(); } break;
- case "undo":
- { this.OnUndo(); } break;
- case "redo":
- { this.OnRedo(); } break;
- case "cut":
- { this.OnCut(); } break;
- case "copy":
- { this.OnCopy(); } break;
- case "paste":
- { this.OnPaste(); } break;
- case "scale":
- { this.canvas.scaleTo(value); }
- break;
- case "lock":
- { this.OnLock(value); } break;
- }
- },
- // 右侧输入框编辑状态时点击编辑区域其他元素,onMessage执行后才执行onUpdateProps方法,通过setTimeout让onUpdateProps先执行
- onMessage(event, value)
- {
- setTimeout(() =>
- {
- switch (event) {
- case 'node':
- case 'addNode':
- case 'line':
- case 'addLine':
- {
- this.nodes = {
- pen: value
- };
- this.isSelected = true;
- this.locked = value.locked;
- }
- break;
- case 'multi':
- {
- this.locked = true;
- if (value && value.length) {
- this.nodes = {
- pens: value
- };
- this.isSelected = true;
- for (const item of value) {
- if (!item.locked) {
- this.locked = false;
- break;
- }
- }
- }
- }
- break;
- case 'space':
- {
- this.nodes = {
- pen: {},
- pens: []
- }
- this.isSelected = false;
- }
- break;
- case 'moveOut':
- {
- this.$refs.rootElement.scrollLeft += 10;
- this.$refs.rootElement.scrollTop += 10;
- }
- break;
- case 'resize':
- {
- if (value) {
- this.canvasHeight = value.height;
- }
- }
- break;
- case 'locked':
- {
- Store.set('locked', value);
- break;
- }
- case "refresh":
- {
- this.canvas.render();
- } break;
- case "setElementColor":
- {
- if (this.canvasOptions) {
- this.canvasOptions.color = value;
- this.canvasOptions.font.color = value;
- }
- if (this.canvas && this.canvas.options) {
- this.canvas.options.color = value;
- this.canvas.options.font.color = value;
- }
- } break;
- case "showGrid":
- {
- if (this.canvas.data) {
- this.canvas.data.grid = value;
- this.canvasGrid = value;
- }
- } break;
- default:
- {
- } break;
- }
- //console.log('onMessage:', event, value);
- }, 50)
- window.n = this.nodes;
- },
- /***
- @param lockNumber {number} 0,1,2(可选) 0:解除lock, 1:可选中不可拖动 2:不可选中 */
- OnLock(lockNumber)
- {
- this.canvas.lock(lockNumber);
- },
- onUpdateProps(node)
- {
- canvas.updateProps(node)
- },
- OnNew(data)
- {
- this.canvas.open();
- this.canvas.data.lineName = this.$store.state.designsetting.lineStyleName;
- this.canvas.data.fromArrowType = this.$store.state.designsetting.fromArrowType;
- this.canvas.data.toArrowType = this.$store.state.designsetting.toArrowType;
- this.canvas.data.bkColor = this.$store.state.designsetting.paintColor;
- this.canvas.render();
- this.canvas.scaleTo(1);
- },
- OnOpen(data)
- {
- this.OnReplace(data)
- },
- OnReplace()
- {
- const input = document.createElement('input')
- input.type = 'file'
- input.onchange = event =>
- {
- const elem = event.target;
- if (elem.files && elem.files[0]) {
- if (elem.files[0].name.indexOf('.json') > 0) {
- this.OpenJson(elem.files[0]);
- } else {
- //this.openZip(elem.files[0]);
- }
- }
- }
- input.click()
- },
- OpenJson(file)
- {
- const name = file.name.replace('.json', '');
- const reader = new FileReader();
- reader.onload = (e) =>
- {
- const text = e.target.result + '';
- try {
- const data = JSON.parse(text);
- data.lineName = this.$store.state.designsetting.lineStyleName;
- data.fromArrowType = this.$store.state.designsetting.fromArrowType;
- data.toArrowType = this.$store.state.designsetting.toArrowType;
- data.name = name;
- //data.scale = 1;
- if (data.bkColor) {
- this.$store.dispatch('setCanvasBackColor', data.bkColor);
- }
- this.canvas.open(data);
- this.canvas.scaleTo(1);
- this.canvas.data.bkColor = data.bkColor;
- this.$store.dispatch('setCanvas', this.canvas);
- } catch (e) {
- console.log(e)
- }
- };
- reader.readAsText(file);
- },
- OnSaveToServer()
- {
- //#region ---------------------------------------不可为空
- if (!this.page.belong || this.page.belong == "") {
- this.$message.error("无法保存,页面-所属环节不可为空");
- this.onMessage('space');
- return;
- }
- if (!this.page.path || this.page.path == "") {
- this.$message.error("无法保存页面,页面路径不可为空");
- this.onMessage('space');
- return;
- }
- if (!this.page.name || this.page.name == "") {
- this.$message.error("无法保存页面,页面名称不可为空");
- this.onMessage('space');
- return;
- }
- //#endregion ------------------------------------------
- /***
- * 第一步 保存页面
- * 第二部 保存路径结构
- */
- var newPathSummary = JSON.parse(JSON.stringify(this.$store.state.pathSummary));
- if (newPathSummary[this.page.belong].page) {
- var newPageNode = {
- "name": this.page.name,
- "path": this.page.path
- }
- newPathSummary[this.page.belong].page[this.page.name] = newPageNode;
- var MonitorJson = {
- id: 0,
- pagepath: "/",
- pagejson: JSON.stringify(newPathSummary),
- }
- monitor.setJsonFile(MonitorJson, res =>
- {
- this.$message.success("提交文件结构" + res.data.msg);
- });
- this.canvas.data.page = this.page;
- var MonitorPageJson = {
- id: 0,
- pagepath: newPathSummary[this.page.belong].path + this.page.path,
- pagejson: JSON.stringify(this.canvas.data),
- }
- monitor.setJsonFile(MonitorPageJson, res =>
- {
- this.$message.success("提交页面" + res.data.msg);
- });
- }
- },
- OnSave(data)
- {
- FileSaver.saveAs(
- new Blob([JSON.stringify(this.canvas.data)], {
- type: 'text/plain;charset=utf-8'
- }),
- `X.json`
- )
- },
- OnSavePng(data)
- {
- this.canvas.saveAsImage('GyeeX.png')
- },
- OnSaveSvg(data)
- {
- const ctx = new C2S(this.canvas.canvas.width + 200, this.canvas.canvas.height + 200)
- for (const item of this.canvas.data.pens) {
- item.render(ctx)
- }
- let mySerializedSVG = ctx.getSerializedSvg()
- mySerializedSVG = mySerializedSVG.replace(
- '<defs/>',
- `<defs>
- <style type="text/css">
- @font-face {
- font-family: 'topology';
- src: url('http://at.alicdn.com/t/font_1331132_h688rvffmbc.ttf?t=1569311680797') format('truetype');
- }
- </style>
- </defs>`
- )
- mySerializedSVG = mySerializedSVG.replace(/--le5le--/g, '&#x')
- const urlObject = window.URL || window
- const export_blob = new Blob([mySerializedSVG])
- const url = urlObject.createObjectURL(export_blob)
- const a = document.createElement('a')
- a.setAttribute('download', 'XGyee.svg')
- a.setAttribute('href', url)
- const evt = document.createEvent('MouseEvents')
- evt.initEvent('click', true, true)
- a.dispatchEvent(evt)
- },
- OnUndo()
- {
- this.canvas.undo()
- },
- OnRedo()
- {
- this.canvas.redo()
- },
- OnCopy()
- {
- this.canvas.copy()
- },
- OnCut()
- {
- this.canvas.cut()
- },
- OnPaste()
- {
- this.canvas.paste()
- },
- onContextMenu(event)
- {
- event.preventDefault()
- event.stopPropagation()
- if (event.clientY + 360 < document.body.clientHeight) {
- this.contextmenu = {
- left: event.clientX + 'px',
- top: event.clientY + 'px'
- }
- } else {
- this.contextmenu = {
- left: event.clientX + 'px',
- bottom: document.body.clientHeight - event.clientY + 'px'
- }
- }
- },
- OnTop()
- {
- if (!this.nodes) {
- return;
- }
- if (this.nodes.pen) {
- this.canvas.top(this.nodes.pen);
- }
- if (this.nodes.pens) {
- for (const item of this.nodes.pens) {
- this.canvas.top(item);
- }
- }
- this.canvas.render();
- },
- OnBottom()
- {
- if (!this.nodes) {
- return;
- }
- if (this.nodes.pen) {
- this.canvas.bottom(this.nodes.pen);
- }
- if (this.nodes.pens) {
- for (const item of this.nodes.pens) {
- this.canvas.bottom(item);
- }
- }
- this.canvas.render();
- },
- OnCombine(stand = false)
- {
- if (!this.nodes || !this.nodes.pens || this.nodes.pens.length < 2) {
- return;
- }
- this.canvas.combine(this.nodes.pens, stand);
- },
- OnUncombine()
- {
- if (!this.nodes || !this.nodes.pen || this.nodes.pen.type) {
- return;
- }
- this.canvas.uncombine(this.nodes.pen);
- this.canvas.render();
- }
- ,//penLocked
- OnPenLock()
- {
- this.penLocked = !this.penLocked;
- if (this.nodes.pen) {
- this.canvas.lockPens([this.nodes.pen], this.penLocked ? 1 : 0);
- }
- if (this.nodes.pens) {
- this.canvas.lockPens(this.nodes.pens, this.penLocked ? 1 : 0);
- }
- this.canvas.render(true);
- },
- OnDelete()
- {
- this.canvas.delete();
- },
- destroyed()
- {
- canvas.destroy()
- },
- },
- }
- </script>
|