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
*env_filename
, WCHAR
*fullname
, DWORD buflen
, WCHAR
**index
, WCHAR
**window
)
102 static const WCHAR helpW
[] = {'\\','h','e','l','p','\\',0};
103 static const WCHAR delimW
[] = {':',':',0};
104 static const WCHAR delim2W
[] = {'>',0};
107 WCHAR
*filename
, *extra
;
109 env_filename
= skip_schema(env_filename
);
111 /* the format is "helpFile[::/index][>window]" */
112 if (index
) *index
= NULL
;
113 if (window
) *window
= NULL
;
115 env_len
= ExpandEnvironmentStringsW(env_filename
, NULL
, 0);
119 filename
= heap_alloc(env_len
* sizeof(WCHAR
));
120 if (filename
== NULL
)
123 ExpandEnvironmentStringsW(env_filename
, filename
, env_len
);
125 extra
= wcsstr(filename
, delim2W
);
130 *window
= strdupW(extra
+1);
133 extra
= wcsstr(filename
, delimW
);
138 *index
= strdupW(extra
+2);
141 GetFullPathNameW(filename
, buflen
, fullname
, NULL
);
142 if (GetFileAttributesW(fullname
) == INVALID_FILE_ATTRIBUTES
)
144 GetWindowsDirectoryW(fullname
, buflen
);
145 lstrcatW(fullname
, helpW
);
146 lstrcatW(fullname
, filename
);
151 return (GetFileAttributesW(fullname
) != INVALID_FILE_ATTRIBUTES
);
154 /******************************************************************
155 * HtmlHelpW (HHCTRL.OCX.15)
157 HWND WINAPI
HtmlHelpW(HWND caller
, LPCWSTR filename
, UINT command
, DWORD_PTR data
)
159 WCHAR fullname
[MAX_PATH
];
161 TRACE("(%p, %s, command=%s, data=%lx)\n",
162 caller
, debugstr_w( filename
),
163 command_to_string( command
), data
);
167 case HH_DISPLAY_TOPIC
:
169 case HH_DISPLAY_INDEX
:
170 case HH_DISPLAY_SEARCH
:{
174 WCHAR
*window
= NULL
;
175 const WCHAR
*index
= NULL
;
176 WCHAR
*default_index
= NULL
;
177 int tab_index
= TAB_CONTENTS
;
182 if (!resolve_filename(filename
, fullname
, MAX_PATH
, &default_index
, &window
))
184 WARN("can't find %s\n", debugstr_w(filename
));
187 index
= default_index
;
190 info
= find_window(window
);
192 info
= CreateHelpViewer(info
, fullname
, caller
);
195 heap_free(default_index
);
201 index
= info
->WinType
.pszFile
;
202 if(!info
->WinType
.pszType
)
203 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
207 /* called to load a specified topic */
210 case HH_DISPLAY_TOPIC
:
214 static const WCHAR delimW
[] = {':',':',0};
215 const WCHAR
*i
= (const WCHAR
*)data
;
217 index
= wcsstr(i
, delimW
);
220 if(memcmp(info
->pCHMInfo
->szFile
, i
, index
-i
))
221 FIXME("Opening a CHM file in the context of another is not supported.\n");
222 index
+= lstrlenW(delimW
);
230 res
= NavigateToChm(info
, info
->pCHMInfo
->szFile
, index
);
231 heap_free(default_index
);
235 ReleaseHelpViewer(info
);
241 case HH_DISPLAY_TOPIC
:
243 tab_index
= TAB_CONTENTS
;
245 case HH_DISPLAY_INDEX
:
246 tab_index
= TAB_INDEX
;
248 FIXME("Should select keyword '%s'.\n", debugstr_w((WCHAR
*)data
));
250 case HH_DISPLAY_SEARCH
:
251 tab_index
= TAB_SEARCH
;
253 FIXME("Should display search specified by HH_FTS_QUERY structure.\n");
256 /* open the requested tab */
257 memset(&nmhdr
, 0, sizeof(nmhdr
));
258 nmhdr
.code
= TCN_SELCHANGE
;
259 SendMessageW(info
->hwndTabCtrl
, TCM_SETCURSEL
, (WPARAM
)info
->tabs
[tab_index
].id
, 0);
260 SendMessageW(info
->WinType
.hwndNavigation
, WM_NOTIFY
, 0, (LPARAM
)&nmhdr
);
262 return info
->WinType
.hwndHelp
;
264 case HH_HELP_CONTEXT
: {
265 WCHAR
*window
= NULL
;
272 if (!resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
))
274 WARN("can't find %s\n", debugstr_w(filename
));
279 info
= find_window(window
);
281 info
= CreateHelpViewer(info
, fullname
, caller
);
288 if(!info
->WinType
.pszType
)
289 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
293 url
= FindContextAlias(info
->pCHMInfo
, data
);
296 if(!data
) /* there may legitimately be no context alias for id 0 */
297 return info
->WinType
.hwndHelp
;
298 ReleaseHelpViewer(info
);
302 NavigateToUrl(info
, url
);
304 return info
->WinType
.hwndHelp
;
306 case HH_PRETRANSLATEMESSAGE
: {
307 static BOOL warned
= FALSE
;
311 FIXME("HH_PRETRANSLATEMESSAGE unimplemented\n");
319 LIST_FOR_EACH_ENTRY_SAFE(info
, next
, &window_list
, HHInfo
, entry
)
321 TRACE("Destroying window %s.\n", debugstr_w(info
->WinType
.pszType
));
322 ReleaseHelpViewer(info
);
326 case HH_SET_WIN_TYPE
: {
327 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
328 WCHAR
*window
= NULL
;
331 if (!filename
&& wintype
->pszType
)
332 window
= strdupW(wintype
->pszType
);
333 else if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
335 WARN("can't find window name: %s\n", debugstr_w(filename
));
338 info
= find_window(window
);
341 info
= heap_alloc_zero(sizeof(HHInfo
));
342 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
343 list_add_tail(&window_list
, &info
->entry
);
348 TRACE("Changing WINTYPE, fsValidMembers=0x%x\n", wintype
->fsValidMembers
);
350 MergeChmProperties(wintype
, info
, TRUE
);
351 UpdateHelpWindow(info
);
354 case HH_GET_WIN_TYPE
: {
355 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
356 WCHAR
*window
= NULL
;
359 if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
361 WARN("can't find window name: %s\n", debugstr_w(filename
));
364 info
= find_window(window
);
367 WARN("Could not find window named %s.\n", debugstr_w(window
));
372 TRACE("Retrieving WINTYPE for %s.\n", debugstr_w(window
));
373 *wintype
= info
->WinType
;
378 FIXME("HH case %s not handled.\n", command_to_string( command
));
384 static void wintypeAtoW(const HH_WINTYPEA
*data
, HH_WINTYPEW
*wdata
, struct wintype_stringsW
*stringsW
)
386 memcpy(wdata
, data
, sizeof(*data
));
387 /* convert all of the ANSI strings to Unicode */
388 wdata
->pszType
= stringsW
->pszType
= strdupAtoW(data
->pszType
);
389 wdata
->pszCaption
= stringsW
->pszCaption
= strdupAtoW(data
->pszCaption
);
390 wdata
->pszToc
= stringsW
->pszToc
= strdupAtoW(data
->pszToc
);
391 wdata
->pszIndex
= stringsW
->pszIndex
= strdupAtoW(data
->pszIndex
);
392 wdata
->pszFile
= stringsW
->pszFile
= strdupAtoW(data
->pszFile
);
393 wdata
->pszHome
= stringsW
->pszHome
= strdupAtoW(data
->pszHome
);
394 wdata
->pszJump1
= stringsW
->pszJump1
= strdupAtoW(data
->pszJump1
);
395 wdata
->pszJump2
= stringsW
->pszJump2
= strdupAtoW(data
->pszJump2
);
396 wdata
->pszUrlJump1
= stringsW
->pszUrlJump1
= strdupAtoW(data
->pszUrlJump1
);
397 wdata
->pszUrlJump2
= stringsW
->pszUrlJump2
= strdupAtoW(data
->pszUrlJump2
);
398 wdata
->pszCustomTabs
= stringsW
->pszCustomTabs
= strdupAtoW(data
->pszCustomTabs
);
401 static void wintypeWtoA(const HH_WINTYPEW
*wdata
, HH_WINTYPEA
*data
, struct wintype_stringsA
*stringsA
)
403 memcpy(data
, wdata
, sizeof(*wdata
));
404 /* convert all of the Unicode strings to ANSI */
405 data
->pszType
= stringsA
->pszType
= strdupWtoA(wdata
->pszType
);
406 data
->pszCaption
= stringsA
->pszCaption
= strdupWtoA(wdata
->pszCaption
);
407 data
->pszToc
= stringsA
->pszToc
= strdupWtoA(wdata
->pszToc
);
408 data
->pszIndex
= stringsA
->pszFile
= strdupWtoA(wdata
->pszIndex
);
409 data
->pszFile
= stringsA
->pszFile
= strdupWtoA(wdata
->pszFile
);
410 data
->pszHome
= stringsA
->pszHome
= strdupWtoA(wdata
->pszHome
);
411 data
->pszJump1
= stringsA
->pszJump1
= strdupWtoA(wdata
->pszJump1
);
412 data
->pszJump2
= stringsA
->pszJump2
= strdupWtoA(wdata
->pszJump2
);
413 data
->pszUrlJump1
= stringsA
->pszUrlJump1
= strdupWtoA(wdata
->pszUrlJump1
);
414 data
->pszUrlJump2
= stringsA
->pszUrlJump2
= strdupWtoA(wdata
->pszUrlJump2
);
415 data
->pszCustomTabs
= stringsA
->pszCustomTabs
= strdupWtoA(wdata
->pszCustomTabs
);
418 /******************************************************************
419 * HtmlHelpA (HHCTRL.OCX.14)
421 HWND WINAPI
HtmlHelpA(HWND caller
, LPCSTR filename
, UINT command
, DWORD_PTR data
)
423 WCHAR
*wfile
= strdupAtoW( filename
);
430 case HH_ALINK_LOOKUP
:
431 case HH_DISPLAY_SEARCH
:
432 case HH_DISPLAY_TEXT_POPUP
:
433 case HH_GET_LAST_ERROR
:
434 case HH_KEYWORD_LOOKUP
:
436 FIXME("structures not handled yet\n");
439 case HH_SET_WIN_TYPE
:
441 struct wintype_stringsW stringsW
;
444 wintypeAtoW((HH_WINTYPEA
*)data
, &wdata
, &stringsW
);
445 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
446 wintype_stringsW_free(&stringsW
);
449 case HH_GET_WIN_TYPE
:
454 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
455 if (!wdata
.pszType
) break;
456 info
= find_window(wdata
.pszType
);
458 wintype_stringsA_free(&info
->stringsA
);
459 wintypeWtoA(&wdata
, (HH_WINTYPEA
*)data
, &info
->stringsA
);
463 case HH_DISPLAY_INDEX
:
464 case HH_DISPLAY_TOPIC
:
466 case HH_GET_WIN_HANDLE
:
467 case HH_SAFE_DISPLAY_TOPIC
:
469 WCHAR
*wdata
= strdupAtoW( (const char *)data
);
470 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)wdata
);
476 case HH_HELP_CONTEXT
:
478 case HH_PRETRANSLATEMESSAGE
:
479 case HH_TP_HELP_CONTEXTMENU
:
480 case HH_TP_HELP_WM_HELP
:
481 case HH_UNINITIALIZE
:
482 /* either scalar or pointer to scalar - do nothing */
486 FIXME("Unknown command: %s (%d)\n", command_to_string(command
), command
);
491 result
= HtmlHelpW( caller
, wfile
, command
, data
);
497 /******************************************************************
498 * doWinMain (HHCTRL.OCX.13)
500 int WINAPI
doWinMain(HINSTANCE hInstance
, LPSTR szCmdLine
)
503 int len
, buflen
, mapid
= -1;
510 /* Parse command line option of the HTML Help command.
512 * Note: The only currently handled action is "mapid",
513 * which corresponds to opening a specific page.
515 while(*szCmdLine
== '-')
520 space
= strchr(ptr
, ' ');
521 if(!strncmp(ptr
, "mapid", space
-ptr
))
525 ptr
+= strlen("mapid")+1;
526 space
= strchr(ptr
, ' ');
527 /* command line ends without number */
530 memcpy(idtxt
, ptr
, space
-ptr
);
531 idtxt
[space
-ptr
] = '\0';
537 FIXME("Unhandled HTML Help command line parameter! (%.*s)\n", (int)(space
-szCmdLine
), szCmdLine
);
542 /* FIXME: Check szCmdLine for bad arguments */
543 if (*szCmdLine
== '\"')
544 endq
= strchr(++szCmdLine
, '\"');
547 len
= endq
- szCmdLine
;
549 len
= strlen(szCmdLine
);
551 /* no filename given */
555 buflen
= MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, NULL
, 0) + 1;
556 filename
= heap_alloc(buflen
* sizeof(WCHAR
));
557 MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, filename
, buflen
);
558 filename
[buflen
-1] = 0;
560 /* Open a specific help topic */
562 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_HELP_CONTEXT
, mapid
);
564 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_DISPLAY_TOPIC
, 0);
570 ERR("Failed to open HTML Help file '%s'.\n", szCmdLine
);
574 while (GetMessageW(&msg
, 0, 0, 0))
576 TranslateMessage(&msg
);
577 DispatchMessageW(&msg
);
583 /******************************************************************
584 * DllGetClassObject (HHCTRL.OCX.@)
586 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
588 FIXME("(%s %s %p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
589 return CLASS_E_CLASSNOTAVAILABLE
;
592 /***********************************************************************
593 * DllRegisterServer (HHCTRL.OCX.@)
595 HRESULT WINAPI
DllRegisterServer(void)
597 return __wine_register_resources( hhctrl_hinstance
);
600 /***********************************************************************
601 * DllUnregisterServer (HHCTRL.OCX.@)
603 HRESULT WINAPI
DllUnregisterServer(void)
605 return __wine_unregister_resources( hhctrl_hinstance
);