• 前端动态生成二维码后合成海报,下载分享


    转载:https://blog.csdn.net/zgh0711/article/details/88192993

    如何生成二维码

    现在前端环境下,要生成二维码,可用的工具库有很多,我这里用的是 qrcode,首先安装

      npm install --save qrcode

    安装完之后 import 导入就能使用了

      import QRCode from 'qrcode'

    一,将生成的二维码通过 canvas 标签展示

    <canvas class="qr" id="qrCode-canvas"></canvas>
    let canvas = document.getElementById('qrCode-canvas')
            QRCode.toCanvas(canvas, this.qrCodeUrl, (error) => {
              if (error) {
                console.log(error)
              } else {
                canvas.style.width = '1.68rem'
                canvas.style.height = '1.68rem'
                });
              }
     })
    

      二维码生成好后,有需要的话可以调整下 canvas 的大小等属性,因为 qrcode 生成的二维码会带有一些默认样式,需要调整下。

    二,先通过 canvas 生成二维码,然后将 canvas 的内容转为 base64 格式以 img 标签展示

    <img class="qr" id="qr-img" :src="dataUrl" alt="邀请二维码"/>
    
    let img = document.getElementById('qr-img')
            let canvas = document.createElement('canvas')
            canvas.width = img.width
            canvas.height = img.height
            //用 canvas 对象和邀请链接生成二维码,并将生成的二维码转为 base64
            QRCode.toCanvas(canvas, this.qrCodeUrl, (error) => {
              if (error) {
                console.log(error)
              } else {
                this.dataUrl = canvas.toDataURL("image/jpeg")
                });
              }
    })

    上面两种方式都可以生成二维码并显示在界面上,区别在于第一种是以 canvas 标签显示,在微信中长按是无法识别的。第二种是以 img 标签显示,在微信中长按会自动识别并可以直接以图片形式分享出去。

    做到这一步,需求已经完成了一半,下一步就是按设计图要求把二维码定位到海报背景上,将二维码和海报合成为一张图片。之所以要合成为一张图片还是为了让用户在微信中操作方便。合成为一张图片后,用户在微信中直接长按合成后的海报就可以分享或者识别二维码。

    合成海报时碰到的坑

    合成海报时需要用到另一个工具库 html2canvas ,基本使用方法也很简单,先获取要合成图片的 dom 对象,然后调用 html2canvas ,它会返回一个 promise,里面的 canvas 就是合成后的图片信息,同样的,这里的 canvas 可以直接显示出来,也可以转成 base64 之后放到 img 标签里显示,但是为了分享或者下载方便,基本都是转成了 base64

    let poster = document.getElementById('poster')
                html2canvas(poster).then(canvas => {
                  this.posterDataUrl = canvas.toDataURL()
    });
    

    到这里,基本需求就完成了。然而世界上的事情往往都不会是一帆风顺的,在测试的时候发现出了问题。

    我前面二维码是用上面说的第二种方法显示的,将二维码的 canvas 转成了 base64 显示在 img 标签上,再将这个二维码 img 和海报背景 img 合成。这样的实现方式在 iOS 和浏览器中都没有问题,然而在安卓版微信里面却是只有海报背景而没有二维码。

    在查了N多资料和尝试了好多遍之后终于发现,是因为我的二维码是 base64 的,如果二维码是以 canvas 形式显示,再去合成海报,将合成后的海报转为 base64 就没有问题了。 所以我在前面写生成二维码的时候写了二种方法。至此,整个需求也就真正的完成了。

    下面是完整代码

    <template>
      <div class="ypl-flex">
        <mt-header title="邀请用户" go-back></mt-header>
        
        <div class="invitePosterPage flex">
          <p class="title">分享专属海报,邀请用户注册,即可成功邀请</p>
          <p class="remark">(注:只能通过此二维码注册用户才可成功邀请)</p>
          
          <img v-if="posterDataUrl" :src="posterDataUrl" class="poster-bg" alt="邀请海报"/>
          <div v-else id="poster" class="flex-row" style="position: relative">
            <img class="poster-bg" src="../../../assets/img/img_poster_bg.png" alt="邀请海报背景"/>
            <canvas class="qr" id="qrCode-canvas"></canvas>
          </div>
        </div>
      </div>
    </template>
    
    <script>
      /**
       * 这是邀请海报组件
       */
      import QRCode from 'qrcode'
      import html2canvas from 'html2canvas';
      
      export default {
        name: 'Poster',
        data () {
          return {
            posterDataUrl:'',
            qrCodeUrl: 'https://www.baidu.com'
          }
        },
      
        async mounted () {
        	this.createQRCode()
        },
        
        methods: {
          createQRCode () {
            //先用 QRCode 生成二维码 canvas,然后用 html2canvas 合成整张海报并转成 base64 显示出来
            let canvas = document.getElementById('qrCode-canvas')
            QRCode.toCanvas(canvas, this.qrCodeUrl, (error) => {
              if (error) {
                console.log(error)
              } else {
                //qrcode 生成的二维码会带有一些默认样式,需要调整下
                canvas.style.width = '1.68rem'
                canvas.style.height = '1.68rem'
                
                let poster = document.getElementById('poster')
                html2canvas(poster).then(canvas => {
                  this.posterDataUrl = canvas.toDataURL()
                });
              }
            })
          }
        },
      }
    </script>
    

      

    这次做这个需求的过程中,查到了一些 html2canvas 的资料,有很多文章都提到 html2canvas 有一些坑,我是没碰到,现在将这些文章列出来,给需要的人。
    html2canvas 用法详解
    Html2canvas - 项目中遇到的那些坑点汇总(更新中…)
    一次 H5 「保存页面为图片」 的踩坑之旅
    微信H5实现网页长按保存图片及识别二维码
    基于html2canvas实现网页保存为图片及图片清晰度优化
    Vue.js结合Canvas制作二维码和图片的合成(qrcanvas + html2canvas)

    然后就是 canvas 转 base64 的 .toDataURL()] 这个方法,这个方法里面其实是可以配置参数的,有需要的可以去看下这个文档
    HTMLCanvasElement.toDataURL()

  • 相关阅读:
    关于php中,记录日志中,将数组转为json信息记录日志时遇到的问题总结
    快速搭建vsftp 服务器并配置指定目录
    Chrome 里的请求报错 "CAUTION: Provisional headers are shown" 是什么意思?
    使用sed,grep 批量修改文件内容
    Linux如何让进程在后台运行的三种方法详解
    redis 安装方式
    beanstalk 安装
    mysql 错误信息
    PostgreSQL LIMIT 子句
    pg_dump和pg-server版本不一致问题的解决办法
  • 原文地址:https://www.cnblogs.com/deng-jie/p/13426170.html
Copyright © 2020-2023  润新知