bootstrap 为 flask 使用人员提供了一个非常优美且有效的前端页面组件,但是完美之处还存在些许缺陷,比如文件的上传功能.而 bootstrap-fileinput 是基于 bootstrap 的控件,非常完美的填补了这个空缺.注意: 本文是基于 bootstrap-fileinput v4.4.2. github 地址: https://github.com/kartik-v/bootstrap-fileinput注意: 本文是主要是以 http://plugins.krajee.com/file-input/demo 示例为基础进行讲解.

创建蓝图 basic

创建方法请参照 flask 项目中使用 bootstrapFileInput(构建篇) 中 lib 蓝图的创建方法,此处不在赘述.

构建基础 html 模板

app/basic/templates/basic_common/base.html 内容如下:

{% block title %}{% endblock %}

{% block css %}

{% endblock %}

{% block js %}

{% endblock %}

{% block content %}

{% endblock %}

base.html 模板引入 css 和 js 时的几个坑

注意 css 和 js 文件的导入顺序

  • 首先需要导入的 js 文件是 jquery.js.
  • 第二需要导入 bootstrap 相关的 css 和 js.
  • 第三需要导入 fileinput 相关的 css 和 js, 请注意项目中的注释, 相关的文件导入也需要有先后顺序的要求.

注意版本问题

  • 此项目所需的 jquery 是 jQuery v2.1.1.
  • 此项目所需的 bootstrap 是 v3.3.7 版本
  • 此项目所需的 fileinput 是 v4.4.2 的版本.

其它版本可能会有所不同.

注意 fileinput 使用模式

fileinput 有两种使用模式,一种是利用 form 提交,一种是 ajax 方式提交.其中 ajax 提交方式,需要从 js 中进行设置, 并将类样式 class 设置为 file-loading. 而非 ajax 提交方式需要引入 form 表单, 类样式 class 需设置为 file, 本基础示例都需要引入 form 表单.

基础示例 1

模板内容

app/basic/templates/example_1.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例1 -- 自动展示缩略图

Select File

{% endblock %}

{% block title %}

基本示例1 -- 自动展示缩略图

{% endblock %}

知识点:

  1. 需要引入 form 表单, 并支持文件上传, 需设置 enctype="multipart/form-data".
  2. 由于 flask 项目是以 html 标签的 name 属性进行选择元素, 该 input 标签中需要设置 name 属性.
  3. class 需要设置为 'file'.
  4. input 标签的 type 属性要设置为 file, 以便支持文件上传.
  5. input 标签由于没有引入 multiple 属性, 故不能实现选择多文件功能.

视图函数

app/basic/views.py 内容如下:

# -*- coding:utf-8 -*-

__author__ = '东方鹗'

from flask import render_template, request, current_app, redirect, url_for

from . import basic

from werkzeug.utils import secure_filename

import os

def allowed_file(filename):

ALLOWED_EXTENSIONS = set(['txt', 'pdf', 'png', 'jpg', 'jpeg', 'gif'])

return '.' in filename and

filename.rsplit('.', 1)[1] in ALLOWED_EXTENSIONS

@basic.route('/example_1', methods=['GET', 'POST'])

def example_1():

if request.method == 'POST':

file = request.files['input-1']

if file and allowed_file(file.filename):

filename = secure_filename(file.filename)

file.save(os.path.join(current_app.config['UPLOAD_FOLDER'], filename))

return render_template('example_1.html')

知识点:

  1. allowed_file 函数是为了检查所上传的文件的格式, 主要实现方法是通过文件后缀来判断文件的格式. 参考文档:http://docs.jinkan.org/docs/flask/patterns/fileuploads.html
  2. secure_filename 函数是为了让用户上传的文件的文件名更安全. 参考文档: http://werkzeug.pocoo.org/docs/0.12/utils/#werkzeug.utils.secure_filename
  3. request.files['input-1'] 是为了获得上传的文件的对象. 此处的 'input-1' 对应的是 html 模板中 input 标签的 name 属性. 获取多个上传的文件需要用到 request.files.getlist('name属性标签'), 可得到一个文件对象的列表.
  4. 上传的路径是在 config.py 文件中设置的 UPLOAD_FOLDER 变量.

展示

fcac0b7cd2aecd540b95e6d6e37918ab.png

基础示例 2

模板内容

app/basic/templates/example_2.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例2 -- 隐藏展示缩略图

Select File

{% endblock %}

{% block title %}

基本示例2 -- 隐藏展示缩略图

{% endblock %}

知识点:

  1. data-show-preview="false" 属性将关闭文件选择后预览的功能, 既缩略图功能.
  2. 实现打开/关闭文件选择后的预览功能也可以在js里进行设置:

$(document).on('ready', function() {

$("#input-2").fileinput({

show-preview: false

});

});

视图函数

views.py 视图函数和示例1基本相同,不在赘述.

展示

45c33bb3b1fe651cac7235c8442f8bf8.png

基础示例 3

模板内容

app/basic/templates/example_3.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例3 -- 利用 file input 属性控制相关选项,如本例可实现多文件上传,但不显示上传按钮

Select File

{% endblock %}

{% block title %}

基本示例3 -- 利用 file input 属性控制相关选项,如本例可实现多文件上传,但不显示上传按钮

{% endblock %}

知识点:

  1. input 标签里有 multiple 属性, 可实现多文件选择的功能.
  2. data-show-upload="false" 属性将不显示上传按钮, 默认为 true.
  3. data-show-caption="true" 属性将显示选择文件的输入框, 默认为 true. 框内将显示将要选择的文件的名称.
  4. 同样可以在 js 里实现相关功能.

$(document).on('ready', function() {

$("#input-3").fileinput({

show-upload: false,

show-caption: true

});

});

视图函数

由于没有上传按钮, 故无需处理选择的文件. 视图函数无需改变.

展示

5b69a3389e259393c096ae2c8ce747fa.png

基础示例 4

模板内容

app/basic/templates/example_4.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例4 -- 设置属性为只读或不可用

Select File

Readonly Input

Disabled Input

{% endblock %}

{% block title %}

基本示例4 -- 设置属性为只读或不可用

{% endblock %}

知识点:

  1. 利用 readonly 或 disabled 属性, 将选择框设置为只读或不可用.

视图函数

由于无法选择文件, 视图函数无需改变.

展示

bf2b61b0282c08a9e022ba03a3643239.png

基础示例 5

模板内容

app/basic/templates/example_5.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例5 -- 设置为单按钮并隐藏文件选择输入框,在上传时显示上传等待图标

Select File

{% endblock %}

{% block title %}

基本示例5 -- 设置为单按钮并隐藏文件选择输入框,在上传时显示上传等待图标

{% endblock %}

知识点:

  1. 通过 js 设置 showCaption 的属性为 false. 你也可以使用 data-show-caption 标签属性来设置.
  2. 注意样式类 class 此处成了 file-loading. 这是一个很积极的变化, 表明你可以使用 ajax 提交方式来上传文件啦. 你把 html 中的 form 表单去掉试试.

视图函数

app/basic/views.py 内容如下:

...

@basic.route('/example_5', methods=['GET', 'POST'])

def example_5():

if request.method == 'POST':

files = request.files.getlist('input-5[]')

for file in files:

if file and allowed_file(file.filename):

filename = secure_filename(file.filename)

file.save(os.path.join(current_app.config['UPLOAD_FOLDER'], filename))

return render_template('example_5.html')

...

知识点

  1. 使用 request.files.getlist() 来获取上传文件的对象列表.

展示

489ea6b70147422bdfad28f24edb3d4c.png

基础示例 6

模板内容

app/basic/templates/example_6.html 内容如下:

{% extends 'basic_common/base.html' %}

{% block content %}

基本示例6 -- 设置最大上传的文件数为10

Select File

{% endblock %}

{% block title %}

基本示例6 -- 设置最大上传的文件数为10

{% endblock %}

知识点:

  1. 通过 js 设置 showUpload 的属性为 false. 你也可以使用 data-show-upload 标签属性来设置.
  2. 设置 maxFileCount 的属性为 10, 表示最多可以一次性上传 10 个文件.
  3. 设置 mainClass 的 属性为 "input-group-lg
Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐