Vue3+TypeScript+printjs实现标签批量打印功能的完整过程

 更新时间:2024年09月03日 09:50:45   作者:温问问  
最近在做vue项目时使用到了print-js打印,这里给大家分享下,这篇文章主要给大家介绍了关于Vue3+TypeScript+printjs实现标签批量打印功能的完整过程,文中通过代码介绍的非常详细,需要的朋友可以参考下

前言:

临时性需求没怎么接触过前端,代码实现有问题及优化点希望大佬可以留言告知一下

开发工具:VS CODE

界面开发:Vue3+TypeScript+ElementPlus

打印组件:Print-JS

前端打印入口图:

标签页面:

打印界面:

实现功能:前端点击"打印标签"弹出打印界面进行打印作业

实现过程:主界面点击"打印标签"调用el-dialog弹窗(预览和直接打印都居于弹窗实现)

标签模板代码:

<template>
    <div class="LabelPrint-List">
        <el-dialog v-model="state.isShowDialog" draggable :close-on-click-modal="false" width="50%" >
            <template #header>
				<div style="color: #fff">
					<el-icon size="16" style="margin-right: 3px; display: inline; vertical-align: middle"> <ele-Edit/> </el-icon>
					<span>标签打印界面</span>
				</div>
			</template>
            <el-row :gutter="10">
                <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" class="mb10">
                     <div v-for="item in state.Datas">
                        <el-card class="box-card" style="width:100mm; height: 90mm;display: block;" >
                            <div :id='item.id?.toString()'>
                                <!-- print-js -->
                                <div class="labelHeadBody">
                                    <div class="labelHeadBodyLeftHead">
                                        <img class="labelHeadBodyLeftHeadimage"  src="/image/点金log.png" fit="fill" />
                                    </div>
                                    <div class="labelHeadBodyRightHead">
                                        <table class="tableHead">
                                            <tr>
                                                <td class="labelHeadBodyRightHeadTd">某某有限公司</td>
                                            </tr>
                                            <tr>
                                                <td class="labelHeadBodyRightHeadTd">物料标识卡</td>
                                            </tr>
                                        </table>
                                    </div>
                                </div>
                                <div class="labelBody">
                                     <table>
                                        <tbody>
                                            <tr><td class="lableBodytdleft">P/N:</td><td class="lableBodytdright">{{ item.produceNo }}</td></tr>
                                            <tr><td class="lableBodytdleft">数量:</td><td class="lableBodytdright">{{ item.quantity }}</td></tr>
                                            <tr><td class="lableBodytdleft">规格:</td><td class="lableBodytdright lableBodytdrightfont">{{ item.platingSpecs }}</td></tr>
                                            <tr><td class="lableBodytdleft">供应商:</td><td class="lableBodytdright">东莞点金</td></tr>
                                            <tr><td class="lableBodytdleft">生产日期:</td><td class="lableBodytdright">{{moment(String(item.createTime)).format('YYYY/MM/DD')}}</td></tr>
                                            <tr><td class="lableBodytdleft">批次单号:</td><td class="lableBodytdright">{{ item.lot }}</td></tr>
                                            <tr><td class="lableBodytdleft">单重:</td><td class="lableBodytdright">{{ item.singleWeight }}</td></tr>
                                            <tr><td class="lableBodytdleft">总重:</td><td class="lableBodytdright">{{ item.sumWeight }}</td></tr>
                                            <tr><td class="lableBodytdleft">标识人:</td><td class="lableBodytdright"></td></tr>
                                        </tbody>
                                     </table>
                                </div>
                                
                            </div>
                        </el-card>
                     </div>
                </el-col>
            </el-row>
            <template #footer>
				<span class="dialog-footer">
					<el-button @click="cancel">取 消</el-button>
                    <el-button style="background-color:red;color:white" @click="print">打 印</el-button>
				</span>
			</template> 
        </el-dialog>
    </div>
</template>

Typescript代码:

printrow 方法中使用nextTick是当el-dialog弹窗DOM加载完成后在调用PrintJS获取需要打印的区域,这个直接打印过程其实会先弹窗然后DOM加载完成后直接调用浏览器打印界面,后面把弹出关闭,如果不加载el-dialog可以通过动态加载html内容来实现直接打印,我这里图方便就用该方法实现了。

printJS({printable:区域id,type:打印类型(pdf\image\html等),style:打印内容的CSS样式})

注意:style参数值按打印区域的HTMLCSS样式构建,调用printJS设置scanStyles:false不会自动加载HTML的CSS样式需要重新给Style参数赋值所以增加了一个printStyle函数,scanStyles默认值是true(会导致打印界面的内容奇奇怪怪,还没去了解详细原因哈哈哈哈)

<script lang="ts" setup>
import { ref,reactive,nextTick } from 'vue';
import { TbProduceOrderNoInfo } from '/@/api-services';
import printJS from 'print-js';
import moment from 'moment';
const props=defineProps({
    title:String
})

const state=reactive({
    isShowDialog:false,
    Datas:[] as Array<TbProduceOrderNoInfo>,
})

const emits = defineEmits(['handleQuery']);
const closeDialog=()=>{
    emits('handleQuery');
    state.isShowDialog=false;
}

const cancel=()=>{
    state.isShowDialog=false;
    closeDialog();
}

//预览+打印
const openDialog=async(row:any)=>{
   
   state.Datas=JSON.parse(JSON.stringify(row));

   state.isShowDialog=true;
}

const print=()=>{
   for(var i=0;i<state.Datas.length;i++){
      printJS({printable:`${state.Datas[i].id}`,type:"html",style:printStyle(),scanStyles:false})
   }
}

//直接打印不预览
const printrow=async(row:any)=>{
    state.Datas=JSON.parse(JSON.stringify(row));
    state.isShowDialog=true;
    //主界面form DOM加载完成
    nextTick(()=>{
            //弹窗加载完成
            nextTick(()=>{
                printJS({printable:`${state.Datas[0].id}`,type:"html",style:printStyle(),scanStyles:false})
                state.isShowDialog=false;
            })
        })
}

//打印界面的CSS样式
const printStyle=()=>{
    return `
.labelHeadBody{
    display: flex;justify-content:space-between;margin: 0; font-size: 16px;width: 100%; height:45px
}
.labelHeadBodyLeftHead{
    width: 30px;
}
.labelHeadBodyRightHead{
    width: 250px; height: 70px;display: flex;justify-content: center;
}
.lableBodytdrightfont{
    font-size:10px
}
.labelHeadBodyRightHeadTd{
    padding: 0;
    font-size: 14px;
    font-weight: bold;
    text-align: center;
    vertical-align: middle;
}
.labelBody{
    margin-left: 5px;
    margin-right: 5px;
}
.lableBodytdleft{
    width: 30%;
    font-weight: bold;
    vertical-align: bottom;
   
}

.lableBodytdright{
    width: 70%; 
    border-bottom: 1px solid;
}
.labelHeadBodyLeftHeadimage{
    width: 70px; height: 40px
}
.tableHead{
    height: 20px;
}
    `;
}



//预览、直接打印
defineExpose({openDialog,printrow})
</script>

标签前端样式代码:

<style>
.labelHeadBody{
    display: flex;justify-content:space-between;margin: 0; font-size: 16px;width: 100%;
}
.labelHeadBodyLeftHead{
    width: 30px;
}
.labelHeadBodyRightHead{
    width: 250px; height: 70px;display: flex;justify-content: center;
}

.labelHeadBodyRightHeadTd{
    padding: 0;
    font-size: 14px;
    font-weight: bold;
    text-align: center;
    vertical-align: middle;
}
.labelBody{
    margin-top: 10px;
    margin-left: 5px;
    margin-right: 5px;
}
.lableBodytdleft{
    width: 30%;
    font-weight: bold;
    vertical-align: bottom;
   
}
.lableBodytdright{
    width: 75%; 
    border-bottom: 1px solid;
}
.labelHeadBodyLeftHeadimage{
    width: 80px; height: 55px
}
.tableHead{
    height: 20px;
}
</style>

最后,如果需要带二维码的同学可以添加qrcode组件,以下是简单的实现(el-image、img标签中图片不显示的问题还没解决,迂回操作直接把生成的二维码图片设置成控件背景来处理,囧.........):

<template #default="scope">
                         <div :style="createQrcode(scope.row.eqNo)" ></div>
                          <!-- <el-image :scr="createQrcode1(scope.row.eqNo)" style="width: 60px;height: 60px;"></el-image> -->
</template>


import QRCode from 'qrcode'

//将生成的二维码设置成div的Style,不知道为嘛el-image绑定base64image图片不显示
const createQrcode=(text:string)=>{
    if(text==""||text==undefined||text==null) return "";
    let url1:any;
    url1="";  
    QRCode.toDataURL(text,(err,url)=>{
        if(err){
            console.error(err);
        }
        else{
            url1=url;
        }
    })
    return `background-image: url(${url1});background-position: center center;background-size: contain;background-repeat: no-repeat;;width:100%;height:60px`;
}

总结 

到此这篇关于Vue3+TypeScript+printjs实现标签批量打印功能的文章就介绍到这了,更多相关Vue3+TypeScript+printjs标签批量打印内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Vue如何设置全局css文件以及css组合器

    Vue如何设置全局css文件以及css组合器

    这篇文章主要介绍了Vue如何设置全局css文件以及css组合器问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-06-06
  • vue配置接口域名方法总结

    vue配置接口域名方法总结

    在本篇文章里小编给大家分享了关于vue配置接口域名方法和相关知识点总结,需要的朋友们跟着操作下。
    2019-05-05
  • vue tab滚动到一定高度,固定在顶部,点击tab切换不同的内容操作

    vue tab滚动到一定高度,固定在顶部,点击tab切换不同的内容操作

    这篇文章主要介绍了vue tab滚动到一定高度,固定在顶部,点击tab切换不同的内容操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07
  • vue登录路由权限管理的项目实践

    vue登录路由权限管理的项目实践

    在开发Web应用程序时,常常需要进行登录验证和权限管理,本文主要介绍了vue登录路由权限管理的项目实践,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • Vue实现点击当前行变色

    Vue实现点击当前行变色

    这篇文章主要为大家详细介绍了Vue实现点击当前行变色,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-12-12
  • Vue检测屏幕变化来改变不同的charts样式实例

    Vue检测屏幕变化来改变不同的charts样式实例

    这篇文章主要介绍了Vue检测屏幕变化来改变不同的charts样式实例,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-10-10
  • 深入浅出分析vue2和vue3的区别

    深入浅出分析vue2和vue3的区别

    这篇文章主要介绍了vue2和vue3的区别,较为详细的分析了vue2与vue3的常见区别与使用方法,需要的朋友可以参考下
    2023-06-06
  • Vue3中Pinia的基本使用笔记

    Vue3中Pinia的基本使用笔记

    Pinia是一个全新的Vue状态管理库,是Vuex的代替者,尤雨溪强势推荐,下面这篇文章主要给大家介绍了关于Vue3中Pinia的基本使用,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2022-10-10
  • 在Vue当中同时配置多个路由文件的方法案例代码

    在Vue当中同时配置多个路由文件的方法案例代码

    这篇文章主要介绍了在Vue当中同时配置多个路由文件的方法,包含具体代码,本文分步骤结合示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-12-12
  • vue3 api自动导入神器推荐

    vue3 api自动导入神器推荐

    在做vue3项目中时,每次使用都需要先进行引入,下面这篇文章主要给大家介绍了关于vue3 api自动导入的相关资料,文中通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-02-02

最新评论