仿腾讯 QQ 和 Skype 通过URL触发自己的程序



本文介绍如何用 URLSearchHook 接口实现自定义Web协议。当浏览器企图去打开一个未知协议的URL地址时,浏览器首先尝试从这个地址得到当前的协议,如果不成功,浏览器将创建在系统中注册的URL Search Hook对象并调用每一个对象的Translate方法,直到地址被转换或所有的URL Search Hook都尝试过。   也就是说,我们可以注册一种目前不存在的协议(类似HTTP),当浏览器遇到新的协议时会自动调用Translate方法来翻译我们的协议,甚至激活我们自己的程序。

如果你电脑中装有QQ,在IE地址栏输入:“tencent://Message/?menu=yes&exe=&uin=13231462”然后[回车],立即可以与我的QQ建立临时会话,如下图:

仿腾讯 QQ 和 Skype 通过URL触发自己的程序_第1张图片 

Skype也有类似的功能。到底是如何实现的呢?看MSDN中有这么一段话:

The IURLSearchHook interface is used by the browser to translate the address of an unknown URL protocol. When attempting to browse to a URL address that does not contain a protocol, the browser will first attempt to determine the correct protocol from the address. If this is not successful, the browser will create URL Search Hook objects and call each object's Translate method until the address is translated or all of the hooks have been queried.

IURLSearchHook接口被浏览器用来转换一个未知的URL协议地址。当浏览器企图去打开一个未知协议的URL地址时,浏览器首先尝试从这个地址得到当前的协议,如果不成功,浏览器将创建在系统中注册的URL Search Hook对象并调用每一个对象的Translate方法,直到地址被转换或所有的URL Search Hook都尝试过。

也就是说,我们可以注册一种目前不存在的协议(类似HTTP),当浏览器遇到新的协议时会自动调用Translate方法来翻译我们的协议,甚至激活我们自己的程序。

以下源代码将实现注册一个新的自定义的Web协议:

view source
print ?
01. //
02. // 注册自定义的Web协议
03. // return : ------------------------------------------------------------------------
04. //      0   -   失败
05. //      1   -   成功
06. //      2   -   已经存在
07. //
08. int RegWebProtocol ( LPCTSTR lpszProtocolName, LPCTSTR lpszAssociatedApp, int nIconIndex/*=0*/ )
09. {
10.     if ( !lpszProtocolName ||
11.          lstrlen(lpszProtocolName) < 1 ||
12.          !lpszAssociatedApp ||
13.          lstrlen(lpszAssociatedApp) < 1 )  
14.         return 0;
15.  
16.     CString csSubKey;
17.     DWORD dwBufSize = 0;
18.  
19.     // 该协议已经存在
20.     HKEY hKey = NULL;
21.     if ( RegOpenKeyEx ( HKEY_CLASSES_ROOT,
22.                 lpszProtocolName,
23.                 0,
24.                 KEY_ALL_ACCESS,
25.                 &hKey ) == ERROR_SUCCESS )
26.     {
27.         return 2;
28.     }
29.     else hKey = NULL;
30.  
31.     // 创建协议子键
32.     if ( !CreateRegisterSubKey ( HKEY_CLASSES_ROOT, lpszProtocolName ) )
33.         return 0;
34.  
35.     // 设置协议描述字符串
36.     CString csProtocolDesc; csProtocolDesc.Format ( _T("%sProtocol"), lpszProtocolName );
37.     dwBufSize = csProtocolDesc.GetLength();
38.     if ( !WriteRegister ( HKEY_CLASSES_ROOT, lpszProtocolName,
39.         _T(""), REG_EXPAND_SZ, (PUCHAR)csProtocolDesc.GetBuffer(0),&dwBufSize) )
40.         return 0;
41.     CString csAppFile; csAppFile.Format ( _T("%s"), lpszAssociatedApp );
42.     dwBufSize = csAppFile.GetLength();
43.     if ( !WriteRegister ( HKEY_CLASSES_ROOT, lpszProtocolName,
44.         _T("URL Protocol"), REG_EXPAND_SZ, (PUCHAR)csAppFile.GetBuffer(0),&dwBufSize) )
45.         return 0;
46.  
47.     // DefaultIcon 子键
48.     csSubKey.Format ( _T("%s\\DefaultIcon"), lpszProtocolName );
49.     if ( !CreateRegisterSubKey ( HKEY_CLASSES_ROOT, csSubKey ) )
50.         return 0;
51.     CString csIconParameter; csIconParameter.Format ( _T("%s,%d"), lpszAssociatedApp, nIconIndex );
52.     dwBufSize = csIconParameter.GetLength();
53.     if ( !WriteRegister ( HKEY_CLASSES_ROOT, csSubKey,
54.         _T(""), REG_EXPAND_SZ, (PUCHAR)csIconParameter.GetBuffer(0),&dwBufSize) )
55.         return 0;
56.  
57.     // shell\open\command 子键
58.     csSubKey.Format ( _T("%s\\shell\\open\\command"), lpszProtocolName );
59.     if ( !CreateRegisterSubKey ( HKEY_CLASSES_ROOT, csSubKey ) )
60.         return 0;
61.     CString csCommand; csCommand.Format ( _T("\"%s\" \"%%1\""), lpszAssociatedApp );
62.     dwBufSize = csCommand.GetLength();
63.     if ( !WriteRegister ( HKEY_CLASSES_ROOT, csSubKey,
64.         _T(""), REG_EXPAND_SZ, (PUCHAR)csCommand.GetBuffer(0),&dwBufSize) )
65.         return 0;
66.  
67.     return 1;
68. }

以下源代码将删除自定义的Web协议:

view source
print ?
01. //
02. // 卸载自定义的Web协议
03. //
04. BOOL UnRegWebProtocol ( LPCTSTR lpszProtocolName )
05. {  
06.     if ( !lpszProtocolName || lstrlen(lpszProtocolName) < 1 )
07.         return FALSE;
08.  
09.     return RegDeleteAllSubKey ( HKEY_CLASSES_ROOT, lpszProtocolName );
10. }

我提供一个详细的源代码可以演示整个过程,程序界面如下:

仿腾讯 QQ 和 Skype 通过URL触发自己的程序_第2张图片

先设置好需要注册的协议名称和关联的执行程序文件名,然后点[Register Protocol]按钮完成注册,点击[Test Protocol]按钮测试注册后的效果。

如本例中注册一个“HXXP”协议,关联程序就是本程序,当点击[Test Protocol]按钮时就会执行该程序的另一个副本,同时[Test URL Parameter]输入的参数也被显示出来了。您也可以将“HXXP://Para1/Para2/Para2”复制到IE地址栏中执行,实现相同的效果,如下图:

仿腾讯 QQ 和 Skype 通过URL触发自己的程序_第3张图片 

你可能感兴趣的:(仿腾讯 QQ 和 Skype 通过URL触发自己的程序)