uni-app小程序生成海报wxml-to-canvas

背景:

小程序项目中遇到需要生成海报并分享的功能,为了实现生成海报功能,起初使用h5端的html-to-canvas,结果明显不能使用,毕竟编译之后不通用,生成失败。
然后尝试在网上查找小程序的替代插件,所以在网上找到wxml2canvas,经过一段时间研究,也一直不能实现功能,晦涩难懂的文档,不完善的详细配置文章导致很多尝试都无济于事。同样的比较知名的github海报插件wxa-plugin-canvasPainter也经过一番研究,都不能解决我们遇到的问题。有兴趣的小伙伴可以自己研究。

于此同时呢,也在微信小程序官方文档中找到了对应的生成图片的方案——wxml-to-canvas。经过一番官网文档的研究之后发现,此方案也只是支持wxml的原声小程序canvas图片生成。而我们项目是uni-app的vue文件,还是无法实现此功能。

然而在频繁网络搜索刷文档之后,不经意间找到了一款迄今为止Fork为0,star为0的牛逼插件——uniapp-wxml-to-canvas,跟上面的插件似乎一样好像能够解决问题,但是又充满不确定因素。github上的文档极其简单,也可以说几乎没有,但是不经意间去查看了源码,终于发现了新大陆。

瑟瑟发抖,绝不靠谱

readme文档能不解释就不解释

延用了骗子文章一惯的标题风格。加上言简意赅的解决方案描述。真是好的东西从不废话。


项目目录清晰明了吸引了我

看项目目录,简洁明了,完全契合我们的项目结构,所以我就一不小心点开了pages目录,从此打开新世界的大门。

/pages/share/share.vue

<template>
  <view class="share-page">
    <view class="share-page-box" id="box" v-if="show" :style="{width: canvasWidth + 'px', height: canvasHeight + 'px' }">
      <wxml-to-canvas class="widget" :width="canvasWidth" :height="canvasHeight"></wxml-to-canvas>
    </view>
    <view class="share-page-box msg-box" v-else :style="{width: canvasWidth + 'px', height: canvasHeight + 'px' }">
      {{msg}}
    </view>
    <view class="share-page-btn" @tap="extraImage">
            <button class="btn-big" :style="getBtnStyle">保存图片</button>
    </view>
        
  </view>
</template>
<script>
const { wxml, style } = require('./DomData.js')
export default {
  name: '',
  data () {
    return {
      show: false, // 是否显示canvas
      canvasWidth: 320, // 默认canvas宽高
      canvasHeight: 480,
      screenWidth: null, // 设备宽度
            name: '',
            pic: '',
            chapter1: '',
            chapter2: '',
            widget: null,
      msg: '加载中,请稍等...', // 提示语
    }
  },
  onLoad (options) {
    console.log('options', options);
    this.name = 'Willam Yang'
    this.pic = 'https://pic1.zhimg.com/80/v2-58fe538a59f870407b1435bfd45893ed_720w.jpeg'
    this.chapter1 = '第一段'
    this.chapter2 = '第二段'
    
    // 获取设备信息
    wx.getSystemInfo({
      success: (res) =>{
        this.screenWidth = res.screenWidth
        this.canvasWidth = this.screenWidth * 0.9
        this.canvasHeight = this.screenWidth * 1.1
        console.log('screenWidth', this.screenWidth)
        this.show = true
        // 数字容器宽度 动态设置 
        setTimeout(() => {
          wx.showLoading({title: '海报生成中...'})
          this.widget = this.selectComponent('.widget')
          this.renderToCanvas()
        }, 1000)
      }
    });
  },
  methods: {
    // wxml 转 canvas
    renderToCanvas () {
      console.log('this.widget', this.widget)
      const _wxml = wxml(this.name, this.pic, this.chapter1, this.chapter2)
      const _style = style(this.screenWidth, this.canvasWidth, this.canvasHeight)
      const p1 = this.widget.renderToCanvas({ wxml: _wxml, style: _style })
      p1.then((res) => {
        console.log('海报生成成功');
        wx.hideLoading()
        // this.container = res
      }).catch((err) => {
        console.log('生成失败')
      })
    },
   // 保存到朋友圈
  extraImage() {
    if (!this.show) {
      wx.showToast({title: '海报生成失败,无法分享到朋友圈', icon: 'none'})
      return
    }
    const p2 = this.widget.canvasToTempFilePath()
    let that = this
    p2.then(result => {
      let path = result.tempFilePath
      wx.getSetting({
        success: res => {
          // 非初始化且未授权的情况,需要再次弹窗提示授权
          if (res.authSetting['scope.writePhotosAlbum'] != undefined && res.authSetting['scope.writePhotosAlbum'] != true) {
            wx.showModal({
              title: '是否授权相册权限',
              content: '需要获取相册权限,请确认授权,否则无法使用相关功能',
              success: res => {
                if (res.confirm) {
                  wx.openSetting({
                    success: dataAu => {
                      if (dataAu.authSetting["scope.writePhotosAlbum"] == true) {
                        wx.showToast({
                          title: '授权成功',
                          icon: 'none',
                          duration: 1000
                        });
                        that.saveIMg(path);
                      } else {
                        wx.showToast({
                          title: '授权失败',
                          icon: 'success',
                          duration: 1000
                        });
                      }
                    }
                  });
                }
              }
            });
          } else {
            // 初始化且未授权,系统默认会弹窗提示授权
            // 非初始化且已授权,也会进入这里
            that.saveIMg(path);
          }
        }
      });
    })
  },
   // 保存到相册
  async saveIMg (tempFilePath) {
    wx.saveImageToPhotosAlbum({
      filePath: tempFilePath,
      success: async (res) => {
        wx.showModal({
          content: '图片已保存,分享给好友吧!',
          showCancel: false,
          confirmText: '好的',
          confirmColor: '#333',
          success: function (res) {
            wx.navigateBack({
              //返回
              delta: 1
            });
          },
          fail: function (res) {
            console.log('res', res);  
          }
        });
      },
  
      fail: function (res) {
        wx.showToast({
          title: '您取消了授权',
          icon: 'none',
          duration: 2000
        })
      }
    });
   }
  }
}
</script>

/pages/share/DomData.js

/**
 *
 *
 * @param {*} number  第几位
 * @param {*} src 名片头像
 * @param {*} name 名片名字
 * @param {*} qrCodeUrl 小程序codeURL图片
 */
const wxml = (name, pic, c1, c2) =>`
<view class="container">
    <image src="`+pic+`"  class="pic"/>
    <text class="name">`+ name +`</text>
    <text class="content">`+ c1 +`</text>
    <text class="content">`+ c2 +`</text>
    <view class="bottom">
        <image src="`+pic+`"  class="qr"/>
        <text class="msg">扫码一起加入学习吧</text>
    </view>
</view>
`

/**
 *
 *
 * @param {*} screenWidth 屏幕宽度
 * @param {*} canvasWidth  画布宽度
 * @param {*} canvasHeight  画布高度
 * @param {*} numberWidth  数字宽度,动态设置
 * @return {*} 
 */
const style = (screenWidth, canvasWidth, canvasHeight) => {
  return {
    "container": {
      width: canvasWidth,
      height: canvasHeight,
      position: 'relative',
      overflow: 'hidden',
            backgroundColor: '#ffffff',
    },
    "name":{
      fontSize: 20,
      color: '#333',
      marginLeft: canvasWidth * 0.08,
      width: canvasWidth * 0.84,
      height: screenWidth * 0.18,
            textAlign: 'center',
    },
    "content": {
            fontSize: 14,
            color: '#333',
      width: canvasWidth * 0.84,
      height: screenWidth * 0.15,
            marginLeft: canvasWidth * 0.08,
    },
    "pic": {
      width: canvasWidth * 0.3,
      height: screenWidth * 0.28,
      marginTop: canvasWidth * 0.1,
      marginLeft: canvasWidth * 0.35,
            marginBottom: canvasWidth * 0.05,
            borderRadius: screenWidth * 0.14,
            overflow: 'hidden',
    },
    "bottom":{
      width: canvasWidth,
      height: screenWidth * 0.2,
      flexDirection: 'row',
      justifyContent: 'self-start',
      alignItems: 'center',
            backgroundColor: '#fafafa',
      position: 'absolute',
      bottom: 0,
      left: 0,
    },
        "qr": {
          width: canvasWidth * 0.14,
          height: screenWidth * 0.14,
          marginLeft: canvasWidth * 0.04,
            marginRight: canvasWidth * 0.04,
        },
    "msg": {
      fontSize: 14,
      color: '#a1a1a1',
      width: canvasWidth * 0.74,
            height: 14,
            textAlign: 'left'
    },
  }
}

module.exports = {
  wxml,
  style
}

到这里,已经大概有一套比较完整的方案已经出来了。

方案:依托微信官方wxml-to-canvas的一套生成方案

1.微信官网地址:

https://developers.weixin.qq.com/miniprogram/dev/platform-capabilities/extended/component-plus/wxml-to-canvas.html
根据官网文档开始安装,配置,引入组件,注意样式一定要详细阅读

2.详细项目配置信息

毕竟我们是uni-app项目,所以跟原生小程序开发还是不一样。所以我们只看微信官网是不能正常实现生成canvas的。详细的uni-app配置信息如下:
一、 在uniapp项目中 构建wxcomponents文件夹
二、下载微信小程序官方的wxml-to-canvas代码片段,将其中的这个两个组件复制到我们自己创建的wxcomponents中

红框内两项

移到项目主目录下

三、修改wxml-to-canvas index.js中 的

module.exports = require("widget-ui");
改为
module.exports = require("../widget-ui/index.js");

四、在globalStyle中配置全局组件

"usingComponents":{
    "wxml-to-canvas": "/wxcomponents/wxml-to-canvas/index"
}

这样就配置好了 剩下的使用方法跟在小程序中使用wxml-to-canvas是一致的 参考官网即可。

3.实现代码

同样的,我们只知道详细配置信息,不知道项目中如何使用也是不能实现生成海报功能,查看详细代码更为重要。
https://github.com/WillamYang/uniapp-wxml-to-canvas

4.重要‼️

一、生成海报不能写在组件中(有异议,没实现)。
二、对象属性值为对应 wxml 标签的 class 驼峰形式。
三、需为每个元素指定 width 和 height 属性,否则会导致布局错误。

整体总结这次生成海报功能,道路非常坎坷,东拼西凑的文章合在一起才拼出来了一套完整实现方案。问题总结如下:
1.微信官网给了我们解决方案,但是只针对wxml,对与uni-app开发来说,只看官方文档很难有实质性的帮助。
2.对于网上搜索的大部分文章,理论写的都很好,苦于没有实践代码,项目中配置但凡缺配少配都不能正常使用。即使配置完成,如何书写实现代码,又无从下手。
3.大部分的相关github仓库代码晦涩难懂,要么文档过于简单,要么代码文件过多,没有一套简洁明了且比较完善的实现方案。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,080评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,422评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,630评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,554评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,662评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,856评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,014评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,752评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,212评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,541评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,687评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,347评论 4 331
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,973评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,777评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,006评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,406评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,576评论 2 349

推荐阅读更多精彩内容