JS VUE 用 canvas 给图片加水印

最近写需求,遇到要给图片加水印的需求。 刚开始想的方案是给图片上覆盖一层水印照片,但是这样的话用户直接下载图片水印也会消失。

后来查资料发现用 canvas 就可以给图片加水印,下面是处理过程。

首先我们要确认图片的格式,我们通过 input 上传的图片格式一般是 File (File 对象是特殊类型的 Blob)即 Blob 格式。

这样的话,我们需要先把 Blob 文件转成 img 标签,先通过 FileReader 读取文件,通过 reader.readAsDataURL 获得文件 Base 64 编码 URL 地址,拿到 URL 后,生成 img 标签。

1. Blob 文件转成 img 标签

js 复制代码
	// blob 文件格式转成 img 标签
    const blobToImg = (blob) => {
      return new Promise((resolve, reject) => {
        const reader = new FileReader()
        reader.readAsDataURL(blob)
        reader.onload = () => {
          let img = new Image()
          img.src = reader.result
          img.addEventListener('load', () => resolve(img))
        }
      })
    }

这里要注意,如果使用 addEventListener 需要先注册事件监听。

js 复制代码
const blobToImg = (blob) => {
      return new Promise((resolve, reject) => {
        const reader = new FileReader()
        reader.addEventListener('load', () => {
          let img = new Image()
          img.src = reader.result
          img.addEventListener('load', () => resolve(img))
        })
        reader.readAsDataURL(blob)
      })
    }

2. img 标签转成 canvas

js 复制代码
 // 将img内容绘制到canvas画布
    imgToCanvas(img) {
      const canvas = document.createElement('canvas')
      canvas.width = img.width
      canvas.height = img.height
      const context = canvas.getContext('2d')
      context.drawImage(img, 0, 0)
      return canvas
    }

3. 在 canvas 画布上绘制水印

水印通过 ctx.createPattern(image, repetition) 方法来进行重复绘制,由于 image 参数要是是 img 标签格式或者 canvas 文件格式,所以我们可以让 UI 老师生成一个水印的图片,或者我们用 canvas 自己画一个水印的图片进行复制。

如果我们需要直接生成图片来进行展示,那我们可以用 canvas.toDataURL() 直接从 canvas 生成图片地址。
如果需要生成 Blob 格式传给后端,那我们可以用 canvas.toBlob() 方法从 canvas 生成 Blob 文件格式,传给后端。

js 复制代码
    // canvas画布上绘制水印
    waterMark(canvas) {
      return new Promise((resolve, reject) => {
        const ctx = canvas.getContext('2d')
        // 绘制水印 canvas
        const canvasWater = this.drawWaterCanvas('已失效')
        // 绘制重复的水印
        ctx.fillStyle = ctx.createPattern(canvasWater, 'repeat')
        ctx.fillRect(0, 0, canvas.width, canvas.height)
        // 这里我需要直接展示,所以就直接生成图片地址
        resolve(canvas.toDataURL())
        
      })
    },
    drawWaterCanvas(str) {
      const canvasWater = document.createElement('canvas')
      canvasWater.width = 400
      canvasWater.height = 400
      const ctxWater = canvasWater.getContext('2d')
      ctxWater.textAlign = 'left'
      ctxWater.textBaseline = 'middle'
      ctxWater.font = '32px Microsoft Yahei'
      ctxWater.fillStyle = 'rgba(0, 0, 0, 0.3)'
      ctxWater.rotate((-20 * Math.PI) / 180)
      ctxWater.fillText(str, 10, 80)
      return canvasWater
    }

4. VUE 中实际应用

html 复制代码
<template>
  <div>
    <img :src="imgUrl" alt="" />
  </div>
</template>

<script>
import loppy from './assets/loppy.jpg'

export default {
  data() {
    return {
      imgUrl: ''
    }
  },
  created() {
    const img = new Image()
    img.src = loppy
    img.onload = async () => {
      const canvas = this.imgToCanvas(img)
      const url = await this.waterMark(canvas)
      this.imgUrl = url
    }
  },
  methods: {
    // 将img内容绘制到canvas画布
    imgToCanvas(img) {
      const canvas = document.createElement('canvas')
      canvas.width = img.width
      canvas.height = img.height
      const context = canvas.getContext('2d')
      context.drawImage(img, 0, 0)
      return canvas
    },
    // canvas画布上绘制水印并转换为blob对象
    waterMark(canvas) {
      return new Promise((resolve) => {
        const ctx = canvas.getContext('2d')
        // 绘制水印 canvas
        const canvasWater = this.drawWaterCanvas('图片已失效')
        // 绘制重复的水印
        ctx.fillStyle = ctx.createPattern(canvasWater, 'repeat')
        ctx.fillRect(0, 0, canvas.width, canvas.height)
        resolve(canvas.toDataURL())
      })
    },
    drawWaterCanvas(str) {
      const canvasWater = document.createElement('canvas')
      canvasWater.width = 500
      canvasWater.height = 500
      const ctxWater = canvasWater.getContext('2d')
      ctxWater.textAlign = 'left'
      ctxWater.textBaseline = 'middle'
      ctxWater.font = '32px Microsoft Yahei'
      ctxWater.fillStyle = 'rgba(0, 0, 0, 0.3)'
      ctxWater.rotate((-20 * Math.PI) / 180)
      ctxWater.fillText(str, 10, 200)
      ctxWater.fillText(new Date().toLocaleString(), 10, 300)
      return canvasWater
    }
  }
}
</script>

<style lang="scss" scoped>
img {
  width: 200px;
  height: 200px;
}
</style>
相关推荐
whltaoin3 小时前
【JAVA全栈项目】弧图图-智能图床 SpringBoot+Vue3 :[框架开荒:一文全步骤打通前后端项目全流程]
java·spring boot·vue·开源项目·全栈·cos
吃饺子不吃馅2 天前
Canvas 如何渲染富文本、图片、SVG 及其 Path 路径?
前端·svg·canvas
清灵xmf2 天前
Vue + TSX 中使用 class 报错 解决方法
vue
专注前端30年2 天前
Vue2 中 v-if 与 v-show 深度对比及实战指南
开发语言·前端·vue
专注前端30年2 天前
【Vue2】基础知识汇总与实战指南
开发语言·前端·vue
纳兹咩programmer3 天前
uni-app踩坑记录-Canvas层级过高遮挡问题
canvas
麦麦大数据4 天前
F039 python五种算法美食推荐可视化大数据系统vue+flask前后端分离架构
python·算法·vue·推荐算法·美食·五种算法
java水泥工4 天前
课程答疑系统|基于SpringBoot和Vue的课程答疑系统(源码+数据库+文档)
spring boot·vue·计算机毕业设计·java毕业设计·大学生毕业设计·课程答疑系统
吃饺子不吃馅4 天前
如何设计一个 Canvas 事件系统?
前端·canvas·图形学
阿洛学长5 天前
高质量 AI 提示词之(从 0-1 开发 Vue 项目)
vue·ai编程·1024程序员节