2 * Implementation of hyperlinking (hlink.dll)
4 * Copyright 2005 Aric Stewart for CodeWeavers
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
21 #include "hlink_private.h"
26 #include "wine/debug.h"
28 WINE_DEFAULT_DEBUG_CHANNEL(hlink
);
30 typedef HRESULT (CALLBACK
*LPFNCREATEINSTANCE
)(IUnknown
*, REFIID
, LPVOID
*);
34 const IClassFactoryVtbl
*lpVtbl
;
35 LPFNCREATEINSTANCE lpfnCI
;
38 BOOL WINAPI
DllMain(HINSTANCE hinstDLL
, DWORD fdwReason
, LPVOID lpvReserved
)
40 TRACE("%p %d %p\n", hinstDLL
, fdwReason
, lpvReserved
);
44 case DLL_WINE_PREATTACH
:
45 return FALSE
; /* prefer native version */
46 case DLL_PROCESS_ATTACH
:
47 DisableThreadLibraryCalls(hinstDLL
);
49 case DLL_PROCESS_DETACH
:
55 /***********************************************************************
56 * DllCanUnloadNow (HLINK.@)
58 HRESULT WINAPI
DllCanUnloadNow( void )
64 /***********************************************************************
65 * HlinkCreateFromMoniker (HLINK.@)
67 HRESULT WINAPI
HlinkCreateFromMoniker( IMoniker
*pimkTrgt
, LPCWSTR pwzLocation
,
68 LPCWSTR pwzFriendlyName
, IHlinkSite
* pihlsite
, DWORD dwSiteData
,
69 IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
74 TRACE("%p %s %s %p %i %p %s %p\n", pimkTrgt
, debugstr_w(pwzLocation
),
75 debugstr_w(pwzFriendlyName
), pihlsite
, dwSiteData
, piunkOuter
,
76 debugstr_guid(riid
), ppvObj
);
78 r
= CoCreateInstance(&CLSID_StdHlink
, piunkOuter
, CLSCTX_INPROC_SERVER
, riid
, (LPVOID
*)&hl
);
83 IHlink_SetStringReference(hl
, HLINKSETF_LOCATION
, NULL
, pwzLocation
);
85 IHlink_SetFriendlyName(hl
, pwzFriendlyName
);
87 IHlink_SetHlinkSite(hl
, pihlsite
, dwSiteData
);
89 IHlink_SetMonikerReference(hl
, 0, pimkTrgt
, pwzLocation
);
93 TRACE("Returning %i\n",r
);
98 /***********************************************************************
99 * HlinkCreateFromString (HLINK.@)
101 HRESULT WINAPI
HlinkCreateFromString( LPCWSTR pwzTarget
, LPCWSTR pwzLocation
,
102 LPCWSTR pwzFriendlyName
, IHlinkSite
* pihlsite
, DWORD dwSiteData
,
103 IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
108 TRACE("%s %s %s %p %i %p %s %p\n", debugstr_w(pwzTarget
),
109 debugstr_w(pwzLocation
), debugstr_w(pwzFriendlyName
), pihlsite
,
110 dwSiteData
, piunkOuter
, debugstr_guid(riid
), ppvObj
);
112 r
= CoCreateInstance(&CLSID_StdHlink
, piunkOuter
, CLSCTX_INPROC_SERVER
, riid
, (LPVOID
*)&hl
);
117 IHlink_SetStringReference(hl
, HLINKSETF_LOCATION
, NULL
, pwzLocation
);
121 IMoniker
*pTgtMk
= NULL
;
122 IBindCtx
*pbc
= NULL
;
125 CreateBindCtx(0, &pbc
);
126 r
= MkParseDisplayName(pbc
, pwzTarget
, &eaten
, &pTgtMk
);
127 IBindCtx_Release(pbc
);
131 LPCWSTR p
= strchrW(pwzTarget
, ':');
132 if (p
&& (p
- pwzTarget
> 1))
133 r
= CreateURLMoniker(NULL
, pwzTarget
, &pTgtMk
);
135 r
= CreateFileMoniker(pwzTarget
,&pTgtMk
);
140 ERR("couldn't create moniker for %s, failed with error 0x%08x\n",
141 debugstr_w(pwzTarget
), r
);
145 IHlink_SetMonikerReference(hl
, 0, pTgtMk
, pwzLocation
);
146 IMoniker_Release(pTgtMk
);
148 IHlink_SetStringReference(hl
, HLINKSETF_TARGET
, pwzTarget
, NULL
);
152 IHlink_SetFriendlyName(hl
, pwzFriendlyName
);
154 IHlink_SetHlinkSite(hl
, pihlsite
, dwSiteData
);
156 TRACE("Returning %i\n",r
);
163 /***********************************************************************
164 * HlinkNavigate (HLINK.@)
166 HRESULT WINAPI
HlinkCreateBrowseContext( IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
170 TRACE("%p %s %p\n", piunkOuter
, debugstr_guid(riid
), ppvObj
);
172 r
= CoCreateInstance(&CLSID_StdHlinkBrowseContext
, piunkOuter
, CLSCTX_INPROC_SERVER
, riid
, ppvObj
);
174 TRACE("returning %i\n",r
);
179 /***********************************************************************
180 * HlinkNavigate (HLINK.@)
182 HRESULT WINAPI
HlinkNavigate(IHlink
*phl
, IHlinkFrame
*phlFrame
,
183 DWORD grfHLNF
, LPBC pbc
, IBindStatusCallback
*pbsc
,
184 IHlinkBrowseContext
*phlbc
)
188 TRACE("%p %p %i %p %p %p\n", phl
, phlFrame
, grfHLNF
, pbc
, pbsc
, phlbc
);
191 r
= IHlinkFrame_Navigate(phlFrame
, grfHLNF
, pbc
, pbsc
, phl
);
193 r
= IHlink_Navigate(phl
, grfHLNF
, pbc
, pbsc
, phlbc
);
198 /***********************************************************************
199 * HlinkOnNavigate (HLINK.@)
201 HRESULT WINAPI
HlinkOnNavigate( IHlinkFrame
*phlFrame
,
202 IHlinkBrowseContext
* phlbc
, DWORD grfHLNF
, IMoniker
*pmkTarget
,
203 LPCWSTR pwzLocation
, LPCWSTR pwzFriendlyName
, ULONG
* puHLID
)
207 TRACE("%p %p %i %p %s %s %p\n",phlFrame
, phlbc
, grfHLNF
, pmkTarget
,
208 debugstr_w(pwzLocation
), debugstr_w(pwzFriendlyName
), puHLID
);
210 r
= IHlinkBrowseContext_OnNavigateHlink(phlbc
, grfHLNF
, pmkTarget
,
211 pwzLocation
, pwzFriendlyName
, puHLID
);
214 r
= IHlinkFrame_OnNavigate(phlFrame
,grfHLNF
,pmkTarget
, pwzLocation
,
220 /***********************************************************************
221 * HlinkCreateFromData (HLINK.@)
223 HRESULT WINAPI
HlinkCreateFromData(IDataObject
*piDataObj
,
224 IHlinkSite
*pihlsite
, DWORD dwSiteData
, IUnknown
*piunkOuter
,
225 REFIID riid
, void **ppvObj
)
227 FIXME("%p %p %d %p %p %p\n",
228 piDataObj
, pihlsite
, dwSiteData
, piunkOuter
, riid
, ppvObj
);
233 /***********************************************************************
234 * HlinkQueryCreateFromData (HLINK.@)
236 HRESULT WINAPI
HlinkQueryCreateFromData(IDataObject
* piDataObj
)
238 FIXME("%p\n", piDataObj
);
242 /***********************************************************************
243 * HlinkNavigateToStringReference (HLINK.@)
245 HRESULT WINAPI
HlinkNavigateToStringReference( LPCWSTR pwzTarget
,
246 LPCWSTR pwzLocation
, IHlinkSite
*pihlsite
, DWORD dwSiteData
,
247 IHlinkFrame
*pihlframe
, DWORD grfHLNF
, LPBC pibc
,
248 IBindStatusCallback
*pibsc
, IHlinkBrowseContext
*pihlbc
)
251 IHlink
*hlink
= NULL
;
253 FIXME("%s %s %p %08x %p %08x %p %p %p\n",
254 debugstr_w(pwzTarget
), debugstr_w(pwzLocation
), pihlsite
,
255 dwSiteData
, pihlframe
, grfHLNF
, pibc
, pibsc
, pihlbc
);
257 r
= HlinkCreateFromString( pwzTarget
, pwzLocation
, NULL
, pihlsite
,
258 dwSiteData
, NULL
, &IID_IHlink
, (LPVOID
*) &hlink
);
260 r
= HlinkNavigate(hlink
, pihlframe
, grfHLNF
, pibc
, pibsc
, pihlbc
);
265 /***********************************************************************
266 * HlinkIsShortcut (HLINK.@)
268 HRESULT WINAPI
HlinkIsShortcut(LPCWSTR pwzFileName
)
272 static const WCHAR url_ext
[] = {'.','u','r','l',0};
274 TRACE("(%s)\n", debugstr_w(pwzFileName
));
279 len
= strlenW(pwzFileName
)-4;
283 return strcmpiW(pwzFileName
+len
, url_ext
) ? S_FALSE
: S_OK
;
286 /***********************************************************************
287 * HlinkGetSpecialReference (HLINK.@)
289 HRESULT WINAPI
HlinkGetSpecialReference(ULONG uReference
, LPWSTR
*ppwzReference
)
291 DWORD res
, type
, size
= 100;
296 static const WCHAR start_pageW
[] = {'S','t','a','r','t',' ','P','a','g','e',0};
297 static const WCHAR search_pageW
[] = {'S','e','a','r','c','h',' ','P','a','g','e',0};
299 static const WCHAR ie_main_keyW
[] =
300 {'S','o','f','t','w','a','r','e',
301 '\\','M','i','c','r','o','s','o','f','t','\\',
302 'I','n','t','e','r','n','e','t',' ','E','x','p','l','o','r','e','r',
303 '\\','M','a','i','n',0};
305 TRACE("(%u %p)\n", uReference
, ppwzReference
);
307 *ppwzReference
= NULL
;
311 value_name
= start_pageW
;
313 case HLSR_SEARCHPAGE
:
314 value_name
= search_pageW
;
316 case HLSR_HISTORYFOLDER
:
322 res
= RegOpenKeyW(HKEY_CURRENT_USER
, ie_main_keyW
, &hkey
);
323 if(res
!= ERROR_SUCCESS
) {
324 WARN("Could not open key: %u\n", res
);
325 return HRESULT_FROM_WIN32(res
);
328 buf
= CoTaskMemAlloc(size
);
329 res
= RegQueryValueExW(hkey
, value_name
, NULL
, &type
, (PBYTE
)buf
, &size
);
330 buf
= CoTaskMemRealloc(buf
, size
);
331 if(res
== ERROR_MORE_DATA
)
332 res
= RegQueryValueExW(hkey
, value_name
, NULL
, &type
, (PBYTE
)buf
, &size
);
334 if(res
!= ERROR_SUCCESS
) {
335 WARN("Could not query value %s: %u\n", debugstr_w(value_name
), res
);
337 return HRESULT_FROM_WIN32(res
);
340 *ppwzReference
= buf
;
344 /***********************************************************************
345 * HlinkTranslateURL (HLINK.@)
347 HRESULT WINAPI
HlinkTranslateURL(LPCWSTR pwzURL
, DWORD grfFlags
, LPWSTR
*ppwzTranslatedURL
)
349 FIXME("(%s %08x %p)\n", debugstr_w(pwzURL
), grfFlags
, ppwzTranslatedURL
);
353 /***********************************************************************
354 * HlinkUpdateStackItem (HLINK.@)
356 HRESULT WINAPI
HlinkUpdateStackItem(IHlinkFrame
*pihlframe
, IHlinkBrowseContext
*pihlbc
,
357 ULONG uHLID
, IMoniker
*pimkTrgt
, LPCWSTR pwzLocation
, LPCWSTR pwzFriendlyName
)
359 FIXME("(%p %p %u %p %s %s)\n", pihlframe
, pihlbc
, uHLID
, pimkTrgt
, debugstr_w(pwzLocation
),
360 debugstr_w(pwzFriendlyName
));
364 /***********************************************************************
365 * HlinkParseDisplayName (HLINK.@)
367 HRESULT WINAPI
HlinkParseDisplayName(LPBC pibc
, LPCWSTR pwzDisplayName
, BOOL fNoForceAbs
,
368 ULONG
*pcchEaten
, IMoniker
**ppimk
)
372 TRACE("(%p %s %x %p %p)\n", pibc
, debugstr_w(pwzDisplayName
), fNoForceAbs
, pcchEaten
, ppimk
);
375 FIXME("Unsupported fNoForceAbs\n");
377 hres
= MkParseDisplayNameEx(pibc
, pwzDisplayName
, pcchEaten
, ppimk
);
381 hres
= MkParseDisplayName(pibc
, pwzDisplayName
, pcchEaten
, ppimk
);
385 hres
= CreateFileMoniker(pwzDisplayName
, ppimk
);
387 *pcchEaten
= strlenW(pwzDisplayName
);
392 /***********************************************************************
393 * HlinkResolveMonikerForData (HLINK.@)
395 HRESULT WINAPI
HlinkResolveMonikerForData(LPMONIKER pimkReference
, DWORD reserved
, LPBC pibc
,
396 ULONG cFmtetc
, FORMATETC
*rgFmtetc
, IBindStatusCallback
*pibsc
, LPMONIKER pimkBase
)
398 LPOLESTR name
= NULL
;
404 TRACE("(%p %x %p %d %p %p %p)\n", pimkReference
, reserved
, pibc
, cFmtetc
, rgFmtetc
, pibsc
, pimkBase
);
406 if(cFmtetc
|| rgFmtetc
|| pimkBase
)
407 FIXME("Unsupported args\n");
409 hres
= RegisterBindStatusCallback(pibc
, pibsc
, NULL
/* FIXME */, 0);
413 hres
= IMoniker_IsSystemMoniker(pimkReference
, &mksys
);
414 if(SUCCEEDED(hres
) && mksys
!= MKSYS_URLMONIKER
)
415 WARN("sysmk = %x\n", mksys
);
417 /* FIXME: What is it for? */
418 CreateBindCtx(0, &bctx
);
419 hres
= IMoniker_GetDisplayName(pimkReference
, bctx
, NULL
, &name
);
420 IBindCtx_Release(bctx
);
421 if(SUCCEEDED(hres
)) {
422 TRACE("got display name %s\n", debugstr_w(name
));
426 return IMoniker_BindToStorage(pimkReference
, pibc
, NULL
, &IID_IUnknown
, &obj
);
429 static HRESULT WINAPI
HLinkCF_fnQueryInterface ( LPCLASSFACTORY iface
,
430 REFIID riid
, LPVOID
*ppvObj
)
432 CFImpl
*This
= (CFImpl
*)iface
;
434 TRACE("(%p)->(%s)\n",This
,debugstr_guid(riid
));
438 if (IsEqualIID(riid
, &IID_IUnknown
) ||
439 IsEqualIID(riid
, &IID_IClassFactory
))
445 TRACE("-- E_NOINTERFACE\n");
446 return E_NOINTERFACE
;
449 static ULONG WINAPI
HLinkCF_fnAddRef (LPCLASSFACTORY iface
)
454 static ULONG WINAPI
HLinkCF_fnRelease(LPCLASSFACTORY iface
)
459 static HRESULT WINAPI
HLinkCF_fnCreateInstance( LPCLASSFACTORY iface
,
460 LPUNKNOWN pUnkOuter
, REFIID riid
, LPVOID
*ppvObject
)
462 CFImpl
*This
= (CFImpl
*)iface
;
464 TRACE("%p->(%p,%s,%p)\n", This
, pUnkOuter
, debugstr_guid(riid
), ppvObject
);
468 return This
->lpfnCI(pUnkOuter
, riid
, ppvObject
);
471 static HRESULT WINAPI
HLinkCF_fnLockServer(LPCLASSFACTORY iface
, BOOL fLock
)
473 FIXME("%p %d\n", iface
, fLock
);
477 static const IClassFactoryVtbl hlcfvt
=
479 HLinkCF_fnQueryInterface
,
482 HLinkCF_fnCreateInstance
,
486 static CFImpl HLink_cf
= { &hlcfvt
, &HLink_Constructor
};
487 static CFImpl HLinkBrowseContext_cf
= { &hlcfvt
, &HLinkBrowseContext_Constructor
};
489 /***********************************************************************
490 * DllGetClassObject (HLINK.@)
492 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID iid
, LPVOID
*ppv
)
494 IClassFactory
*pcf
= NULL
;
496 TRACE("%s %s %p\n", debugstr_guid(rclsid
), debugstr_guid(iid
), ppv
);
502 if (IsEqualIID(rclsid
, &CLSID_StdHlink
))
503 pcf
= (IClassFactory
*) &HLink_cf
;
504 else if (IsEqualIID(rclsid
, &CLSID_StdHlinkBrowseContext
))
505 pcf
= (IClassFactory
*) &HLinkBrowseContext_cf
;
507 return CLASS_E_CLASSNOTAVAILABLE
;
509 return IClassFactory_QueryInterface(pcf
, iid
, ppv
);
512 static HRESULT
register_clsid(LPCGUID guid
)
514 static const WCHAR clsid
[] =
515 {'C','L','S','I','D','\\',0};
516 static const WCHAR ips
[] =
517 {'\\','I','n','p','r','o','c','S','e','r','v','e','r','3','2',0};
518 static const WCHAR hlink
[] =
519 {'h','l','i','n','k','.','d','l','l',0};
520 static const WCHAR threading_model
[] =
521 {'T','h','r','e','a','d','i','n','g','M','o','d','e','l',0};
522 static const WCHAR apartment
[] =
523 {'A','p','a','r','t','m','e','n','t',0};
528 lstrcpyW(path
, clsid
);
529 StringFromGUID2(guid
, &path
[6], 80);
531 r
= RegCreateKeyW(HKEY_CLASSES_ROOT
, path
, &key
);
532 if (r
!= ERROR_SUCCESS
)
535 RegSetValueExW(key
, NULL
, 0, REG_SZ
, (const BYTE
*)hlink
, sizeof hlink
);
536 RegSetValueExW(key
, threading_model
, 0, REG_SZ
, (const BYTE
*)apartment
, sizeof apartment
);
542 /***********************************************************************
543 * DllRegisterServer (HLINK.@)
545 HRESULT WINAPI
DllRegisterServer(void)
549 r
= register_clsid(&CLSID_StdHlink
);
551 r
= register_clsid(&CLSID_StdHlinkBrowseContext
);