| /* |
| * Copyright 2007 Jacek Caban for CodeWeavers |
| * |
| * 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 "wine/debug.h" |
| |
| WINE_DEFAULT_DEBUG_CHANNEL(htmlhelp); |
| |
| #define BLOCK_SIZE 0x1000 |
| |
| typedef enum { |
| INSERT_NEXT, |
| INSERT_CHILD |
| } insert_type_t; |
| |
| typedef struct { |
| char *buf; |
| int size; |
| int len; |
| } strbuf_t; |
| |
| static void strbuf_init(strbuf_t *buf) |
| { |
| buf->size = 8; |
| buf->len = 0; |
| buf->buf = hhctrl_alloc(buf->size); |
| } |
| |
| static void strbuf_zero(strbuf_t *buf) |
| { |
| buf->len = 0; |
| } |
| |
| static void strbuf_free(strbuf_t *buf) |
| { |
| hhctrl_free(buf->buf); |
| } |
| |
| static void strbuf_append(strbuf_t *buf, const char *data, int len) |
| { |
| if(buf->len+len > buf->size) { |
| buf->size = buf->len+len; |
| buf->buf = hhctrl_realloc(buf->buf, buf->size); |
| } |
| |
| memcpy(buf->buf+buf->len, data, len); |
| buf->len += len; |
| } |
| |
| typedef struct { |
| IStream *str; |
| char buf[BLOCK_SIZE]; |
| ULONG size; |
| ULONG p; |
| } stream_t; |
| |
| static void stream_init(stream_t *stream, IStream *str) |
| { |
| memset(stream, 0, sizeof(stream_t)); |
| stream->str = str; |
| } |
| |
| static BOOL stream_chr(stream_t *stream, strbuf_t *buf, char c) |
| { |
| BOOL b = TRUE; |
| ULONG i; |
| |
| while(b) { |
| for(i=stream->p; i<stream->size; i++) { |
| if(stream->buf[i] == c) { |
| b = FALSE; |
| break; |
| } |
| } |
| |
| if(buf && i > stream->p) |
| strbuf_append(buf, stream->buf+stream->p, i-stream->p); |
| stream->p = i; |
| |
| if(stream->p == stream->size) { |
| stream->p = 0; |
| IStream_Read(stream->str, stream->buf, sizeof(stream->buf), &stream->size); |
| if(!stream->size) |
| break; |
| } |
| } |
| |
| return stream->size != 0; |
| } |
| |
| static void get_node_name(strbuf_t *node, strbuf_t *name) |
| { |
| const char *ptr = node->buf+1; |
| |
| strbuf_zero(name); |
| |
| while(*ptr != '>' && !isspace(*ptr)) |
| ptr++; |
| |
| strbuf_append(name, node->buf+1, ptr-node->buf-1); |
| strbuf_append(name, "", 1); |
| } |
| |
| static BOOL next_node(stream_t *stream, strbuf_t *buf) |
| { |
| if(!stream_chr(stream, NULL, '<')) |
| return FALSE; |
| |
| if(!stream_chr(stream, buf, '>')) |
| return FALSE; |
| |
| strbuf_append(buf, ">", 2); |
| |
| return TRUE; |
| } |
| |
| static const char *get_attr(const char *node, const char *name, int *len) |
| { |
| const char *ptr, *ptr2; |
| char name_buf[32]; |
| int nlen; |
| |
| nlen = strlen(name); |
| memcpy(name_buf, name, nlen); |
| name_buf[nlen++] = '='; |
| name_buf[nlen++] = '\"'; |
| name_buf[nlen] = 0; |
| |
| ptr = strstr(node, name_buf); |
| if(!ptr) { |
| WARN("name not found\n"); |
| return NULL; |
| } |
| |
| ptr += nlen; |
| ptr2 = strchr(ptr, '\"'); |
| if(!ptr2) |
| return NULL; |
| |
| *len = ptr2-ptr; |
| return ptr; |
| } |
| |
| static void parse_obj_node_param(ContentItem *item, ContentItem *hhc_root, const char *text) |
| { |
| const char *ptr; |
| LPWSTR *param, merge; |
| int len, wlen; |
| |
| ptr = get_attr(text, "name", &len); |
| if(!ptr) { |
| WARN("name attr not found\n"); |
| return; |
| } |
| |
| if(!strncasecmp("name", ptr, len)) { |
| param = &item->name; |
| }else if(!strncasecmp("merge", ptr, len)) { |
| param = &merge; |
| }else if(!strncasecmp("local", ptr, len)) { |
| param = &item->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 = hhctrl_alloc((wlen+1)*sizeof(WCHAR)); |
| MultiByteToWideChar(CP_ACP, 0, ptr, len, *param, wlen); |
| (*param)[wlen] = 0; |
| |
| if(param == &merge) { |
| SetChmPath(&item->merge, hhc_root->merge.chm_file, merge); |
| hhctrl_free(merge); |
| } |
| } |
| |
| static ContentItem *parse_hhc(HHInfo*,IStream*,ContentItem*,insert_type_t*); |
| |
| static ContentItem *insert_item(ContentItem *item, ContentItem *new_item, insert_type_t insert_type) |
| { |
| if(!item) |
| return new_item; |
| |
| switch(insert_type) { |
| case INSERT_NEXT: |
| item->next = new_item; |
| return new_item; |
| case INSERT_CHILD: |
| if(item->child) { |
| ContentItem *iter = item->child; |
| while(iter->next) |
| iter = iter->next; |
| iter->next = new_item; |
| }else { |
| item->child = new_item; |
| } |
| return item; |
| } |
| |
| return NULL; |
| } |
| |
| static ContentItem *parse_sitemap_object(HHInfo *info, stream_t *stream, ContentItem *hhc_root, |
| insert_type_t *insert_type) |
| { |
| strbuf_t node, node_name; |
| ContentItem *item; |
| |
| *insert_type = INSERT_NEXT; |
| |
| strbuf_init(&node); |
| strbuf_init(&node_name); |
| |
| item = hhctrl_alloc_zero(sizeof(ContentItem)); |
| |
| while(next_node(stream, &node)) { |
| get_node_name(&node, &node_name); |
| |
| TRACE("%s\n", node.buf); |
| |
| if(!strcasecmp(node_name.buf, "/object")) |
| break; |
| if(!strcasecmp(node_name.buf, "param")) |
| parse_obj_node_param(item, hhc_root, node.buf); |
| |
| strbuf_zero(&node); |
| } |
| |
| strbuf_free(&node); |
| strbuf_free(&node_name); |
| |
| if(item->merge.chm_index) { |
| IStream *merge_stream; |
| |
| merge_stream = GetChmStream(info->pCHMInfo, item->merge.chm_file, &item->merge); |
| if(merge_stream) { |
| item->child = parse_hhc(info, merge_stream, hhc_root, insert_type); |
| IStream_Release(merge_stream); |
| }else { |
| WARN("Could not get %s::%s stream\n", debugstr_w(item->merge.chm_file), |
| debugstr_w(item->merge.chm_file)); |
| } |
| |
| } |
| |
| return item; |
| } |
| |
| static ContentItem *parse_ul(HHInfo *info, stream_t *stream, ContentItem *hhc_root) |
| { |
| strbuf_t node, node_name; |
| ContentItem *ret = NULL, *prev = NULL, *new_item = NULL; |
| insert_type_t it; |
| |
| 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)) { |
| new_item = parse_sitemap_object(info, stream, hhc_root, &it); |
| prev = insert_item(prev, new_item, it); |
| if(!ret) |
| ret = prev; |
| } |
| }else if(!strcasecmp(node_name.buf, "ul")) { |
| new_item = parse_ul(info, stream, hhc_root); |
| insert_item(prev, new_item, INSERT_CHILD); |
| }else if(!strcasecmp(node_name.buf, "/ul")) { |
| break; |
| } |
| |
| strbuf_zero(&node); |
| } |
| |
| strbuf_free(&node); |
| strbuf_free(&node_name); |
| |
| return ret; |
| } |
| |
| static ContentItem *parse_hhc(HHInfo *info, IStream *str, ContentItem *hhc_root, |
| insert_type_t *insert_type) |
| { |
| stream_t stream; |
| strbuf_t node, node_name; |
| ContentItem *ret = NULL, *prev = NULL; |
| |
| *insert_type = INSERT_NEXT; |
| |
| 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, "ul")) { |
| ContentItem *item = parse_ul(info, &stream, hhc_root); |
| prev = insert_item(prev, item, INSERT_CHILD); |
| if(!ret) |
| ret = prev; |
| *insert_type = INSERT_CHILD; |
| } |
| |
| strbuf_zero(&node); |
| } |
| |
| strbuf_free(&node); |
| strbuf_free(&node_name); |
| |
| return ret; |
| } |
| |
| static void insert_content_item(HWND hwnd, ContentItem *parent, ContentItem *item) |
| { |
| TVINSERTSTRUCTW tvis; |
| |
| memset(&tvis, 0, sizeof(tvis)); |
| tvis.u.item.mask = TVIF_TEXT|TVIF_PARAM; |
| tvis.u.item.cchTextMax = strlenW(item->name)+1; |
| tvis.u.item.pszText = item->name; |
| tvis.u.item.lParam = (LPARAM)item; |
| tvis.hParent = parent ? parent->id : 0; |
| tvis.hInsertAfter = TVI_LAST; |
| |
| item->id = (HTREEITEM)SendMessageW(hwnd, TVM_INSERTITEMW, 0, (LPARAM)&tvis); |
| } |
| |
| static void fill_content_tree(HWND hwnd, ContentItem *parent, ContentItem *item) |
| { |
| while(item) { |
| if(item->name) { |
| insert_content_item(hwnd, parent, item); |
| fill_content_tree(hwnd, item, item->child); |
| }else { |
| fill_content_tree(hwnd, parent, item->child); |
| } |
| item = item->next; |
| } |
| } |
| |
| static void set_item_parents(ContentItem *parent, ContentItem *item) |
| { |
| while(item) { |
| item->parent = parent; |
| set_item_parents(item, item->child); |
| item = item->next; |
| } |
| } |
| |
| void InitContent(HHInfo *info) |
| { |
| IStream *stream; |
| insert_type_t insert_type; |
| |
| info->content = hhctrl_alloc_zero(sizeof(ContentItem)); |
| SetChmPath(&info->content->merge, info->pCHMInfo->szFile, info->WinType.pszToc); |
| |
| stream = GetChmStream(info->pCHMInfo, info->pCHMInfo->szFile, &info->content->merge); |
| if(!stream) { |
| TRACE("Could not get content stream\n"); |
| return; |
| } |
| |
| info->content->child = parse_hhc(info, stream, info->content, &insert_type); |
| IStream_Release(stream); |
| |
| set_item_parents(NULL, info->content); |
| fill_content_tree(info->tabs[TAB_CONTENTS].hwnd, NULL, info->content); |
| } |
| |
| static void free_content_item(ContentItem *item) |
| { |
| ContentItem *next; |
| |
| while(item) { |
| next = item->next; |
| |
| free_content_item(item->child); |
| |
| hhctrl_free(item->name); |
| hhctrl_free(item->local); |
| hhctrl_free(item->merge.chm_file); |
| hhctrl_free(item->merge.chm_index); |
| |
| item = next; |
| } |
| } |
| |
| void ReleaseContent(HHInfo *info) |
| { |
| free_content_item(info->content); |
| } |