react表格行下载文件方法总结

一、前言

下载文件时,后台接口返回的响应体是文件流格式的,前端接收时如果不进行处理,就会无法正确下载文件(有可能会直接打开文件等)。

在此记录下react的表格行使用file-saver下载文件的方法。(注意不同项目可能有差异,供参考

样例截图:
在这里插入图片描述

二、react表格行使用file-saver下载相关代码(以本人项目为例)

1.首先,本人的一个js文件里有个表格标签,AdvancedTable

import AdvancedTable from '@/components/AdvancedTable';
……
          <AdvancedTable
            namespace="myPageModels"
            queryPara={queryPara}
            data={data}
            columns={this.getColumns()}
            rowSelection={rowSelection}
            loading={loading}
          />

其中,namespace是表格名称相关的;
queryPara是表格查询参数相关的,里面有例如 queryPara.page = 0; queryPara.sort = 'id,desc'; queryPara.size = 10;这些,点击翻页的时候框架会用到;
data就是表格数据,格式是[{"id":1,"name":"第一行"},{"id":2,"name":"第二行"}]这样,就是个JsonList数组;
this.getColumns()是表格行数据展示的方法,下载按钮就在这里实现的,后续继续详述;

rowSelection是表格行是否允许选择的配置,可以配置成单选、多选、不允许选择等,也可以设置被选中的监听方法;

loading是表格加载是否完成的标志位,可以让框架控制,也可以自己写逻辑、开始设置为false、当进行某个操作成功后再设置为true,就能实现表格一直转圈、当操作成功后才显示加载完毕。

2.this.getColumns()是表格行数据展示的方法,其中有下载按钮,样例如下:

import {Button, Divider, Icon, Collapse, Popconfirm, Modal, message, Switch, Drawer} from 'antd';


……

  getColumns = () => {
    const { dispatch } = this.props;

    const {
      pageSeriesModels: { isAdmin },
    } = this.props;

    const { nowPageValue } = this.state;

    const columns = [
      {……},
      
      {
        title: '下载',
        dataIndex: 'download',
        align: 'center',
        width: 100,
        render: (text, record) => (
          <Fragment>
            <Fragment>
              <a
                onClick={() =>
                  {
                    let fileSuffix = '';
                    try{
                    fileSuffix = record.documentName.substring(record.documentName.length-3,record.documentName.length).toLowerCase()
                    } catch(e){}
                    console.log("fileSuffix",fileSuffix)
                    if(fileSuffix === 'pdf'){
                      dispatch({
                        type: 'pageSeriesModels/downloadPdf',
                        paramFileId: record.id,
                        paramType: nowPageValue,
                        paramDocumentName: record.documentName,

                        //这个callback实际上写是写了,但是不知道为什么没有被调用
                        callback: (resp) => {
                          console.log(resp);
                          Modal.success({
                            centered: true,
                            content: '正在下载,请稍后',
                            okText: formatMessage({ id: 'global.ok' }),
                          });
                        },
                      })
                    }else{
					  message.error('文件格式错误,无法下载');
					}

                  }
                }
              >
                <Icon type="download" /> <FormattedMessage id="下载" />
              </a>
            </Fragment>
            
            {isAdmin ? <Divider type="vertical" /> : null }

            
            
          </Fragment>
        ),
      },
    ];
    return columns;
  };

这段的意思是,表格里会展示一个下载按钮;
点击下载按钮后,会先判断下文件名后缀是不是pdf,如果不是pdf,就弹出个提示信息"文件格式错误,无法下载";
如果是pdf,就会调用dispatch方法,执行另一个js里的下载方法/downloadPdf,传入的参数是 paramFileId: record.id, paramType: nowPageValue, paramDocumentName: record.documentName,;其中record.id就是表格的数据data里的id,paramType是一个type(个人项目用的,值是1),paramDocumentName同理,是表格data里的文件名。

3.dispatch里的 type: 'pageSeriesModels/downloadPdf',就会调用另一个js里的方法,样例如下:

import * as services from '@/services/api';

export default {
  namespace: 'pageSeriesModels',

  ……

  effects: {
    *downloadPdf({ paramFileId,paramType,paramDocumentName, callback }, { call, put }) {
      //const value = {};
      //value.fileId = paramFileId;
      //value.type = paramType;
      const response = yield call(services.down1, `/api/downloadPdf?fileId=${paramFileId}&type=${paramType}`, null, paramDocumentName);
      if (response) {
        if (callback) callback(response);
      }
    },

}

dispatch的用法,这里就不详述了(可以看本人之前的文章或者百度);
这个方法的意思是,调用services.down1方法,传入入参;
如果执行后有返回值response,并且有回调方法callback(刚才调用dispatch时传入的),就执行回调方法。

4.services.down1方法,又在另一个自己写的js文件里(上方import了),样例如下:

import { stringify } from 'qs';
import { message } from 'antd';
import { saveAs } from 'file-saver';
import fetch from 'dva/fetch';

……

export async function down1(url, values, filename) {
  return fetch(`/commonHeadUrl${url}`, {
    method: 'POST',
    headers: {
      Accept: 'application/json',
      'Content-Type': 'application/x-msdownload; charset=utf-8',
      Authorization: `Bearer ${sessionStorage.getItem('token')}`,
    },
  }).then(response => {
    if (response.status !== 200 && response.status !== 201) {
      message.error('出错了,请将功能路径告知管理员');
    } else {
      response.blob().then(blob => {
        if (blob.size === 0) {
          message.error('下载失败,请将功能路径告知管理员');
        } else {
          saveAs(blob, filename);
        }
      });
    }
  });
}

这段代码的意思是,设置了为post请求、header参数,然后请求/commonHeadUrl/api/downloadPdf?fileId=${paramFileId}&type=${paramType}这个地址;(没有显式写域名,会自动带上当前域名)
如果返回文件流,就使用saveAs方法下载,就实现浏览器下载了。

其中,saveAs就是file-saver里的方法,如果没有,可以使用命令装下:

npm install file-saver

其它组件如果没有的,可以先百度下,如果是框架的组件,那就也用npm i 下载下。

到此就可以用file-saver下载文件了。

三、备注

1.本人的vue项目中,经常使用axios发送get与post等请求

npm install axios

2.本人的react项目中,使用fetch发送get与post等请求

(1)fetch:原生函数,不再使用 XmlHttpRequest 对象提交 ajax 请求。

(2)老版本浏览器可能不支持。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

k8s入门实战-Service

k8s入门实战-Service Service 和 Label Service 通过一组 Pod 路由通信。Service 是一种抽象&#xff0c;它允许 Pod 死亡并在 Kubernetes 中复制&#xff0c;而不会影响应用程序。在依赖的 Pod (如应用程序中的前端和后端组件)之间进行发现和路由是由Kubernetes Service 处理…

day03 MyBatis 核心

mapper接口和原理 之前的持久层组成部分:UserMapper.xmlIUserDAOUserDAOimpl 使用mapper接口:UserMapper.xmlUserMaper接口 mapper接口的好处; 避免持久层里面传入参数错误:以前里面写错了不会报错,只有等到运行代码才能看到错误,第二个参数的类型是Objiect MAPPer使用注意…

unix环境高级编程 第一章 UNIX基础知识 Go实现代码

ls命令的Go语言实现 package mainimport ("fmt""os" )func main() {if len(os.Args) ! 2 {panic("参数数量不足")}targetPath : os.Args[1]if dirList, err : os.ReadDir(targetPath); err nil {for _, dirInfo : range dirList {fmt.Println(…

淡季不淡,满帮一季度净利创历史新高的背后原因是什么?

进入五月&#xff0c;经济复苏的成果越发体现在很多基础行业的表现中。经济的“大动脉”货运行业&#xff0c;也迎来一份新答卷。 北京时间5月22日美股盘前&#xff0c;数字货运平台满帮集团&#xff08;NYSE:YMM&#xff0c;简称&#xff1a;满帮&#xff09;&#xff0c;发布…

自动化测试用例怎么写?最全自动化测试用例设计编写指南...

目录&#xff1a;导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结&#xff08;尾部小惊喜&#xff09; 前言 Python自动化测试&…

python---变量(3)

求字符串的长度 使用len来求字符串中有几个字符 字符串的拼接 此时是把a2字符串拼接到a1字符串的末尾&#xff0c;得到更大的字符串&#xff0c;对于原来的a1和a2是没有影响的&#xff01; 不能把字符串和数字混合相加&#xff01; 这个时候程序就会报错&#xff0c;不能…

GO语言并发编程入门:Goroutine、Channel、Context、并发安全、GMP调度模型

GO语言并发编程入门&#xff1a;Goroutine、Channel、Context、并发安全、GMP调度模型 1.GO并发介绍 并发&#xff1a;多线程程序在一个核的cpu上运行。 并行&#xff1a;多线程程序在多个核的cpu上运行。 由上可知并发不是并行&#xff0c;并行是直接利用多核实现多线程的运…

【分布式文件存储】MinIO部署及实现文件上传下载

目录 概述 MinIO集群部署 准备docker-compose.yml 测试启动 MinIO用户管理 Buckets管理 创建Buckets MinIO客户端 引入依赖 文件上传下载Demo 调用API碰到的问题 概述 MinIO | 高性能, Kubernetes 原生对象存储 MinIO是全球领先的对象存储先锋&#xff0c;目前在全世…

照相机标定

一.相机标定的原理 1.1 相机如何成像&#xff1a; 相机成像系统中&#xff0c;共包含四个坐标系&#xff1a;世界坐标系、相机坐标系、图像坐标系、像素坐标系。 1.1.1 世界坐标系&#xff1a; 世界坐标系&#xff08;world coordinate&#xff09;&#xff0c;也称为测量坐…

Cobalt Strike工具基本使用

Cobalt Strike 安装启动启动server端启动client目标机器连接 工具基使用用户驱动攻击屏幕截图进程列表键盘记录文件管理远程vnc远程代理端口扫描 生成后门被攻击者运行后门文件后查看结果 钓鱼攻击信息收集网站克隆文件下载 安装 网盘地址&#xff1a;链接&#xff1a;https:/…

PyG的Planetoid无法直接下载Cora等数据集的解决方法

问题描述&#xff1a; 在使用PyG的时候&#xff0c;通常会涉及到一些公共数据集的下载&#xff0c;由于网络问题&#xff0c;导致无法下载出现以下问题&#xff1a; 尝试了很多的方法都没有成功&#xff08;主要是个人比较菜&#xff01;&#xff09;。但是皇天不负有心人&am…

文心一言和ChatGPT最全对比

文心一言和ChatGPT都是基于深度学习技术的自然语言处理模型&#xff0c;有各自的优势和使用场景&#xff0c;无法简单地比较 ChatGPT 和文心一言哪一个功能更强大&#xff0c;它们各自具有优势和局限性&#xff0c;需要根据具体需求进行选择&#xff0c;以下一些具体对比&#…

分布式网络通信框架(十)——Mprpc框架使用示例

发布一个服务提供远程调用方法的流程 若想要发布一个服务提供一些远程调用方法&#xff0c;步骤如下&#xff1a; 先在protobuf文件中添加参数和返回值的message 类型&#xff0c;然后再添加希望提供的服务 service 类型&#xff08;如UserServiceRpc&#xff09;和 其中的方…

C++内存管理和模板

文章目录 内存管理new和delete 函数模板隐式实例化显式实例化 类模板 内存管理 有时候我们需要动态的申请内存,比如队列,栈,二叉树等数据结构,我们一开始并不知道要存储多少个数据,也就是不确定究竟要多大的内存,给小了存不下,给大了会造成内存资源的浪费,因此动态的申请内存是…

原工程运行正常,重新复制一份后再 npm install 后再运行就报错的解决办法

原工程&#xff0c;运行正常 将刚刚的工程复制一份呢&#xff0c;重新 npm install 再 npm run serve 就报错 出现这个问题十之八九都是依赖的问题。有可能是因为这个工程里面之前安装过一些东西&#xff0c;后来莫名其妙的就把 package.json 里面相关的依赖给删掉了。但由于原…

【数据结构】顺序二叉树的实现—以堆的实现为例、堆的调整、堆的创建、堆的插入和删除、堆排序

文章目录 1.堆的概念及结构2.堆的实现&#xff08;以大堆为例&#xff09;2.1堆的插入2.1.1堆的向上调整算法 2.2堆的删除2.2.1堆的向下调整算法 2.3堆的创建2.4有关建堆的时间复杂度 3.堆排序4.C语言堆实现源码 1.堆的概念及结构 堆就是顺序结构二叉树。 如果有一个关键码的集…

jetcache参考文档

jetcache简介 https://github.com/alibaba/jetcache/blob/master/docs/CN/GettingStarted.md 简介 JetCache是一个基于Java的缓存系统封装&#xff0c;提供统一的API和注解来简化缓存的使用。 JetCache提供了比SpringCache更加强大的注解&#xff0c;可以原生的支持TTL、两级…

shell变量类型含义 长方形面积计算 截取相应字段方法

1.总结变量的类型及含义&#xff1f; 2.实现课堂案例计算长方形面积&#xff1f;&#xff08;6种方式&#xff09; 3.定义变量urlhttps://blog.csdn.net/weixin_45029822/article/details/103568815&#xff08;通过多种方法实现&#xff09; 1&#xff09;截取网站访问的协议 …

文心一言 VS 讯飞星火 VS chatgpt (22)-- 算法导论4.2 2题

二、为Strassen算法编写伪代码。 文心一言&#xff1a; Strassen算法是一种用于矩阵乘法的算法&#xff0c;它的核心思想是通过不断将原始矩阵分割成较小的子矩阵来实现高效的乘法运算。以下是Strassen算法的伪代码&#xff1a; function Strassen(A, B): if A[0][0] 0 or…

Springboot整合WebSocket

目录 1.WebSocket 简介 2.WebSocket 实战 1.WebSocket 简介 WebSocket是一种在单个TCP连接上进行全双工通信的协议。WebSocket使得客户端和服务器之间的数据交换变得更加简单&#xff0c;允许服务端主动向客户端推送数据。在WebSocket API中&#xff0c;浏览器和服务器只需要…