基于 oracle 的 flask 项目(五)——报表下载

在国企,很多领导是按部就班的晋升上来的,他们的年龄及经历使他们至今还没有掌握电脑的基础知识,因此,纸质报表是他们掌握全局的依据,必备的工具。将报表下载下来,然后能稍许进行修饰,这是必须的。

项目描述

  • 将页面原始呈现的数据及搜索后的数据以 excel 报表的形式下载下来。

知识难点

单一的 url 页面中会呈现出不同的数据,需要能够把不同的数据下载下来,只需要用到 ajax 的知识,在同一个页面中以 json 的形式传递数据。

实现页面模板

不再赘述,仅仅是在页面上加载一个下载按钮,详情看源码。

实现下载功能的试图函数

views.py 的内容如下:

...
...
import os
from collections import OrderedDict
from config import basedir
from pyexcel_xls import save_data, get_data
...
...



@show.route('/<path:filename>', methods=['GET', 'POST'])
def download_xls(filename):
    data = OrderedDict()
    data_path = os.path.join(excel_path, filename)

    num = 0
    if '01.xls' in filename:
        header_data = ['序号', '部门', '角色', '员工', '电话', '客户', '月份',
                       '本月资产余额  ', '上月资产余额', '新增余额']
        body_data = [header_data]
        for t in json.loads(request.get_data()):
            num += 1
            body_data.append([num, t["department"], t["role"], t["staff_name"], t["phone"], t["guest_name"],
                              t["month"], t["balance"], t["last_balance"],
                              float('%.2f' % t["balance"]) - float('%.2f' % t["last_balance"])])
        data.update({'01报表': body_data})

    if '02.xls' in filename:
        header_data = ['序号', '部门', '角色', '员工', '电话', '月份', '管户数'
                       '本月资产余额  ', '上月资产余额', '新增余额']
        body_data = [header_data]
        for t in json.loads(request.get_data()):
            num += 1
            body_data.append([num, t["department"], t["role"], t["staff_name"], t["staff_phone"],
                              t["month"], t["balance"], t["last_balance"],
                              float('%.2f' % t["balance"]) - float('%.2f' % t["last_balance"])])
        data.update({'02报表': body_data})
    save_data(data_path, data)
    return jsonify({"data": "ok"})
  • 很明显,通过 return 返回的 jsonify 数据,大家应该知道,我这是要用 json 了。

  • 而其中的两个 filename 的判断,我是因为要做两个页面的下载链接,所以对 filename 进行了一个判断。

  • 代码中使用了 collections 的有序字典 OrderedDict, pyexcel_xls的相关模块,同时也设置了要生成的 xls 文件的生成路径及文件名——data_path = os.path.join(excel_path, filename)

实现 ajax 功能

既然是 ajax 那就是 javascript 咯,还得返回 html 模板。在 flask 项目中,模板的变量都是封装在 jinja2 中,我们写 javascript 的时候,不能单独再建立一个 js 文件,导入到模板中,因为,无法获得 Jinja2 变量。还得直接在 html 中写 javasript。

01.html 中实现 ajax 下载

{% block js %}
    {{ super() }} <!-- 继承 block js 里的原始 js 文件 -->
    <script>
        $SCRIPT_ROOT = {{ request.script_root|tojson|safe }}; <!-- flask ajax 需要进行该项设置 -->
        function downloadxls() { <!-- 创建 downloadxls 函数,页面上的导出报表按钮需要调用这个函数 -->
            var database = new Object();
            database = {{ database|safe }}; <!-- 调用 jinja2 中的变量,使用 safe ,防止出现多重冒号。 -->
            var data = JSON.stringify(database);
            $.ajax({
                type: "POST",
                url: "{{ url_for('show.download_xls', filename='%s_01.xls' % current_user.phone) }}",<!-- 生成 xls 文件的 url 链接 -->
                contentType: "application/json; charset=utf-8",
                data: data, <!-- 返回到后台的 json 数据 -->
                success: function (msg) {
                    if (msg.data === 'ok'){ <!-- 后台送来的 'ok' 数据 -->
                        //window.open("{{ url_for('show.static', filename='excel_files/%s_01.xls' % current_user.phone) }}");<!-- 用 js 打开 xls 文件,可以进行下载,但是会出现页面闪现的现象,界面不算友好。 -->
                        var $eleForm = $("<form method='get'></form>");<!-- 使用 form 表单的形式进行文件下载,不会出现页面闪现的现象。 -->
                        $eleForm.attr("action", "{{ url_for('show.static', filename='excel_files/%s_01.xls' % current_user.phone) }}");
                        $(document.body).append($eleForm);
                        //提交表单,实现下载
                        $eleForm.submit();
                    }
                }
            })
        }
    </script>
{% endblock %}

详情请看注释。

02.html 中实现 ajax 下载

不在赘述,基本上是和 02.html 页面一样。

至此,你可以实验一下,你的项目是否可以实现报表下载了。

很不幸,我们忘了一个重要的事情,那就是 js 中 database 是继承自 jinja2 变量,它是否是你所想的那种 json 数据呢。答案是否定的。结果是这样的:

018.png

那么我们需要继续工作咯。

使 sqlalchemy 数据 json 化

models.py 中建立一个类:

class AlchemyJsonEncoder(json.JSONEncoder):
    def default(self, obj):
        # 判断是否是Query
        if isinstance(obj, Query):
            # 定义一个字典数组
            fields = []
            # 检索结果集的行记录
            for rec in obj:
                # 定义一个字典对象
                record = {}
                # 检索记录中的成员
                for field in [x for x in dir(rec) if
                              # 过滤属性
                              not x.startswith('_')
                              # 过滤掉方法属性
                              and hasattr(rec.__getattribute__(x), '__call__') == False
                              # 过滤掉不需要的属性
                              and x != 'metadata']:
                    try:
                        record[field] = rec.__getattribute__(field)
                    except TypeError:
                        record[field] = None
                fields.append(record)
            # 返回字典数组
            return fields
        # 其他类型的数据按照默认的方式序列化成JSON
        return json.JSONEncoder.default(self, obj)

详细功能请看注释。

在 01 , 02 两个 url 链接的试图函数中,加入如下内容:

views.py 中增加

return render_template('show/01.html', data=data, searchForm=search_form, database=json.dumps(database, cls=AlchemyJsonEncoder))

此时,在查看我们要传输的 json 数据:

019.png

至此,你可以使用下载功能了,不管你的页面中的数据如何变化,你下载的报表都会显示你当前页面的数据。

源码下载

提示: 让这么多的数据在页面中使用明文传输,是一种极其不安全的行为,这是网络爬虫的肥沃矿场。当然,如果你使用的环境是我这样的内部 web 环境,那就另当别论。不过如果你有更好的解决方法,那么可以告诉我,不胜感激。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 201,784评论 5 474
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,745评论 2 378
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 148,702评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,229评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,245评论 5 363
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,376评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,798评论 3 393
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,471评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,655评论 1 295
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,485评论 2 318
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,535评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,235评论 3 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,793评论 3 304
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,863评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,096评论 1 258
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,654评论 2 348
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,233评论 2 341