JS实现弹出下载对话框及常见文件类型的下载

编程入门 行业动态 更新时间:2024-10-27 10:34:30
原文链接:http://wwwblogs/liuxianan/p/js-download.html
S要实现下载功能,一般都是这么几个过程:生成下载的URL,动态创建一个A标签,并将其href指向生成的URL,然后触发A标签的单击事件,这样就会弹出下载对话框,从而实现了一个下载的功能

1.写在前面

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

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

2.几个备用知识点

2.1. JS触发单击事件

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

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

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

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

2.2. HTML5的download属性

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

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

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

补充说明:

file:///模式下貌似不生效;

链接指向一些第三方链接时也不会生效,具体有待研究;

2.3. JS弹出下载对话框

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

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

?
1 2 openDownloadDialog(url, saveName) openDownloadDialog(blob, saveName)

代码如下:

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

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

3.1. JS生成CSV文件并下载

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

3.1.1. CSV格式示例

如下:

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

excel打开效果如下:

3.1.2. 初次尝试

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

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

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

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

3.1.3. 解决CSV乱码问题

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

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

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

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

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

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

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

3.1.5. 最后总结

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

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

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

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

3.2. JS实现纯文本的下载保存

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

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

3.3. JS实现图片的下载保存

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

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

扩展

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

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

以上所述是小编给大家介绍的JS实现弹出下载对话框及常见文件类型的下载,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对脚本之家网站的支持!


更多推荐

JS实现弹出下载对话框及常见文件类型的下载

本文发布于:2023-06-13 22:39:00,感谢您对本站的认可!
本文链接:https://www.elefans.com/category/jswz/34/1412626.html
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。
本文标签:弹出   对话框   文件类型   常见   JS

发布评论

评论列表 (有 0 条评论)
草根站长

>www.elefans.com

编程频道|电子爱好者 - 技术资讯及电子产品介绍!