热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

小程序canvas绘图并保存到系统相册

开始实现之前先上个效果图tips网络图片需先配置download域名,可通过wx.getImageInfo转为临时路径;个人习惯问题,我习

开始实现之前先上个效果图

tips

  1. 网络图片需先配置download域名,可通过wx.getImageInfo转为临时路径;
  2. 个人习惯问题,我习惯使用async-await语法,所以需要引入regenerator这个库,使用方式可网上查。

一、封装通用微信api返回为Promise对象

/datas/common.js

// 封装获取微信图片信息。
export const getWxImageInfo = (imgPath) => {return new Promise((resolve, reject) => {wx.getImageInfo({src: imgPath,success: res => {resolve(res)},fail: res => {reject(res)}})})
}// 封装获取节点选择器信息
export const getSelectQurey = (queryStr) => {return new Promise(resolve => {var query = wx.createSelectorQuery();query.select(queryStr).boundingClientRect();query.exec(res => {resolve(res)})})
}// 封装把画布导出生成指定大小的图片
export const canvasToTempFilePath = (width, height, canvasId, fileType = 'jpg') => {return new Promise((resolve, reject) => {wx.canvasToTempFilePath({width,height,canvasId,fileType,success: res => {resolve(res)},fail: res => {reject(res)}})})
}// 封装保存图片到系统相册
export const saveImageToPhotosAlbum = (filePath) => {return new Promise((resolve, reject) => {wx.saveImageToPhotosAlbum({filePath,success: res => {resolve(res)},fail: res => {reject(res)}})})
}

二、视图的实现

.wxml


"icon-download" catchtap="getCanvas">点击生成图片
'shade' wx:if="{{isShowCanvas}}">'qr-code'>'qr-canvas' canvas-id="qrCanvas" id="qrCanvas">'qr-btn'>'qr-btn-save' catchtap='saveImageToPhotosAlbumFunc'>保存图片,分享到朋友圈'qr-btn-cancel' catchtap='hideCanvas'>取消

.wxss

/* 查看大图 */.shade {width: 100%;height: 100%;background-color: rgba(240, 235, 235, 0.5);position: fixed;z-index: 100;top: 0;left: 0;
}.qr-code {width: 600rpx;height: 1000rpx;background-color: #fff;position: absolute;top: 50%;left: 50%;transform: translate(-50%, -50%); /* margin: 30rpx auto; */
}.qr-canvas {display: block;background-color: #fff;margin: 0 auto;width: 600rpx;height: 900rpx;
}.qr-btn {width: 600rpx;height: 100rpx;line-height: 100rpx;margin: 0 auto;font-size: 28rpx;color: #fff;display: flex;background-color: #658dc5;
}.qr-btn-save {flex: 0 0 500rpx;text-align: center;border-right: 1rpx solid #fff;
}.qr-btn-cancel {text-align: center;flex: 0 0 100rpx;
}

三、创建canvas并保存到系统相册

tips

  1. 商品图是正方形的,所以这里商品图的宽高都用canvas的宽
  2. 文字不能换行,这里只是简单的处理了一下
  3. 注意: wx.canvasToTempFilePath(Object object, Object this) 这个的使用,文档有一句话需要注意的:“把当前画布指定区域的内容导出生成指定大小的图片。在 draw() 回调里调用该方法才能保证图片导出成功。”

const app = getApp()
const regeneratorRuntime = app.globalData.regeneratorRuntimeconst
const util = require('../../utils/util.js')
import {getSelectQurey,getWxImageInfo,canvasToTempFilePath,saveImageToPhotosAlbum
} from &#39;../../datas/common.js&#39;Page({data: {isShowCanvas: false, // 是否显示canvas wxaCode: &#39;https://xxx..jpg&#39;, // 商品小程序码goodsImageUrl: &#39;https://xxx..jpg&#39;, // 商品图片 canvasTempFilePath: &#39;&#39;, // canvas导出生成图片的临时路径 },// 点击显示要生成的canvas getCanvas(e) {if (!this.data.wxaCode) {util.showToast(&#39;二维码生成失败&#39;);return;}this.setData({isShowCanvas: true}, () &#61;> {this.createCanvas();})},// 隐藏canvas hideCanvas() {this.setData({isShowCanvas: false})},// 创建canvas async createCanvas() {wx.showLoading({title: &#39;图片生成中...&#39;})const _this &#61; this// 创建节点选择器 const res &#61; await getSelectQurey(&#39;#qrCanvas&#39;);// canvas的宽高 const cvWidth &#61; res[0].width;const cvHeight &#61; res[0].height;const cvSubValue &#61; cvHeight - cvWidthconst qrWidth &#61; cvSubValue / 1.5const qrMargin &#61; (cvSubValue - qrWidth) / 2const qrX &#61; cvWidth - qrWidth - qrMargin / 2const qrY &#61; cvWidth &#43; qrMarginconst shopNameY &#61; cvWidth &#43; cvSubValue - qrWidth// 二维码网络图片转临时路径 let qrImagePath &#61; &#39;&#39;;try {const wxaCode &#61; _this.data.wxaCode;const qrImage &#61; await getWxImageInfo(wxaCode);qrImagePath &#61; qrImage.path} catch (e) {wx.hideLoading();this.hideCanvas();util.showToast(&#39;二维码生成失败&#39;);return;}// 商品网络图片转临时路径 let goodsImagePath &#61; &#39;/images/default_goods.png&#39;;const goodsImage &#61; _this.data.goodsImageUrl;if (goodsImage) {const goodsImageRes &#61; await getWxImageInfo(goodsImage);goodsImagePath &#61; goodsImageRes.path;}// 创建canvas var ctx &#61; wx.createCanvasContext(&#39;qrCanvas&#39;, _this);// 设置背景 ctx.setFillStyle(&#39;#fff&#39;);ctx.fillRect(0, 0, cvWidth, cvHeight);// 设置商品图片 商品图宽高是一样的 ctx.drawImage(goodsImagePath, 0, 0, cvWidth, cvWidth);// 设置二维码图片 ctx.drawImage(qrImagePath, qrX, qrY, qrWidth, qrWidth);// 设置店铺名称 const shopName &#61; &#39;我是店铺名称&#39;;ctx.setFillStyle(&#39;black&#39;)ctx.setFontSize(16)ctx.setTextAlign(&#39;left&#39;)ctx.fillText(shopName, 10, shopNameY, cvWidth - qrWidth);// 设置商品名称 文字不能换行&#xff0c;这里只是简单的处理了一下 const goodsName &#61; &#39;一个名字很长很长的商品就问你怕不怕&#39;;let goodsName1 &#61; &#39;&#39;;let goodsName2 &#61; &#39;&#39;;ctx.setFillStyle(&#39;black&#39;)ctx.setFontSize(14)ctx.setTextAlign(&#39;left&#39;)if (goodsName.length <&#61; 10) {ctx.fillText(goodsName, 10, shopNameY &#43; 30, cvWidth - qrWidth);} elseif (goodsName.length > 10 && goodsName.length <&#61; 22) {goodsName1 &#61; goodsName.substring(0, 10);goodsName2 &#61; goodsName.substring(10);ctx.fillText(goodsName1, 10, shopNameY &#43; 30, cvWidth - qrWidth);ctx.fillText(goodsName2, 10, shopNameY &#43; 50, cvWidth - qrWidth);} else {goodsName1 &#61; goodsName.substring(0, 10);goodsName2 &#61; goodsName.substring(10, 22) &#43; &#39;...&#39;;ctx.fillText(goodsName1, 10, shopNameY &#43; 30, cvWidth - qrWidth);ctx.fillText(goodsName2, 10, shopNameY &#43; 50, cvWidth - qrWidth);}// 设置提示 const tipText &#61; &#39;长按识别小程序&#xff0c;马上下单&#xff01;&#39;;ctx.setFillStyle(&#39;gray&#39;)ctx.setFontSize(8)ctx.setTextAlign(&#39;center&#39;)ctx.fillText(tipText, cvWidth / 2, cvHeight - 10);// 完成 ctx.draw(false, () &#61;> {wx.hideLoading();_this.canvasToTempFilePathFunc(cvWidth, cvHeight, &#39;qrCanvas&#39;)});},// 把当前画布指定区域的内容导出生成指定大小的图片 async canvasToTempFilePathFunc(cvWidth, cvHeight, qrCanvas) {try {let res &#61; await canvasToTempFilePath(cvWidth, cvHeight, qrCanvas);this.setData({canvasTempFilePath: res.tempFilePath});} catch (error) {console.log(error);util.showToast(error.errMsg);}},// 保存图片到本地 async saveImageToPhotosAlbumFunc() {try {let res &#61; await saveImageToPhotosAlbum(this.data.canvasTempFilePath);console.log(res);this.hideCanvas();util.showToast(&#39;图片保存成功&#39;);} catch (err) {console.log(err);}}
})

写得比较简单&#xff0c;因为主要是方便自己做记录的&#xff0c;所以也没有考虑到过多的使用场景。


转:https://juejin.im/post/5d0ae2715188251ac11f70c1



推荐阅读
  • 本文详细介绍了Java中org.eclipse.ui.forms.widgets.ExpandableComposite类的addExpansionListener()方法,并提供了多个实际代码示例,帮助开发者更好地理解和使用该方法。这些示例来源于多个知名开源项目,具有很高的参考价值。 ... [详细]
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • Python自动化处理:从Word文档提取内容并生成带水印的PDF
    本文介绍如何利用Python实现从特定网站下载Word文档,去除水印并添加自定义水印,最终将文档转换为PDF格式。该方法适用于批量处理和自动化需求。 ... [详细]
  • 解决JAX-WS动态客户端工厂弃用问题并迁移到XFire
    在处理Java项目中的JAR包冲突时,我们遇到了JaxWsDynamicClientFactory被弃用的问题,并成功将其迁移到org.codehaus.xfire.client。本文详细介绍了这一过程及解决方案。 ... [详细]
  • 本文介绍了如何利用JavaScript或jQuery来判断网页中的文本框是否处于焦点状态,以及如何检测鼠标是否悬停在指定的HTML元素上。 ... [详细]
  • 导航栏样式练习:项目实例解析
    本文详细介绍了如何创建一个具有动态效果的导航栏,包括HTML、CSS和JavaScript代码的实现,并附有详细的说明和效果图。 ... [详细]
  • 使用 Azure Service Principal 和 Microsoft Graph API 获取 AAD 用户列表
    本文介绍了一段通用代码示例,该代码不仅能够操作 Azure Active Directory (AAD),还可以通过 Azure Service Principal 的授权访问和管理 Azure 订阅资源。Azure 的架构可以分为两个层级:AAD 和 Subscription。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • Android LED 数字字体的应用与实现
    本文介绍了一种适用于 Android 应用的 LED 数字字体(digital font),并详细描述了其在 UI 设计中的应用场景及其实现方法。这种字体常用于视频、广告倒计时等场景,能够增强视觉效果。 ... [详细]
  • RecyclerView初步学习(一)
    RecyclerView初步学习(一)ReCyclerView提供了一种插件式的编程模式,除了提供ViewHolder缓存模式,还可以自定义动画,分割符,布局样式,相比于传统的ListVi ... [详细]
  • 掌握远程执行Linux脚本和命令的技巧
    本文将详细介绍如何利用Python的Paramiko库实现远程执行Linux脚本和命令,帮助读者快速掌握这一实用技能。通过具体的示例和详尽的解释,让初学者也能轻松上手。 ... [详细]
  • 优化局域网SSH连接延迟问题的解决方案
    本文介绍了解决局域网内SSH连接到服务器时出现长时间等待问题的方法。通过调整配置和优化网络设置,可以显著缩短SSH连接的时间。 ... [详细]
  • 本文介绍如何在PostgreSQL数据库中正确插入和处理JSON数据类型,确保数据完整性和避免常见错误。 ... [详细]
  • 本文介绍了多个关于JavaScript的书籍资源、实用工具和编程实例,涵盖从入门到进阶的各个阶段,帮助读者全面提升JavaScript编程能力。 ... [详细]
author-avatar
天若无雨666
这个菇凉很宅,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有