|
|||||||||||||||||||||||||||||||
|
|||||||||||||||||||||||||||||||
|
Announcements
Want a new Job?
Chapters
Services
Feature Zones
|
IntroductionAfter Oct. 2001, you cannot get help by pressing F1 in VC6 with latest MSDN. Microsoft has changed the help format from CHM to document explorer.
DetailsFirst, we must know how VC6 brings up the help after you press F1. This is easy if you have a debugger such as Soft-ICE. If you don't install MSDN, after you press F1, VC6 will popup a message box which says you haven't installed MSDN. Set a breakpoint at Then we use Dependency Walker ("depends" in Visual Studio Tools) to see what functions are exported. We will see Use VC6 to create a simple console project. Add the following line to your .cpp file. #import "C:\Program Files\Common Files\Microsoft Shared\VS98\vshelp.dll" And build your project. Open the Debug directory, and you will find "vshelp.tlh" and "vshelp.tli". Open "vshelp.tlh", you will see: struct __declspec(uuid("854d7ac0-bc3d-11d0-b421-00a0c90f9dc4"))
IVsHelpSystem : IUnknown
{
//
// Wrapper methods for error-handling
//
HRESULT KeywordSearch (
LPWSTR pszKeyword,
long dwFlags,
long dwReserved );
HRESULT ALinkSearch (
LPWSTR pszALink,
long dwFlags,
long dwReserved );
...
The function Then how to do this? Since it's a COM interface, it's easy to hook it. Basically a COM interface is a C++ class with a virtual table. It has a pointer to a table that contains the address of all virtual functions. And this table is shared by all instances. So we only need to change the function address in the table. And this interface will look like: struct _IVsHelpSystemVtbl{ HRESULT (STDMETHODCALLTYPE *QueryInterface)(IUnknown * This, REFIID riid, void **ppvObject); ULONG (STDMETHODCALLTYPE *AddRef)(IUnknown * This); ULONG (STDMETHODCALLTYPE *Release)(IUnknown * This); HRESULT (STDMETHODCALLTYPE *KeywordSearch)(IUnknown * This, LPWSTR pszKeyword, long dwFlags, long dwReserved); }; struct _IVsHelpSystem { struct _IVsHelpSystemVtbl *lpVtbl; }; What we need to do now is create an instance of this COM. And we will get the address of the table. Now we need to know how to bring up the help of the latest MSDN. It seems it's not documented. It's also a COM. Add the following line to your .cpp file. You will get the COM definition. #import "C:\Program Files\Common Files\Microsoft Shared\MSEnv\vshelp.tlb" In "vshelp.tlh", you can find function: HRESULT DisplayTopicFromF1Keyword ( _bstr_t pszKeyword ); Obviously, this is what we need. So in the function Here is the code to hook the function HRESULT hr = theHelp.CreateInstance(__uuidof(VsHelp::DExploreAppObj)); if (SUCCEEDED(hr)) { HRESULT hr = vc6Help.CreateInstance( __uuidof(VsHelpServices::VsHelpServices)); if (SUCCEEDED(hr)) { iHelp = (_IVsHelpSystem *)vc6Help.GetInterfacePtr(); TRACE1("iHelp = %x\n", iHelp); TRACE1("lpVtbl = %x\n", iHelp->lpVtbl); TRACE1("KeywordSearch = %x\n", iHelp->lpVtbl->KeywordSearch); OldKeywordSearch = iHelp->lpVtbl->KeywordSearch; DWORD dwOldProtect; if (VirtualProtect(iHelp->lpVtbl, sizeof( _IVsHelpSystemVtbl), PAGE_READWRITE, &dwOldProtect)) iHelp->lpVtbl->KeywordSearch = MyKeywordSearch; } } Here is the code of HRESULT __stdcall MyKeywordSearch(IUnknown * This, LPWSTR pszKeyword, long dwFlags, long dwReserved) { ASSERT(theHelp != NULL); LONG lResult; HKEY hKey; lResult = RegOpenKeyEx(HKEY_CURRENT_USER, "Software\\Microsoft\\Dependency Walker\\External Help", 0, KEY_READ, &hKey); if (lResult == ERROR_SUCCESS) { CHAR szCollection[MAX_PATH]; DWORD cbCollection = MAX_PATH; lResult = RegQueryValueEx(hKey, "Collection", NULL, NULL, (LPBYTE)szCollection, &cbCollection); if (lResult == ERROR_SUCCESS) { TRACE1("use collection: %s\n", szCollection); if (stricmp(szCollection, "Online") == 0) { WCHAR szURL[1024]; DWORD dwURL = sizeof(szURL); lResult = RegQueryValueExW(hKey, L"URL", NULL, NULL, (LPBYTE)szURL, &dwURL); if (lResult == ERROR_SUCCESS) { WCHAR *p = wcsstr(szURL, L"%1"); if (p) { p[1] = L's'; WCHAR szLink[1024]; if (_snwprintf(szLink, sizeof(szLink)/sizeof(WCHAR), szURL, pszKeyword) > 0) { TRACE1("use collection: %S\n", szLink); ShellExecuteW(NULL, L"open", szLink, NULL, NULL, SW_SHOWNORMAL); RegCloseKey(hKey); return S_OK; } } } } /* end Online */ else { HKEY hHelp; lResult = RegOpenKeyEx(HKEY_LOCAL_MACHINE, "SOFTWARE\\Microsoft\\MSDN\\7.0\\Help\\0x0409", 0, KEY_READ|KEY_ENUMERATE_SUB_KEYS, &hHelp); if (lResult == ERROR_SUCCESS) { DWORD dwIndex = 0; CHAR szGuid[MAX_PATH]; while (RegEnumKey(hHelp, dwIndex++, szGuid, MAX_PATH) == ERROR_SUCCESS) { HKEY hGuid; if (RegOpenKeyEx(hHelp, szGuid, 0, KEY_READ, &hGuid) != ERROR_SUCCESS) continue; CHAR szCollection2[MAX_PATH]; DWORD cbCollection2 = MAX_PATH; lResult = RegQueryValueEx(hGuid, NULL, NULL, NULL, (LPBYTE)szCollection2, &cbCollection2); if (lResult == ERROR_SUCCESS && stricmp(szCollection, szCollection2) == 0) { cbCollection2 = MAX_PATH; lResult = RegQueryValueEx(hGuid, "Filename", NULL, NULL, (LPBYTE)szCollection2, &cbCollection2); if (lResult == ERROR_SUCCESS && strnicmp(szCollection2, "ms-help://", sizeof("ms-help://")-1) == 0) { TRACE1("use collection: %s\n", szCollection2); theHelp->SetCollection(szCollection2, ""); RegCloseKey(hGuid); break; } } RegCloseKey(hGuid); } /* end RegEnumKey(hHelp) */ RegCloseKey(hHelp); } } /* end not Online */ } RegCloseKey(hKey); } theHelp->SyncIndex(pszKeyword, 1); theHelp->DisplayTopicFromF1Keyword(pszKeyword); return S_OK; } Easy, right? There are two things you must keep in mind. First, we must declare the vc6Help in global space and release it when the program quits. This is because the vshelp.dll will be freed if there is no more instance. If so, then what we modified will be gone with it. Then second, we must call Sometimes, you may have a couple of help collections installed. After you press F1, it may not show the help collection you want. How to set the default help collection?
How to use it?Click VC6's menu "Tools"->"Customize"->"Add-ins and Macro Files"->"Browse". Then choose "VSNetHelp.dll", and click "Close". Move the caret to a keyword, press F1. You got it!
|
||||||||||||||||||||||||||||||