• [python][flask] Flask 图片上传与下载例子(支持漂亮的拖拽上传)




    1、效果预览

    我们基于 Flask 官方指导工程,增加一个图片拖拽上传功能,效果如下:


    2、新增逻辑概览

    我们在官方指导工程 https://github.com/pallets/flask/tree/2.1.1/examples/tutorial/flaskr 上进行增加代码,改动如下:

    ➜  flaskr git:(main) ✗ tree
    .
    ├── static
    │   ├── file
    │   │   ├── css
    │   │   │   └── upload.css           <-  增加图片上传的 CSS
    │   │   ├── img
    │   │   │   ├── 20220525004341_22.png
    │   │   │   └── 20220529231518_76.png
    │   │   └── js
    │   │       └── upload.js            <- 增加图片上传的 JS
    │   └── style.css
    ├── templates
    │   ├── auth
    │   │   ├── login.html
    │   │   └── register.html
    │   ├── base.html
    │   ├── blog
    │   │   ├── create.html
    │   │   ├── index.html
    │   │   └── update.html
    │   └── tuchuang                     <- 增加图片上传的 html
    │       └── upload.html
    ├── auth.py
    ├── blog.py
    ├── db.py
    ├── __init__.py
    ├── schema.sql
    └── tuchuang.py                      <- 增加图床 python 蓝图
    
    9 directories, 18 files
    

    由于 flask 官方 Demo 基于蓝图设计,这给我们新增逻辑带来了很大的方便。关于官方 Demo 的介绍,可以参考我的《Flask 入门(以一个博客后台为例)


    3、tuchuang.py 逻辑介绍

    3.1 图片上传

    1)该接口采用 POST 方法,需要登录;
    2)接着,检查请求中是否有 'file' 关键词,然后取出文件,判断文件是否为空或是否合法;
    3)最后,将上传的图片保存(采用秒级别的时间戳+随机数重命名);
    4)该接口在上传图片成功后,返回该图片的链接;如果不成功,返回 upload.html 页面;

    @bp.route('/', methods=['GET', 'POST'])
    @login_required
    def upload_file():
        if request.method == 'POST':
            # check if the post request has the file part
            if 'file' not in request.files:
                flash('No file part')
                return redirect(request.url)
            file = request.files['file']
            # If the user does not select a file, the browser submits an
            # empty file without a filename.
            if file.filename == '':
                flash('No selected file')
                return redirect(request.url)
            if file and allowed_file(file.filename):
                # 获取安全的文件名 正常文件名
                filename = secure_filename(file.filename)
                
                # 生成随机数
                random_num = random.randint(0, 100)
                # f.filename.rsplit('.', 1)[1] 获取文件的后缀
                filename = datetime.now().strftime("%Y%m%d%H%M%S") + "_" + str(random_num) + "." + filename.rsplit('.', 1)[1]
                file_path = app.config['UPLOAD_FOLDER']    # basedir 代表获取当前位置的绝对路径
                
                # 如果文件夹不存在,就创建文件夹
                if not os.path.exists(file_path):
    	            os.makedirs(file_path)
    	
                file.save(os.path.join(file_path, filename))
                return redirect(url_for('tuchuang.download_file', name=filename))
        return render_template("tuchuang/upload.html")
    

    3.2 图片合法检查

    上述代码中有一个合法检测的函数 allowed_file,用于检查上传图片的后缀是否在允许列表:

    basedir = os.path.abspath(os.path.dirname(__file__))                 # 获取当前文件所在目录
    UPLOAD_FOLDER = basedir+'/static/file/img'                           # 计算图片文件存放目录
    ALLOWED_EXTENSIONS = {'txt', 'pdf', 'png', 'jpg', 'jpeg', 'gif'}     # 设置可上传图片后缀 
    
    app = Flask(__name__)
    app.config['UPLOAD_FOLDER'] = UPLOAD_FOLDER
    bp = Blueprint("tuchuang", __name__, url_prefix="/tuchuang")
    
    def allowed_file(filename):                                          # 检查上传图片是否在可上传图片允许列表
        return '.' in filename and \
               filename.rsplit('.', 1)[1].lower() in ALLOWED_EXTENSIONS
    

    3.3 图片下载

    图片下载比较简单,就是调用 send_from_directory 函数,就能够把 static 目录下的对应文件发出:(我们一般把各种用于外面访问的静态图片、JS、CSS 等放在 static 文件中)

    @bp.route('/download/<name>')
    def download_file(name):
        return send_from_directory(app.config["UPLOAD_FOLDER"], name)
    

    4、__init__.py 逻辑介绍

    由于我们采用蓝图设计,因此需要稍微修改下 __init__.py 文件,来将 tuchuang.py 加入:

    • MAX_CONTENT_LENGTH=16 * 1000 * 1000 上传图片大小限制
    • from flaskr import auth, blog, tuchuang
    • app.register_blueprint(tuchuang.bp) 将 tuchuang 加入蓝图
    • app.add_url_rule("/download/<name>", endpoint="download_file", build_only=True)

    5、upload.html 介绍

    5.1 upload Jinja 模板介绍

    • Jinja 引用外部 css:<link rel="stylesheet" href="{{ url_for('static', filename='file/css/upload.css') }}">
    • Jinja 引用外部 js:<script type="text/javascript" src="{{ url_for('static', filename='file/js/upload.js') }}"></script>
    • 该 Jinja 模板实现了两种图片上传交互:
      • 普通版,采用 file select 框 + submit 按钮,实现图片上传:
        <form method=post enctype=multipart/form-data>
            <input type=file name=file>
            <input type=submit value=Upload>
        </form>
        
      • 拖拽版(需要借助 JS,CSS),在 <div id="drop-area"> 内实现

    下面是 tuchuang/upload.html 完整代码:

    <!doctype html>
    <link rel="stylesheet" href="{{ url_for('static', filename='file/css/upload.css') }}">
    <script type="text/javascript"  src="{{ url_for('static', filename='file/js/upload.js') }}"></script>
    <title>Upload new File</title>
    <h1>Upload new File</h1>
    <form method=post enctype=multipart/form-data>
        <input type=file name=file>
        <input type=submit value=Upload>
    </form>
    <div id="drop-area">
        <form class="my-form">
            <p>Upload multiple files with the file dialog or by dragging and dropping images onto the dashed region</p>
            <input type="file" id="fileElem" multiple accept="image/*" onchange="handleFiles(this.files)">
            <label class="button" for="fileElem">Select some files</label>
            <div id="gallery"></div>
            <progress id="progress-bar" max=100 value=0></progress>
        </form>
    </div>
    

    5.2 upload css 介绍(虚线框)

    下面是拖拽需要用到的 CSS,大家暂时浏览下,之后结合 JS 就明白了:

    ➜  css git:(main) ✗ cat upload.css            
    #drop-area {
        border: 2px dashed #ccc;
        border-radius: 20px;
        width: 480px;
        font-family: sans-serif;
        margin: 100px auto;
        padding: 20px;
    }
    #drop-area.highlight {
        border-color: purple;
    }
    p {
        margin-top: 0;
    }
    .my-form {
        margin-bottom: 10px;
    }
    #gallery {
        margin-top: 10px;
    }
    #gallery img {
        width: 150px;
        margin-bottom: 10px;
        margin-right: 10px;
        vertical-align: middle;
    }
    .button {
        display: inline-block;
        padding: 10px;
        background: #ccc;
        cursor: pointer;
        border-radius: 5px;
        border: 1px solid #ccc;
    }
    .button:hover {
        background: #ddd;
    }
    #fileElem {
        display: none;
    }
    

    5.3 upload js 介绍(拖拽)

    5.3.1 JS 拖拽框架

    JS 代码主要基于 window.onload + 拖拽事件实现,大致框架如下:

    ➜  js git:(main) ✗ cat upload.js 
    window.onload=function(){
        var dropArea = document.getElementById('drop-area')
    
        // 阻止默认行为
        ;['dragenter', 'dragover', 'dragleave', 'drop'].forEach(eventName => {
            dropArea.addEventListener(eventName, preventDefaults, false)
        })
    
        function preventDefaults (e) {
            e.preventDefault()
            e.stopPropagation()
        }
    
        // 增加事件,鼠标拖入边框高亮,拖出边框变为原来样子
        ;['dragenter', 'dragover'].forEach(eventName => {
            dropArea.addEventListener(eventName, highlight, false)
        })
    
        ;['dragleave', 'drop'].forEach(eventName => {
            dropArea.addEventListener(eventName, unhighlight, false)
        })
    
        function highlight(e) {
            dropArea.classList.add('highlight')
        }
    
        function unhighlight(e) {
            dropArea.classList.remove('highlight')
        }
    
        // 增加事件,鼠标放下,之后准备上传图片
        dropArea.addEventListener('drop', handleDrop, false)
    
        function handleDrop(e) {
        	// 之后准备上传图片
        }
    }
    

    window.onload() 方法用于在网页加载完毕后立刻执行的操作,即当 HTML 文档加载完毕后,立刻执行某个方法。

    为什么使用 window.onload()?

    因为 JavaScript 中的函数方法需要在 HTML 文档渲染完成后才可以使用,如果没有渲染完成,此时的 DOM 树是不完整的,这样在调用一些 JavaScript 代码时就可能报出"undefined"错误。


    5.3.2 JS 图片上传
    function handleDrop(e) {
    	// 从拖拽放下事件中获取拖拽的文件
        let dt = e.dataTransfer
        let files = dt.files
    
    	// 调用图片处理函数,对图片进行处理
        handleFiles(files)
    }
    
    function handleFiles(files) {
    	// 对于多个图片,循环调用 uploadFile 函数,进行上传
        ([...files]).forEach(uploadFile)
    }
    
    function uploadFile(file) {
    	// JS 合成表单,利用 POST 方法,实现上传(部署在远端时,要改下下面的 url)
        let url = 'http://127.0.0.1:5000/tuchuang/'
        let formData = new FormData()
    
        formData.append('file', file)
    
        fetch(url, {
            method: 'POST',
            body: formData
        })
            .then(progressDone) // <- Add `progressDone` call here
            .catch(() => { /* Error. Inform the user */ })
    }
    

    Fetch API 提供了一个 JavaScript接口,用于访问和操纵HTTP管道的部分,例如请求和响应。它还提供了一个全局 fetch()方法,该方法提供了一种简单,合理的方式来跨网络异步获取资源。详细介绍参考《参考链接[8]》:

    • 1.进行 fetch 请求 参考;
    • 2.支持的请求参数参考;
    • 3.发送带凭据的请求参考;
    • 4.上传 JSON 数据参考;
    • 5.上传文件参考;
    • 6.上传多个文件参考;
    • 7.检测请求是否成功参考;
    • 8.自定义请求对象参考;
    • 9.Headers参考;
    • 10.Guard参考;
    • 11.Response 对象参考;
    • 12.Body参考;
    • 13.特性检测参考;

    该文章讲的比较好,大家可以跳转过去学习下~


    5.3.3 JS 图片上传进度条

    想要带有进度条,我们需要修改下 handleFiles 函数:

    var filesDone = 0
    var filesToDo = 0
    var progressBar = document.getElementById('progress-bar')
    
    ...
    
    // 预览
    function previewFile(file) {
        let reader = new FileReader()
        reader.readAsDataURL(file)
        reader.onloadend = function() {
            let img = document.createElement('img')
            img.src = reader.result
            document.getElementById('gallery').appendChild(img)
        }
    }
    
    // 进度条初始化,fileDone 置 0,filesToDo 置需要上传图片总数
    function initializeProgress(numfiles) {
        progressBar.value = 0
        filesDone = 0
        filesToDo = numfiles
    }
    
    // 注意,该函作为 fetch 的返回回调函数,意思是每次传输完成一个图片,进度条进行相应变化
    function progressDone() {
        filesDone++
        progressBar.value = filesDone / filesToDo * 100
    }
    
    function handleFiles(files) {
        files = [...files]
        initializeProgress(files.length) 
        files.forEach(uploadFile)
        files.forEach(previewFile)
    }
    

    6、后记

    本文涉及到的源代码在 GITHUB,后续我会基于该工程加入各种有意思的功能。
    此外,之前的两篇文章列在下面,可能对您理解本文有帮助:


    参考链接

    [1]. 本文代码 GITHUB
    [2]. 在HTML中引入CSS的几种方式介绍
    [3]. python Flask中html模版中如何引用css,js等资源
    [4]. HTML引入JS的两种方法
    [5]. 使用Flask引用HTML中的.js文件的静态资源问题
    [6]. Flask 官方指导 Uploading Files
    [7]. JavaScript window.onload
    [8]. JavaScript使用 Fetch
    [9]. 本文 JS+CSS 参考



    : 这篇是在大家熟悉 flaskr 的指导项目之后,实现一个图片上传和下载的案例...

    如果觉得不错,帮忙点个支持哈~

  • 相关阅读:
    Azure KeyVault(四)另类在 .NET Core 上操作 Secrets 的类库方法-----Azure.Security.KeyVault.Secrets
    C语言判断闰年(ZZULIOJ1028: I love 闰年!)
    MNN编译
    JAVA 中的代码生成包 CGLIB (Code Generation Library)
    你想要的快捷回复短语
    Node.js 入门教程 23 使用 npm 的语义版本控制 & 24 卸载 npm 软件包 & 25 npm 全局或本地的软件包
    11、Vue的生命周期
    基于PHP+MySQL的企业宣传展示网
    LQR 控制器
    突破卡脖子技术 AVS3标准在世界杯实现移动端规模化商用
  • 原文地址:https://www.cnblogs.com/zjutlitao/p/16325464.html