|
最近一個(gè)項(xiàng)目開發(fā)要用到php技術(shù)導(dǎo)出Word文檔,比較了幾種方案,首先是使用Microsoft Office自帶的ActiveX/COM組件,比如Word.Application,這種方式的優(yōu)點(diǎn)是格式兼容度高,可以生成純doc的Word2003格式文檔,缺點(diǎn)一是比較占資源(調(diào)用會(huì)啟動(dòng)一個(gè)WINWORD.EXE進(jìn)程),不適合Web多用戶訪問(wèn)使用;二是php這種Web開發(fā)技術(shù)大多數(shù)是跑在Linux服務(wù)器上,當(dāng)然也就無(wú)法使用Windows下的技術(shù)了,平臺(tái)可移植和兼容性不好。
第二種生成Word的方案是生成Word兼容的網(wǎng)頁(yè)格式,然后以Word方式打開,這種方案總體上感覺怪怪的,畢竟文件格式是HTML的,而且格式兼容度不好,不過(guò)這種方式的優(yōu)點(diǎn)是節(jié)省服務(wù)器資源,能夠快速生成;最后一種方案也就是今天的主角,采用phpWord生成Word2007(docx)格式的文檔,現(xiàn)在基本上微軟Office Word 2003以后的版本均兼容這種格式了,對(duì)于2003版本來(lái)說(shuō),僅需要下載安裝個(gè)兼容格式包(下載地址),也能正常打開這類文件,當(dāng)然如果你使用的是最新版本的Office(包括但不限于Office 2007、Office 2010)則不需要安裝此格式包。
好了,下面我就介紹一下phpWord,大家可以通過(guò)訪問(wèn)項(xiàng)目主頁(yè)下載并獲得關(guān)于項(xiàng)目的更多信息。
我在使用過(guò)程中主要遇到了中文亂碼的問(wèn)題,結(jié)合網(wǎng)上大神們的指導(dǎo),通過(guò)下面的方式解決了這類問(wèn)題,希望對(duì)大家有所幫助。
1、增加?xùn)|亞字體支持
打開并編輯路徑/Writer/Word2007/Base.php文件內(nèi)容,大概在第349行(行數(shù)隨著版本可能會(huì)有變化)大概函數(shù)_writeTextStyle內(nèi)添加:
$objWriter->writeAttribute('w:eastAsia', $font)
比如我的修改片段基本是下面這樣:
// Fontif($font != 'Arial') { $objWriter->startElement('w:rFonts'); $objWriter->writeAttribute('w:eastAsia', $font); // 添加這行 $objWriter->writeAttribute('w:ascii', $font); $objWriter->writeAttribute('w:hAnsi', $font); $objWriter->writeAttribute('w:cs', $font); $objWriter->endElement();}
2. 解決中文亂碼問(wèn)題
編輯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);}
調(diào)用方式如下:
$document->setValue('Template', iconv('utf-8', 'GB2312//IGNORE', '中文'));
上面的代碼主要解決模板的問(wèn)題,下面同樣的道理,解決Section添加文本的問(wèn)題,找到代碼$givenText = utf8_encode($text);,刪除或者注釋掉這行代碼,添加$givenText = iconv('gbk', 'utf-8', $text);,比如代碼如下:
/** * Add a Text Element * * @param string $text * @param mixed $styleFont * @param mixed $styleParagraph * @return phpWord_Section_Text */public function addText($text, $styleFont = null, $styleParagraph = null) { //$givenText = utf8_encode($text); $givenText = iconv('gbk', 'utf-8', $text); // 注釋掉上面行后添加這行 $text = new phpWord_Section_Text($givenText, $styleFont, $styleParagraph); $this->_elementCollection[] = $text; return $text;}
調(diào)用方式和上面的模板調(diào)用大同小異,這邊就不列舉了。
折騰了這么多,突然發(fā)現(xiàn)網(wǎng)上還有另外一個(gè)版本的phpWord,項(xiàng)目類名大小寫上略有不同,隸屬于phpOffice/phpWord,GitHub項(xiàng)目地址(文檔)。這個(gè)版本的phpWord內(nèi)容更加豐富,支持的功能也比較多(包括行間距,縮進(jìn)和首行縮進(jìn)等),最后我也采取的這個(gè)版本的phpWord,值得注意的是這兩個(gè)版本的phpWord在API接口上基本一致,可以通用。但是有些API,在phpOffice/phpWord里是不推薦的,比如createSection需要改成addSection,另外應(yīng)用這個(gè)版本的phpWord不需要像上面那樣做任何中文支持的修改,比較省事。
這兩個(gè)phpWord項(xiàng)目的官方都提供了較詳細(xì)的使用例子和文檔,這里就不介紹了。最后提示的是:在模板模式下loadTemplate,只能使用setValue等模板操作方法,不能再添加段落或者段落修改了。這個(gè)略有不便。
對(duì)于phpOffice/phpWord我提供一個(gè)簡(jiǎn)單的例子供參考(當(dāng)然官方例子更多):
require_once 'phpOffice/phpWord/phpWord.php'; // 包含頭文件use phpOffice\phpWord\Autoloader;use phpOffice\phpWord\Settings;use phpOffice\phpWord\IOFactory; require_once __DIR__ . '/phpOffice/phpWord/Autoloader.php';Autoloader::register();Settings::loadConfig(); // Create a new phpWord Object$phpWord = new \phpOffice\phpWord\phpWord();$phpWordHelper= new \phpOffice\phpWord\Shared\Font(); $phpWord->setDefaultFontName('仿宋'); // 全局字體$phpWord->setDefaultFontSize(16); // 全局字號(hào)為3號(hào) // 設(shè)置文檔的屬性,這些在對(duì)文檔右擊屬性可以看到,也可以省去這些步驟$properties = $phpWord->getDocumentProperties();$properties->setCreator('張三'); // 創(chuàng)建者$properties->setCompany('某公司'); // 公司$properties->setTitle('某某文檔'); // 標(biāo)題$properties->setDescription('http://wangye.org'); // 描述$properties->setLastModifiedBy('李四'); // 最后修改$properties->setCreated( time() ); // 創(chuàng)建時(shí)間$properties->setModified( time() ); // 修改時(shí)間 // 添加3號(hào)仿宋字體到'FangSong16pt'留著下面使用$phpWord->addFontStyle('FangSong16pt', array('name'=>'仿宋', 'size'=>16)); // 添加段落樣式到'Normal'以備下面使用$phpWord->addParagraphStyle( 'Normal',array( 'align'=>'both', 'spaceBefore' => 0, 'spaceAfter' => 0, 'spacing'=>$phpWordHelper->pointSizeToTwips(2.8), 'lineHeight' => 1.19, // 行間距 'indentation' => array( // 首行縮進(jìn) 'firstLine' => $phpWordHelper->pointSizeToTwips(32) ) )); // Section樣式:上3.5厘米、下3.8厘米、左3厘米、右3厘米,頁(yè)腳3厘米// 注意這里厘米(centimeter)要轉(zhuǎn)換為twips單位$sectionStyle = array( 'orientation' => null, 'marginLeft' => $phpWordHelper->centimeterSizeToTwips(3), 'marginRight' => $phpWordHelper->centimeterSizeToTwips(3), 'marginTop' => $phpWordHelper->centimeterSizeToTwips(3.5), 'marginBottom' => $phpWordHelper->centimeterSizeToTwips(3.8), 'pageNumberingStart' => 1, // 頁(yè)碼從1開始 'footerHeight' => $phpWordHelper->centimeterSizeToTwips(3),); $section = $phpWord->addSection($sectionStyle); // 添加一節(jié) // 下面這句是輸入文檔內(nèi)容,注意這里用到了剛才我們添加的// 字體樣式FangSong16pt和段落樣式Normal$section->addText('文檔內(nèi)容', 'FangSong16pt', 'Normal');$section->addTextBreak(1); // 新起一個(gè)空白段落 $objWriter = IOFactory::createWriter($phpWord, 'Word2007');$objWriter->save('/path/to/file'); // 保存到/path/to/file路徑下
總結(jié)
1、用模板word生成word中文亂碼解決方案:打開phpword/Template.php文件,找到$replace = utf8_encode($replace);將其改為$replace =iconv('gbk', 'utf-8', $replace); 即可。
2、直接生成word文檔,調(diào)用addText對(duì)象時(shí)中文亂碼解決方案:打開phpword/Section.php文件,找到$givenText = utf8_encode($text);將其改為$givenText = iconv('gbk', 'utf-8', $text);即可。
3、貌似其他方法也類似第解決。
4、注意php文件采用gbk哦。反正我的顯示中文了。在網(wǎng)上找了好久,研究了半天才搞定。
php技術(shù):phpword插件導(dǎo)出word文件時(shí)中文亂碼問(wèn)題處理方案,轉(zhuǎn)載需保留來(lái)源!
鄭重聲明:本文版權(quán)歸原作者所有,轉(zhuǎn)載文章僅為傳播更多信息之目的,如作者信息標(biāo)記有誤,請(qǐng)第一時(shí)間聯(lián)系我們修改或刪除,多謝。