实现效果

请添加图片描述

实现代码

1.给img添加属性

<div class="in_code">
   <img
       :src="verificationImg"
        @click="acquireVerification"
   />
</div>

src

src与js中定义的属性双向绑定,图片属性的来源

:src="verificationImg"

@click

实现点击图片换一张验证码

@click="acquireVerification"

2.在进入需要验证码页面查询

因为我这里是通过对话框实现的

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-VUK1D7cd-1646028123555)(assets/验证码图片-20220227085640-z6s71j1.jpg)]

所以,在对话框打开的时候调用了 获取验证码的接口

openDiag(val, val2) {
      this.dialogVisible = val
      this.activeName = val2
      this.acquireVerification() // 就是这个接口
},

3.点击图片,获取验证码

async acquireVerification(){
    const data = await api.verificationCode.verificationCode();
}

接口返回的数据

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-YppW9EYA-1646028123555)(assets/接口返回的数据-20220227085707-jbe1zuq.jpg)]

{
  "code": 200,
  "message": "验证码",
  "data": {
    "image": "",
    "vCode": 1645923302817
  }
}

4.获取时间戳并保存

this.vCode = data.data.vCode.toString();

通过时间戳转换,我获取到了当前时间

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-GWz1mRBy-1646028123556)(assets/image-20220227085759-9hb7sxw.png)]

因为此时用户还没有登录,所以对于后端来说,不知道该用户是谁。所以只能给一个临时的身份认证。即给一个毫秒级的时间戳

此时在后端,每次点击图片获取到验证码的时候,都已经构建了一个key-value键值对,每一个时间戳对应一个验证码,对应一个人

5.获取image

image:是图像

网络中,会通过base64编码,以字符串形式发送图片。所以现在我们需要解码



这一串为Data URL scheme语法

Base64 - MDN Web Docs Glossary: Definitions of Web-related terms | MDN (mozilla.org)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-K18E6abD-1646028123556)(assets/image-20220227090737-55kr0in.png)]

data:文件类型,数据

现在我们将base64格式的图片转换为文件

  • 以逗号分割
let arr = dataUrl.split(',')

即可获得逗号前面的编码类型和编码

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gkC5NAYe-1646028123556)(assets/以逗号分隔-20220227090410-0newl29.jpg)]

  • 通过正则的惰性匹配获取内容

尽可能少的匹配字符

let type = arr[0].match(/:.*?);/)[1]

这里的意思就是从:;尽可能少的匹配字符

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-xiU4yR5a-1646028123557)(assets/正则惰性匹配-20220227091908-4xia2z9.jpg)]

  • 通过/获取后缀
let suffix = type.split('/')[1]

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-7zLUkWXT-1646028123557)(assets/获取后缀-20220227091918-goe6hb4.jpg)]

  • 通过atob转换解码base64文件

Base64的编码与解码 - 术语表 | MDN (mozilla.org)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-gpSYBURI-1646028123557)(assets/image-20220227092052-y3nu0if.png)]

let decodeBase64 = atob(arr[1])

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-r2obaE9b-1646028123557)(assets/atob解码base64-20220227092154-slryz3n.jpg)]

类型为string

  • 获取长度后,循环赋值到8位无符号整数数组内

每一个base64字符都代表6bit。因此,4个base64字符可以转换成3个字节(即占数组3位)

Uint8Array - JavaScript | MDN (mozilla.org)

创建之后,获取解码后的每一位unicode值

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-vaqEd5y7-1646028123558)(assets/image-20220227093643-2jrx0y7.png)]

let n = decodeBase64 .length
let u8Arr = new Unit8Array(n);
while(n --){
    u8Arr = decodeBase64.charCodeAt(n)
}
  • 返回一个file对象

File - Web API 接口参考 | MDN (mozilla.org)

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-6w4BmRId-1646028123558)(assets/image-20220227101117-wwzw7x3.png)]

return new File([u8Arr] , `${fileNmae}.${suffix}`,{
	type: type
})

base64转换成文件-完整代码

base64ImgtoFile(dataurl, filename = 'file') {
      debugger
      let arr = dataurl.split(',')
      let type = arr[0].match(/:(.*?);/)[1]
      let suffix = mime.split('/')[1]

      let decodeBase64 = atob(arr[1])
      let n = bstr.length
      let u8Arr = new Uint8Array(n)
      while (n--) {
        u8Arr[n] = decodeBase64.charCodeAt(n)
      }
      return new File([u8Arr], `${filename}.${suffix}`, {
        type: type,
      })
    },

6.获取imageURL

  • 创建一个blob

blob在MDN中的定义是一个不可变、原始数据的类文件对象。它的数据可以按照文本或二进制格式进行兑取,也可以转换成ReadableStream来用来数据操作

Blob - Web API 接口参考 | MDN (mozilla.org)

现在我们创建一个Blob对象,利用Blob()构造函数

let blob = new Blob([imgFile], { type:  'image/png'})
  • 获取到图片的src地址
let imageUrl = (window.URL || window.wekitURL).createObjectURL(blob)
  • 再赋值即可
this.verificationImg = imageUrl

完整代码

<div class="in_code">
    <img :src="verificationImg" @click="acquireVerification" />
</div>

……

export default {
  data() {
	verificationImg: '',
   }

async acquireVerification() {
      const data = await api.verificationCode.verificationCode()
      this.vCode = data.data.vCode.toString()
      let base64Img = data.data.image
      var imgFile = this.base64ImgtoFile(base64Img)
      // this.verificationImg = imgFile
      let blob = new Blob([imgFile], { type: 'image/png' })
      let imageUrl = (window.URL || window.webkitURL).createObjectURL(blob)
      this.verificationImg = imageUrl
    },

base64ImgtoFile(dataurl, filename = 'file') {
      debugger
      let arr = dataurl.split(',')
      let type = arr[0].match(/:(.*?);/)[1]
      let suffix = mime.split('/')[1]

      let decodeBase64 = atob(arr[1])
      let n = bstr.length
      let u8Arr = new Uint8Array(n)
      while (n--) {
        u8Arr[n] = decodeBase64.charCodeAt(n)
      }
      return new File([u8Arr], `${filename}.${suffix}`, {
        type: type,
      })
    },

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐