分类: C/C++
2008-03-10 20:00:19
如何在注册表中查找默认浏览器位置的定义?我需要知道EXE文件的路径和名称以便启动一个应用程序会话。我的目的很简单,就是打开默认的浏览器,这样用户能够象普通程序一样使用它,而不是在我设计的程序窗口内浏览因特网。 Rolf Wenger
据我所知,在Window中没有专门指定默认浏览器的注册表键值或设定值。即使专家也很难弄清楚整个注册表,更何况常人。我知道可能存在这样一个键值, HKCU\System\Mumble\Bletch\Blah\Gak\DefaultBrowser 如果你知道这样的键值,请写信告诉我。不过,我知道一个简单的解决办法,那就是查找哪个程序和HTML文件相关联。在Window操作系统中HTML文件的后缀通常为.htm和.html,所以你要做的就是查找HKCR/.htm的键值。如果你查找了会找到下面的键值: HKEY_CLASSES_ROOT\.htm="htmlfile" 再根据这个键值查找HKCR/htmlfilm的条目,你会找到下面的键值: [HKEY_CLASSES_ROOT\htmlfile\shell\open\command] @="\"C:\\PROGRA~1\\INTERN~1\\iexplore.exe\" -nohome" 这个键值表明Microsoft Internet Explorer (iexplore.exe)是用来打开.htm文件的程序。(-nohome开关标志告诉IE浏览器不要打开主页)如果默认的浏览器是Netscape,这个条目会是这样: [HKEY_CLASSES_ROOT\htmlfile\shell\open\command] @="\"C:\\PROGRA~1\\NETSCAPE\\netscape.exe\". 我的回答满意吗? Rolf Wenger 有两种方式可以改变窗口中的光标:你可以在注册窗口类时声明一个全局光标(HCURSOR)作为WNDCLASS结构的一部分,或者通过处理WM_SETCURSOR消息来手工设置光标。标准的MFC程序采用第一种方式设置光标,它自动为主窗口注册一个箭头光标。你可以在主窗口或子窗口中通过处理WM_SETCURSOR消息来改写这个行为。 // 在按钮类中处理WM_SETCURSOR消息 BOOL CMyButton::OnSetCursor(CWnd* pWnd, UINT nHitTest, UINT msg) { ::SetCursor(m_hMyCursor); 不管什么时候,当用户将鼠标移动到按钮上并且鼠标没有被捕获时,Windows 会发送一条WM_SETCURSOR消息给按钮。它传递一个窗口句柄——即鼠标指针指向的窗口, 此时就是按钮本身;击中测试码——即在WM_NCHITTEST消息中使用的HTXXX码(见 Figure 1);和一个触发事件的消息ID,比如说它触发了WM_MOUSEMOVE事件。设置鼠标 光标的最佳机会就是在处理WM_SETCURSOR消息的时候。如果要这么做,你必须返回TRUE以阻止窗口默认的处理过程。 Hans Zwahlen 唉,这个问题没有特别令人满意的答案,只有一些曲线救国的办法。该问题的实质在于Windows中的每个资源必须从属于某个模块(EXE或DLL),而在每一个模块中,特定资源不能有相同的名称或ID。就DLL(动态链接库)而言不存在这样的问题,因为DLL本身就有和主程序不同的HINSTANCE句柄。但对静态链接库来说,所有资源在同一个EXE文件内共存,就像一个大家庭的成员共同生活在一起一样。当然,像家庭成员可能发生冲突一样,资源 也存在发生冲突的可能性。MFC是如何解决这个问题的呢?它采用800磅重的大猩猩的方法,“我是800磅重的大猩猩,身强体壮,你们打不赢我,最好乖乖听我的。”据我所知MFC没有任何内置的对话框,但有一些通用的菜单项ID,像ID_FILE_OPEN。这些ID也被用作菜单项提示字符串的资源ID。所有MFC的ID值都比0xE000大。最好不要使用大于该值的ID,否则产生冲突就是你的问题。 这种方法对编写操作系统的人来说可能是适合的,但如果你只是一个努力赚钱养家糊口的人,当顾客抱怨你的ID和他的程序有冲突的时候,最好不要“嘴硬”, 你必须要想方设法解决这个问题,要不然怎么办? 我曾经用过的一个办法是:把库中所有资源ID规定为一个已初始化基本值的偏移量,在发生冲突时程序员能够改变这个基本值。可以在一个特殊的头文件中规定这个值,然后程序员必须同时在主程序文件和资源文件中包含它。 // 在主应用程序rc文件 #include "libres.h" // ID 标识符 #include "libres.rc" // 资源这里libres.rc是包含库中所有对话框、光标和其他资源的资源文件。头文件libres.h以基本值偏移量的方式规定了libres.res中使用的资源ID。 // 在libres.h文件中 #ifndef LIBBASEID #define LIBBASEID 2000 // 或其他值 #endif #define IDR_FOO (LIBBASEID+1) #define IDR_BAR (LIBBASEID+2) // 等等只要改变LIBBASEID的值,使用这个库的程序员可以方便地重新映射所有资源ID,例如: // 在主程序rc文件 #define LIBBASEID 3000 #include "libres.h" #include "libres.rc"现在所有的ID从3000开始而不是2000,在实际当中这个办法很有效,但它有一个麻烦的问题,那就是它需要用户重新编译库以改变ID。如果重新编译库是不可行的(也许你不想提供源代码),还有一些其 它的办法。要是程序员改变了LIBBASEID,只有当他直接使用IDR_FOO或其他资源ID时,才需要重新编译库。也就是说,在库中直接引用了这些资源ID: BOOL SomeLibFn(...) { DialogBox(..., IDR_FOO, ...); }可以不直接引用这些资源ID,方式之一就是把它们作为调用函数的参数传递。 BOOL SomeLibFn(..., UINT nDlgID) { DialogBox(..., nDlgID, ...); }现在你不用担心库中的资源ID会发生冲突,因为主程序一定支持它们。改变LIBBASEID值的用户不用也不能重新编译库,因为所有ID是作为函数值传入的。当然这个方法要求编写主程序的程序员必须编辑库中的rc文件以设置ID,还是有点麻烦。更好的办法是以全局静态变量而不是固定的ID值作为偏移量的基本值。 // 在头文件中 extern UINT LibBaseId; #define IDR_FOO (LibBaseId+1); #define IDR_BAR (LibBaseId+2); // 等等 // 在库中主模块 UINT LibBaseId = 2000;假如现在发生冲突,应用程序可以在启动时改变LibBaseId值(也许是通过一个函数),然后重新编译主程序即可。唉,但这对RC文件不起作用。我曾经说过RC文件的语法并不是纯C/C++语法,而只是其一个子集,所以RC文件不知道extern和UINT是什么。要使资源编译器理解这些符号,你需要使用另外的头文件,或者采用更简便的方法,使用宏RC_INVOKED,这样这些符号就可以放在同一个头文件当中。 // 在libres.h头文件中 #ifdef RC_INVOKED #ifndef LibBaseId #define LibBaseId 2000 #endif #else extern UINT LibBaseId; #endif #define IDR_FOO (LibBaseId+1); #define IDR_BAR (LibBaseId+2); // 等等一般来说,包含在RC文件中的头文件不能包含除#define以外的C代码。采用这种方法,使用你的库的程序员必须做两件事来重新映射库中的资源ID。第一步,在把libres.h包含到主RC文件之前,他们必须重新#define LibBaseId一个新值。第二步,他们必须在主应用程序的启动代码中(比如在CWinApp::InitInstance函数中)将LibBaseId初始化为同样的值。最后,有另外一种更简单的解决资源ID冲突的办法:使用字符串而不是数值来映射资源ID。 // 在libres.h头文件中 #define IDR_FOO "MyApp_IDR_FOO" #define IDR_BAR "MyApp_IDR_BAR" //等等使用字符串映射资源ID的效率并不高,因为不仅它们占用空间多,而且查找字符串映射的资源比查找整数映射的资源慢。如果你库中有几百个用于字符串或其他使用频率高的资源ID,我建议你使用整数映射ID并且采用前面提出的一种解决方法。但假如你库中的资源就是几个对话框,那使用字符串映射资源ID是再好不过了。没有人会注意多用了5毫秒显示对话框,并且资源ID冲突的可能性几乎为零。 |
作者简介 Paul DiLascia,《Windows++: Writing Reusable Windows Code in C++》(Addison-Wesley出版公司1992出版)的作者,是一名自由作家和顾问。你可以发邮件到askpd@pobox.com或登录和他取得联系。 |