vue3.0搭配.net core實(shí)現(xiàn)文件上傳組件
在開發(fā)Web應(yīng)用程序中,文件上傳是經(jīng)常用到的一個(gè)功能。
在Jquery時(shí)代,做上傳功能,一般找jQuery插件就夠了,很少有人去探究上傳文件插件到底是怎么做的。
簡單列一下我們要做的技術(shù)點(diǎn)和功能點(diǎn)
使用技術(shù)
客戶端使用vue.js 3.0,并使用vue3新增的功能:Composition API ,服務(wù)器使用asp.net core
功能點(diǎn)
- 標(biāo)簽美化
- 文件預(yù)覽
- 文件上傳
- 服務(wù)器接收文件
文件選擇美化
在標(biāo)準(zhǔn)的html文件選擇標(biāo)簽,是十分不美觀的。大概就是下圖的樣子
但是我們的設(shè)計(jì)師的設(shè)計(jì)圖可不是這樣的啊,所以第一步是選擇美化一下樣式。
標(biāo)簽美化
找遍整個(gè)搜索引擎,美化文件選擇標(biāo)簽只有兩種方法
- 設(shè)置input標(biāo)簽透明度為0,然后定位一個(gè)其他的容易修改樣式的標(biāo)簽到透明度度為0的input標(biāo)簽上。
- 設(shè)置input標(biāo)簽的display為none,然后使用JavaScript來觸發(fā)當(dāng)前input的點(diǎn)擊事件。
因?yàn)楣P者最近在做基于vue.js 3.0的項(xiàng)目,需要自己自定義很多UI組件,所以參考了layui element ,它們都是使用第二種方式來美化文件選擇標(biāo)簽。
假設(shè)我們UI設(shè)計(jì)圖是上圖的樣式,如果需要美化,只需要隱藏文件選擇的Input標(biāo)簽。然后放置一個(gè)按鈕,然后設(shè)置按鈕的樣式為設(shè)計(jì)圖上的樣式即可
<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點(diǎn)擊的時(shí)候,使用JavaScript去點(diǎn)擊隱藏的input標(biāo)簽
<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中要獲取到標(biāo)簽的ref,不能使用this.$refs來獲取。當(dāng)然,你如果喜歡使用vue2的options api。那依然可以使用this.$refs來獲取標(biāo)簽的el
只需要簡單的觸發(fā)input的click事件,就可以使瀏覽器彈出文件選擇框了。
文件預(yù)覽
基本上所有的文件上傳組件,都有預(yù)覽上傳圖片的功能。本文所寫的上傳組件當(dāng)然也不例外。
監(jiān)聽input標(biāo)簽的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內(nèi)核等高版本瀏覽器中,無法像低版本瀏覽器一樣,能獲得文件的具體磁盤路徑。如果像以前用文件路徑去獲取文件。只能獲得一個(gè) C:\fakepath"+文件名的路徑。無法獲取到真實(shí)文件路徑。據(jù)說可以通過某些方法獲取真實(shí)路徑。我試過,沒成功。有興趣的朋友可以試試。
文件上傳
選擇文件后,我們需要把文件保存到到服務(wù)器。在傳統(tǒng)的多頁面web程序中,只需要設(shè)置按鈕的type為submit,然后使用form表單直接提交文件和表單信息到服務(wù)器去。
但是我們做單頁面程序,一般來說是通過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("上傳進(jìn)度:" + percent); } }; xhr.onerror = () => { console.log("上傳文件錯(cuò)誤"); }; xhr.ontimeout = () => { console.log("上傳超時(shí)"); }; xhr.send(form); };
在頁面上新增一個(gè)按鈕,用來手動(dòng)觸發(fā)上傳
<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>
點(diǎn)擊 立即上傳 按鈕,觸發(fā)上傳
const uploadClick = () => { data.files.forEach((file) => { uploadServer(file); }); };
服務(wù)器接收
在服務(wù)器編程中,我們使用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運(yùn)行asp.net core服務(wù)端。然后點(diǎn)擊上傳,你以為就上傳成功了嗎?
不!沒那么簡單。如果如果vue程序和asp.net core程序,不在同一個(gè)域名下,你還得處理上傳跨域問題。當(dāng)然這個(gè)問題在asp.net core中是非常簡單的。只需要簡單配置一下即可
如果在IIS或者Nginx下,就需要修改對應(yīng)站點(diǎn)的配置文件了。當(dāng)然具體服務(wù)器軟件的配置不在本篇文章的討論之下。有需要的同學(xué)可以私下交流
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();之前才會(huì)生效。
上傳成功后,你就會(huì)在服務(wù)器的wwwroot的files文件夾中看到上傳的圖片文件了。
本文完成了基本的功能,起一個(gè)拋磚引玉的作用。更多功能,如:文件類型限制,文件大小限制等,可以根據(jù)使用場景自定義擴(kuò)展
本篇vue 3.0文件上傳組件開發(fā)到這里就結(jié)束了。
以上就是vue3.0搭配.net core實(shí)現(xiàn)文件上傳組件的詳細(xì)內(nèi)容,更多關(guān)于vue3.0 文件上傳的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
vue項(xiàng)目打包以及優(yōu)化的實(shí)現(xiàn)步驟
項(xiàng)目完成,我們會(huì)將項(xiàng)目進(jìn)行上線,為了提升性能,我們往往會(huì)進(jìn)行一些優(yōu)化處理,本文主要介紹了vue項(xiàng)目打包以及優(yōu)化的實(shí)現(xiàn)步驟,感興趣的可以了解一下2021-07-07nuxt 實(shí)現(xiàn)在其它js文件中使用store的方式
這篇文章主要介紹了nuxt 實(shí)現(xiàn)在其它js文件中使用store的方式,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-11-11Vue SSR 即時(shí)編譯技術(shù)的實(shí)現(xiàn)
這篇文章主要介紹了Vue SSR 即時(shí)編譯技術(shù)的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-05-05