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
;
44 extern struct list window_list
;
46 BOOL WINAPI
DllMain(HINSTANCE hInstance
, DWORD fdwReason
, LPVOID lpvReserved
)
48 TRACE("(%p,%d,%p)\n", hInstance
, fdwReason
, lpvReserved
);
52 case DLL_PROCESS_ATTACH
:
53 hhctrl_hinstance
= hInstance
;
54 DisableThreadLibraryCalls(hInstance
);
60 static const char *command_to_string(UINT command
)
62 #define X(x) case x: return #x
65 X( HH_DISPLAY_TOPIC
);
67 X( HH_DISPLAY_INDEX
);
68 X( HH_DISPLAY_SEARCH
);
71 X( HH_GET_WIN_HANDLE
);
72 X( HH_ENUM_INFO_TYPE
);
73 X( HH_SET_INFO_TYPE
);
78 X( HH_KEYWORD_LOOKUP
);
79 X( HH_DISPLAY_TEXT_POPUP
);
81 X( HH_TP_HELP_CONTEXTMENU
);
82 X( HH_TP_HELP_WM_HELP
);
85 X( HH_GET_LAST_ERROR
);
86 X( HH_ENUM_CATEGORY
);
87 X( HH_ENUM_CATEGORY_IT
);
88 X( HH_RESET_IT_FILTER
);
89 X( HH_SET_INCLUSIVE_FILTER
);
90 X( HH_SET_EXCLUSIVE_FILTER
);
93 X( HH_SAFE_DISPLAY_TOPIC
);
94 X( HH_PRETRANSLATEMESSAGE
);
95 X( HH_SET_GLOBAL_PROPERTY
);
96 default: return "???";
101 static BOOL
resolve_filename(const WCHAR
*filename
, WCHAR
*fullname
, DWORD buflen
, WCHAR
**index
, WCHAR
**window
)
104 WCHAR chm_file
[MAX_PATH
];
106 static const WCHAR helpW
[] = {'\\','h','e','l','p','\\',0};
107 static const WCHAR delimW
[] = {':',':',0};
108 static const WCHAR delim2W
[] = {'>',0};
110 filename
= skip_schema(filename
);
112 /* the format is "helpFile[::/index][>window]" */
113 if (index
) *index
= NULL
;
114 if (window
) *window
= NULL
;
116 extra
= strstrW(filename
, delim2W
);
119 memcpy(chm_file
, filename
, (extra
-filename
)*sizeof(WCHAR
));
120 chm_file
[extra
-filename
] = 0;
123 *window
= strdupW(extra
+1);
126 extra
= strstrW(filename
, delimW
);
129 if (filename
!= chm_file
)
130 memcpy(chm_file
, filename
, (extra
-filename
)*sizeof(WCHAR
));
131 chm_file
[extra
-filename
] = 0;
134 *index
= strdupW(extra
+2);
137 GetFullPathNameW(filename
, buflen
, fullname
, NULL
);
138 if (GetFileAttributesW(fullname
) == INVALID_FILE_ATTRIBUTES
)
140 GetWindowsDirectoryW(fullname
, buflen
);
141 strcatW(fullname
, helpW
);
142 strcatW(fullname
, filename
);
144 return (GetFileAttributesW(fullname
) != INVALID_FILE_ATTRIBUTES
);
147 /******************************************************************
148 * HtmlHelpW (HHCTRL.OCX.15)
150 HWND WINAPI
HtmlHelpW(HWND caller
, LPCWSTR filename
, UINT command
, DWORD_PTR data
)
152 WCHAR fullname
[MAX_PATH
];
154 TRACE("(%p, %s, command=%s, data=%lx)\n",
155 caller
, debugstr_w( filename
),
156 command_to_string( command
), data
);
160 case HH_DISPLAY_TOPIC
:
162 case HH_DISPLAY_INDEX
:
163 case HH_DISPLAY_SEARCH
:{
167 WCHAR
*window
= NULL
;
168 const WCHAR
*index
= NULL
;
169 WCHAR
*default_index
= NULL
;
170 int tab_index
= TAB_CONTENTS
;
175 if (!resolve_filename(filename
, fullname
, MAX_PATH
, &default_index
, &window
))
177 WARN("can't find %s\n", debugstr_w(filename
));
180 index
= default_index
;
183 info
= find_window(window
);
185 info
= CreateHelpViewer(info
, fullname
, caller
);
188 heap_free(default_index
);
194 index
= info
->WinType
.pszFile
;
195 if(!info
->WinType
.pszType
)
196 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
200 /* called to load a specified topic */
203 case HH_DISPLAY_TOPIC
:
207 static const WCHAR delimW
[] = {':',':',0};
208 const WCHAR
*i
= (const WCHAR
*)data
;
210 index
= strstrW(i
, delimW
);
213 if(memcmp(info
->pCHMInfo
->szFile
, i
, index
-i
))
214 FIXME("Opening a CHM file in the context of another is not supported.\n");
215 index
+= strlenW(delimW
);
223 res
= NavigateToChm(info
, info
->pCHMInfo
->szFile
, index
);
224 heap_free(default_index
);
228 ReleaseHelpViewer(info
);
234 case HH_DISPLAY_TOPIC
:
236 tab_index
= TAB_CONTENTS
;
238 case HH_DISPLAY_INDEX
:
239 tab_index
= TAB_INDEX
;
241 FIXME("Should select keyword '%s'.\n", debugstr_w((WCHAR
*)data
));
243 case HH_DISPLAY_SEARCH
:
244 tab_index
= TAB_SEARCH
;
246 FIXME("Should display search specified by HH_FTS_QUERY structure.\n");
249 /* open the requested tab */
250 memset(&nmhdr
, 0, sizeof(nmhdr
));
251 nmhdr
.code
= TCN_SELCHANGE
;
252 SendMessageW(info
->hwndTabCtrl
, TCM_SETCURSEL
, (WPARAM
)info
->tabs
[tab_index
].id
, 0);
253 SendMessageW(info
->WinType
.hwndNavigation
, WM_NOTIFY
, 0, (LPARAM
)&nmhdr
);
255 return info
->WinType
.hwndHelp
;
257 case HH_HELP_CONTEXT
: {
258 WCHAR
*window
= NULL
;
265 if (!resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
))
267 WARN("can't find %s\n", debugstr_w(filename
));
272 info
= find_window(window
);
274 info
= CreateHelpViewer(info
, fullname
, caller
);
281 if(!info
->WinType
.pszType
)
282 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
286 url
= FindContextAlias(info
->pCHMInfo
, data
);
289 if(!data
) /* there may legitimately be no context alias for id 0 */
290 return info
->WinType
.hwndHelp
;
291 ReleaseHelpViewer(info
);
295 NavigateToUrl(info
, url
);
297 return info
->WinType
.hwndHelp
;
299 case HH_PRETRANSLATEMESSAGE
: {
300 static BOOL warned
= FALSE
;
304 FIXME("HH_PRETRANSLATEMESSAGE unimplemented\n");
312 LIST_FOR_EACH_ENTRY_SAFE(info
, next
, &window_list
, HHInfo
, entry
)
314 TRACE("Destroying window %s.\n", debugstr_w(info
->WinType
.pszType
));
315 ReleaseHelpViewer(info
);
319 case HH_SET_WIN_TYPE
: {
320 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
321 WCHAR
*window
= NULL
;
324 if (!filename
&& wintype
->pszType
)
325 window
= strdupW(wintype
->pszType
);
326 else if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
328 WARN("can't find window name: %s\n", debugstr_w(filename
));
331 info
= find_window(window
);
334 info
= heap_alloc_zero(sizeof(HHInfo
));
335 info
->WinType
.pszType
= info
->stringsW
.pszType
= window
;
336 list_add_tail(&window_list
, &info
->entry
);
341 TRACE("Changing WINTYPE, fsValidMembers=0x%x\n", wintype
->fsValidMembers
);
343 MergeChmProperties(wintype
, info
, TRUE
);
344 UpdateHelpWindow(info
);
347 case HH_GET_WIN_TYPE
: {
348 HH_WINTYPEW
*wintype
= (HH_WINTYPEW
*)data
;
349 WCHAR
*window
= NULL
;
352 if (!filename
|| !resolve_filename(filename
, fullname
, MAX_PATH
, NULL
, &window
) || !window
)
354 WARN("can't find window name: %s\n", debugstr_w(filename
));
357 info
= find_window(window
);
360 WARN("Could not find window named %s.\n", debugstr_w(window
));
365 TRACE("Retrieving WINTYPE for %s.\n", debugstr_w(window
));
366 *wintype
= info
->WinType
;
371 FIXME("HH case %s not handled.\n", command_to_string( command
));
377 static void wintypeAtoW(const HH_WINTYPEA
*data
, HH_WINTYPEW
*wdata
, struct wintype_stringsW
*stringsW
)
379 memcpy(wdata
, data
, sizeof(*data
));
380 /* convert all of the ANSI strings to Unicode */
381 wdata
->pszType
= stringsW
->pszType
= strdupAtoW(data
->pszType
);
382 wdata
->pszCaption
= stringsW
->pszCaption
= strdupAtoW(data
->pszCaption
);
383 wdata
->pszToc
= stringsW
->pszToc
= strdupAtoW(data
->pszToc
);
384 wdata
->pszIndex
= stringsW
->pszIndex
= strdupAtoW(data
->pszIndex
);
385 wdata
->pszFile
= stringsW
->pszFile
= strdupAtoW(data
->pszFile
);
386 wdata
->pszHome
= stringsW
->pszHome
= strdupAtoW(data
->pszHome
);
387 wdata
->pszJump1
= stringsW
->pszJump1
= strdupAtoW(data
->pszJump1
);
388 wdata
->pszJump2
= stringsW
->pszJump2
= strdupAtoW(data
->pszJump2
);
389 wdata
->pszUrlJump1
= stringsW
->pszUrlJump1
= strdupAtoW(data
->pszUrlJump1
);
390 wdata
->pszUrlJump2
= stringsW
->pszUrlJump2
= strdupAtoW(data
->pszUrlJump2
);
391 wdata
->pszCustomTabs
= stringsW
->pszCustomTabs
= strdupAtoW(data
->pszCustomTabs
);
394 static void wintypeWtoA(const HH_WINTYPEW
*wdata
, HH_WINTYPEA
*data
, struct wintype_stringsA
*stringsA
)
396 memcpy(data
, wdata
, sizeof(*wdata
));
397 /* convert all of the Unicode strings to ANSI */
398 data
->pszType
= stringsA
->pszType
= strdupWtoA(wdata
->pszType
);
399 data
->pszCaption
= stringsA
->pszCaption
= strdupWtoA(wdata
->pszCaption
);
400 data
->pszToc
= stringsA
->pszToc
= strdupWtoA(wdata
->pszToc
);
401 data
->pszIndex
= stringsA
->pszFile
= strdupWtoA(wdata
->pszIndex
);
402 data
->pszFile
= stringsA
->pszFile
= strdupWtoA(wdata
->pszFile
);
403 data
->pszHome
= stringsA
->pszHome
= strdupWtoA(wdata
->pszHome
);
404 data
->pszJump1
= stringsA
->pszJump1
= strdupWtoA(wdata
->pszJump1
);
405 data
->pszJump2
= stringsA
->pszJump2
= strdupWtoA(wdata
->pszJump2
);
406 data
->pszUrlJump1
= stringsA
->pszUrlJump1
= strdupWtoA(wdata
->pszUrlJump1
);
407 data
->pszUrlJump2
= stringsA
->pszUrlJump2
= strdupWtoA(wdata
->pszUrlJump2
);
408 data
->pszCustomTabs
= stringsA
->pszCustomTabs
= strdupWtoA(wdata
->pszCustomTabs
);
411 /******************************************************************
412 * HtmlHelpA (HHCTRL.OCX.14)
414 HWND WINAPI
HtmlHelpA(HWND caller
, LPCSTR filename
, UINT command
, DWORD_PTR data
)
416 WCHAR
*wfile
= strdupAtoW( filename
);
423 case HH_ALINK_LOOKUP
:
424 case HH_DISPLAY_SEARCH
:
425 case HH_DISPLAY_TEXT_POPUP
:
426 case HH_GET_LAST_ERROR
:
427 case HH_KEYWORD_LOOKUP
:
429 FIXME("structures not handled yet\n");
432 case HH_SET_WIN_TYPE
:
434 struct wintype_stringsW stringsW
;
437 wintypeAtoW((HH_WINTYPEA
*)data
, &wdata
, &stringsW
);
438 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
439 wintype_stringsW_free(&stringsW
);
442 case HH_GET_WIN_TYPE
:
447 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)&wdata
);
448 if (!wdata
.pszType
) break;
449 info
= find_window(wdata
.pszType
);
451 wintype_stringsA_free(&info
->stringsA
);
452 wintypeWtoA(&wdata
, (HH_WINTYPEA
*)data
, &info
->stringsA
);
456 case HH_DISPLAY_INDEX
:
457 case HH_DISPLAY_TOPIC
:
459 case HH_GET_WIN_HANDLE
:
460 case HH_SAFE_DISPLAY_TOPIC
:
462 WCHAR
*wdata
= strdupAtoW( (const char *)data
);
463 result
= HtmlHelpW( caller
, wfile
, command
, (DWORD_PTR
)wdata
);
469 case HH_HELP_CONTEXT
:
471 case HH_PRETRANSLATEMESSAGE
:
472 case HH_TP_HELP_CONTEXTMENU
:
473 case HH_TP_HELP_WM_HELP
:
474 case HH_UNINITIALIZE
:
475 /* either scalar or pointer to scalar - do nothing */
479 FIXME("Unknown command: %s (%d)\n", command_to_string(command
), command
);
484 result
= HtmlHelpW( caller
, wfile
, command
, data
);
490 /******************************************************************
491 * doWinMain (HHCTRL.OCX.13)
493 int WINAPI
doWinMain(HINSTANCE hInstance
, LPSTR szCmdLine
)
496 int len
, buflen
, mapid
= -1;
503 /* Parse command line option of the HTML Help command.
505 * Note: The only currently handled action is "mapid",
506 * which corresponds to opening a specific page.
508 while(*szCmdLine
== '-')
513 space
= strchr(ptr
, ' ');
514 if(!strncmp(ptr
, "mapid", space
-ptr
))
518 ptr
+= strlen("mapid")+1;
519 space
= strchr(ptr
, ' ');
520 /* command line ends without number */
523 memcpy(idtxt
, ptr
, space
-ptr
);
524 idtxt
[space
-ptr
] = '\0';
530 FIXME("Unhandled HTML Help command line parameter! (%.*s)\n", (int)(space
-szCmdLine
), szCmdLine
);
535 /* FIXME: Check szCmdLine for bad arguments */
536 if (*szCmdLine
== '\"')
537 endq
= strchr(++szCmdLine
, '\"');
540 len
= endq
- szCmdLine
;
542 len
= strlen(szCmdLine
);
544 /* no filename given */
548 buflen
= MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, NULL
, 0) + 1;
549 filename
= heap_alloc(buflen
* sizeof(WCHAR
));
550 MultiByteToWideChar(CP_ACP
, 0, szCmdLine
, len
, filename
, buflen
);
551 filename
[buflen
-1] = 0;
553 /* Open a specific help topic */
555 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_HELP_CONTEXT
, mapid
);
557 hwnd
= HtmlHelpW(GetDesktopWindow(), filename
, HH_DISPLAY_TOPIC
, 0);
563 ERR("Failed to open HTML Help file '%s'.\n", szCmdLine
);
567 while (GetMessageW(&msg
, 0, 0, 0))
569 TranslateMessage(&msg
);
570 DispatchMessageW(&msg
);
576 /******************************************************************
577 * DllGetClassObject (HHCTRL.OCX.@)
579 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID riid
, LPVOID
*ppv
)
581 FIXME("(%s %s %p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
), ppv
);
582 return CLASS_E_CLASSNOTAVAILABLE
;
585 /***********************************************************************
586 * DllRegisterServer (HHCTRL.OCX.@)
588 HRESULT WINAPI
DllRegisterServer(void)
590 return __wine_register_resources( hhctrl_hinstance
);
593 /***********************************************************************
594 * DllUnregisterServer (HHCTRL.OCX.@)
596 HRESULT WINAPI
DllUnregisterServer(void)
598 return __wine_unregister_resources( hhctrl_hinstance
);