用phpword处理docx模板时候始终发生神奇的BUG,就是复制原版例子里的${Value1}进自己的模板然后替换是没问题的,但是只要一改动这个变量文字,PHP做相应替换就失效了。

用了下残废百度无果,一怒翻起google,准确度高多了。

原来有2个PHPWord项目:

其中适用比较广的是PHPOffice项目下的一个子项目

Git地址:https://github.com/PHPOffice/PHPWord

查看官方的文档:

http://phpword.readthedocs.org/en/latest/templates-processing.html?highlight=replace

具体如何用模板做替换:

[php]  view plain copy print ?
  1. include_once ('exec/lib/phpword/src/PhpWord/PHPWord.php');
  2. use PhpOffice\PhpWord\Autoloader;
  3. use PhpOffice\PhpWord\Settings;
  4. use PhpOffice\PhpWord\IOFactory;
  5. include_once ('exec/lib/phpword/src/PhpWord/Autoloader.php');
  6. Autoloader::register();
  7. Settings::loadConfig();
  8. // Create a new PHPWord Object
  9. $PHPWord = new \PhpOffice\PhpWord\PhpWord();
  10. $templateProcessor = new \PhpOffice\PhpWord\TemplateProcessor('statics/template/adminPositive.docx');
  11. $templateProcessor->setValue('xm1', '姓名');
  12. $templateProcessor->setValue('zw1', '公务员');
  13. $templateProcessor->setValue('sfz1', '360281199909090009');
  14. $templateProcessor->setValue('gz1', '统发');
  15. //$templateProcessor->setValue('Street', 'Coming-Undone-Street 32');
  16. $templateProcessor->saveAs('test.docx');

这里要注意的是换行问题。

另外copy一篇比较nice的博文做个备份

转载自:http://wangye.org/blog/archives/943/

===============================================================================================

最近一个项目开发要用到PHP技术导出Word文档,比较了几种方案,首先是使用Microsoft Office自带的ActiveX/COM组件,比如Word.Application,这种方式的优点是格式兼容度高,可以生成纯doc的Word2003格式文档,缺点一是比较占资源(调用会启动一个WINWORD.EXE进程),不适合Web多用户访问使用;二是PHP这种Web开发技术大多数是跑在Linux服务器上,当然也就无法使用Windows下的技术了,平台可移植和兼容性不好。第二种生成Word的方案是生成Word兼容的网页格式,然后以Word方式打开,这种方案总体上感觉怪怪的,毕竟文件格式是HTML的,而且格式兼容度不好,不过这种方式的优点是节省服务器资源,能够快速生成;最后一种方案也就是今天的主角,采用PHPWord生成Word2007(docx)格式的文档,现在基本上微软Office Word 2003以后的版本均兼容这种格式了,对于2003版本来说,仅需要下载安装个兼容格式包(下载地址),也能正常打开这类文件,当然如果你使用的是最新版本的Office(包括但不限于Office 2007、Office 2010)则不需要安装此格式包。

好了,下面我就介绍一下PHPWord,大家可以通过访问项目主页下载并获得关于项目的更多信息。

我在使用过程中主要遇到了中文乱码的问题,结合网上大神们的指导,通过下面的方式解决了这类问题,希望对大家有所帮助。

1、增加东亚字体支持

打开并编辑路径/Writer/Word2007/Base.php文件内容,大概在第349行(行数随着版本可能会有变化)大概函数_writeTextStyle内添加:

[php]  view plain copy print ?
  1. $objWriter->writeAttribute('w:eastAsia', $font)

比如我的修改片段基本是下面这样:

[php]  view plain copy print ?
  1. // Font
  2. if($font != 'Arial') {
  3. $objWriter->startElement('w:rFonts');
  4. $objWriter->writeAttribute('w:eastAsia', $font); // 添加这行
  5. $objWriter->writeAttribute('w:ascii', $font);
  6. $objWriter->writeAttribute('w:hAnsi', $font);
  7. $objWriter->writeAttribute('w:cs', $font);
  8. $objWriter->endElement();
  9. }

2. 解决中文乱码问题

编辑PHPWord/Template.php,找到代码$replace = utf8_encode($replace);,删除或者注释掉这行代码,添加$replace = iconv( 'gbk','utf-8', $replace);,比如代码改为如下:

[php]  view plain copy print ?
  1. /**
  2. * Set a Template value
  3. *
  4. * @param mixed $search
  5. * @param mixed $replace
  6. */
  7. public function setValue($search, $replace) {
  8. if(substr($search, 0, 2) !== '${' && substr($search, -1) !== '}') {
  9. $search = '${'.$search.'}';
  10. }
  11. if(!is_array($replace)) {
  12. //$replace = utf8_encode($replace);
  13. $replace =iconv('gbk', 'utf-8', $replace); // 注释掉上面行后添加这行
  14. }
  15. $this->_documentXML = str_replace($search, $replace, $this->_documentXML);
  16. }

调用方式如下:

[php]  view plain copy print ?
  1. $document->setValue('Template', iconv('utf-8', 'GB2312//IGNORE', '中文'));

上面的代码主要解决模板的问题,下面同样的道理,解决Section添加文本的问题,找到代码$givenText = utf8_encode($text);,删除或者注释掉这行代码,添加$givenText = iconv('gbk', 'utf-8', $text);,比如代码如下:

[php]  view plain copy print ?
  1. /**
  2. * Add a Text Element
  3. *
  4. * @param string $text
  5. * @param mixed $styleFont
  6. * @param mixed $styleParagraph
  7. * @return PHPWord_Section_Text
  8. */
  9. public function addText($text, $styleFont = null, $styleParagraph = null) {
  10. //$givenText = utf8_encode($text);
  11. $givenText = iconv('gbk', 'utf-8', $text); // 注释掉上面行后添加这行
  12. $text = new PHPWord_Section_Text($givenText, $styleFont, $styleParagraph);
  13. $this->_elementCollection[] = $text;
  14. return $text;
  15. }

调用方式和上面的模板调用大同小异,这边就不列举了。

折腾了这么多,突然发现网上还有另外一个版本的PhpWord,项目类名大小写上略有不同,隶属于PHPOffice/PHPWord,GitHub项目地址(文档)。这个版本的PHPWord内容更加丰富,支持的功能也比较多(包括行间距,缩进和首行缩进等),最后我也采取的这个版本的PHPWord,值得注意的是这两个版本的PHPWord在API接口上基本一致,可以通用。但是有些API,在PHPOffice/PHPWord里是不推荐的,比如createSection需要改成addSection,另外应用这个版本的PHPWord不需要像上面那样做任何中文支持的修改,比较省事。

这两个PHPWord项目的官方都提供了较详细的使用例子和文档,这里就不介绍了。最后提示的是:在模板模式下loadTemplate,只能使用setValue等模板操作方法,不能再添加段落或者段落修改了。这个略有不便。

参考文档:phpword 部分BUG修改笔记及心得

对于PHPOffice/PHPWord我提供一个简单的例子供参考(当然官方例子更多):

[php]  view plain copy print ?
  1. require_once 'PhpOffice/PhpWord/PhpWord.php'; // 包含头文件
  2. use PhpOffice\PhpWord\Autoloader;
  3. use PhpOffice\PhpWord\Settings;
  4. use PhpOffice\PhpWord\IOFactory;
  5. require_once __DIR__ . '/PhpOffice/PhpWord/Autoloader.php';
  6. Autoloader::register();
  7. Settings::loadConfig();
  8. // Create a new PHPWord Object
  9. $PHPWord = new \PhpOffice\PhpWord\PhpWord();
  10. $PHPWordHelper= new \PhpOffice\PhpWord\Shared\Font();
  11. $PHPWord->setDefaultFontName('仿宋'); // 全局字体
  12. $PHPWord->setDefaultFontSize(16);     // 全局字号为3号
  13. // 设置文档的属性,这些在对文档右击属性可以看到,也可以省去这些步骤
  14. $properties = $PHPWord->getDocumentProperties();
  15. $properties->setCreator('张三');   // 创建者
  16. $properties->setCompany('某公司'); // 公司
  17. $properties->setTitle('某某文档'); // 标题
  18. $properties->setDescription('http://wangye.org'); // 描述
  19. $properties->setLastModifiedBy('李四'); // 最后修改
  20. $properties->setCreated( time() );      // 创建时间
  21. $properties->setModified( time() );     // 修改时间
  22. // 添加3号仿宋字体到'FangSong16pt'留着下面使用
  23. $PHPWord->addFontStyle('FangSong16pt', array('name'=>'仿宋', 'size'=>16));
  24. // 添加段落样式到'Normal'以备下面使用
  25. $PHPWord->addParagraphStyle(
  26. 'Normal',array(
  27. 'align'=>'both',
  28. 'spaceBefore' => 0,
  29. 'spaceAfter' => 0,
  30. 'spacing'=>$PHPWordHelper->pointSizeToTwips(2.8),
  31. 'lineHeight' => 1.19,  // 行间距
  32. 'indentation' => array( // 首行缩进
  33. 'firstLine' => $PHPWordHelper->pointSizeToTwips(32)
  34. )
  35. )
  36. );
  37. // Section样式:上3.5厘米、下3.8厘米、左3厘米、右3厘米,页脚3厘米
  38. // 注意这里厘米(centimeter)要转换为twips单位
  39. $sectionStyle = array(
  40. 'orientation' => null,
  41. 'marginLeft' => $PHPWordHelper->centimeterSizeToTwips(3),
  42. 'marginRight' => $PHPWordHelper->centimeterSizeToTwips(3),
  43. 'marginTop' => $PHPWordHelper->centimeterSizeToTwips(3.5),
  44. 'marginBottom' => $PHPWordHelper->centimeterSizeToTwips(3.8),
  45. 'pageNumberingStart' => 1, // 页码从1开始
  46. 'footerHeight' => $PHPWordHelper->centimeterSizeToTwips(3),
  47. );
  48. $section = $PHPWord->addSection($sectionStyle); // 添加一节
  49. // 下面这句是输入文档内容,注意这里用到了刚才我们添加的
  50. // 字体样式FangSong16pt和段落样式Normal
  51. $section->addText('文档内容', 'FangSong16pt', 'Normal');
  52. $section->addTextBreak(1); // 新起一个空白段落
  53. $objWriter = IOFactory::createWriter($PHPWord, 'Word2007');
  54. $objWriter->save('/path/to/file'); // 保存到/path/to/file路径下

PHPWord利用模板替换字符串生成精确的word文档相关推荐

  1. 利用poi读取word模板文件生成新的word文档

    利用poi读取word模板文件生成新的word文档 利用poi读取word模板文件,并回填逻辑数据,生成并导出需要的word文档源码.解决模板读取异常问题,提供wordUtils工具类(各种功能实现) ...

  2. java word模板poi生成文件_利用poi读取word模板文件生成新的word文档

    利用poi读取word模板文件生成新的word文档 利用poi读取word模板文件,并回填逻辑数据,生成并导出需要的word文档源码.解决模板读取异常问题,提供wordUtils工具类(各种功能实现) ...

  3. 把Excel数据填充word模板生成多份word文档

    有些事情,你想记得的就会记得.有些事情,你想忘记的就会忘记,如果忘记不了,那就不要忘记了,因为忘记是不需要努力的. Model_Car.cs代码 public class Model_Car{publ ...

  4. Word邮件合并功能详解:合并后生成多个word文档,删除空白页

    Word邮件合并功能详解:合并后生成多个word文档,删除空白页 最近在实习,干了很多打杂得工作,所以office软件用的很多很多,瞬间觉得自己可以去裸考计算机二级了哈哈哈哈哈哈.今天因为工作用到了邮 ...

  5. 自定义maven插件:自动生成API的word文档

    继上次开发完Maven插件开发:根据库表生成实体类&根据实体类生成库表之后,博主对开发maven插件喜爱得一塌糊涂.这不,今天给大家带来了<自定义maven插件:自动生成API的word ...

  6. excel明细生成多个word文档,比邮件合并好用100倍。

    需求:把excel明细批量生成word文档,每条明细生成1个文档. 将上面的WIFI用户名.密码填到word文档中相应位置,并且需要插入图片. 通常用的是word邮件合并功能,但是步骤多,麻烦.插件简 ...

  7. Excel明细数据生成多个word文档,Excel魔方轻松完成

    场景: 我有这样一个明细表,需要按某列关键字,拆分到word文档中. word文档是下面的样子: 通常来说,如果只有型号.编号等数据,可以用邮件合并来完成. 但是,明细表有多对一的关系,也就是说,一个 ...

  8. springboot生成 .docx的word文档

    java springboot 通过word文档下载带有表格的.docx文件 try { //获取Word模板,模板存放路径在项目的resources目录下 InputStream ins = new ...

  9. ftl文件模板图片_使用Freemarker导出Word文档(包含图片)代码实现及总结

    本篇是关于利用FreeMarker导出Word的实现步骤. 优点:采用FreeMarker是导出Word的最佳实现,非常的灵活,能够按照自己指定的样式设置并输出内容,操作简单方便,代码实现也容易.代码 ...

最新文章

  1. Python爬虫破解有道翻译
  2. NetBeans 时事通讯(刊号 # 55 - May 06, 2009)
  3. python安装numpy-Python使用pip安装Numpy模块
  4. java jespa_Jespa实际运用的一点心得
  5. s3c6410 uboot代码分析《二》
  6. 随想录(学习nxp rt1052 soc)
  7. 【万里征程——Windows App开发】如何使用粘贴板
  8. 使用bind()和connect()函数
  9. (7)数据分析-秩和检验
  10. 驱动miniPCIE网络模块EC20硬件电路详解
  11. 字节跳动的第一场败仗:烧光20亿,悟空问答终落幕
  12. 实验吧——天网管理系统
  13. [转]ZBrush3官方中文教程 一
  14. 2019年Android GMS 认证 boot logo 最新要求
  15. 【Python零基础到入门】Python基础语法篇——数字(Number) 学习 【文末送书】
  16. 3DMAX软件可以运用到哪些行业?次世代游戏建模怎么样?
  17. Introduce·传播学核心期刊推荐之《现代传播(中国传媒大学学报)》
  18. 学人工智能有前途吗?人工智能前景-AI就业方向
  19. PostgreSQL集群方案-citus
  20. 八种酒吧里最时尚的喝酒法

热门文章

  1. 解决windows电脑蓝屏的方法
  2. 一起来学习Java浮点类型
  3. Minecraft 从入门到入坑(边玩游戏边学编程)
  4. Spring boot 事物管理
  5. Python应用 | 我喜欢看什么美剧(一)
  6. 可视计算机应用期末考试,职称计算机考试photoshop考试习题复习
  7. zcmu-5066: 黑暗长廊
  8. acer p3 171 装android,宏碁P3:平板PC一秒切换_Acer P3-171-3322Y2G06as_笔记本评测-中关村在线...
  9. Windows Mobile与symbian智能手机系统的比较
  10. 提莫攻击的中毒持续时间