手把手教你撸一个小程序带太阳码的海报分享

简介: 我们都知道,微信小程序目前还不支持转发朋友圈,可能现在Android是支持了,iOS还不支持,但总的来说还不能支持普遍机型。这样假如我们需要推荐某个心仪的商品到朋友圈就没法分享出去,于是就可以使用生成海报的形式,让商品详情页的信息显示在一张图片上,保存到手机相册,然后发朋友圈,朋友可以长按识别海报上的小程序码直达该商品详情页面,从而达到如同直接分享商品详情页的效果。

1、前言

我们都知道,微信小程序目前还不支持转发朋友圈,可能现在Android是支持了,iOS还不支持,但总的来说还不能支持普遍机型。这样假如我们需要推荐某个心仪的商品到朋友圈就没法分享出去,于是就可以使用生成海报的形式,让商品详情页的信息显示在一张图片上,保存到手机相册,然后发朋友圈,朋友可以长按识别海报上的小程序码直达该商品详情页面,从而达到如同直接分享商品详情页的效果。

image.png

2、思路

之所以用户扫码能直接进入到我们指定的页面,主要就是要靠海报上的那个太阳码,也就是小程序码,其他海报上的信息或者漂亮的样式纯粹就是为了渲染海报图片而已。

1、当用户点击按钮生成海报时,我们要通过后台接口生成并返回小程序码,要记得传给后台要塞入小程序码的参数,比如这里要进入商品详情页就需要塞入商品id,如果还需要知道是谁分享的,那还要塞入用户id。(这里有个问题留给大家思考,因为生成小程序码的接口scene字段长度最多是32位,如果我们需要传的参数位数超过32该怎么办,特别是数据库主键是以uuid形式存储的时候,那么传2个参数的时候就会超过而无法调用接口)

2、保存海报按钮,要检测用户是否已授权保存到手机相册,如果已授权,则显示保存海报按钮,如果未授权则显示授权并保存海报按钮

3、用户扫小程序码,解析码中scene参数,获取到先前塞入的参数,比如商品id,用户id,再调用商品详情接口渲染数据即可。

3、效果

image.png

4、实现

分享图片的components组件

index.wxml

  <view class="container">
    <view class="canvas-wrapper">
      <image class="icon" src="/images/icons/close.png" bindtap="onClose"></image>
      <image class="shareImage" src="{{tempFilePath}}"></image>
    </view>
    <view class="button-wrapper">
      <ly-button type="warning" i-class="custom-button" shape="circle" lang wx:if="{{saveImageAuth !== '1'}}" bindtap="saveImage" circle>保存海报</ly-button>
      <ly-button type="warning" i-class="custom-button" shape="circle" lang wx:else bindtap="bindOpenSetting" circle>授权并保存海报</ly-button>
    </view>
  </view>
</v-mask>
<view class="hideCanvas">
  <canvas canvas-id="shareCanvas" style="width: 750px;height: 1125px; zoom: {{unit}}"></canvas>
</view>

index.js

Component({
  /**
   * 组件的属性列表
   */
  properties: {
    detail: Object
  },

  /**
   * 组件的初始数据
   */
  data: {
    unit: 1, // 比例
    saveImageAuth: '1', // 权限
    tempFilePath: '', // 临时图片地址
  },
  lifetimes: {
    attached() {
      // 检查授权权限
      this.checkSaveImageAuth();
      // //获取用户设备信息,屏幕宽度
      wx.getSystemInfo({
        success: res => {
          console.log('getSystemInfo', res)
          this.setData({
            unit: res.windowWidth / 750 * 0.8,
            ratio: res.pixelRatio
          })
        }
      })
      this.drawCanvas()
    }
  },
  /**
   * 组件的方法列表
   */
  methods: {
    // 封装的下载图片函数
    downLoadImage(url) {
      return new Promise((resolve, reject) => {
        wx.getImageInfo({
          src: url,
          success(res) {
            resolve(res.path)
          },
          fail(err) {
            reject(err)
          },
          complete() {
            console.log('complete')
          }
        })
      })
    },
    // 封装的下载云存储文件函数
    downLoadCloudFile(id) {
      return new Promise((resolve, reject) => {
        wx.cloud.downloadFile({
          fileID: id,
          success: res => {
            // 返回临时文件路径
            resolve(res.tempFilePath)
          },
          fail: err => {
            console.log('err', err)
          }
        })
      })
    },
    bindOpenSetting(e) {
      let _this = this
      wx.openSetting({
        success(res) {
          if (res.authSetting['scope.writePhotosAlbum']) {
            _this.saveImage()
          } else {
            wx.showToast({
              title: '请授权保存相册权限,才能为您生成分享图',
              icon: 'none'
            })
          }
        }
      })
      console.log('检查权限', e)
    },
    // 检查用户授权权限
    checkSaveImageAuth() {

      let _this = this
      wx.getSetting({
        success(res) {
          console.log('检查用户授权权限', res)
          let auth = ''
          // 有权限
          if (res.authSetting['scope.writePhotosAlbum'] === true) {
            console.log('2')
            auth = '2'
            // 无权限
          } else if (res.authSetting['scope.writePhotosAlbum'] === false) {
            console.log('1')
            auth = '1'
          } else {
            // 未设置
            console.log('0')
            auth = '0'
          }
          _this.setData({
            saveImageAuth: auth
          })
        }
      })
    },
    drawCanvas() {
      let detail = this.properties.detail;
      let mallType = detail.mallType;
      console.log('drawCanvas',detail);
      let ctx = wx.createCanvasContext('shareCanvas', this)
      let titleImage = this.downLoadImage('https://ysd-1300312604.cos.ap-shanghai.myqcloud.com/goods/goods_editor/20201022/c7bda103dcc24ce688743ec824b3dea0.png');
      let productImage = this.downLoadImage(detail.productImage);
      let erCodeImage = this.downLoadImage(detail.erCodeImage);
      Promise.all([titleImage, productImage, erCodeImage]).then(imgs => {
        console.log('imgs', imgs)
        // 全部图片下载成功
        let bgWidth = 750;
        let bgHeight = 1.5 * 750;
        // 绘制白底背景
        ctx.setFillStyle('#fff')
        ctx.fillRect(0, 0, bgWidth, bgHeight)
        console.log('全部图片下载成功')
        // 绘制顶部标题图片
        let titleOffLeft = (750 - 652) / 2;
        let titleOffTop = 60;
        let titleWidth = 652;
        let titleHeight = 50;
        // todo 偏移量需要再处理一下
        ctx.drawImage(imgs[0], titleOffLeft, titleOffTop, titleWidth, titleHeight);
        // 绘制产品图片
        let productOffLeft = 30;
        let productOffTop = 155;
        let productWidth = 690;
        let productHeight = 380;
        ctx.drawImage(imgs[1], productOffLeft, productOffTop, productWidth, productHeight)
        // 绘制矩形边框
        ctx.lineWidth = 1;
        ctx.strokeStyle = '#ccc';
        ctx.rect(30, 156, 690, 670);
        ctx.stroke();
        // 绘制标题
        ctx.setFillStyle('#333') // 文字颜色

        ctx.font = `${32}px PingFang`; // 文字字号

        let title1 = detail.title;
        let title2 = '';
        if (title1.length > 40) {
          title2 = title1.substring(20, 39) + '...'
        } else if (title1.length > 20) {
          title2 = title1.substring(20)
        }
        title1 = title1.substring(0, 20)
        let textWidth = ctx.measureText(title1).width;
        let canvasWidthNoPadding = 750 - 56 * 2;
        ctx.fillText(title1, 56, 595)
        ctx.fillText(title2, 56, 630)
        // 绘制副标题
        ctx.setFillStyle('#ccc');
        ctx.font = `28px PingFang`;
        let subTitle = detail.subTitle;
        if (subTitle.length > 23) {
          subTitle = subTitle.substring(0, 22) + '...'
        }
        ctx.fillText(subTitle, 56, 695)
        // 绘制价格
        let priceIcon = '券后¥ ';
        let price = String(detail.price); // 价格
        let priceTail = '优惠券¥ ';
        if (mallType == 3) {
          priceIcon = '折后 ';
        }
        let couponAmount = String(detail.couponAmount);
        ctx.font = '28px PingFang';

        ctx.setFillStyle('#08B4DE');
        ctx.fillText(priceIcon, 56, 770 );
        let offsetPriceIcon = ctx.measureText(priceIcon).width;
        ctx.font = '36px PingFang';
        ctx.fillText(price, (56 + offsetPriceIcon), 770 );

        if (detail.couponAmount != 0) {
          let offsetPrice = ctx.measureText(price).width;
          console.log('offsetPrice', offsetPrice)
          if (mallType !=3) {
            ctx.font = '28px PingFang';
            ctx.fillText(priceTail, (360 + offsetPriceIcon + offsetPrice), 770 );
          }
          
          ctx.font = '36px PingFang';
          let offsetPriceTailIcon = ctx.measureText(priceTail).width;
          ctx.fillText(mallType !=3 ? couponAmount : couponAmount + ' 折', (330 + offsetPriceIcon + offsetPrice + offsetPriceTailIcon), 770 );
        }
      
        let linePriceIcon = (mallType != 4 ? mallType != 1 ? mallType != 2 ? mallType != 3 ? '拼多多': '唯品会' :'苏宁': '京东': '淘宝') + '价 ';
        let linePrice = String(detail.linePrice); // 价格
        let saleNum = String(detail.saleNum) + '人已购';
        ctx.font = '28px PingFang';
        ctx.setFillStyle('#666');
        ctx.fillText(linePriceIcon, 56, 818 );
        let offsetLinePriceIcon = ctx.measureText(linePriceIcon).width;
        ctx.font = '36px PingFang';
        ctx.fillText(linePrice, (56 + offsetLinePriceIcon), 818 );
        if (mallType != 3 && detail.saleNum != 0) {
          let offsetLinePrice = ctx.measureText(linePrice).width;
          ctx.font = '28px PingFang';
          ctx.fillText(saleNum, (330 + offsetLinePriceIcon + offsetLinePrice), 818 );
        }
      
        // 绘制海报语信息
        let saleTitle = '先领券再购物,即领即用,让你立省到家';
        let saleName = detail.saleName;
        let salePhone = String(detail.salePhone);
        ctx.font = '24px PingFang';
        ctx.setFillStyle('#08B4DE');
        ctx.fillText(saleTitle, 30 , 930 );
        ctx.setFillStyle('#666');
        ctx.fillText(saleName, 30, 972 );
        let saleNameWidth = ctx.measureText(saleName).width;
        ctx.fillText(salePhone, 50 + saleNameWidth, 972 );
        // 绘店铺信息
        ctx.setFillStyle('#08B4DE');
        let supplierName = `店铺名称:${detail.mallName}`;
        let supplierNameWidth = ctx.measureText(supplierName).width;
        ctx.fillText(supplierName, 50, 1036 );
        this.roundRect(ctx, 30, 1006 , (supplierNameWidth + 40), 40, 20)
        // 绘制二维码
        let ercodeDesc = '长按识别小程序码领券';
        ctx.setFillStyle('#bbb');
        ctx.font = '22px PingFang';
        let ercodeDescWidth = ctx.measureText(ercodeDesc).width;
        let offsetErCode = (750 - 30) - ercodeDescWidth;
        ctx.fillText(ercodeDesc, offsetErCode, 1078);
        console.log('imgs', imgs[2])
        ctx.drawImage(imgs[2], 540 , 860, 165, 165)
        ctx.draw(true, () => {
          let _this = this;
          wx.canvasToTempFilePath({
            x: 0,
            y: 0,
            canvasId: 'shareCanvas',
            quality: 1.0,
            fileType: 'jpg',
            success(res) {
              console.log('生成海报成功')
              console.log('res', res)
              _this.setData({
                tempFilePath: res.tempFilePath
              })
            },
            fail(err) {
              console.log('err', err)
              wx.showToast({
                title: '海报生成失败',
                icon: 'none',
              })
              _this.triggerEvent('complete');
            },
            complete() {
              
            }
          }, _this)
        })
      }).catch(err => {
        console.log(err)
      })
    },
    onClose() {
      wx.hideLoading();
      this.triggerEvent('complete');
    },
    // 保存图片
    saveImage() {
      let _this = this;
      let tempFilePath = this.data.tempFilePath;
      if (!tempFilePath) {
        wx.showToast({
          title: '海报生成失败',
          icon: 'none',
        })
        return;
      }
      wx.saveImageToPhotosAlbum({
        filePath: tempFilePath,
        success() {
          wx.showToast({
            title: '已保存到相册,您可将海报分享到朋友圈',
            icon: 'none'
          })
          _this.triggerEvent('complete');
        },
        fail() {
          wx.showToast({
            title: '海报保存失败',
            icon: 'none',
          })
          _this.checkSaveImageAuth()
        },
        complete() {
        }
      })
    },
    // 绘制圆角矩形
    roundRect(ctx, x, y, w, h, r) {
      if (w < 2 * r) {
        r = w / 2;
      }
      if (h < 2 * r) {
        r = h / 2;
      }
      ctx.beginPath();
      ctx.setStrokeStyle('#ff7800');
      ctx.setFillStyle('transparent')
      ctx.setLineWidth(0.5);
      ctx.moveTo(x + r, y);
      ctx.arcTo(x + w, y, x + w, y + h, r);
      ctx.arcTo(x + w, y + h, x, y + h, r);
      ctx.arcTo(x, y + h, x, y, r);
      ctx.arcTo(x, y, x + w, y, r);
      ctx.stroke();
      ctx.closePath();
    },
  },

})

在需要海报的页面引入组件,比如要在detail模块引入,则需要在detail.json文件声明组件

{
  "usingComponents": {
    "v-share-image": "/components/share-image/index"
  }
}

然后在页面detail.wxml引用即可

山水有相逢,来日皆可期,谢谢阅读,我们再会

我手中的金箍棒,上能通天,下能探海

上一篇:基于element-ui封装的upload组件

相关文章
|
8月前
|
小程序 前端开发 API
微信小程序保存海报的过程
微信小程序保存海报的过程
124 0
|
3月前
|
前端开发 小程序 JavaScript
小程序海报,极简的实现方式
小程序海报,极简的实现方式
50 2
小程序海报,极简的实现方式
|
3月前
|
前端开发 小程序 JavaScript
小程序 canvas 生成海报 一次搞掂
小程序 canvas 生成海报 一次搞掂
51 1
|
6月前
|
运维 小程序 前端开发
好的商业模式-----小程序定制资料,加一张好看的海报,在推广中就可以找到用户中了,云服务部署收5000,部署是一种服务,定制化,游戏开发创者,仲裁劳务会剪视频好,提供服务,想增加一些新功能收费,会说
好的商业模式-----小程序定制资料,加一张好看的海报,在推广中就可以找到用户中了,云服务部署收5000,部署是一种服务,定制化,游戏开发创者,仲裁劳务会剪视频好,提供服务,想增加一些新功能收费,会说
|
6月前
|
前端开发 小程序
【微信小程序-原生开发】实用教程20 - 生成海报(实战范例为生成活动海报,内含生成指定页面的小程序二维码,保存图片到手机,canvas 系列教程)
【微信小程序-原生开发】实用教程20 - 生成海报(实战范例为生成活动海报,内含生成指定页面的小程序二维码,保存图片到手机,canvas 系列教程)
432 0
|
小程序 程序员 开发工具
小程序生成二维码海报的组件-wxa-plugin-canvas
小程序生成二维码海报的组件-wxa-plugin-canvas
119 0
|
8月前
|
小程序 API
微信小程序保存海报的过程
微信小程序保存海报的过程
94 0
|
前端开发 小程序 JavaScript
|
小程序 前端开发 JavaScript
微信小程序自动生成打卡海报
微信小程序自动生成打卡海报
190 0
|
前端开发 小程序 API
微信小程序——二维码推广海报
微信小程序——二维码推广海报
174 0