手记

js不借助后端,多文件拖拽压缩上传,支持选择文件夹

今天产品经理又提了一个需求.

在系统中上传文件时,需要支持多文件和文件夹上传,并且需要在文件上传时需要将多文件压缩成zip包,下载的时候,直接下载zip包

听到这个需求,我的第一反应就是js应该没有强大吧,因为压缩和访问文件夹,涉及到了文件系统的读取和修改,后来经过一番探索还真让我给实现出来了.

主要用到的库是 jszip

 则  则这里简单对jszip做下简单介绍,更详细的功能和api请移步官网.

jszip是是一个创建,读取和写入.zip文件的js库, api优化,简单
浏览器支持

实现思路如下:

1:用户选中文件或文件夹后,获取文件对象,

2:遍历获取的文件对象 放入实例化的zip对象中

3:使用generateAsync()方法生成文件, 通过formdata提交到服务端

代码如下:

此案例支持拖拽上传多个文件, 支持选择多个文件,选择单个文件夹  此外可以使用 file-saver库的saveAs对zip文件包保存到本地

<template><div class="home"><img alt="Vue logo" src="../assets/logo.png">文件压缩上传案例<label name="myfile" for="myfile" :class="{'is-dragover': dragover}"@drop.prevent="onDrop"@dragover.prevent="onDragover"@dragleave.prevent="dragover = false"><input type="file" id="myfile" multiple  webkitdirectory  @change="selectFile" /></label></div></template><script>// @ is an alias to /srcimport JSZip from 'jszip'import axios from 'axios'import { saveAs } from 'file-saver'export default {name: 'home',components: {},data () {return {dragover: false}},mounted () {},methods: {selectFile (even) {var files = even.target.filesthis.commmon(files)},onDrop (even) {this.dragover = falsevar files = even.dataTransfer.filesthis.commmon(files)},commmon (files) {var zip = new JSZip()for (var i = 0; i < files.length; i++) {var f = files[i]zip.file(f.name, f)}zip.generateAsync({ type: 'blob' }).then(function (content) {saveAs(content, 'example.zip')let formData = new FormData()formData.append('file', content)axios({method: 'post',url: '/framework/file/create',data: formData,withCredentials: true, // 默认的headers: {'Content-Type': 'application/x-www-form-urlencoded'}})})},onDragover () {this.dragover = true}}}</script><style lang="scss">.is-dragover{border: 2px solid red !important}</style>

 

这里说一下拖拽上传文件, 主要是用到的原生事件是这三个

 ondrop      ondragover      ondragleave

加上prevent 可以防止拖拽过程,浏览器 直接打开文件,阻止事件默认行为.

ondragover的事件上可以处理文件拖拽到了可放置的元素上,对用户 进行友好提示.

ondrop     事件是文件拖拽到了元素上,松开鼠标时触发, 这个时候可以通过事件拿到拖拽的文件列表 使用even.dataTransfer.files 

 

jszip库的常用api是这两个

 

file(name, data [,options]) :创建zip文件,可以放入多个文件,支持多种文件格式String/ArrayBuffer/Uint8Array/Buffer/Blob/Promise/Nodejs stream,

name type description
name string the name of the file. You can specify folders in the name : the folder separator is a forward slash (“/”).
data String/ArrayBuffer/Uint8Array/Buffer/Blob/Promise/Nodejs stream the content of the file.
options object the options.

 

 

Content of options :

name type default description
base64 boolean false set to true if the data is base64 encoded. For example image data from a <canvas> element. Plain text and HTML do not need this option. More.
binary boolean false set to true if the data should be treated as raw content, false if this is a text. If base64 is used, this defaults to true, if the data is not a string, this will be set to trueMore.
date date the current date the last modification date. More.
compression string null If set, specifies compression method to use for this specific file. If not, the default file compression will be used, see generateAsync(options)More.
compressionOptions object null the options to use when compressing the file, see generateAsync(options)More.
comment string null The comment for this file. More.
optimizedBinaryString boolean false Set to true if (and only if) the input is a “binary string” and has already been prepared with a 0xFF mask.
createFolders boolean true Set to true if folders in the file path should be automatically created, otherwise there will only be virtual folders that represent the path to the file. More.
unixPermissions 16 bits number null The UNIX permissions of the file, if any. More.
dosPermissions 6 bits number null The DOS permissions of the file, if any. More.
dir boolean false Set to true if this is a directory and content should be ignored. More.

generateAsync(options[, onUpdate]): 生成一个完整的zip的文件在当前文件目录 返回一个promise

Arguments

name type default description
options object   the options to generate the zip file :
options.type string   The type of zip to return, see below for the other types. RequiredMore.
options.compression string STORE (no compression) the default file compression method to use. More.
options.compressionOptions object null the options to use when compressing the file. More.
options.comment string   The comment to use for the zip file. More.
options.mimeType string application/zip mime-type for the generated file. More.
options.platform string DOS The platform to use when generating the zip file. More.
options.encodeFileName function encode with UTF-8 the function to encode the file name / comment. More.
options.streamFiles boolean false Stream the files and create file descriptors, see below. More.
onUpdate function   The optional function called on each internal update with the metadata. More.

type option

Possible values for type :

  • base64: the result will be a string, the binary in a base64 form.

  • binarystring (or string, deprecated): the result will be a string in “binary” form, using 1 byte per char (2 bytes).

  • array: the result will be an Array of bytes (numbers between 0 and 255) containing the zip.

  • uint8array: the result will be a Uint8Array containing the zip. This requires a compatible browser.

  • arraybuffer: the result will be a ArrayBuffer containing the zip. This requires a compatible browser.

  • blob: the result will be a Blob containing the zip. This requires a compatible browser.

  • nodebuffer: the result will be a nodejs Buffer containing the zip. This requires nodejs.

 

能不能做,要想看这个库的api具不具备将文件转化为zip文件,通过以上两个api,可以得知这个库是支持的.

 

jszip库api

 

另外jszip库也支持读取本地和远程的zip文件返回内部文件目录,文件名.等信息,有兴趣的朋友可以去尝试一下

 

 

1人推荐
随时随地看视频
慕课网APP