這次我要講述的是在React-Flask框架上開發(fā)上傳組件的技巧。我目前主要以React開發(fā)前端,在這個(gè)過程中認(rèn)識(shí)到了許多有趣的前端UI框架——React-Bootstrap、Ant Design、Material UI、Bulma等。而比較流行的上傳組件也不少,而目前用戶比較多的是jQuery-File-Upload和Dropzone,而成長速度快的新晉有Uppy和filepond。比較惋惜的是Fine-Uploader的作者自2018年后就決定不再維護(hù)了,原因作為后來者的我就不多過問了,但請(qǐng)各位尊重每一位開源作者的勞動(dòng)成果。
這里我選擇React-Dropzone,原因如下:
通過yarn將react-dropzone和引入:
yarn add react-dropzone axios
前端js如下(如有缺失,請(qǐng)自行修改):
import React, { useState, useCallback, useEffect, } from 'react'; import {useDropzone} from 'react-dropzone'; import "./dropzone.styles.css" import InfiniteScroll from 'react-infinite-scroller'; import { List, message, // Avatar, Spin, } from 'antd'; import axios from 'axios'; /** * 計(jì)算文件大小 * @param {*} bytes * @param {*} decimals * @returns */ function formatBytes(bytes, decimals = 2) { if (bytes === 0) return '0 Bytes'; const k = 1024; const dm = decimals 0 ? 0 : decimals; const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']; const i = Math.floor(Math.log(bytes) / Math.log(k)); return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]; } /** * Dropzone 上傳文件 * @param {*} props * @returns */ function DropzoneUpload(props) { const [files, setFiles] = useState([]) const [loading, setLoading] = useState(false); const [hasMore, setHasMore] = useState(true); const onDrop = useCallback(acceptedFiles => { setLoading(true); const formData = new FormData(); smallFiles.forEach(file => { formData.append("files", file); }); axios({ method: 'POST', url: '/api/files/multiplefiles', data: formData, headers: { "Content-Type": "multipart/form-data", } }) then(resp => { addFiles(acceptedFiles); setLoading(false); }); }, [files]); // Dropzone setting const { getRootProps, getInputProps } = useDropzone({ multiple:true, onDrop, }); // 刪除附件 const removeFile = file => { const newFiles = [...files] newFiles.splice(newFiles.indexOf(file), 1) setFiles(newFiles) } useEffect(() => { // init uploader files setFiles([]) },[]) return ( section className="container"> div {...getRootProps({className: 'dropzone'})}> input {...getInputProps()} /> p>拖動(dòng)文件或點(diǎn)擊選擇文件😊/p> /div> div className="demo-infinite-container"> InfiniteScroll initialLoad={false} pageStart={0} loadMore={handleInfiniteOnLoad} hasMore={!loading hasMore} useWindow= {false} > List dataSource={files} renderItem={item=> ( List.Item actions={[ // a key="list-loadmore-edit">編輯/a>, a key="list-loadmore-delete" onClick={removeFile}>刪除/a> ]} // extra={ // } key={item.path}> List.Item.Meta avatar={ > { !!item.type ['image/gif', 'image/jpeg', 'image/png'].includes(item.type) img width={100} alt='logo' src={item.preview} /> } /> } title={item.path} description={formatBytes(item.size)} /> /List.Item> )} > {loading hasMore ( div className="demo-loading-container"> Spin /> /div> )} /List> /InfiniteScroll> /div> /section> ); }
flask代碼:
def multiplefiles(): if 'files' not in request.files: return jsonify({'message': '沒有文件!'}), 200 files = request.files.getlist('files') for file in files: if file: # 通過拼音解決secure_filename中文問題 filename = secure_filename(''.join(lazy_pinyin(file.filename)) Path(UPLOAD_FOLDER + '/' + file_info['dir_path']).mkdir(parents=True, exist_ok=True) file.save(os.path.join(UPLOAD_FOLDER + '/' + file_info['dir_path'], filename)) return jsonify({'message': '保存成功!!'})
通過file.slice()方法生成文件的chunks。不要用Promise.all容易產(chǎn)生非順序型的請(qǐng)求,導(dǎo)致文件損壞。
js代碼:
const promiseArray = largeFiles.map(file => new Promise((resolve, reject) => { const chunkSize = CHUNK_SIZE; const chunks = Math.ceil(file.size / chunkSize); let chunk = 0; let chunkArray = new Array(); while (chunk = chunks) { let offset = chunk * chunkSize; let slice = file.slice(offset, offset+chunkSize) chunkArray.push([slice, offset]) ++chunk; } const chunkUploadPromises = (slice, offset) => { const largeFileData = new FormData(); largeFileData.append('largeFileData', slice) return new Promise((resolve, reject) => { axios({ method: 'POST', url: '/api/files/largefile', data: largeFileData, headers: { "Content-Type": "multipart/form-data" } }) .then(resp => { console.log(resp); resolve(resp); }) .catch(err => { reject(err); }) }) }; chunkArray.reduce( (previousPromise, [nextChunk, nextOffset]) => { return previousPromise.then(() => { return chunkUploadPromises(nextChunk, nextOffset); }); }, Promise.resolve()); resolve(); }))
flask代碼:
filename = secure_filename(''.join(lazy_pinyin(filename))) Path(UPLOAD_FOLDER + '/' + file_info['dir_path']).mkdir(parents=True, exist_ok=True) save_path = os.path.join(UPLOAD_FOLDER + '/' + file_info['dir_path'], filename) # rm file if exists if offset == 0 and save_path.exists(filename): os.remove(filename) try: with open(save_path, 'ab') as f: f.seek(offset) f.write(file.stream.read()) print("time: "+ str(datetime.now())+" offset: " + str(offset)) except OSError: return jsonify({'Could not write to file'}), 500
文件傳輸一直都是HTTP的痛點(diǎn),尤其是大文件傳輸。最好的方式是自己做個(gè)Client,通過FTP和FTPS的協(xié)議進(jìn)行傳輸。第二種來自于大廠很中心化的方法,通過文件的checksum來確定文件是否已經(jīng)上傳了,來營造秒傳的效果。第三種來自去中心化的Bittorrent的方法每一個(gè)用戶做文件種子,提供文件傳輸?shù)妮o助,目前國內(nèi)并沒有普及使用。
到此這篇關(guān)于Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼的文章就介紹到這了,更多相關(guān)Python React-Dropzone上傳組件內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
標(biāo)簽:漯河 林芝 寧夏 大同 盤錦 南平 普洱 海南
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Python基于React-Dropzone實(shí)現(xiàn)上傳組件的示例代碼》,本文關(guān)鍵詞 Python,基于,React-Dropzone,實(shí)現(xiàn),;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。