vue下载页面生成pdf,及打印

一、下载页面生成pdf

1、安装对应的依赖包

npm install html2canvas;
npm install jspdf --save

2、在utils中创建一个htmlToPdf.js文件,并在main.js中全局引用

import html2Canvas from 'html2canvas'
import JsPDF from 'jspdf'
export default {
  install(Vue, options) {
    Vue.prototype.getPdf = function () {
      var title = this.htmlTitle  //DPF标题
        html2Canvas(document.querySelector('#printTest'), {
          allowTaint: true,
          taintTest: false,
          useCORS: true,
          y:0, // 对Y轴进行裁切
          x:0, // 对x轴进行裁切
          // width:1200, 设置宽度
          // height:5000, 设置高度
          dpi: window.devicePixelRatio * 4, //将分辨率提高到特定的DPI 提高四倍
          scale: 4 //按比例增加分辨率 
        }).then(function (canvas) {
          let contentWidth = canvas.width
          let contentHeight = canvas.height
          let pageHeight = contentWidth / 592.28 * 841.89
          let leftHeight = contentHeight
          let position = 0
          let imgWidth = 595.28
          let imgHeight = 592.28 / contentWidth * contentHeight
          let pageData = canvas.toDataURL('image/jpeg', 1.0)
          let PDF = new JsPDF('', 'pt', 'a4')
          if (leftHeight < pageHeight) {
            PDF.addImage(pageData, 'JPEG', 0, 0, imgWidth, imgHeight)
          } else {
            while (leftHeight > 0) {
              PDF.addImage(pageData, 'JPEG', 0, position, imgWidth, imgHeight)
              leftHeight -= pageHeight
              position -= 841.89
              if (leftHeight > 0) {
              PDF.addPage()
            }
          }
        }
        PDF.save(title + '.pdf')
      })
    }
  }
}
// main.js
import htmlToPdf from '@/utils/htmlToPdf'
Vue.use(htmlToPdf)

3、在vue页面中添加标签,id与js文件中一致,声明变量指定pdf名字

<div id="printTest">生成pdf的内容</div>
data() {
	return {
    htmlTitle: '需要下载的pdf文件名'
  }
}

4、添加事件

<el-button type="primary" @click="getPdf()">下 载</el-button>

二、打印

1、创建一个print.js文件

//print.js
// 打印类属性、方法定义
/* eslint-disable */
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 () {
    var content = this.getStyle() + this.getHtml();
    this.writeIframe(content);
  },
  extend: function (obj, obj2) {
    for (var k in obj2) {
      obj[k] = obj2[k];
    }
    return obj;
  },
  getStyle: function () {
    var str = "",
      styles = document.querySelectorAll('style,link');
    for (var i = 0; i < styles.length; i++) {
      str += styles[i].outerHTML;
    }
    str += "<style>" + (this.options.noPrint ? this.options.noPrint : '.no-print') + "{display:none;}</style>";
    // 去除height:100%样式,解决分页下,样式混乱问题
    str += "<style>html,body,div{height: auto!important;}</style>";

    return str;
  },
  getHtml: function () {
    var inputs = document.querySelectorAll('input');
    var textareas = document.querySelectorAll('textarea');
    var selects = document.querySelectorAll('select');
    var canvass = document.querySelectorAll('canvas');
    for (var 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 (var k2 = 0; k2 < textareas.length; k2++) {
      if (textareas[k2].type == 'textarea') {
        textareas[k2].innerHTML = textareas[k2].value
      }
    }
    for (var k3 = 0; k3 < selects.length; k3++) {
      if (selects[k3].type == 'select-one') {
        var child = selects[k3].children;
        for (var i in child) {
          if (child[i].tagName == 'OPTION') {
            if (child[i].selected == true) {
              child[i].setAttribute('selected', "selected")
            } else {
              child[i].removeAttribute('selected')
            }
          }
        }
      }
    }
    //canvass echars图表转为图片
    for (var k4 = 0; k4 < canvass.length; k4++) {
      var imageURL = canvass[k4].toDataURL("image/png");
      var img = document.createElement("img");
      img.src = imageURL;
      img.setAttribute('style', 'max-width: 100%;');
      img.className = 'isNeedRemove'
      // canvass[k4].style.display = 'none'
      // canvass[k4].parentNode.style.width = '100%'
      // canvass[k4].parentNode.style.textAlign = 'center'
      canvass[k4].parentNode.insertBefore(img,canvass[k4].nextElementSibling);
    }
    // 包裹要打印的元素
    // fix: https://github.com/xyl66/vuePlugs_printjs/issues/36
    // return this.wrapperRefDom(this.dom).outerHTML;
    return this.dom.outerHTML;
  },
  // 向父级元素循环,包裹当前需要打印的元素
  // 防止根级别开头的 css 选择器不生效
  wrapperRefDom: function (refDom) {
    let prevDom = null
    let currDom = refDom
    while (currDom && currDom.tagName.toLowerCase() !== 'body') {
      if (prevDom) {
        let element = currDom.cloneNode(false)
        element.appendChild(prevDom)
        prevDom = element
      } else {
        prevDom = currDom.cloneNode(true)
      }

      currDom = currDom.parentElement
    }

    return currDom.tagName.toLowerCase() === 'body' ? currDom : prevDom
  },
  writeIframe: function (content) {
    var w, doc, iframe = document.createElement('iframe'),
      f = document.body.appendChild(iframe);
    iframe.id = "myIframe";
    //iframe.style = "position:absolute;width:0;height:0;top:-10px;left:-10px;";
    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();
    var _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 MyPlugin = {}
MyPlugin.install = function (Vue, options) {
  // 4. 添加实例方法
  Vue.prototype.$print = Print
}
export default MyPlugin

也可以安装依赖:npm install vue-print-nb --save

2、在main.js中引入,并全局挂载

import Print from '@/assets/js/print.js'

Vue.use(Print);

3、添加标签

<div id="printTest" ref="print">打印的内容</div>
<el-button type="success" @click="getPrint">打 印</el-button>

4、方法

 getPrint () {
      //加ref读取标签,no-print则不参与打印,打印调用如下:
      this.$print(this.$refs.print);
      //因为每次打印都会多一个img(将echarts的canvas转为img),我的print.js里头已经配置给这个img里头加的classname为isNeedRemove,所以每次打印完需去除,我只有一张echarts、去除的数量根据业务需求定。
      this.$nextTick(() => {
        let arr = document.getElementsByClassName('isNeedRemove');
        if (arr.length) arr[0].remove();
      });
    },

5、修改样式(直接打印样式可能会显示不出来)

@page {
  size: auto A4 landscape;
  margin: 20px auto;
}
@media print {
  // 这里写自己需要的样式
  #printTest {}
}

如有更好的建议请在评论区留下宝贵的意见!!!

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Vue动态生成条码标签打印模板可视化可以通过以下步骤实现: 1. 首先,需要使用Vue框架搭建一个项目,可以使用Vue CLI来创建一个新的项目。 2. 在Vue项目中,可以使用第三方组件库来实现可视化的界面设计,例如Element UI或Vuetify。通过这些组件库,可以方便地创建表格、表单和布局等元素。 3. 在Vue组件中,可以使用JavaScript库来生成条码标签。例如,可以使用JsBarcode库来生成条码图像,该库可以根据输入的数据生成各种类型的条码,如Code 39、Code 128等。将生成的条码图像添加到Vue组件中,并根据需要进行样式调整。 4. 在Vue组件中,可以通过表单输入或者从后端获取数据来生成条码标签。用户可以在可视化界面中输入条码相关信息,然后点击生成按钮来生成条码标签。可以使用Vue的数据绑定功能,将用户输入的数据与条码生成的相关参数进行关联,确保生成的条码标签与用户输入的数据一致。 5. 在Vue组件中,可以实现打印功能。可以使用浏览器提供的打印功能,通过JavaScript代码调用浏览器的打印API来实现打印功能。在进行打印前,可以设置打印样式,并将需要打印的内容添加到打印队列中。 6. 最后,在可视化界面中,可以通过添加按钮或菜单来触发打印功能。用户可以选择需要打印的条码标签,然后点击打印按钮来打印条码标签。 通过以上步骤,可以实现Vue动态生成条码标签打印模板可视化。用户可以方便地在可视化界面中输入相关信息,生成条码标签,并通过打印功能将生成的条码标签打印出来。这种可视化的方式可以提高工作效率,减少人工操作的繁琐性。 ### 回答2: Vue 是一种用于构建交互式用户界面的 JavaScript 框架。它提供了一种简洁的方式来创建动态生成条码标签并打印模板的可视化视图。 首先,我们可以使用 Vue 组件来创建一个条码标签模板。该模板可以包含动态生成条码的逻辑,以及需要打印的标签样式。 为了动态生成条码,我们可以使用第三方库,如 jsbarcode。这个库提供了一个简单的 API 来生成各种类型的条码,例如一维码和二维码。我们可以在 Vue 组件中调用该库的方法,传递所需的信息以生成条码。 在可视化方面,Vue 提供了一种响应式的数据绑定机制,以及一个强大的模板语法。我们可以使用这些特性来实现动态更新条码标签的可视化效果。例如,我们可以使用 v-for 指令来循环生成多个条码标签,根据数据源动态渲染每个标签的内容。 为了打印模板,我们可以使用第三方库,如 jsPDF。这个库提供了一个用于生成 PDF 文件的 API,我们可以在 Vue 组件中调用该库的方法,传递待打印内容和样式参数,生成 PDF 文件,然后将其下载打印出来。 总的来说,Vue 可以帮助我们以一种可视化的方式动态生成条码标签,并提供了丰富的工具和库来处理打印模板的需求。该框架的简洁性和灵活性使得开发者可以轻松地实现这些功能,并为用户提供优雅而高效的界面体验。 ### 回答3: Vue是一种流行的JavaScript框架,它可以用于构建现代化的web应用程序。在Vue中动态生成条码标签的方法有多种,下面将介绍一种实现方式并实现可视化打印模板。 首先,我们可以使用一个开源的JavaScript库叫做JsBarcode来生成条码。我们可以通过在Vue组件中引入该库,并传入相应的数据来动态生成条码。 在Vue组件中,我们可以使用一个data属性来存储条码数据,并通过Vue的双向绑定来显示和更新该数据。 在模板中,我们可以使用一个<input>元素来输入条码数据,并使用一个<button>元素来触发生成条码的方法。当用户点击按钮时,我们可以获取<input>元素的值,并将其传递给JsBarcode库来生成条码。 在生成条码后,我们可以使用一个<img>元素来显示生成的条码图片。 此外,对于打印模板的可视化,我们可以使用Vue的计算属性和条件渲染来根据不同的所需显示和隐藏相应的组件或元素,例如显示条码和打印按钮。 当用户点击打印按钮时,我们可以使用window.print()方法来打印当前页面上的内容。 通过以上步骤,我们可以实现一个Vue动态生成条码标签的可视化打印模板。在该模板中,用户可以输入条码数据,点击按钮生成条码并将其显示出来,最后可以选择打印该标签。这样就实现了一个功能完成的Vue动态生成条码标签打印模板。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值