vue:处理base64格式文件pdf、图片预览

news2024/12/24 21:03:25

一、需求:后端返回是base64数据,需要前端处理来展示文件。

二、实现方法:

解释一下这段代码的功能:

  1. preview(item) 是一个函数,接受一个参数 item,其中包含了文件的相关信息。

  2. )首先,通过条件语句 if (item.type == 'pdf') 检查文件类型是否为 'pdf'。

  3. )如果文件类型是 'pdf',则执行以下操作:

    • 使用 webAPI.server.get() 方法发送一个请求,请求的URL是通过 this.previewUrl.format(item.id) 构建的。
    • 在请求成功后,使用 then 处理响应数据,将返回的 base64 字符串解码为字节数组。
    • 创建一个表示 PDF 文件的 Blob 对象,然后通过 URL.createObjectURL(blob) 创建一个用于在浏览器中预览的 URL。
    • 最后,调用 parent.layerAPI.openResultWin() 打开一个窗口,显示文件名和预览的文件 URL。(这一步对应第3点的代码)
  4. )如果文件类型不是 'pdf',则继续检查是否属于图像格式('png', 'gif', 'jpg', 'bmp', 'svg')。

  5. )如果文件类型是图像格式,执行以下操作:

    • 与 'pdf' 类似,发送一个异步请求获取文件内容,并处理响应数据。
    • 解码 base64 字符串为字节数组,创建一个表示图像的 Blob 对象。
    • 使用 URL.createObjectURL(blob) 创建用于在浏览器中预览的 URL。
    • 使用 this.$nextTick() 来确保在组件更新后执行代码。
    • 最后,如果存在 this.$refs.myImg,将图像的 showViewer 属性设置为 true,用于显示图像查看器。
    •   <el-image

            class="my-img"

            v-if="imageUrl"

            ref="myImg"

            :src="imageUrl"

            :preview-src-list="[imageUrl]"

          >

          </el-image>

  6. )如果文件类型既不是 'pdf' 也不是图像格式,则可能会显示一条警告消息,(// this.$message.warning('暂不支持该格式预览');)。

总体而言,该函数用于处理文件预览的逻辑,支持预览 'pdf' 文件和一些图像格式。

   <iframe
            width="100%"
            height="604px"
            :src="PDFUrl"
             v-if="PDFUrl"
       />



   handleInitPDF() {
      webAPI.server
        .get({
          url: this.getPdfUrl.format(this.id),
        })
        .then((res) => {
          if (res.code == 0) {
            if (res.data) {
              const base64Str = res.data;
              const byteCharacters = atob(base64Str);
              const byteNumbers = new Array(byteCharacters.length);
              for (let i = 0; i < byteCharacters.length; i++) {
                byteNumbers[i] = byteCharacters.charCodeAt(i);
              }
              const byteArray = new Uint8Array(byteNumbers);
              const blob = new Blob([byteArray], { type: 'application/pdf' });
              this.newurl = URL.createObjectURL(blob);
             
            }
            this.PDFUrl = this.newurl;
           
          }
        });
      this.PDFvisiable = true;
    },

 三、如果要直接打开弹窗,与预览图片:

通俗的理解为:计算机是无法直接对base64数据进行处理的,不论是其他类型字符串、数字什么的也好,计算机可以理解的语言其实是二进制数据。因此我们需要将信息转化成计算机可以理解的二进制数据,所以需要先解码,此时我们得到的是原始的二进制 数据,但是由于我们最终需要的是一个url,因此我们需要继续对这个原始二进制数据处理。首先使用Unicode 编码,这一步是为了处理一些特殊的文件格式,比如 PDF,可能某些文件格式使用非常规的编码方式,需要在 JavaScript 中进行适当的转换,以便后续处理。接着类型化数组来表示二进制数据,再将类型化数组(如 Uint8Array)转换为 Blob 对象,因为类型化数组处理过的数据会更适合blob方法处理会更高效,Blob 对象是一种表示二进制大对象的标准化方式,此时我们得到的二进制数据就是标准化的我们所需要的二进制计算机可以理解的数据了,就可以使用方法得到url了。

   preview(item) {
      if (item.type == 'pdf') {
        webAPI.server
          .get({
            url: this.previewUrl.format(item.id),
          })
          .then((res) => {
            const base64Str = res.data;
            const byteCharacters = atob(base64Str);
            const byteNumbers = new Array(byteCharacters.length);
            for (let i = 0; i < byteCharacters.length; i++) {
              byteNumbers[i] = byteCharacters.charCodeAt(i);
            }
            const byteArray = new Uint8Array(byteNumbers);
            const blob = new Blob([byteArray], { type: 'application/pdf' });
            this.fileUrl = URL.createObjectURL(blob);
            parent.layerAPI.openResultWin(    //直接打开弹窗
              '查看文件:' + item.fileName,
              this.fileUrl
            );
          });
        return;
      }
      let imgFormat = ['png', 'gif', 'jpg', 'bmp', 'svg'];
      if (imgFormat.includes(item.type)) {
        webAPI.server
          .get({
            url: this.previewUrl.format(item.id),
          })
          .then((res) => {
            const base64Str = res.data;
            const byteCharacters = atob(base64Str);
            const byteNumbers = new Array(byteCharacters.length);
            for (let i = 0; i < byteCharacters.length; i++) {
              byteNumbers[i] = byteCharacters.charCodeAt(i);
            }
            const byteArray = new Uint8Array(byteNumbers);
            const blob = new Blob([byteArray]);
            this.imageUrl = URL.createObjectURL(blob);
            this.$nextTick(() => {
              if (this.$refs.myImg) {
                this.$refs.myImg.showViewer = true;
              }
            });
          });
        return;
      }
      // this.$message.warning('暂不支持该格式预览');
    },

四、详细解释一下具体的使用到的函数方法:

1)atob()

在JavaScript中,atob() 函数用于解码Base64编码的字符串。在这里,const byteCharacters = atob(base64Str); 这一行代码的目的是将Base64编码的字符串 base64Str 解码为原始的二进制数据,存储在 byteCharacters 变量中。

解释一下具体的步骤:

  1. atob() 函数是Base64解码函数,它接受一个Base64编码的字符串作为参数,并返回解码后的原始二进制数据。

  2. base64Str 是一个包含Base64编码数据的字符串,通常是从服务器端获取的文件内容以Base64编码的形式传输。

  3. 调用 atob(base64Str) 将Base64编码的字符串解码为原始的二进制数据,并将结果存储在 byteCharacters 变量中。

在文件预览的上下文中,这一步是为了将服务器返回的Base64编码的文件内容解码为原始二进制数据,以便后续可以处理和使用这些数据,比如创建Blob对象用于文件预览。

2)new Array()

在这一步 const byteNumbers = new Array(byteCharacters.length); 中,代码创建了一个新的数组 byteNumbers,其长度为 byteCharacters.length。这一步的意义和作用:

  1. byteCharacters 是之前通过 atob() 函数解码得到的原始二进制数据,通常是代表文件内容的字节数据。

  2. byteCharacters.length 返回 byteCharacters 数组的长度,即字节数。

  3. new Array(byteCharacters.length) 创建了一个新的数组,其长度为 byteCharacters.length。这个数组用于存储将字节数据转换为数字的结果

  4. 在 JavaScript 中,new Array(length) 创建一个具有指定长度的新数组。在这里,我们使用 byteCharacters.length 作为数组的长度。

这一步的目的是为了准备一个数组,以便后续将字节数据转换为数字。每个元素将用于存储一个字节的数值。这通常是为了进一步处理二进制数据,例如将它们用于创建 Blob 对象、ArrayBuffer 等。后续代码可能会遍历字节数据,将每个字节的数值存储在数组中的相应位置。

3)charCodeAt()

这个循环的目的是遍历 byteCharacters 数组中的每个字符,获取每个字符的 Unicode 编码,并将这些编码存储到 byteNumbers 数组中。意义和作用:

  1. for 循环通过变量 i 从 0 开始逐步增加,直到 byteCharacters.length - 1。这样可以遍历 byteCharacters 数组中的每个字符。

  2. byteCharacters.charCodeAt(i) 是一个字符串方法,它返回给定位置(索引 i)的字符的 Unicode 编码。Unicode 编码是一个数字,表示字符在 Unicode 字符集中的位置

  3. 对于每个字符,循环将其 Unicode 编码存储在 byteNumbers 数组的相应位置(索引 i)。

这个过程的目的是将原始二进制数据(以字符串形式存在的字符集)转换为对应的数字表示形式。后续代码可能需要使用这些数字,例如用于创建 Uint8Array 或其他二进制数据结构。在这个具体的代码中,这些数字可能被用于创建表示文件内容的二进制数据。

Unicode 编码: 在处理一些特殊的文件格式,比如 PDF,可能会涉及到将二进制数据按照一定规则转换为 Unicode 编码。这是因为某些文件格式可能使用非常规的编码方式,需要在 JavaScript 中进行适当的转换,以便后续处理。

4)Uint8Array()

这行代码的目的是创建一个 Uint8Array 对象,该对象包含了之前收集的字节数据的数值表示。具体的意义和作用:

  1. Uint8Array 是 JavaScript 中的一种类型化数组(TypedArray),它表示一个包含 8 位无符号整数的二进制数据缓冲区。

  2. byteNumbers 是之前通过遍历 byteCharacters 字符串得到的包含每个字符 Unicode 编码的数组。

  3. new Uint8Array(byteNumbers) 创建了一个新的 Uint8Array 对象,该对象的内容是由 byteNumbers 数组提供的数值。

  4. 这样得到的 byteArray 对象实际上是一个包含了文件内容的二进制数据,其中每个元素都是一个 8 位无符号整数,对应于之前字节数据的数值表示。

这一步的目的是将之前的字符编码数组转换为类型化数组,以便后续可以更方便地进行二进制数据的处理和操作。通常,这样的数组可以用于创建诸如 Blob 对象、ArrayBuffer 等二进制数据结构。在这个具体的代码中,byteArray 可能会用于创建表示文件内容的 Blob 对象,用于文件预览等操作。

5)Blob()

这行代码的目的是创建一个 Blob 对象,用于表示二进制大对象,其中包含了之前转换的文件内容的二进制数据。意义和作用:

  1. Blob 是 JavaScript 中的一个对象,用于表示二进制大对象(Binary Large Object)。它通常用于存储二进制数据,比如文件内容。

  2. new Blob([byteArray], { type: 'application/pdf' }) 创建了一个新的 Blob 对象,其中 byteArray 是之前创建的 Uint8Array 对象,它包含了文件内容的二进制数据。

  3. 第一个参数 [byteArray] 是一个数组,表示 Blob 对象的内容。在这里,我们将 byteArray 放入数组中,作为 Blob 对象的二进制数据内容。

  4. 第二个参数 { type: 'application/pdf' } 指定了 Blob 对象的 MIME 类型。在这个具体的例子中,类型被指定为 'application/pdf',表明这是一个 PDF 文件。

  5. 注意:对于图片而言,浏览器通常可以通过文件扩展名等方式自动识别图像类型,因此在创建 Blob 对象时不强制指定类型是合理的。因为浏览器在处理图像时通常会根据文件的内容自动识别图像格式。

  6. 而对于预览 PDF 文件的情况,指定了 { type: 'application/pdf' } 作为 MIME 类型。这是因为 PDF 文件不一定包含文件扩展名等能够让浏览器自动识别的信息,因此通过指定明确的 MIME 类型可以确保正确地将二进制数据解释为 PDF 文件。

    对于其他类型的文件,你可以根据需要指定适当的 MIME 类型。例如,Word 文档通常使用 'application/msword''application/vnd.openxmlformats-officedocument.wordprocessingml.document' 等类型。指定正确的 MIME 类型有助于确保浏览器正确解释和处理文件内容。如果未指定类型,浏览器将尝试根据文件内容自动确定,但这不总是可靠的。

这一步的目的是创建一个表示文件内容的 Blob 对象,方便后续在浏览器中进行预览。通常,这个 Blob 对象可以被用于创建一个 URL,然后用于打开文件预览窗口或其他相关操作。

6)URL.createObjectURL()

这行代码的目的是使用 URL.createObjectURL() 方法创建一个包含 Blob 对象内容的 URL。具体的意义和作用:

  1. URL.createObjectURL(blob) 是一个 Web API 方法,它接受一个 Blob 对象作为参数,并返回一个表示该 Blob 对象内容的 URL

  2. blob 是之前创建的包含文件内容的 Blob 对象。

  3. URL.createObjectURL(blob)blob 对象转换为一个 URL,该 URL 可以用于在浏览器中访问 Blob 对象的内容。

  4. this.fileUrl = ... 将生成的 URL 存储在对象的 fileUrl 属性中。这个属性可能被后续代码用于在浏览器中进行文件预览。

这一步的目的是为了获得一个用于预览文件内容的 URL,通常用于将文件内容嵌入到页面中或者在新窗口中打开。

7)为什么在这个过程中会选择使用类型化数组(Uint8Array

  1. 二进制数据的表示: 在JavaScript中,通常使用字符串表示文本数据,而不是直接操作二进制数据。但在处理文件、图像等二进制数据时,直接使用字符串可能不够高效。

  2. Uint8Array: Uint8Array 是一种类型化数组,它表示一个包含 8 位无符号整数的二进制数据缓冲区。这意味着每个元素都可以存储一个字节的数据。

  3. 二进制数据的处理: 将二进制数据存储在 Uint8Array 中,相比于字符串,提供了更直接、更高效的方式来处理二进制数据。你可以方便地遍历、修改、截取等操作,而无需担心字符编码等细节。

  4. Blob 构造: Blob 构造函数通常接受一个类型化数组作为参数,因为它能更好地适应二进制数据的结构。Uint8Array 就是这样的一种类型化数组,可以直接用于创建 Blob 对象。

  5. 二进制数据的标准化: 使用 Uint8Array 作为中间步骤,有助于确保二进制数据在进行Blob构造等操作时,处于一种标准化和易处理的形式。

短而言之,Uint8Array 提供了一种更直接、更高效地处理二进制数据的方式,适合在文件、图像等场景下进行操作。在这个具体的代码中,它有助于确保二进制数据的结构和格式得到正确处理,以便后续的 Blob 构造和 URL 创建。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1394985.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot的自定义starter和SpringBoot Starter机制,以及综合案例和通用模块-短信发送,基于AOP技术实现日志切面

目录 1.SpringBoot Starter机制 1.1.什么是SpringBoot Starter 1.2.为什么要使用SpringBoot Starter 1.3.应用场景 1.4.自动加载核心注解说明 2.综合案例 2.1.命名规范 2.2.通用模块-短信发送 2.2.1.创建配置类Properties 2.2.2.编写短信业务功能 2.2.3.创建自动配置…

基于Python+django影片数据爬取与数据分析设计与实现

目录 一、 前言介绍&#xff1a; 二 、功能设计&#xff1a; 三、功能实现&#xff1a; 系统登录实现 管理员实现 用户模块实现 四、库表设计&#xff1a; 五、关键代码&#xff1a; 六、论文参考&#xff1a; 七、其他案例&#xff1a; 八、源码获取&#xff1a; 一…

各省快递量数据, shp+excel,2001-2021年,已实现数据可视化

基本信息. 数据名称: 各省快递量数据 数据格式: shpexcel 数据时间&#xff1a;2001-2021年 数据几何类型: 面 数据坐标系: WGS84 数据来源&#xff1a;网络公开数据 数据字段&#xff1a; 序号字段名称字段说明1a_2001快递量/万件_2001年2a_2002快递量/万件_2002年3…

FairyGUI Day 1 导入FairyGUI

FairyGUI Unity3d引擎版本&#xff1a;Uinty3d 20233.2.3f1 1、从资产商店中将FairyGUI购入我的资产中&#xff0c;目前是免费的。 2、从我的资产中将FairyGUI导入到当前项目中。 3、我遇到的问题&#xff0c;我的Assets下有两个文件夹分别是Resources和Scenes&#xff0c;导…

AEB滤镜再破碎,安全焦虑「解不开」?

不久前&#xff0c;理想L7重大交通事故&#xff0c;再次引发了公众对AEB的热议。 根据理想汽车公布的事故视频显示&#xff0c;碰撞发生前3秒&#xff0c;车速在178km/h时驾驶员采取了制动措施&#xff0c;但车速大幅超出AEB&#xff08;自动紧急刹车系统&#xff09;的工作范…

【笔记】认识电机

认识电机 电机一些概念永磁同步电机电机的效率转矩永磁体定子和转子励磁电磁感应定律 AC Optimal Power Flow功能快捷键合理的创建标题&#xff0c;有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居…

esp32-idf eclipse 定时器的使用demo

esp32定时器的使用demo 1、介绍 ESP32芯片包含两个硬件定时器组。每组有两个通用硬件定时器。它们都是基于16位预分频器和64位自动重载功能的向上向下计数器的64位通用定时器。 2、API接口函数 创建定时器函数&#xff1a; esp_timer_create(); esp_err_t esp_timer_create …

P4学习(四)实验一:Basic Forwarding

目录 一.前置知识二.实验过程记录1.找到实验文件2.拓扑图3.明确实验内容4.实验初体验 三. 编写解决方案1.Parse部分1.1 Code1.2 知识点解析 2.Ingress部分2.1 Code2.2 知识点解析 3.Deparse部分3.1 Code3.2 知识点 四.实验完成测试 一.前置知识 Linux基础命令(vim)V!Model的架…

大模型多卡训练原理

背景知识 深度学习涉及大量矩阵运算&#xff0c;而矩阵运算可以并行计算。 一、数据并行 每张卡加载不同的数据&#xff0c;将计算结果合并 存在问题&#xff1a;每个显卡都加载了模型&#xff0c;浪费了一定空间 二、模型并行&#xff1a;适合模型特别大的情况 1、串行计算…

Java Springboot SSE如何判断客户端能否正常接收消息

目录 背景解决方案思路代码代码解释 Java反射知识点补充 背景 当新建一个 emitter 对象的时候, 它的默认超时时间是 30s. SseEmitter emitter new SseEmitter(); 但是很多情况下, 默认30s的时间太短, 需要把 emitter 对象的超时时间设置成不超时, 也就是永久有效. private …

RHEL8 Samba服务器详细配置用户模式

任务&#xff1a; 配置server01为samba服务器&#xff0c;samba服务器的/companydata/sales为共享目录&#xff0c;共享名为sales&#xff0c;里面创建测试文件test_share.tar&#xff0c;创建用户组sales&#xff0c;创建组内用户sale1&#xff0c;要求配置用户模式访问&#…

py爬虫入门笔记(request.get的使用)

文章目录 Day11. 了解浏览器开发者工具2. Get请求http://baidu.com3. Post请求https://fanyi.baidu.com/sug4. 肯德基小作业 Day21. 正则表达式2. 使用re模块3. 爬取豆瓣电影Top250的第一页4. 爬取豆瓣电影Top250所有的250部电影信息 Day31. xpath的使用2. 认识下载照片线程池的…

【翻译】在Qt Designer中创建主窗口(Main Windows)

原文地址&#xff1a;https://doc.qt.io/qt-6/designer-creating-mainwindows.html Qt Designer 可用于为不同用途创建用户界面&#xff0c;并为每个用户界面提供不同类型的模板。主窗口模板用于创建具有菜单栏、工具栏和停靠窗口部件的应用程序窗口。 通过打开文件菜单并选择…

工程档案数字化的意义

工程档案数字化可以提高档案管理效率、节约资源成本、保护档案安全、提高档案可持续性、提升检索与利用的便捷性&#xff0c;促进信息共享与合作&#xff0c;具有重要的意义和价值。 1. 提高档案管理效率&#xff1a;数字化档案可以通过电子方式进行存储、检索和共享&#xff0…

企业网盘的价值:为什么企业需要它?

企业网盘因其主打的文件管理协作功能&#xff0c;正好符合信息时代高速发展下企业的需要&#xff0c;能够帮助企业集中管理文件数据&#xff0c;提供便捷的文件协作服务&#xff0c;一跃成为近两年企业服务类产品榜单中的一匹黑马。 企业网盘真的这么好用吗&#xff1f;企业真…

ant-desgin的table的上移、下移

文章目录 html部分函数部分 html部分 <a-table :columns"columns" :data-source"dataList" :loading"listLoading" :pagination"false"><template #bodyCell"{ column, record, index }"><template v-if&qu…

class_10:this关键字

this关键字是指向调用对象的指针 #include <iostream> #include <iostream> using namespace std;class Car{ public://成员数据string brand; //品牌int year; //年限//构造函数名与类名相同Car(string brand,int year){cout<<"构造函数中&#…

字面量(java)

字面量类型&#xff1a; 整数类型&#xff1a;不带小数的数字&#xff0c;如666、-88 小数类型&#xff1a;带小数点的数字&#xff0c;如13.14、-5.21 字符串类型&#xff1a;用双引号引起来的内容&#xff0c;如"HelloWorld"&#xff0c;""," &q…

HCIA交换技术

VLAN的作用&#xff08;只记录MAC&#xff09;&#xff1a; 路由器和交换机协同工作&#xff0c;为了解决广播域带来的问题&#xff0c;人们引入了VLAN&#xff08;virtual local area network&#xff09;&#xff0c;即虚拟局域网技术&#xff1a;通过在交换机上部署VLAN&…

【软件测试】学习笔记-精准测试

软件测试行业从最开始的手工测试到自动化测试&#xff0c;从黑盒测试到白盒测试&#xff0c;测试理念和技术都发生了日新月异的变化。现如今&#xff0c;几乎所有的软件公司都有一套强大且复杂的自动化测试用例&#xff0c;用来夜以继日地保证产品的正确性和稳定性。 然而&…