|  | /* | 
|  | * Copyright 2007 Jacek Caban for CodeWeavers | 
|  | * Copyright 2010 Erich Hoover | 
|  | * | 
|  | * This library is free software; you can redistribute it and/or | 
|  | * modify it under the terms of the GNU Lesser General Public | 
|  | * License as published by the Free Software Foundation; either | 
|  | * version 2.1 of the License, or (at your option) any later version. | 
|  | * | 
|  | * This library is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU | 
|  | * Lesser General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU Lesser General Public | 
|  | * License along with this library; if not, write to the Free Software | 
|  | * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
|  | */ | 
|  |  | 
|  | #define NONAMELESSUNION | 
|  | #define NONAMELESSSTRUCT | 
|  |  | 
|  | #include "hhctrl.h" | 
|  | #include "stream.h" | 
|  |  | 
|  | #include "wine/debug.h" | 
|  |  | 
|  | WINE_DEFAULT_DEBUG_CHANNEL(htmlhelp); | 
|  |  | 
|  | /* Fill the TreeView object corresponding to the Index items */ | 
|  | static void fill_index_tree(HWND hwnd, IndexItem *item) | 
|  | { | 
|  | int index = 0; | 
|  | LVITEMW lvi; | 
|  |  | 
|  | while(item) { | 
|  | TRACE("tree debug: %s\n", debugstr_w(item->keyword)); | 
|  |  | 
|  | if(!item->keyword) | 
|  | { | 
|  | FIXME("HTML Help index item has no keyword.\n"); | 
|  | item = item->next; | 
|  | continue; | 
|  | } | 
|  | memset(&lvi, 0, sizeof(lvi)); | 
|  | lvi.iItem = index++; | 
|  | lvi.mask = LVIF_TEXT|LVIF_PARAM|LVIF_INDENT; | 
|  | lvi.iIndent = item->indentLevel; | 
|  | lvi.cchTextMax = strlenW(item->keyword)+1; | 
|  | lvi.pszText = item->keyword; | 
|  | lvi.lParam = (LPARAM)item; | 
|  | item->id = (HTREEITEM)SendMessageW(hwnd, LVM_INSERTITEMW, 0, (LPARAM)&lvi); | 
|  | item = item->next; | 
|  | } | 
|  | } | 
|  |  | 
|  | /* Parse the attributes correspond to a list item, including sub-topics. | 
|  | * | 
|  | * Each list item has, at minimum, a param of type "keyword" and two | 
|  | * parameters corresponding to a "sub-topic."  For each sub-topic there | 
|  | * must be a "name" param and a "local" param, if there is only one | 
|  | * sub-topic then there isn't really a sub-topic, the index will jump | 
|  | * directly to the requested item. | 
|  | */ | 
|  | static void parse_index_obj_node_param(IndexItem *item, const char *text) | 
|  | { | 
|  | const char *ptr; | 
|  | LPWSTR *param; | 
|  | int len, wlen; | 
|  |  | 
|  | ptr = get_attr(text, "name", &len); | 
|  | if(!ptr) { | 
|  | WARN("name attr not found\n"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | /* Allocate a new sub-item, either on the first run or whenever a | 
|  | * sub-topic has filled out both the "name" and "local" params. | 
|  | */ | 
|  | if(item->itemFlags == 0x11 && (!strncasecmp("name", ptr, len) || !strncasecmp("local", ptr, len))) { | 
|  | item->nItems++; | 
|  | item->items = heap_realloc(item->items, sizeof(IndexSubItem)*item->nItems); | 
|  | item->items[item->nItems-1].name = NULL; | 
|  | item->items[item->nItems-1].local = NULL; | 
|  | item->itemFlags = 0x00; | 
|  | } | 
|  | if(!strncasecmp("keyword", ptr, len)) { | 
|  | param = &item->keyword; | 
|  | }else if(!item->keyword && !strncasecmp("name", ptr, len)) { | 
|  | /* Some HTML Help index files use an additional "name" parameter | 
|  | * rather than the "keyword" parameter.  In this case, the first | 
|  | * occurrence of the "name" parameter is the keyword. | 
|  | */ | 
|  | param = &item->keyword; | 
|  | }else if(!strncasecmp("name", ptr, len)) { | 
|  | item->itemFlags |= 0x01; | 
|  | param = &item->items[item->nItems-1].name; | 
|  | }else if(!strncasecmp("local", ptr, len)) { | 
|  | item->itemFlags |= 0x10; | 
|  | param = &item->items[item->nItems-1].local; | 
|  | }else { | 
|  | WARN("unhandled param %s\n", debugstr_an(ptr, len)); | 
|  | return; | 
|  | } | 
|  |  | 
|  | ptr = get_attr(text, "value", &len); | 
|  | if(!ptr) { | 
|  | WARN("value attr not found\n"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | wlen = MultiByteToWideChar(CP_ACP, 0, ptr, len, NULL, 0); | 
|  | *param = heap_alloc((wlen+1)*sizeof(WCHAR)); | 
|  | MultiByteToWideChar(CP_ACP, 0, ptr, len, *param, wlen); | 
|  | (*param)[wlen] = 0; | 
|  | } | 
|  |  | 
|  | /* Parse the object tag corresponding to a list item. | 
|  | * | 
|  | * At this step we look for all of the "param" child tags, using this information | 
|  | * to build up the information about the list item.  When we reach the </object> | 
|  | * tag we know that we've finished parsing this list item. | 
|  | */ | 
|  | static IndexItem *parse_index_sitemap_object(HHInfo *info, stream_t *stream) | 
|  | { | 
|  | strbuf_t node, node_name; | 
|  | IndexItem *item; | 
|  |  | 
|  | strbuf_init(&node); | 
|  | strbuf_init(&node_name); | 
|  |  | 
|  | item = heap_alloc_zero(sizeof(IndexItem)); | 
|  | item->nItems = 0; | 
|  | item->items = heap_alloc_zero(0); | 
|  | item->itemFlags = 0x11; | 
|  |  | 
|  | while(next_node(stream, &node)) { | 
|  | get_node_name(&node, &node_name); | 
|  |  | 
|  | TRACE("%s\n", node.buf); | 
|  |  | 
|  | if(!strcasecmp(node_name.buf, "param")) { | 
|  | parse_index_obj_node_param(item, node.buf); | 
|  | }else if(!strcasecmp(node_name.buf, "/object")) { | 
|  | break; | 
|  | }else { | 
|  | WARN("Unhandled tag! %s\n", node_name.buf); | 
|  | } | 
|  |  | 
|  | strbuf_zero(&node); | 
|  | } | 
|  |  | 
|  | strbuf_free(&node); | 
|  | strbuf_free(&node_name); | 
|  |  | 
|  | return item; | 
|  | } | 
|  |  | 
|  | /* Parse the HTML list item node corresponding to a specific help entry. | 
|  | * | 
|  | * At this stage we look for the only child tag we expect to find under | 
|  | * the list item: the <OBJECT> tag.  We also only expect to find object | 
|  | * tags with the "type" attribute set to "text/sitemap". | 
|  | */ | 
|  | static IndexItem *parse_li(HHInfo *info, stream_t *stream) | 
|  | { | 
|  | strbuf_t node, node_name; | 
|  | IndexItem *ret = NULL; | 
|  |  | 
|  | strbuf_init(&node); | 
|  | strbuf_init(&node_name); | 
|  |  | 
|  | while(next_node(stream, &node)) { | 
|  | get_node_name(&node, &node_name); | 
|  |  | 
|  | TRACE("%s\n", node.buf); | 
|  |  | 
|  | if(!strcasecmp(node_name.buf, "object")) { | 
|  | const char *ptr; | 
|  | int len; | 
|  |  | 
|  | static const char sz_text_sitemap[] = "text/sitemap"; | 
|  |  | 
|  | ptr = get_attr(node.buf, "type", &len); | 
|  |  | 
|  | if(ptr && len == sizeof(sz_text_sitemap)-1 | 
|  | && !memcmp(ptr, sz_text_sitemap, len)) { | 
|  | ret = parse_index_sitemap_object(info, stream); | 
|  | break; | 
|  | } | 
|  | }else { | 
|  | WARN("Unhandled tag! %s\n", node_name.buf); | 
|  | } | 
|  |  | 
|  | strbuf_zero(&node); | 
|  | } | 
|  |  | 
|  | strbuf_free(&node); | 
|  | strbuf_free(&node_name); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | /* Parse the HTML Help page corresponding to all of the Index items. | 
|  | * | 
|  | * At this high-level stage we locate out each HTML list item tag. | 
|  | * Since there is no end-tag for the <LI> item, we must hope that | 
|  | * the <LI> entry is parsed correctly or tags might get lost. | 
|  | * | 
|  | * Within each entry it is also possible to encounter an additional | 
|  | * <UL> tag.  When this occurs the tag indicates that the topics | 
|  | * contained within it are related to the parent <LI> topic and | 
|  | * should be inset by an indent. | 
|  | */ | 
|  | static void parse_hhindex(HHInfo *info, IStream *str, IndexItem *item) | 
|  | { | 
|  | stream_t stream; | 
|  | strbuf_t node, node_name; | 
|  | int indent_level = -1; | 
|  |  | 
|  | strbuf_init(&node); | 
|  | strbuf_init(&node_name); | 
|  |  | 
|  | stream_init(&stream, str); | 
|  |  | 
|  | while(next_node(&stream, &node)) { | 
|  | get_node_name(&node, &node_name); | 
|  |  | 
|  | TRACE("%s\n", node.buf); | 
|  |  | 
|  | if(!strcasecmp(node_name.buf, "li")) { | 
|  | item->next = parse_li(info, &stream); | 
|  | item->next->merge = item->merge; | 
|  | item = item->next; | 
|  | item->indentLevel = indent_level; | 
|  | }else if(!strcasecmp(node_name.buf, "ul")) { | 
|  | indent_level++; | 
|  | }else if(!strcasecmp(node_name.buf, "/ul")) { | 
|  | indent_level--; | 
|  | }else { | 
|  | WARN("Unhandled tag! %s\n", node_name.buf); | 
|  | } | 
|  |  | 
|  | strbuf_zero(&node); | 
|  | } | 
|  |  | 
|  | strbuf_free(&node); | 
|  | strbuf_free(&node_name); | 
|  | } | 
|  |  | 
|  | /* Initialize the HTML Help Index tab */ | 
|  | void InitIndex(HHInfo *info) | 
|  | { | 
|  | IStream *stream; | 
|  |  | 
|  | info->index = heap_alloc_zero(sizeof(IndexItem)); | 
|  | info->index->nItems = 0; | 
|  | SetChmPath(&info->index->merge, info->pCHMInfo->szFile, info->WinType.pszIndex); | 
|  |  | 
|  | stream = GetChmStream(info->pCHMInfo, info->pCHMInfo->szFile, &info->index->merge); | 
|  | if(!stream) { | 
|  | TRACE("Could not get index stream\n"); | 
|  | return; | 
|  | } | 
|  |  | 
|  | parse_hhindex(info, stream, info->index); | 
|  | IStream_Release(stream); | 
|  |  | 
|  | fill_index_tree(info->tabs[TAB_INDEX].hwnd, info->index->next); | 
|  | } | 
|  |  | 
|  | /* Free all of the Index items, including all of the "sub-items" that | 
|  | * correspond to different sub-topics. | 
|  | */ | 
|  | void ReleaseIndex(HHInfo *info) | 
|  | { | 
|  | IndexItem *item = info->index, *next; | 
|  | int i; | 
|  |  | 
|  | if(!item) return; | 
|  | /* Note: item->merge is identical for all items, only free once */ | 
|  | heap_free(item->merge.chm_file); | 
|  | heap_free(item->merge.chm_index); | 
|  | while(item) { | 
|  | next = item->next; | 
|  |  | 
|  | heap_free(item->keyword); | 
|  | for(i=0;i<item->nItems;i++) { | 
|  | heap_free(item->items[i].name); | 
|  | heap_free(item->items[i].local); | 
|  | } | 
|  | heap_free(item->items); | 
|  |  | 
|  | item = next; | 
|  | } | 
|  | } |