Header.vue 40 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469
  1. <template>
  2. <div class="app-header">
  3. <a class="logo" :href="enterprise.home" target="_blank">
  4. <img src="/favicon.ico" />
  5. <span> {{ enterprise.name }}</span>
  6. </a>
  7. <t-dropdown
  8. :minColumnWidth="200"
  9. :maxHeight="560"
  10. :delay2="[10, 150]"
  11. overlayClassName="header-dropdown"
  12. trigger1="click"
  13. >
  14. <a> 文件 </a>
  15. <t-dropdown-menu>
  16. <t-dropdown-item @click="newFile">
  17. <a>新建文件</a>
  18. </t-dropdown-item>
  19. <t-dropdown-item @click="load(true)">
  20. <a>打开文件</a>
  21. </t-dropdown-item>
  22. <t-dropdown-item divider="true" @click="load">
  23. <a>导入文件</a>
  24. </t-dropdown-item>
  25. <t-dropdown-item>
  26. <a @click="save()">保存</a>
  27. </t-dropdown-item>
  28. <t-dropdown-item>
  29. <a @click="save(SaveType.SaveAs,'',undefined,1)">另保存</a>
  30. </t-dropdown-item>
  31. <t-dropdown-item divider="true">
  32. <a @click="downloadJson">下载JSON文件</a>
  33. </t-dropdown-item>
  34. <t-dropdown-item>
  35. <a @click="downloadZip">
  36. <div class="flex">
  37. 导出为ZIP文件 <span class="flex-grow"></span>
  38. <span><label>VIP</label></span>
  39. </div>
  40. </a>
  41. </t-dropdown-item>
  42. <t-dropdown-item>
  43. <a @click="downloadHtml">
  44. <div class="flex">
  45. 下载离线部署包 <span class="flex-grow"></span>
  46. <span><label>VIP</label></span>
  47. </div>
  48. </a>
  49. </t-dropdown-item>
  50. <t-dropdown-item>
  51. <a @click="downloadVue3">
  52. <div class="flex">
  53. 下载Vue3组件包 <span class="flex-grow"></span>
  54. <span><label>VIP</label></span>
  55. </div>
  56. </a>
  57. </t-dropdown-item>
  58. <t-dropdown-item>
  59. <a @click="downloadVue2">
  60. <div class="flex">
  61. 下载Vue2组件包 <span class="flex-grow"></span>
  62. <span><label>VIP</label></span>
  63. </div>
  64. </a>
  65. </t-dropdown-item>
  66. <t-dropdown-item divider="true">
  67. <a @click="downloadReact">
  68. <div class="flex">
  69. 下载React组件包 <span class="flex-grow"></span>
  70. <span><label>VIP</label></span>
  71. </div>
  72. </a>
  73. </t-dropdown-item>
  74. <t-dropdown-item>
  75. <a @click="downloadPng">下载为PNG</a>
  76. </t-dropdown-item>
  77. <t-dropdown-item divider="true">
  78. <a @click="downloadSvg">下载为SVG</a>
  79. </t-dropdown-item>
  80. <t-dropdown-item>
  81. <a @click="switchTheme('light')">明亮主题</a>
  82. </t-dropdown-item>
  83. <t-dropdown-item>
  84. <a @click="switchTheme('dark')">暗黑主题</a>
  85. </t-dropdown-item>
  86. </t-dropdown-menu>
  87. </t-dropdown>
  88. <t-dropdown
  89. :minColumnWidth="180"
  90. :maxHeight="500"
  91. :delay2="[10, 150]"
  92. overlayClassName="header-dropdown"
  93. >
  94. <a> 编辑 </a>
  95. <t-dropdown-menu>
  96. <t-dropdown-item>
  97. <a @click="onUndo">
  98. <div class="flex">
  99. 撤销 <span class="flex-grow"></span> Ctrl + Z
  100. </div>
  101. </a>
  102. </t-dropdown-item>
  103. <t-dropdown-item divider="true">
  104. <a @click="onRedo">
  105. <div class="flex">
  106. 恢复 <span class="flex-grow"></span> Ctrl + Y
  107. </div>
  108. </a>
  109. </t-dropdown-item>
  110. <t-dropdown-item>
  111. <a @click="onCut">
  112. <div class="flex">
  113. 剪切 <span class="flex-grow"></span> Ctrl + X
  114. </div>
  115. </a>
  116. </t-dropdown-item>
  117. <t-dropdown-item>
  118. <a @click="onCopy">
  119. <div class="flex">
  120. 复制 <span class="flex-grow"></span> Ctrl + C
  121. </div>
  122. </a>
  123. </t-dropdown-item>
  124. <t-dropdown-item divider="true">
  125. <a @click="onPaste">
  126. <div class="flex">
  127. 粘贴 <span class="flex-grow"></span> Ctrl + V
  128. </div>
  129. </a>
  130. </t-dropdown-item>
  131. <t-dropdown-item>
  132. <a @click="onAll">
  133. <div class="flex">
  134. 全选 <span class="flex-grow"></span> Ctrl + A
  135. </div>
  136. </a>
  137. </t-dropdown-item>
  138. <t-dropdown-item>
  139. <a @click="onDelete">
  140. <div class="flex">删除 <span class="flex-grow"></span> DELETE</div>
  141. </a>
  142. </t-dropdown-item>
  143. </t-dropdown-menu>
  144. </t-dropdown>
  145. <t-dropdown
  146. :minColumnWidth="180"
  147. :maxHeight="500"
  148. :delay2="[10, 150]"
  149. overlayClassName="header-dropdown"
  150. >
  151. <a> 工具 </a>
  152. <t-dropdown-menu>
  153. <t-dropdown-item>
  154. <a @click="onScaleWindow">窗口大小</a>
  155. </t-dropdown-item>
  156. <t-dropdown-item>
  157. <a @click="onScaleUp">放大</a>
  158. </t-dropdown-item>
  159. <t-dropdown-item>
  160. <a @click="onScaleDown">缩小</a>
  161. </t-dropdown-item>
  162. <t-dropdown-item divider="true">
  163. <a @click="onScaleFull">100%视图</a>
  164. </t-dropdown-item>
  165. <t-dropdown-item>
  166. <a @click="showMap">
  167. <div class="flex middle">
  168. 鹰眼地图 <span class="flex-grow"></span>
  169. <!-- <t-icon v-show="map" name="check" /> -->
  170. <check-icon v-show="map"/>
  171. </div>
  172. </a>
  173. </t-dropdown-item>
  174. <t-dropdown-item divider="true">
  175. <a @click="showMagnifier">
  176. <div class="flex middle">
  177. 放大镜 <span class="flex-grow"></span>
  178. <!-- <t-icon v-show="magnifier" name="check" /> -->
  179. <check-icon v-show="magnifier"/>
  180. </div>
  181. </a>
  182. </t-dropdown-item>
  183. <t-dropdown-item>
  184. <a @click="onAutoAnchor">
  185. <div class="flex middle">
  186. 自动锚点 <span class="flex-grow"></span>
  187. <!-- <t-icon v-show="autoAnchor" name="check" /> -->
  188. <check-icon v-show="autoAnchor"/>
  189. </div>
  190. </a>
  191. </t-dropdown-item>
  192. <t-dropdown-item divider="true">
  193. <a @click="onDisableAnchor">
  194. <div class="flex middle">
  195. 显示锚点 <span class="flex-grow"></span>
  196. <!-- <t-icon v-show="showAnchor" name="check" /> -->
  197. <check-icon v-show="showAnchor"/>
  198. </div>
  199. </a>
  200. </t-dropdown-item>
  201. <t-dropdown-item>
  202. <a @click="onToggleAnchor">
  203. <div
  204. class="flex"
  205. :style="{
  206. color: showAnchor ? '' : '#4f5b75',
  207. }"
  208. >
  209. 添加/删除锚点 <span class="flex-grow"></span> A
  210. </div>
  211. </a>
  212. </t-dropdown-item>
  213. </t-dropdown-menu>
  214. </t-dropdown>
  215. <t-dropdown
  216. :minColumnWidth="180"
  217. :maxHeight="500"
  218. :delay2="[10, 150]"
  219. overlayClassName="header-dropdown"
  220. >
  221. <a> 帮助 </a>
  222. <t-dropdown-menu>
  223. <t-dropdown-item
  224. v-for="item in enterprise.helps"
  225. :divider="item.divider"
  226. >
  227. <a :href="item.url" target="_blank">{{ item.name }}</a>
  228. </t-dropdown-item>
  229. </t-dropdown-menu>
  230. </t-dropdown>
  231. <div style="width: 148px; flex-shrink: 0"></div>
  232. <input v-model="data.name" @input="onInputName" />
  233. <a :href="enterprise.account" target="_blank">
  234. <!-- <t-icon name="home" /> -->
  235. <home-icon/>
  236. 账户中心
  237. </a>
  238. <a :href="enterprise['v']" target="_blank" class="active">
  239. <!-- <t-icon name="desktop" /> -->
  240. <desktop-icon/>
  241. 大屏可视化
  242. </a>
  243. <a :href="enterprise['3d']" target="_blank">
  244. <!-- <t-icon name="control-platform" /> -->
  245. <control-platform-icon/>
  246. 3D可视化
  247. </a>
  248. <a :href="enterprise['2d']" target="_blank">
  249. <!-- <t-icon name="app" /> -->
  250. <app-icon/>
  251. 2D可视化
  252. </a>
  253. <t-dropdown
  254. v-if="user.id"
  255. :minColumnWidth="200"
  256. :maxHeight="500"
  257. :delay2="[10, 150]"
  258. overlayClassName="custom-dropdown header"
  259. >
  260. <a style="margin-left: 32px; margin-right: 12px">
  261. <t-avatar
  262. size="small"
  263. :image="user.avatarUrl ? user.avatarUrl : baseUrl + 'img/avatar.png'"
  264. />
  265. </a>
  266. <t-dropdown-menu>
  267. <t-dropdown-item divider="true">
  268. <a :href="enterprise.account">
  269. {{ user.username }}
  270. <label
  271. class="ml-16 vip-label"
  272. :style="{
  273. color: user.limit > 1 ? '#ff4000' : '#D5C781',
  274. background: user.limit > 1 ? '#ff400030' : '#D5C78133',
  275. }"
  276. >VIP</label
  277. >
  278. </a>
  279. </t-dropdown-item>
  280. <t-dropdown-item divider="true">
  281. <a :href="`${enterprise.account}/v`" target="_blank"> 我的大屏 </a>
  282. </t-dropdown-item>
  283. <t-dropdown-item>
  284. <a :href="`${enterprise.account}/account/teams`" target="_blank">
  285. 我的团队
  286. </a>
  287. </t-dropdown-item>
  288. <t-dropdown-item>
  289. <a :href="`${enterprise.account}/account/info`" target="_blank">
  290. 账号信息
  291. </a>
  292. </t-dropdown-item>
  293. <t-dropdown-item divider="true">
  294. <a :href="`${enterprise.account}/account/security`" target="_blank">
  295. 安全设置
  296. </a>
  297. </t-dropdown-item>
  298. <t-dropdown-item>
  299. <a @click="logout">退出</a>
  300. </t-dropdown-item>
  301. </t-dropdown-menu>
  302. </t-dropdown>
  303. <div class="flex middle" v-else>
  304. <a class="button primary solid" style="width: 80px" :href="login()">
  305. 登录
  306. </a>
  307. </div>
  308. </div>
  309. </template>
  310. <script lang="ts" setup>
  311. import { reactive, ref, onBeforeMount, onUnmounted, nextTick } from 'vue';
  312. import { useRouter, useRoute } from 'vue-router';
  313. import { useUser } from '@/services/user';
  314. import { MessagePlugin } from 'tdesign-vue-next';
  315. import {
  316. Meta2dBackData,
  317. dealwithFormatbeforeOpen,
  318. gotoAccount,
  319. checkData,
  320. } from '@/services/utils';
  321. import { readFile } from '@/services/file';
  322. import { compareVersion, baseVer, upgrade } from '@/services/upgrade';
  323. import { parseSvg } from '@meta2d/svg';
  324. import { Pen, getGlobalColor, isShowChild } from '@meta2d/core';
  325. import { cdn, upCdn } from '@/services/api';
  326. import JSZip from 'jszip';
  327. import axios from 'axios';
  328. import { switchTheme } from '@/services/theme';
  329. import { noLoginTip } from '@/services/utils';
  330. import {
  331. save,
  332. blank,
  333. newFile,
  334. SaveType,
  335. onScaleFull,
  336. onScaleWindow,
  337. showMagnifier,
  338. showMap,
  339. drawPen,
  340. map,
  341. magnifier,
  342. useDot,
  343. delAttrs,
  344. useAssets,
  345. } from '@/services/common';
  346. import { useEnterprise } from '@/services/enterprise';
  347. import { CheckIcon,HomeIcon ,DesktopIcon ,ControlPlatformIcon,AppIcon} from 'tdesign-icons-vue-next';
  348. import { getDownloadList } from '@/services/download';
  349. const { enterprise } = useEnterprise();
  350. const router = useRouter();
  351. const route = useRoute();
  352. const baseUrl = import.meta.env.BASE_URL || '/';
  353. // const { assets, getAssets } = useAssets();
  354. const { user, signout } = useUser();
  355. const { setDot } = useDot();
  356. const data = reactive({
  357. name: '空白文件',
  358. });
  359. onBeforeMount(async () => {
  360. // getAssets();
  361. });
  362. const logout = () => {
  363. signout();
  364. meta2d.emit('logout');
  365. };
  366. const onInputName = () => {
  367. (meta2d.store.data as Meta2dBackData).name = data.name;
  368. setDot();
  369. };
  370. const initMeta2dName = () => {
  371. data.name = (meta2d.store.data as Meta2dBackData).name || '';
  372. };
  373. let downloadList = new Set();
  374. let iframeNum = 0;
  375. let compareNum = 0;
  376. nextTick(() => {
  377. meta2d.on('opened', initMeta2dName);
  378. window.addEventListener('message', function (e) {
  379. if (typeof e.data !== 'string'||!e.data||e.data.startsWith('setImmediate')) {
  380. return;
  381. }
  382. try {
  383. let data = JSON.parse(e.data);
  384. if (typeof data === 'object') {
  385. if(data.type && data.name==='download'){
  386. downloadList = new Set([...downloadList, ...data.data]);
  387. compareNum+=1;
  388. if(compareNum>=iframeNum){
  389. saveDownload();
  390. }
  391. }
  392. meta2d.emit(data.name);
  393. } else {
  394. meta2d.emit(data);
  395. }
  396. } catch (e) {
  397. console.info(e);
  398. }
  399. });
  400. });
  401. onUnmounted(() => {
  402. meta2d.off('opened', initMeta2dName);
  403. });
  404. const login = () => {
  405. return `${enterprise['account']}?cb=${encodeURIComponent(location.href)}`;
  406. };
  407. function load(isNew = false) {
  408. const input = document.createElement('input');
  409. input.type = 'file';
  410. input.onchange = (event) => {
  411. const elem = event.target as HTMLInputElement;
  412. if (elem.files && elem.files[0]) {
  413. blank();
  414. // 路由跳转 可能在 openFile 后执行
  415. if (elem.files[0].name.endsWith('.json')) {
  416. openJson(elem.files[0]);
  417. if (isNew) {
  418. router.push({
  419. path: '/',
  420. query: {
  421. r: Date.now() + '',
  422. },
  423. });
  424. }
  425. } else if (elem.files[0].name.endsWith('.svg')) {
  426. MessagePlugin.info(
  427. '可二次编辑但转换存在损失,若作为图片使用,请使用右侧属性面板的上传图片功能'
  428. );
  429. openSvg(elem.files[0]);
  430. } else if (elem.files[0].name.endsWith('.zip')) {
  431. router.push({
  432. path: '/',
  433. query: {
  434. r: Date.now() + '',
  435. },
  436. });
  437. setTimeout(() => {
  438. openZip(elem.files[0]);
  439. }, 500);
  440. } else {
  441. MessagePlugin.info('打开文件只支持 json,svg,zip 格式');
  442. }
  443. }
  444. };
  445. input.click();
  446. }
  447. const openJson = async (file: File) => {
  448. const text = await readFile(file);
  449. try {
  450. let data: Meta2dBackData = JSON.parse(text);
  451. if (!data.name) {
  452. data.name = file.name.replace('.json', '');
  453. }
  454. if (!data.version || compareVersion(data.version, baseVer) === -1) {
  455. // 如果版本号不存在或者版本号 version < 1.0.0
  456. data = upgrade(data, baseVer);
  457. }
  458. dealwithFormatbeforeOpen(data);
  459. for (const k of delAttrs) {
  460. delete (data as any)[k];
  461. }
  462. meta2d.open(data);
  463. } catch (e) {
  464. console.error(e);
  465. }
  466. };
  467. const openSvg = async (file: File) => {
  468. const text = await readFile(file);
  469. const pens: Pen[] = parseSvg(text);
  470. meta2d.canvas.addCaches = pens;
  471. MessagePlugin.info('svg转换成功,请点击画布决定放置位置');
  472. };
  473. const openZip = async (file: File) => {
  474. if (!(user && user.id)) {
  475. MessagePlugin.warning(noLoginTip);
  476. return;
  477. }
  478. // if (!user.isVip) {
  479. // gotoAccount();
  480. // return;
  481. // }
  482. if (!user.vip) {
  483. MessagePlugin.info('需要开通普通会员~');
  484. gotoAccount();
  485. return;
  486. }
  487. const { default: JSZip } = await import('jszip');
  488. const zip = new JSZip();
  489. await zip.loadAsync(file);
  490. let dataStr = '';
  491. for (const key in zip.files) {
  492. if (zip.files[key].dir) {
  493. continue;
  494. }
  495. if (key.endsWith('.json')) {
  496. // 认为只有一个 json 文件
  497. // dataStr = await zip.file(key).async('string');
  498. break;
  499. }
  500. }
  501. if (!dataStr) {
  502. return false;
  503. }
  504. for (const key in zip.files) {
  505. if (zip.files[key].dir) {
  506. continue;
  507. }
  508. // let _png = key.indexOf('/png');
  509. // let _img = key.indexOf('/img');
  510. // let _image = key.indexOf('/image');
  511. // let _file = key.indexOf('/file');
  512. let _keyLower = key.toLowerCase();
  513. // if (!key.endsWith('.json') && (_png !== -1 || _img !== -1 || _image !== -1 || _file !== -1)) {
  514. if (
  515. _keyLower.endsWith('.png') ||
  516. _keyLower.endsWith('.svg') ||
  517. _keyLower.endsWith('.gif') ||
  518. _keyLower.endsWith('.jpg') ||
  519. _keyLower.endsWith('.jpeg')
  520. ) {
  521. let filename = key.substr(key.lastIndexOf('/') + 1);
  522. const extPos = filename.lastIndexOf('.');
  523. let ext = '';
  524. if (extPos > 0) {
  525. ext = filename.substr(extPos);
  526. }
  527. filename = filename.substring(0, extPos > 8 ? 8 : extPos);
  528. // 上传文件
  529. const result: any = {};
  530. // await upload(
  531. // // await zip.file(key).async('blob'),
  532. // true,
  533. // filename + ext,
  534. // "/2D/默认"
  535. // );
  536. let _key = key;
  537. // if (_png) {
  538. // _key = key.substring(_png);
  539. // } else if (_image) {
  540. // _key = key.substring(_png);
  541. // } else if (_img) {
  542. // _key = key.substring(_img);
  543. // } else if (_file) {
  544. // _key = key.substring(_file);
  545. // }
  546. if (result) {
  547. if (dataStr.replaceAll) {
  548. //'le5le.meta2d'
  549. dataStr = dataStr.replaceAll(_key.slice(12), result.url);
  550. } else {
  551. while (dataStr.includes(_key)) {
  552. dataStr = dataStr.replace(_key.slice(12), result.url);
  553. // 正则 gm 在特殊情况下报错,例如如下场景
  554. /**
  555. *    
  556. const data = '{"image":"/image/materials/IoT-Chemical(化学)/Air stripper 2(汽提塔2).svg"}';
  557. const key = '/image/materials/IoT-Chemical(化学)/Air stripper 2(汽提塔2).svg';
  558. data.replace(key, '123');
  559. data.replaceAll(key, '123')
  560. data.replace(new RegExp(key, 'gm'), '123');
  561. data.replace(new RegExp(key, 'g'), '123');
  562. */
  563. }
  564. }
  565. }
  566. }
  567. }
  568. try {
  569. let data: Meta2dBackData = JSON.parse(dataStr);
  570. if (data) {
  571. if (!data.name) {
  572. data.name = file.name.replace('.zip', '');
  573. }
  574. if (!data.version || compareVersion(data.version, baseVer) === -1) {
  575. // 如果版本号不存在或者版本号 version < 1.0.0
  576. data = upgrade(data, baseVer);
  577. }
  578. dealwithFormatbeforeOpen(data);
  579. const delAttrs = [
  580. 'userId',
  581. 'shared',
  582. 'team',
  583. 'owner',
  584. 'username',
  585. 'editor',
  586. 'editorId',
  587. 'editorName',
  588. 'createdAt',
  589. 'folder',
  590. 'image',
  591. 'id',
  592. '_id',
  593. 'view',
  594. 'updatedAt',
  595. 'star',
  596. 'recommend',
  597. ];
  598. for (const k of delAttrs) {
  599. delete (data as any)[k];
  600. }
  601. meta2d.open(data);
  602. }
  603. } catch (e) {
  604. return false;
  605. }
  606. };
  607. const downloadJson = () => {
  608. const data: Meta2dBackData = meta2d.data();
  609. if (data._id) delete data._id;
  610. if (data.id) delete data.id;
  611. checkData(data);
  612. import('file-saver').then(({ saveAs }) => {
  613. saveAs(
  614. new Blob(
  615. [JSON.stringify(data).replaceAll(cdn, '').replaceAll(upCdn, '')],
  616. {
  617. type: 'text/plain;charset=utf-8',
  618. }
  619. ),
  620. `${data.name || 'le5le.meta2d'}.json`
  621. );
  622. });
  623. };
  624. const downloadZip = async () => {
  625. if (!(user && user.id)) {
  626. MessagePlugin.warning(noLoginTip);
  627. return;
  628. }
  629. // if (!user.isVip) {
  630. // // gotoAccount();
  631. // return;
  632. // }
  633. if (!user.vip) {
  634. MessagePlugin.info('需要开通普通会员~');
  635. gotoAccount();
  636. return;
  637. }
  638. MessagePlugin.info('正在下载打包中,可能需要几分钟,请耐心等待...');
  639. const [{ default: JSZip }, { saveAs }] = await Promise.all([
  640. import('jszip'),
  641. import('file-saver'),
  642. ]);
  643. const zip: any = new JSZip();
  644. const data: Meta2dBackData = meta2d.data();
  645. let _fileName =
  646. (data.name && data.name.replace(/\//g, '_').replace(/:/g, '_')) ||
  647. 'le5le.meta2d';
  648. const _zip = zip.folder(`${_fileName}`);
  649. if (data._id) delete data._id;
  650. if (data.id) delete data.id;
  651. checkData(data);
  652. _zip.file(
  653. `${_fileName}.json`,
  654. JSON.stringify(data).replaceAll(cdn, '').replaceAll(upCdn, '')
  655. );
  656. await zipBkImg(_zip);
  657. await zipImages(_zip, meta2d.store.data.pens);
  658. const blob = await zip.generateAsync({ type: 'blob' });
  659. saveAs(blob, `${_fileName}.zip`);
  660. };
  661. const zip3D = (name:string) => {
  662. const pen_3d = meta2d.store.data.pens.filter(
  663. (pen) =>
  664. pen.name === 'iframe' &&
  665. (pen.tags.includes('meta3d') || pen.iframe.indexOf('3d') !== -1)
  666. );
  667. if (pen_3d && pen_3d.length) {
  668. //存在3d场景
  669. pen_3d.forEach((pen) => {
  670. //发送消息
  671. let params = queryURLParams(pen.iframe.split('?')[1]);
  672. (
  673. pen.calculative.singleton.div.children[0] as HTMLIFrameElement
  674. ).contentWindow.postMessage(
  675. JSON.stringify({
  676. name,
  677. id: params.id,
  678. }),
  679. '*'
  680. );
  681. });
  682. }
  683. };
  684. function queryURLParams(value?: string) {
  685. let url = value || window.location.href.split('?')[1];
  686. const urlSearchParams = new URLSearchParams(url);
  687. const params = Object.fromEntries(urlSearchParams.entries());
  688. return params;
  689. }
  690. const downloadHtml = async () => {
  691. if (!(user && user.id)) {
  692. MessagePlugin.warning(noLoginTip);
  693. return;
  694. }
  695. if(user.vipDesc !== '超级会员'&&user.vipDesc !=='旗舰会员'){
  696. MessagePlugin.info('需要开通超级会员~');
  697. gotoAccount();
  698. return
  699. }
  700. MessagePlugin.info('正在下载打包中,可能需要几分钟,请耐心等待...');
  701. compareNum = 0;
  702. const meta2dData = meta2d.data();
  703. const pen_3d = meta2dData.pens.filter(
  704. (pen) =>
  705. pen.name === 'iframe' &&
  706. (pen.iframe.indexOf('3d.le5le.com') !== -1)
  707. );
  708. if (pen_3d && pen_3d.length) {
  709. //存在3d场景
  710. if(pen_3d.length===1){
  711. let params = queryURLParams(pen_3d[0].iframe.split('?')[1]);
  712. meta2d.store.pens[pen_3d[0].id].calculative.singleton.div.children[0].contentWindow.postMessage(
  713. JSON.stringify({
  714. name:'deploy',
  715. // id: params.id,
  716. type:1, //用于区分是系统消息
  717. path:`3d`
  718. }),
  719. '*'
  720. );
  721. pen_3d.iframe = '/view?data=3d';
  722. }else{
  723. pen_3d.forEach((pen) => {
  724. //发送消息
  725. let params = queryURLParams(pen.iframe.split('?')[1]);
  726. (
  727. meta2d.store.pens[pen.id].calculative.singleton.div.children[0] as HTMLIFrameElement
  728. ).contentWindow.postMessage(
  729. JSON.stringify({
  730. name:'deploy',
  731. // id: params.id,
  732. type:1,
  733. path:`3d-${params.id}`
  734. }),
  735. '*'
  736. );
  737. pen.iframe = `/view?data=3d-${params.id}`;
  738. });
  739. }
  740. iframeNum+= pen_3d.length;
  741. }
  742. const pen_2d = meta2dData.pens.filter(
  743. (pen) =>
  744. pen.name === 'iframe' &&
  745. (pen.iframe.indexOf('2d.le5le.com') !== -1)
  746. );
  747. if (pen_2d && pen_2d.length) {
  748. //存在3d场景
  749. if(pen_2d.length===1){
  750. let params = queryURLParams(pen_2d[0].iframe.split('?')[1]);
  751. meta2d.store.pens[pen_2d[0].id].calculative.singleton.div.children[0].contentWindow.postMessage(
  752. JSON.stringify({
  753. name:'downloadHtml',
  754. id: params.id,
  755. type:1,
  756. path:`2d`
  757. }),
  758. '*'
  759. );
  760. pen_2d[0].iframe = '/view?data=2d'
  761. }else{
  762. pen_2d.forEach((pen) => {
  763. //发送消息
  764. let params = queryURLParams(pen.iframe.split('?')[1]);
  765. (
  766. meta2d.store.pens[pen.id].calculative.singleton.div.children[0] as HTMLIFrameElement
  767. ).contentWindow.postMessage(
  768. JSON.stringify({
  769. name:'downloadHtml',
  770. // id: params.id,
  771. type:1,
  772. path:`2d-${params.id}`
  773. }),
  774. '*'
  775. );
  776. });
  777. pen.iframe = `/view?data=2d-${params.id}`
  778. }
  779. iframeNum+= pen_2d.length;
  780. }
  781. const pen_v = meta2dData.pens.filter(
  782. (pen) =>
  783. pen.name === 'iframe' &&
  784. (pen.iframe.indexOf('v.le5le.com') !== -1)
  785. );
  786. if (pen_v && pen_v.length) {
  787. //存在3d场景
  788. pen_v.forEach((pen) => {
  789. //发送消息
  790. let params = queryURLParams(pen.iframe.split('?')[1]);
  791. (
  792. meta2d.store.pens[pen.id].calculative.singleton.div.children[0] as HTMLIFrameElement
  793. ).contentWindow.postMessage(
  794. JSON.stringify({
  795. name:'downloadHtml',
  796. // id: params.id,
  797. type:1,
  798. path:`v-${params.id}`
  799. }),
  800. '*'
  801. );
  802. pen.iframe = `/view?data=v-${params.id}`
  803. });
  804. iframeNum+= pen_v.length;
  805. }
  806. downloadList = getDownloadList(meta2dData);
  807. if(iframeNum===0){
  808. //如果没有嵌入场景
  809. saveDownload();
  810. }else{
  811. setTimeout(()=>{
  812. if(compareNum < iframeNum){
  813. //message阻塞/报错的情况
  814. saveDownload();
  815. }
  816. },10000);
  817. }
  818. }
  819. const saveDownload = async()=>{
  820. const [{ default: JSZip }, { saveAs }] = await Promise.all([
  821. import('jszip'),
  822. import('file-saver'),
  823. ]);
  824. const zip = new JSZip();
  825. await Promise.all([...downloadList].map(async(item)=>{
  826. if(item.url){
  827. //接口请求
  828. const res: Blob = await axios.get(item.url, {
  829. responseType: 'blob',
  830. });
  831. zip.file(item.path, res, { createFolders: true });
  832. }else if(item.data){
  833. //直接写数据
  834. zip.file(
  835. item.path,
  836. item.data, { createFolders: true }
  837. );
  838. }
  839. }));
  840. let _fileName =
  841. (data.name && data.name.replace(/\//g, '_').replace(/:/g, '_')) ||
  842. 'le5le.meta2d';
  843. const blob = await zip.generateAsync({ type: 'blob' });
  844. saveAs(blob, `${_fileName}.zip`);
  845. }
  846. const _downloadHtml = async () => {
  847. if (!(user && user.id)) {
  848. MessagePlugin.warning(noLoginTip);
  849. return;
  850. }
  851. // if (!user.isVip) {
  852. // gotoAccount();
  853. // return;
  854. // }
  855. if(user.vipDesc !== '超级会员'&&user.vipDesc !=='旗舰会员'){
  856. MessagePlugin.info('需要开通超级会员~');
  857. gotoAccount();
  858. return
  859. }
  860. frameFlag = -1;
  861. MessagePlugin.info('正在下载打包中,可能需要几分钟,请耐心等待...');
  862. zip3D('deploy');
  863. const data: Meta2dBackData = meta2d.data();
  864. if (data._id) delete data._id;
  865. if (data.id) delete data.id;
  866. if (data.image) delete data.image;
  867. checkData(data);
  868. const [{ default: JSZip }, { saveAs }] = await Promise.all([
  869. import('jszip'),
  870. import('file-saver'),
  871. ]);
  872. const zip = new JSZip();
  873. let _fileName =
  874. (data.name && data.name.replace(/\//g, '_').replace(/:/g, '_')) ||
  875. 'le5le.meta2d';
  876. //处理cdn图片地址
  877. const _zip: any = zip.folder(`${_fileName}`);
  878. _zip.file(
  879. 'data.json',
  880. JSON.stringify(data).replaceAll(cdn, '').replaceAll(upCdn, '')
  881. );
  882. await Promise.all([
  883. zipBkImg(_zip),
  884. zipImages(_zip, meta2d.store.data.pens),
  885. zipFiles(_zip),
  886. ]);
  887. const blob = await zip.generateAsync({ type: 'blob' });
  888. saveAs(blob, `${_fileName}.zip`);
  889. };
  890. async function zipBkImg(zip: JSZip) {
  891. let img = meta2d.store.data.bkImage;
  892. if (img) {
  893. if (img.startsWith('/') || img.startsWith(cdn) || img.startsWith(upCdn)) {
  894. await zipImage(zip, img);
  895. }
  896. }
  897. }
  898. enum Frame {
  899. vue2,
  900. vue3,
  901. react,
  902. }
  903. const downloadVue3 = async () => {
  904. downloadAsFrame(Frame.vue3);
  905. };
  906. const downloadVue2 = async () => {
  907. downloadAsFrame(Frame.vue2);
  908. };
  909. const downloadReact = async () => {
  910. downloadAsFrame(Frame.react);
  911. };
  912. let frameFlag = -1;
  913. async function downloadAsFrame(type: Frame) {
  914. if (!(user && user.id)) {
  915. MessagePlugin.warning(noLoginTip);
  916. return;
  917. }
  918. // if (!user.isVip) {
  919. // gotoAccount();
  920. // return;
  921. // }
  922. if(user.vipDesc !=='旗舰会员'){
  923. MessagePlugin.info('需要开通旗舰会员~');
  924. gotoAccount();
  925. return
  926. }
  927. frameFlag = type;
  928. MessagePlugin.info('正在下载打包中,可能需要几分钟,请耐心等待...');
  929. zip3D(type===Frame.vue3?'toVue3':(type===Frame.vue2?'toVue2':'toReact'));
  930. const data: Meta2dBackData = meta2d.data();
  931. if (data._id) delete data._id;
  932. if (data.id) delete data.id;
  933. if (data.image) delete data.image;
  934. checkData(data);
  935. const [{ default: JSZip }, { saveAs }] = await Promise.all([
  936. import('jszip'),
  937. import('file-saver'),
  938. ]);
  939. const zip = new JSZip();
  940. let _fileName =
  941. (data.name && data.name.replace(/\//g, '_').replace(/:/g, '_')) ||
  942. 'le5le.meta2d';
  943. const _zip: any = zip.folder(`${_fileName}`);
  944. _zip.file(
  945. `${
  946. type === Frame.vue3
  947. ? 'meta2d-vue3'
  948. : type === Frame.vue2
  949. ? 'meta2d-vue2'
  950. : 'meta2d-react'
  951. }/public/json/data.json`,
  952. JSON.stringify(data).replaceAll(cdn, '').replaceAll(upCdn, '')
  953. );
  954. await Promise.all([
  955. zipJs(_zip),
  956. zipBkImg(_zip),
  957. zipImages(_zip, meta2d.store.data.pens),
  958. type === Frame.vue3
  959. ? zipVue3Files(_zip)
  960. : type === Frame.vue2
  961. ? zipVue2Files(_zip)
  962. : zipReactFiles(_zip),
  963. ]);
  964. const blob = await zip.generateAsync({ type: 'blob' });
  965. saveAs(blob, `${_fileName}.zip`);
  966. frameFlag = -1;
  967. }
  968. async function zipJs(zip: JSZip) {
  969. const files = ['/view/js/marked.min.js', '/view/js/lcjs.iife.js'];
  970. await Promise.all(
  971. files.map(async (filePath) => {
  972. const res: Blob = await axios.get(filePath, {
  973. responseType: 'blob',
  974. });
  975. zip.file(
  976. `${
  977. frameFlag === Frame.vue3
  978. ? 'meta2d-vue3'
  979. : frameFlag === Frame.vue2
  980. ? 'meta2d-vue2'
  981. : 'meta2d-react'
  982. }/public` + filePath.replace('/view', ''),
  983. res,
  984. { createFolders: true }
  985. );
  986. })
  987. );
  988. }
  989. async function _zipVue3Files(zip: JSZip) {
  990. const files = [
  991. '/view/js/marked.min.js',
  992. '/view/js/lcjs.iife.js',
  993. '/view/vue3/Meta2d.vue',
  994. '/view/index.html',
  995. '/view/js/meta2d.js',
  996. '/view/使用说明.md',
  997. ] as const;
  998. // 文件同时加载
  999. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1000. }
  1001. //新
  1002. async function zipVue3Files(zip: JSZip) {
  1003. const files = [
  1004. '/view/meta2d-vue3/src/components/Meta2d.vue',
  1005. '/view/meta2d-vue3/src/App.vue',
  1006. '/view/meta2d-vue3/src/main.js',
  1007. '/view/meta2d-vue3/src/style.css',
  1008. '/view/meta2d-vue3/index.html',
  1009. '/view/meta2d-vue3/package.json',
  1010. '/view/meta2d-vue3/README.md',
  1011. '/view/meta2d-vue3/vite.config.js',
  1012. ] as const;
  1013. // 文件同时加载
  1014. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1015. }
  1016. async function _zipVue2Files(zip: JSZip) {
  1017. const files = [
  1018. '/view/js/marked.min.js',
  1019. '/view/js/lcjs.iife.js',
  1020. '/view/vue2/Meta2d.vue',
  1021. '/view/index.html',
  1022. '/view/js/meta2d.js',
  1023. '/view/使用说明.md',
  1024. ] as const;
  1025. // 文件同时加载
  1026. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1027. }
  1028. async function zipVue2Files(zip: JSZip) {
  1029. const files = [
  1030. '/view/meta2d-vue2/src/components/Meta2d.vue',
  1031. '/view/meta2d-vue2/src/App.vue',
  1032. '/view/meta2d-vue2/src/main.js',
  1033. // '/view/meta2d-vue2/src/style.css',
  1034. '/view/meta2d-vue2/public/index.html',
  1035. '/view/meta2d-vue2/package.json',
  1036. '/view/meta2d-vue2/README.md',
  1037. // '/view/meta2d-vue3/vite.config.js',
  1038. ] as const;
  1039. // 文件同时加载
  1040. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1041. }
  1042. async function _zipReactFiles(zip: JSZip) {
  1043. const files = [
  1044. '/view/js/marked.min.js',
  1045. '/view/js/lcjs.iife.js',
  1046. '/view/react/Meta2d.jsx',
  1047. '/view/react/Meta2d.css',
  1048. '/view/index.html',
  1049. '/view/js/meta2d.js',
  1050. '/view/使用说明.md',
  1051. ] as const;
  1052. // 文件同时加载
  1053. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1054. }
  1055. async function zipReactFiles(zip: JSZip) {
  1056. const files = [
  1057. '/view/meta2d-react/src/index.css',
  1058. '/view/meta2d-react/src/index.js',
  1059. '/view/meta2d-react/src/Meta2d.css',
  1060. '/view/meta2d-react/src/Meta2d.jsx',
  1061. '/view/meta2d-react/package.json',
  1062. '/view/meta2d-react/README.md',
  1063. '/view/meta2d-react/public/index.html',
  1064. ] as const;
  1065. // 文件同时加载
  1066. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1067. }
  1068. async function zipFiles(zip: JSZip) {
  1069. const files = [
  1070. '/view/js/marked.min.js',
  1071. '/view/js/lcjs.iife.js',
  1072. '/view/js/index.js',
  1073. '/view/js/meta2d.js',
  1074. '/view/index.html',
  1075. '/view/index.css',
  1076. '/view/favicon.ico',
  1077. '/view/使用说明.pdf',
  1078. ] as const;
  1079. // 文件同时加载
  1080. await Promise.all(files.map((filePath) => zipFile(zip, filePath)));
  1081. }
  1082. async function zipFile(zip: JSZip, filePath: string) {
  1083. const res: Blob = await axios.get((cdn ? cdn + '/v' : import.meta.env.BASE_URL.slice(0,-1)) + filePath, {
  1084. responseType: 'blob',
  1085. });
  1086. zip.file(filePath.replace('/view', ''), res, { createFolders: true });
  1087. }
  1088. /**
  1089. * 图片放到 zip 里
  1090. * @param pens 可以是非具有 calculative 的 pen
  1091. */
  1092. async function zipImages(zip: JSZip, pens: Pen[]) {
  1093. if (!pens) {
  1094. return;
  1095. }
  1096. // 不止 image 上有图片, strokeImage ,backgroundImage 也有图片
  1097. const imageKeys = [
  1098. {
  1099. string: 'image',
  1100. },
  1101. { string: 'strokeImage' },
  1102. { string: 'backgroundImage' },
  1103. ] as const;
  1104. const images: string[] = [];
  1105. for (const pen of pens) {
  1106. for (const i of imageKeys) {
  1107. const image = pen[i.string];
  1108. if (image) {
  1109. // HTMLImageElement 无法精确控制图片格式
  1110. if (
  1111. image.startsWith('/') ||
  1112. image.startsWith(cdn) ||
  1113. image.startsWith(upCdn)
  1114. ) {
  1115. // 只考虑相对路径下的 image ,绝对路径图片无需下载
  1116. if (!images.includes(image)) {
  1117. images.push(image);
  1118. }
  1119. }
  1120. }
  1121. }
  1122. // 无需递归遍历子节点,现在所有的节点都在外层
  1123. }
  1124. await Promise.all(images.map((image) => zipImage(zip, image)));
  1125. }
  1126. async function zipImage(zip: JSZip, image: string) {
  1127. const res: Blob = await axios.get(image, {
  1128. responseType: 'blob',
  1129. params: {
  1130. isZip: true,
  1131. },
  1132. });
  1133. zip.file(
  1134. (frameFlag === -1
  1135. ? ''
  1136. : `${
  1137. frameFlag === Frame.vue3
  1138. ? 'meta2d-vue3'
  1139. : frameFlag === Frame.vue2
  1140. ? 'meta2d-vue2'
  1141. : 'meta2d-react'
  1142. }/public`) + (cdn ? image.replace(cdn, '').replace(upCdn, '') : image),
  1143. res,
  1144. {
  1145. createFolders: true,
  1146. }
  1147. );
  1148. }
  1149. const downloadImageTips =
  1150. '无法下载,宽度不合法,画布可能没有画笔/画布大小超出浏览器最大限制';
  1151. const downloadPng = () => {
  1152. if (!meta2d.store.data.pens.length) {
  1153. MessagePlugin.warning(downloadImageTips);
  1154. return;
  1155. }
  1156. try {
  1157. meta2d.downloadPng();
  1158. } catch (e) {
  1159. MessagePlugin.warning(downloadImageTips);
  1160. }
  1161. };
  1162. async function getIconDefs(url: string) {
  1163. let res: any = await axios.get(url);
  1164. let str = res.match(/@font-face([\s\S]*?)\}/)[1];
  1165. str = `@font-face ${str} }`;
  1166. return str;
  1167. }
  1168. const downloadSvg = async () => {
  1169. // await import('@/assets/canvas2svg');
  1170. for (const pen of meta2d.store.data.pens) {
  1171. if (pen.calculative.img) {
  1172. //重新生成绘制图片
  1173. pen.onRenderPenRaw?.(pen);
  1174. }
  1175. }
  1176. if (!C2S) {
  1177. MessagePlugin.error('请先加载乐吾乐官网下的canvas2svg.js');
  1178. return;
  1179. }
  1180. const rect: any = meta2d.getRect();
  1181. if (!isFinite(rect.width)) {
  1182. MessagePlugin.error(downloadImageTips);
  1183. return;
  1184. }
  1185. rect.x -= 10;
  1186. rect.y -= 10;
  1187. const ctx = new C2S(rect.width + 20, rect.height + 20);
  1188. ctx.textBaseline = 'middle';
  1189. ctx.strokeStyle = getGlobalColor(meta2d.store);
  1190. for (const pen of meta2d.store.data.pens) {
  1191. // 不使用 calculative.inView 的原因是,如果 pen 在 view 之外,那么它的 calculative.inView 为 false,但是它的绘制还是需要的
  1192. if (!isShowChild(pen, meta2d.store) || pen.visible == false) {
  1193. continue;
  1194. }
  1195. meta2d.renderPenRaw(ctx, pen, rect);
  1196. }
  1197. let mySerializedSVG = ctx.getSerializedSvg();
  1198. let icon_pens = meta2d.store.data.pens.filter(
  1199. (item) => item.iconFamily && item.icon
  1200. );
  1201. if (icon_pens && icon_pens.length > 0) {
  1202. let iconList = [
  1203. '/icon/国家电网/iconfont.css',
  1204. '/icon/电气工程/iconfont.css',
  1205. '/icon/通用图标/iconfont.css',
  1206. ];
  1207. let defsList: any = await Promise.all(
  1208. iconList.map((item) => getIconDefs(item))
  1209. );
  1210. mySerializedSVG = mySerializedSVG.replace(
  1211. '<defs/>',
  1212. `<defs>
  1213. <style type="text/css">
  1214. ${defsList.join('\n')}
  1215. </style>
  1216. {{bk}}
  1217. </defs>
  1218. {{bkRect}}`
  1219. );
  1220. }
  1221. /* mySerializedSVG = mySerializedSVG.replace(
  1222. '<defs/>',
  1223. `<defs>
  1224. <style type="text/css">
  1225. @font-face {
  1226. font-family: 'ticon';
  1227. src: url('icon/通用图标/iconfont.ttf') format('truetype');
  1228. }
  1229. </style>
  1230. {{bk}}
  1231. </defs>
  1232. {{bkRect}}`
  1233. );
  1234. */
  1235. if (meta2d.store.data.background) {
  1236. mySerializedSVG = mySerializedSVG.replace('{{bk}}', '');
  1237. mySerializedSVG = mySerializedSVG.replace(
  1238. '{{bkRect}}',
  1239. `<rect x="0" y="0" width="100%" height="100%" fill="${meta2d.store.data.background}"></rect>`
  1240. );
  1241. } else {
  1242. mySerializedSVG = mySerializedSVG.replace('{{bk}}', '');
  1243. mySerializedSVG = mySerializedSVG.replace('{{bkRect}}', '');
  1244. }
  1245. mySerializedSVG = mySerializedSVG.replace(/--le5le--/g, '&#x');
  1246. const urlObject: any = (window as any).URL || window;
  1247. const export_blob = new Blob([mySerializedSVG]);
  1248. const url = urlObject.createObjectURL(export_blob);
  1249. const a = document.createElement('a');
  1250. a.setAttribute(
  1251. 'download',
  1252. `${(meta2d.store.data as Meta2dBackData).name || 'le5le.meta2d'}.svg`
  1253. );
  1254. a.setAttribute('href', url);
  1255. const evt = document.createEvent('MouseEvents');
  1256. evt.initEvent('click', true, true);
  1257. a.dispatchEvent(evt);
  1258. };
  1259. const onUndo = () => {
  1260. meta2d.undo();
  1261. };
  1262. const onRedo = () => {
  1263. meta2d.redo();
  1264. };
  1265. const onCut = () => {
  1266. meta2d.cut();
  1267. };
  1268. const onCopy = () => {
  1269. meta2d.copy();
  1270. };
  1271. const onPaste = () => {
  1272. meta2d.paste();
  1273. };
  1274. const onAll = () => {
  1275. meta2d.activeAll();
  1276. };
  1277. const onDelete = () => {
  1278. meta2d.delete();
  1279. };
  1280. const onToggleAnchor = () => {
  1281. //取消连线状态
  1282. // meta2d.store.options.disableAnchor = false;
  1283. if (!meta2d.store.options.disableAnchor) {
  1284. meta2d.canvas.drawingLineName && drawPen();
  1285. meta2d.toggleAnchorMode();
  1286. }
  1287. };
  1288. const onAddAnchorHand = () => {
  1289. meta2d.addAnchorHand();
  1290. };
  1291. const onRemoveAnchorHand = () => {
  1292. meta2d.removeAnchorHand();
  1293. };
  1294. const onToggleAnchorHand = () => {
  1295. meta2d.toggleAnchorHand();
  1296. };
  1297. const onScaleUp = () => {
  1298. const _scale = meta2d.store.data.scale + 0.1;
  1299. meta2d.scale(_scale);
  1300. };
  1301. const onScaleDown = () => {
  1302. const _scale = meta2d.store.data.scale - 0.1;
  1303. meta2d.scale(_scale);
  1304. };
  1305. const autoAnchor = ref(true);
  1306. const onAutoAnchor = () => {
  1307. meta2d.store.options.autoAnchor = !meta2d.store.options.autoAnchor;
  1308. autoAnchor.value = meta2d.store.options.autoAnchor;
  1309. };
  1310. const showAnchor = ref(false);
  1311. const onDisableAnchor = () => {
  1312. meta2d.store.options.disableAnchor = !meta2d.store.options.disableAnchor;
  1313. changeDisableAnchor();
  1314. };
  1315. const changeDisableAnchor = () => {
  1316. const { disableAnchor, autoAnchor } = meta2d.store.options;
  1317. showAnchor.value = !disableAnchor || false;
  1318. if (disableAnchor && autoAnchor) {
  1319. // 禁用瞄点开了,需要关闭自动瞄点
  1320. onAutoAnchor();
  1321. }
  1322. };
  1323. </script>
  1324. <style lang="postcss" scoped>
  1325. .app-header {
  1326. display: flex;
  1327. height: 40px;
  1328. background-color: var(--color-background);
  1329. position: relative;
  1330. z-index: 2;
  1331. .logo {
  1332. display: flex;
  1333. padding: 0 16px;
  1334. align-items: center;
  1335. font-size: 14px;
  1336. font-weight: 500;
  1337. img {
  1338. height: 20px;
  1339. margin-right: 6px;
  1340. }
  1341. }
  1342. a {
  1343. display: flex;
  1344. padding: 0 8px;
  1345. margin: 0 8px;
  1346. align-items: center;
  1347. color: var(--color);
  1348. text-decoration: none;
  1349. white-space: nowrap;
  1350. &:hover {
  1351. color: var(--color-primary);
  1352. }
  1353. svg {
  1354. font-size: 15px;
  1355. margin: 2px 4px 0 0;
  1356. }
  1357. &.active {
  1358. background-color: var(--color-primary);
  1359. color: #ffffff;
  1360. }
  1361. }
  1362. input {
  1363. font-size: var(--font-size);
  1364. flex-grow: 1;
  1365. background: none;
  1366. outline: none;
  1367. border: none;
  1368. text-align: center;
  1369. color: var(--color-title);
  1370. }
  1371. }
  1372. </style>