Common upload component development
Before developing the upload component we need to know:
- FormData API required for uploading files
- DragOver triggered when a file is dragged to a region
- DragLeave File leaves the drag area
- Drop when the file is moved to a valid destination
First, implement a basic upload process:
Basic upload process, click the button to select, complete upload
The code is as follows:
<template>
<div class="app-container">
<! -- Use change event -->
<input type="file" @change="handleFileChange">
</div>
</template>
<script lang="ts">
import { defineComponent } from 'vue'
import axios from 'axios'
export default defineComponent({
name: 'App'.setup() {
const handleFileChange = (e: Event) = > {
// Assert to HTMLInputElement
const target = e.target as HTMLInputElement
const files = target.files
if(files) {
const uploadedFile = files[0]
const formData = new FormData()
formData.append('file', uploadedFile)
// Use Node to simulate the upload interface
axios.post('http://localhost:3001/upload', formData, {
headers: {
"Content-Type": 'multipart/form-data'
}
}).then(resp= > {
console.log('resp', resp)
}).catch(error= >{})}}return {
handleFileChange
}
}
})
</script>
<style>
.page-title {
color: #fff;
}
</style>
Copy the code
The results are as follows:
Now that we’ve done the basic upload processing, it’s relatively simple, so let’s createUploader.vue
File is used to encapsulateUpload
Components.
We need to implement the following functionality
Custom templates
- Optimize the style to hide the input
- Click on the
<div class="upload-area" @click="triggerUpload"></div>
Start with JSinput
theclick
The event - Handling upload Status
The code is as follows:
<template>
<div class="file-upload">
<div class="upload-area" @click="triggerUpload"></div>
<span v-if="fileStatus==='loading'">Are uploading</span>
<span v-else-if="fileStatus==='success'">Uploaded successfully</span>
<span v-else-if="fileStatus==='error'">Upload failed</span>
<span v-else>Click on the upload</span>
<input ref="fileInput" type="file" name="file" style="display: none" />
</div></template> <script lang="ts"> import { computed, defineComponent, PropType, reactive, ref } from 'vue' import axios from 'axios' type UploadStatus = 'ready' | 'loading' | 'success' | 'error' export default DefineComponent ({props: {action: {// url address type: String, required: Setup (props) {// input instance object, Through association with ref = "the fileInput" get input instance objects const fileInput = ref < null | HTMLInputElement > (null) const fileStatus = Ref <UploadStatus>('ready') // 1. Div const triggerUpload = () => {if(fileInput.value) {fileinput.value.click ()}} // Use div to trigger input change const handleFileChange = (e: Event) => { const target = e.target as HTMLInputElement const files = target.files if(files) { const uploadedFile = files[0] const formData = new FormData() formData.append('file', UploadedFile) readyfile. status = 'loading' // To load axios.post(props. { "Content-Type": 'multipart/form-data' } }).then(resp=> { console.log('resp', Resp) readyfile. status = 'success' // Upload success set the status to SUCCESS}). Catch (error=> {readyfile. status = 'error' // // }} return {fileInput, triggerUpload, handleFileChange, fileStatus}}) </script>Copy the code
Now that we have optimized the upload component style and handled the upload state, we need to deal with the file upload list
Supports file upload list
To handle the file upload list we need to have the following implementation:
- Show file name
- state
- Can be deleted
- Display upload Progress
- It is possible to have a richer display
Make some changes from the code in the previous step:
<template>
<div class="file-upload">
<div class="upload-area" @click="triggerUpload"></div>
<! -- Click Upload mode -->
<slot v-if="isUploading" name='loading'>
<button disabled>Are uploading</button>
</slot>
<! -- Upload completed state -->
<slot name="uploaded" v-else-if="lastFileData && lastFileData.loaded" :uploadData="lastFileData.data">
<button disabled>Click on the upload</button>
</slot>
<! -- Default state -->
<slot v-else name='default'>
<! <button disabled> </button> -->
<span>Click on the upload</span>
</slot>
<input ref="fileInput" type="file" name="file" style="display: none" />
<! - shows the fileList -- -- >
<ul>
<li
v-for="file in filesList"
:key="file.uid"
:class="`uploaded-file upload-${file.status}`"
>
<span class="filname">
{{ file.name }}
</span>
<button class="delete-icon" @click="reomveFile(file.uid)">Del</button>
</li>
</ul>
</div></template> <script lang="ts"> import { last } from 'lodash-es' import { v4 as uuidv4 } from 'uuid'; / / define upload state type UploadStatus = 'ready' | 'loading' | 'success' | 'error' / / step1 definition to upload the file object interface class export interface UploadFile { uid: string; // File id size: number; // File size name: string; // File name status: UploadStatus; // Upload status raw: File; / / file progress? : string; // File upload progress resp? : any; // Does the server return the data URL? Export default defineComponent({props: {action: {// URL address type: string, required: Setup (props) {// input instance object, Through associated with ref = "the fileInput" get input instance objects const fileInput = ref < null | HTMLInputElement > (null) / / step2 upload file list const filesList = Ref <UploadFile[]>([]) // step4-1 Determine whether a constant isUploading = computed(()=> {return Fileslist.value.some ((file)=>file.status==='loading')}) // Step 4-2 Get the last const lastFileData = computed(()=>{const lastFile = last(filesList.value) if(! lastFile) return false return { loaded: lastFile? .status === 'success', data: lastFile? .resp}}) // 1. Div click event const triggerUpload = () => {if(fileInput.value) {fileinput.value.click ()}} // Div to trigger input change const handleFileChange = (e: Event) => { const target = e.target as HTMLInputElement const files = target.files if(files) { const uploadedFile = File [0] const formData = new formData () formdata.append ('file', uploadedFile) // FileObj = reactive<UploadFile>({uid: uuid(); Size: uploadedfile. size, name: uploadedfile. name, status: 'loading', raw: uploadedFile }) filesList.value.push(fileObj) axios.post(props.actions, formData, { headers: { "Content-Type": 'multipart/form-data'}, //step6: (progressEvent)=> { const complete = (progressEvent.loaded / progressEvent.total * 100 | 0) + '%' fileObj.progress = complete } }).then(resp=> { console.log('resp', Resp) fileobj.status = 'success'}).catch(error=> {fileobj.status = 'error'}).finally(()=> {// A picture cannot be uploaded repeatedly If (fileinput.value) {fileinput.value. Value = ''}})}} const reomveFile = (uid: string) => { filesList.value = filesList.value.filter(file=>file.uid! ==uid) } return { fileInput, triggerUpload, handleFileChange, fileStatus, isUploading, filesList, lastFileData } } }) </script>Copy the code
- First we define the upload file object interface class
UploadFile
- Created a
filesList
Responsive object - FileStatus was removed because in
UploadFile
We already have the definition in the interfacestatus
- Modify the state in the template using
computed
To determine whether it is in the uploaded state and add a slot for customization - Show uploaded pictures
- Handling upload Progress
- Processing to delete
Note: If we select the same image, we will not upload the image, because we use the input change event, so we need to set the value of the input to null after uploading
Supports a series of lifecycle hook events, upload events
beforeUpload
<template> ... </template> <script lang="ts"> import { last } from 'lodash-es' import { v4 as uuidv4 } from 'uuid'; / / define upload state type UploadStatus = 'ready' | 'loading' | 'success' | 'error' / / upload files to a Boolean or promsie definition and accept a File type CheckUpload = ()=> boolean | Promise<File> export interface UploadFile { ... } export default defineComponent({props: {action: {// URL address type: String, required: true}, beforeUpload: {type: Function as PropType<CheckUpload> } }, setup(props) { const fileInput = ref<null | HTMLInputElement>(null) const filesList = ref<UploadFile[]>([]) // step4-1 Check whether you are uploading const isUploading = computed(()=> {return fileslist.value.some ((file)=>file.status==='loading')}) const lastFileData = computed(()=>{ const lastFile = last(filesList.value) if(! lastFile) return false return { loaded: lastFile? .status === 'success', data: lastFile? .resp } }) const triggerUpload = () => { if(fileInput.value) { fileInput.value.click() } } const handleFileChange = (e: Event) => { const target = e.target as HTMLInputElement const files = target.files const uploadedFile = files[0] if(props.beforeUpload) { const result = props.beforeUpload(uploadedFile) if(result && result instanceof Promise) { result.then((processFile)=> { if(uploadedFile instanceof File) { postFile(uploadedFile) } }).catch(error=>{ console.error(error) }) } esle if(result===true) { postFile(uploadedFile) } }else{ postFile(uploadedFile) } } const reomveFile = (uid: string) => { filesList.value = filesList.value.filter(file=>file.uid! Const postFile = (readyFile:UploadFile)=> {const formData = new formData () formdata.append ('file', readyFile.raw) readyFile.status = 'loading' axios.post(props.action, formData, { headers: { "Content-Type": 'multipart/form-data' }, onUploadProgress: (progressEvent) = > {const complete = (progressEvent. The loaded/progressEvent. Total * 100 | 0) + '%'. / / the console log (' upload '+ complete) readyFile.progress = complete } }, ).then(resp=> { console.log('resp', resp) // fileStatus.value = 'success' readyFile.status = 'success' readyFile.resp = resp.data }).catch(error=> { // Filestatus. value = 'error' readyfile.status = 'error'}).finally(()=> {// A picture cannot be uploaded repeatedly bug if(fileInput.value) { fileInput.value.value = '' } }) } return { fileInput, triggerUpload, handleFileChange, fileStatus, isUploading, filesList, lastFileData } } }) </script>Copy the code
Implementation steps:
- Define the property beforeUpload at poops and also define the upload File as Boolean or promsie and accept a File
- Encapsulate the original upload method as postFile
- The rest of the process is processed according to the beforeUpload result returned
OnProgress, onSuccess, onError, onChange are similar
Drag and drop support
The general process is as follows:
- Dragover and dragleave add and remove corresponding classes
- Drop retrieves the file being dragged, deletes the class, and triggers an upload
- Trigger only if drag is true
Note that the specific usage of V-ON can be referred to the code in the official vUE document:
<template>
<div class="file-upload">
<div
:class="['upload-area', drag && isDragOver ? 'is-dragover': '' ]"
v-on="events"
>
<! -- Click Upload mode -->
<slot v-if="isUploading" name='loading'>
<button disabled>Are uploading</button>
</slot>
<! -- Upload completed state -->
<slot name="uploaded" v-else-if="lastFileData && lastFileData.loaded" :uploadData="lastFileData.data">
<button disabled>Click on the upload</button>
</slot>
<! -- Default state -->
<slot v-else name='default'>
<! <button disabled> </button> -->
<span>Click on the upload</span>
</slot>
</div>
<input ref="fileInput" type="file" name="file" style="display: none" @change="handleFileChange" />
<ul>
<li
v-for="file in filesList"
:key="file.uid"
:class="`uploaded-file upload-${file.status}`"
>
<img
v-if="file.url && listType === 'picture'"
class="upload-list-thumbnail"
:src="file.url"
:alt="file.name"
>
<span class="filname">
{{ file.name }}
</span>
<span class="progress">
{{ file.progress }}
</span>
<button class="delete-icon" @click="reomveFile(file.uid)">Del</button>
</li>
</ul>
</div>
</template><script lang="ts"> import { computed, defineComponent, PropType, reactive, ref } from 'vue' import axios from 'axios' import { v4 as uuidv4 } from 'uuid'; import { last } from 'lodash-es' type UploadStatus = 'ready' | 'loading' | 'success' | 'error' type fileListType = 'text' | 'picture' type CheckUpload = ()=> boolean | Promise<File> export interface UploadFile { uid: string; // File id size: number; // File size name: string; // File name status: UploadStatus; // Upload status raw: File; / / file progress? : string; resp? : any; // Does the server return the data URL? : String // corresponding to the displayed URL} // dragOver triggered when the file is dragged to the region // dragLeave file leaves the drag region // Drop event gets the file being dragged to delete class and triggers upload // only valid if drag is set Export Default defineComponent({name: 'Uploader', props: {action: {// URL address type: String, required: Type: Function as PropType<CheckUpload>}, drag: {// Whether to drag type: Boolean, default: False}, autoUpload: {type: Boolean, default: true}, listType: {type: String as PropType<fileListType>, default: 'text' } }, setup(props) { const fileInput = ref<null | HTMLInputElement>(null) const fileStatus = ref<UploadStatus>('ready') // Const filesList = ref<UploadFile[]>([]) // Upload-area display const isDragOver = ref< Boolean >(false) const triggerUpload = () => {if(fileInput.value) { fileInput.value.click() } } let events: {[key:string]: (e: any)=>void} = { 'click': triggerUpload, } // only one file in the loading state isUploading const isUploading = computed(()=>{return fileslist.value.some ((file)=> File-status ==='loading')}) // Get the last item of the uploaded file const lastFileData = computed(()=>{const lastFile = last(fileslist.value) if(! lastFile) return false return { loaded: lastFile? .status === 'success', data: lastFile? .resp}}) // Handle dragover, dragleave const handleDrag = (e: DragEvent,over: Boolean)=> {// prevent default event e.preventDefault() // dragover to true, Value = over} const handleDrop = (e: DragEvent) => { e.preventDefault() isDragOver.value = false if(e.dataTransfer) { beforeUploadCheck(e.dataTransfer.files) } } if(props.drag){ events = { ... events, 'dragover': (e: DragEvent) => { handleDrag(e, true)}, 'dragleave': (e: DragEvent) => {handleDrag(e, false)}, 'drop': handleDrop} // console.log(events)} const reomveFile = (uid: uid) string)=> { filesList.value = filesList.value.filter(file=>file.uid! ==uid) } const postFile = (readyFile:UploadFile) => { const formData = new FormData() formData.append('file', Readyfile. raw) readyfile. status = 'loading' // After selecting a file and pushing it into a storage object { "Content-Type": 'multipart/form-data' }, onUploadProgress: (progressEvent) = > {const complete = (progressEvent. The loaded/progressEvent. Total * 100 | 0) + '%'. / / the console log (' upload '+ complete) readyFile.progress = complete } }, ).then(resp=> { console.log('resp', resp) // fileStatus.value = 'success' readyFile.status = 'success' readyFile.resp = resp.data }).catch(error=> { // Filestatus. value = 'error' readyfile.status = 'error'}).finally(()=> {// A picture cannot be uploaded repeatedly bug if(fileInput.value) { fileInput.value.value = '' } }) } const addFileToList = (uploadedFile: File) => { const fileObj = reactive<UploadFile>({ uid: uuidv4(), size: uploadedFile.size, name: uploadedFile.name, status: 'ready', raw: UploadedFile}) // Handle image format and show if(props. ListType ==='picture') {// try {// fileObj URL.createObjectURL(uploadedFile) // }catch(err) { // console.error('upload file err', err) // } const fileReader = new FileReader() fileReader.readAsDataURL(uploadedFile) fileReader.addEventListener('load', ()=> { fileObj.url = fileReader.result as string }) } filesList.value.push(fileObj) if(props.autoUpload) { postFile(fileObj) } } const uploadFiles = ()=> { // filesList.value.filter(file => file.status === 'ready').forEach(readyFile => postFile(readyFile)) filesList.value.filter(file => file.status === 'ready').forEach(readyFile=>postFile(readyFile)) } const beforeUploadCheck = (files: null | FileList ) => { if(files) { fileStatus.value = 'loading' const uploadedFile = files[0] if(props.beofreUpload) { Const result = props. BeofreUpload () // if(result && result instanceof Promise) {result.then(processedFile=> { if(processedFile instanceof File) { addFileToList(processedFile) } else { throw new Error('beforeUpload promise should return file object') } }).catch(err=> { console.log(err) }) } else if(result === true) { addFileToList(uploadedFile) } } else { addFileToList(uploadedFile) } } } const handleFileChange = (e: Event) => { const target = e.target as HTMLInputElement const files = target.files beforeUploadCheck(files) } return { fileInput, triggerUpload, handleFileChange, isUploading, filesList, reomveFile, lastFileData, beforeUploadCheck, isDragOver, events } } }) </script>Copy the code
The above is the basic upload general components based on VUe3 implementation with the upload interface code attached:
// Import modules
const Koa = require('koa');
const fs = require('fs');
const path = require('path');
const router = require('koa-router') ();const koaBody = require('koa-body');
const static = require('koa-static');
const cors = require('koa2-cors')
/ / instantiate
const app = new Koa();
app.use(koaBody({
multipart: true.// Support file upload
formidable: {
maxFieldsSize: 2 * 1024 * 1024.// The maximum file size is 2 MB
multipart: true // Whether multipart-formdate forms are supported}}));const uploadUrl = "http://localhost:3001/static/upload";
// Configure the route
router.get('/'.(ctx) = > {
// Set the header type. If not, the page will be downloaded directly
ctx.type = 'html';
// Read the file
const pathUrl = path.join(__dirname, '/static/upload.html');
ctx.body = fs.createReadStream(pathUrl);
});
// Upload the file
router.post('/upload'.(ctx) = > {
// Get the uploaded file
const file = ctx.request.files.file;
console.log(file);
// Read the file stream
const fileReader = fs.createReadStream(file.path);
// Set the file saving path
const filePath = path.join(__dirname, '/static/upload/');
// Assemble into absolute paths
const fileResource = filePath + ` /${file.name}`;
/** * Use createWriteStream to write data, and then use pipe to concatenate */
const writeStream = fs.createWriteStream(fileResource);
// determine if the /static/upload folder exists and create one if it does not
if(! fs.existsSync(filePath)) { fs.mkdir(filePath,(err) = > {
if (err) {
throw new Error(err);
} else {
fileReader.pipe(writeStream);
ctx.body = {
url: uploadUrl + ` /${file.name}`.code: 0.message: 'Upload successful 1'}; }}); }else {
fileReader.pipe(writeStream);
ctx.body = {
url: uploadUrl + ` /${file.name}`.code: 0.message: 'Upload successful 1'}; }});// Configure the static resource path
app.use(static(path.join(__dirname)));
// Enable cross-domain
app.use(cors())
// Start the route
app.use(router.routes()).use(router.allowedMethods());
// Listen on the port number
app.listen(3001.() = > {
console.log('server is listen in 3001');
});
Copy the code
Write in the last
This is just a simple implementation, of course we can also add custom headers, custom transfer data, accecpt, etc