前段时间公司发布新需求,要求用户点击按钮可以导出pdf或者html到本地,pdf中要包含可点击跳转的目录,要分页记录页码,还有页眉和页脚,和后台的小哥哥配合试了好多方法,最终完成的效果还不错,在这里做个记录。
##需求
##言归正传
因为设计也比较给力,直接把pdf样式设计成模板了,
所以前端的工作就是按照样式生成html页面扔给后台。
我之前也看了很多博客,查找了很多工具和插件,下面先做个简单介绍。
{{ title }}
@page {
size: A4;/*设置导出的pdf的大小*/
border-bottom: 0.25pt solid #666;/*页脚处的横线*/
border-top: 0.25pt solid #666;/*页眉处的横线*/
@top-left {
content: "我是左侧的页眉内容";
vertical-align: bottom;
color: #333;
font-size: 9pt;
margin: 30pt 0 10pt 0;
}
@top-center {
content: "我是居中的页眉";
vertical-align: bottom;
color: #333;
margin: 30pt 0 10pt 0;
}
@top-right {
content: "我是右侧的页眉内容";
vertical-align: bottom;
color: #333;
font-size: 9pt;
margin: 30pt 0 10pt 0;
}
@bottom-right {/*在页面的右下角生成页码*/
content: counter(page)
" / " counter(pages);
vertical-align: top;
margin: 10pt 0 30pt 0;
color: #333;
font-size: 9pt;
}
@bottom-left {
content: "我是左侧页脚";
color: #333;
font-size: 9pt;
vertical-align: top;
margin: 10pt 0 30pt 0;
}
@bottom-center {
content: "我是居中的页脚";
color: #333;
font-size: 9pt;
vertical-align: top;
margin: 10pt 0 30pt 0;
}
}
@page:first { /*设置封面*/
margin: 0pt;/*让封面的背景图占满整页*/
padding: 0pt;/*让封面的背景图占满整页*/
@top-left {/*设置首页的页眉页脚内容为空*/
content: "";
}
@top-right {/*设置首页的页眉页脚内容为空*/
content: "";
display:none;
}
@bottom-right {/*设置首页的页眉页脚内容为空*/
content:"";
}
@bottom-left {/*设置首页的页眉页脚内容为空*/
content: "";
}
}
}
# 单页的大小: 595 * 842 (px)
# 首页如果占满一整页,则大小:800 * 1142 (px)
# table表格添加border 需要用 css的方式:border:1px solid #000;
# pre显示代码并自动换行:
width: 595px;
white-space: pre-line; //合并空格
word-break: break-word;
word-wrap: break-word; // 必须,不可舍弃
# 不支持css3 的一些属性,如: translate scale rotate ....
# 设计稿按照普通屏大小来进行设计,不要按照高倍屏!不要按照高倍屏!!不要按照高倍屏!!!
# 打印选项里已经设计了留白,所以设计稿左右不必留白,内容充满即可
from weasyprint import HTML
# 生成pdf的文件名称,数组中的print.css是打印样式,index.css是pdf文件的样式
HTML(string='这是一个测试
').write_pdf("test_11_22_test12.pdf", stylesheets=['pdf/print.css','pdf/index.css'])
# 将百度首页导出为pdf
HTML('http://www.baidu.com').write_pdf("baidu.pdf", stylesheets=['pdf/print.css'])
我是页面的头部
h1{
bookmark-level:none;
}
关于bookmark,这是css中的书签,网上的资料不多,感兴趣的小伙伴可以参考点击查看详情
好了,以上内容如果理解了,就可以顺利生成pdf了,下面来看看如何点击按钮进行下载,先上代码
var opt = {
type: 'HEAD',//设置请求方式
url:'/api/html_pdf?format=json',//请求路径
headers:{'Authorization':token},
data:{work_name:work_name,flag:flag},
complete:function(xhr, data) {
var dis = xhr.getResponseHeader("Content-Disposition")
download(dis)
}
}
$('.downPdf').bind('click',function(){
$.ajax(opt);
})
function getFileName(headers) { //获取responseHeaders中的文件名称
return headers.replace('attachment;filename=', '');
}
function download(res) {
var a = document.createElement('a'); //创建一个a链接
var filename = getFileName(res); //pdf文件的名称
var url = '/media/down_pdf/' + filename; //服务器中的pdf文件存储位置
a.href = url; //将文件地址付给a链接
a.setAttribute('download','report.pdf');//设置a链接的download属性,并命名下载的pdf文件为report.pdf
a.click(); //触发a链接的下载动作
}
服务端向浏览器发送文件时,如果是浏览器支持的文件类型,一般会默认被浏览器打开,比如txt、jpg、视频以及音频等,但如果需要提示用户保存,就要利用 Content-Disposition 进行处理:
Response.AppendHeader("Content-Disposition","attachment;filename=FileName.pdf");
Response.AppendHeader("Content-Type","application/actet-stream");
Response.AppendHeader("Content-Type","application/pdf");