当前位置: 移动技术网 > IT编程>脚本编程>vue.js > vue3.0搭配.net core实现文件上传组件

vue3.0搭配.net core实现文件上传组件

2020年10月29日  | 移动技术网IT编程  | 我要评论
在开发web应用程序中,文件上传是经常用到的一个功能。在jquery时代,做上传功能,一般找jquery插件就够了,很少有人去探究上传文件插件到底是怎么做的。简单列一下我们要做的技术点和功能点使用技术

在开发web应用程序中,文件上传是经常用到的一个功能。
在jquery时代,做上传功能,一般找jquery插件就够了,很少有人去探究上传文件插件到底是怎么做的。
简单列一下我们要做的技术点和功能点

使用技术

客户端使用vue.js 3.0,并使用vue3新增的功能:composition api ,服务器使用asp.net core

功能点

  1. 标签美化
  2. 文件预览
  3. 文件上传
  4. 服务器接收文件

文件选择美化

在标准的html文件选择标签,是十分不美观的。大概就是下图的样子

但是我们的设计师的设计图可不是这样的啊,所以第一步是选择美化一下样式。

标签美化

找遍整个搜索引擎,美化文件选择标签只有两种方法

  1. 设置input标签透明度为0,然后定位一个其他的容易修改样式的标签到透明度度为0的input标签上。
  2. 设置input标签的display为none,然后使用javascript来触发当前input的点击事件。

因为笔者最近在做基于vue.js 3.0的项目,需要自己自定义很多ui组件,所以参考了layui element ,它们都是使用第二种方式来美化文件选择标签。

假设我们ui设计图是上图的样式,如果需要美化,只需要隐藏文件选择的input标签。然后放置一个按钮,然后设置按钮的样式为设计图上的样式即可

 <div class="uploader">
  <button>选择文件</button>
  <input type="file" placeholder="请选择文件" />
 </div>
.uploader {
 display: inline-block;
 button {
  background: #4e6ef2;
  color: aliceblue;
  padding: 5px;
  outline: none;
  border: none;
  &:hover {
   opacity: 0.8;
  }
  &:active {
   opacity: 1;
  }
 }
 input {
  display: none;
 }
}

美化完成组件后,我们需要用在button点击的时候,使用javascript去点击隐藏的input标签

<template>
 <div class="uploader">
  <button @click="btnclick">选择文件</button>
  <input type="file" placeholder="请选择文件" ref="fileselector" />
 </div>
</template>
 
<script>
import { ref } from "vue";
export default {
 name: "uploader",
 setup() {
  const fileselector = ref(null);
  const btnclick = () => {
   fileselector.value.click();
  };
  return {
   fileselector,
   btnclick,
  };
 },
};
</script>

在composition api中要获取到标签的ref,不能使用this.$refs来获取。当然,你如果喜欢使用vue2的options api。那依然可以使用this.$refs来获取标签的el
只需要简单的触发input的click事件,就可以使浏览器弹出文件选择框了。

文件预览

基本上所有的文件上传组件,都有预览上传图片的功能。本文所写的上传组件当然也不例外。
监听input标签的change事件,获取到files对象。然后使用filereader读取文件信息。

const filechange = (e) => {
   let files = e.target.files;
   console.log(files);
   for (let i = 0; i < files.length; i++) {
    let file = files[i];
    var filereader = new filereader();
    filereader.addeventlistener(
     "load",
     (event) => {
      console.log(event);
      data.imglist.push({
       base64: event.target.result,
      });
     },
     false
    );
    filereader.readasdataurl(file);
   }
  };

在chromium内核等高版本浏览器中,无法像低版本浏览器一样,能获得文件的具体磁盘路径。如果像以前用文件路径去获取文件。只能获得一个 c:\fakepath"+文件名的路径。无法获取到真实文件路径。据说可以通过某些方法获取真实路径。我试过,没成功。有兴趣的朋友可以试试。

文件上传

选择文件后,我们需要把文件保存到到服务器。在传统的多页面web程序中,只需要设置按钮的type为submit,然后使用form表单直接提交文件和表单信息到服务器去。
但是我们做单页面程序,一般来说是通过javascript的ajax去上传文件。

 const uploadserver = (file) => {
   var form = new formdata();
   form.append("file", file);
   var xhr = new xmlhttprequest();
   xhr.open("post", props.server);
   xhr.onreadystatechange = () => {
    if (xhr.readystate == 4 && xhr.status == 200) {
     var res = json.parse(xhr.responsetext);
     console.log("上传成功");
     data.logs.push({
      log: res,
     });
    }
   };
   xhr.upload.onprogress = (event) => {
    if (event.lengthcomputable) {
     var percent = (event.loaded / event.total) * 100;
     console.log("上传进度:" + percent);
    }
   };
   xhr.onerror = () => {
    console.log("上传文件错误");
   };
   xhr.ontimeout = () => {
    console.log("上传超时");
   };
   xhr.send(form);
  };

在页面上新增一个按钮,用来手动触发上传

 <div class="uploader">
  <button @click="btnclick">选择文件</button>
  <button @click="uploadclick">立即上传</button>
  <input
   type="file"
   placeholder="请选择文件"
   ref="fileselector"
   @change="filechange"
   multiple
  />
  <div class="image-list">
   <img v-for="(item, i) in data.imglist" :key="i" :src="item.base64" />
  </div>
  <div class="log">
   <p v-for="(item, i) in data.logs" :key="i">{{ item.log }}</p>
  </div>
 </div>

点击 立即上传 按钮,触发上传

	const uploadclick = () => {
   data.files.foreach((file) => {
    uploadserver(file);
   });
  };

服务器接收

在服务器编程中,我们使用c#来接收上传的文件。

 /// <summary>
    /// 上传
    /// </summary>
    /// <param name="files"></param>
    /// <returns></returns>
    [httppost("/upload")]
    public async task<iactionresult> upload([fromservices] iwebhostenvironment host)
    {
      var files = request.form.files;
      long size = files.sum(f => f.length);
      list<string> list = new list<string>();
      foreach (var formfile in files)
      {
        if (formfile.length > 0)
        {
          var path = path.combine(host.webrootpath, "files");
          
          if (!directory.exists(path))
          {
            directory.createdirectory(path);
          }
          string filename = $"{guid.newguid():n}{path.getextension(formfile.filename)}";
          path = path.combine(path, filename);
          var filepath = path;
 
          using var stream = system.io.file.create(filepath);
          await formfile.copytoasync(stream);
          var c = path.volumeseparatorchar;
          list.add($"{request.scheme}://{request.host.value}/{path.combine("files", filename).replace(path.directoryseparatorchar, path.altdirectoryseparatorchar)}");
        }
      }
 
      return ok(new { list = list, size });
    }
  

使用dotnet run运行asp.net core服务端。然后点击上传,你以为就上传成功了吗?
不!没那么简单。如果如果vue程序和asp.net core程序,不在同一个域名下,你还得处理上传跨域问题。当然这个问题在asp.net core中是非常简单的。只需要简单配置一下即可

如果在iis或者nginx下,就需要修改对应站点的配置文件了。当然具体服务器软件的配置不在本篇文章的讨论之下。有需要的同学可以私下交流
asp.net core跨域处理

 app.usecors(options =>
      {
        options.withorigins("http://localhost:3000", "http://127.0.0.1", "http://localhost:8080"); // 允许特定ip跨域
        options.allowanyheader();
        options.allowanymethod();
        options.allowcredentials();
      });

以上配置必须要放在app.usestaticfiles();之前才会生效。

上传成功后,你就会在服务器的wwwroot的files文件夹中看到上传的图片文件了。

本文完成了基本的功能,起一个抛砖引玉的作用。更多功能,如:文件类型限制,文件大小限制等,可以根据使用场景自定义扩展

本篇vue 3.0文件上传组件开发到这里就结束了。

以上就是vue3.0搭配.net core实现文件上传组件的详细内容,更多关于vue3.0 文件上传的资料请关注移动技术网其它相关文章!

如您对本文有疑问或者有任何想说的,请点击进行留言回复,万千网友为您解惑!

相关文章:

验证码:
移动技术网