2 * hhctrl implementation
4 * Copyright 2004 Krzysztof Foltman
5 * Copyright 2007 Jacek Caban for CodeWeavers
7 * This library is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
12 * This library is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with this library; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
22 #include "wine/debug.h"
39 WINE_DEFAULT_DEBUG_CHANNEL(htmlhelp
);
41 HINSTANCE hhctrl_hinstance
;
42 BOOL hh_process
= FALSE
;
45 BOOL WINAPI
DllMain(HINSTANCE hInstance
, DWORD fdwReason
, LPVOID lpvReserved
)
47 TRACE("(%p,%d,%p)\n", hInstance
, fdwReason
, lpvReserved
);
51 case DLL_PROCESS_ATTACH
:
52 hhctrl_hinstance
= hInstance
;
53 DisableThreadLibraryCalls(hInstance
);
59 static const char *command_to_string(UINT command
)
61 #define X(x) case x: return #x
64 X( HH_DISPLAY_TOPIC
);
66 X( HH_DISPLAY_INDEX
);
67 X( HH_DISPLAY_SEARCH
);
70 X( HH_GET_WIN_HANDLE
);
71 X( HH_ENUM_INFO_TYPE
);
72 X( HH_SET_INFO_TYPE
);
77 X( HH_KEYWORD_LOOKUP
);
78 X( HH_DISPLAY_TEXT_POPUP
);
80 X( HH_TP_HELP_CONTEXTMENU
);
81 X( HH_TP_HELP_WM_HELP
);
84 X( HH_GET_LAST_ERROR
);
85 X( HH_ENUM_CATEGORY
);
86 X( HH_ENUM_CATEGORY_IT
);
87 X( HH_RESET_IT_FILTER
);
88 X( HH_SET_INCLUSIVE_FILTER
);
89 X( HH_SET_EXCLUSIVE_FILTER
);
92 X( HH_SAFE_DISPLAY_TOPIC
);
93 X( HH_PRETRANSLATEMESSAGE
);
94 X( HH_SET_GLOBAL_PROPERTY
);
95 default: return "???";
100 static BOOL
resolve_filename(const WCHAR
*filename
, WCHAR
*fullname
, DWORD buflen
, WCHAR
**index
, WCHAR
**window
)
103 WCHAR chm_file
[MAX_PATH
];
105 static const WCHAR helpW
[] = {'\\','h','e','l','p','\\',0};
106 static const WCHAR delimW
[] = {':',':',0};
107 static const WCHAR delim2W
[] = {'>',0};
109 filename
= skip_schema(filename
);
111 /* the format is "helpFile[::/index][>window]" */
112 if (index
) *index
= NULL
;
113 if (window
) *window
= NULL
;
115 extra
= strstrW(filename
, delim2W
);
118 memcpy(chm_file
, filename
, (extra
-filename
)*sizeof(WCHAR
));
119 chm_file
[extra
-filename
] = 0;
122 *window
= strdupW(extra
+1);
125 extra
= strstrW(filename
, delimW
);
128 if (filename
!= chm_file
)
129 memcpy(chm_file
, filename
, (extra
-filename
)*sizeof(WCHAR
));
130 chm_file
[extra
-filename
] = 0;
133 *index
= strdupW(extra
+2);
136 GetFullPathNameW(filename
, buflen
, fullname
, NULL
);
137 if (GetFileAttributesW(fullname
) == INVALID_FILE_ATTRIBUTES
)
139 GetWindowsDirectoryW(fullname
, buflen
);
140 strcatW(fullname
, helpW
);
141 strcatW(fullname
, filename
);
143 return (GetFileAttributesW(fullname
) != INVALID_FILE_ATTRIBUTES
);
146 /******************************************************************
147 * HtmlHelpW (HHCTRL.OCX.15)
149 HWND WINAPI
HtmlHelpW(HWND caller
, LPCWSTR filename
, UINT command
, DWORD_PTR data
)
151 WCHAR fullname
[MAX_PATH
];
153 TRACE("(%p, %s, command=%s, data=%lx)\n",
154 caller
, debugstr_w( filename
),
155 command_to_string( command
), data
);
159 case HH_DISPLAY_TOPIC
:
161 case HH_DISPLAY_INDEX
:
162 case HH_DISPLAY_SEARCH
:{
166 WCHAR
*window
= NULL
;
167 const WCHAR
*index
= NULL
;
168 WCHAR
*default_index
= NULL
;
169 int tab_index
= TAB_CONTENTS
;
174 if (!resolve_filename(filename
, fullname
, MAX_PATH
, &default_index
, &window
))
176 WARN("can't find %s\n", debugstr_w(filename
));
179 index
= default_index
;
182 info
= find_window(window
);
184 info
= CreateHelpViewer(info
, fullname
, caller
);
187 heap_free(default_index
);
193 index
= info
->WinType
.pszFile
;
194 if(!info
->WinType
.pszType
)
195 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
199 /* called to load a specified topic */
202 case HH_DISPLAY_TOPIC
:
206 static const WCHAR delimW
[] = {':',':',0};
207 const WCHAR
*i
= (const WCHAR
*)data
;
209 index
= strstrW(i
, delimW
);
212 if(memcmp(info
->pCHMInfo
->szFile
, i
, index
-i
))
213 FIXME("Opening a CHM file in the context of another is not supported.\n");
214 index
+= strlenW(delimW
);
222 res
= NavigateToChm(info
, info
->pCHMInfo
->szFile
, index
);
223 heap_free(default_index
);
227 ReleaseHelpViewer(info
);
233 case HH_DISPLAY_TOPIC
:
235 tab_index
= TAB_CONTENTS
;
237 case HH_DISPLAY_INDEX
:
238 tab_index
= TAB_INDEX
;
240 FIXME("Should select keyword '%s'.\n", debugstr_w((WCHAR
*)data
));
242 case HH_DISPLAY_SEARCH
:
243 tab_index
= TAB_SEARCH
;
245 FIXME("Should display search specified by HH_FTS_QUERY structure.\n");
248 /* open the requested tab */
249 memset(&nmhdr
, 0, sizeof(nmhdr
));
250 nmhdr
.code
= TCN_SELCHANGE
;
251 SendMessageW(info
->hwndTabCtrl
, TCM_SETCURSEL
, (WPARAM
)info
->tabs
[tab_index
].id
, 0);
252 SendMessageW(info
->WinType
.hwndNavigation
, WM_NOTIFY
, 0, (LPARAM
)&nmhdr
);
254 return info
->WinType
.hwndHelp
;
256 case HH_HELP_CONTEXT
: {
257 WCHAR
*window
= NULL
;
264 if (!resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
))
266 WARN("can't find %s\n", debugstr_w(filename
));
271 info
= find_window(window
);
273 info
= CreateHelpViewer(info
, fullname
, caller
);
280 if(!info
->WinType
.pszType
)
281 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
285 url
= FindContextAlias(info
->pCHMInfo
, data
);
288 if(!data
) /* there may legitimately be no context alias for id 0 */
289 return info
->WinType
.hwndHelp
;
290 ReleaseHelpViewer(info
);
294 NavigateToUrl(info
, url
);
296 return info
->WinType
.hwndHelp
;
298 case HH_PRETRANSLATEMESSAGE
: {
299 static BOOL warned
= FALSE
;
303 FIXME("HH_PRETRANSLATEMESSAGE unimplemented\n");
311 LIST_FOR_EACH_ENTRY_SAFE(info
, next
, &window_list
, HHInfo
, entry
)
313 TRACE("Destroying window %s.\n", debugstr_w(info
->WinType
.pszType
));
314 ReleaseHelpViewer(info
);
318 case HH_SET_WIN_TYPE
: {
319 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
320 WCHAR
*window
= NULL
;
323 if (!filename
&& wintype
->pszType
)
324 window
= strdupW(wintype
->pszType
);
325 else if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
327 WARN("can't find window name: %s\n", debugstr_w(filename
));
330 info
= find_window(window
);
333 info
= heap_alloc_zero(sizeof(HHInfo
));
334 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
335 list_add_tail(&window_list
, &info
->entry
);
340 TRACE("Changing WINTYPE, fsValidMembers=0x%x\n", wintype
->fsValidMembers
);
342 MergeChmProperties(wintype
, info
, TRUE
);
343 UpdateHelpWindow(info
);
346 case HH_GET_WIN_TYPE
: {
347 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
348 WCHAR
*window
= NULL
;
351 if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
353 WARN("can't find window name: %s\n", debugstr_w(filename
));
356 info
= find_window(window
);
359 WARN("Could not find window named %s.\n", debugstr_w(window
));
364 TRACE("Retrieving WINTYPE for %s.\n", debugstr_w(window
));
365 *wintype
= info
->WinType
;
370 FIXME("HH case %s not handled.\n", command_to_string( command
));
376 static void wintypeAtoW(const HH_WINTYPEA
*data
, HH_WINTYPEW
*wdata
, struct wintype_stringsW
*stringsW
)
378 memcpy(wdata
, data
, sizeof(*data
));
379 /* convert all of the ANSI strings to Unicode */
380 wdata
->pszType
= stringsW
->pszType
= strdupAtoW(data
->pszType
);
381 wdata
->pszCaption
= stringsW
->pszCaption
= strdupAtoW(data
->pszCaption
);
382 wdata
->pszToc
= stringsW
->pszToc
= strdupAtoW(data
->pszToc
);
383 wdata
->pszIndex
= stringsW
->pszIndex
= strdupAtoW(data
->pszIndex
);
384 wdata
->pszFile
= stringsW
->pszFile
= strdupAtoW(data
->pszFile
);
385 wdata
->pszHome
= stringsW
->pszHome
= strdupAtoW(data
->pszHome
);
386 wdata
->pszJump1
= stringsW
->pszJump1
= strdupAtoW(data
->pszJump1
);
387 wdata
->pszJump2
= stringsW
->pszJump2
= strdupAtoW(data
->pszJump2
);
388 wdata
->pszUrlJump1
= stringsW
->pszUrlJump1
= strdupAtoW(data
->pszUrlJump1
);
389 wdata
->pszUrlJump2
= stringsW
->pszUrlJump2
= strdupAtoW(data
->pszUrlJump2
);
390 wdata
->pszCustomTabs
= stringsW
->pszCustomTabs
= strdupAtoW(data
->pszCustomTabs
);
393 static void wintypeWtoA(const HH_WINTYPEW
*wdata
, HH_WINTYPEA
*data
, struct wintype_stringsA
*stringsA
)
395 memcpy(data
, wdata
, sizeof(*wdata
));
396 /* convert all of the Unicode strings to ANSI */
397 data
->pszType
= stringsA
->pszType
= strdupWtoA(wdata
->pszType
);
398 data
->pszCaption
= stringsA
->pszCaption
= strdupWtoA(wdata
->pszCaption
);
399 data
->pszToc
= stringsA
->pszToc
= strdupWtoA(wdata
->pszToc
);
400 data
->pszIndex
= stringsA
->pszFile
= strdupWtoA(wdata
->pszIndex
);
401 data
->pszFile
= stringsA
->pszFile
= strdupWtoA(wdata
->pszFile
);
402 data
->pszHome
= stringsA
->pszHome
= strdupWtoA(wdata
->pszHome
);
403 data
->pszJump1
= stringsA
->pszJump1
= strdupWtoA(wdata
->pszJump1
);
404 data
->pszJump2
= stringsA
->pszJump2
= strdupWtoA(wdata
->pszJump2
);
405 data
->pszUrlJump1
= stringsA
->pszUrlJump1
= strdupWtoA(wdata
->pszUrlJump1
);
406 data
->pszUrlJump2
= stringsA
->pszUrlJump2
= strdupWtoA(wdata
->pszUrlJump2
);
407 data
->pszCustomTabs
= stringsA
->pszCustomTabs
= strdupWtoA(wdata
->pszCustomTabs
);
410 /******************************************************************
411 * HtmlHelpA (HHCTRL.OCX.14)
413 HWND WINAPI
HtmlHelpA(HWND caller
, LPCSTR filename
, UINT command
, DWORD_PTR data
)
415 WCHAR
*wfile
= strdupAtoW( filename
);
422 case HH_ALINK_LOOKUP
:
423 case HH_DISPLAY_SEARCH
:
424 case HH_DISPLAY_TEXT_POPUP
:
425 case HH_GET_LAST_ERROR
:
426 case HH_KEYWORD_LOOKUP
:
428 FIXME("structures not handled yet\n");
431 case HH_SET_WIN_TYPE
:
433 struct wintype_stringsW stringsW
;
436 wintypeAtoW((HH_WINTYPEA
*)data
, &wdata
, &stringsW
);
437 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
438 wintype_stringsW_free(&stringsW
);
441 case HH_GET_WIN_TYPE
:
446 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
447 if (!wdata
.pszType
) break;
448 info
= find_window(wdata
.pszType
);
450 wintype_stringsA_free(&info
->stringsA
);
451 wintypeWtoA(&wdata
, (HH_WINTYPEA
*)data
, &info
->stringsA
);
455 case HH_DISPLAY_INDEX
:
456 case HH_DISPLAY_TOPIC
:
458 case HH_GET_WIN_HANDLE
:
459 case HH_SAFE_DISPLAY_TOPIC
:
461 WCHAR
*wdata
= strdupAtoW( (const char *)data
);
462 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)wdata
);
468 case HH_HELP_CONTEXT
:
470 case HH_PRETRANSLATEMESSAGE
:
471 case HH_TP_HELP_CONTEXTMENU
:
472 case HH_TP_HELP_WM_HELP
:
473 case HH_UNINITIALIZE
:
474 /* either scalar or pointer to scalar - do nothing */
478 FIXME("Unknown command: %s (%d)\n", command_to_string(command
), command
);
483 result
= HtmlHelpW( caller
, wfile
, command
, data
);
489 /******************************************************************
490 * doWinMain (HHCTRL.OCX.13)
492 int WINAPI
doWinMain(HINSTANCE hInstance
, LPSTR szCmdLine
)
495 int len
, buflen
, mapid
= -1;
502 /* Parse command line option of the HTML Help command.
504 * Note: The only currently handled action is "mapid",
505 * which corresponds to opening a specific page.
507 while(*szCmdLine
== '-')
512 space
= strchr(ptr
, ' ');
513 if(!strncmp(ptr
, "mapid", space
-ptr
))
517 ptr
+= strlen("mapid")+1;
518 space
= strchr(ptr
, ' ');
519 /* command line ends without number */
522 memcpy(idtxt
, ptr
, space
-ptr
);
523 idtxt
[space
-ptr
] = '\0';
529 FIXME("Unhandled HTML Help command line parameter! (%.*s)\n", (int)(space
-szCmdLine
), szCmdLine
);
534 /* FIXME: Check szCmdLine for bad arguments */
535 if (*szCmdLine
== '\"')
536 endq
= strchr(++szCmdLine
, '\"');
539 len
= endq
- szCmdLine
;
541 len
= strlen(szCmdLine
);
543 /* no filename given */
547 buflen
= MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, NULL
, 0) + 1;
548 filename
= heap_alloc(buflen
* sizeof(WCHAR
));
549 MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, filename
, buflen
);
550 filename
[buflen
-1] = 0;
552 /* Open a specific help topic */
554 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_HELP_CONTEXT
, mapid
);
556 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_DISPLAY_TOPIC
, 0);
562 ERR("Failed to open HTML Help file '%s'.\n", szCmdLine
);
566 while (GetMessageW(&msg
, 0, 0, 0))
568 TranslateMessage(&msg
);
569 DispatchMessageW(&msg
);
575 /******************************************************************
576 * DllGetClassObject (HHCTRL.OCX.@)
578 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
580 FIXME("(%s %s %p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
581 return CLASS_E_CLASSNOTAVAILABLE
;
584 /***********************************************************************
585 * DllRegisterServer (HHCTRL.OCX.@)
587 HRESULT WINAPI
DllRegisterServer(void)
589 return __wine_register_resources( hhctrl_hinstance
);
592 /***********************************************************************
593 * DllUnregisterServer (HHCTRL.OCX.@)
595 HRESULT WINAPI
DllUnregisterServer(void)
597 return __wine_unregister_resources( hhctrl_hinstance
);