create.html 41 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102
  1. {__NOLAYOUT__}
  2. <!doctype html>
  3. <html lang="en">
  4. <head>
  5. <meta charset="UTF-8">
  6. <meta name="viewport"
  7. content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
  8. <meta http-equiv="X-UA-Compatible" content="ie=edge">
  9. <title>自定义海报</title>
  10. <link rel="stylesheet" href="//unpkg.com/element-ui/lib/theme-chalk/index.css">
  11. <link rel="stylesheet" href="//at.alicdn.com/t/font_2759699_mqr3liwciis.css" media="all">
  12. <style>
  13. @font-face {
  14. font-family: 'SourceHanSans';
  15. font-display: swap;
  16. src: url('/assets/addons/posters/SourceHanSansCN-Regular.ttf');
  17. }
  18. body {
  19. margin: 0;
  20. padding: 0;
  21. }
  22. .container {
  23. max-width: 1200px;
  24. position: relative;
  25. margin: 0px auto;
  26. }
  27. .box-shadow-h {
  28. cursor: pointer;
  29. }
  30. .box-shadow, .box-shadow-h:hover {
  31. box-shadow: 0 3px 1px -2px rgba(0, 0, 0, .1), 0 2px 2px 0 rgba(0, 0, 0, .1), 0 1px 5px 1px rgba(0, 0, 0, .1) !important;
  32. }
  33. .content {
  34. width: 100%;
  35. height: 100vh;
  36. display: flex;
  37. align-items: center;
  38. position: relative;
  39. margin: 0 auto;
  40. }
  41. .poster {
  42. position: relative;
  43. border-radius: 3px;
  44. border: 1px double #DFECFD;
  45. z-index: 9;
  46. }
  47. .config {
  48. width: 350px;
  49. min-height: 750px;
  50. position: absolute;
  51. margin-left: 100px;
  52. }
  53. .config .title {
  54. margin: 10px auto 20px;
  55. font-size: 24px;
  56. font-weight: bold;
  57. }
  58. .config .name {
  59. margin: 10px auto;
  60. font-size: 15px;
  61. font-weight: bold;
  62. position: relative;
  63. padding-left: 10px;
  64. }
  65. .config .name:after {
  66. content: '';
  67. width: 4px;
  68. height: 100%;
  69. position: absolute;
  70. left: 0;
  71. border-radius: 5px;
  72. background-color: #4988FD;
  73. }
  74. .config .box {
  75. margin-bottom: 20px;
  76. }
  77. .config .box.form .el-form-item {
  78. margin-bottom: 10px;
  79. }
  80. .config .box.types {
  81. text-align: center;
  82. }
  83. .config .types .type {
  84. display: inline-block;
  85. text-align: center;
  86. padding: 6px 12px;
  87. border-radius: 10px;
  88. width: 70px;
  89. position: relative;
  90. }
  91. .config .types .type > .iconfont {
  92. font-size: 45px;
  93. }
  94. .config .types .type > .tag {
  95. font-size: 12px;
  96. color: #797676;
  97. }
  98. .el-upload--picture-card {
  99. width: 80px;
  100. height: 80px;
  101. line-height: 84px;
  102. }
  103. .no-select {
  104. -webkit-user-select: none;
  105. -moz-user-select: none;
  106. -ms-user-select: none;
  107. user-select: none;
  108. }
  109. .drag {
  110. position: absolute;
  111. }
  112. .drag > .text.space > .text-content {
  113. word-break: break-all;
  114. white-space: pre-wrap;
  115. }
  116. .drag > .text.ellipsis {
  117. overflow: hidden;
  118. white-space: nowrap;
  119. text-overflow: ellipsis;
  120. }
  121. .drag > .text > .text-content {
  122. font-family: SourceHanSans !important;
  123. margin: 0;
  124. }
  125. .drag > .el-icon-close {
  126. display: none;
  127. cursor: pointer;
  128. position: absolute;
  129. bottom: -20px;
  130. font-size: 18px;
  131. left: 50%;
  132. color: #000;
  133. transform: translate(-50%);
  134. }
  135. .drag.current > .el-icon-close {
  136. display: block;
  137. }
  138. .drag > img {
  139. width: 100%;
  140. height: 100%;
  141. }
  142. /*覆盖图片防止被选中*/
  143. .poster .cover {
  144. cursor: move;
  145. top: 0;
  146. left: 0;
  147. right: 0;
  148. bottom: 0;
  149. position: absolute;
  150. user-select: none;
  151. }
  152. .drag.current > .cover, .poster.current {
  153. border: 1px dotted #000;
  154. }
  155. .marker {
  156. font-size: 12px;
  157. line-height: 20px;
  158. color: #818489;
  159. }
  160. .input-label {
  161. font-size: 12px;
  162. }
  163. .scale {
  164. position: absolute;
  165. background: #fff;
  166. border: 1px solid #000;
  167. width: 7px;
  168. height: 7px;
  169. z-index: 1;
  170. display: none;
  171. }
  172. .poster.current > .cover > .scale, .drag.current > .cover > .scale {
  173. display: block;
  174. }
  175. .scale-nw {
  176. top: -3.5px;
  177. left: -3.5px;
  178. cursor: nw-resize;
  179. border-radius: 50%;
  180. }
  181. .scale-ne {
  182. top: -3.5px;
  183. right: -3.5px;
  184. cursor: ne-resize;
  185. border-radius: 50%;
  186. }
  187. .scale-sw {
  188. bottom: -3.5px;
  189. left: -3.5px;
  190. cursor: sw-resize;
  191. border-radius: 50%;
  192. }
  193. .scale-se {
  194. bottom: -3.5px;
  195. right: -3.5px;
  196. cursor: se-resize;
  197. border-radius: 50%;
  198. }
  199. .scale-n {
  200. top: -3.5px;
  201. left: 50%;
  202. margin-left: -3.5px;
  203. cursor: n-resize;
  204. }
  205. .scale-e {
  206. right: -3px;
  207. top: 50%;
  208. margin-top: -3.5px;
  209. cursor: e-resize;
  210. }
  211. .scale-s {
  212. bottom: -3px;
  213. left: 50%;
  214. margin-left: -3.5px;
  215. cursor: s-resize;
  216. }
  217. .scale-w {
  218. left: -3.5px;
  219. top: 50%;
  220. margin-top: -3.5px;
  221. cursor: w-resize;
  222. }
  223. #manageIframe {
  224. width: 100%;
  225. height: 500px;
  226. }
  227. .manageDialog .el-dialog__body {
  228. padding: 0 10px;
  229. }
  230. .manageDialog > .el-dialog {
  231. min-width: 815px;
  232. }
  233. .submit {
  234. padding-left: 80px;
  235. }
  236. .content .lists {
  237. z-index: 6;
  238. min-height: 750px;
  239. min-height: 750px;
  240. position: absolute;
  241. padding-top: 10px;
  242. margin-left: 1px;
  243. }
  244. .content .lists .item {
  245. border: 1px solid #cfcfd4;
  246. border-left: none;
  247. padding: 0px 2px;
  248. margin-bottom: 10px;
  249. box-shadow: 0 2px 4px 0 rgb(0 0 0 / 5%);
  250. border-radius: 0px 100px 100px 0px;
  251. width: 50px;
  252. transition: width .3s;
  253. cursor: pointer;
  254. min-height: 40px;
  255. line-height: 40px;
  256. position: relative;
  257. }
  258. .content .lists.item.bg {
  259. color: #000000;
  260. }
  261. .content .lists .item .el-icon-close {
  262. position: absolute;
  263. right: 5px;
  264. top: 50%;
  265. transform: translateY(-50%);
  266. display: none;
  267. }
  268. .content .lists .item.current .el-icon-close {
  269. display: inline;
  270. }
  271. .content .lists .item.current {
  272. border-color: #88888b;
  273. width: 90px;
  274. }
  275. .content .lists .item > .image {
  276. display: block;
  277. width: 30px;
  278. margin: 5px 5px;
  279. user-select: none;
  280. }
  281. .content .lists .item > .text {
  282. overflow: hidden;
  283. white-space: nowrap;
  284. max-width: 65px;
  285. /*text-overflow: ellipsis;*/
  286. }
  287. </style>
  288. </head>
  289. <body>
  290. <div id="app">
  291. <div class="container">
  292. <div class="content">
  293. <div class="poster" :class="{current: -1 === currentIndex}" @mousedown.stop="tapBg"
  294. :style="bgStyle">
  295. <div class="drag"
  296. @click.stop="void(0)"
  297. v-for="(item, index) in materials"
  298. @mousedown.stop="mousedown($event, index)" :key="index"
  299. :class="{current: index === currentIndex}"
  300. :style="makeMaterialStyle(item)">
  301. <img v-if="item.type === 'qr'" src="/assets/addons/posters/img/qrcode.png"
  302. alt="">
  303. <img v-else-if="item.type === 'image'" alt=""
  304. :style="{borderRadius: item.config.radius + 'px'}"
  305. :src="(!item.generate && item.config.image) ? item.config.image : '/assets/addons/posters/img/image.png'">
  306. <div v-else-if="item.type === 'text'" class="text"
  307. :class="item.config.overflow">
  308. <pre class="text-content"
  309. :style="{lineHeight: item.config.lineHeight + 'px'}">{{ item.config.text }}</pre>
  310. </div>
  311. <i class="el-icon-close" @click.stop="delPoster(index)"></i>
  312. <div class="cover">
  313. <div v-if="item.type !== 'text'" class="scale scale-nw"
  314. @mousedown.stop="shape($event, 'nw', false, true)"></div>
  315. <div v-if="item.type !== 'text'" class="scale scale-ne"
  316. @mousedown.stop="shape($event, 'ne', false, true)"></div>
  317. <div v-if="item.type !== 'text'" class="scale scale-sw"
  318. @mousedown.stop="shape($event, 'sw', false, true)"></div>
  319. <div v-if="item.type !== 'text'" class="scale scale-se"
  320. @mousedown.stop="shape($event, 'se', false, true)"></div>
  321. <div v-if="item.type === 'image' " class="scale scale-n"
  322. @mousedown.stop="shape($event, 'n')"></div>
  323. <div v-if="item.type !== 'qr'" class="scale scale-e"
  324. @mousedown.stop="shape($event, 'e')"></div>
  325. <div v-if="item.type === 'image'" class="scale scale-s"
  326. @mousedown.stop="shape($event, 's')"></div>
  327. <div v-if="item.type !== 'qr'" class="scale scale-w"
  328. @mousedown.stop="shape($event, 'w')"></div>
  329. </div>
  330. </div>
  331. <div class="cover">
  332. <div class="scale scale-n" @mousedown.stop="shape($event, 'n', true)"></div>
  333. <div class="scale scale-e" @mousedown.stop="shape($event, 'e', true)"></div>
  334. <div class="scale scale-s" @mousedown.stop="shape($event, 's', true)"></div>
  335. <div class="scale scale-w" @mousedown.stop="shape($event, 'w', true)"></div>
  336. </div>
  337. </div>
  338. <div class="lists" :style="{left: bg.width + 'px'}">
  339. <div class="item bg" :class="{current: currentIndex < 0}"
  340. :style="{backgroundColor: bg.color}" @click="setCurrent(-1)">背景
  341. </div>
  342. <div id="parentDrag">
  343. <div class="item" @click="setCurrent(materials.length - index - 1)"
  344. :class="{current: reverseCurrentIndex === index}"
  345. v-for="(item, index) in reverseMaterials"
  346. :key="index"
  347. :data-id="index"
  348. >
  349. <img v-if="item.type === 'image'" class="image"
  350. :style="{borderRadius: item.config.radius + 'px'}"
  351. :src="(!item.generate && item.config.image) ? item.config.image : '/assets/addons/posters/img/image.png'"
  352. alt="图片"/>
  353. <img v-if="item.type === 'qr'" class="image"
  354. src="/assets/addons/posters/img/qrcode.png" alt="二维码"/>
  355. <div v-if="item.type === 'text'" class="text">{{ item.config.text }}</div>
  356. <i class="el-icon-close" @click.stop="delPoster(index)"></i>
  357. </div>
  358. </div>
  359. </div>
  360. <div class="config" :style="{left: bg.width + 'px'}">
  361. <div class="title">自定义海报</div>
  362. <div class="name">设计素材</div>
  363. <div class="box types">
  364. <div class="type box-shadow-h" v-for="(item, index) in materialTypes"
  365. :key="index" @click="addPoster(item.value)">
  366. <i class="iconfont" :class="item.icon"></i>
  367. <div class="tag">{{item.title}}</div>
  368. </div>
  369. </div>
  370. <div v-if="current">
  371. <div class="name">素材配置</div>
  372. <div class="box form">
  373. <el-form label-width="80px">
  374. <el-form-item label="动态">
  375. <el-switch v-model="current.generate"></el-switch>
  376. <div class="marker">后台动态生成内容</div>
  377. </el-form-item>
  378. <el-form-item v-if="current.generate" label="变量">
  379. <div>{{current.type}}_{{currentIndex}}</div>
  380. <div class="marker">动态替换 {{current.type}}_{{currentIndex}} 的值</div>
  381. </el-form-item>
  382. <el-form-item v-if="current.type === 'image' && !current.generate"
  383. label="图片">
  384. <el-button type="primary" size="small" plain
  385. @click.stop="manageVisible = true">选择图片
  386. </el-button>
  387. <div>
  388. <img v-if="current.config.image" :src="current.config.image"
  389. style="height: 100px">
  390. </div>
  391. </el-form-item>
  392. <el-dialog class="manageDialog" title="选择图片"
  393. :visible.sync="manageVisible" :width="maxWidth">
  394. <iframe id="manageIframe"
  395. :src="manageUrl"
  396. frameborder="0"></iframe>
  397. <span slot="footer" class="dialog-footer">
  398. <el-button @click.stop="manageVisible = false">取 消</el-button>
  399. <el-button type="primary"
  400. @click.stop="selectImage">确 定</el-button>
  401. </span>
  402. </el-dialog>
  403. <el-form-item v-if="current.type === 'text' || current.type === 'qr'"
  404. label="内容">
  405. <el-input type="textarea" autosize size="small"
  406. v-model="current.config.text" clearable></el-input>
  407. <div class="marker" v-if="current.generate">动态替换内容中的 {:变量} 值</div>
  408. </el-form-item>
  409. <el-form-item label="居中">
  410. <el-button type="primary" size="small" plain @click="center"> 左右居中
  411. </el-button>
  412. </el-form-item>
  413. <el-form-item v-if="current.type === 'qr'" label="边框">
  414. <el-slider v-model="current.config.margin" :min="0"
  415. :max="current.config.width"></el-slider>
  416. </el-form-item>
  417. <el-form-item v-if="current.type === 'image'" label="尺寸">
  418. <div>
  419. <span class="input-label">宽度(px): </span>
  420. <el-input-number size="small" style="width: 120px" :min="1"
  421. :max="bg.width"
  422. v-model="current.config.width"></el-input-number>
  423. <el-tooltip class="item" effect="dark" content="重置"
  424. placement="top">
  425. <el-button size="small" icon="el-icon-refresh"
  426. @click.stop="refreshImage" circle></el-button>
  427. </el-tooltip>
  428. </div>
  429. <div>
  430. <span class="input-label">高度(px): </span>
  431. <el-input-number size="small" style="width: 120px" :min="1"
  432. :max="bg.height"
  433. v-model="current.config.height"/>
  434. </div>
  435. </el-form-item>
  436. <el-form-item v-if="current.type === 'text'" label="超出">
  437. <el-radio-group v-model="current.config.overflow" size="small">
  438. <el-radio-button v-for="(item, index) in overflows" :key="index"
  439. :label="item.value">{{item.title}}
  440. </el-radio-button>
  441. </el-radio-group>
  442. <el-input v-if="current.config.overflow === 'ellipsis'"
  443. v-model="current.config.overflow_text" size="small"
  444. placeholder="超出部分替换文本" clearable
  445. style="width: 150px"></el-input>
  446. </el-form-item>
  447. <el-form-item v-if="current.type === 'text' || current.type === 'qr'"
  448. label="宽度">
  449. <el-slider v-model="current.config.width" :min="1"
  450. :max="bg.width"></el-slider>
  451. </el-form-item>
  452. <el-form-item v-if="current.type === 'image'" label="圆角">
  453. <el-slider v-model="current.config.radius"
  454. :max="current.config.width / 2"></el-slider>
  455. </el-form-item>
  456. <el-form-item v-if="current.type === 'image' || current.type === 'qr'"
  457. label="透明度">
  458. <el-slider v-model="current.config.opacity"/>
  459. </el-form-item>
  460. <el-form-item v-if="current.type === 'text'" label="大小">
  461. <el-slider v-model="current.config.fontSize" :min="10"
  462. :max="100"></el-slider>
  463. </el-form-item>
  464. <el-form-item v-if="current.type === 'text'" label="行高">
  465. <el-slider v-model="current.config.lineHeight" :min="10"
  466. :max="100"></el-slider>
  467. </el-form-item>
  468. <el-form-item v-if="current.type === 'text'" label="颜色">
  469. <el-color-picker color-format="rgb" show-alpha
  470. v-model="current.config.color"></el-color-picker>
  471. </el-form-item>
  472. </el-form>
  473. </div>
  474. </div>
  475. <div v-else>
  476. <div class="name">海报背景</div>
  477. <div class="box form">
  478. <el-form label-width="80px">
  479. <el-form-item label="标题">
  480. <el-input v-model="title" size="small" clearable/>
  481. </el-form-item>
  482. <el-form-item label="背景色">
  483. <el-color-picker color-format="rgb"
  484. v-model="bg.color"></el-color-picker>
  485. </el-form-item>
  486. <el-form-item label="尺寸">
  487. <div>
  488. <span class="input-label">宽度(px): </span>
  489. <el-input-number size="small" :min="1" :max="maxWidth"
  490. v-model="bg.width"/>
  491. </div>
  492. <div>
  493. <span class="input-label">高度(px): </span>
  494. <el-input-number size="small" :min="1" :max="maxHeight"
  495. v-model="bg.height"/>
  496. </div>
  497. </el-form-item>
  498. </el-form>
  499. </div>
  500. </div>
  501. <div class="submit">
  502. <el-button type="primary" @click="submit">保存</el-button>
  503. </div>
  504. </div>
  505. </div>
  506. </div>
  507. </div>
  508. <script src="//cdn.staticfile.org/vue/2.6.9/vue.min.js"></script>
  509. <script src="//unpkg.com/element-ui/lib/index.js"></script>
  510. <script src="//cdn.staticfile.org/axios/0.21.1/axios.min.js"></script>
  511. <script src="//cdn.staticfile.org/Sortable/1.14.0/Sortable.min.js"></script>
  512. <script>
  513. const materialsDefault = {
  514. image: {
  515. type: 'image',
  516. generate: true,
  517. zIndex: 1,
  518. config: {
  519. image: null,
  520. left: 0,
  521. top: 0,
  522. width: 80,
  523. height: 80,
  524. radius: 0,
  525. opacity: 100, //透明度
  526. }
  527. },
  528. qr: {
  529. type: 'qr',
  530. generate: true,
  531. zIndex: 1,
  532. config: {
  533. text: 'https://baidu.com/s?wd={\:id}',
  534. left: 0,
  535. top: 0,
  536. width: 100,
  537. margin: 2,
  538. opacity: 100,
  539. }
  540. },
  541. text: {
  542. type: 'text',
  543. generate: true,
  544. zIndex: 1,
  545. config: {
  546. text: '自定义文本{\:name}',
  547. left: 0,
  548. top: 0,
  549. width: 270,
  550. fontSize: 20,
  551. lineHeight: 20,
  552. overflow: 'space',
  553. overflow_text: '',
  554. color: 'rgba(0, 0, 0, 1)'
  555. }
  556. }
  557. }
  558. const clone = function () {
  559. return function f(obj) {
  560. if (!obj) {
  561. return obj;
  562. }
  563. if (obj instanceof Date) {
  564. return new Date(obj);
  565. }
  566. if (obj instanceof RegExp) {
  567. return new RegExp(obj);
  568. }
  569. if (obj === null) {
  570. return obj;
  571. }
  572. if (typeof obj !== 'object') {
  573. return obj;
  574. }
  575. let result = obj instanceof Array ? [] : {};
  576. for (let key in obj) {
  577. if (obj.hasOwnProperty(key)) {
  578. result[key] = typeof obj[key] === 'object' ? f(obj[key]) : obj[key];
  579. }
  580. }
  581. return result;
  582. }
  583. }()
  584. const CREATE = !!"{$create|default=true}"
  585. let id = "{$id|default=0}",
  586. manageUrl = "{:url('/general/attachment/select', ['mimetype'=>'image/*'])}"
  587. const app = new Vue({
  588. el: '#app',
  589. data() {
  590. return {
  591. title: '',
  592. maxWidth: 750,
  593. maxHeight: 750,
  594. bg: {
  595. color: 'rgb(255,255,255)',
  596. width: 422,
  597. height: 750,
  598. },
  599. overflows: [
  600. {title: '换行', value: 'space'},
  601. {title: '省略', value: 'ellipsis'},
  602. ],
  603. currentIndex: -1,
  604. materialTypes: {
  605. image: {
  606. icon: 'icon-pic',
  607. title: '图片',
  608. value: 'image'
  609. },
  610. qr: {
  611. icon: 'icon-qr',
  612. title: '二维码',
  613. value: 'qr'
  614. },
  615. text: {
  616. icon: 'icon-input',
  617. title: '文字',
  618. value: 'text',
  619. },
  620. },
  621. materials: [],
  622. manageVisible: false,
  623. sortable: null,
  624. }
  625. },
  626. computed: {
  627. manageUrl() {
  628. return this.manageVisible ? manageUrl : ''
  629. },
  630. current() {
  631. return this.materials[this.currentIndex]
  632. },
  633. bgStyle() {
  634. return {
  635. width: this.bg.width + 'px',
  636. height: this.bg.height + 'px',
  637. backgroundColor: this.bg.color
  638. }
  639. },
  640. reverseCurrentIndex(){
  641. return this.currentIndex >= 0 ? this.materials.length - this.currentIndex - 1 : -1 ;
  642. },
  643. reverseMaterials(){
  644. let materials = []
  645. this.materials.forEach(v => {
  646. materials.unshift(v)
  647. })
  648. return materials
  649. }
  650. },
  651. created() {
  652. if (!CREATE) {
  653. this.init();
  654. }
  655. },
  656. mounted() {
  657. this.keyDown()
  658. const that = this
  659. this.sortable = Sortable.create(document.getElementById('parentDrag'), {
  660. animation: 150,
  661. onEnd: function (evt) {
  662. let length = that.materials.length
  663. this.toArray().forEach( (i, k) => {
  664. let index = length - i - 1
  665. that.materials[index].zIndex = length - k
  666. })
  667. },
  668. });
  669. },
  670. methods: {
  671. shape(downEvent, direction, bg = false, scale = false) {
  672. let config = bg ? this.bg : this.current.config
  673. let maxWidth = bg ? this.maxWidth : this.bg.width
  674. let maxHeight = bg ? this.maxHeight : this.bg.height
  675. let startX = downEvent.clientX
  676. let startY = downEvent.clientY
  677. let height = config['height'] || 0
  678. let width = config['width'] || 0
  679. let top = config['top'] || 0
  680. let left = config['left'] || 0
  681. let containerLeft = document.getElementsByClassName('container')[0].offsetLeft
  682. let containerTop = document.getElementsByClassName('poster')[0].offsetTop
  683. let move = moveEvent => {
  684. let currX = moveEvent.clientX
  685. let currY = moveEvent.clientY
  686. let disY = currY - startY
  687. let disX = currX - startX
  688. let hasN = /n/.test(direction)
  689. let hasS = /s/.test(direction)
  690. let hasW = /w/.test(direction)
  691. let hasE = /e/.test(direction)
  692. let newWidth = +width + (hasW ? -disX : hasE ? disX : 1)
  693. let newHeight = +height + (hasN ? -disY : hasS ? disY : 1)
  694. newWidth = newWidth > 0 ? (newWidth > maxWidth ? maxWidth : newWidth) : 1
  695. newHeight = newHeight > 0 ? (newHeight > maxHeight ? maxHeight : newHeight) : 1
  696. newHeight = scale ? parseInt(height * (newWidth / width)) : newHeight
  697. let checkW = startX - containerLeft - width + newWidth > maxWidth
  698. let checkN = startY - containerTop - height + newHeight > maxHeight
  699. hasW = hasW || checkW
  700. hasN = hasN || checkN
  701. config['height'] !== undefined && (config['height'] = newHeight)
  702. config['width'] !== undefined && (config['width'] = newWidth)
  703. config['left'] !== undefined && hasW && (config['left'] = this.calcLocal(+left + disX, true))
  704. config['top'] !== undefined && hasN && (config['top'] = this.calcLocal(+top + ((scale && !checkN) ? height - newHeight : disY), false))
  705. }
  706. let up = () => {
  707. document.removeEventListener('mousemove', move)
  708. document.removeEventListener('mouseup', up)
  709. }
  710. document.addEventListener('mousemove', move)
  711. document.addEventListener('mouseup', up)
  712. },
  713. refreshImage() {
  714. this.setImage(this.current.config.image)
  715. },
  716. selectImage() {
  717. const iframe = document.getElementById('manageIframe').contentWindow
  718. let selected = iframe.surface_selection
  719. if (selected.length > 1) {
  720. this.$message.error('只能选择一张图片');
  721. return;
  722. } else if (selected.length < 1) {
  723. return;
  724. }
  725. this.setImage(selected[0].url)
  726. this.manageVisible = false
  727. },
  728. setImage(url) {
  729. const img = new Image()
  730. img.src = url
  731. img.onload = () => {
  732. this.current.config.image = url
  733. if (img.width > this.bg.width || img.height > this.bg.height) {
  734. let wScale = this.bg.width / img.width,
  735. hScale = this.bg.height / img.height,
  736. scale = wScale < hScale ? wScale : hScale
  737. this.current.config.width = Math.round(img.width * scale)
  738. this.current.config.height = Math.round(img.height * scale)
  739. } else {
  740. this.current.config.width = img.width
  741. this.current.config.height = img.height
  742. }
  743. if (this.current.config.width + this.current.config.left > this.bg.width) {
  744. this.current.config.left = 0
  745. } else if (this.current.config.height + this.current.config.top > this.bg.height) {
  746. this.current.config.top = 0
  747. }
  748. }
  749. },
  750. keyDown() {
  751. document.onkeydown = (e) => {
  752. let e1 = e || event || window.event || arguments.callee.caller.arguments[0]
  753. if (e1 && this.current) {
  754. let step = 1
  755. switch (e1.keyCode) {
  756. case 46: // 删除
  757. this.delPoster()
  758. break;
  759. case 37: // ←
  760. this.current.config.left = this.calcLocal(this.current.config.left - step, true)
  761. break;
  762. case 38: // ↑
  763. this.current.config.top = this.calcLocal(this.current.config.top - step, false)
  764. break;
  765. case 39: // →
  766. this.current.config.left = this.calcLocal(this.current.config.left + step, true)
  767. break;
  768. case 40: // ↓
  769. this.current.config.top = this.calcLocal(this.current.config.top + step, false)
  770. break;
  771. }
  772. }
  773. }
  774. },
  775. center() {
  776. this.current.config.left = (this.bg.width - this.current.config.width) / 2
  777. },
  778. tapBg() {
  779. this.currentIndex = -1
  780. },
  781. addPoster(value) {
  782. let material = clone(materialsDefault[value])
  783. switch (material.type) {
  784. case 'text':
  785. material.config.width = this.bg.width
  786. break;
  787. }
  788. this.materials.push(material)
  789. this.currentIndex = this.materials.length - 1
  790. this.materials.forEach( (item, k) => { item.zIndex = k + 1 })
  791. },
  792. delPoster(index = null) {
  793. this.materials.splice(index === null ? this.currentIndex : index, 1)
  794. },
  795. makeMaterialStyle(item) {
  796. let style = {
  797. zIndex: item.zIndex,
  798. left: item.config.left + 'px',
  799. top: item.config.top + 'px',
  800. width: item.config.width + 'px'
  801. }
  802. switch (item.type) {
  803. case 'image':
  804. style.height = item.config.height + 'px'
  805. style.borderRadius = item.config.radius + 'px'
  806. style.opacity = item.config.opacity / 100
  807. break;
  808. case 'text':
  809. style.fontSize = item.config.fontSize + 'px'
  810. style.color = item.config.color
  811. break;
  812. case 'qr':
  813. if (item.config.margin > 0) {
  814. style.width = item.config.width - 2 * item.config.margin + 'px'
  815. style.padding = item.config.margin + 'px'
  816. style.backgroundColor = '#fff'
  817. }
  818. style.height = style.width
  819. style.opacity = item.config.opacity / 100
  820. break;
  821. }
  822. return style
  823. },
  824. calcLocal(val, direction = true, current = null) {
  825. if (null === current) {
  826. current = this.current
  827. }
  828. if (direction) {
  829. let size = current.config.width
  830. return val > this.bg.width - size ? this.bg.width - size : (val < 0 ? 0 : val)
  831. } else {
  832. let size = 0
  833. switch (current.type) {
  834. case 'image':
  835. size = current.config.height
  836. break;
  837. case 'text':
  838. size = current.config.fontSize
  839. break;
  840. case 'qr':
  841. size = current.config.width
  842. break;
  843. }
  844. return val > this.bg.height - size ? this.bg.height - size : (val < 0 ? 0 : val)
  845. }
  846. },
  847. setCurrent(index) {
  848. this.currentIndex = index
  849. },
  850. mousedown(downEvent, index) {
  851. this.setCurrent(index)
  852. let startTop = this.current.config.top,
  853. startLeft = this.current.config.left,
  854. clientX = downEvent.clientX,
  855. clientY = downEvent.clientY
  856. let move = moveEvent => {
  857. let currX = moveEvent.clientX
  858. let currY = moveEvent.clientY
  859. this.current.config.left = this.calcLocal(currX - clientX + startLeft, true)
  860. this.current.config.top = this.calcLocal(currY - clientY + startTop, false)
  861. }
  862. let up = () => {
  863. document.removeEventListener('mousemove', move)
  864. document.removeEventListener('mouseup', up)
  865. }
  866. document.addEventListener('mousemove', move)
  867. document.addEventListener('mouseup', up)
  868. },
  869. colorToVal(color) {
  870. return color.match(/\d+,\d+,\d+/g)
  871. },
  872. setError(err, index = -1) {
  873. if (!isNaN(err)) {
  874. index = err
  875. } else {
  876. this.$message.error(err);
  877. }
  878. this.currentIndex = parseInt(index)
  879. },
  880. checkMaterials() {
  881. if (!this.title) {
  882. this.setError('请设置标题', -1);
  883. return false;
  884. }
  885. for (let i in this.materials) {
  886. let v = this.materials[i],
  887. c = v.config
  888. switch (v.type) {
  889. case 'image':
  890. if (!v.generate && !c.image) {
  891. this.setError('请选择素材图片', i);
  892. return false;
  893. }
  894. break;
  895. case 'qr':
  896. if (!c.text) {
  897. this.setError('请设置二维码内容', i);
  898. return false;
  899. }
  900. break;
  901. case 'text':
  902. if (!v.generate && !c.text) {
  903. this.setError('请设置文本内容', i);
  904. return false;
  905. }
  906. break;
  907. }
  908. }
  909. return true;
  910. },
  911. submit() {
  912. if (true !== this.checkMaterials()) {
  913. return;
  914. }
  915. axios.post('', {title: this.title, bg: this.bg, materials: this.materials}, {
  916. headers: {'X-Requested-With': 'XMLHttpRequest'},
  917. }).then(response => {
  918. let res = response.data
  919. if (res.code === 1) {
  920. let parent = window.parent
  921. if (parent && parent.layer) {
  922. parent.$("#table").bootstrapTable('refresh', {});
  923. parent.layer.close(parent.layer.getFrameIndex(window.name))
  924. } else {
  925. if (CREATE) {
  926. this.reset();
  927. }
  928. this.$message.success(res.msg);
  929. }
  930. } else {
  931. this.$message.error(res.msg);
  932. }
  933. }).catch(error => {
  934. console.log(error);
  935. });
  936. },
  937. init() {
  938. axios.get("{:url('detail')}", {
  939. params: {id},
  940. headers: {'X-Requested-With': 'XMLHttpRequest'},
  941. }).then(response => {
  942. let res = response.data
  943. if (res.code === 1) {
  944. this.bg = res.data.bg
  945. this.title = res.data.title
  946. res.data.materials.forEach((v, k) => {
  947. v.zIndex = k + 1
  948. })
  949. this.materials = res.data.materials
  950. } else {
  951. this.$message.error(res.msg);
  952. }
  953. }).catch(error => {
  954. console.log(error);
  955. });
  956. },
  957. reset() {
  958. this.bg = {
  959. color: 'rgb(255,255,255)',
  960. width: 422,
  961. height: 750,
  962. };
  963. this.title = '';
  964. this.currentIndex = -1;
  965. this.materials = [];
  966. },
  967. }
  968. })
  969. // 本页面不需要Layer 只需要子页面文件选择 重写Layer方法
  970. const Layer = {
  971. getFrameIndex() {
  972. return 0
  973. },
  974. close(index) {
  975. }
  976. }
  977. let config = {$config | json_encode};
  978. let cdnurl = config.upload.cdnurl;
  979. window.$ = function () {
  980. return {
  981. data() {
  982. const sel = function (data) {
  983. let url = data.url
  984. url = (cdnurl && url.indexOf(cdnurl) === 0) ? url : cdnurl + url;
  985. app.setImage(url)
  986. app.manageVisible = false
  987. }
  988. return sel;
  989. }
  990. }
  991. }
  992. window.Layer = Layer
  993. </script>
  994. </body>
  995. </html>