uniApp小程序保存canvas图片

最近写了需求 用 canvas 保存 2 张图片叠在一起 并保存手机相册 先上代码

javascript 复制代码
<wd-button type="primary" class="download-btn" @click="downloadImage">下载图片</wd-button>

<canvas
    canvas-id="mergeCanvas"
    style="position: absolute; left: -9999px; width: 300px; height: 400px"
    width="300"
    height="300"
  ></canvas>
//这里的canvas的dom元素 如果不想展示的话 可以用上面的style

// 下载图片
const downloadImage = async () => {
  uni.getSetting({
    success(res) {
      console.log(res.authSetting, 'res.authSetting')

      if (!res.authSetting['scope.writePhotosAlbum']) {
        uni.authorize({
          scope: 'scope.writePhotosAlbum',
          success() {
            saveImg()
          },
        })
      } else {
        saveImg()
      }
    },
  })
}

// 合并图片方法
const mergeImages = async () => {
  try {
    return new Promise((resolve, reject) => {
      const ctx = uni.createCanvasContext('mergeCanvas', proxy) // 使用 proxy
      // 获取底图信息
      uni.getImageInfo({
        src: 'https://xxx.com',
        success: (baseRes) => {
          const width = 300
          const height = 400
          // 绘制底图
          ctx.drawImage(baseRes.path, 0, 0, width, height)
          // 获取叠加图信息
          uni.getImageInfo({
            src: data.deviceInfo.image,
            success: (overlayRes) => {
              // 绘制叠加图
              ctx.drawImage(overlayRes.path, 55, 105, 187, 187)

              // 添加码牌编号
              ctx.setFontSize(16) // 设置字体大小
              ctx.setFillStyle('black') // 设置字体颜色
              ctx.fillText(`LSF${data.deviceInfo.merchantCodeId}`, width * 0.32, height * 0.74) // 在底部添加码牌编号

              // 执行绘制操作
              setTimeout(() => {
                ctx.draw(false, () => {
                  uni.canvasToTempFilePath({
                    canvasId: 'mergeCanvas',
                    success: (res) => {
                      resolve(res.tempFilePath)
                    },
                    fail: (err) => {
                      reject(err)
                    },
                  })
                })
              }, 100)
            },
            fail: (err) => {
              console.error('叠加图加载失败:', err)
              reject(err)
            },
          })
        },
        fail: (err) => {
          console.error('底图加载失败:', err)
          reject(err)
        },
      })
    })
  } catch (error) {
    console.log(error, 'error')
  }
}

const saveImg = async () => {
  try {
    uni.showLoading({ title: '图片保存中...' })

    const mergedImagePath = await mergeImages()

    // 保存图片到相册
    uni.saveImageToPhotosAlbum({
      filePath: mergedImagePath,
      success: () => {
        uni.showToast({ title: '保存成功', icon: 'success' })
      },
      fail: (err) => {
        console.log(err, 'err')

        if (err.errMsg === 'saveImageToPhotosAlbum:fail auth deny') {
          uni.openSetting({
            success(settingdata) {
              if (settingdata.authSetting['scope.writePhotosAlbum']) {
                uni.showToast({
                  title: '您已授权成功,请重新保存',
                  icon: 'success',
                  duration: 2000,
                })
              } else {
                uni.showToast({
                  title: '尚未授权,无法保存海报',
                  icon: 'none',
                  duration: 2000,
                })
              }
            },
          })
        }
      },
      complete: () => {
        uni.hideLoading() // 确保在操作完成后隐藏加载提示
      },
    })
  } catch (error) {
    console.log(error)
    uni.showToast({ title: '图片生成失败', icon: 'none' })
    // uni.hideLoading() // 确保在捕获错误时也隐藏加载提示
  }
}

代码看起来没问题 但上线后就下载不了 体验版可以下载 找了很久 发现 需要去设置一下微信小程序的用户隐私保护指引

这里显示已更新就行 没有处理的 需要先处理一下 才能发起权限

相关推荐
LaughingZhu1 小时前
Product Hunt 每日热榜 | 2026-05-21
前端·人工智能·经验分享·chatgpt·html
怕浪猫1 小时前
Electron 开发实战(一):从零入门核心基础与环境搭建
前端·electron·ai编程
小鹏linux2 小时前
Ubuntu 22.04 部署开源免费具有精美现代web页面的Casdoor账号管理系统
linux·前端·ubuntu·开源·堡垒机
前端若水3 小时前
会话管理:创建、切换、删除对话历史
前端·人工智能·python·react.js
Bigger3 小时前
mini-cc:一个轻量级 AI 编程助手的诞生
前端·ai编程·claude
涵涵(互关)3 小时前
Naive-ui树型选择器只显示根节点
前端·ui·vue
BY组态3 小时前
Ricon组态系统最佳实践:从零开始构建物联网监控平台
前端·物联网·iot·web组态·组态
BY组态4 小时前
Ricon组态系统vs传统组态软件:为什么选择新一代Web组态平台
前端·物联网·iot·web组态·组态
SoaringHeart4 小时前
Flutter进阶:OverlayEntry 插入图层管理器 NOverlayZIndexManager
前端·flutter
放下华子我只抽RuiKe54 小时前
React 从入门到生产(四):自定义 Hook
前端·javascript·人工智能·深度学习·react.js·自然语言处理·前端框架