1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102 |
- {__NOLAYOUT__}
- <!doctype html>
- <html lang="en">
- <head>
- <meta charset="UTF-8">
- <meta name="viewport"
- content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0">
- <meta http-equiv="X-UA-Compatible" content="ie=edge">
- <title>自定义海报</title>
- <link rel="stylesheet" href="//unpkg.com/element-ui/lib/theme-chalk/index.css">
- <link rel="stylesheet" href="//at.alicdn.com/t/font_2759699_mqr3liwciis.css" media="all">
- <style>
- @font-face {
- font-family: 'SourceHanSans';
- font-display: swap;
- src: url('/assets/addons/posters/SourceHanSansCN-Regular.ttf');
- }
- body {
- margin: 0;
- padding: 0;
- }
- .container {
- max-width: 1200px;
- position: relative;
- margin: 0px auto;
- }
- .box-shadow-h {
- cursor: pointer;
- }
- .box-shadow, .box-shadow-h:hover {
- 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;
- }
- .content {
- width: 100%;
- height: 100vh;
- display: flex;
- align-items: center;
- position: relative;
- margin: 0 auto;
- }
- .poster {
- position: relative;
- border-radius: 3px;
- border: 1px double #DFECFD;
- z-index: 9;
- }
- .config {
- width: 350px;
- min-height: 750px;
- position: absolute;
- margin-left: 100px;
- }
- .config .title {
- margin: 10px auto 20px;
- font-size: 24px;
- font-weight: bold;
- }
- .config .name {
- margin: 10px auto;
- font-size: 15px;
- font-weight: bold;
- position: relative;
- padding-left: 10px;
- }
- .config .name:after {
- content: '';
- width: 4px;
- height: 100%;
- position: absolute;
- left: 0;
- border-radius: 5px;
- background-color: #4988FD;
- }
- .config .box {
- margin-bottom: 20px;
- }
- .config .box.form .el-form-item {
- margin-bottom: 10px;
- }
- .config .box.types {
- text-align: center;
- }
- .config .types .type {
- display: inline-block;
- text-align: center;
- padding: 6px 12px;
- border-radius: 10px;
- width: 70px;
- position: relative;
- }
- .config .types .type > .iconfont {
- font-size: 45px;
- }
- .config .types .type > .tag {
- font-size: 12px;
- color: #797676;
- }
- .el-upload--picture-card {
- width: 80px;
- height: 80px;
- line-height: 84px;
- }
- .no-select {
- -webkit-user-select: none;
- -moz-user-select: none;
- -ms-user-select: none;
- user-select: none;
- }
- .drag {
- position: absolute;
- }
- .drag > .text.space > .text-content {
- word-break: break-all;
- white-space: pre-wrap;
- }
- .drag > .text.ellipsis {
- overflow: hidden;
- white-space: nowrap;
- text-overflow: ellipsis;
- }
- .drag > .text > .text-content {
- font-family: SourceHanSans !important;
- margin: 0;
- }
- .drag > .el-icon-close {
- display: none;
- cursor: pointer;
- position: absolute;
- bottom: -20px;
- font-size: 18px;
- left: 50%;
- color: #000;
- transform: translate(-50%);
- }
- .drag.current > .el-icon-close {
- display: block;
- }
- .drag > img {
- width: 100%;
- height: 100%;
- }
- /*覆盖图片防止被选中*/
- .poster .cover {
- cursor: move;
- top: 0;
- left: 0;
- right: 0;
- bottom: 0;
- position: absolute;
- user-select: none;
- }
- .drag.current > .cover, .poster.current {
- border: 1px dotted #000;
- }
- .marker {
- font-size: 12px;
- line-height: 20px;
- color: #818489;
- }
- .input-label {
- font-size: 12px;
- }
- .scale {
- position: absolute;
- background: #fff;
- border: 1px solid #000;
- width: 7px;
- height: 7px;
- z-index: 1;
- display: none;
- }
- .poster.current > .cover > .scale, .drag.current > .cover > .scale {
- display: block;
- }
- .scale-nw {
- top: -3.5px;
- left: -3.5px;
- cursor: nw-resize;
- border-radius: 50%;
- }
- .scale-ne {
- top: -3.5px;
- right: -3.5px;
- cursor: ne-resize;
- border-radius: 50%;
- }
- .scale-sw {
- bottom: -3.5px;
- left: -3.5px;
- cursor: sw-resize;
- border-radius: 50%;
- }
- .scale-se {
- bottom: -3.5px;
- right: -3.5px;
- cursor: se-resize;
- border-radius: 50%;
- }
- .scale-n {
- top: -3.5px;
- left: 50%;
- margin-left: -3.5px;
- cursor: n-resize;
- }
- .scale-e {
- right: -3px;
- top: 50%;
- margin-top: -3.5px;
- cursor: e-resize;
- }
- .scale-s {
- bottom: -3px;
- left: 50%;
- margin-left: -3.5px;
- cursor: s-resize;
- }
- .scale-w {
- left: -3.5px;
- top: 50%;
- margin-top: -3.5px;
- cursor: w-resize;
- }
- #manageIframe {
- width: 100%;
- height: 500px;
- }
- .manageDialog .el-dialog__body {
- padding: 0 10px;
- }
- .manageDialog > .el-dialog {
- min-width: 815px;
- }
- .submit {
- padding-left: 80px;
- }
- .content .lists {
- z-index: 6;
- min-height: 750px;
- min-height: 750px;
- position: absolute;
- padding-top: 10px;
- margin-left: 1px;
- }
- .content .lists .item {
- border: 1px solid #cfcfd4;
- border-left: none;
- padding: 0px 2px;
- margin-bottom: 10px;
- box-shadow: 0 2px 4px 0 rgb(0 0 0 / 5%);
- border-radius: 0px 100px 100px 0px;
- width: 50px;
- transition: width .3s;
- cursor: pointer;
- min-height: 40px;
- line-height: 40px;
- position: relative;
- }
- .content .lists.item.bg {
- color: #000000;
- }
- .content .lists .item .el-icon-close {
- position: absolute;
- right: 5px;
- top: 50%;
- transform: translateY(-50%);
- display: none;
- }
- .content .lists .item.current .el-icon-close {
- display: inline;
- }
- .content .lists .item.current {
- border-color: #88888b;
- width: 90px;
- }
- .content .lists .item > .image {
- display: block;
- width: 30px;
- margin: 5px 5px;
- user-select: none;
- }
- .content .lists .item > .text {
- overflow: hidden;
- white-space: nowrap;
- max-width: 65px;
- /*text-overflow: ellipsis;*/
- }
- </style>
- </head>
- <body>
- <div id="app">
- <div class="container">
- <div class="content">
- <div class="poster" :class="{current: -1 === currentIndex}" @mousedown.stop="tapBg"
- :style="bgStyle">
- <div class="drag"
- @click.stop="void(0)"
- v-for="(item, index) in materials"
- @mousedown.stop="mousedown($event, index)" :key="index"
- :class="{current: index === currentIndex}"
- :style="makeMaterialStyle(item)">
- <img v-if="item.type === 'qr'" src="/assets/addons/posters/img/qrcode.png"
- alt="">
- <img v-else-if="item.type === 'image'" alt=""
- :style="{borderRadius: item.config.radius + 'px'}"
- :src="(!item.generate && item.config.image) ? item.config.image : '/assets/addons/posters/img/image.png'">
- <div v-else-if="item.type === 'text'" class="text"
- :class="item.config.overflow">
- <pre class="text-content"
- :style="{lineHeight: item.config.lineHeight + 'px'}">{{ item.config.text }}</pre>
- </div>
- <i class="el-icon-close" @click.stop="delPoster(index)"></i>
- <div class="cover">
- <div v-if="item.type !== 'text'" class="scale scale-nw"
- @mousedown.stop="shape($event, 'nw', false, true)"></div>
- <div v-if="item.type !== 'text'" class="scale scale-ne"
- @mousedown.stop="shape($event, 'ne', false, true)"></div>
- <div v-if="item.type !== 'text'" class="scale scale-sw"
- @mousedown.stop="shape($event, 'sw', false, true)"></div>
- <div v-if="item.type !== 'text'" class="scale scale-se"
- @mousedown.stop="shape($event, 'se', false, true)"></div>
- <div v-if="item.type === 'image' " class="scale scale-n"
- @mousedown.stop="shape($event, 'n')"></div>
- <div v-if="item.type !== 'qr'" class="scale scale-e"
- @mousedown.stop="shape($event, 'e')"></div>
- <div v-if="item.type === 'image'" class="scale scale-s"
- @mousedown.stop="shape($event, 's')"></div>
- <div v-if="item.type !== 'qr'" class="scale scale-w"
- @mousedown.stop="shape($event, 'w')"></div>
- </div>
- </div>
- <div class="cover">
- <div class="scale scale-n" @mousedown.stop="shape($event, 'n', true)"></div>
- <div class="scale scale-e" @mousedown.stop="shape($event, 'e', true)"></div>
- <div class="scale scale-s" @mousedown.stop="shape($event, 's', true)"></div>
- <div class="scale scale-w" @mousedown.stop="shape($event, 'w', true)"></div>
- </div>
- </div>
- <div class="lists" :style="{left: bg.width + 'px'}">
- <div class="item bg" :class="{current: currentIndex < 0}"
- :style="{backgroundColor: bg.color}" @click="setCurrent(-1)">背景
- </div>
- <div id="parentDrag">
- <div class="item" @click="setCurrent(materials.length - index - 1)"
- :class="{current: reverseCurrentIndex === index}"
- v-for="(item, index) in reverseMaterials"
- :key="index"
- :data-id="index"
- >
- <img v-if="item.type === 'image'" class="image"
- :style="{borderRadius: item.config.radius + 'px'}"
- :src="(!item.generate && item.config.image) ? item.config.image : '/assets/addons/posters/img/image.png'"
- alt="图片"/>
- <img v-if="item.type === 'qr'" class="image"
- src="/assets/addons/posters/img/qrcode.png" alt="二维码"/>
- <div v-if="item.type === 'text'" class="text">{{ item.config.text }}</div>
- <i class="el-icon-close" @click.stop="delPoster(index)"></i>
- </div>
- </div>
- </div>
- <div class="config" :style="{left: bg.width + 'px'}">
- <div class="title">自定义海报</div>
- <div class="name">设计素材</div>
- <div class="box types">
- <div class="type box-shadow-h" v-for="(item, index) in materialTypes"
- :key="index" @click="addPoster(item.value)">
- <i class="iconfont" :class="item.icon"></i>
- <div class="tag">{{item.title}}</div>
- </div>
- </div>
- <div v-if="current">
- <div class="name">素材配置</div>
- <div class="box form">
- <el-form label-width="80px">
- <el-form-item label="动态">
- <el-switch v-model="current.generate"></el-switch>
- <div class="marker">后台动态生成内容</div>
- </el-form-item>
- <el-form-item v-if="current.generate" label="变量">
- <div>{{current.type}}_{{currentIndex}}</div>
- <div class="marker">动态替换 {{current.type}}_{{currentIndex}} 的值</div>
- </el-form-item>
- <el-form-item v-if="current.type === 'image' && !current.generate"
- label="图片">
- <el-button type="primary" size="small" plain
- @click.stop="manageVisible = true">选择图片
- </el-button>
- <div>
- <img v-if="current.config.image" :src="current.config.image"
- style="height: 100px">
- </div>
- </el-form-item>
- <el-dialog class="manageDialog" title="选择图片"
- :visible.sync="manageVisible" :width="maxWidth">
- <iframe id="manageIframe"
- :src="manageUrl"
- frameborder="0"></iframe>
- <span slot="footer" class="dialog-footer">
- <el-button @click.stop="manageVisible = false">取 消</el-button>
- <el-button type="primary"
- @click.stop="selectImage">确 定</el-button>
- </span>
- </el-dialog>
- <el-form-item v-if="current.type === 'text' || current.type === 'qr'"
- label="内容">
- <el-input type="textarea" autosize size="small"
- v-model="current.config.text" clearable></el-input>
- <div class="marker" v-if="current.generate">动态替换内容中的 {:变量} 值</div>
- </el-form-item>
- <el-form-item label="居中">
- <el-button type="primary" size="small" plain @click="center"> 左右居中
- </el-button>
- </el-form-item>
- <el-form-item v-if="current.type === 'qr'" label="边框">
- <el-slider v-model="current.config.margin" :min="0"
- :max="current.config.width"></el-slider>
- </el-form-item>
- <el-form-item v-if="current.type === 'image'" label="尺寸">
- <div>
- <span class="input-label">宽度(px): </span>
- <el-input-number size="small" style="width: 120px" :min="1"
- :max="bg.width"
- v-model="current.config.width"></el-input-number>
- <el-tooltip class="item" effect="dark" content="重置"
- placement="top">
- <el-button size="small" icon="el-icon-refresh"
- @click.stop="refreshImage" circle></el-button>
- </el-tooltip>
- </div>
- <div>
- <span class="input-label">高度(px): </span>
- <el-input-number size="small" style="width: 120px" :min="1"
- :max="bg.height"
- v-model="current.config.height"/>
- </div>
- </el-form-item>
- <el-form-item v-if="current.type === 'text'" label="超出">
- <el-radio-group v-model="current.config.overflow" size="small">
- <el-radio-button v-for="(item, index) in overflows" :key="index"
- :label="item.value">{{item.title}}
- </el-radio-button>
- </el-radio-group>
- <el-input v-if="current.config.overflow === 'ellipsis'"
- v-model="current.config.overflow_text" size="small"
- placeholder="超出部分替换文本" clearable
- style="width: 150px"></el-input>
- </el-form-item>
- <el-form-item v-if="current.type === 'text' || current.type === 'qr'"
- label="宽度">
- <el-slider v-model="current.config.width" :min="1"
- :max="bg.width"></el-slider>
- </el-form-item>
- <el-form-item v-if="current.type === 'image'" label="圆角">
- <el-slider v-model="current.config.radius"
- :max="current.config.width / 2"></el-slider>
- </el-form-item>
- <el-form-item v-if="current.type === 'image' || current.type === 'qr'"
- label="透明度">
- <el-slider v-model="current.config.opacity"/>
- </el-form-item>
- <el-form-item v-if="current.type === 'text'" label="大小">
- <el-slider v-model="current.config.fontSize" :min="10"
- :max="100"></el-slider>
- </el-form-item>
- <el-form-item v-if="current.type === 'text'" label="行高">
- <el-slider v-model="current.config.lineHeight" :min="10"
- :max="100"></el-slider>
- </el-form-item>
- <el-form-item v-if="current.type === 'text'" label="颜色">
- <el-color-picker color-format="rgb" show-alpha
- v-model="current.config.color"></el-color-picker>
- </el-form-item>
- </el-form>
- </div>
- </div>
- <div v-else>
- <div class="name">海报背景</div>
- <div class="box form">
- <el-form label-width="80px">
- <el-form-item label="标题">
- <el-input v-model="title" size="small" clearable/>
- </el-form-item>
- <el-form-item label="背景色">
- <el-color-picker color-format="rgb"
- v-model="bg.color"></el-color-picker>
- </el-form-item>
- <el-form-item label="尺寸">
- <div>
- <span class="input-label">宽度(px): </span>
- <el-input-number size="small" :min="1" :max="maxWidth"
- v-model="bg.width"/>
- </div>
- <div>
- <span class="input-label">高度(px): </span>
- <el-input-number size="small" :min="1" :max="maxHeight"
- v-model="bg.height"/>
- </div>
- </el-form-item>
- </el-form>
- </div>
- </div>
- <div class="submit">
- <el-button type="primary" @click="submit">保存</el-button>
- </div>
- </div>
- </div>
- </div>
- </div>
- <script src="//cdn.staticfile.org/vue/2.6.9/vue.min.js"></script>
- <script src="//unpkg.com/element-ui/lib/index.js"></script>
- <script src="//cdn.staticfile.org/axios/0.21.1/axios.min.js"></script>
- <script src="//cdn.staticfile.org/Sortable/1.14.0/Sortable.min.js"></script>
- <script>
- const materialsDefault = {
- image: {
- type: 'image',
- generate: true,
- zIndex: 1,
- config: {
- image: null,
- left: 0,
- top: 0,
- width: 80,
- height: 80,
- radius: 0,
- opacity: 100, //透明度
- }
- },
- qr: {
- type: 'qr',
- generate: true,
- zIndex: 1,
- config: {
- text: 'https://baidu.com/s?wd={\:id}',
- left: 0,
- top: 0,
- width: 100,
- margin: 2,
- opacity: 100,
- }
- },
- text: {
- type: 'text',
- generate: true,
- zIndex: 1,
- config: {
- text: '自定义文本{\:name}',
- left: 0,
- top: 0,
- width: 270,
- fontSize: 20,
- lineHeight: 20,
- overflow: 'space',
- overflow_text: '',
- color: 'rgba(0, 0, 0, 1)'
- }
- }
- }
- const clone = function () {
- return function f(obj) {
- if (!obj) {
- return obj;
- }
- if (obj instanceof Date) {
- return new Date(obj);
- }
- if (obj instanceof RegExp) {
- return new RegExp(obj);
- }
- if (obj === null) {
- return obj;
- }
- if (typeof obj !== 'object') {
- return obj;
- }
- let result = obj instanceof Array ? [] : {};
- for (let key in obj) {
- if (obj.hasOwnProperty(key)) {
- result[key] = typeof obj[key] === 'object' ? f(obj[key]) : obj[key];
- }
- }
- return result;
- }
- }()
- const CREATE = !!"{$create|default=true}"
- let id = "{$id|default=0}",
- manageUrl = "{:url('/general/attachment/select', ['mimetype'=>'image/*'])}"
- const app = new Vue({
- el: '#app',
- data() {
- return {
- title: '',
- maxWidth: 750,
- maxHeight: 750,
- bg: {
- color: 'rgb(255,255,255)',
- width: 422,
- height: 750,
- },
- overflows: [
- {title: '换行', value: 'space'},
- {title: '省略', value: 'ellipsis'},
- ],
- currentIndex: -1,
- materialTypes: {
- image: {
- icon: 'icon-pic',
- title: '图片',
- value: 'image'
- },
- qr: {
- icon: 'icon-qr',
- title: '二维码',
- value: 'qr'
- },
- text: {
- icon: 'icon-input',
- title: '文字',
- value: 'text',
- },
- },
- materials: [],
- manageVisible: false,
- sortable: null,
- }
- },
- computed: {
- manageUrl() {
- return this.manageVisible ? manageUrl : ''
- },
- current() {
- return this.materials[this.currentIndex]
- },
- bgStyle() {
- return {
- width: this.bg.width + 'px',
- height: this.bg.height + 'px',
- backgroundColor: this.bg.color
- }
- },
- reverseCurrentIndex(){
- return this.currentIndex >= 0 ? this.materials.length - this.currentIndex - 1 : -1 ;
- },
- reverseMaterials(){
- let materials = []
- this.materials.forEach(v => {
- materials.unshift(v)
- })
- return materials
- }
- },
- created() {
- if (!CREATE) {
- this.init();
- }
- },
- mounted() {
- this.keyDown()
- const that = this
- this.sortable = Sortable.create(document.getElementById('parentDrag'), {
- animation: 150,
- onEnd: function (evt) {
- let length = that.materials.length
- this.toArray().forEach( (i, k) => {
- let index = length - i - 1
- that.materials[index].zIndex = length - k
- })
- },
- });
- },
- methods: {
- shape(downEvent, direction, bg = false, scale = false) {
- let config = bg ? this.bg : this.current.config
- let maxWidth = bg ? this.maxWidth : this.bg.width
- let maxHeight = bg ? this.maxHeight : this.bg.height
- let startX = downEvent.clientX
- let startY = downEvent.clientY
- let height = config['height'] || 0
- let width = config['width'] || 0
- let top = config['top'] || 0
- let left = config['left'] || 0
- let containerLeft = document.getElementsByClassName('container')[0].offsetLeft
- let containerTop = document.getElementsByClassName('poster')[0].offsetTop
- let move = moveEvent => {
- let currX = moveEvent.clientX
- let currY = moveEvent.clientY
- let disY = currY - startY
- let disX = currX - startX
- let hasN = /n/.test(direction)
- let hasS = /s/.test(direction)
- let hasW = /w/.test(direction)
- let hasE = /e/.test(direction)
- let newWidth = +width + (hasW ? -disX : hasE ? disX : 1)
- let newHeight = +height + (hasN ? -disY : hasS ? disY : 1)
- newWidth = newWidth > 0 ? (newWidth > maxWidth ? maxWidth : newWidth) : 1
- newHeight = newHeight > 0 ? (newHeight > maxHeight ? maxHeight : newHeight) : 1
- newHeight = scale ? parseInt(height * (newWidth / width)) : newHeight
- let checkW = startX - containerLeft - width + newWidth > maxWidth
- let checkN = startY - containerTop - height + newHeight > maxHeight
- hasW = hasW || checkW
- hasN = hasN || checkN
- config['height'] !== undefined && (config['height'] = newHeight)
- config['width'] !== undefined && (config['width'] = newWidth)
- config['left'] !== undefined && hasW && (config['left'] = this.calcLocal(+left + disX, true))
- config['top'] !== undefined && hasN && (config['top'] = this.calcLocal(+top + ((scale && !checkN) ? height - newHeight : disY), false))
- }
- let up = () => {
- document.removeEventListener('mousemove', move)
- document.removeEventListener('mouseup', up)
- }
- document.addEventListener('mousemove', move)
- document.addEventListener('mouseup', up)
- },
- refreshImage() {
- this.setImage(this.current.config.image)
- },
- selectImage() {
- const iframe = document.getElementById('manageIframe').contentWindow
- let selected = iframe.surface_selection
- if (selected.length > 1) {
- this.$message.error('只能选择一张图片');
- return;
- } else if (selected.length < 1) {
- return;
- }
- this.setImage(selected[0].url)
- this.manageVisible = false
- },
- setImage(url) {
- const img = new Image()
- img.src = url
- img.onload = () => {
- this.current.config.image = url
- if (img.width > this.bg.width || img.height > this.bg.height) {
- let wScale = this.bg.width / img.width,
- hScale = this.bg.height / img.height,
- scale = wScale < hScale ? wScale : hScale
- this.current.config.width = Math.round(img.width * scale)
- this.current.config.height = Math.round(img.height * scale)
- } else {
- this.current.config.width = img.width
- this.current.config.height = img.height
- }
- if (this.current.config.width + this.current.config.left > this.bg.width) {
- this.current.config.left = 0
- } else if (this.current.config.height + this.current.config.top > this.bg.height) {
- this.current.config.top = 0
- }
- }
- },
- keyDown() {
- document.onkeydown = (e) => {
- let e1 = e || event || window.event || arguments.callee.caller.arguments[0]
- if (e1 && this.current) {
- let step = 1
- switch (e1.keyCode) {
- case 46: // 删除
- this.delPoster()
- break;
- case 37: // ←
- this.current.config.left = this.calcLocal(this.current.config.left - step, true)
- break;
- case 38: // ↑
- this.current.config.top = this.calcLocal(this.current.config.top - step, false)
- break;
- case 39: // →
- this.current.config.left = this.calcLocal(this.current.config.left + step, true)
- break;
- case 40: // ↓
- this.current.config.top = this.calcLocal(this.current.config.top + step, false)
- break;
- }
- }
- }
- },
- center() {
- this.current.config.left = (this.bg.width - this.current.config.width) / 2
- },
- tapBg() {
- this.currentIndex = -1
- },
- addPoster(value) {
- let material = clone(materialsDefault[value])
- switch (material.type) {
- case 'text':
- material.config.width = this.bg.width
- break;
- }
- this.materials.push(material)
- this.currentIndex = this.materials.length - 1
- this.materials.forEach( (item, k) => { item.zIndex = k + 1 })
- },
- delPoster(index = null) {
- this.materials.splice(index === null ? this.currentIndex : index, 1)
- },
- makeMaterialStyle(item) {
- let style = {
- zIndex: item.zIndex,
- left: item.config.left + 'px',
- top: item.config.top + 'px',
- width: item.config.width + 'px'
- }
- switch (item.type) {
- case 'image':
- style.height = item.config.height + 'px'
- style.borderRadius = item.config.radius + 'px'
- style.opacity = item.config.opacity / 100
- break;
- case 'text':
- style.fontSize = item.config.fontSize + 'px'
- style.color = item.config.color
- break;
- case 'qr':
- if (item.config.margin > 0) {
- style.width = item.config.width - 2 * item.config.margin + 'px'
- style.padding = item.config.margin + 'px'
- style.backgroundColor = '#fff'
- }
- style.height = style.width
- style.opacity = item.config.opacity / 100
- break;
- }
- return style
- },
- calcLocal(val, direction = true, current = null) {
- if (null === current) {
- current = this.current
- }
- if (direction) {
- let size = current.config.width
- return val > this.bg.width - size ? this.bg.width - size : (val < 0 ? 0 : val)
- } else {
- let size = 0
- switch (current.type) {
- case 'image':
- size = current.config.height
- break;
- case 'text':
- size = current.config.fontSize
- break;
- case 'qr':
- size = current.config.width
- break;
- }
- return val > this.bg.height - size ? this.bg.height - size : (val < 0 ? 0 : val)
- }
- },
- setCurrent(index) {
- this.currentIndex = index
- },
- mousedown(downEvent, index) {
- this.setCurrent(index)
- let startTop = this.current.config.top,
- startLeft = this.current.config.left,
- clientX = downEvent.clientX,
- clientY = downEvent.clientY
- let move = moveEvent => {
- let currX = moveEvent.clientX
- let currY = moveEvent.clientY
- this.current.config.left = this.calcLocal(currX - clientX + startLeft, true)
- this.current.config.top = this.calcLocal(currY - clientY + startTop, false)
- }
- let up = () => {
- document.removeEventListener('mousemove', move)
- document.removeEventListener('mouseup', up)
- }
- document.addEventListener('mousemove', move)
- document.addEventListener('mouseup', up)
- },
- colorToVal(color) {
- return color.match(/\d+,\d+,\d+/g)
- },
- setError(err, index = -1) {
- if (!isNaN(err)) {
- index = err
- } else {
- this.$message.error(err);
- }
- this.currentIndex = parseInt(index)
- },
- checkMaterials() {
- if (!this.title) {
- this.setError('请设置标题', -1);
- return false;
- }
- for (let i in this.materials) {
- let v = this.materials[i],
- c = v.config
- switch (v.type) {
- case 'image':
- if (!v.generate && !c.image) {
- this.setError('请选择素材图片', i);
- return false;
- }
- break;
- case 'qr':
- if (!c.text) {
- this.setError('请设置二维码内容', i);
- return false;
- }
- break;
- case 'text':
- if (!v.generate && !c.text) {
- this.setError('请设置文本内容', i);
- return false;
- }
- break;
- }
- }
- return true;
- },
- submit() {
- if (true !== this.checkMaterials()) {
- return;
- }
- axios.post('', {title: this.title, bg: this.bg, materials: this.materials}, {
- headers: {'X-Requested-With': 'XMLHttpRequest'},
- }).then(response => {
- let res = response.data
- if (res.code === 1) {
- let parent = window.parent
- if (parent && parent.layer) {
- parent.$("#table").bootstrapTable('refresh', {});
- parent.layer.close(parent.layer.getFrameIndex(window.name))
- } else {
- if (CREATE) {
- this.reset();
- }
- this.$message.success(res.msg);
- }
- } else {
- this.$message.error(res.msg);
- }
- }).catch(error => {
- console.log(error);
- });
- },
- init() {
- axios.get("{:url('detail')}", {
- params: {id},
- headers: {'X-Requested-With': 'XMLHttpRequest'},
- }).then(response => {
- let res = response.data
- if (res.code === 1) {
- this.bg = res.data.bg
- this.title = res.data.title
- res.data.materials.forEach((v, k) => {
- v.zIndex = k + 1
- })
- this.materials = res.data.materials
- } else {
- this.$message.error(res.msg);
- }
- }).catch(error => {
- console.log(error);
- });
- },
- reset() {
- this.bg = {
- color: 'rgb(255,255,255)',
- width: 422,
- height: 750,
- };
- this.title = '';
- this.currentIndex = -1;
- this.materials = [];
- },
- }
- })
- // 本页面不需要Layer 只需要子页面文件选择 重写Layer方法
- const Layer = {
- getFrameIndex() {
- return 0
- },
- close(index) {
- }
- }
- let config = {$config | json_encode};
- let cdnurl = config.upload.cdnurl;
- window.$ = function () {
- return {
- data() {
- const sel = function (data) {
- let url = data.url
- url = (cdnurl && url.indexOf(cdnurl) === 0) ? url : cdnurl + url;
- app.setImage(url)
- app.manageVisible = false
- }
- return sel;
- }
- }
- }
- window.Layer = Layer
- </script>
- </body>
- </html>
|