黑马程序员技术交流社区

标题: 【上海校区】element-ui Upload 上传组件源码分析整理笔记 [打印本页]

作者: 梦缠绕的时候    时间: 2019-9-3 08:39
标题: 【上海校区】element-ui Upload 上传组件源码分析整理笔记
简单写了部分注释,upload-dragger.vue(拖拽上传时显示此组件)、upload-list.vue(已上传文件列表)源码暂未添加多少注释,等有空再补充,先记下来...
index.vue<script>import UploadList from './upload-list';import Upload from './upload';import ElProgress from 'element-ui/packages/progress';import Migrating from 'element-ui/src/mixins/migrating';function noop() {}export default {  name: 'ElUpload',  mixins: [Migrating],  components: {    ElProgress,    UploadList,    Upload  },  provide() {    return {      uploader: this    };  },  inject: {    elForm: {      default: ''    }  },  props: {    action: { //必选参数,上传的地址      type: String,      required: true    },    headers: { //设置上传的请求头部      type: Object,      default() {        return {};      }    },    data: Object, //上传时附带的额外参数    multiple: Boolean, //是否支持多选文件    name: { //上传的文件字段名      type: String,      default: 'file'    },    drag: Boolean, //是否启用拖拽上传    dragger: Boolean,    withCredentials: Boolean, //支持发送 cookie 凭证信息    showFileList: { //是否显示已上传文件列表      type: Boolean,      default: true    },    accept: String, //接受上传的文件类型(thumbnail-mode 模式下此参数无效)    type: {      type: String,      default: 'select'    },    beforeUpload: Function, //上传文件之前的钩子,参数为上传的文件,若返回 false 或者返回 Promise 且被 reject,则停止上传。    beforeRemove: Function, //删除文件之前的钩子,参数为上传的文件和文件列表,若返回 false 或者返回 Promise 且被 reject,则停止上传。    onRemove: { //文件列表移除文件时的钩子      type: Function,      default: noop    },    onChange: { //文件状态改变时的钩子,添加文件、上传成功和上传失败时都会被调用      type: Function,      default: noop    },    onPreview: { //点击文件列表中已上传的文件时的钩子      type: Function    },    onSuccess: { //文件上传成功时的钩子      type: Function,      default: noop    },    onProgress: { //文件上传时的钩子      type: Function,      default: noop    },    onError: { //文件上传失败时的钩子      type: Function,      default: noop    },    fileList: { //上传的文件列表, 例如: [{name: 'food.jpg', url: 'https://xxx.cdn.com/xxx.jpg'}]      type: Array,      default() {        return [];      }    },    autoUpload: { //是否在选取文件后立即进行上传      type: Boolean,      default: true    },    listType: { //文件列表的类型      type: String,      default: 'text' // text,picture,picture-card    },    httpRequest: Function, //覆盖默认的上传行为,可以自定义上传的实现    disabled: Boolean, //是否禁用    limit: Number, //最大允许上传个数    onExceed: { //文件超出个数限制时的钩子      type: Function,      default: noop    }  },  data() {    return {      uploadFiles: [],      dragOver: false,      draging: false,      tempIndex: 1    };  },  computed: {    uploadDisabled() {      return this.disabled || (this.elForm || {}).disabled;    }  },  watch: {    fileList: {      immediate: true,      handler(fileList) {        this.uploadFiles = fileList.map(item => {          item.uid = item.uid || (Date.now() + this.tempIndex++);          item.status = item.status || 'success';          return item;        });      }    }  },  methods: {    //文件上传之前调用的方法    handleStart(rawFile) {      rawFile.uid = Date.now() + this.tempIndex++;      let file = {        status: 'ready',        name: rawFile.name,        size: rawFile.size,        percentage: 0,        uid: rawFile.uid,        raw: rawFile      };      //判断文件列表类型      if (this.listType === 'picture-card' || this.listType === 'picture') {        try {          file.url = URL.createObjectURL(rawFile);        } catch (err) {          console.error('[Element Error][Upload]', err);          return;        }      }      this.uploadFiles.push(file);      this.onChange(file, this.uploadFiles);    },    handleProgress(ev, rawFile) {      const file = this.getFile(rawFile);      this.onProgress(ev, file, this.uploadFiles);      file.status = 'uploading';      file.percentage = ev.percent || 0;    },    //文件上传成功后改用该方法,在该方法中调用用户设置的on-success和on-change方法,并将对应的参数传递出去    handleSuccess(res, rawFile) {      const file = this.getFile(rawFile);      if (file) {        file.status = 'success';        file.response = res;        this.onSuccess(res, file, this.uploadFiles);        this.onChange(file, this.uploadFiles);      }    },    //文件上传失败后改用该方法,在该方法中调用用户设置的on-error和on-change方法,并将对应的参数传递出去    handleError(err, rawFile) {      const file = this.getFile(rawFile);      const fileList = this.uploadFiles;      file.status = 'fail';      fileList.splice(fileList.indexOf(file), 1);      this.onError(err, file, this.uploadFiles);      this.onChange(file, this.uploadFiles);    },    //文件列表移除文件时调用该方法    handleRemove(file, raw) {      if (raw) {        file = this.getFile(raw);      }      let doRemove = () => {        this.abort(file);        let fileList = this.uploadFiles;        fileList.splice(fileList.indexOf(file), 1);        this.onRemove(file, fileList);      };      if (!this.beforeRemove) {        doRemove();      } else if (typeof this.beforeRemove === 'function') {        const before = this.beforeRemove(file, this.uploadFiles);        if (before && before.then) {          before.then(() => {            doRemove();          }, noop);        } else if (before !== false) {          doRemove();        }      }    },    getFile(rawFile) {      let fileList = this.uploadFiles;      let target;      fileList.every(item => {        target = rawFile.uid === item.uid ? item : null;        return !target;      });      return target;    },    abort(file) {      this.$refs['upload-inner'].abort(file);    },    clearFiles() {      this.uploadFiles = [];    },    submit() {      this.uploadFiles        .filter(file => file.status === 'ready')        .forEach(file => {          this.$refs['upload-inner'].upload(file.raw);        });    },    getMigratingConfig() {      return {        props: {          'default-file-list': 'default-file-list is renamed to file-list.',          'show-upload-list': 'show-upload-list is renamed to show-file-list.',          'thumbnail-mode': 'thumbnail-mode has been deprecated, you can implement the same effect according to this case: http://element.eleme.io/#/zh-CN/ ... u-xiang-shang-chuan        }      };    }  },  beforeDestroy() {    this.uploadFiles.forEach(file => {      if (file.url && file.url.indexOf('blob:') === 0) {        URL.revokeObjectURL(file.url);      }    });  },  render(h) {    let uploadList;    //如果用户设置showFileList为true,则显示上传文件列表    if (this.showFileList) {      uploadList = (        <UploadList          disabled={this.uploadDisabled}          listType={this.listType}          files={this.uploadFiles}          on-remove={this.handleRemove}          handlePreview={this.onPreview}>        </UploadList>      );    }    const uploadData = {      props: {        type: this.type,        drag: this.drag,        action: this.action,        multiple: this.multiple,        'before-upload': this.beforeUpload,        'with-credentials': this.withCredentials,        headers: this.headers,        name: this.name,        data: this.data,        accept: this.accept,        fileList: this.uploadFiles,        autoUpload: this.autoUpload,        listType: this.listType,        disabled: this.uploadDisabled,        limit: this.limit,        'on-exceed': this.onExceed,        'on-start': this.handleStart,        'on-progress': this.handleProgress,        'on-success': this.handleSuccess,        'on-error': this.handleError,        'on-preview': this.onPreview,        'on-remove': this.handleRemove,        'http-request': this.httpRequest      },      ref: 'upload-inner'    };    const trigger = this.$slots.trigger || this.$slots.default;    const uploadComponent = <upload {...uploadData}>{trigger}</upload>;    return (      <div>        { this.listType === 'picture-card' ? uploadList : ''}        {          this.$slots.trigger            ? [uploadComponent, this.$slots.default]            : uploadComponent        }        {this.$slots.tip}        { this.listType !== 'picture-card' ? uploadList : ''}      </div>    );  }};</script>upload.vue<script>import ajax from './ajax';import UploadDragger from './upload-dragger.vue';export default {  inject: ['uploader'],  components: {    UploadDragger  },  props: {    type: String,    action: { //必选参数,上传的地址      type: String,      required: true    },    name: { //上传的文件字段名      type: String,      default: 'file'    },    data: Object, //上传时附带的额外参数    headers: Object, //设置上传的请求头部    withCredentials: Boolean, //支持发送 cookie 凭证信息    multiple: Boolean, //是否支持多选文件    accept: String, //接受上传的文件类型(thumbnail-mode 模式下此参数无效)    onStart: Function,    onProgress: Function, //文件上传时的钩子    onSuccess: Function, //文件上传成功时的钩子    onError: Function, //文件上传失败时的钩子    beforeUpload: Function, //上传文件之前的钩子,参数为上传的文件,若返回 false 或者返回 Promise 且被 reject,则停止上传。    drag: Boolean, //是否启用拖拽上传    onPreview: { //点击文件列表中已上传的文件时的钩子      type: Function,      default: function() {}    },    onRemove: { //文件列表移除文件时的钩子      type: Function,      default: function() {}    },    fileList: Array, //上传的文件列表, 例如: [{name: 'food.jpg', url: 'https://xxx.cdn.com/xxx.jpg'}]    autoUpload: Boolean, //是否在选取文件后立即进行上传    listType: String, //文件列表的类型    httpRequest: { //覆盖默认的上传行为,可以自定义上传的实现      type: Function,      default: ajax    },    disabled: Boolean,//是否禁用    limit: Number,//最大允许上传个数    onExceed: Function //文件超出个数限制时的钩子  },  data() {    return {      mouseover: false,      reqs: {}    };  },  methods: {    isImage(str) {      return str.indexOf('image') !== -1;    },    handleChange(ev) {      const files = ev.target.files;      if (!files) return;      this.uploadFiles(files);    },    uploadFiles(files) {      //文件超出个数限制时,调用onExceed钩子函数      if (this.limit && this.fileList.length + files.length > this.limit) {        this.onExceed && this.onExceed(files, this.fileList);        return;      }      //将files转成数组      let postFiles = Array.prototype.slice.call(files);      if (!this.multiple) { postFiles = postFiles.slice(0, 1); }      if (postFiles.length === 0) { return; }      postFiles.forEach(rawFile => {        this.onStart(rawFile);        //选取文件后调用upload方法立即进行上传文件        if (this.autoUpload) this.upload(rawFile);      });    },    upload(rawFile) {      this.$refs.input.value = null;      //beforeUpload 上传文件之前的钩子不存在就直接调用post上传文件      if (!this.beforeUpload) {        return this.post(rawFile);      }      // beforeUpload 上传文件之前的钩子,参数为上传的文件,若返回 false 或者返回 Promise 且被 reject,则停止上传      const before = this.beforeUpload(rawFile);      // 在调用beforeUpload钩子后返回的是true,则继续上传      if (before && before.then) {        before.then(processedFile => {          //processedFile转成对象          const fileType = Object.prototype.toString.call(processedFile);          if (fileType === '[object File]' || fileType === '[object Blob]') {            if (fileType === '[object Blob]') {              processedFile = new File([processedFile], rawFile.name, {                type: rawFile.type              });            }            for (const p in rawFile) {              if (rawFile.hasOwnProperty(p)) {                processedFile[p] = rawFile[p];              }            }            this.post(processedFile);          } else {            this.post(rawFile);          }        }, () => {          this.onRemove(null, rawFile);        });      } else if (before !== false) { //调用beforeUpload之后没有返回值,此时before为undefined,继续上传        this.post(rawFile);      } else {  //调用beforeUpload之后返回值为false,则不再继续上传并移除文件        this.onRemove(null, rawFile);      }    },    abort(file) {      const { reqs } = this;      if (file) {        let uid = file;        if (file.uid) uid = file.uid;        if (reqs[uid]) {          reqs[uid].abort();        }      } else {        Object.keys(reqs).forEach((uid) => {          if (reqs[uid]) reqs[uid].abort();          delete reqs[uid];        });      }    },    //上传文件过程的方法    post(rawFile) {      const { uid } = rawFile;      const options = {        headers: this.headers,        withCredentials: this.withCredentials,        file: rawFile,        data: this.data,        filename: this.name,        action: this.action,        onProgress: e => { //文件上传时的钩子函数          this.onProgress(e, rawFile);        },        onSuccess: res => { //文件上传成功的钩子函数          //上传成功调用onSuccess方法,即index.vue中的handleSuccess方法          this.onSuccess(res, rawFile);          delete this.reqs[uid];        },        onError: err => { //文件上传失败的钩子函数          this.onError(err, rawFile);          delete this.reqs[uid];        }      };      //httpRequest可以自定义上传文件,如果没有定义,默认通过ajax文件中的方法上传      const req = this.httpRequest(options);      this.reqs[uid] = req;      if (req && req.then) {        req.then(options.onSuccess, options.onError);      }    },    handleClick() {      //点击组件时调用input的click方法      if (!this.disabled) {        this.$refs.input.value = null;        this.$refs.input.click();      }    },    handleKeydown(e) {      if (e.target !== e.currentTarget) return;      //如果当前按下的是回车键和空格键,调用handleClick事件      if (e.keyCode === 13 || e.keyCode === 32) {        this.handleClick();      }    }  },  render(h) {    let {      handleClick,      drag,      name,      handleChange,      multiple,      accept,      listType,      uploadFiles,      disabled,      handleKeydown    } = this;    const data = {      class: {        'el-upload': true      },      on: {        click: handleClick,        keydown: handleKeydown      }    };    data.class[`el-upload--${listType}`] = true;    return (      //判断是否允许拖拽,允许的话显示upload-dragger组件,不允许就显示所有插槽中的节点      <div {...data} tabindex="0" >        {          drag            ? <upload-dragger disabled={disabled} on-file={uploadFiles}>{this.$slots.default}</upload-dragger>            : this.$slots.default        }        <input class="el-upload__input" type="file" ref="input" name={name} on-change={handleChange} multiple={multiple} accept={accept}></input>      </div>    );  }};</script>ajax.jsfunction getError(action, option, xhr) {  let msg;  if (xhr.response) {    msg = `${xhr.response.error || xhr.response}`;  } else if (xhr.responseText) {    msg = `${xhr.responseText}`;  } else {    msg = `fail to post ${action} ${xhr.status}`;  }  const err = new Error(msg);  err.status = xhr.status;  err.method = 'post';  err.url = action;  return err;}function getBody(xhr) {  const text = xhr.responseText || xhr.response;  if (!text) {    return text;  }  try {    return JSON.parse(text);  } catch (e) {    return text;  }}//默认的上传文件的方法export default function upload(option) {  //XMLHttpRequest 对象用于在后台与服务器交换数据。  if (typeof XMLHttpRequest === 'undefined') {    return;  }  //创建XMLHttpRequest对象  const xhr = new XMLHttpRequest();  const action = option.action; //上传的地址  //XMLHttpRequest.upload 属性返回一个 XMLHttpRequestUpload对象,用来表示上传的进度。这个对象是不透明的,但是作为一个XMLHttpRequestEventTarget,可以通过对其绑定事件来追踪它的进度。  if (xhr.upload) {    //上传进度调用方法,上传过程中会频繁调用该方法    xhr.upload.onprogress = function progress(e) {      if (e.total > 0) {        // e.total是需要传输的总字节,e.loaded是已经传输的字节        e.percent = e.loaded / e.total * 100;      }      //调文件上传时的钩子函数      option.onProgress(e);    };  }  // 创建一个FormData 对象  const formData = new FormData();  //用户设置了上传时附带的额外参数时  if (option.data) {    Object.keys(option.data).forEach(key => {      // 添加一个新值到 formData 对象内的一个已存在的键中,如果键不存在则会添加该键。      formData.append(key, option.data[key]);    });  }  formData.append(option.filename, option.file, option.file.name);  //请求出错  xhr.onerror = function error(e) {    option.onError(e);  };  //请求成功回调函数  xhr.onload = function onload() {    if (xhr.status < 200 || xhr.status >= 300) {      return option.onError(getError(action, option, xhr));    }    //调用upload.vue文件中的onSuccess方法,将上传接口返回值作为参数传递    option.onSuccess(getBody(xhr));  };  //初始化请求  xhr.open('post', action, true);  if (option.withCredentials && 'withCredentials' in xhr) {    xhr.withCredentials = true;  }  const headers = option.headers || {};  for (let item in headers) {    if (headers.hasOwnProperty(item) && headers[item] !== null) {      //设置请求头      xhr.setRequestHeader(item, headers[item]);    }  }  //发送请求  xhr.send(formData);  return xhr;}upload-dragger.vue<template>  <!--拖拽上传时显示此组件-->  <div    class="el-upload-dragger"    :class="{      'is-dragover': dragover    }"    @drop.prevent="onDrop"    @dragover.prevent="onDragover"    @dragleave.prevent="dragover = false"  >    <slot></slot>  </div></template><script>  export default {    name: 'ElUploadDrag',    props: {      disabled: Boolean    },    inject: {      uploader: {        default: ''      }    },    data() {      return {        dragover: false      };    },    methods: {      onDragover() {        if (!this.disabled) {          this.dragover = true;        }      },      onDrop(e) {        if (this.disabled || !this.uploader) return;        //接受上传的文件类型(thumbnail-mode 模式下此参数无效),此处判断该文件是都符合能上传的类型        const accept = this.uploader.accept;        this.dragover = false;        if (!accept) {          this.$emit('file', e.dataTransfer.files);          return;        }        this.$emit('file', [].slice.call(e.dataTransfer.files).filter(file => {          const { type, name } = file;          //获取文件名后缀,与设置的文件类型进行对比          const extension = name.indexOf('.') > -1            ? `.${ name.split('.').pop() }`            : '';          const baseType = type.replace(/\/.*$/, '');          return accept.split(',')            .map(type => type.trim())            .filter(type => type)            .some(acceptedType => {              if (/\..+$/.test(acceptedType)) {                //文件名后缀与设置的文件类型进行对比                return extension === acceptedType;              }              if (/\/\*$/.test(acceptedType)) {                return baseType === acceptedType.replace(/\/\*$/, '');              }              if (/^[^\/]+\/[^\/]+$/.test(acceptedType)) {                return type === acceptedType;              }              return false;            });        }));      }    }  };</script>upload-list.vue<template>  <!--这里主要显示已上传文件列表-->  <transition-group    tag="ul"    :class="[      'el-upload-list',      'el-upload-list--' + listType,      { 'is-disabled': disabled }    ]"    name="el-list">    <li      v-for="file in files"      :class="['el-upload-list__item', 'is-' + file.status, focusing ? 'focusing' : '']"      :key="file.uid"      tabindex="0"      @keydown.delete="!disabled && $emit('remove', file)"      @focus="focusing = true"      @blur="focusing = false"      @click="focusing = false"    >      <img        class="el-upload-list__item-thumbnail"        v-if="file.status !== 'uploading' && ['picture-card', 'picture'].indexOf(listType) > -1"        :src="file.url" alt=""      >      <a class="el-upload-list__item-name" @click="handleClick(file)">        <i class="el-icon-document"></i>{{file.name}}      </a>      <label class="el-upload-list__item-status-label">        <i :class="{          'el-icon-upload-success': true,          'el-icon-circle-check': listType === 'text',          'el-icon-check': ['picture-card', 'picture'].indexOf(listType) > -1        }"></i>      </label>      <i class="el-icon-close" v-if="!disabled" @click="$emit('remove', file)"></i>      <i class="el-icon-close-tip" v-if="!disabled">{{ t('el.upload.deleteTip') }}</i> <!--因为close按钮只在li:focus的时候 display, li blur后就不存在了,所以键盘导航时永远无法 focus到 close按钮上-->      <el-progress        v-if="file.status === 'uploading'"        :type="listType === 'picture-card' ? 'circle' : 'line'"        :stroke-width="listType === 'picture-card' ? 6 : 2"        :percentage="parsePercentage(file.percentage)">      </el-progress>      <span class="el-upload-list__item-actions" v-if="listType === 'picture-card'">        <span          class="el-upload-list__item-preview"          v-if="handlePreview && listType === 'picture-card'"          @click="handlePreview(file)"        >          <i class="el-icon-zoom-in"></i>        </span>        <span          v-if="!disabled"          class="el-upload-list__item-delete"          @click="$emit('remove', file)"        >          <i class="el-icon-delete"></i>        </span>      </span>    </li>  </transition-group></template><script>  import Locale from 'element-ui/src/mixins/locale';  import ElProgress from 'element-ui/packages/progress';  export default {    name: 'ElUploadList',    mixins: [Locale],    data() {      return {        focusing: false      };    },    components: { ElProgress },    props: {      files: {        type: Array,        default() {          return [];        }      },      disabled: {        type: Boolean,        default: false      },      handlePreview: Function,      listType: String    },    methods: {      parsePercentage(val) {        return parseInt(val, 10);      },      handleClick(file) {        this.handlePreview && this.handlePreview(file);      }    }  };</script>

作者: 梦缠绕的时候    时间: 2019-9-3 08:39
有任何问题欢迎在评论区留言
作者: 梦缠绕的时候    时间: 2019-9-3 08:39
或者添加学姐微信
DKA-2018




欢迎光临 黑马程序员技术交流社区 (http://bbs.itheima.com/) 黑马程序员IT技术论坛 X3.2