转:Generating PDFs from Web Pages on the Fly with jsPDF
The Portable Document Format has been one the major innovations in the fields of desktop publishing and office automations.
It’s widely used in web publishing too, but unfortunately very often in wrong ways – like using it to replace contents that should have been built with HTML. This causes many problems regarding usability, accessibility, SEO and so on.
However, there are some cases in which PDF files are required: when a document needs to be archived and must be used outside the web (for example an invoice) or when you need a deep control on printing.
It was just the need to control printing that brought me to research a way to easily generate a PDF.
The purpose of this article is not to simply explain how a PDF can be created (there are many easy way to do this), but also to focus on the circumstances where a PDF file can solve a problem, and how a simple tool like jsPDF can help with this.
Dealing with Printing
Anyone who has dealt with CSS printing rules knows how difficult it is to achieve a decent level of cross-browser compatibility (take a look, for example, at thePage-break support table at Can I Use). Therefore, when I need to build something that must be printed, I always try to avoid CSS, and the simplest solution is to use PDF.
I’m not talking here about the simple conversion of HTML to PDF. (I’ve tried several tools of that type, but none of them has fully satisfied me.) My goal is to have complete control over the positioning and size of elements, page breaks and so on.
In the past I’ve often used FPDF, a PHP tool that can easily give you such controls and that can be easily expanded with many plugins.
Unfortunately, the library seems to be abandoned (its last version dates back to 2011) (Update: actually, the latest version appears to be from December 2015), but thanks to some JavaScript libraries, we now have the ability to build PDF files directly in the clients (thus making their generation faster).
When I started my project, some months ago, I searched for a JS library, and finally I found two candidates: jsPDF and pdfmake. pdfmake seems to be well documented and very easy to use, but since it was a beta version, I decided for jsPDF.
PDF Building with jsPDF
The jsPDF documentation is fairly minimal, consisting of a single page along with some demos, and a little more information in the source file (or in its jsDoc pages), so keep in mind that using it for complex projects can be a little hard in the beginning.
Anyway, jsPDF is very easy for basic PDF files generation. Take a look to a simple “Hello World” example:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="utf-8">
<title>Hello world</title>
</head>
<body>
<h1>Hello world</h1>
<script type="text/javascript" src="https://cdnjs.cloudflare.com/ajax/libs/jspdf/1.0.272/jspdf.debug.js"></script>
<script type="text/javascript">
var pdf = new jsPDF();
pdf.text(30, 30, 'Hello world!');
pdf.save('hello_world.pdf');
</script>
</body>
</html>
This HTML page generates a one-page PDF file and saves it on your computer. First you have to link to the jsPDF library (in this case, from cdnjs.com), then a jsPDF instance is created, a line of text is added, and the result is saved as hello_world.pdf
.
Note that I’ve used the 1.0.272 version, and that it’s not the latest: at the time of writing this, the most recent version is the 1.1.135, but it has many issues, so I am still using the previous one.
You can see how extremely simple it is to build a basic PDF file (you can find more examples at the jsPDF site).
Let’s try to build something harder.
The Flyer Project
The user interface allows the user to insert some basic data (a title, an abstract and a price). Optionally, an image can be added, otherwise a grey-boxed Special Offer
title is displayed.
Other data (the agency name and its website URL and logo) are embedded in the application code.
The PDF can be previewed in a iframe (except Explorer or Edge) or directly downloaded.
When the Update preview
or the Download
buttons are clicked, the PDF is generated using jsPDF and passed to the iframe as a data URI string or saved to disk, as in the above example.
The PDF generation first creates a new instance a jsPDF object with these options: portrait orientation (p
), millimeters units (mm
), ‘A4’ format.
var pdf = new jsPDF('p', 'mm', 'a4');
Images are added using the addImage
function. Note that every object placed in the PDF page must be exactly positioned. You have to take care of the coordinates of each object using the units declared.
// pdf.addImage(base64_source, image format, X, Y, width, height)
pdf.addImage(agency_logo.src, 'PNG', logo_sizes.centered_x, _y, logo_sizes.w, logo_sizes.h);
Images must be Base64 encoded: the agency logo is embedded in the script in this format, while the image loaded by the user is encoded using the readAsDataURL
method in the$('#flyer-image').change
listener.
The title is added using the textAlign
function. Note that this function is not part of the jsPDF core, but, as suggested by the author in his examples, the library can be easily expanded using its API. You can find the textAlign()
function at the top of flyer builder script:
pdf.textAlign(flyer_title, {align: "center"}, 0, _y);
This function calculates the X coordinate of the the text string to make it centered, and then calls the native text()
method:
pdf.text(text string, X, Y);
To change text properties, you can use the setFontSize()
, setFont()
, setTextColor()
andsetFontType()
methods.
To set a 20pt Times Bold red string, for example, you need to type this:
pdf.setFontSize(20);
pdf.setFont("times");
pdf.setFontType("bold");
pdf.setTextColor(255, 0, 0);
pdf.text(10,10, 'This is a 20pt Times Bold red string');
The Special offer
grey box and the price circle use two similar methods: roundedRect()
andcircle()
. Both of them require top-left coordinates, size values (the width and height in the first case and the radius in the second one):
pdf.roundedRect( X, Y, width, height, radius along X axis, radius along Y axis, style);
pdf.circle( X, Y, radius, style);
The style
parameters refers to the fill and stroke properties of the object. Valid styles are: S
[default] for stroke, F
for fill, and DF
(or FD
) for fill and stroke.
Fill and stroke properties must be set in advance using setFillColor
and setDrawColor
, which require a RGB value and setLineWidth
that requires the line width value in the unit declared at inception of PDF document.
The complete code is available in the CodePen demo:
HTML
<div class="container-fluid flyer-builder">
<h1>Flyer Builder</h1>
<p><em>Forked from <a target="_blank" href="http://codepen.io/massimo-cassandro/pen/qOrJNx">this original Pen by Massimo Cassandro</a>, for his SitePoint article on <a target="_blank" href="http://sitepoint.com/generating-pdfs-from-web-pages-on-the-fly-with-jspdf">Generating PDFs from Web Pages on the Fly with jsPDF</a>. The preview doesn't work with Chrome or Safari in this demo since it has been loaded in a iframe. Use Firefox or try at <a href="http://www.primominuto.net/articoli/pdf_flyer/flyer.html">www.primominuto.net/articoli/pdf_flyer/flyer.html</a></em></p> <div class="row">
<div class="col-sm-6 loc_form"> <div class="form-group">
<label for="flyer-image" class="control-label">Image</label>
<input id="flyer-image" type="file" tabindex="1">
</div> <div class="row">
<div class="col-sm-7">
<div class="form-group">
<label for="flyer-title" class="control-label">Title</label>
<input required class="form-control" id="flyer-title" placeholder="Main title" value="Really incredible!!" maxlength="255" type="text" tabindex="2">
</div>
</div>
<div class="col-sm-3">
<div class="form-group">
<label for="flyer-title-size" class="control-label">Size (pt)</label>
<input required class="form-control" id="flyer-title-size" value="60" min="1" step="1" type="number" tabindex="3" title="Title size: tune it to fit the available space">
</div>
</div>
<div class="col-sm-2">
<div class="form-group">
<label for="flyer-title-color" class="control-label">Color</label>
<input required class="form-control" id="flyer-title-color" value="#0080FF" type="color" tabindex="4" title="Title color">
</div>
</div>
</div> <div class="form-group">
<label class="control-label" for="flyer-description">Description</label>
<textarea class="form-control" id="flyer-description" placeholder="Insert a short description taking care of the available space">Lorem ipsum dolor sit amet, consectetur adipisicing elit. Nemo, nihil officia neque ad expedita consequatur quae! Voluptate, incidunt, earum, sit, eveniet harum ratione expedita quibusdam possimus sed laboriosam dolore ut recusandae eos. Ipsa,natus pariatur iste dolorum optio nostrum consectetur!</textarea>
</div> <div class="row">
<div class="col-sm-4">
<div class="form-group">
<label for="flyer-price" class="control-label">Price</label>
<input required class="form-control" id="flyer-price" placeholder="Enter price (w/o decimals)" value="100" type="number" tabindex="5" step="any" min="0">
</div>
</div>
<div class="col-sm-3">
<div class="form-group">
<label for="flyer-price-currency" class="control-label">Currency</label>
<select required class="form-control" id="flyer-price-currency" tabindex="6">
<option value="€">€</option>
<option value="$">$</option>
<option value="£">£</option>
</select>
</div>
</div>
<div class="col-sm-2">
<div class="form-group">
<label for="flyer-price-color" class="control-label">Color</label>
<input required class="form-control" id="flyer-price-color" value="#cc0000" type="color" tabindex="8" title="Price color">
</div>
</div>
</div>
<hr>
<div class="row">
<div class="col-sm-6">
<div class="form-group">
<button id="flyer_preview_btn" type="button" class="btn btn-primary btn-block" tabindex="9">Update preview</button>
</div>
</div>
<div class="col-sm-6">
<div class="form-group text-right">
<button id="flyer_download_btn" type="button" class="btn btn-default btn-xs btn-block" tabindex="10">Download</button>
</div>
</div>
</div> </div> <div class="col-sm-6">
<iframe id="pdf_preview" type="application/pdf" src=""></iframe>
</div> </div>
</div>
css
.flyer-builder {
position: relative;
height: 100%; & iframe,
.no_iframe {
width:100%;
height:400px;
border:1px solid #666;
background-color: #ddd;
} & .no_iframe {
& > div {
//margin:0 .3em;
width:80%;
position: absolute;
top:50%;
left:50%;
transform: translate(-50%, -50%);
font-weight: normal;
}
} @media all and (min-width:768px) {
& .btn.btn-block {
display: inline;
width: auto;
}
}
}
js
(function() {
"use strict"; /*
Refs:
http://mrrio.github.io/jsPDF/
https://github.com/MrRio/jsPDF
https://mrrio.github.io/jsPDF/doc/symbols/jsPDF.html
*/ // some variables
var agency_logo = {
src: '',
w: 800,
h: 285
},
agency_name = 'Travel & Holidays',
agency_site_url = 'www.travelandholidays.com',
footer = agency_name + ' - ' + agency_site_url, page_size = 'a4',
page_width = 210, // mm
page_margin = 10, // mm
content_width = page_width - (page_margin * 2), // available width for the content _y, _x, // drawing coord
color_array, _string, lineHeight, y_correction, // some variables vspace = 10 // standard vertical space between elements ; // some variables
var can_display_preview = true, // if true a preview of the PDF can be displayed in the iframe,
// this value is set to false if the browser can't display the preview
preview_container = $('#pdf_preview'),
update_preview_button = $('#flyer_preview_btn'),
download_button = $('#flyer_download_btn'); // preview can be displayed?
if (navigator.msSaveBlob) { // older IE
update_preview_button.prop('disabled', true);
can_display_preview = false;
preview_container.replaceWith(
'<div class="no_iframe">' +
'<div>' +
"The preview can't be displayed" +
'</div>' +
'</div>'
);
} // utilities
var hex2rgb = function(hex_string) {
if (/^#/.test(hex_string)) {
hex_string = hex_string.substr(1);
}
if (hex_string.length === 3) {
hex_string = hex_string.replace(/\w/, function(match) {
return String(match) + String(match);
});
} return {
red: parseInt(hex_string.substr(0, 2), 16),
green: parseInt(hex_string.substr(2, 2), 16),
blue: parseInt(hex_string.substr(4, 2), 16)
};
}, px2mm = function(pixel) {
// px to inches
var inches = pixel / 72;
return inches * 25.4;
}, mm2px = function(millimeters) {
// mm to inches
var inches = millimeters / 25.4;
return inches * 72;
}, // function to calculate and check img sizes
imgSizes = function(img_w, img_h, img_mm_w) {
/*
img_w and img_h represent the original image size, in pixel
img_mm_w is the desidered rendered image size, in millimeters */ if (img_mm_w > content_width) { // this should be never used...
img_mm_w = content_width;
} if (mm2px(img_mm_w) > img_w) {
throw 'The `img_mm_w` parameter is too big';
} var img_mm_h = Math.round((px2mm(img_h) * img_mm_w) / px2mm(img_w)); return {
w: img_mm_w,
h: img_mm_h,
centered_x: (page_width - img_mm_w) / 2
};
}; try { // image reading
// More info at https://developer.mozilla.org/en-US/docs/Web/API/FileReader/readAsDataURL
// for more examples about file api
// take a look at https://scotch.io/tutorials/use-the-html5-file-api-to-work-with-files-locally-in-the-browser var flyer_img = $('#flyer-image'),
img_data = null; $('#flyer-image').change(function() { // temporary disabling buttons while parsing image
update_preview_button.prop('disabled', true);
download_button.prop('disabled', true); /*
getting the file flyer_img[0] : transforms the jQuery reference to a DOM object
files[0] : refers to the file the the user has chosen use `console.log(user_file);` to show some info about the file */ var user_file = flyer_img[0].files[0]; img_data = {
type: user_file.type === 'image/jpeg' ? 'JPEG' : 'PNG' // maybe you should add some controls to prevent loading of other file types
}; var reader = new FileReader();
reader.onload = function(event) {
img_data.src = event.target.result; // we need this to get img dimensions
var user_img = new Image();
user_img.onload = function() {
img_data.w = user_img.width;
img_data.h = user_img.height; // restoring buttons
download_button.prop('disabled', false);
if (can_display_preview) {
update_preview_button.prop('disabled', false);
}
};
user_img.src = img_data.src;
}; //when the file is read it triggers the onload event above.
reader.readAsDataURL(user_file);
}); //!pdf builder
var createPDF = function(update_preview) {
/*
update_preview:
==> true: shows pdf online
==> false: downloads the pdf
*/ _y = page_margin; // vertical starting point // form data
var flyer_title = $('#flyer-title').val(),
flyer_title_size = $('#flyer-title-size').val(),
flyer_title_color = $('#flyer-title-color').val(),
flyer_description = $('#flyer-description').val(),
flyer_price = $('#flyer-price').val(),
flyer_price_currency = $('#flyer-price-currency').val(),
flyer_price_color = $('#flyer-price-color').val(); var pdf = new jsPDF('p', 'mm', page_size),
text_baseline, // some colors:
light_grey = 237,
grey = 128,
black = 0,
white = 255; // Optional - set properties of the document
pdf.setProperties({
title: flyer_title,
subject: footer,
author: 'me',
creator: 'Flyer Builder & jsPDF'
}); // !logo
var logo_sizes = imgSizes(agency_logo.w, agency_logo.h, 60);
pdf.addImage(agency_logo.src, 'PNG', logo_sizes.centered_x, _y, logo_sizes.w, logo_sizes.h); // fonts initializing
pdf.setFont("helvetica");
pdf.setFontType("bold"); // !main title
color_array = hex2rgb(flyer_title_color);
pdf.setTextColor(color_array.red, color_array.green, color_array.blue); pdf.setFontSize(flyer_title_size); lineHeight = px2mm(pdf.getLineHeight(flyer_title)); _y += (logo_sizes.h + vspace + lineHeight); pdf.textAlign(flyer_title, {
align: "center"
}, 0, _y); _y += vspace; // !user image
if (img_data) {
var img_sizes = imgSizes(img_data.w, img_data.h, content_width);
pdf.addImage(img_data.src, img_data.type, img_sizes.centered_x, _y, img_sizes.w, img_sizes.h);
_y += img_sizes.h; } else {
// if we haven't an image, a grey box with a text is added var box_height = 80; pdf.setFillColor(light_grey);
pdf.roundedRect(page_margin, _y, content_width, box_height, 5, 5, 'F');
pdf.setFontSize(60);
pdf.setTextColor(white);
_string = 'SPECIAL OFFER';
lineHeight = px2mm(pdf.getLineHeight(_string)); // y_correction: value to be added to y coord of the grey box to have text vertically centered
// it is empirically calculated adding 1/3 of text line height to half box height
y_correction = box_height / 2 + lineHeight / 3; pdf.textAlign(_string, {
align: "center"
}, 0, _y + y_correction); _y += box_height;
} // !price
// first: creating a circle that overlaps the bottom side of the image
var circle_radius = 30;
color_array = hex2rgb(flyer_price_color);
pdf.setFillColor(color_array.red, color_array.green, color_array.blue); // _x and _y refer to center of the circle
_x = content_width - circle_radius; // circle ends at `page_margin` millimeters from the image right side pdf.circle(_x, _y, circle_radius, 'F'); // circle overlaps image for 1/2 of its height pdf.setFontSize(60);
pdf.setFont("times");
pdf.setFontType("bold"); _string = flyer_price_currency + parseInt(flyer_price); // decimals are removed lineHeight = px2mm(pdf.getLineHeight(_string));
y_correction = lineHeight / 3; pdf.setTextColor(white);
pdf.textAlign(_string, {
align: "centerAtX"
}, _x, _y + y_correction); // !description
if (flyer_description) {
pdf.setFontSize(20);
pdf.setFont("helvetica");
pdf.setFontType("italic");
pdf.setTextColor(grey); var lineWidth = content_width - (circle_radius * 2) - (page_margin * 2);
_y += page_margin; var line_height = 12; // mm var description_lines = pdf.splitTextToSize(flyer_description, lineWidth);
//pdf.text(page_margin, _y, description_lines); // doesn't allows to change line spacing for (var i = 0; i < description_lines.length; i++) {
pdf.text(page_margin, _y, description_lines[i]);
_y += line_height;
} } // !footer
_y = 287;
pdf.setFontSize(9);
pdf.setFontType("normal");
pdf.setTextColor(black);
pdf.textAlign(footer, {
align: "center"
}, 0, _y); // ****************************
// output
if (update_preview) {
preview_container.attr('src', pdf.output('datauristring'));
} else {
pdf.save('flyer ' + flyer_title + '.pdf');
} }; // end createPDF // !buttons
update_preview_button.click(function() {
createPDF(true);
}); $('#flyer_download_btn').click(function() {
createPDF(false);
}); } catch (e) {
console.log(e);
} })();
Conclusion
This basic example shows how it’s possible to build a very basic flyer with jsPDF.
Its use could be easy, but the lack of a complete documentation makes every step really complicated.
I’m still looking around for other solutions, keeping my eye on others like pdfmake. But ultimately, I think the only really definitive solution can be better browser support for printing CSS rules!
Have you used jsPDF or something similar? What was your experience? Let me know in the comments.
转:Generating PDFs from Web Pages on the Fly with jsPDF的更多相关文章
- 如何在ASP.NET Web站点中统一页面布局[Creating a Consistent Layout in ASP.NET Web Pages(Razor) Sites]
如何在ASP.NET Web站点中统一页面布局[Creating a Consistent Layout in ASP.NET Web Pages(Razor) Sites] 一.布局页面介绍[Abo ...
- Displaying Data in a Chart with ASP.NET Web Pages (Razor)
This article explains how to use a chart to display data in an ASP.NET Web Pages (Razor) website by ...
- Web Pages - Efficient Paging Without The WebGrid
Web Pages - Efficient Paging Without The WebGrid If you want to display your data over a number of p ...
- Web Pages razor 学习
1. Web Pages razor Web Pages 是三种 ASP.NET 编程模型中的一种,用于创建 ASP.NET 网站和 web 应用程序. 其他两种编程模型是 Web Forms 和 M ...
- ASP.NET MVC3 系列教程 – Web Pages 1.0
http://www.cnblogs.com/highend/archive/2011/04/14/aspnet_mvc3_web_pages.html I:Web Pages 1.0中以“_”开头的 ...
- Web网页中动态数据区域的识别与抽取 Dynamical Data Regions Identification and Extraction in Web Pages
Web网页中动态数据区域的识别与抽取 Dynamical Data Regions Identification and Extraction in Web Pages Web网页中动态数据区域的识别 ...
- 五张图概括 什么是 ASP 、 ASP.NET (Web Pages,Web Forms ,MVC )
当你看懂下面这五张图,我相信你对于学习.NET Web开发路线将不陌生! 来源: http://www.w3 ...
- 安卓,网页控件,显示网页 Android, web controls, display web pages
安卓,网页控件,显示网页Android, web controls, display web pages 作者:韩梦飞沙 Author:han_meng_fei_sha 邮箱:313134555@qq ...
- ASP.NET —— Web Pages
为简单起见,新建一个空的web工程,再新建一个MVC的视图(.cshtml),因为WP是单页面模型,所以以后就在这个页面中进行试验. Razor语法简介: 变量可用var或者其确切类型声明. 遍历fo ...
随机推荐
- Git、Github和GitLab的区别及与SVN的比较
个人理解: SVN适合领导啊,大家一起在加班,看你进度什么的,git则不必如此,忙完传上来完活. 一.含义: 百度上这样介绍的: Git(读音为/gɪt/.)是一个开源的分布式版本控制系统,可以有效. ...
- storm中的topology-worker-executor-task
调度角色 调度方法 自定义调度 1 调度角色 任务角色结构 上图是JStorm中一个topology对应的任务执行结构,其中worker是进程,executor对应于线程,task对应着spout ...
- 外观模式及php实现
外观模式: 外观模式(Facade Pattern):外部与一个子系统的通信必须通过一个统一的外观对象进行,为子系统中的一组接口提供一个一致的界面,外观模式定义了一个高层接口,这个接口使得这一子系统更 ...
- 好吧,不说闲言碎语,不抱怨,好好工作,好好学习,多总结。记录一下昨天做vuejs的心得
1.做了两个bat文件,一个是直接定位到vuejs项目并且运行,另一个就是打包 run.bat d:cd wwwcd vuecd dtbpmcnpm run devpause build.bat cd ...
- Android中的GreenDao框架修改数据库的存储路径
目前android中比较热门的数据库框架有greenDAO.OrmLite.AndrORM,其中我比较喜欢用GreenDao,其运行效率最高,内存消耗最少,性能最佳.具体怎么使用GreenDao,网上 ...
- C++拾遗(七)——关联容器
关联容器(Associative containers)支持通过键来高效地查找和读取元素.两个基本的关联容器类型是 map 和set.map 的元素以键-值(key-value)对的形式组织:键用作元 ...
- JSON 序列化格式
一.C#处理简单json数据json数据: 复制代码代码如下: {"result":"0","res_info":"ok" ...
- Android开发出现 StackOverflowError
问题:StackOverflowError 在HTC或者摩托罗拉的手机上测试出现 StackOverflowError 的错误. 06-12 10:28:31.750: E/AndroidRuntim ...
- vue axios 请求 https 的特殊处理
最近遇到自签发的CA证书,在前端axios请求https请求时,无法自动加载证书. 解决方法:将无法加载的请求在浏览器新窗口手动加载,选择继续连接. 重新加载,问题解决. 根本原因:因为自签发证书,浏 ...
- 《队长说得队》【Alpha】Scrum meeting 4
项目 内容 这个作业属于哪个课程 >>2016级计算机科学与工程学院软件工程(西北师范大学) 这个作业的要求在哪里 >>实验十二 团队作业8:软件测试与ALPHA冲刺 团队名称 ...