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"
27 #include "wine/debug.h"
29 WINE_DEFAULT_DEBUG_CHANNEL(hlink
);
31 typedef HRESULT (*LPFNCREATEINSTANCE
)(IUnknown
*, REFIID
, LPVOID
*);
35 IClassFactory IClassFactory_iface
;
36 LPFNCREATEINSTANCE lpfnCI
;
39 static inline CFImpl
*impl_from_IClassFactory(IClassFactory
*iface
)
41 return CONTAINING_RECORD(iface
, CFImpl
, IClassFactory_iface
);
44 /***********************************************************************
45 * HlinkCreateFromMoniker (HLINK.@)
47 HRESULT WINAPI
HlinkCreateFromMoniker( IMoniker
*pimkTrgt
, LPCWSTR pwzLocation
,
48 LPCWSTR pwzFriendlyName
, IHlinkSite
* pihlsite
, DWORD dwSiteData
,
49 IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
54 TRACE("%p %s %s %p %li %p %s %p\n", pimkTrgt
, debugstr_w(pwzLocation
),
55 debugstr_w(pwzFriendlyName
), pihlsite
, dwSiteData
, piunkOuter
,
56 debugstr_guid(riid
), ppvObj
);
58 hr
= CoCreateInstance(&CLSID_StdHlink
, piunkOuter
, CLSCTX_INPROC_SERVER
, &IID_IHlink
, (LPVOID
*)&hl
);
62 IHlink_SetMonikerReference(hl
, HLINKSETF_LOCATION
| HLINKSETF_TARGET
, pimkTrgt
, pwzLocation
);
65 IHlink_SetFriendlyName(hl
, pwzFriendlyName
);
67 IHlink_SetHlinkSite(hl
, pihlsite
, dwSiteData
);
69 hr
= IHlink_QueryInterface(hl
, riid
, ppvObj
);
75 /***********************************************************************
76 * HlinkCreateFromString (HLINK.@)
78 HRESULT WINAPI
HlinkCreateFromString( LPCWSTR pwzTarget
, LPCWSTR pwzLocation
,
79 LPCWSTR pwzFriendlyName
, IHlinkSite
* pihlsite
, DWORD dwSiteData
,
80 IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
87 TRACE("%s %s %s %p %li %p %s %p\n", debugstr_w(pwzTarget
),
88 debugstr_w(pwzLocation
), debugstr_w(pwzFriendlyName
), pihlsite
,
89 dwSiteData
, piunkOuter
, debugstr_guid(riid
), ppvObj
);
91 hr
= CoCreateInstance(&CLSID_StdHlink
, piunkOuter
, CLSCTX_INPROC_SERVER
, &IID_IHlink
, (void **)&hl
);
97 hash
= wcschr(pwzTarget
, '#');
100 if (hash
== pwzTarget
)
104 int tgt_len
= hash
- pwzTarget
;
105 tgt
= malloc((tgt_len
+ 1) * sizeof(WCHAR
));
107 return E_OUTOFMEMORY
;
108 memcpy(tgt
, pwzTarget
, tgt_len
* sizeof(WCHAR
));
118 tgt
= wcsdup(pwzTarget
);
120 return E_OUTOFMEMORY
;
130 IHlink_SetStringReference(hl
, HLINKSETF_TARGET
| HLINKSETF_LOCATION
, tgt
, loc
);
135 IHlink_SetFriendlyName(hl
, pwzFriendlyName
);
138 IHlink_SetHlinkSite(hl
, pihlsite
, dwSiteData
);
140 hr
= IHlink_QueryInterface(hl
, riid
, ppvObj
);
147 /***********************************************************************
148 * HlinkCreateBrowseContext (HLINK.@)
150 HRESULT WINAPI
HlinkCreateBrowseContext( IUnknown
* piunkOuter
, REFIID riid
, void** ppvObj
)
152 TRACE("%p %s %p\n", piunkOuter
, debugstr_guid(riid
), ppvObj
);
153 return CoCreateInstance(&CLSID_StdHlinkBrowseContext
, piunkOuter
, CLSCTX_INPROC_SERVER
, riid
, ppvObj
);
156 /***********************************************************************
157 * HlinkNavigate (HLINK.@)
159 HRESULT WINAPI
HlinkNavigate(IHlink
*phl
, IHlinkFrame
*phlFrame
,
160 DWORD grfHLNF
, LPBC pbc
, IBindStatusCallback
*pbsc
,
161 IHlinkBrowseContext
*phlbc
)
165 TRACE("%p %p %li %p %p %p\n", phl
, phlFrame
, grfHLNF
, pbc
, pbsc
, phlbc
);
168 r
= IHlinkFrame_Navigate(phlFrame
, grfHLNF
, pbc
, pbsc
, phl
);
170 r
= IHlink_Navigate(phl
, grfHLNF
, pbc
, pbsc
, phlbc
);
175 /***********************************************************************
176 * HlinkOnNavigate (HLINK.@)
178 HRESULT WINAPI
HlinkOnNavigate( IHlinkFrame
*phlFrame
,
179 IHlinkBrowseContext
* phlbc
, DWORD grfHLNF
, IMoniker
*pmkTarget
,
180 LPCWSTR pwzLocation
, LPCWSTR pwzFriendlyName
, ULONG
* puHLID
)
184 TRACE("%p %p %li %p %s %s %p\n",phlFrame
, phlbc
, grfHLNF
, pmkTarget
,
185 debugstr_w(pwzLocation
), debugstr_w(pwzFriendlyName
), puHLID
);
187 r
= IHlinkBrowseContext_OnNavigateHlink(phlbc
, grfHLNF
, pmkTarget
,
188 pwzLocation
, pwzFriendlyName
, puHLID
);
191 r
= IHlinkFrame_OnNavigate(phlFrame
,grfHLNF
,pmkTarget
, pwzLocation
,
197 /***********************************************************************
198 * HlinkCreateFromData (HLINK.@)
200 HRESULT WINAPI
HlinkCreateFromData(IDataObject
*piDataObj
,
201 IHlinkSite
*pihlsite
, DWORD dwSiteData
, IUnknown
*piunkOuter
,
202 REFIID riid
, void **ppvObj
)
204 FIXME("%p, %p, %ld, %p, %s, %p\n", piDataObj
, pihlsite
, dwSiteData
,
205 piunkOuter
, debugstr_guid(riid
), ppvObj
);
210 /***********************************************************************
211 * HlinkQueryCreateFromData (HLINK.@)
213 HRESULT WINAPI
HlinkQueryCreateFromData(IDataObject
* piDataObj
)
215 FIXME("%p\n", piDataObj
);
219 /***********************************************************************
220 * HlinkNavigateToStringReference (HLINK.@)
222 HRESULT WINAPI
HlinkNavigateToStringReference( LPCWSTR pwzTarget
,
223 LPCWSTR pwzLocation
, IHlinkSite
*pihlsite
, DWORD dwSiteData
,
224 IHlinkFrame
*pihlframe
, DWORD grfHLNF
, LPBC pibc
,
225 IBindStatusCallback
*pibsc
, IHlinkBrowseContext
*pihlbc
)
228 IHlink
*hlink
= NULL
;
230 TRACE("%s %s %p %08lx %p %08lx %p %p %p\n",
231 debugstr_w(pwzTarget
), debugstr_w(pwzLocation
), pihlsite
,
232 dwSiteData
, pihlframe
, grfHLNF
, pibc
, pibsc
, pihlbc
);
234 r
= HlinkCreateFromString( pwzTarget
, pwzLocation
, NULL
, pihlsite
,
235 dwSiteData
, NULL
, &IID_IHlink
, (LPVOID
*) &hlink
);
237 r
= HlinkNavigate(hlink
, pihlframe
, grfHLNF
, pibc
, pibsc
, pihlbc
);
238 IHlink_Release(hlink
);
244 /***********************************************************************
245 * HlinkIsShortcut (HLINK.@)
247 HRESULT WINAPI
HlinkIsShortcut(LPCWSTR pwzFileName
)
251 TRACE("(%s)\n", debugstr_w(pwzFileName
));
256 len
= lstrlenW(pwzFileName
)-4;
260 return wcsicmp(pwzFileName
+len
, L
".url") ? S_FALSE
: S_OK
;
263 /***********************************************************************
264 * HlinkGetSpecialReference (HLINK.@)
266 HRESULT WINAPI
HlinkGetSpecialReference(ULONG uReference
, LPWSTR
*ppwzReference
)
268 DWORD res
, type
, size
= 100;
273 TRACE("(%lu %p)\n", uReference
, ppwzReference
);
275 *ppwzReference
= NULL
;
279 value_name
= L
"Start Page";
281 case HLSR_SEARCHPAGE
:
282 value_name
= L
"Search Page";
284 case HLSR_HISTORYFOLDER
:
290 res
= RegOpenKeyW(HKEY_CURRENT_USER
, L
"Software\\Microsoft\\Internet Explorer\\Main", &hkey
);
291 if(res
!= ERROR_SUCCESS
) {
292 WARN("Could not open key: %lu\n", res
);
293 return HRESULT_FROM_WIN32(res
);
296 buf
= CoTaskMemAlloc(size
);
297 res
= RegQueryValueExW(hkey
, value_name
, NULL
, &type
, (PBYTE
)buf
, &size
);
298 buf
= CoTaskMemRealloc(buf
, size
);
299 if(res
== ERROR_MORE_DATA
)
300 res
= RegQueryValueExW(hkey
, value_name
, NULL
, &type
, (PBYTE
)buf
, &size
);
302 if(res
!= ERROR_SUCCESS
) {
303 WARN("Could not query value %s: %lu\n", debugstr_w(value_name
), res
);
305 return HRESULT_FROM_WIN32(res
);
308 *ppwzReference
= buf
;
312 /***********************************************************************
313 * HlinkTranslateURL (HLINK.@)
315 HRESULT WINAPI
HlinkTranslateURL(LPCWSTR pwzURL
, DWORD grfFlags
, LPWSTR
*ppwzTranslatedURL
)
317 FIXME("(%s %08lx %p)\n", debugstr_w(pwzURL
), grfFlags
, ppwzTranslatedURL
);
321 /***********************************************************************
322 * HlinkUpdateStackItem (HLINK.@)
324 HRESULT WINAPI
HlinkUpdateStackItem(IHlinkFrame
*frame
, IHlinkBrowseContext
*bc
,
325 ULONG hlid
, IMoniker
*target
, LPCWSTR location
, LPCWSTR friendly_name
)
329 TRACE("(%p %p 0x%lx %p %s %s)\n", frame
, bc
, hlid
, target
, debugstr_w(location
), debugstr_w(friendly_name
));
331 if (!frame
&& !bc
) return E_INVALIDARG
;
334 hr
= IHlinkFrame_UpdateHlink(frame
, hlid
, target
, location
, friendly_name
);
336 hr
= IHlinkBrowseContext_UpdateHlink(bc
, hlid
, target
, location
, friendly_name
);
341 /***********************************************************************
342 * HlinkParseDisplayName (HLINK.@)
344 HRESULT WINAPI
HlinkParseDisplayName(LPBC pibc
, LPCWSTR pwzDisplayName
, BOOL fNoForceAbs
,
345 ULONG
*pcchEaten
, IMoniker
**ppimk
)
347 ULONG eaten
= 0, len
;
350 TRACE("(%p %s %x %p %p)\n", pibc
, debugstr_w(pwzDisplayName
), fNoForceAbs
, pcchEaten
, ppimk
);
353 FIXME("Unsupported fNoForceAbs\n");
355 len
= ARRAY_SIZE(L
"file:") - 1;
356 if(!wcsnicmp(pwzDisplayName
, L
"file:", len
)) {
357 pwzDisplayName
+= len
;
360 while(*pwzDisplayName
== '/') {
365 hres
= MkParseDisplayNameEx(pibc
, pwzDisplayName
, pcchEaten
, ppimk
);
369 hres
= MkParseDisplayName(pibc
, pwzDisplayName
, pcchEaten
, ppimk
);
374 hres
= CreateFileMoniker(pwzDisplayName
, ppimk
);
376 *pcchEaten
= eaten
+ lstrlenW(pwzDisplayName
);
381 /***********************************************************************
382 * HlinkResolveMonikerForData (HLINK.@)
384 HRESULT WINAPI
HlinkResolveMonikerForData(LPMONIKER pimkReference
, DWORD reserved
, LPBC pibc
,
385 ULONG cFmtetc
, FORMATETC
*rgFmtetc
, IBindStatusCallback
*pibsc
, LPMONIKER pimkBase
)
387 LPOLESTR name
= NULL
;
393 TRACE("(%p %lx %p %ld %p %p %p)\n", pimkReference
, reserved
, pibc
, cFmtetc
, rgFmtetc
, pibsc
, pimkBase
);
395 if(cFmtetc
|| rgFmtetc
|| pimkBase
)
396 FIXME("Unsupported args\n");
398 hres
= RegisterBindStatusCallback(pibc
, pibsc
, NULL
/* FIXME */, 0);
402 hres
= IMoniker_IsSystemMoniker(pimkReference
, &mksys
);
403 if(SUCCEEDED(hres
) && mksys
!= MKSYS_URLMONIKER
)
404 WARN("sysmk = %lx\n", mksys
);
406 /* FIXME: What is it for? */
407 CreateBindCtx(0, &bctx
);
408 hres
= IMoniker_GetDisplayName(pimkReference
, bctx
, NULL
, &name
);
409 IBindCtx_Release(bctx
);
410 if(SUCCEEDED(hres
)) {
411 TRACE("got display name %s\n", debugstr_w(name
));
415 return IMoniker_BindToStorage(pimkReference
, pibc
, NULL
, &IID_IUnknown
, &obj
);
418 /***********************************************************************
419 * HlinkClone (HLINK.@)
421 HRESULT WINAPI
HlinkClone(IHlink
*hlink
, REFIID riid
, IHlinkSite
*hls
,
422 DWORD site_data
, void **obj
)
424 IMoniker
*mk
, *clone_mk
= NULL
;
425 WCHAR
*loc
, *name
= NULL
;
428 if(!hlink
|| !riid
|| !obj
)
433 hres
= IHlink_GetMonikerReference(hlink
, HLINKGETREF_DEFAULT
, &mk
, &loc
);
441 hres
= CreateStreamOnHGlobal(NULL
, TRUE
, &strm
);
443 IMoniker_Release(mk
);
447 hres
= OleSaveToStream((IPersistStream
*)mk
, strm
);
449 IStream_Release(strm
);
450 IMoniker_Release(mk
);
453 IMoniker_Release(mk
);
456 hres
= IStream_Seek(strm
, lgint
, STREAM_SEEK_SET
, NULL
);
458 IStream_Release(strm
);
462 hres
= OleLoadFromStream(strm
, &IID_IMoniker
, (void**)&clone_mk
);
463 IStream_Release(strm
);
468 hres
= IHlink_GetFriendlyName(hlink
, HLFNAMEF_DEFAULT
, &name
);
472 hres
= HlinkCreateFromMoniker(clone_mk
, loc
, name
, hls
, site_data
, NULL
,
477 IMoniker_Release(clone_mk
);
483 static HRESULT WINAPI
HLinkCF_fnQueryInterface ( LPCLASSFACTORY iface
,
484 REFIID riid
, LPVOID
*ppvObj
)
486 CFImpl
*This
= impl_from_IClassFactory(iface
);
488 TRACE("(%p)->(%s, %p)\n", This
, debugstr_guid(riid
), ppvObj
);
492 if (IsEqualIID(riid
, &IID_IUnknown
) ||
493 IsEqualIID(riid
, &IID_IClassFactory
))
495 *ppvObj
= &This
->IClassFactory_iface
;
499 TRACE("-- E_NOINTERFACE\n");
500 return E_NOINTERFACE
;
503 static ULONG WINAPI
HLinkCF_fnAddRef (LPCLASSFACTORY iface
)
508 static ULONG WINAPI
HLinkCF_fnRelease(LPCLASSFACTORY iface
)
513 static HRESULT WINAPI
HLinkCF_fnCreateInstance( LPCLASSFACTORY iface
,
514 LPUNKNOWN pUnkOuter
, REFIID riid
, LPVOID
*ppvObject
)
516 CFImpl
*This
= impl_from_IClassFactory(iface
);
518 TRACE("%p->(%p,%s,%p)\n", This
, pUnkOuter
, debugstr_guid(riid
), ppvObject
);
522 return This
->lpfnCI(pUnkOuter
, riid
, ppvObject
);
525 static HRESULT WINAPI
HLinkCF_fnLockServer(LPCLASSFACTORY iface
, BOOL fLock
)
527 FIXME("%p %d\n", iface
, fLock
);
531 static const IClassFactoryVtbl hlcfvt
=
533 HLinkCF_fnQueryInterface
,
536 HLinkCF_fnCreateInstance
,
540 static CFImpl HLink_cf
= { { &hlcfvt
}, HLink_Constructor
};
541 static CFImpl HLinkBrowseContext_cf
= { { &hlcfvt
}, HLinkBrowseContext_Constructor
};
543 /***********************************************************************
544 * DllGetClassObject (HLINK.@)
546 HRESULT WINAPI
DllGetClassObject(REFCLSID rclsid
, REFIID iid
, LPVOID
*ppv
)
548 IClassFactory
*pcf
= NULL
;
550 TRACE("%s %s %p\n", debugstr_guid(rclsid
), debugstr_guid(iid
), ppv
);
556 if (IsEqualIID(rclsid
, &CLSID_StdHlink
))
557 pcf
= &HLink_cf
.IClassFactory_iface
;
558 else if (IsEqualIID(rclsid
, &CLSID_StdHlinkBrowseContext
))
559 pcf
= &HLinkBrowseContext_cf
.IClassFactory_iface
;
561 return CLASS_E_CLASSNOTAVAILABLE
;
563 return IClassFactory_QueryInterface(pcf
, iid
, ppv
);