1 /* 2 * Implementation of hyperlinking (hlink.dll) 3 * 4 * Copyright 2005 Aric Stewart for CodeWeavers 5 * 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. 10 * 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. 15 * 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 19 */ 20 21 #include "hlink_private.h" 22 23 #include "winreg.h" 24 #include "rpcproxy.h" 25 #include "hlguids.h" 26 27 #include "wine/debug.h" 28 29 WINE_DEFAULT_DEBUG_CHANNEL(hlink); 30 31 static HINSTANCE instance; 32 33 typedef HRESULT (*LPFNCREATEINSTANCE)(IUnknown*, REFIID, LPVOID*); 34 35 typedef struct 36 { 37 IClassFactory IClassFactory_iface; 38 LPFNCREATEINSTANCE lpfnCI; 39 } CFImpl; 40 41 static inline CFImpl *impl_from_IClassFactory(IClassFactory *iface) 42 { 43 return CONTAINING_RECORD(iface, CFImpl, IClassFactory_iface); 44 } 45 46 BOOL WINAPI DllMain(HINSTANCE hinstDLL, DWORD fdwReason, LPVOID lpvReserved) 47 { 48 TRACE("%p %d %p\n", hinstDLL, fdwReason, lpvReserved); 49 50 switch (fdwReason) 51 { 52 case DLL_PROCESS_ATTACH: 53 instance = hinstDLL; 54 DisableThreadLibraryCalls(hinstDLL); 55 break; 56 } 57 return TRUE; 58 } 59 60 /*********************************************************************** 61 * DllCanUnloadNow (HLINK.@) 62 */ 63 HRESULT WINAPI DllCanUnloadNow( void ) 64 { 65 return S_FALSE; 66 } 67 68 /*********************************************************************** 69 * HlinkCreateFromMoniker (HLINK.@) 70 */ 71 HRESULT WINAPI HlinkCreateFromMoniker( IMoniker *pimkTrgt, LPCWSTR pwzLocation, 72 LPCWSTR pwzFriendlyName, IHlinkSite* pihlsite, DWORD dwSiteData, 73 IUnknown* piunkOuter, REFIID riid, void** ppvObj) 74 { 75 IHlink *hl = NULL; 76 HRESULT r; 77 78 TRACE("%p %s %s %p %i %p %s %p\n", pimkTrgt, debugstr_w(pwzLocation), 79 debugstr_w(pwzFriendlyName), pihlsite, dwSiteData, piunkOuter, 80 debugstr_guid(riid), ppvObj); 81 82 r = CoCreateInstance(&CLSID_StdHlink, piunkOuter, CLSCTX_INPROC_SERVER, riid, (LPVOID*)&hl); 83 if (FAILED(r)) 84 return r; 85 86 IHlink_SetMonikerReference(hl, HLINKSETF_LOCATION | HLINKSETF_TARGET, pimkTrgt, pwzLocation); 87 88 if (pwzFriendlyName) 89 IHlink_SetFriendlyName(hl, pwzFriendlyName); 90 if (pihlsite) 91 IHlink_SetHlinkSite(hl, pihlsite, dwSiteData); 92 93 *ppvObj = hl; 94 95 TRACE("Returning %i\n",r); 96 97 return r; 98 } 99 100 /*********************************************************************** 101 * HlinkCreateFromString (HLINK.@) 102 */ 103 HRESULT WINAPI HlinkCreateFromString( LPCWSTR pwzTarget, LPCWSTR pwzLocation, 104 LPCWSTR pwzFriendlyName, IHlinkSite* pihlsite, DWORD dwSiteData, 105 IUnknown* piunkOuter, REFIID riid, void** ppvObj) 106 { 107 IHlink *hl = NULL; 108 HRESULT r; 109 WCHAR *hash, *tgt; 110 const WCHAR *loc; 111 112 TRACE("%s %s %s %p %i %p %s %p\n", debugstr_w(pwzTarget), 113 debugstr_w(pwzLocation), debugstr_w(pwzFriendlyName), pihlsite, 114 dwSiteData, piunkOuter, debugstr_guid(riid), ppvObj); 115 116 r = CoCreateInstance(&CLSID_StdHlink, piunkOuter, CLSCTX_INPROC_SERVER, riid, (LPVOID*)&hl); 117 if (FAILED(r)) 118 return r; 119 120 if (pwzTarget) 121 { 122 hash = wcschr(pwzTarget, '#'); 123 if (hash) 124 { 125 if (hash == pwzTarget) 126 tgt = NULL; 127 else 128 { 129 int tgt_len = hash - pwzTarget; 130 tgt = heap_alloc((tgt_len + 1) * sizeof(WCHAR)); 131 if (!tgt) 132 return E_OUTOFMEMORY; 133 memcpy(tgt, pwzTarget, tgt_len * sizeof(WCHAR)); 134 tgt[tgt_len] = 0; 135 } 136 if (!pwzLocation) 137 loc = hash + 1; 138 else 139 loc = pwzLocation; 140 } 141 else 142 { 143 tgt = hlink_strdupW(pwzTarget); 144 if (!tgt) 145 return E_OUTOFMEMORY; 146 loc = pwzLocation; 147 } 148 } 149 else 150 { 151 tgt = NULL; 152 loc = pwzLocation; 153 } 154 155 IHlink_SetStringReference(hl, HLINKSETF_TARGET | HLINKSETF_LOCATION, tgt, loc); 156 157 heap_free(tgt); 158 159 if (pwzFriendlyName) 160 IHlink_SetFriendlyName(hl, pwzFriendlyName); 161 162 if (pihlsite) 163 IHlink_SetHlinkSite(hl, pihlsite, dwSiteData); 164 165 TRACE("Returning %i\n",r); 166 *ppvObj = hl; 167 168 return r; 169 } 170 171 172 /*********************************************************************** 173 * HlinkCreateBrowseContext (HLINK.@) 174 */ 175 HRESULT WINAPI HlinkCreateBrowseContext( IUnknown* piunkOuter, REFIID riid, void** ppvObj) 176 { 177 TRACE("%p %s %p\n", piunkOuter, debugstr_guid(riid), ppvObj); 178 return CoCreateInstance(&CLSID_StdHlinkBrowseContext, piunkOuter, CLSCTX_INPROC_SERVER, riid, ppvObj); 179 } 180 181 /*********************************************************************** 182 * HlinkNavigate (HLINK.@) 183 */ 184 HRESULT WINAPI HlinkNavigate(IHlink *phl, IHlinkFrame *phlFrame, 185 DWORD grfHLNF, LPBC pbc, IBindStatusCallback *pbsc, 186 IHlinkBrowseContext *phlbc) 187 { 188 HRESULT r = S_OK; 189 190 TRACE("%p %p %i %p %p %p\n", phl, phlFrame, grfHLNF, pbc, pbsc, phlbc); 191 192 if (phlFrame) 193 r = IHlinkFrame_Navigate(phlFrame, grfHLNF, pbc, pbsc, phl); 194 else if (phl) 195 r = IHlink_Navigate(phl, grfHLNF, pbc, pbsc, phlbc); 196 197 return r; 198 } 199 200 /*********************************************************************** 201 * HlinkOnNavigate (HLINK.@) 202 */ 203 HRESULT WINAPI HlinkOnNavigate( IHlinkFrame *phlFrame, 204 IHlinkBrowseContext* phlbc, DWORD grfHLNF, IMoniker *pmkTarget, 205 LPCWSTR pwzLocation, LPCWSTR pwzFriendlyName, ULONG* puHLID) 206 { 207 HRESULT r; 208 209 TRACE("%p %p %i %p %s %s %p\n",phlFrame, phlbc, grfHLNF, pmkTarget, 210 debugstr_w(pwzLocation), debugstr_w(pwzFriendlyName), puHLID); 211 212 r = IHlinkBrowseContext_OnNavigateHlink(phlbc, grfHLNF, pmkTarget, 213 pwzLocation, pwzFriendlyName, puHLID); 214 215 if (phlFrame) 216 r = IHlinkFrame_OnNavigate(phlFrame,grfHLNF,pmkTarget, pwzLocation, 217 pwzFriendlyName, 0); 218 219 return r; 220 } 221 222 /*********************************************************************** 223 * HlinkCreateFromData (HLINK.@) 224 */ 225 HRESULT WINAPI HlinkCreateFromData(IDataObject *piDataObj, 226 IHlinkSite *pihlsite, DWORD dwSiteData, IUnknown *piunkOuter, 227 REFIID riid, void **ppvObj) 228 { 229 FIXME("%p, %p, %d, %p, %s, %p\n", piDataObj, pihlsite, dwSiteData, 230 piunkOuter, debugstr_guid(riid), ppvObj); 231 *ppvObj = NULL; 232 return E_NOTIMPL; 233 } 234 235 /*********************************************************************** 236 * HlinkQueryCreateFromData (HLINK.@) 237 */ 238 HRESULT WINAPI HlinkQueryCreateFromData(IDataObject* piDataObj) 239 { 240 FIXME("%p\n", piDataObj); 241 return E_NOTIMPL; 242 } 243 244 /*********************************************************************** 245 * HlinkNavigateToStringReference (HLINK.@) 246 */ 247 HRESULT WINAPI HlinkNavigateToStringReference( LPCWSTR pwzTarget, 248 LPCWSTR pwzLocation, IHlinkSite *pihlsite, DWORD dwSiteData, 249 IHlinkFrame *pihlframe, DWORD grfHLNF, LPBC pibc, 250 IBindStatusCallback *pibsc, IHlinkBrowseContext *pihlbc) 251 { 252 HRESULT r; 253 IHlink *hlink = NULL; 254 255 TRACE("%s %s %p %08x %p %08x %p %p %p\n", 256 debugstr_w(pwzTarget), debugstr_w(pwzLocation), pihlsite, 257 dwSiteData, pihlframe, grfHLNF, pibc, pibsc, pihlbc); 258 259 r = HlinkCreateFromString( pwzTarget, pwzLocation, NULL, pihlsite, 260 dwSiteData, NULL, &IID_IHlink, (LPVOID*) &hlink ); 261 if (SUCCEEDED(r)) { 262 r = HlinkNavigate(hlink, pihlframe, grfHLNF, pibc, pibsc, pihlbc); 263 IHlink_Release(hlink); 264 } 265 266 return r; 267 } 268 269 /*********************************************************************** 270 * HlinkIsShortcut (HLINK.@) 271 */ 272 HRESULT WINAPI HlinkIsShortcut(LPCWSTR pwzFileName) 273 { 274 int len; 275 276 static const WCHAR url_ext[] = {'.','u','r','l',0}; 277 278 TRACE("(%s)\n", debugstr_w(pwzFileName)); 279 280 if(!pwzFileName) 281 return E_INVALIDARG; 282 283 len = lstrlenW(pwzFileName)-4; 284 if(len < 0) 285 return S_FALSE; 286 287 return wcsicmp(pwzFileName+len, url_ext) ? S_FALSE : S_OK; 288 } 289 290 /*********************************************************************** 291 * HlinkGetSpecialReference (HLINK.@) 292 */ 293 HRESULT WINAPI HlinkGetSpecialReference(ULONG uReference, LPWSTR *ppwzReference) 294 { 295 DWORD res, type, size = 100; 296 LPCWSTR value_name; 297 WCHAR *buf; 298 HKEY hkey; 299 300 static const WCHAR start_pageW[] = {'S','t','a','r','t',' ','P','a','g','e',0}; 301 static const WCHAR search_pageW[] = {'S','e','a','r','c','h',' ','P','a','g','e',0}; 302 303 static const WCHAR ie_main_keyW[] = 304 {'S','o','f','t','w','a','r','e', 305 '\\','M','i','c','r','o','s','o','f','t','\\', 306 'I','n','t','e','r','n','e','t',' ','E','x','p','l','o','r','e','r', 307 '\\','M','a','i','n',0}; 308 309 TRACE("(%u %p)\n", uReference, ppwzReference); 310 311 *ppwzReference = NULL; 312 313 switch(uReference) { 314 case HLSR_HOME: 315 value_name = start_pageW; 316 break; 317 case HLSR_SEARCHPAGE: 318 value_name = search_pageW; 319 break; 320 case HLSR_HISTORYFOLDER: 321 return E_NOTIMPL; 322 default: 323 return E_INVALIDARG; 324 } 325 326 res = RegOpenKeyW(HKEY_CURRENT_USER, ie_main_keyW, &hkey); 327 if(res != ERROR_SUCCESS) { 328 WARN("Could not open key: %u\n", res); 329 return HRESULT_FROM_WIN32(res); 330 } 331 332 buf = CoTaskMemAlloc(size); 333 res = RegQueryValueExW(hkey, value_name, NULL, &type, (PBYTE)buf, &size); 334 buf = CoTaskMemRealloc(buf, size); 335 if(res == ERROR_MORE_DATA) 336 res = RegQueryValueExW(hkey, value_name, NULL, &type, (PBYTE)buf, &size); 337 RegCloseKey(hkey); 338 if(res != ERROR_SUCCESS) { 339 WARN("Could not query value %s: %u\n", debugstr_w(value_name), res); 340 CoTaskMemFree(buf); 341 return HRESULT_FROM_WIN32(res); 342 } 343 344 *ppwzReference = buf; 345 return S_OK; 346 } 347 348 /*********************************************************************** 349 * HlinkTranslateURL (HLINK.@) 350 */ 351 HRESULT WINAPI HlinkTranslateURL(LPCWSTR pwzURL, DWORD grfFlags, LPWSTR *ppwzTranslatedURL) 352 { 353 FIXME("(%s %08x %p)\n", debugstr_w(pwzURL), grfFlags, ppwzTranslatedURL); 354 return E_NOTIMPL; 355 } 356 357 /*********************************************************************** 358 * HlinkUpdateStackItem (HLINK.@) 359 */ 360 HRESULT WINAPI HlinkUpdateStackItem(IHlinkFrame *frame, IHlinkBrowseContext *bc, 361 ULONG hlid, IMoniker *target, LPCWSTR location, LPCWSTR friendly_name) 362 { 363 HRESULT hr; 364 365 TRACE("(%p %p 0x%x %p %s %s)\n", frame, bc, hlid, target, debugstr_w(location), debugstr_w(friendly_name)); 366 367 if (!frame && !bc) return E_INVALIDARG; 368 369 if (frame) 370 hr = IHlinkFrame_UpdateHlink(frame, hlid, target, location, friendly_name); 371 else 372 hr = IHlinkBrowseContext_UpdateHlink(bc, hlid, target, location, friendly_name); 373 374 return hr; 375 } 376 377 /*********************************************************************** 378 * HlinkParseDisplayName (HLINK.@) 379 */ 380 HRESULT WINAPI HlinkParseDisplayName(LPBC pibc, LPCWSTR pwzDisplayName, BOOL fNoForceAbs, 381 ULONG *pcchEaten, IMoniker **ppimk) 382 { 383 static const WCHAR file_colonW[] = {'f','i','l','e',':'}; 384 ULONG eaten = 0; 385 HRESULT hres; 386 387 TRACE("(%p %s %x %p %p)\n", pibc, debugstr_w(pwzDisplayName), fNoForceAbs, pcchEaten, ppimk); 388 389 if(fNoForceAbs) 390 FIXME("Unsupported fNoForceAbs\n"); 391 392 if(!_wcsnicmp(pwzDisplayName, file_colonW, ARRAY_SIZE(file_colonW))) { 393 pwzDisplayName += ARRAY_SIZE(file_colonW); 394 eaten += ARRAY_SIZE(file_colonW); 395 396 while(*pwzDisplayName == '/') { 397 pwzDisplayName++; 398 eaten++; 399 } 400 }else { 401 hres = MkParseDisplayNameEx(pibc, pwzDisplayName, pcchEaten, ppimk); 402 if(SUCCEEDED(hres)) 403 return hres; 404 405 hres = MkParseDisplayName(pibc, pwzDisplayName, pcchEaten, ppimk); 406 if(SUCCEEDED(hres)) 407 return hres; 408 } 409 410 hres = CreateFileMoniker(pwzDisplayName, ppimk); 411 if(SUCCEEDED(hres)) 412 *pcchEaten = eaten + lstrlenW(pwzDisplayName); 413 414 return hres; 415 } 416 417 /*********************************************************************** 418 * HlinkResolveMonikerForData (HLINK.@) 419 */ 420 HRESULT WINAPI HlinkResolveMonikerForData(LPMONIKER pimkReference, DWORD reserved, LPBC pibc, 421 ULONG cFmtetc, FORMATETC *rgFmtetc, IBindStatusCallback *pibsc, LPMONIKER pimkBase) 422 { 423 LPOLESTR name = NULL; 424 IBindCtx *bctx; 425 DWORD mksys = 0; 426 void *obj = NULL; 427 HRESULT hres; 428 429 TRACE("(%p %x %p %d %p %p %p)\n", pimkReference, reserved, pibc, cFmtetc, rgFmtetc, pibsc, pimkBase); 430 431 if(cFmtetc || rgFmtetc || pimkBase) 432 FIXME("Unsupported args\n"); 433 434 hres = RegisterBindStatusCallback(pibc, pibsc, NULL /* FIXME */, 0); 435 if(FAILED(hres)) 436 return hres; 437 438 hres = IMoniker_IsSystemMoniker(pimkReference, &mksys); 439 if(SUCCEEDED(hres) && mksys != MKSYS_URLMONIKER) 440 WARN("sysmk = %x\n", mksys); 441 442 /* FIXME: What is it for? */ 443 CreateBindCtx(0, &bctx); 444 hres = IMoniker_GetDisplayName(pimkReference, bctx, NULL, &name); 445 IBindCtx_Release(bctx); 446 if(SUCCEEDED(hres)) { 447 TRACE("got display name %s\n", debugstr_w(name)); 448 CoTaskMemFree(name); 449 } 450 451 return IMoniker_BindToStorage(pimkReference, pibc, NULL, &IID_IUnknown, &obj); 452 } 453 454 /*********************************************************************** 455 * HlinkClone (HLINK.@) 456 */ 457 HRESULT WINAPI HlinkClone(IHlink *hlink, REFIID riid, IHlinkSite *hls, 458 DWORD site_data, void **obj) 459 { 460 IMoniker *mk, *clone_mk = NULL; 461 WCHAR *loc, *name = NULL; 462 HRESULT hres; 463 464 if(!hlink || !riid || !obj) 465 return E_INVALIDARG; 466 467 *obj = NULL; 468 469 hres = IHlink_GetMonikerReference(hlink, HLINKGETREF_DEFAULT, &mk, &loc); 470 if(FAILED(hres)) 471 return hres; 472 473 if(mk) { 474 IStream *strm; 475 LARGE_INTEGER lgint; 476 477 hres = CreateStreamOnHGlobal(NULL, TRUE, &strm); 478 if(FAILED(hres)) { 479 IMoniker_Release(mk); 480 goto cleanup; 481 } 482 483 hres = OleSaveToStream((IPersistStream*)mk, strm); 484 if(FAILED(hres)) { 485 IStream_Release(strm); 486 IMoniker_Release(mk); 487 goto cleanup; 488 } 489 IMoniker_Release(mk); 490 491 lgint.QuadPart = 0; 492 hres = IStream_Seek(strm, lgint, STREAM_SEEK_SET, NULL); 493 if(FAILED(hres)) { 494 IStream_Release(strm); 495 goto cleanup; 496 } 497 498 hres = OleLoadFromStream(strm, &IID_IMoniker, (void**)&clone_mk); 499 IStream_Release(strm); 500 if(FAILED(hres)) 501 goto cleanup; 502 } 503 504 hres = IHlink_GetFriendlyName(hlink, HLFNAMEF_DEFAULT, &name); 505 if(FAILED(hres)) 506 goto cleanup; 507 508 hres = HlinkCreateFromMoniker(clone_mk, loc, name, hls, site_data, NULL, 509 &IID_IHlink, obj); 510 511 cleanup: 512 if(clone_mk) 513 IMoniker_Release(clone_mk); 514 CoTaskMemFree(loc); 515 CoTaskMemFree(name); 516 return hres; 517 } 518 519 static HRESULT WINAPI HLinkCF_fnQueryInterface ( LPCLASSFACTORY iface, 520 REFIID riid, LPVOID *ppvObj) 521 { 522 CFImpl *This = impl_from_IClassFactory(iface); 523 524 TRACE("(%p)->(%s, %p)\n", This, debugstr_guid(riid), ppvObj); 525 526 *ppvObj = NULL; 527 528 if (IsEqualIID(riid, &IID_IUnknown) || 529 IsEqualIID(riid, &IID_IClassFactory)) 530 { 531 *ppvObj = &This->IClassFactory_iface; 532 return S_OK; 533 } 534 535 TRACE("-- E_NOINTERFACE\n"); 536 return E_NOINTERFACE; 537 } 538 539 static ULONG WINAPI HLinkCF_fnAddRef (LPCLASSFACTORY iface) 540 { 541 return 2; 542 } 543 544 static ULONG WINAPI HLinkCF_fnRelease(LPCLASSFACTORY iface) 545 { 546 return 1; 547 } 548 549 static HRESULT WINAPI HLinkCF_fnCreateInstance( LPCLASSFACTORY iface, 550 LPUNKNOWN pUnkOuter, REFIID riid, LPVOID *ppvObject) 551 { 552 CFImpl *This = impl_from_IClassFactory(iface); 553 554 TRACE("%p->(%p,%s,%p)\n", This, pUnkOuter, debugstr_guid(riid), ppvObject); 555 556 *ppvObject = NULL; 557 558 return This->lpfnCI(pUnkOuter, riid, ppvObject); 559 } 560 561 static HRESULT WINAPI HLinkCF_fnLockServer(LPCLASSFACTORY iface, BOOL fLock) 562 { 563 FIXME("%p %d\n", iface, fLock); 564 return E_NOTIMPL; 565 } 566 567 static const IClassFactoryVtbl hlcfvt = 568 { 569 HLinkCF_fnQueryInterface, 570 HLinkCF_fnAddRef, 571 HLinkCF_fnRelease, 572 HLinkCF_fnCreateInstance, 573 HLinkCF_fnLockServer 574 }; 575 576 static CFImpl HLink_cf = { { &hlcfvt }, HLink_Constructor }; 577 static CFImpl HLinkBrowseContext_cf = { { &hlcfvt }, HLinkBrowseContext_Constructor }; 578 579 /*********************************************************************** 580 * DllGetClassObject (HLINK.@) 581 */ 582 HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID iid, LPVOID *ppv) 583 { 584 IClassFactory *pcf = NULL; 585 586 TRACE("%s %s %p\n", debugstr_guid(rclsid), debugstr_guid(iid), ppv); 587 588 if (!ppv) 589 return E_INVALIDARG; 590 *ppv = NULL; 591 592 if (IsEqualIID(rclsid, &CLSID_StdHlink)) 593 pcf = &HLink_cf.IClassFactory_iface; 594 else if (IsEqualIID(rclsid, &CLSID_StdHlinkBrowseContext)) 595 pcf = &HLinkBrowseContext_cf.IClassFactory_iface; 596 else 597 return CLASS_E_CLASSNOTAVAILABLE; 598 599 return IClassFactory_QueryInterface(pcf, iid, ppv); 600 } 601 602 /*********************************************************************** 603 * DllRegisterServer (HLINK.@) 604 */ 605 HRESULT WINAPI DllRegisterServer(void) 606 { 607 return __wine_register_resources( instance ); 608 } 609 610 /*********************************************************************** 611 * DllUnregisterServer (HLINK.@) 612 */ 613 HRESULT WINAPI DllUnregisterServer(void) 614 { 615 return __wine_unregister_resources( instance ); 616 } 617