前端实现给图片添加水印

一、利用Canvas

::: tip

实现步骤:

  • 上传图片,转换为base64格式的数据,利用Image加载图片
  • 利用canvas写入图片,然后绘制水印
  • 最后通过canvas输出添加水印后的base64数据

:::

1. 本地读取图像文件渲染到img标签

::: tip

​ 通过intpt[type=file]读取文件,将会得到已被Blob对象,借助FileReader.readAsDataURL读取Blob内容并得到Base64编码

​ 读取操作是个异步操作,读取完成后会触发load事件,为了方便之后的使用,使用一个Promise封装这个操作

:::

// blob转图像
const blobToImg = (blob) => {
    return new Promise((resolve,reject) => {
        const reader = new FileReader()
        reader.addEventListener('load',() => {
            const img = new Image()
            img.src = reader.result
            img.addEventListener('load',() => resolve(img))
        })
        reader.readAsDataURL(blob)
    })
}

2. 将img的内容绘制到canvas中

::: tip

​ drawImage这个方法可以传入多个参数,以定义绘制的图像的范围,这里传入的0, 0定义从图像左上角开始绘制,后面可以继续传入两个参数来定义图像的绘制终点

:::

const 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画布上添加水印并转换为Blob对象

const canvasWatermark = (canvas,text) => {
    return new Promise((resolve,reject) => {
        const ctx = canvas.getContext('2d')
        // 设置填充字号和字体,样式
          ctx.font = "24px 宋体"
          ctx.fillStyle = "#FFC82C"
          // 设置右对齐
          ctx.textAlign = 'right'
          // 在指定位置绘制文字,这里指定距离右下角20坐标的地方
          ctx.fillText(text, canvas.width - 20, canvas.height - 20)
          canvas.toBlob(blob => resolve(blob))
    })
}

4. 使用方法

  <!-- 只接受图片类型文件 图片上传-->
  <input type="file" id="upload" accept="image/*">
  <!-- 添加水印后展示区域 -->
  <div id="container"></div>
	const inputEl = document.getElementById('upload')
    const container = document.getElementById('container')
    inputEl.addEventListener('change', async (e) => {
      const waterImg = await imgWaterMark(e.target.files[0], 'Ocean')
      container.appendChild(waterImg)
    })
    // 读取本地文件渲染到img
    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)
      })
    }
    // 将img内容绘制到canvas画布
    const imgToCanvas = (img) => {
      let canvas = document.createElement('canvas')
      canvas.width = img.width
      canvas.height = img.height
      let context = canvas.getContext('2d')
      context.drawImage(img, 0, 0)
      return canvas
    }
    // canvas画布上绘制水印并转换为blob对象
    const waterMark = (canvas, text) => {
      return new Promise((resolve, reject) => {
        let ctx = canvas.getContext('2d')
        // 设置填充的字号 字体 样式
        ctx.font = "30px Josefin Slab"
        ctx.fillStyle = "#fff"
        // 右对齐
        ctx.textAlign = 'right'
        // 指定位置绘制文字
        ctx.fillText(text, canvas.width - 20, canvas.height - 20)
        canvas.toBlob(blob => resolve(blob))
      })
    }
    // 绘制水印并返回添加水印后的图片
    const imgWaterMark = async (file, text) => {
      let img = await blobToImg(file)
      let canvas = imgToCanvas(img)
      let blob = await waterMark(canvas, text)
      // 此处将Blob读取到img标签,并在dom内渲染出来;如果是上传文件,可以将blob添加到FormData中
      let newImage = await blobToImg(blob)
      return newImage
    }

二、利用第三方库

::: tip

​ 例如可以使用watermark.js

​ 地址和使用方法详情见:watermark.js

:::

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要在Flask应用程序中返回图像流以供前端展示,需要使用Python的Pillow库和Flask的send_file功能。以下是如何实现的步骤: 1. 从Pillow库中导入Image模块。 ```python from PIL import Image ``` 2. 打开图像并将其转换为二进制数据。 ```python img = Image.open('image.png') img_byte_arr = io.BytesIO() img.save(img_byte_arr, format='PNG') img_byte_arr = img_byte_arr.getvalue() ``` 3. 使用Flask的send_file函数发送图像数据。 ```python from flask import Flask, send_file app = Flask(__name__) @app.route('/show_image') def show_image(): return send_file(io.BytesIO(img_byte_arr), mimetype='image/png') ``` 4. 在前端中通过img标签展示图像。 ```html <img src="{{ url_for('show_image') }}"> ``` 这样,当访问/show_image URL时,将返回包含图像的流数据,然后在前端通过img标签展示图像。 ### 回答2: Flask 是一款基于 Python 的微型 Web 框架,在实现 Web 应用中广受欢迎。在 Web 应用中,常常需要展示图片等静态资源,本文将介绍如何使用 Flask 实现返回图片流给前端展示。 首先,我们需要在 Flask 中引入相关库。在 Flask 中,我们使用 Pillow 库处理图片,使用 io 库将图片转化为二进制流。代码如下: ``` from flask import Flask, make_response, send_file from io import BytesIO from PIL import Image app = Flask(__name__) ``` 以上代码中,我们引入了 Flask、io 以及 Pillow 库。 接着,我们需要定义一个路由函数,用于响应用户请求,返回图片流。路由函数的具体实现如下: ``` @app.route('/image') def image(): # 打开一张图片 image_path = './test.jpg' with open(image_path, 'rb') as f: image_data = f.read() image = Image.open(BytesIO(image_data)) # 转化为二进制流 img_io = BytesIO() image.save(img_io, 'JPEG') img_io.seek(0) # 返回二进制流 response = make_response(img_io.getvalue()) response.headers['Content-Type'] = 'image/jpeg' return response ``` 第一步,我们使用 open 函数打开一张图片,并读取二进制数据。在示例代码中,我们将图片文件放在了代码根目录下,文件名为 test.jpg。 第二步,我们使用 Pillow 库将读取到的数据转换为 Image 对象,方便后续操作。 第三步,我们将 Image 对象转化为二进制流。在示例代码中,我们使用 BytesIO 对象将二进制数据存储起来,并将存储的数据格式转换为 JPEG 格式。 第四步,在将二进制流返回给前端之前,需要将其封装成一个 Response 对象。在代码中,我们使用 Flask 提供的 make_response 函数创建一个 Response 对象,并设置其 Content-Type 为 image/jpeg。 最后,我们将 Response 对象返回给前端即可实现图片流的展示。可以通过浏览器访问路由 /image,在页面上展示图片。 以上是使用 Flask 实现返回图片流给前端展示的方法,通过参考示例代码及相关库函数的用法,我们可以很快地实现图片、音视频等二进制文件返回给前端的功能。 ### 回答3: 在Flask中,我们可以使用send_file()函数来返回一个图片(或者其他媒体文件)到前端展示。send_file()可以从本地文件系统中或者是通过网络获取的文件流来向客户端发送数据。 在Python中,我们可以使用Pillow或者OpenCV等库来处理图片,并将其转化为二进制数据流,方便在Flask框架中进行传输。 下面是一个简单的例子,它展示了如何在Flask中返回一张图片前端展示: ```python from flask import Flask, send_file from PIL import Image import io app = Flask(__name__) @app.route('/getImage') def getImage(): # 读取图片文件 img = Image.open("test.jpg") # 将图片转化为二进制数据流 img_io = io.BytesIO() img.save(img_io, 'JPEG', quality=70) img_io.seek(0) # 返回二进制数据流给前端 return send_file(img_io, mimetype='image/jpeg') ``` 在这个例子中,我们首先使用Pillow库中的Image来读取一张图片文件。随后,我们将这张图片转化为二进制数据流,通过BytesIO类将其封装成可以在Flask中进行传输的对象。最后,我们利用send_file()函数将这个对象返回给前端展示。 需要注意的是,我们在调用send_file函数的时候设置mimetype为'image/jpeg',以便让浏览器正确解析这个数据流。如果我们返回的是其他类型的媒体文件,那么我们应该相应地修改mimetype参数。 在实际应用中,我们可能需要对图片进行裁剪、缩放、旋转、添加水印等处理,我们可以使用Pillow或者OpenCV等库来完成这些任务。我们只需要将处理后的图片通过类似上述代码的方式发送给前端即可。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值