Skip to content
html
<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="UTF-8">
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <title>Document</title>
  <style>
    .upload-container {
      width: 500px;
      height: 100%;
      border: 1px solid #ebebeb;
      margin: 0 auto;
      margin-top: 50px;
      box-shadow: 3px 3px rgba(0, 0, 0, .3);
      padding: 0 30px;
      padding-bottom: 50px;
    }

    .upload-container .upload-btn {
      margin-top: 10px;
    }

    .upload-container .upload-btn .disabled {
      background-color: #dddddd !important;
    }

    .upload-container .upload-btn .select, .submit{
      height: 32px;
      background-color: #409eff;
      font-size: 12px;
      border: none;
      border-radius: 3px;
      color: #fff;
    }

    .upload-container .upload-btn .submit {
      margin-left: 10px;
    }
    
    .upload-container .upload-tip {
      font-size: 12px;
      color: #606266;
      margin-top: 10px;
    }

    .upload-container .upload-abbre {
      margin-top: 20px;
    }
  </style>
</head>
<body>
  <section class="upload-container">
    <input name="upload-inp" type="file" style="display: none;">
    <div class="upload-btn">
      <button class="select">点击上传</button>
      <button class="submit">上传服务器</button>
    </div>
    <div class="upload-tip">
      只能上传JPG/PNG/JPEG/WEBP文件, 且大小不能超过500kb
    </div>
    <div class="upload-abbre">
      <img src="" alt="">
    </div>
  </section>

  <script src="https://cdn.bootcdn.net/ajax/libs/axios/0.21.1/axios.js"></script>
  <script src="https://cdn.bootcdn.net/ajax/libs/qs/6.10.1/qs.js"></script>
  <script src="https://cdn.bootcdn.net/ajax/libs/spark-md5/3.0.0/spark-md5.js"></script>
  <script src="./request.js"></script>
  <script>
    let uploadContainer = document.querySelector('.upload-container'),
      uploadInp = uploadContainer.querySelector('[name=upload-inp]'),
      uploadBtn = uploadContainer.querySelector('.upload-btn'),
      uploadAbbre = uploadContainer.querySelector('.upload-abbre'),
      abbre = uploadAbbre.querySelector('img'),
      select = uploadBtn.querySelector('.select'),
      submit = uploadBtn.querySelector('.submit'),
      _file = null

    // 点击上传文件按钮, 触发input的点击事件
    select.addEventListener('click', function () {
      if (isDisabled()) {
        // 按钮禁用
        return
      }

      if (_file) {
        alert('一次只能上传一张图片~')
        return
      }

      uploadInp.click()
    })

    // 禁用按钮
    function isDisabled () {
      return select.classList.contains('disabled') || submit.classList.contains('disabled')
    }

    // 清除abbre
    function clearAbbre () {
      _file = null
      uploadInp.value = ''
      abbre.src = ''
    }

    // 图片转化为base64
    function img2Base64 (file) {
      return new Promise((resolve, reject) => {
        let fileReader = new FileReader
        fileReader.readAsDataURL(file) // => 会异步转化成Base64 => Promise封装
        fileReader.onload = function (event) {
          // 转化结果保存在event.target.result中
          resolve(event.target.result)
        }
      })
    }

    // 图片转化为Buffer
    function img2Buffer (file) {
      return new Promise((resolve, reject) => {
        let fileReader = new FileReader
        fileReader.readAsArrayBuffer(file) // => 会异步转化成Base64 => Promise封装
        fileReader.onload = function (event) {
          // 转化结果保存在event.target.result中
          resolve(event.target.result)
        }
      })
    }

    // 根据内容生成hash名
    function hashHandler (buffer) {
      let spark = new SparkMD5.ArrayBuffer(),
        hash = null,
        suffix = /\.([a-zA-Z0-9]+)$/.exec(_file.name)[0]

      spark.append(buffer)
      hash = spark.end()
      return {
        buffer,
        hash,
        suffix,
        filename: hash + suffix
      }
    }

    // 选择图片时的处理
    uploadInp.addEventListener('change', async function () {
      _file = this.files[0]

      if (!_file) {
        return
      }

      // 限制图片类型
      if (!/(jpg|png|jpeg|webp)/i.test(_file.type)) {
        alert('上传的图片格式不正确~')
        return
      }

      // 限制文件大小
      if (_file.size > 500 * 1024) {
        alert('文件太大了~')
        return
      }

      // 显示缩略图 img.src设置为base64即可
      abbre.src = await img2Base64(_file)
    })

    // 上传图片
    submit.addEventListener('click', async function () {
      if (isDisabled()) {
        // 按钮禁用
        return
      }

      if (!_file) {
        alert('你还没有选择图片~')
        return
      }

      // 上传过程中禁用按钮
      select.classList.toggle('disabled')
      submit.classList.toggle('disabled')

      let buffer = await img2Buffer(_file),
        { filename} = hashHandler(buffer)

      let formData = new FormData
      formData.append('file', _file)
      formData.append('filename', filename)

      // 上传图片
      instance.post('/upload-single-hash', formData)
        .then(response => {
          if (response.status === 0) {
            // 上传成功
            alert('上传成功~')
          } else {
            return Promise.reject(response.statusText)
          }
        })
        .catch(reason => {
          // 上传失败 response.status = 1 / 响应码>=300
          alert('上传失败, 请稍后再试~')
        })
        .finally(() => {
          clearAbbre()
          _file = null
          // 取消禁用
          select.classList.toggle('disabled')
          submit.classList.toggle('disabled')
        })
    })
  </script>
</body>
</html>