vue实现PS效果,鼠标拖拽指令、十字辅助线、鼠标选点、打印页面指定内容、生成随机id、颜色选择器、div上输入文字(类似QQ截图输入文字)、vue图片上传转base64...

29 篇文章 3 订阅
18 篇文章 4 订阅

因为涵盖的小细节比较多,所以代码比较长,建议大家复制运行来进行对应功能查看,我在代码中添加了大量的注释,以供大家阅读代码

最终效果

部分细节

1、鼠标选点时可能会超出底图位置

2、图片的background-size在图片上传前是无效的,需要在上传之后设置

3、元素高度取值问题、与父元素相对位置取值问题

......

这些问题我在代码中都有详细的注释

代码实现

index.vue

<template>
    <div id="img-add-content">
        <div class="add-content-img" ref="moveScope">
            <div style="position:relative;">
                <div @click="getMouseTP"
                    @mousemove="getMouseTPSubline"
                    @mouseover="setMoveFlag"
                    style="display: inline-block;"
                    ref="imgBox">
                    <img :src="imgAndContent.backgroundUrl"
                        style="pointer-events: none;"
                        :style="{'height':imgAndContent.backgroundHeight,'width':imgAndContent.backgroundWidth}"
                        alt=""/>
                </div>
                <div id="sublineX" style="pointer-events: none;" :style="{'width':imgAndContent.backgroundWidth}" ref="sublineX"></div>
                <div id="sublineY" style="pointer-events: none;" :style="{'height':imgAndContent.backgroundHeight}" ref="sublineY"></div>
            </div>
            <div class="content-item" v-for="(item,index) in imgAndContent.formList" :key="item.key">
                <div :id="item.key" v-if="item.form.contentType==='0'" v-drag
                    :style="{
                        'position':'absolute',
                        'left':item.form.distanceX ? item.form.distanceX + 'px' : '0px',
                        'top':item.form.distanceY ? item.form.distanceY + 'px' : '0px',
                        'width':item.form.contentWidth ? item.form.contentWidth + 'px' : 'auto',
                        'height':item.form.content=='' ? (item.form.contentSize || '0') + 'px' : '',
                        'fontSize':item.form.contentSize + 'px',
                        'color':item.form.contentColor,
                        'overflow':'hidden'
                    }"
                    :class="!previewType ? 'preview' : ''">
                    <!-- 设置 onselectstart="return false" style="-moz-user-select:none;" -->
                    <!-- 防止用户鼠标误操作造成双击选中文本,影响拖拽操作 -->
                    <div :id="item.key" @input="divIninput(index,item.key)" :ref="'divIninput'+item.key" contenteditable="true" style="outline: none;"></div>
                    <!-- <el-input wrap="physical" class="textarea" type="textarea" autosize v-model="item.form.content"></el-input> -->
                    <!-- <div onselectstart="return false" style="-moz-user-select:none">{{ item.form.content }}{{ index }}</div> -->
                </div>
                <div v-if="item.form.contentType==='1'" v-drag
                    :style="{
                        'position':'absolute',
                        'left':item.form.distanceX ? item.form.distanceX + 'px' : '0px',
                        'top':item.form.distanceY ? item.form.distanceY + 'px' : '0px',
                        'width':item.form.contentWidth ? item.form.contentWidth + 'px' : 'auto',
                        'height':item.form.imgHeight + 'px',
                        'background':`url(${item.form.fileList.length>0 ? item.form.fileList[0].url : ''})`,
                        'backgroundSize': item.form.fileSize,
                    }"
                    :class="!previewType ? 'preview' : ''">
                    <div :id="item.key" style="height: 100%;width: 100%;">
                        <!-- 'background':`url(${item.form.contentImgUrl})`, -->
                        <!-- <img :src="item.content" alt="" style="width:100%;height:100%;pointer-events: none;"/> -->
                    </div>
                </div>
            </div>
        </div>
        <div class="add-content-text">
            <div class="add-content-text-btn">
                <el-button @click="addContentConfig">新增</el-button>
                <el-button @click="submitContentConfig">保存</el-button>
                <el-button @click="openPrint">打印</el-button>
                <el-switch
                    v-model="previewType"
                    active-text="预览"
                    inactive-text="编辑">
                </el-switch>
            </div>
            <div v-for="(item,index) in imgAndContent.formList" :key="item.key" class="content-element-item">
                <el-form :rules="rules" :ref="'form'+ item.key" :model="item.form" label-width="180px">
                    <el-form-item label="鼠标选点">
                        <el-switch
                            v-model="item.form.mousePT"
                            @change="changeMousePTForm(item.form)"
                            active-color="#13ce66"
                            inactive-color="#ff4949">
                        </el-switch>
                    </el-form-item>
                    <el-form-item label="类型">
                        <el-select v-model="item.form.contentType" @change="handleChangeType(item.form,index,'form'+ item.key)" placeholder="请选择类型">
                            <el-option label="文字" :value="'0'"></el-option>
                            <el-option label="图片" :value="'1'"></el-option>
                        </el-select>
                    </el-form-item>
                    <el-form-item label="距图片X轴距离" prop="distanceX">
                        <el-input v-model="item.form.distanceX" @change="beyondScope(item.form,index,item.key)" placeholder="请输入X轴距离"></el-input>
                    </el-form-item>
                    <el-form-item label="距图片Y轴距离" prop="distanceY">
                        <el-input v-model="item.form.distanceY" @change="beyondScope(item.form,index,item.key)" placeholder="请输入Y轴距离"></el-input>
                    </el-form-item>

                    <el-form-item :label="item.form.contentType==='1' ? '图片宽度' : '内容宽度'" prop="contentWidth">
                        <el-input v-model.trim="item.form.contentWidth" @change="beyondScope(item.form,index,item.key)" placeholder="请输入"></el-input>
                    </el-form-item>
                    <el-form-item v-if="item.form.contentType==='0'" label="文本建议宽度">
                        <span class="text-suggest">文本宽度 = 文本内容个数 * 字体大小</span>
                    </el-form-item>

                    <!-- 1、文字类型 -->
                    <el-form-item :key="item.key+'contentSize'" v-if="item.form.contentType==='0'" label="文字大小" prop="contentSize">
                        <el-input v-model.trim="item.form.contentSize" @change="beyondScope(item.form,index,item.key)" placeholder="请输入文字大小"></el-input>
                    </el-form-item>
                    <el-form-item style="width:300px;" :key="item.key+'contentColor'" v-if="item.form.contentType==='0'" label="文字颜色">
                        <el-input type="color" v-model="item.form.contentColor" placeholder="请输入文字大小"></el-input>
                    </el-form-item>
                    <el-form-item :key="item.key+'content'" v-if="item.form.contentType==='0'" label="内容" prop="content">
                        <el-input type="textarea" v-model="item.form.content" @input="setDivContent(item.form,index,item.key)" placeholder="请输入内容"></el-input>
                    </el-form-item>
                    <!-- 2、图片类型 -->
                    <el-form-item :key="item.key+'imgHeight'" v-if="item.form.contentType==='1'" label="图片高度" prop="imgHeight">
                        <el-input v-model.trim="item.form.imgHeight" @change="beyondScope(item.form,index,item.key)" placeholder="请输入"></el-input>
                    </el-form-item>
                    <!-- <el-form-item :key="item.key+'contentImgUrl'" v-if="item.form.contentType==='1'" label="图片地址" prop="contentImgUrl">
                        <el-input v-model="item.form.contentImgUrl" placeholder="请输入内容"></el-input>
                    </el-form-item> -->
                    <el-form-item :key="item.key+'fileList'" v-if="item.form.contentType==='1'" label="上传图片" prop="fileList">
                        <el-upload
                            class="upload-demo"
                            list-type="picture"
                            action="#"
                            accept=".jpg, .png"
                            :limit="1"
                            :auto-upload="false"
                            :file-list="item.form.fileList"
                            :on-change="(file)=>getFile(file,index,item.key)"
                            :on-remove="()=>handleUploadRemove(index)">
                            <el-button size="small" type="primary">选择图片上传</el-button>
                            <div slot="tip" class="el-upload__tip">只能上传一张jpg/png文件</div>
                        </el-upload>
                    </el-form-item>
                    <el-form-item>
                        <el-button @click="delContentConfig(index)">删除</el-button>
                    </el-form-item>
                </el-form>
            </div>
        </div>
    </div>
</template>
<script>
    export default {
        name: 'ImgAddContent',

        data () {
            let checkNumber = (rule, value, callback) => {
                if (/^(?:[1-9]\d*)$/.test(value) == false) {
                    callback(new Error('请输入正整数'));
                } else {
                    callback();
                }
            };
            return {
                // 预览、编辑状态切换
                previewType:true,
                moveFlag:true,
                imgAndContent:{
                    // 需要保存背景图的宽高,防止页面像素改变时定位的元素位置变化
                    backgroundWidth:'',
                    backgroundHeight:'',
                    backgroundUrl:'',
                    formList:[],
                    fileSize:''
                },
                generateContentList:[{}],
                rules:{
                    content: [
                        { required: true, message: '请输入文字内容', trigger: 'blur' },
                    ],
                    // 必填,且必须为正整数
                    contentSize: [
                        { required: true, validator: checkNumber, trigger: 'blur' }
                    ],
                    // 必填,且必须为正整数
                    distanceX: [
                        { required: true, validator: checkNumber, trigger: 'blur' }
                    ],
                    // 必填,且必须为正整数
                    distanceY: [
                        { required: true, validator: checkNumber, trigger: 'blur' }
                    ],
                    // 宽度可填可不填,如果填必须为正整数
                    contentWidth:[
                        { required: true, validator: checkNumber, trigger: 'blur' }
                    ],
                    // 图片高度可填可不填,如果填必须为正整数
                    imgHeight:[
                        { required: true, validator: checkNumber, trigger: 'blur' }
                    ],
                    // 必填
                    fileList:[
                        { required: true, message: '请上传图片', trigger: 'change' },
                    ],
                },
            }
        },
        directives: {
            // 拖拽指令
            drag: (el,binding,vnode)=> {
                let dragBox = el; //获取当前元素

                // 获取移动区域
                let moveScope = vnode.context.$refs.moveScope;

                dragBox.onmousedown = e => {
                    // vnode.context就是this
                    // 获取到图片,并取它的宽高
                    let imgBox = vnode.context.$refs.imgBox;

                    // 定义当前元素的id
                    let dragBoxId = null;
                    // 定义当前元素距离图片左上角的坐标
                    let dragBoxX = null;
                    let dragBoxY = null;


                    //算出鼠标相对元素的位置
                    let disX = e.clientX - dragBox.offsetLeft;
                    let disY = e.clientY - dragBox.offsetTop;
                    moveScope.onmousemove = e => {
                        // 在拖拽时隐藏辅助线,否则可能影响用户拖拽操作
                        vnode.context.$refs.sublineX.style.display = 'none';
                        vnode.context.$refs.sublineY.style.display = 'none';

                        //用鼠标的位置减去鼠标相对元素的位置,得到元素的位置
                        let left = e.clientX - disX;
                        let top = e.clientY - disY;
                        // 移动当前元素
                        if (left > imgBox.offsetWidth - el.offsetWidth) {
                            dragBox.style.left = imgBox.offsetWidth - el.offsetWidth + 'px';
                        }else if(left < 0){
                            dragBox.style.left = 0 + 'px';
                        }else{
                            dragBox.style.left = left + 'px';
                        }

                        if (top > imgBox.offsetHeight - el.offsetHeight) {
                            dragBox.style.top = imgBox.offsetHeight - el.offsetHeight + 'px';
                        }else if(top < 0){
                            dragBox.style.top = 0 + 'px';
                        }else{
                            dragBox.style.top = top + 'px';
                        }

                        // 获取到当前元素dragBox距离左上角的值,赋值给当前元素的xy坐标
                        dragBoxId = e.target.id;
                        dragBoxX = dragBox.style.left;
                        dragBoxY = dragBox.style.top;
                    };
                    moveScope.onmouseup = e => {
                        handleMoveUp(e)
                    };
                    moveScope.onmouseout = e => {
                        handleMoveUp(e)
                    };
                    // 鼠标移出目标区域和鼠标弹起事件处理函数
                    function handleMoveUp(e) {
                        // 通过dragBoxId找到对应的form表单
                        vnode.context.imgAndContent.formList.forEach(item=>{
                            if (item.key===dragBoxId) {
                                // 在赋值时,要将px删掉
                                item.form.distanceX = Number(dragBoxX.substring(0,dragBoxX.length - 2));
                                item.form.distanceY = Number(dragBoxY.substring(0,dragBoxY.length - 2));
                            }
                        });
                        // 在拖拽操作结束时,显示辅助线
                        vnode.context.$refs.sublineX.style.display = 'block';
                        vnode.context.$refs.sublineY.style.display = 'block';
                        //鼠标弹起来的时候不再移动
                        moveScope.onmousemove = null;
                        //预防鼠标弹起来后还会循环(即预防鼠标放上去的时候还会移动)
                        moveScope.onmouseup = null;
                        // 预防移除后仍触发
                        moveScope.onmouseout = null;
                    }
                };
            }
        },
        mounted () {

        },
        created() {
            this.getContentList();
        },
        methods: {
            getFile(file,index,key) {
                // 必须要清空列表和设置的filesize,否则上传的动画效果,和图片展示效果会存在问题
                this.imgAndContent.formList[index].form.fileList = [];
                this.imgAndContent.formList[index].form.fileSize = '';
                // 移除图片的校验结果
                this.$refs['form'+key][0].clearValidate('fileList');
                this.getBase64(file.raw).then(res => {
                    const params = res
                    this.imgAndContent.formList[index].form.fileList.push({name:file.name,url:params})
                    // 需要再图片设置之后设置background-size,否则无效
                    this.imgAndContent.formList[index].form.fileSize = '100% 100%';
                })
            },
            // 图片转base64
            getBase64(file) {
                return new Promise(function(resolve, reject) {
                    let reader = new FileReader();
                    let imgResult = '';
                    reader.readAsDataURL(file);
                    reader.onload = function() {
                        imgResult = reader.result;
                    };
                    reader.onerror = function(error) {
                        reject(error);
                    };
                    reader.onloadend = function() {
                        resolve(imgResult);
                    };
                });
            },
            // 移除图片
            handleUploadRemove(index) {
                this.imgAndContent.formList[index].form.fileList = [];
                // 需要再图片设置之后设置background-size,否则无效
                this.imgAndContent.formList[index].form.fileSize = '';
            },


            // div键入事件
            divIninput(index,key){
                let innerText = this.$refs['divIninput'+key][0].innerText;
                this.imgAndContent.formList[index].form.content = innerText;
                this.beyondScope(this.imgAndContent.formList[index].form,index,key)
            },
            // 获取后台数据
            getContentList(){
                // 数据模拟
                const timer = setTimeout(()=>{
                    this.imgAndContent = {
                        backgroundWidth:'1000px',
                        backgroundHeight:'600px',
                        backgroundUrl:'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fup.enterdesk.com%2Fedpic_source%2F53%2F0a%2Fda%2F530adad966630fce548cd408237ff200.jpg&refer=http%3A%2F%2Fup.enterdesk.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=jpeg?sec=1641370458&t=06759ab43457fab4f297cdb05e6bd329',
                        formList:[
                            {
                                key:'12132',
                                form:{
                                    contentType:'0',
                                    content:'测试回显',
                                    contentSize:20,
                                    distanceX:'420',
                                    distanceY:'220',
                                    mousePT:false,
                                    contentWidth:80,
                                    imgHeight:20,
                                    // contentImgUrl:'http://sssss',
                                    contentColor:'#012596',
                                    fileList:[],
                                    fileSize:'100% 100%'
                                }
                            },
                            {
                                key:'dasds',
                                form:{
                                    contentType:'1',
                                    content:'',
                                    contentSize:'',
                                    distanceX:'920',
                                    distanceY:'220',
                                    mousePT:false,
                                    contentWidth:80,
                                    imgHeight:80,
                                    // contentImgUrl:'https://gimg2.baidu.com/image_search/src=http%3A%2F%2Fb-ssl.duitang.com%2Fuploads%2Fitem%2F201811%2F25%2F20181125001855_tsfsl.png&refer=http%3A%2F%2Fb-ssl.duitang.com&app=2002&size=f9999,10000&q=a80&n=0&g=0n&fmt=jpeg?sec=1641023635&t=e0c4a3b0b887912c019ef2cb9689eb1a',
                                    contentColor:'#000000',
                                    fileList:[
                                        {
                                            name:'远眺',
                                            url:''
                                        }
                                    ],
                                    fileSize:'100% 100%'
                                }
                            },
                        ]
                    }
                    this.imgAndContent.formList.forEach((item,index)=>{
                        this.beyondScope(item.form,index,item.key);
                        // 回显左侧文本数据
                        if (item.form.contentType==='0') {
                            this.$nextTick(()=>{
                                this.$refs['divIninput'+item.key][0].innerHTML = item.form.content;
                            })
                        }
                    })
                    clearTimeout(timer);
                },2000)
            },
            // 在类型切换时
            // 1、文本切换为图片:contentSize、content 清空
            // 2、图片切换为文本:imgHeight、contentImgUrl 清空
            handleChangeType(form,index,key){
                if (form.contentType==='1') {
                    this.imgAndContent.formList[index].form.contentSize = '';
                    this.imgAndContent.formList[index].form.content = '';
                    this.imgAndContent.formList[index].form.contentColor = '';
                }else{
                    this.imgAndContent.formList[index].form.imgHeight = '';
                    this.imgAndContent.formList[index].form.fileList = [];
                }
                // 将form表单校验提示清空
                this.$refs[key][0].clearValidate();
            },
            // 生成一个随机key
            generateID(length){
                return Number(Math.random().toString().substr(3,length) + Date.now()).toString(36);
            },
            // 当鼠标移入到图片区域时,此时辅助线可移动
            setMoveFlag(){
                this.moveFlag = true;
            },
            // 只能存在一个鼠标选点的form表单
            changeMousePTForm(form){
                // 当某一个的值变为true时,其他的都要变为false
                this.imgAndContent.formList.filter(item=>{
                    return item.form!==form
                }).forEach(item=>{
                    item.form.mousePT = false;
                })
            },
            // 添加一个内容
            addContentConfig(){
                this.$set(this.imgAndContent.formList,this.imgAndContent.formList.length,{
                    key:this.generateID(3),
                    form:{
                        contentType:'0',
                        content:'',
                        contentSize:12,
                        contentColor:'#000000',
                        distanceX:null,
                        distanceY:null,
                        mousePT:false,
                        contentWidth:20,
                        imgHeight:20,
                        fileBase64:'',
                        fileList:[]
                    }
                })
            },
            // 打印
            openPrint(){
                this.$print(this.$refs.moveScope);
            },
            // 提交编辑好的数据
            submitContentConfig(){
                let resList = []
                this.imgAndContent.formList.forEach(form=>{
                    resList.push(this.checkForm('form'+form.key));
                })
                Promise.all(resList)
                    .then(() => {
                        console.log('校验成功');
                        console.log(JSON.stringify(this.imgAndContent.formList));
                    })
                    .catch(() => {
                        console.log('校验失败');
                        console.log(this.imgAndContent.formList);
                    });

            },
            // 表单校验
            checkForm(formName) {
                return new Promise((resolve, reject) => {
                    this.$refs[formName][0].validate(valid => {
                        if (valid) {
                            resolve();
                        } else reject();
                    });
                });
            },
            // 删除一个内容
            delContentConfig(index){
                // formList的index对应的就是generateContentList的index
                // 1、删除generateContentList的index项
                this.generateContentList.splice(index,1);
                // 2、删除formList的对应index项
                this.imgAndContent.formList.splice(index,1);
            },
            // 鼠标选点
            getMouseTP(e){
                this.imgAndContent.formList.forEach((item,index)=>{
                    if (item.form.mousePT) {
                        item.form.distanceX = e.layerX || e.offsetX;
                        item.form.distanceY = e.layerY || e.offsetY;
                        // 选点之后要重新挪动元素位置,判断是否在范围内
                        this.beyondScope(item.form,index,item.key)
                        // 在点击之后,应该保持辅助线位置
                        // 当点击之后继续点击应该恢复辅助线的位置移动
                        // 并将辅助线移动到目前鼠标位置
                        if (!this.moveFlag) {
                            this.moveFlag = true;
                            this.getMouseTPSubline({layerX:null,layerY:null,offsetX:null,offsetY:null},item.form.distanceX,item.form.distanceY);
                        }else{
                            this.moveFlag = false;
                        }
                    }
                })
            },
            setDivContent(form,index,key){
                this.$refs['divIninput'+key][0].innerText = form.content;
                this.beyondScope(form,index,key)
            },
            // 防止鼠标选点元素超出范围
            // 需要在生成元素的时候调用
            // 需要参数:当前的form
            // 需要拿到 当前生成元素的宽高、底图的宽高
            beyondScope(form,index,key){
                console.log(form,index,key);
                this.$nextTick(()=>{
                    let imgBox = this.$refs.imgBox;
                    let element = document.getElementById(key);
                    // 如果元素距离顶部的距离top值,加上元素的高度 大于 图片的高度
                    // 则元素距离顶部的距离left值 应该等于 图片高度 减去 元素的高度
                    // X轴同理
                    if (Number(form.distanceY) + element.offsetHeight > imgBox.offsetHeight) {
                        this.imgAndContent.formList[index].form.distanceY = imgBox.offsetHeight - element.offsetHeight;
                    }
                    if (Number(form.distanceX) + element.offsetWidth > imgBox.offsetWidth) {
                        this.imgAndContent.formList[index].form.distanceX = imgBox.offsetWidth - element.offsetWidth;
                    }
                })
            },

            // 鼠标选点辅助线
            getMouseTPSubline(e,distanceX,distanceY){
                if (!this.moveFlag) {
                    return;
                }
                let x = e.layerX || e.offsetX || distanceX;
                let y = e.layerY || e.offsetY || distanceY;
                this.$refs.sublineX.style.top = y + 'px';
                this.$refs.sublineY.style.left = x + 'px';
                if (!!distanceX || !!distanceY) {
                    this.moveFlag = false;
                }
            }
        }
    }
</script>

<style scoped lang="less">
    @import "~common/css/reset.css";

    #img-add-content{
        position: relative;
        width: 100%;
        height: 100vh;
        .add-content-img{
            position: relative;
            left: 0;
            top: 0;
            width: 50%;
            height: 100%;
            background: rgb(130, 195, 197);
            overflow-y: auto;
            // img{
            //     max-width: 100%;
            // }
            .content-item{
                cursor: move;
                .preview{
                    // border: 1px solid #EE473A;
                    box-shadow: 0 0 1px 1px #EE473A;
                }
                .textarea{
                    /deep/ .el-textarea__inner{
                        padding: 0px;
                        background: transparent;
                        border: none;
                        margin: 0;
                        resize:none;
                        overflow:hidden;
                    }
                }
            }
            #sublineX,#sublineY{
                background: #ddd;
                position: absolute;
                z-index: 10;
                pointer-events: none;
            }
            #sublineX{
                left: 0;
                width: 100%;
                height: 1px;
            }
            #sublineY{
                top: 0;
                width: 1px;
                height: 100%;
            }
        }
        .add-content-text{
            position: absolute;
            right: 0;
            top: 0;
            width: 50%;
            height: 100%;
            overflow-y: auto;
            background: #ccc;
            .text-suggest{
                color: #EE473A;
                font-size: inherit;
            }
            .add-content-text-btn{
                position: sticky;
                top: 0;
                z-index: 10;
                background: #ccc;
            }
            .content-element-item{
                border-bottom: 3px solid #eee;
            }
        }
    }
</style>
<style lang="less">
</style>

print.js

// 使用ref获取dom节点
// 不需要打印的节点处理
// this.$print(this.$refs.print,{'no-print':'.do-not-print'})
const Print = function(dom, options) {
    if (!(this instanceof Print)) return new Print(dom, options);

    this.options = this.extend({
        'noPrint': '.no-print'
    }, options);

    if ((typeof dom) === 'string') {
        this.dom = document.querySelector(dom);
    } else {
        this.isDOM(dom)
        this.dom = this.isDOM(dom) ? dom : dom.$el;
    }

    this.init();
};
Print.prototype = {
    init: function() {
        let content = this.getStyle() + this.getHtml();
        this.writeIframe(content);
    },
    extend: function(obj, obj2) {
        for (let k in obj2) {
            obj[k] = obj2[k];
        }
        return obj;
    },

    getStyle: function() {
        let str = '';
        let styles = document.querySelectorAll('style,link');
        for (let i = 0; i < styles.length; i++) {
            str += styles[i].outerHTML;
        }
        str += '<style>' + (this.options.noPrint ? this.options.noPrint : '.no-print') + '{display:none;}</style>';

        return str;
    },

    getHtml: function() {
        let inputs = document.querySelectorAll('input');
        let textareas = document.querySelectorAll('textarea');
        let selects = document.querySelectorAll('select');

        for (let k = 0; k < inputs.length; k++) {
            if (inputs[k].type == 'checkbox' || inputs[k].type == 'radio') {
                if (inputs[k].checked == true) {
                    inputs[k].setAttribute('checked', 'checked')
                } else {
                    inputs[k].removeAttribute('checked')
                }
            } else if (inputs[k].type == 'text') {
                inputs[k].setAttribute('value', inputs[k].value)
            } else {
                inputs[k].setAttribute('value', inputs[k].value)
            }
        }

        for (let k2 = 0; k2 < textareas.length; k2++) {
            if (textareas[k2].type == 'textarea') {
                textareas[k2].innerHTML = textareas[k2].value
            }
        }

        for (let k3 = 0; k3 < selects.length; k3++) {
            if (selects[k3].type == 'select-one') {
                let child = selects[k3].children;
                for (let i in child) {
                    if (child[i].tagName == 'OPTION') {
                        if (child[i].selected == true) {
                            child[i].setAttribute('selected', 'selected')
                        } else {
                            child[i].removeAttribute('selected')
                        }
                    }
                }
            }
        }

        return this.dom.outerHTML;
    },

    writeIframe: function(content) {
        let w, doc, iframe = document.createElement('iframe'),
            f = document.body.appendChild(iframe);
        iframe.id = 'myIframe';
        iframe.setAttribute('style', 'position:absolute;width:0;height:0;top:-10px;left:-10px;');
        w = f.contentWindow || f.contentDocument;
        doc = f.contentDocument || f.contentWindow.document;
        doc.open();
        doc.write(content);
        doc.close();
        let _this = this
        iframe.onload = function() {
            _this.toPrint(w);
            setTimeout(function() {
                document.body.removeChild(iframe)
            }, 100)
        }
    },

    toPrint: function(frameWindow) {
        try {
            setTimeout(function() {
                frameWindow.focus();
                try {
                    if (!frameWindow.document.execCommand('print', false, null)) {
                        frameWindow.print();
                    }
                } catch (e) {
                    frameWindow.print();
                }
                frameWindow.close();
            }, 10);
        } catch (err) {
            console.log('err', err);
        }
    },
    isDOM: (typeof HTMLElement === 'object') ?
        function(obj) {
            return obj instanceof HTMLElement;
        } : function(obj) {
            return obj && typeof obj === 'object' && obj.nodeType === 1 && typeof obj.nodeName === 'string';
        }
};
const PrintElement = {}
// 实现一个插件,必须实现它的install方法
PrintElement.install = function(Vue, options) {
    Vue.prototype.$print = Print
}
export default PrintElement;

app.js

import PrintElement from './print'

// ...

Vue.use(PrintElement);

// ...

  • 6
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值