因工作需要,使用了版本比较旧的PHPWord项目

官方已不见维护更新,上次版本更新是在Fri Jul 8, 2011 at 8:00 AM

如果PHP版本>=5.3.3,强烈推荐使用PHPOffice/PHPWord这个开源项目

本篇针对的为旧版本的PHPWord

基本安装

问题总结

Autoloader自动加载部分情况下失败

在使用Yii 1配置自动加载时无法正常加载类库,需对其PHPWord/Autoloader.php做部分调整,这儿借鉴了PHPExcel的Autoloader:

/**

* PHPWord_Autoloader

*/

class PHPWord_Autoloader

{

/**

* Register the Autoloader with SPL

*

*/

public static function Register() {

$functions = spl_autoload_functions();

foreach ( $functions as $function)

spl_autoload_unregister($function);

$functions = array_merge(array(array('PHPWord_Autoloader','Load')),$functions);

foreach ( $functions as $function)

$x = spl_autoload_register($function);

return $x;

} // function Register()

/**

* Autoload a class identified by name

*

* @param string $pClassName Name of the object to load

*/

public static function Load($pClassName){

if ((class_exists($pClassName,FALSE)) || (strpos($pClassName, 'PHPWord') !== 0)) {

// Either already loaded, or not a PHPWord class request

return FALSE;

}

$pClassFilePath = PHPWORD_BASE_PATH .

str_replace('_',DIRECTORY_SEPARATOR,$pClassName) .

'.php';

if ((file_exists($pClassFilePath) === FALSE) || (is_readable($pClassFilePath) === FALSE)) {

// Can't load

return FALSE;

}

require($pClassFilePath);

} // function Load()

}

模板替换时无法识别模板标签

表现

使用/复制官方样例的模板文件替换正常

自己手动敲出模板标签替换异常

原因

PHPWord的替换规则是将Word文件解析成XML进行替换处理,当Word解析成XML时字符分离了,导致匹配不上模板标签;

具体分析可参考一下资料:

解决办法

改进Template类:

可参考Github: Arisse/PHPWord_CloneRow对Template类进行改造。

因为下面仍需要修改Template类,这儿暂时就不贴代码了,下面一并贴出改造后的代码。

中文乱码

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

/**

* Set a Template value

*

* @param mixed $search

* @param mixed $replace

*/

public function setValue($search, $replace) {

if(substr($search, 0, 2) !== '${' && substr($search, -1) !== '}') {

$search = '${'.$search.'}';

}

if(!is_array($replace)) {

//$replace = utf8_encode($replace);

$replace =iconv('gbk', 'utf-8', $replace); // 注释掉上面行后添加这行

}

$this->_documentXML = str_replace($search, $replace, $this->_documentXML);

}

空格输出

在想要输出换行的地方用
代替即可.

标记符号输出

参考百度经验: 如何在word中选择打钩的方框

仅以输出□和☑为例,其它符号与之类似。

注:PHP文件需要使用UTF-8编码

在Word文件中按照参考文件方式插入☑;

复制符号到PHP文件;

正常的输出替换。

具体代码见如下的项目代码。

Template类代码

// code

/**

* Set a Template value

*

* @param mixed $search

* @param mixed $replace

*/

public function setValue($search, $replace, $limit=-1) {

if(substr($search, 0, 1) !== '{' && substr($search, -1) !== '}') {

$search = '{'.$search.'}';

}

if(!is_array($replace)) {

// $replace = utf8_encode($replace);

// $replace = iconv( 'gbk','utf-8', $replace);

$replace = str_replace("\n","
",$replace);

}

preg_match_all('/\{[^}]+\}/', $this->_documentXML, $matches);

foreach ($matches[0] as $k => $match) {

$no_tag = strip_tags($match);

if ($no_tag == $search) {

$match = '{'.$match.'}';

$this->_documentXML = preg_replace($match, $replace, $this->_documentXML, $limit);

if ($limit == 1) {

break;

}

}

}

}

// code

项目代码

// @author Heier xheier@outlook.com

public function actionExportPersonTable() {

// 获取数据部分代码

// ...

$PHPWord = new PHPWord();

// Word模板目录

$personBasePath = Yii::app()->basePath.'/person/';

// 删除目录下临时文件-十分钟以前

$this->delfile( $personBasePath, 10 );

// 模板文件名

$tempName = $personBasePath . '/moban.docx';

$word = $PHPWord->loadTemplate( $tempName );

// 项目使用的是GBK编码,需要做转换

$username = iconv('gbk', 'utf-8', getUserNameById($personData[0]['user_id']) );

$personal_type = $personData[0]['personal_type'];

// 模板替换开始

// 可以输出打勾的方框

$deptA=$deptBP=$deptB=$deptC=$deptD = '□';

if( $DirectorLevel == 'A' ) {

$deptA = '☑';

} elseif( $DirectorLevel == 'B+' ) {

$deptBP = '☑';

} elseif( $DirectorLevel == 'B' ) {

$deptB = '☑';

} elseif( $DirectorLevel == 'C' ) {

$deptC = '☑';

} elseif( $DirectorLevel == 'D' ) {

$deptD = '☑';

}

$word->setValue('deptA', $deptA);

$word->setValue('deptBP', $deptBP);

$word->setValue('deptB', $deptB);

$word->setValue('deptC', $deptC);

$word->setValue('deptD', $deptD);

// 设置其它替换

// ...

// 生成临时文件以供下载

$tmpFileName = md5( time().'Heier' );

$word->save($personBasePath . '/' . $tmpFileName .'.docx');

$file = $personBasePath . '/' . $tmpFileName .'.docx';

// 下载Word文件

ob_start(); //打开缓冲区

$fp = fopen($file,"r");

$file_size = filesize($file);

$downFileName = 'XXX.docx';

header("Cache-Control: public");

header("Content-type: application/octet-stream");

header("Accept-Ranges: bytes");

header("Content-Disposition: attachment; filename={$downFileName}");

header("Pragma:no-cache");

header("Expires:0");

$buffer = 1024;

$file_count = 0;

//向浏览输出回数据

while(!feof($fp) && $file_count < $file_size){

$file_con = fread($fp,$buffer);

$file_count += $buffer;

echo $file_con;

}

ob_end_flush();//输出全部内容到浏览器

}

参考文档汇总

关于我

php 模板替换,使用PHPWord对Word文件做模板替换相关推荐

  1. php读取word模板文件,使用PHPWord对Word文件做模板替换

    文章排版有点乱,建议点击左下角的"阅读原文"查看. 因工作需要,使用了版本比较旧的 PHPWord项目 官方已不见维护更新,上次版本更新是在 Fri Jul 8, 2011 at ...

  2. (五)、JAVA基于OPENXML的word文档插入、合并、替换操作系列之word文件合并[支持多文件]

    (五).JAVA基于OPENXML的word文档插入.合并.替换操作系列之word文件合并[支持多文件] 二.word合并的多种方案简单比较 三.基于Open Xml WordprocessingML ...

  3. java全文检索word中的内容_对服务器上所有Word文件做全文检索的解决方案-Java

    一.背景介绍 Word文档与日常办公密不可分,在实际应用中,当某一文档服务器中有很多Word文档,假如有成千上万个文档时,用户查找打开包含某些指定关键字的文档就变得很困难,目前这一问题没有好的解决方案 ...

  4. Java使用Word的模板引擎 Poi-tl操控导出word文件

    Poi-tl介绍 poi-tl是一个基于Apache POI的Java库,用于操作Microsoft Office文档,包括Word文档(.docx).Excel电子表格(.xlsx)和PowerPo ...

  5. 微软crm在哪新建审批模板_如何在Microsoft Word中创建模板

    微软crm在哪新建审批模板 Templates let you configure all the relevant settings you want pre-applied to document ...

  6. word文件做一半未响应_Win7的word为什么老是出现未响应?

    展开全部 1.按Win+R键,打开运行,输入winWord /safe,点击确定进入无加载项word; 2.进入word界面,点击文件; 3.点62616964757a686964616fe4b893 ...

  7. php代码实现对word文件的查找与替换,ThinkPHP5使用phpword实现文件模板字符替换

    PhpWord文档 想实现更多功能的朋友可以仔细参考文档 一.使用环境 目前是在ThinkPHP5.0.21中使用的PhpWord,其他使用环境引入方式可能会有一点不同. 使用composer安装,直 ...

  8. Java:使用POI实现word的docx文件的模板功能

    一:场景  通过Word模板来实现动态的word生成 二: 基本要求  1:替换文本中的内容  2:替换表格中的内容(不用动态生成表格)  3:替换后的内容应该与替换前的内容格式相同  4:模板修改方 ...

  9. java word 模板_java根据模板生成word文件

    JAVA生成word模板程序步骤 1. 将freemarker-2.3.13.jar复制到项目\WEB-INF\lib目录下 2. 编辑模板文件 (1) 将DOC文件另存为xml文件,将xml文件在e ...

最新文章

  1. 允许使用抽象类类型 isearchboxinfo 的对象_Java学习5-设计模式+抽象类/方法
  2. Envi 4.7 破解安装及下载(转)
  3. selenium python自动化测试 ddt数据驱动
  4. Linux磁盘管理:LVM逻辑卷管理
  5. JSP中文乱码解决方案了解和TOMCAT中文乱码解决
  6. oracle逗号隔开行转列_oralce逗号分割变多行-Oracle
  7. 209. 长度最小的子数组(中等 数组 滑动窗口)
  8. el-input 纯数字输入 限制长度 限制最大值方法
  9. GT性能测试Android版使用说明
  10. Linux最常用的关机命令介绍!
  11. c语言求幸运数字程序,算法题挑选幸运数字,该如何处理
  12. 解读广告投放效果数据
  13. android 酷狗音乐 ip,“音乐+IP”融合模式 夯实酷狗音乐原创硬实力
  14. python爬虫简单实例-爬取17K小说网小说
  15. 文本检测之DBNet,DBNet++
  16. 手把手教你读财报----银行业---第十四课
  17. Hadoop Single Node Setup(hadoop本地模式和伪分布式模式安装-官方文档翻译 2.7.3)
  18. 地砖中间高四边低_地砖留缝多少适合 主要是由这4种要素决策的
  19. consul--基础--04--命令
  20. 猎头推荐成功一个人竟然收年薪一半,也太赚了吧

热门文章

  1. 私有云方案——利用阿里云云解析实现DDNS
  2. ASP.NET Core 使用UrlFirewall对请求进行过滤
  3. 十二个 ASP.NET Core 例子
  4. StackExchange.Redis客户端读写主从配置,以及哨兵配置
  5. 用 Visual Studio Code 在 macOS 上创建首个 ASP.NET Core 应用程序
  6. [.NET Core].NET Core R2安装及示例教程
  7. django12:form 组件/渲染标签/数据校验/钩子函数/
  8. “Visual Studio 启动不能打开上次打开的文件” 最正确的解决姿势
  9. [转]2020年5月程序员工资统计,平均14542元
  10. Visual Studio listView控件绑定SQL Server数据库并动态显示数据,调整列宽