使用php在win下生成chm文档

jerry 2015年11月18日 收藏
一个类和hhc.exe还有hha.dll
用于生成包含html目录的chm项目文件 ,然后通过hhp项目文件和内容文件生成 .chm手册。
<?php
    /* 函数 listDirTree( $dirName = null )
    ** 功能 列出目录下所有文件及子目录
    ** 参数 $dirName 目录名称
    ** 返回 目录结构数组 false为失败
    */

    function listDir($dirName = null) {
        if (empty($dirName))
            exit("IBFileSystem: directory is empty.");
        if (is_dir($dirName)) {
            if ($dh = opendir($dirName)) {
                $tree = array();
                while (( $file = readdir($dh) ) !== false) {
                    if ($file != "." && $file != "..") {
                        $filePath = $dirName . DIRECTORY_SEPARATOR . $file;
                        if (is_dir($filePath)) { //为目录,递归
                            $tree2 =listDir($filePath);
                            $tree = $tree2? array_merge($tree,$tree2):$tree;
                        } else { //为文件,添加到当前数组
                            $tree[] = $filePath;
                        }
                    }
                }
                closedir($dh);
            } else {
                exit("IBFileSystem: can not open directory $dirName.");
            }

            //返回当前的$tree
            $tree = array_unique($tree);
            natsort($tree);
            return $tree;
        } else {
            exit("IBFileSystem: $dirName is not a directory.");
        }
    }

    function listDirTree($dirName = null,$remove) {
        if (empty($dirName))
            exit("IBFileSystem: directory is empty.");
        if (is_dir($dirName)) {
            if ($dh = opendir($dirName)) {
                $tree = array();
                while (( $file = readdir($dh) ) !== false) {
                    if ($file != "." && $file != ".." && stripos($remove, $file) === false) {
                        $filePath = $dirName . DIRECTORY_SEPARATOR . $file;
                        if (is_dir($filePath)) { //为目录,递归
                            $arr = listDirTree($filePath,$remove);
                            natsort($arr);
                            $tree[$file] = $arr;
                        } else { //为文件,添加到当前数组
                            $tree[] = $filePath;
                        }
                    }
                }
                closedir($dh);
            } else {
                exit("IBFileSystem: can not open directory $dirName.");
            }
            
            //返回当前的$tree
            return $tree;
        } else {
            exit("IBFileSystem: $dirName is not a directory.");
        }
    }

    function cmp($a,$b){
        $a = (int)$a;
        $b = (int)$b;
        if($a == $b)    return 0;
        return ($a>$b)? 1:-1;
    }


    class chmBuilder{
        // const version = 0.1;
        public $chm_name;
        public $chm_path;
        public $chm_hhp;
        public $chm_hhc;
        public $chm_hhk;
        public $chm_uninclude_dirs;
        public $chm_uninclude_files;
        public $chm_image_type;
        public $chm_first_open;
        public $chm_title;

        public function __construct($chm_name='your_chm',$chm_path='',$chm_uninclude_dirs,$chm_uninclude_files){
            $this->chm_name = $chm_name;
            $this->chm_path = $chm_path;
            $this->chm_uninclude_dirs = $chm_uninclude_dirs;
            $this->chm_uninclude_files = $chm_uninclude_files;
            $this->chm_image_type = 'Folder';
        }

        public function build(){
            $this->buildHhp();
            $this->buildHhc();
            $this->buildHhk();
        }

        public function buildHhp(){
            $manual_files = listDir($this->chm_path);
            $files = implode(PHP_EOL, $manual_files);
            $this->chm_first_open = iconv('UTF-8', 'GB2312', $this->chm_first_open);
            $this->chm_title = iconv('UTF-8', 'GB2312', $this->chm_title);
            $tpl = <<<eof
[OPTIONS]
Compatibility=1.1 or later
Compiled file={$this->chm_path}/{$this->chm_name}.chm
Contents file={$this->chm_hhc}.hhc
COPYRIGHT=www.thinkphp.cn
Display compile progress=No
Default topic={$this->chm_first_open}
Error log file=chm_builder.Log
Full-text search=Yes
Index file={$this->chm_hhk}.hhk
ImageType={$this->chm_image_type}
Language=0x804
Title={$this->chm_title}

[FILES]
{$files}
eof;
            file_put_contents("{$this->chm_path}/{$this->chm_hhp}.hhp", $tpl);
        }

        public function buildHhc(){
            $list = array();
            $file_tree = listDirTree($this->chm_path,"{$this->chm_hhp} {$this->chm_uninclude_dirs}{$this->chm_uninclude_files}");
            uksort($file_tree, 'cmp');
            foreach ($file_tree as $key => $value) {
                if(is_string($value)){
                    $title = explode(DIRECTORY_SEPARATOR, $value);
                    $title = array_pop($title);
                    $title = rtrim($title,'.html');
                    $list[] = <<<eof
    <LI><OBJECT type="text/sitemap">
        <param name="Name" value="{$title}">
        <param name="Local" value="{$value}">
        </OBJECT>
eof;
                }else{
                    $child = array();
                    foreach ($value as $k => $val) {
                        $title = explode(DIRECTORY_SEPARATOR, $val);
                        $title = array_pop($title);
                        $title = rtrim($title,'.html');
                        $child[] = <<<eof
        <LI><OBJECT type="text/sitemap">
            <param name="Name" value="{$title}">
            <param name="Local" value="{$val}">
            <param name="ImageNumber" value="9">
            </OBJECT>
eof;
                    }
                    $child = implode(PHP_EOL, $child);
                    $list[] = <<<eof
    <LI> <OBJECT type="text/sitemap">
        <param name="Name" value="{$key}">
        <param name="ImageNumber" value="1">
        </OBJECT>
    <UL>    
{$child}
    </UL>    
eof;
                }
            }
            $list = implode(PHP_EOL, $list);
            $tpl = <<<eof
<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML//EN">
<HTML>
<HEAD>
<meta name="GENERATOR" content="yangweijie code-tech.diandian.com">
<!-- Sitemap 1.0 -->
</HEAD><BODY>
<OBJECT type="text/site properties">
    <param name="ExWindow Styles" value="0x200">
    <param name="Window Styles" value="0x800025">
    <param name="Font" value="MS Sans Serif,10,0">
</OBJECT>
<UL>
{$list}
</UL>
</BODY></HTML>
eof;
            file_put_contents("{$this->chm_path}/{$this->chm_hhc}.hhc", $tpl);
        }

        public function buildHhk(){
            $list = array();
            $file_tree = listDir($this->chm_path);
            foreach ($file_tree as $key => $value) {
                if(is_string($value)){
                    if(stripos($value, '.html')){
                        $title = explode(DIRECTORY_SEPARATOR, $value);
                        $title = array_pop($title);
                        $title = rtrim($title,'.html');
                        $list[] = <<<eof
    <LI><OBJECT type="text/sitemap">
        <param name="Name" value="{$title}">
        <param name="Local" value="{$value}">
        </OBJECT>
eof;
                    }
                }
            }
            $list = implode(PHP_EOL, $list);
            $tpl = <<<eof
<!DOCTYPE HTML PUBLIC "-//IETF//DTD HTML//EN">
<HTML>
<HEAD>
<meta name="GENERATOR" content="yangweijie code-tech.diandian.com">
<!-- Sitemap 1.0 -->
</HEAD><BODY>
<UL>
{$list}
</UL>
</BODY></HTML>
eof;
            file_put_contents("{$this->chm_path}/{$this->chm_hhk}.hhk", $tpl);
        }

        public function makeChm(){
            if(!is_file("{$this->chm_path}/{$this->chm_hhp}.hhp"))    
                return "build error:can't generate *.hhp file!";
            $command = "hhc {$this->chm_path}/{$this->chm_hhp}.hhp";
            system($command);
            if(file_exists("{$this->chm_path}/{$this->chm_name}.chm"))
                return true;
            else
                return 'generate chm failed!';
        }
    }

?>
使用方法,放到要生成目录的外面 定义好路径,手册名,不包含目录,不包含文件 字符串(空格分割),设置好一些属性后, 将hhc.exe的位置加入环境变量path中,cmd 里调用 执行的index.php 可以看到生成的信息,或者错误
index.php
<?php
    header('Content-type:text/plain;charset=utf-8');
    error_reporting(E_ERROR);
    ini_set('memory_limit', '30M');
    include 'chm_builder.php';
    $chm = new chmBuilder('ThinkPHP manual',__DIR__.DIRECTORY_SEPARATOR.'manual','public ','.DS_Store Thumbs.db book.tpl');
    $chm->chm_hhp = 'index';
    $chm->chm_hhc = 'index';
    $chm->chm_first_open = $chm->chm_path.DIRECTORY_SEPARATOR.'序言.html';
    $chm->chm_hhk = 'index';
    $chm->chm_title = 'ThinkPHP 3.1.2官方手册';
    $chm->build();
    //$chm->makeChm();
?>
这个可以配合ThinkPHP Sublime 插件来生成手册,目前排序方面有点问题,故没集成到插件里去。目前只支持二级分类,多级的大家递归时tab缩进好个是就行了,用手册里第一层目录和单文件名作为章节,里面的文件作为子章节