Drupal下如何把包含Highcharts的页面输出到PDF

来自:http://www.oschina.net/question/554557_106965

本文PDF版下载:Drupal下如何把包含Highcharts的页面输出到PDF

 

Highcharts是一种通过JavaScript在浏览器客户端生成图的库,由于缺乏生成把HTML转换成PDF的Javascript库,生成PDF的工作只能放在服务器端做。但是服务器端如何比较完美地解释和渲染HTML、Javascript、CSS呢?答案在PhantomJS。本文基于Drupal的权限和主题样式,讲述如何把这一系列的技术组合起来,实现包含Highcharts的页面输出到PDF。


1     什么是Highcharts

Highcharts是一种使用纯HTML5/Javacript在浏览器端生成图表的JavaScript库,支持所有现代浏览器,如ChromeFirefoxIEiPad/iPhone上的浏览器(Safari)等。特别强调支持IE6(连IE6都支持,还有什么好担心的呢?)。全球前100强的企业中有超过35家使用了该技术。

 

2013322日,3.0版本正式发布。

 

相比起在服务器后台生成图表的技术,在前端生成可以生成动态、有交互效果的图表。

 

对于非商业用途,Highcharts是免费的,商业用途的费用可查询如下页面:http://shop.highsoft.com/highcharts.html

 

以下是Highcharts的官方网站:http://www.highcharts.com/

 

2     Drupal中如何使用Highcharts

首先下载Highcharts库,放到Drupalsites/all/libraries下,

 

然后在需要使用Highcharts的页面中添加如下一行,即可使用

drupal_add_js(libraries_get_path('highcharts') . '/js/highcharts.js');

 

余下就是参考HighchartsAPI了,与Drupal无关。

 

Drupal本身也有Highcharts模块,主要用于Views数据的图表展示:参考http://drupal.org/project/highcharts

3     Highcharts导出成PDF的若干种方法

3.1     利用API把单幅Highcharts导出成静态图再重新组合页面

Highcharts本身提供导出成图片的方法(APIhttp://docs.highcharts.com/#export-module),但是它和整个页面的排版脱离,要把一幅幅Highcharts图导出后再和页面其它元素组合起来输出PDF工作量非常大。并且这也设计把Highcharts导出的静态图从外部(浏览器)传回PHP

 

Highcharts提供了在服务器端导出静态图的方案(事实上它使用了我们之后会讨论的PhantomJS方案):

http://www.highcharts.com/component/content/article/2-news/52-serverside-generated-charts

3.2     html2canvas

这是一个通过JS截屏的工具,但是它并非真正截屏,而是模拟。笔者并没有尝试使用,可以遇见,在不同浏览器,不同窗口大小会有不同效果,而且,我们希望PDF版会隐藏页面的一些元素(比如菜单、导航等)

 

软件主页:http://html2canvas.hertzen.com/

3.3     wkhtmltopdf

这是一个非常接近的解决方案,项目的目的很直接,就是在服务器端运行一个Webkit内核的命令,把HTML转换成PDFDrupal/PHP里需要做的只是执行一条system语句。笔者一度以为这是最终的解决方案,但随着使用的深入,发现项目缺乏维护,遇到一个编码问题一直没有被修复,页面的渲染效果似乎也和Chrome有差异。它可能是基于一个比较旧的Webit内核。鉴于以上原因笔者还是没有采用该方案。

 

软件主页:https://code.google.com/p/wkhtmltopdf/

3.4     PhantomJS

这是笔者最终采用的方案,它是一个基于Webkit的命令行可运行的浏览器,目前处于比较好的维护状态,它可以用于自动化测试。

 

软件主页:

http://phantomjs.org/

 

使用方法如下:

1.     PhantomJS下载到服务器,并置于$PATH中,笔者是做一个符号链接到/usr/bin/phantomjs

2.     创建如下JS脚本:generate_pdf.js

// This file is NOT a browser-run javascript but PhantonJS script

 

var system = require('system');

var html_path = system.args[1];

var pdf_path = system.args[2];

 

var page = require('webpage').create();

page.paperSize = {

  format: 'A4',

  orientation: 'landscape',

  border: '1cm'

};

 

page.open(html_path, function () {

    page.render(pdf_path);

    phantom.exit();

});

3.     运行如下命令即可把HTML页面输出到PDF

$ phantomjs generate_pdf.js http://www.google.com google.pdf

 

使用PhantomJS生成的PDF比较大,我想它很可能每一页就是一张图片。

4     如何在Drupal实现权限控制

如果需要生成PDF的页面不能匿名访问,那么对于在命令行直接运行phantomjs是个问题,可以通过以下两种方式解决:

4.1     运行phantomjs时设置登录用户的cookie

笔者并没有试验过这种方法,印象中相关cookie值在Drupal里的计算可能和用户来源有关,不确定直接把用户的cookie传给phantomjs是否能正常通过Drupal的身份验证。这种方法调试起来也比较麻烦。

4.2     在页面URL中增加身份验证的参数

这个方法的前提是该页面是自己写的,只有这样我们才能改变其权限验证以及页面的参数。具体方法是把用户的uidsession ID以参数形式传给页面,作为身份验证的参数,判断过程可参考以下代码:

  // permission checking

  if (!user_access('access content')) {

    $auth_uid = arg(4);

    $auth_sid = arg(5);

    if (isset($auth_uid) && isset($auth_sid)) {

      // check permission, used for authentication of print

      $account = user_load($auth_uid);

      if (user_access('access content', $account)) {

        $result = db_query('SELECT uid, sid

          FROM {sessions}

          WHERE uid = :uid AND sid = :sid',

          array(':uid' => $auth_uid, ':sid' => $auth_sid));

        if ($result->rowCount() == 0) {

          drupal_access_denied ();

          return;

        }

      } else {

        drupal_access_denied();

        return;

      }

    } else {

      drupal_access_denied();

      return;

    }

  }

5     如何在Drupal为页面添加打印相关样式

首先,我们在页面中应该通过某种方式来区分当前正在生产的页面是正常浏览状态,还是处于生成PDF的状态。当处于生成PDF状态时,我们添加如下一行:(假设我们使用zen主题,其它主题请按需修改其打印样式路径)

drupal_add_css(drupal_get_path('theme', 'zen') . '/css/print.css');

 

这样我们就能获得额外的打印渲染样式,print.css中比较明显的效果是把内容区以外的东西都隐藏掉了。我们也可为页面的PDF版再通过drupal_add_css的方法添加额外的样式。

6     打印的分页问题

分页的样式是page-break-after, page-break-before, page-break-inside,如果我们想在制定类的元素后分页可添加如下样式:

.bkaft {

  page-break-after:always;

}

 

然后在需要分页的元素中添加类:

<div class=”bkaft”>…</div>

 

参考:http://www.w3schools.com/cssref/pr_print_pageba.asp

 

值得注意的是,page-break*属性不能用于position:absolutefloat元素,不能用在table和有边框的元素内。Chrome分页成功的情况似乎要少于IE

7     PDF字体问题

在部分Linux系统中,PhantomJS可能无法渲染中文,这是和系统的字体有关。对于Open SUSE Linux,我们可以使用以下命令解决:

$ sudo zypper install *font*

 

有些Linux系统的默认字体不好看,我们可以通过强制方式指定字体。下面的"WenQuanYi Zen Hei"是相对比较好看的一中。.page-xxx需要根据页面具体情况修改。!important是较少使用的CSS属性,它相当于把该样式设置为最高优先级,非迫不得已不要使用,因为难以再写一些样式来覆盖它。

.page-xxx * {

  font-family: "WenQuanYi Zen Hei" !important;

}

 

如果想使用其它非系统自带字体(注意:这可能存在版权问题),可以把相关字体文件复制到/usr/share/fonts目录下的一个自定义子目录,然后通过以下命令更新字体:

# cd /usr/share/fonts/windows

# mkfontscale

# mkfontdir

# fc-cache

 

8     HighchartsIE8中渲染性能低下的问题

IE8对于渲染Highcharts异常地慢,解决方案是让IE8IE7兼容模式下运行(可能会快10倍),Drupal中通过以下代码可实现:

    $meta_ie_render_engine = array(

        '#type' => 'html_tag',

        '#tag' => 'meta',

        '#attributes' => array(

            'http-equiv' => 'X-UA-Compatible',

            'content' =>  'IE=7',

        ),

        '#weight' => -9999,

    );

   

    // Add header meta tag for IE to head

    drupal_add_html_head($meta_ie_render_engine, 'meta_ie_render_engine');

9     参考资料

1.        Highcharts主页:http://www.highcharts.com/

2.        Drupal Highcharts模块:http://drupal.org/project/highcharts

3.        html2canvas主页:http://html2canvas.hertzen.com/

4.        wkhtmltopdf主页:https://code.google.com/p/wkhtmltopdf/

5.        PhantomJS主页:http://phantomjs.org/


本资料由"知未科技"赞助制作

 

Sponsored By Curious-Tech.com


 

你可能感兴趣的:(Drupal下如何把包含Highcharts的页面输出到PDF)