写在前面

JS要实现下载功能,一般都是这么几个过程:生成下载的URL,动态创建一个A标签,并将其href指向生成的URL,然后触发A标签的单击事件,这样就会弹出下载对话框,从而实现了一个下载的功能。

这里所说的下载,有时候也可以理解为保存。出于安全考虑,JS肯定无法直接调用FileAPI写文件到磁盘,但是却可以通过下载来变相实现保存功能。

几个备用知识点

JS触发单击事件

既然是用A标签模拟,那么肯定要知道JS如何主动触发单击事件。

最简单的触发单击事件肯定是elem.click(),平时在不需要考虑兼容性的场合我都是这么干的,但是毕竟这个方法有兼容性(具体兼容性如何没做过测试),所以还是要掌握一个通用的方法。

以下代码是网上比较容易找到的一段代码,我在前面加了一段MouseEvent的判断:

  1. /**
  2. * 触发单击事件
  3. * @param elem 需要触发事件的DOM对象
  4. */
  5. function fireClickEvent(elem)
  6. {
  7. var event;
  8. if(window.MouseEvent) event = new MouseEvent('click');
  9. else
  10. {
  11. event = document.createEvent('MouseEvents');
  12. event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
  13. }
  14. elem.dispatchEvent(event);
  15. }

HTML5的download属性

这个属性很重要,它可以指定下载文件名,并且可以告诉浏览器目标链接是一个下载链接,不是一个普通链接,我们看下面代码就能看出区别了:

  1. <a href="data:text/txt;charset=utf-8,测试下载纯文本" download="测试.txt" >下载1</a>
  2. <a href="data:text/txt;charset=utf-8,测试下载纯文本">下载2</a>

可以发现,下载1按钮能够实现下载,点击下载2链接时直接在浏览器打开文件内容了。

补充说明:

  • file:///模式下貌似不生效;
  • 链接指向一些第三方链接时也不会生效,具体有待研究;

JS弹出下载对话框

假如给我们的不是一个下载地址而是一个blob对象,我们可以通过URL.createObjectURL来给blob对象生成临时URL,并且可以利用HTML5的download属性来指定下载的文件名,好家伙,有了这2个东西我们就可以实现一个“万能”的弹出下载对话框方法了。

综上所述,我又在fireClickEvent的基础上继续简单封装了一个openDownloadDialog方法,使用如下:

  • openDownloadDialog(url, saveName)
  • openDownloadDialog(blob, saveName)

代码如下:

  1. /**
  2. * 通用的打开下载对话框方法,没有测试过具体兼容性
  3. * @param url 下载地址,也可以是一个blob对象,必选
  4. * @param saveName 保存文件名,可选
  5. */
  6. function openDownloadDialog(url, saveName)
  7. {
  8. if(typeof url == 'object' && url instanceof Blob)
  9. {
  10. url = URL.createObjectURL(url); // 创建blob地址
  11. }
  12. var aLink = document.createElement('a');
  13. aLink.href = url;
  14. aLink.download = saveName || ''; // HTML5新增的属性,指定保存文件名,可以不要后缀,注意,file:///模式下不会生效
  15. var event;
  16. if(window.MouseEvent) event = new MouseEvent('click');
  17. else
  18. {
  19. event = document.createEvent('MouseEvents');
  20. event.initMouseEvent('click', true, false, window, 0, 0, 0, 0, 0, false, false, false, false, 0, null);
  21. }
  22. aLink.dispatchEvent(event);
  23. }

JS实现常见文件类型的下载

JS生成CSV文件并下载

csv是一种逗号分隔的表格文件格式,可以很好的被Excel支持,由于其文件格式简单,所以经常用在简单的表格上面。最重要的是它是一种纯文本格式,可以很轻松地用JS来生成而不借助第三方库。

CSV格式示例

如下:

  1. 姓名,期中成绩,期末成绩
  2. 张三,58,95
  3. 李四,98,74
  4. 王二,47,38
  5. 刘能,15,100
  6. 黄五,87,68

excel打开效果如下:

初次尝试

首先想到的是使用data:text/txt;来实现,先看一下下载纯文本:

  1. <a download="测试.txt" href="data:text/txt;charset=utf-8,测试下载纯文本">下载</a>

以上代码没毛病,然后再换成csv。换csv的最大问题就是如何处理换行,很简单,用encodeURIComponent编码一下就可以了:

  1. <button onclick="test()">下载CSV</button>
  2. <script>
  3. function test()
  4. {
  5. var csv = '姓名,期中成绩,期末成绩\n张三,58,95\n李四,98,74';
  6. var a = document.createElement('a');
  7. a.href = 'data:text/txt;charset=utf-8,'+encodeURIComponent(csv);
  8. a.download = '测试.csv';
  9. a.click(); // 这里偷个懒,直接用click模拟
  10. }
  11. </script>

解决CSV乱码问题

虽然我们用的是UTF-8编码,下载后你会发现,用文本编辑器打开没问题,但是用Excel打开乱码:

别急,原因就是少了一个\ufeffBOM头,改成这样就没问题了:

  1. <button onclick="test()">下载CSV</button>
  2. <script>
  3. function test()
  4. {
  5. var csv = '姓名,期中成绩,期末成绩\n张三,58,95\n李四,98,74';
  6. var a = document.createElement('a');
  7. a.href = 'data:text/txt;charset=utf-8,\ufeff'+encodeURIComponent(csv);
  8. a.download = '测试.csv';
  9. a.click(); // 这里偷个懒,直接用click模拟
  10. }
  11. </script>

继续解决下载文件名的问题

大部分浏览器可能都没啥问题,但是一些比较老的Chrome可能下载的时候指定的download就是不生效,此时可以用blob来解决:

  1. var csv = '姓名,期中成绩,期末成绩\n张三,58,95\n李四,98,74';
  2. var blob = new Blob(['\ufeff' + data], {type: 'text/csv,charset=UTF-8'});
  3. openDownloadDialog(blob, '测试.csv');

建议一般情况下都用这种方法,稳妥一点。

最后总结

不考虑兼容性的保存CSV方法:

  1. /**
  2. * 保存CSV文件
  3. * @params csv csv文件内容
  4. * @params saveName 保存的文件名
  5. */
  6. function saveCSV(csv, saveName)
  7. {
  8. var a = document.createElement('a');
  9. a.href = 'data:text/csv;charset=utf-8,\ufeff' + encodeURIComponent(csv);
  10. a.download = saveName;
  11. a.click();
  12. }

考虑兼容性的保存CSV方法:

  1. /**
  2. * 保存CSV文件
  3. * @params csv csv文件内容
  4. * @params saveName 保存的文件名
  5. */
  6. function saveCSV(csv, saveName)
  7. {
  8. var blob = new Blob(['\ufeff' + csv], {type: 'text/csv,charset=UTF-8'});
  9. openDownloadDialog(blob, saveName);
  10. }

JS实现纯文本的下载保存

掌握了csv,再去下载纯文本基本上就没啥问题了,就是换一下文件类型而已:

  1. var csv = '你好,我是小茗同学!\n测试换行!';
  2. var blob = new Blob([data], {type: 'text/txt,charset=UTF-8'});
  3. openDownloadDialog(blob, '测试.csv');

JS实现图片的下载保存

网页上一般要保存图片都是从canvas里面拿到的图片数据,通过toDataURL转换为base64数据:

  1. /**
  2. * 将某个canvas保存为图片
  3. * @param canvasObj canvas对象
  4. * @param saveName 保存的名称
  5. * @param type 保存的图片格式,如 image/png
  6. * @param quality 图片质量,可选0-1
  7. */
  8. function saveImage(canvasObj, saveName, type, quality)
  9. {
  10. if(!canvasObj) return;
  11. type = type || 'image/png';
  12. quality = quality || 0.92;
  13. var url = canvasObj.toDataURL(type, quality).replace(/image\/.*?;/, 'image/octet-stream;');
  14. openDownloadDialog(url, saveName);
  15. }

扩展

关于文件保存,不嫌麻烦的话,GitHub上面有个比较出名的库:https://github.com/eligrey/FileSaver.js/

demo:https://eligrey.com/demos/FileSaver.js/

JS弹出下载对话框以及实现常见文件类型的下载的更多相关文章

  1. jquery简易版xwPop.js弹出消息对话框插件

    xwPop.js弹出消息对话框插件是一款含有多种情景模式的原生模态消息对话框代码,可用于替代浏览器默认的alert弹出对话框,它提供各种参数和方法,功能非常强大.目前已经在项目中有应用到xwpop开发 ...

  2. a链接中 JS弹出确认对话框方法

    一种: <a href="javascript:if(confirm('确实要删除该内容吗?'))location='http://www.google.com'">弹 ...

  3. js 弹出QQ对话框

    首先打开下面链接,开通QQ推广. http://shang.qq.com/v3/index.html 然后在页面代码中写入 <a target="_blank" href=& ...

  4. js弹出QQ对话框在线交谈

    <div style="position:absolute; top:110px; right:220px; z-index:2;"> <a target=&qu ...

  5. js弹出框、对话框、提示框、弹窗总结

    一.JS的三种最常见的对话框 //====================== JS最常用三种弹出对话框 ======================== //弹出对话框并输出一段提示信息 funct ...

  6. 【转】js弹出框、对话框、提示框、弹窗总结

    js弹出框.对话框.提示框.弹窗总结 一.js的三种最常见的对话框 //====================== JS最常用三种弹出对话框 ======================== //弹 ...

  7. js js弹出框、对话框、提示框、弹窗总结

    js弹出框.对话框.提示框.弹窗总结 一.JS的三种最常见的对话框 //====================== JS最常用三种弹出对话框 ======================== //弹 ...

  8. 转载 js弹出框、对话框、提示框、弹窗总结

    转载:https://blog.csdn.net/huileiforever/article/details/9464659 一.JS的三种最常见的对话框   //================== ...

  9. JavaScript学习总结(6)——js弹出框、对话框、提示框、弹窗总结

    一.JS的三种最常见的对话框 [javascript] view plaincopy //====================== JS最常用三种弹出对话框 =================== ...

随机推荐

  1. 开涛spring3(6.5) - AOP 之 6.5 AspectJ切入点语法详解

    6.5.1  Spring AOP支持的AspectJ切入点指示符 切入点指示符用来指示切入点表达式目的,,在Spring AOP中目前只有执行方法这一个连接点,Spring AOP支持的Aspect ...

  2. Servlet 详解

    1.什么是 Servlet? Java Servlet 是运行在 Web 服务器或应用服务器上的程序,它是作为来自 Web 浏览器或其他 HTTP 客户端的请求和 HTTP 服务器上的数据库或应用程序 ...

  3. net.sf.json.JSONException: java.lang.reflect.InvocationTargetException Caused by: java.lang.IllegalArgumentException at java.sql.Date.getHours(Unknown Source)

    数据库字段类型为Date,转成JSON格式会有问题,解决方案如下: json-lib有一个配置类JsonConfig通过JsonConfig可以注册一个字段处理器实现JsonValueProcesso ...

  4. hdu_A Walk Through the Forest ——迪杰特斯拉+dfs

    A Walk Through the Forest Time Limit : 2000/1000ms (Java/Other)   Memory Limit : 65536/32768K (Java/ ...

  5. 原生js实现图片网格式渐显、渐隐效果

    写正文前先吐槽一下:端午放假完第一天去某千人以上公司面试前端工程师,第一轮是我应聘职位的部门小领导,谈的不错,面试主要围绕要用到的技术来:第二轮来了我要说的正主,我了个去,问的问题一个和前端无关,问我 ...

  6. 模板C++ 03图论算法 2最短路之全源最短路(Floyd)

    3.2最短路之全源最短路(Floyd) 这个算法用于求所有点对的最短距离.比调用n次SPFA的优点在于代码简单,时间复杂度为O(n^3).[无法计算含有负环的图] 依次扫描每一点(k),并以该点作为中 ...

  7. Akka(7): FSM:通过状态变化来转换运算行为

    在上篇讨论里我们提到了become/unbecome.由于它们本质上是堆栈操作,所以只能在较少的状态切换下才能保证堆栈操作的协调及维持程序的清晰逻辑.对于比较复杂的程序流程,Akka提供了FSM:一种 ...

  8. Function.prototyoe.call.apply

    刚刚在一个群里看到有人问 Function.prototype.call.apply(obj, args) 如何理解,觉得挺有意思的.刚开始被惯性思维干扰了,一直都是 call 和 apply 分开用 ...

  9. Nodejs的http模块

    一.http服务器    我们知道传统的HTTP服务器是由Aphche.Nginx.IIS之类的软件来搭建的,但是Nodejs并不需要,Nodejs提供了http模块,自身就可以用来构建服务器,例如: ...

  10. [1] [转]软件架构之三层架构和MVC的关系

    注:本文章内所有内容都来自互联网,本人主要是起了一个收集的作用 又看到有人在问三层架构和MVC的关系,感觉这种问题有点教条化了.因为它们都在逻辑上将应用程序划为三块,凑了一个数字3,就有人非要把它们联 ...