项目背景

实现一款交互屏桌面应用软件,类似医院那张种给用户操作办理业务的应用程序。操作业务中在充值后可供用户打印小票。

开发框架:electron-vue

vue版本:v2.6.14

electron版本:v17.2.0

node版本:v16.13.0

实现流程:

electron调用python exe Electron调用西门子_javascript

ps:提一下主结构的版本,是因为在electron中很多情况下出现各种问题是由版本问题导致的。

实现

1.获取系统打印机列表

在主进程窗口,通过webContents对象的getPrintersAsync()方法,获取到系统打印机列表,在主进程中定义监听对象,供渲染进程监听

ipcMain.on('getPrinterList', async (event) => {
  const list = await mainWindow.webContents.getPrintersAsync();
  mainWindow.webContents.send('getPrinterList', list);
});

在渲染进程中通过ipcRender,来触发监听事件,通过once来处理返回的打印机数据。

import { ipcRenderer } from "electron";
    ...
methods:{
    handlePrinter() {
      ipcRenderer.send("getPrinterList");
      ipcRenderer.once("getPrinterList", (event, data) => {
        console.log(data);
        data.forEach((item) => {
          if (item.isDefault) {
            this.printerName = item.name;
          }
        });
        console.log("this.printerName: ", this.printerName);
        this.printRender();
      });
    },
}

2.使用webview标签,实现打印

定义webview,在src中绑定打印的页面index.html,我们定义这个页面到static这个静态文件目录下面,因为这个页面的文件不会被打包导致打印失败。需要值得注意的是,得给webview添加2个属性:

nodeintegration

 webpreferences="contextIsolation=no"

<webview
  id="printWebview"
  ref="printWebview"
  :src="fullPath"
  nodeintegration
  webpreferences="contextIsolation=no"
  style="width: 80mm; visibility: hidden"
/>

在index.html页面中我们定义一个监听事件,来监听是否触发打印,并将需求打印的内容填充到index.html页面中。完成后,通过sendToHost方法,发送一个消息webview-print-do来通知打印。

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Document</title>
    <style scoped>
      body,
      html {
        padding: 0;
        margin: 0;
        font-size: 12px;
      }

      @page {
        margin: 0px;
      }
      ul,li{
        list-style: none;
      }
      ul{
        padding-bottom: 40px;
      }
      .title{
        text-align: center;
      }
      li{
        margin-top: 10px;
      }
      li>span:first-of-type{
        display: inline-block;
        text-align: left;
      }

      .div1 {
        overflow: hidden;
      }
    </style>
  </head>

  <body id="bd"></body>

  <script type="module">
    const { ipcRenderer } = require('electron')
    window.onload = () => {
      ipcRenderer.on('webview-print-render', (event, info) => {
      console.log(event,info);
      // 执行渲染
      document.getElementById('bd').innerHTML = info.html
      ipcRenderer.sendToHost('webview-print-do')
    })
    }
    
  </script>
</html>
printRender() {
      this.messageBox = this.$message({
        type:'warning',
        message: "打印中,请稍后",
        duration: 0,
      });
      // 获取<webview>节点
      const webview = this.$refs.printWebview;
      console.log(webview);
      webview.send("webview-print-render", {
        printName: this.printerName,
        html:
          "<ul>" +
          '<li class="title">欢迎光临浩森体育场馆</li>' +
          "<li><span>会员名:</span><span>" +
          this.taxInfo.customerName +
          "</span></li>" +
          "<li><span>会员手机号:</span><span>" +
          this.taxInfo.customerPhone +
          "</span></li>" +
          "<li><span>消费类型:</span><span>"+ this.tradeTypeName +"</span></li>" +
          "<li><span>订单创建时间:</span><span>" +
          this.taxInfo.createTime +
          "</span></li>" +
          "<li><span>单号:</span><span>" +
          this.taxInfo.orderNo +
          "</span></li>" +
          "<li><span>消费金额:</span><span>" +
          this.taxInfo.totalFee +
          "</span>元</li>" +
          "<li><span>实际支付:</span><span>" +
          this.taxInfo.realFee +
          "</span>元</li>" +
          "<li><span>支付方式:</span><span>" +
          this.taxInfo.channelName +
          "</span></li>" +
          "<li><span>支付时间:</span><span>" +
          this.taxInfo.payTime +
          "</span></li>" +
          "</ul>",
      });
    },

在打印页面,通过给webview添加监听,来判断是否可以打印了。如果监听到webview-print-do这个消息,然后就可以开始执行打印了

import { ipcRenderer } from "electron";
    ...
methods:{
    handlePrinter() {
      ipcRenderer.send("getPrinterList");
      ipcRenderer.once("getPrinterList", (event, data) => {
        data.forEach((item) => {
          if (item.isDefault) {
            this.printerName = item.name;
          }
        });
        this.printRender();
      });
    },
}
  ...

mounted() {
    const webview = this.$refs.printWebview;
    webview.addEventListener("ipc-message", (event) => {
      if (event.channel === "webview-print-do") {
        webview
          .print({
            silent: true,//静默打印
            printBackground: true,
            deviceName: this.printerName, //打印机名称
          })
          .then((res) => {})
          .catch((err) => {})
          .finally(() => {
            this.messageBox.close();
          });
      }
    });
  },

3.注意事项 

有几个问题需要注意

第一点:不要纠结index.html的内容,可以把这个文件看做是承载你打印内容的纸张,不用去看这个文件内是否报错

第二个点:webview中使用src路径,在本地测试的时候 可以通过../../这样的方式去找到index.html,但是在打包过后,这种访问方式是不行的,需要你通过path去获取到它的目录。具体的方法是这样的:

...
import path from "path";

export default {
    
    data(){
        return {
            fullPath: path.join(__static, "print.html"),
        }
    }
    ...
    
}

第三个点:容易被忽视的,在前面也提到了,我使用的版本中,需要给webview标签加上2个属性,才能成功实现打印功能

nodeintegration

 webpreferences="contextIsolation=no"

简单解释一下,第一个属性nodeintegration,是为了让webview访问的页面具有Node集成, 并且可以使用像 require 和 process 这样的node APIs 去访问低层系统资源。

第二个属性webpreferences="contextIsolation=no"

也是同样的目的,为了在index.html中使用require。

大概就这么多,踩了好几个坑,比较重要的应该都记下来了。如果看到底的你还有什么问题,很欢迎讨论咨询,共同解决问题。