flask+ajax 实现同时删除文件列表和服务器上的文件

问题背景

        在使用flask+dropzone开发多文件上传功能时,发现dropzone控件只能删除列表里的文件,而不能删除已经上传到服务器的文件。因此想要通过修改js代码解决该问题。

        观察一下addRemoveLinks的代码,是通过js在页面添加了一个指向“javasccript:undefined”的超链接(即不做任何处理),然后通过监听removedFile事件将列表里的文件移除。

if (this.options.addRemoveLinks) {
   file._removeLink = Dropzone.createElement("<a class=\"dz-remove\" href=\"javascript:undefined;\" data-dz-remove>" +this.options.dictRemoveFile + "</a>");
   file.previewElement.appendChild(file._removeLink);
}

        一个比较朴素的想法是,修改href="/remove"指向flask的路由,并且在对应函数中进行文件删除:

if (this.options.addRemoveLinks) {
   file._removeLink = Dropzone.createElement("<a class=\"dz-remove\" href=\"/remove\" data-dz-remove>" +this.options.dictRemoveFile + "</a>");
   file.previewElement.appendChild(file._removeLink);
}
from flask import request, render_template
from config import UPLOAD_FOLDERS #这是我自己配置的存储路径
import os
@app.route('/remove')
def remove():
    _file = request.files.get('file')
    pth = os.path.join(UPLOAD_FOLDERS,_file)
    if os.path.exists(pth):
        os.remove(pth)
    return render_template('upload.html')

        确实可以实现删除,但render_template函数会重新渲染upload页面,导致列表里的其他文件也不再显示:

        这不符合用户使用需求。

解决方案

        考虑使用flask+ajax实现页面的局部更新。

        照例先观察dropzone.js里关于removedfile的源代码:

removedfile: function(file) {
    var _ref;
    if (file.previewElement) {
      if ((_ref = file.previewElement) != null) {
            _ref.parentNode.removeChild(file.previewElement);
    }
   }
   return this._updateMaxFilesReachedClass();
}

        事件监听函数:

this.on("removedfile", (function(_this) {
      return function() {
          return _this.updateTotalUploadProgress();
      };

        一个非常简单的解决方案是在removedfile函数里添加ajax代码段:

removedfile: function(file) {
    var _ref;
    $.ajax({
        type:'POST',
        url:'remove',
        data:{'filename':file.name},
        success:function(){
            /* success responese */        
        },
        error:function(){
            /* error responese */          
        }    
    });    
    if (file.previewElement) {
      if ((_ref = file.previewElement) != null) {
            _ref.parentNode.removeChild(file.previewElement);
    }
   }
   return this._updateMaxFilesReachedClass();
}

        然后修改flask服务器上对应的后台处理代码:

@app.route('/remove', methods=['POST'])
def remove():
    _file = request.form.get('filename')
    pth = os.path.join(UPLOAD_FOLDERS,_file)
    if os.path.exists(pth):
        os.remove(pth)
        return '',200
    return 'File %s not found' % _file, 404

            需要注意的是:1.这里文件名的获取方式不再是通过request.files而是request.form; 2.ajax使用post提交时,flask函数里必须设置methods=['POST'],否则会报错405(不允许的操作);3. 返回信息应当进行进一步处理(比如jsonfy),我这里偷个懒,简单地写个框架出来。

        之后运行就可以得到目标效果啦!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值