xref: /reactos/dll/win32/shell32/CShellLink.cpp (revision 63bb46a2)
1 /*
2  *
3  *      Copyright 1997  Marcus Meissner
4  *      Copyright 1998  Juergen Schmied
5  *      Copyright 2005  Mike McCormack
6  *      Copyright 2009  Andrew Hill
7  *      Copyright 2013  Dominik Hornung
8  *      Copyright 2017  Hermes Belusca-Maito
9  *      Copyright 2018-2024 Katayama Hirofumi MZ <katayama.hirofumi.mz@gmail.com>
10  *
11  * This library is free software; you can redistribute it and/or
12  * modify it under the terms of the GNU Lesser General Public
13  * License as published by the Free Software Foundation; either
14  * version 2.1 of the License, or (at your option) any later version.
15  *
16  * This library is distributed in the hope that it will be useful,
17  * but WITHOUT ANY WARRANTY; without even the implied warranty of
18  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
19  * Lesser General Public License for more details.
20  *
21  * You should have received a copy of the GNU Lesser General Public
22  * License along with this library; if not, write to the Free Software
23  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
24  *
25  * NOTES
26  *   Nearly complete information about the binary formats
27  *   of .lnk files available at http://www.wotsit.org
28  *
29  *  You can use winedump to examine the contents of a link file:
30  *   winedump lnk sc.lnk
31  *
32  *  MSI advertised shortcuts are totally undocumented.  They provide an
33  *   icon for a program that is not yet installed, and invoke MSI to
34  *   install the program when the shortcut is clicked on.  They are
35  *   created by passing a special string to SetPath, and the information
36  *   in that string is parsed an stored.
37  */
38 /*
39  * In the following is listed more documentation about the Shell Link file format,
40  * as well as its interface.
41  *
42  * General introduction about "Shell Links" (MSDN):
43  *   https://msdn.microsoft.com/en-us/library/windows/desktop/bb776891(v=vs.85).aspx
44  *
45  *
46  * Details of the file format:
47  *
48  * - Official MSDN documentation "[MS-SHLLINK]: Shell Link (.LNK) Binary File Format":
49  *   https://msdn.microsoft.com/en-us/library/dd871305.aspx
50  *
51  * - Forensics:
52  *   http://forensicswiki.org/wiki/LNK
53  *   http://computerforensics.parsonage.co.uk/downloads/TheMeaningofLIFE.pdf
54  *   https://ithreats.files.wordpress.com/2009/05/lnk_the_windows_shortcut_file_format.pdf
55  *   https://github.com/libyal/liblnk/blob/master/documentation/Windows%20Shortcut%20File%20(LNK)%20format.asciidoc
56  *
57  * - List of possible shell link header flags (SHELL_LINK_DATA_FLAGS enumeration):
58  *   https://msdn.microsoft.com/en-us/library/windows/desktop/bb762540(v=vs.85).aspx
59  *   https://msdn.microsoft.com/en-us/library/dd891314.aspx
60  *
61  *
62  * In addition to storing its target by using a PIDL, a shell link file also
63  * stores metadata to make the shell able to track the link target, in situations
64  * where the link target is moved amongst local or network directories, or moved
65  * to different volumes. For this, two structures are used:
66  *
67  * - The first and oldest one (from NewShell/WinNT4) is the "LinkInfo" structure,
68  *   stored in a serialized manner at the beginning of the shell link file:
69  *   https://msdn.microsoft.com/en-us/library/dd871404.aspx
70  *   The official API for manipulating this is located in LINKINFO.DLL .
71  *
72  * - The second, more recent one, is an extra binary block appended to the
73  *   extra-data list of the shell link file: this is the "TrackerDataBlock":
74  *   https://msdn.microsoft.com/en-us/library/dd891376.aspx
75  *   Its purpose is for link tracking, and works in coordination with the
76  *   "Distributed Link Tracking" service ('TrkWks' client, 'TrkSvr' server).
77  *   See a detailed explanation at:
78  *   http://www.serverwatch.com/tutorials/article.php/1476701/Searching-for-the-Missing-Link-Distributed-Link-Tracking.htm
79  *
80  *
81  * MSI installations most of the time create so-called "advertised shortcuts".
82  * They provide an icon for a program that may not be installed yet, and invoke
83  * MSI to install the program when the shortcut is opened (resolved).
84  * The philosophy of this approach is explained in detail inside the MSDN article
85  * "Application Resiliency: Unlock the Hidden Features of Windows Installer"
86  * (by Michael Sanford), here:
87  *   https://msdn.microsoft.com/en-us/library/aa302344.aspx
88  *
89  * This functionality is implemented by adding a binary "Darwin" data block
90  * of type "EXP_DARWIN_LINK", signature EXP_DARWIN_ID_SIG == 0xA0000006,
91  * to the shell link file:
92  *   https://msdn.microsoft.com/en-us/library/dd871369.aspx
93  * or, this could be done more simply by specifying a special link target path
94  * with the IShellLink::SetPath() function. Defining the following GUID:
95  *   SHELL32_AdvtShortcutComponent = "::{9db1186e-40df-11d1-aa8c-00c04fb67863}:"
96  * setting a target of the form:
97  *   "::{SHELL32_AdvtShortcutComponent}:<MSI_App_ID>"
98  * would automatically create the necessary binary block.
99  *
100  * With that, the target of the shortcut now becomes the MSI data. The latter
101  * is parsed from MSI and retrieved by the shell that then can run the program.
102  *
103  * This MSI functionality, dubbed "link blessing", actually originates from an
104  * older technology introduced in Internet Explorer 3 (and now obsolete since
105  * Internet Explorer 7), called "MS Internet Component Download (MSICD)", see
106  * this MSDN introductory article:
107  *   https://msdn.microsoft.com/en-us/library/aa741198(v=vs.85).aspx
108  * and leveraged in Internet Explorer 4 with "Software Update Channels", see:
109  *   https://msdn.microsoft.com/en-us/library/aa740931(v=vs.85).aspx
110  * Applications supporting this technology could present shell links having
111  * a special target, see subsection "Modifying the Shortcut" in the article:
112  *   https://msdn.microsoft.com/en-us/library/aa741201(v=vs.85).aspx#pub_shor
113  *
114  * Similarly as for the MSI shortcuts, these MSICD shortcuts are created by
115  * specifying a special link target path with the IShellLink::SetPath() function,
116  * defining the following GUID:
117  *   SHELL32_AdvtShortcutProduct = "::{9db1186f-40df-11d1-aa8c-00c04fb67863}:"
118  * and setting a target of the form:
119  *   "::{SHELL32_AdvtShortcutProduct}:<AppName>::<Path>" .
120  * A tool, called "blesslnk.exe", was also provided for automatizing the process;
121  * its ReadMe can be found in the (now outdated) MS "Internet Client SDK" (INetSDK,
122  * for MS Windows 95 and NT), whose contents can be read at:
123  *   http://www.msfn.org/board/topic/145352-new-windows-lnk-vulnerability/?page=4#comment-944223
124  * The MS INetSDK can be found at:
125  *   https://web.archive.org/web/20100924000013/http://support.microsoft.com/kb/177877
126  *
127  * Internally the shell link target of these MSICD shortcuts is converted into
128  * a binary data block of a type similar to Darwin / "EXP_DARWIN_LINK", but with
129  * a different signature EXP_LOGO3_ID_SIG == 0xA0000007 . Such shell links are
130  * called "Logo3" shortcuts. They were evoked in this user comment in "The Old
131  * New Thing" blog:
132  *   https://blogs.msdn.microsoft.com/oldnewthing/20121210-00/?p=5883#comment-1025083
133  *
134  * The shell exports the API 'SoftwareUpdateMessageBox' (in shdocvw.dll) that
135  * displays a message when an update for an application supporting this
136  * technology is available.
137  *
138  */
139 
140 #include "precomp.h"
141 
142 #include <appmgmt.h>
143 
144 WINE_DEFAULT_DEBUG_CHANNEL(shell);
145 
146 /*
147  * Allows to define whether or not Windows-compatible behaviour
148  * should be adopted when setting and retrieving icon location paths.
149  * See CShellLink::SetIconLocation(LPCWSTR pszIconPath, INT iIcon)
150  * for more details.
151  */
152 #define ICON_LINK_WINDOWS_COMPAT
153 
154 #define SHLINK_LOCAL  0
155 #define SHLINK_REMOTE 1
156 
157 /* link file formats */
158 
159 #include "pshpack1.h"
160 
161 struct LOCATION_INFO
162 {
163     DWORD        dwTotalSize;
164     DWORD        dwHeaderSize;
165     DWORD        dwFlags;
166     DWORD        dwVolTableOfs;
167     DWORD        dwLocalPathOfs;
168     DWORD        dwNetworkVolTableOfs;
169     DWORD        dwFinalPathOfs;
170 };
171 
172 struct LOCAL_VOLUME_INFO
173 {
174     DWORD        dwSize;
175     DWORD        dwType;
176     DWORD        dwVolSerial;
177     DWORD        dwVolLabelOfs;
178 };
179 
180 struct volume_info
181 {
182     DWORD        type;
183     DWORD        serial;
184     WCHAR        label[12];  /* assume 8.3 */
185 };
186 
187 #include "poppack.h"
188 
189 /* IShellLink Implementation */
190 
191 static HRESULT ShellLink_UpdatePath(LPCWSTR sPathRel, LPCWSTR path, LPCWSTR sWorkDir, LPWSTR* psPath);
192 
193 /* strdup on the process heap */
194 static LPWSTR __inline HEAP_strdupAtoW(HANDLE heap, DWORD flags, LPCSTR str)
195 {
196     INT len;
197     LPWSTR p;
198 
199     assert(str);
200 
201     len = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0);
202     p = (LPWSTR)HeapAlloc(heap, flags, len * sizeof(WCHAR));
203     if (!p)
204         return p;
205     MultiByteToWideChar(CP_ACP, 0, str, -1, p, len);
206     return p;
207 }
208 
209 static LPWSTR __inline strdupW(LPCWSTR src)
210 {
211     LPWSTR dest;
212     if (!src) return NULL;
213     dest = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wcslen(src) + 1) * sizeof(WCHAR));
214     if (dest)
215         wcscpy(dest, src);
216     return dest;
217 }
218 
219 // TODO: Use it for constructor & destructor too
220 VOID CShellLink::Reset()
221 {
222     ILFree(m_pPidl);
223     m_pPidl = NULL;
224 
225     HeapFree(GetProcessHeap(), 0, m_sPath);
226     m_sPath = NULL;
227     ZeroMemory(&volume, sizeof(volume));
228 
229     HeapFree(GetProcessHeap(), 0, m_sDescription);
230     m_sDescription = NULL;
231     HeapFree(GetProcessHeap(), 0, m_sPathRel);
232     m_sPathRel = NULL;
233     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
234     m_sWorkDir = NULL;
235     HeapFree(GetProcessHeap(), 0, m_sArgs);
236     m_sArgs = NULL;
237     HeapFree(GetProcessHeap(), 0, m_sIcoPath);
238     m_sIcoPath = NULL;
239 
240     m_bRunAs = FALSE;
241     m_bDirty = FALSE;
242 
243     if (m_pDBList)
244         SHFreeDataBlockList(m_pDBList);
245     m_pDBList = NULL;
246 
247     /**/sProduct = sComponent = NULL;/**/
248 }
249 
250 CShellLink::CShellLink()
251 {
252     m_Header.dwSize = sizeof(m_Header);
253     m_Header.clsid = CLSID_ShellLink;
254     m_Header.dwFlags = 0;
255 
256     m_Header.dwFileAttributes = 0;
257     ZeroMemory(&m_Header.ftCreationTime, sizeof(m_Header.ftCreationTime));
258     ZeroMemory(&m_Header.ftLastAccessTime, sizeof(m_Header.ftLastAccessTime));
259     ZeroMemory(&m_Header.ftLastWriteTime, sizeof(m_Header.ftLastWriteTime));
260     m_Header.nFileSizeLow = 0;
261 
262     m_Header.nIconIndex = 0;
263     m_Header.nShowCommand = SW_SHOWNORMAL;
264     m_Header.wHotKey = 0;
265 
266     m_pPidl = NULL;
267 
268     m_sPath = NULL;
269     ZeroMemory(&volume, sizeof(volume));
270 
271     m_sDescription = NULL;
272     m_sPathRel = NULL;
273     m_sWorkDir = NULL;
274     m_sArgs = NULL;
275     m_sIcoPath = NULL;
276     m_bRunAs = FALSE;
277     m_bDirty = FALSE;
278     m_pDBList = NULL;
279     m_bInInit = FALSE;
280     m_hIcon = NULL;
281     m_idCmdFirst = 0;
282 
283     m_sLinkPath = NULL;
284 
285     /**/sProduct = sComponent = NULL;/**/
286 }
287 
288 CShellLink::~CShellLink()
289 {
290     TRACE("-- destroying IShellLink(%p)\n", this);
291 
292     ILFree(m_pPidl);
293 
294     HeapFree(GetProcessHeap(), 0, m_sPath);
295 
296     HeapFree(GetProcessHeap(), 0, m_sDescription);
297     HeapFree(GetProcessHeap(), 0, m_sPathRel);
298     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
299     HeapFree(GetProcessHeap(), 0, m_sArgs);
300     HeapFree(GetProcessHeap(), 0, m_sIcoPath);
301     HeapFree(GetProcessHeap(), 0, m_sLinkPath);
302     SHFreeDataBlockList(m_pDBList);
303 }
304 
305 HRESULT STDMETHODCALLTYPE CShellLink::GetClassID(CLSID *pclsid)
306 {
307     TRACE("%p %p\n", this, pclsid);
308 
309     if (pclsid == NULL)
310         return E_POINTER;
311     *pclsid = CLSID_ShellLink;
312     return S_OK;
313 }
314 
315 /************************************************************************
316  * IPersistStream_IsDirty (IPersistStream)
317  */
318 HRESULT STDMETHODCALLTYPE CShellLink::IsDirty()
319 {
320     TRACE("(%p)\n", this);
321     return (m_bDirty ? S_OK : S_FALSE);
322 }
323 
324 HRESULT STDMETHODCALLTYPE CShellLink::Load(LPCOLESTR pszFileName, DWORD dwMode)
325 {
326     TRACE("(%p, %s, %x)\n", this, debugstr_w(pszFileName), dwMode);
327 
328     if (dwMode == 0)
329         dwMode = STGM_READ | STGM_SHARE_DENY_WRITE;
330 
331     CComPtr<IStream> stm;
332     HRESULT hr = SHCreateStreamOnFileW(pszFileName, dwMode, &stm);
333     if (SUCCEEDED(hr))
334     {
335         HeapFree(GetProcessHeap(), 0, m_sLinkPath);
336         m_sLinkPath = strdupW(pszFileName);
337         hr = Load(stm);
338         ShellLink_UpdatePath(m_sPathRel, pszFileName, m_sWorkDir, &m_sPath);
339         m_bDirty = FALSE;
340     }
341     TRACE("-- returning hr %08x\n", hr);
342     return hr;
343 }
344 
345 HRESULT STDMETHODCALLTYPE CShellLink::Save(LPCOLESTR pszFileName, BOOL fRemember)
346 {
347     BOOL bAlreadyExists;
348     WCHAR szFullPath[MAX_PATH];
349 
350     TRACE("(%p)->(%s)\n", this, debugstr_w(pszFileName));
351 
352     if (!pszFileName)
353         return E_FAIL;
354 
355     bAlreadyExists = PathFileExistsW(pszFileName);
356 
357     CComPtr<IStream> stm;
358     HRESULT hr = SHCreateStreamOnFileW(pszFileName, STGM_READWRITE | STGM_CREATE | STGM_SHARE_EXCLUSIVE, &stm);
359     if (SUCCEEDED(hr))
360     {
361         hr = Save(stm, FALSE);
362 
363         if (SUCCEEDED(hr))
364         {
365             GetFullPathNameW(pszFileName, _countof(szFullPath), szFullPath, NULL);
366             if (bAlreadyExists)
367                 SHChangeNotify(SHCNE_UPDATEITEM, SHCNF_PATHW, szFullPath, NULL);
368             else
369                 SHChangeNotify(SHCNE_CREATE, SHCNF_PATHW, szFullPath, NULL);
370 
371             if (m_sLinkPath)
372                 HeapFree(GetProcessHeap(), 0, m_sLinkPath);
373 
374             m_sLinkPath = strdupW(pszFileName);
375             m_bDirty = FALSE;
376         }
377         else
378         {
379             DeleteFileW(pszFileName);
380             WARN("Failed to create shortcut %s\n", debugstr_w(pszFileName));
381         }
382     }
383 
384     return hr;
385 }
386 
387 HRESULT STDMETHODCALLTYPE CShellLink::SaveCompleted(LPCOLESTR pszFileName)
388 {
389     FIXME("(%p)->(%s)\n", this, debugstr_w(pszFileName));
390     return S_OK;
391 }
392 
393 HRESULT STDMETHODCALLTYPE CShellLink::GetCurFile(LPOLESTR *ppszFileName)
394 {
395     *ppszFileName = NULL;
396 
397     if (!m_sLinkPath)
398     {
399         /* IPersistFile::GetCurFile called before IPersistFile::Save */
400         return S_FALSE;
401     }
402 
403     *ppszFileName = (LPOLESTR)CoTaskMemAlloc((wcslen(m_sLinkPath) + 1) * sizeof(WCHAR));
404     if (!*ppszFileName)
405     {
406         /* out of memory */
407         return E_OUTOFMEMORY;
408     }
409 
410     /* copy last saved filename */
411     wcscpy(*ppszFileName, m_sLinkPath);
412 
413     return S_OK;
414 }
415 
416 static HRESULT Stream_LoadString(IStream* stm, BOOL unicode, LPWSTR *pstr)
417 {
418     TRACE("%p\n", stm);
419 
420     USHORT len;
421     DWORD count = 0;
422     HRESULT hr = stm->Read(&len, sizeof(len), &count);
423     if (FAILED(hr) || count != sizeof(len))
424         return E_FAIL;
425 
426     if (unicode)
427         len *= sizeof(WCHAR);
428 
429     TRACE("reading %d\n", len);
430     LPSTR temp = (LPSTR)HeapAlloc(GetProcessHeap(), 0, len + sizeof(WCHAR));
431     if (!temp)
432         return E_OUTOFMEMORY;
433     count = 0;
434     hr = stm->Read(temp, len, &count);
435     if (FAILED(hr) || count != len)
436     {
437         HeapFree(GetProcessHeap(), 0, temp);
438         return E_FAIL;
439     }
440 
441     TRACE("read %s\n", debugstr_an(temp, len));
442 
443     /* convert to unicode if necessary */
444     LPWSTR str;
445     if (!unicode)
446     {
447         count = MultiByteToWideChar(CP_ACP, 0, temp, len, NULL, 0);
448         str = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (count + 1) * sizeof(WCHAR));
449         if (!str)
450         {
451             HeapFree(GetProcessHeap(), 0, temp);
452             return E_OUTOFMEMORY;
453         }
454         MultiByteToWideChar(CP_ACP, 0, temp, len, str, count);
455         HeapFree(GetProcessHeap(), 0, temp);
456     }
457     else
458     {
459         count /= sizeof(WCHAR);
460         str = (LPWSTR)temp;
461     }
462     str[count] = 0;
463 
464     *pstr = str;
465 
466     return S_OK;
467 }
468 
469 
470 /*
471  * NOTE: The following 5 functions are part of LINKINFO.DLL
472  */
473 static BOOL ShellLink_GetVolumeInfo(LPCWSTR path, CShellLink::volume_info *volume)
474 {
475     WCHAR drive[4] = { path[0], ':', '\\', 0 };
476 
477     volume->type = GetDriveTypeW(drive);
478     BOOL bRet = GetVolumeInformationW(drive, volume->label, _countof(volume->label), &volume->serial, NULL, NULL, NULL, 0);
479     TRACE("ret = %d type %d serial %08x name %s\n", bRet,
480           volume->type, volume->serial, debugstr_w(volume->label));
481     return bRet;
482 }
483 
484 static HRESULT Stream_ReadChunk(IStream* stm, LPVOID *data)
485 {
486     struct sized_chunk
487     {
488         DWORD size;
489         unsigned char data[1];
490     } *chunk;
491 
492     TRACE("%p\n", stm);
493 
494     DWORD size;
495     ULONG count;
496     HRESULT hr = stm->Read(&size, sizeof(size), &count);
497     if (FAILED(hr) || count != sizeof(size))
498         return E_FAIL;
499 
500     chunk = static_cast<sized_chunk *>(HeapAlloc(GetProcessHeap(), 0, size));
501     if (!chunk)
502         return E_OUTOFMEMORY;
503 
504     chunk->size = size;
505     hr = stm->Read(chunk->data, size - sizeof(size), &count);
506     if (FAILED(hr) || count != (size - sizeof(size)))
507     {
508         HeapFree(GetProcessHeap(), 0, chunk);
509         return E_FAIL;
510     }
511 
512     TRACE("Read %d bytes\n", chunk->size);
513 
514     *data = chunk;
515 
516     return S_OK;
517 }
518 
519 static BOOL Stream_LoadVolume(LOCAL_VOLUME_INFO *vol, CShellLink::volume_info *volume)
520 {
521     volume->serial = vol->dwVolSerial;
522     volume->type = vol->dwType;
523 
524     if (!vol->dwVolLabelOfs)
525         return FALSE;
526     if (vol->dwSize <= vol->dwVolLabelOfs)
527         return FALSE;
528     INT len = vol->dwSize - vol->dwVolLabelOfs;
529 
530     LPSTR label = (LPSTR)vol;
531     label += vol->dwVolLabelOfs;
532     MultiByteToWideChar(CP_ACP, 0, label, len, volume->label, _countof(volume->label));
533 
534     return TRUE;
535 }
536 
537 static LPWSTR Stream_LoadPath(LPCSTR p, DWORD maxlen)
538 {
539     UINT len = 0;
540 
541     while (len < maxlen && p[len])
542         len++;
543 
544     UINT wlen = MultiByteToWideChar(CP_ACP, 0, p, len, NULL, 0);
545     LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, (wlen + 1) * sizeof(WCHAR));
546     if (!path)
547         return NULL;
548     MultiByteToWideChar(CP_ACP, 0, p, len, path, wlen);
549     path[wlen] = 0;
550 
551     return path;
552 }
553 
554 static HRESULT Stream_LoadLocation(IStream *stm,
555                                    CShellLink::volume_info *volume, LPWSTR *path)
556 {
557     char *p = NULL;
558     HRESULT hr = Stream_ReadChunk(stm, (LPVOID*) &p);
559     if (FAILED(hr))
560         return hr;
561 
562     LOCATION_INFO *loc = reinterpret_cast<LOCATION_INFO *>(p);
563     if (loc->dwTotalSize < sizeof(LOCATION_INFO))
564     {
565         HeapFree(GetProcessHeap(), 0, p);
566         return E_FAIL;
567     }
568 
569     /* if there's valid local volume information, load it */
570     if (loc->dwVolTableOfs &&
571         ((loc->dwVolTableOfs + sizeof(LOCAL_VOLUME_INFO)) <= loc->dwTotalSize))
572     {
573         LOCAL_VOLUME_INFO *volume_info;
574 
575         volume_info = (LOCAL_VOLUME_INFO*) &p[loc->dwVolTableOfs];
576         Stream_LoadVolume(volume_info, volume);
577     }
578 
579     /* if there's a local path, load it */
580     DWORD n = loc->dwLocalPathOfs;
581     if (n && n < loc->dwTotalSize)
582         *path = Stream_LoadPath(&p[n], loc->dwTotalSize - n);
583 
584     TRACE("type %d serial %08x name %s path %s\n", volume->type,
585           volume->serial, debugstr_w(volume->label), debugstr_w(*path));
586 
587     HeapFree(GetProcessHeap(), 0, p);
588     return S_OK;
589 }
590 
591 
592 /*
593  * The format of the advertised shortcut info is:
594  *
595  *  Offset     Description
596  *  ------     -----------
597  *    0          Length of the block (4 bytes, usually 0x314)
598  *    4          tag (dword)
599  *    8          string data in ASCII
600  *    8+0x104    string data in UNICODE
601  *
602  * In the original Win32 implementation the buffers are not initialized
603  * to zero, so data trailing the string is random garbage.
604  */
605 HRESULT CShellLink::GetAdvertiseInfo(LPWSTR *str, DWORD dwSig)
606 {
607     LPEXP_DARWIN_LINK pInfo;
608 
609     *str = NULL;
610 
611     pInfo = (LPEXP_DARWIN_LINK)SHFindDataBlock(m_pDBList, dwSig);
612     if (!pInfo)
613         return E_FAIL;
614 
615     /* Make sure that the size of the structure is valid */
616     if (pInfo->dbh.cbSize != sizeof(*pInfo))
617     {
618         ERR("Ooops. This structure is not as expected...\n");
619         return E_FAIL;
620     }
621 
622     TRACE("dwSig %08x  string = '%s'\n", pInfo->dbh.dwSignature, debugstr_w(pInfo->szwDarwinID));
623 
624     *str = pInfo->szwDarwinID;
625     return S_OK;
626 }
627 
628 /************************************************************************
629  * IPersistStream_Load (IPersistStream)
630  */
631 HRESULT STDMETHODCALLTYPE CShellLink::Load(IStream *stm)
632 {
633     TRACE("%p %p\n", this, stm);
634 
635     if (!stm)
636         return STG_E_INVALIDPOINTER;
637 
638     /* Free all the old stuff */
639     Reset();
640 
641     ULONG dwBytesRead = 0;
642     HRESULT hr = stm->Read(&m_Header, sizeof(m_Header), &dwBytesRead);
643     if (FAILED(hr))
644         return hr;
645 
646     if (dwBytesRead != sizeof(m_Header))
647         return E_FAIL;
648     if (m_Header.dwSize != sizeof(m_Header))
649         return E_FAIL;
650     if (!IsEqualIID(m_Header.clsid, CLSID_ShellLink))
651         return E_FAIL;
652 
653     /* Load the new data in order */
654 
655     if (TRACE_ON(shell))
656     {
657         SYSTEMTIME stCreationTime;
658         SYSTEMTIME stLastAccessTime;
659         SYSTEMTIME stLastWriteTime;
660         WCHAR sTemp[MAX_PATH];
661 
662         FileTimeToSystemTime(&m_Header.ftCreationTime, &stCreationTime);
663         FileTimeToSystemTime(&m_Header.ftLastAccessTime, &stLastAccessTime);
664         FileTimeToSystemTime(&m_Header.ftLastWriteTime, &stLastWriteTime);
665 
666         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stCreationTime,
667                        NULL, sTemp, _countof(sTemp));
668         TRACE("-- stCreationTime: %s\n", debugstr_w(sTemp));
669         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stLastAccessTime,
670                        NULL, sTemp, _countof(sTemp));
671         TRACE("-- stLastAccessTime: %s\n", debugstr_w(sTemp));
672         GetDateFormatW(LOCALE_USER_DEFAULT, DATE_SHORTDATE, &stLastWriteTime,
673                        NULL, sTemp, _countof(sTemp));
674         TRACE("-- stLastWriteTime: %s\n", debugstr_w(sTemp));
675     }
676 
677     /* load all the new stuff */
678     if (m_Header.dwFlags & SLDF_HAS_ID_LIST)
679     {
680         hr = ILLoadFromStream(stm, &m_pPidl);
681         if (FAILED(hr))
682             return hr;
683     }
684     pdump(m_pPidl);
685 
686     /* Load the location information... */
687     if (m_Header.dwFlags & SLDF_HAS_LINK_INFO)
688     {
689         hr = Stream_LoadLocation(stm, &volume, &m_sPath);
690         if (FAILED(hr))
691             return hr;
692     }
693     /* ... but if it is required not to use it, clear it */
694     if (m_Header.dwFlags & SLDF_FORCE_NO_LINKINFO)
695     {
696         HeapFree(GetProcessHeap(), 0, m_sPath);
697         m_sPath = NULL;
698         ZeroMemory(&volume, sizeof(volume));
699     }
700 
701     BOOL unicode = !!(m_Header.dwFlags & SLDF_UNICODE);
702 
703     if (m_Header.dwFlags & SLDF_HAS_NAME)
704     {
705         hr = Stream_LoadString(stm, unicode, &m_sDescription);
706         if (FAILED(hr))
707             return hr;
708         TRACE("Description  -> %s\n", debugstr_w(m_sDescription));
709     }
710 
711     if (m_Header.dwFlags & SLDF_HAS_RELPATH)
712     {
713         hr = Stream_LoadString(stm, unicode, &m_sPathRel);
714         if (FAILED(hr))
715             return hr;
716         TRACE("Relative Path-> %s\n", debugstr_w(m_sPathRel));
717     }
718 
719     if (m_Header.dwFlags & SLDF_HAS_WORKINGDIR)
720     {
721         hr = Stream_LoadString(stm, unicode, &m_sWorkDir);
722         if (FAILED(hr))
723             return hr;
724         PathRemoveBackslash(m_sWorkDir);
725         TRACE("Working Dir  -> %s\n", debugstr_w(m_sWorkDir));
726     }
727 
728     if (m_Header.dwFlags & SLDF_HAS_ARGS)
729     {
730         hr = Stream_LoadString(stm, unicode, &m_sArgs);
731         if (FAILED(hr))
732             return hr;
733         TRACE("Arguments    -> %s\n", debugstr_w(m_sArgs));
734     }
735 
736     if (m_Header.dwFlags & SLDF_HAS_ICONLOCATION)
737     {
738         hr = Stream_LoadString(stm, unicode, &m_sIcoPath);
739         if (FAILED(hr))
740             return hr;
741         TRACE("Icon file    -> %s\n", debugstr_w(m_sIcoPath));
742     }
743 
744     /* Now load the optional data block list */
745     hr = SHReadDataBlockList(stm, &m_pDBList);
746     if (FAILED(hr)) // FIXME: Should we fail?
747         return hr;
748 
749     LPEXP_SPECIAL_FOLDER pSpecial = (LPEXP_SPECIAL_FOLDER)SHFindDataBlock(m_pDBList, EXP_SPECIAL_FOLDER_SIG);
750     if (pSpecial && pSpecial->cbSize == sizeof(*pSpecial) && ILGetSize(m_pPidl) > pSpecial->cbOffset)
751     {
752         if (LPITEMIDLIST folder = SHCloneSpecialIDList(NULL, pSpecial->idSpecialFolder, FALSE))
753         {
754             LPITEMIDLIST pidl = ILCombine(folder, (LPITEMIDLIST)((char*)m_pPidl + pSpecial->cbOffset));
755             if (pidl)
756             {
757                 ILFree(m_pPidl);
758                 m_pPidl = pidl;
759                 TRACE("Replaced pidl base with CSIDL %u up to %ub.\n", pSpecial->idSpecialFolder, pSpecial->cbOffset);
760             }
761             ILFree(folder);
762         }
763     }
764 
765     if (TRACE_ON(shell))
766     {
767 #if (NTDDI_VERSION < NTDDI_LONGHORN)
768         if (m_Header.dwFlags & SLDF_HAS_LOGO3ID)
769         {
770             hr = GetAdvertiseInfo(&sProduct, EXP_LOGO3_ID_SIG);
771             if (SUCCEEDED(hr))
772                 TRACE("Product      -> %s\n", debugstr_w(sProduct));
773         }
774 #endif
775         if (m_Header.dwFlags & SLDF_HAS_DARWINID)
776         {
777             hr = GetAdvertiseInfo(&sComponent, EXP_DARWIN_ID_SIG);
778             if (SUCCEEDED(hr))
779                 TRACE("Component    -> %s\n", debugstr_w(sComponent));
780         }
781     }
782 
783     if (m_Header.dwFlags & SLDF_RUNAS_USER)
784         m_bRunAs = TRUE;
785     else
786         m_bRunAs = FALSE;
787 
788     TRACE("OK\n");
789 
790     pdump(m_pPidl);
791 
792     return S_OK;
793 }
794 
795 /************************************************************************
796  * Stream_WriteString
797  *
798  * Helper function for IPersistStream_Save. Writes a unicode string
799  *  with terminating nul byte to a stream, preceded by the its length.
800  */
801 static HRESULT Stream_WriteString(IStream* stm, LPCWSTR str)
802 {
803     SIZE_T length;
804     USHORT len;
805     DWORD count;
806 
807     length = wcslen(str) + 1;
808     if (length > MAXUSHORT)
809     {
810         return E_INVALIDARG;
811     }
812 
813     len = (USHORT)length;
814     HRESULT hr = stm->Write(&len, sizeof(len), &count);
815     if (FAILED(hr))
816         return hr;
817 
818     length *= sizeof(WCHAR);
819 
820     hr = stm->Write(str, (ULONG)length, &count);
821     if (FAILED(hr))
822         return hr;
823 
824     return S_OK;
825 }
826 
827 /************************************************************************
828  * Stream_WriteLocationInfo
829  *
830  * Writes the location info to a stream
831  *
832  * FIXME: One day we might want to write the network volume information
833  *        and the final path.
834  *        Figure out how Windows deals with unicode paths here.
835  */
836 static HRESULT Stream_WriteLocationInfo(IStream* stm, LPCWSTR path,
837         CShellLink::volume_info *volume)
838 {
839     LOCAL_VOLUME_INFO *vol;
840     LOCATION_INFO *loc;
841 
842     TRACE("%p %s %p\n", stm, debugstr_w(path), volume);
843 
844     /* figure out the size of everything */
845     DWORD label_size = WideCharToMultiByte(CP_ACP, 0, volume->label, -1,
846                                       NULL, 0, NULL, NULL);
847     DWORD path_size = WideCharToMultiByte(CP_ACP, 0, path, -1,
848                                      NULL, 0, NULL, NULL);
849     DWORD volume_info_size = sizeof(*vol) + label_size;
850     DWORD final_path_size = 1;
851     DWORD total_size = sizeof(*loc) + volume_info_size + path_size + final_path_size;
852 
853     /* create pointers to everything */
854     loc = static_cast<LOCATION_INFO *>(HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, total_size));
855     vol = (LOCAL_VOLUME_INFO*) &loc[1];
856     LPSTR szLabel = (LPSTR) &vol[1];
857     LPSTR szPath = &szLabel[label_size];
858     LPSTR szFinalPath = &szPath[path_size];
859 
860     /* fill in the location information header */
861     loc->dwTotalSize = total_size;
862     loc->dwHeaderSize = sizeof(*loc);
863     loc->dwFlags = 1;
864     loc->dwVolTableOfs = sizeof(*loc);
865     loc->dwLocalPathOfs = sizeof(*loc) + volume_info_size;
866     loc->dwNetworkVolTableOfs = 0;
867     loc->dwFinalPathOfs = sizeof(*loc) + volume_info_size + path_size;
868 
869     /* fill in the volume information */
870     vol->dwSize = volume_info_size;
871     vol->dwType = volume->type;
872     vol->dwVolSerial = volume->serial;
873     vol->dwVolLabelOfs = sizeof(*vol);
874 
875     /* copy in the strings */
876     WideCharToMultiByte(CP_ACP, 0, volume->label, -1,
877                          szLabel, label_size, NULL, NULL);
878     WideCharToMultiByte(CP_ACP, 0, path, -1,
879                          szPath, path_size, NULL, NULL);
880     *szFinalPath = 0;
881 
882     ULONG count = 0;
883     HRESULT hr = stm->Write(loc, total_size, &count);
884     HeapFree(GetProcessHeap(), 0, loc);
885 
886     return hr;
887 }
888 
889 /************************************************************************
890  * IPersistStream_Save (IPersistStream)
891  *
892  * FIXME: makes assumptions about byte order
893  */
894 HRESULT STDMETHODCALLTYPE CShellLink::Save(IStream *stm, BOOL fClearDirty)
895 {
896     TRACE("%p %p %x\n", this, stm, fClearDirty);
897 
898     m_Header.dwSize = sizeof(m_Header);
899     m_Header.clsid = CLSID_ShellLink;
900 
901     /* Store target attributes */
902     WIN32_FIND_DATAW wfd = {};
903     WCHAR FsTarget[MAX_PATH];
904     if (GetPath(FsTarget, _countof(FsTarget), NULL, 0) == S_OK && PathFileExistsW(FsTarget))
905     {
906         HANDLE hFind = FindFirstFileW(FsTarget, &wfd);
907         if (hFind != INVALID_HANDLE_VALUE)
908             FindClose(hFind);
909     }
910     m_Header.dwFileAttributes = wfd.dwFileAttributes;
911     m_Header.ftCreationTime = wfd.ftCreationTime;
912     m_Header.ftLastAccessTime = wfd.ftLastAccessTime;
913     m_Header.ftLastWriteTime = wfd.ftLastWriteTime;
914     m_Header.nFileSizeLow = wfd.nFileSizeLow;
915 
916     /*
917      * Reset the flags: keep only the flags related to data blocks as they were
918      * already set in accordance by the different mutator member functions.
919      * The other flags will be determined now by the presence or absence of data.
920      */
921     m_Header.dwFlags &= (SLDF_RUN_WITH_SHIMLAYER | SLDF_RUNAS_USER |
922                          SLDF_RUN_IN_SEPARATE | SLDF_HAS_DARWINID |
923 #if (NTDDI_VERSION < NTDDI_LONGHORN)
924                          SLDF_HAS_LOGO3ID |
925 #endif
926                          SLDF_HAS_EXP_ICON_SZ | SLDF_HAS_EXP_SZ);
927     // TODO: When we will support Vista+ functionality, add other flags to this list.
928 
929     /* The stored strings are in UNICODE */
930     m_Header.dwFlags |= SLDF_UNICODE;
931 
932     if (m_pPidl)
933         m_Header.dwFlags |= SLDF_HAS_ID_LIST;
934     if (m_sPath && *m_sPath && !(m_Header.dwFlags & SLDF_FORCE_NO_LINKINFO))
935         m_Header.dwFlags |= SLDF_HAS_LINK_INFO;
936     if (m_sDescription && *m_sDescription)
937         m_Header.dwFlags |= SLDF_HAS_NAME;
938     if (m_sPathRel && *m_sPathRel)
939         m_Header.dwFlags |= SLDF_HAS_RELPATH;
940     if (m_sWorkDir && *m_sWorkDir)
941         m_Header.dwFlags |= SLDF_HAS_WORKINGDIR;
942     if (m_sArgs && *m_sArgs)
943         m_Header.dwFlags |= SLDF_HAS_ARGS;
944     if (m_sIcoPath && *m_sIcoPath)
945         m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
946     if (m_bRunAs)
947         m_Header.dwFlags |= SLDF_RUNAS_USER;
948 
949     /* Write the shortcut header */
950     ULONG count;
951     HRESULT hr = stm->Write(&m_Header, sizeof(m_Header), &count);
952     if (FAILED(hr))
953     {
954         ERR("Write failed\n");
955         return hr;
956     }
957 
958     /* Save the data in order */
959 
960     if (m_pPidl)
961     {
962         hr = ILSaveToStream(stm, m_pPidl);
963         if (FAILED(hr))
964         {
965             ERR("Failed to write PIDL\n");
966             return hr;
967         }
968     }
969 
970     if (m_Header.dwFlags & SLDF_HAS_LINK_INFO)
971     {
972         hr = Stream_WriteLocationInfo(stm, m_sPath, &volume);
973         if (FAILED(hr))
974             return hr;
975     }
976 
977     if (m_Header.dwFlags & SLDF_HAS_NAME)
978     {
979         hr = Stream_WriteString(stm, m_sDescription);
980         if (FAILED(hr))
981             return hr;
982     }
983 
984     if (m_Header.dwFlags & SLDF_HAS_RELPATH)
985     {
986         hr = Stream_WriteString(stm, m_sPathRel);
987         if (FAILED(hr))
988             return hr;
989     }
990 
991     if (m_Header.dwFlags & SLDF_HAS_WORKINGDIR)
992     {
993         hr = Stream_WriteString(stm, m_sWorkDir);
994         if (FAILED(hr))
995             return hr;
996     }
997 
998     if (m_Header.dwFlags & SLDF_HAS_ARGS)
999     {
1000         hr = Stream_WriteString(stm, m_sArgs);
1001         if (FAILED(hr))
1002             return hr;
1003     }
1004 
1005     if (m_Header.dwFlags & SLDF_HAS_ICONLOCATION)
1006     {
1007         hr = Stream_WriteString(stm, m_sIcoPath);
1008         if (FAILED(hr))
1009             return hr;
1010     }
1011 
1012     /*
1013      * Now save the data block list.
1014      *
1015      * NOTE that both advertised Product and Component are already saved
1016      * inside Logo3 and Darwin data blocks in the m_pDBList list, and the
1017      * m_Header.dwFlags is suitably initialized.
1018      */
1019     hr = SHWriteDataBlockList(stm, m_pDBList);
1020     if (FAILED(hr))
1021         return hr;
1022 
1023     /* Clear the dirty bit if requested */
1024     if (fClearDirty)
1025         m_bDirty = FALSE;
1026 
1027     return hr;
1028 }
1029 
1030 /************************************************************************
1031  * IPersistStream_GetSizeMax (IPersistStream)
1032  */
1033 HRESULT STDMETHODCALLTYPE CShellLink::GetSizeMax(ULARGE_INTEGER *pcbSize)
1034 {
1035     TRACE("(%p)\n", this);
1036     return E_NOTIMPL;
1037 }
1038 
1039 static BOOL SHELL_ExistsFileW(LPCWSTR path)
1040 {
1041     if (INVALID_FILE_ATTRIBUTES == GetFileAttributesW(path))
1042         return FALSE;
1043 
1044     return TRUE;
1045 }
1046 
1047 /**************************************************************************
1048  *  ShellLink_UpdatePath
1049  *    update absolute path in sPath using relative path in sPathRel
1050  */
1051 static HRESULT ShellLink_UpdatePath(LPCWSTR sPathRel, LPCWSTR path, LPCWSTR sWorkDir, LPWSTR* psPath)
1052 {
1053     if (!path || !psPath)
1054         return E_INVALIDARG;
1055 
1056     if (!*psPath && sPathRel)
1057     {
1058         WCHAR buffer[2*MAX_PATH], abs_path[2*MAX_PATH];
1059         LPWSTR final = NULL;
1060 
1061         /* first try if [directory of link file] + [relative path] finds an existing file */
1062 
1063         GetFullPathNameW(path, MAX_PATH * 2, buffer, &final);
1064         if (!final)
1065             final = buffer;
1066         wcscpy(final, sPathRel);
1067 
1068         *abs_path = '\0';
1069 
1070         if (SHELL_ExistsFileW(buffer))
1071         {
1072             if (!GetFullPathNameW(buffer, MAX_PATH, abs_path, &final))
1073                 wcscpy(abs_path, buffer);
1074         }
1075         else
1076         {
1077             /* try if [working directory] + [relative path] finds an existing file */
1078             if (sWorkDir)
1079             {
1080                 wcscpy(buffer, sWorkDir);
1081                 wcscpy(PathAddBackslashW(buffer), sPathRel);
1082 
1083                 if (SHELL_ExistsFileW(buffer))
1084                     if (!GetFullPathNameW(buffer, MAX_PATH, abs_path, &final))
1085                         wcscpy(abs_path, buffer);
1086             }
1087         }
1088 
1089         /* FIXME: This is even not enough - not all shell links can be resolved using this algorithm. */
1090         if (!*abs_path)
1091             wcscpy(abs_path, sPathRel);
1092 
1093         *psPath = strdupW(abs_path);
1094         if (!*psPath)
1095             return E_OUTOFMEMORY;
1096     }
1097 
1098     return S_OK;
1099 }
1100 
1101 HRESULT STDMETHODCALLTYPE CShellLink::GetPath(LPSTR pszFile, INT cchMaxPath, WIN32_FIND_DATAA *pfd, DWORD fFlags)
1102 {
1103     HRESULT hr;
1104     LPWSTR pszFileW;
1105     WIN32_FIND_DATAW wfd;
1106 
1107     TRACE("(%p)->(pfile=%p len=%u find_data=%p flags=%u)(%s)\n",
1108           this, pszFile, cchMaxPath, pfd, fFlags, debugstr_w(m_sPath));
1109 
1110     /* Allocate a temporary UNICODE buffer */
1111     pszFileW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, max(cchMaxPath, MAX_PATH) * sizeof(WCHAR));
1112     if (!pszFileW)
1113         return E_OUTOFMEMORY;
1114 
1115     /* Call the UNICODE function */
1116     hr = GetPath(pszFileW, cchMaxPath, &wfd, fFlags);
1117 
1118     /* Convert the file path back to ANSI */
1119     WideCharToMultiByte(CP_ACP, 0, pszFileW, -1,
1120                         pszFile, cchMaxPath, NULL, NULL);
1121 
1122     /* Free the temporary buffer */
1123     HeapFree(GetProcessHeap(), 0, pszFileW);
1124 
1125     if (pfd)
1126     {
1127         ZeroMemory(pfd, sizeof(*pfd));
1128 
1129         /* Copy the file data if a file path was returned */
1130         if (*pszFile)
1131         {
1132             DWORD len;
1133 
1134             /* Copy the fixed part */
1135             CopyMemory(pfd, &wfd, FIELD_OFFSET(WIN32_FIND_DATAA, cFileName));
1136 
1137             /* Convert the file names to ANSI */
1138             len = lstrlenW(wfd.cFileName);
1139             WideCharToMultiByte(CP_ACP, 0, wfd.cFileName, len + 1,
1140                                 pfd->cFileName, sizeof(pfd->cFileName), NULL, NULL);
1141             len = lstrlenW(wfd.cAlternateFileName);
1142             WideCharToMultiByte(CP_ACP, 0, wfd.cAlternateFileName, len + 1,
1143                                 pfd->cAlternateFileName, sizeof(pfd->cAlternateFileName), NULL, NULL);
1144         }
1145     }
1146 
1147     return hr;
1148 }
1149 
1150 HRESULT STDMETHODCALLTYPE CShellLink::GetIDList(PIDLIST_ABSOLUTE *ppidl)
1151 {
1152     TRACE("(%p)->(ppidl=%p)\n", this, ppidl);
1153 
1154     if (!m_pPidl)
1155     {
1156         *ppidl = NULL;
1157         return S_FALSE;
1158     }
1159 
1160     *ppidl = ILClone(m_pPidl);
1161     return S_OK;
1162 }
1163 
1164 HRESULT STDMETHODCALLTYPE CShellLink::SetIDList(PCIDLIST_ABSOLUTE pidl)
1165 {
1166     TRACE("(%p)->(pidl=%p)\n", this, pidl);
1167     return SetTargetFromPIDLOrPath(pidl, NULL);
1168 }
1169 
1170 HRESULT STDMETHODCALLTYPE CShellLink::GetDescription(LPSTR pszName, INT cchMaxName)
1171 {
1172     TRACE("(%p)->(%p len=%u)\n", this, pszName, cchMaxName);
1173 
1174     if (cchMaxName)
1175         *pszName = 0;
1176 
1177     if (m_sDescription)
1178         WideCharToMultiByte(CP_ACP, 0, m_sDescription, -1,
1179                              pszName, cchMaxName, NULL, NULL);
1180 
1181     return S_OK;
1182 }
1183 
1184 HRESULT STDMETHODCALLTYPE CShellLink::SetDescription(LPCSTR pszName)
1185 {
1186     TRACE("(%p)->(pName=%s)\n", this, pszName);
1187 
1188     HeapFree(GetProcessHeap(), 0, m_sDescription);
1189     m_sDescription = NULL;
1190 
1191     if (pszName)
1192     {
1193         m_sDescription = HEAP_strdupAtoW(GetProcessHeap(), 0, pszName);
1194         if (!m_sDescription)
1195             return E_OUTOFMEMORY;
1196     }
1197     m_bDirty = TRUE;
1198 
1199     return S_OK;
1200 }
1201 
1202 HRESULT STDMETHODCALLTYPE CShellLink::GetWorkingDirectory(LPSTR pszDir, INT cchMaxPath)
1203 {
1204     TRACE("(%p)->(%p len=%u)\n", this, pszDir, cchMaxPath);
1205 
1206     if (cchMaxPath)
1207         *pszDir = 0;
1208 
1209     if (m_sWorkDir)
1210         WideCharToMultiByte(CP_ACP, 0, m_sWorkDir, -1,
1211                              pszDir, cchMaxPath, NULL, NULL);
1212 
1213     return S_OK;
1214 }
1215 
1216 HRESULT STDMETHODCALLTYPE CShellLink::SetWorkingDirectory(LPCSTR pszDir)
1217 {
1218     TRACE("(%p)->(dir=%s)\n", this, pszDir);
1219 
1220     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
1221     m_sWorkDir = NULL;
1222 
1223     if (pszDir)
1224     {
1225         m_sWorkDir = HEAP_strdupAtoW(GetProcessHeap(), 0, pszDir);
1226         if (!m_sWorkDir)
1227             return E_OUTOFMEMORY;
1228     }
1229     m_bDirty = TRUE;
1230 
1231     return S_OK;
1232 }
1233 
1234 HRESULT STDMETHODCALLTYPE CShellLink::GetArguments(LPSTR pszArgs, INT cchMaxPath)
1235 {
1236     TRACE("(%p)->(%p len=%u)\n", this, pszArgs, cchMaxPath);
1237 
1238     if (cchMaxPath)
1239         *pszArgs = 0;
1240 
1241     if (m_sArgs)
1242         WideCharToMultiByte(CP_ACP, 0, m_sArgs, -1,
1243                              pszArgs, cchMaxPath, NULL, NULL);
1244 
1245     return S_OK;
1246 }
1247 
1248 HRESULT STDMETHODCALLTYPE CShellLink::SetArguments(LPCSTR pszArgs)
1249 {
1250     TRACE("(%p)->(args=%s)\n", this, pszArgs);
1251 
1252     HeapFree(GetProcessHeap(), 0, m_sArgs);
1253     m_sArgs = NULL;
1254 
1255     if (pszArgs)
1256     {
1257         m_sArgs = HEAP_strdupAtoW(GetProcessHeap(), 0, pszArgs);
1258         if (!m_sArgs)
1259             return E_OUTOFMEMORY;
1260     }
1261     m_bDirty = TRUE;
1262 
1263     return S_OK;
1264 }
1265 
1266 HRESULT STDMETHODCALLTYPE CShellLink::GetHotkey(WORD *pwHotkey)
1267 {
1268     TRACE("(%p)->(%p)(0x%08x)\n", this, pwHotkey, m_Header.wHotKey);
1269     *pwHotkey = m_Header.wHotKey;
1270     return S_OK;
1271 }
1272 
1273 HRESULT STDMETHODCALLTYPE CShellLink::SetHotkey(WORD wHotkey)
1274 {
1275     TRACE("(%p)->(hotkey=%x)\n", this, wHotkey);
1276 
1277     m_Header.wHotKey = wHotkey;
1278     m_bDirty = TRUE;
1279 
1280     return S_OK;
1281 }
1282 
1283 HRESULT STDMETHODCALLTYPE CShellLink::GetShowCmd(INT *piShowCmd)
1284 {
1285     TRACE("(%p)->(%p) %d\n", this, piShowCmd, m_Header.nShowCommand);
1286     *piShowCmd = m_Header.nShowCommand;
1287     return S_OK;
1288 }
1289 
1290 HRESULT STDMETHODCALLTYPE CShellLink::SetShowCmd(INT iShowCmd)
1291 {
1292     TRACE("(%p) %d\n", this, iShowCmd);
1293 
1294     m_Header.nShowCommand = iShowCmd;
1295     m_bDirty = TRUE;
1296 
1297     return S_OK;
1298 }
1299 
1300 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(LPSTR pszIconPath, INT cchIconPath, INT *piIcon)
1301 {
1302     HRESULT hr;
1303     LPWSTR pszIconPathW;
1304 
1305     TRACE("(%p)->(%p len=%u iicon=%p)\n", this, pszIconPath, cchIconPath, piIcon);
1306 
1307     /* Allocate a temporary UNICODE buffer */
1308     pszIconPathW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, cchIconPath * sizeof(WCHAR));
1309     if (!pszIconPathW)
1310         return E_OUTOFMEMORY;
1311 
1312     /* Call the UNICODE function */
1313     hr = GetIconLocation(pszIconPathW, cchIconPath, piIcon);
1314 
1315     /* Convert the file path back to ANSI */
1316     WideCharToMultiByte(CP_ACP, 0, pszIconPathW, -1,
1317                         pszIconPath, cchIconPath, NULL, NULL);
1318 
1319     /* Free the temporary buffer */
1320     HeapFree(GetProcessHeap(), 0, pszIconPathW);
1321 
1322     return hr;
1323 }
1324 
1325 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(UINT uFlags, PSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1326 {
1327     HRESULT hr;
1328     LPWSTR pszIconFileW;
1329 
1330     TRACE("(%p)->(%u %p len=%u piIndex=%p pwFlags=%p)\n", this, uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1331 
1332     /* Allocate a temporary UNICODE buffer */
1333     pszIconFileW = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, cchMax * sizeof(WCHAR));
1334     if (!pszIconFileW)
1335         return E_OUTOFMEMORY;
1336 
1337     /* Call the UNICODE function */
1338     hr = GetIconLocation(uFlags, pszIconFileW, cchMax, piIndex, pwFlags);
1339 
1340     /* Convert the file path back to ANSI */
1341     WideCharToMultiByte(CP_ACP, 0, pszIconFileW, -1,
1342                         pszIconFile, cchMax, NULL, NULL);
1343 
1344     /* Free the temporary buffer */
1345     HeapFree(GetProcessHeap(), 0, pszIconFileW);
1346 
1347     return hr;
1348 }
1349 
1350 HRESULT STDMETHODCALLTYPE CShellLink::Extract(PCSTR pszFile, UINT nIconIndex, HICON *phiconLarge, HICON *phiconSmall, UINT nIconSize)
1351 {
1352     TRACE("(%p)->(path=%s iicon=%u)\n", this, pszFile, nIconIndex);
1353 
1354     LPWSTR str = NULL;
1355     if (pszFile)
1356     {
1357         str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszFile);
1358         if (!str)
1359             return E_OUTOFMEMORY;
1360     }
1361 
1362     HRESULT hr = Extract(str, nIconIndex, phiconLarge, phiconSmall, nIconSize);
1363 
1364     if (str)
1365         HeapFree(GetProcessHeap(), 0, str);
1366 
1367     return hr;
1368 }
1369 
1370 HRESULT STDMETHODCALLTYPE CShellLink::SetIconLocation(LPCSTR pszIconPath, INT iIcon)
1371 {
1372     TRACE("(%p)->(path=%s iicon=%u)\n", this, pszIconPath, iIcon);
1373 
1374     LPWSTR str = NULL;
1375     if (pszIconPath)
1376     {
1377         str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszIconPath);
1378         if (!str)
1379             return E_OUTOFMEMORY;
1380     }
1381 
1382     HRESULT hr = SetIconLocation(str, iIcon);
1383 
1384     if (str)
1385         HeapFree(GetProcessHeap(), 0, str);
1386 
1387     return hr;
1388 }
1389 
1390 HRESULT STDMETHODCALLTYPE CShellLink::SetRelativePath(LPCSTR pszPathRel, DWORD dwReserved)
1391 {
1392     TRACE("(%p)->(path=%s %x)\n", this, pszPathRel, dwReserved);
1393 
1394     HeapFree(GetProcessHeap(), 0, m_sPathRel);
1395     m_sPathRel = NULL;
1396 
1397     if (pszPathRel)
1398     {
1399         m_sPathRel = HEAP_strdupAtoW(GetProcessHeap(), 0, pszPathRel);
1400         m_bDirty = TRUE;
1401     }
1402 
1403     return ShellLink_UpdatePath(m_sPathRel, m_sPath, m_sWorkDir, &m_sPath);
1404 }
1405 
1406 static LPWSTR
1407 shelllink_get_msi_component_path(LPWSTR component)
1408 {
1409     DWORD Result, sz = 0;
1410 
1411     Result = CommandLineFromMsiDescriptor(component, NULL, &sz);
1412     if (Result != ERROR_SUCCESS)
1413         return NULL;
1414 
1415     sz++;
1416     LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, sz * sizeof(WCHAR));
1417     Result = CommandLineFromMsiDescriptor(component, path, &sz);
1418     if (Result != ERROR_SUCCESS)
1419     {
1420         HeapFree(GetProcessHeap(), 0, path);
1421         path = NULL;
1422     }
1423 
1424     TRACE("returning %s\n", debugstr_w(path));
1425 
1426     return path;
1427 }
1428 
1429 HRESULT STDMETHODCALLTYPE CShellLink::Resolve(HWND hwnd, DWORD fFlags)
1430 {
1431     HRESULT hr = S_OK;
1432     BOOL bSuccess;
1433 
1434     TRACE("(%p)->(hwnd=%p flags=%x)\n", this, hwnd, fFlags);
1435 
1436     /* FIXME: use IResolveShellLink interface? */
1437 
1438     // FIXME: See InvokeCommand().
1439 
1440 #if (NTDDI_VERSION < NTDDI_LONGHORN)
1441     // NOTE: For Logo3 (EXP_LOGO3_ID_SIG), check also for SHRestricted(REST_NOLOGO3CHANNELNOTIFY)
1442     if (m_Header.dwFlags & SLDF_HAS_LOGO3ID)
1443     {
1444         FIXME("Logo3 links are not supported yet!\n");
1445         return E_FAIL;
1446     }
1447 #endif
1448 
1449     /* Resolve Darwin (MSI) target */
1450     if (m_Header.dwFlags & SLDF_HAS_DARWINID)
1451     {
1452         LPWSTR component = NULL;
1453         hr = GetAdvertiseInfo(&component, EXP_DARWIN_ID_SIG);
1454         if (FAILED(hr))
1455             return E_FAIL;
1456 
1457         /* Clear the cached path */
1458         HeapFree(GetProcessHeap(), 0, m_sPath);
1459         m_sPath = shelllink_get_msi_component_path(component);
1460         if (!m_sPath)
1461             return E_FAIL;
1462     }
1463 
1464     if (!m_sPath && m_pPidl)
1465     {
1466         WCHAR buffer[MAX_PATH];
1467 
1468         bSuccess = SHGetPathFromIDListW(m_pPidl, buffer);
1469         if (bSuccess && *buffer)
1470         {
1471             m_sPath = strdupW(buffer);
1472             if (!m_sPath)
1473                 return E_OUTOFMEMORY;
1474 
1475             m_bDirty = TRUE;
1476         }
1477         else
1478         {
1479             hr = S_OK;    /* don't report an error occurred while just caching information */
1480         }
1481     }
1482 
1483     // FIXME: Strange to do that here...
1484     if (!m_sIcoPath && m_sPath)
1485     {
1486         m_sIcoPath = strdupW(m_sPath);
1487         if (!m_sIcoPath)
1488             return E_OUTOFMEMORY;
1489 
1490         m_Header.nIconIndex = 0;
1491 
1492         m_bDirty = TRUE;
1493     }
1494 
1495     return hr;
1496 }
1497 
1498 HRESULT STDMETHODCALLTYPE CShellLink::SetPath(LPCSTR pszFile)
1499 {
1500     TRACE("(%p)->(path=%s)\n", this, pszFile);
1501 
1502     if (!pszFile)
1503         return E_INVALIDARG;
1504 
1505     LPWSTR str = HEAP_strdupAtoW(GetProcessHeap(), 0, pszFile);
1506     if (!str)
1507         return E_OUTOFMEMORY;
1508 
1509     HRESULT hr = SetPath(str);
1510     HeapFree(GetProcessHeap(), 0, str);
1511 
1512     return hr;
1513 }
1514 
1515 HRESULT STDMETHODCALLTYPE CShellLink::GetPath(LPWSTR pszFile, INT cchMaxPath, WIN32_FIND_DATAW *pfd, DWORD fFlags)
1516 {
1517     WCHAR buffer[MAX_PATH];
1518 
1519     TRACE("(%p)->(pfile=%p len=%u find_data=%p flags=%u)(%s)\n",
1520           this, pszFile, cchMaxPath, pfd, fFlags, debugstr_w(m_sPath));
1521 
1522     if (cchMaxPath)
1523         *pszFile = 0;
1524     // FIXME: What if cchMaxPath == 0 , or pszFile == NULL ??
1525 
1526     // FIXME: What about Darwin??
1527 
1528     /*
1529      * Retrieve the path to the target from the PIDL (if we have one).
1530      * NOTE: Do NOT use the cached path (m_sPath from link info).
1531      */
1532     if (m_pPidl && SHGetPathFromIDListW(m_pPidl, buffer))
1533     {
1534         if (fFlags & SLGP_SHORTPATH)
1535             GetShortPathNameW(buffer, buffer, _countof(buffer));
1536         // FIXME: Add support for SLGP_UNCPRIORITY
1537     }
1538     else
1539     {
1540         *buffer = 0;
1541     }
1542 
1543     /* If we have a FindData structure, initialize it */
1544     if (pfd)
1545     {
1546         ZeroMemory(pfd, sizeof(*pfd));
1547 
1548         /* Copy the file data if the target is a file path */
1549         if (*buffer)
1550         {
1551             pfd->dwFileAttributes = m_Header.dwFileAttributes;
1552             pfd->ftCreationTime   = m_Header.ftCreationTime;
1553             pfd->ftLastAccessTime = m_Header.ftLastAccessTime;
1554             pfd->ftLastWriteTime  = m_Header.ftLastWriteTime;
1555             pfd->nFileSizeHigh    = 0;
1556             pfd->nFileSizeLow     = m_Header.nFileSizeLow;
1557 
1558             /*
1559              * Build temporarily a short path in pfd->cFileName (of size MAX_PATH),
1560              * then extract and store the short file name in pfd->cAlternateFileName.
1561              */
1562             GetShortPathNameW(buffer, pfd->cFileName, _countof(pfd->cFileName));
1563             lstrcpynW(pfd->cAlternateFileName,
1564                       PathFindFileNameW(pfd->cFileName),
1565                       _countof(pfd->cAlternateFileName));
1566 
1567             /* Now extract and store the long file name in pfd->cFileName */
1568             lstrcpynW(pfd->cFileName,
1569                       PathFindFileNameW(buffer),
1570                       _countof(pfd->cFileName));
1571         }
1572     }
1573 
1574     /* Finally check if we have a raw path the user actually wants to retrieve */
1575     if ((fFlags & SLGP_RAWPATH) && (m_Header.dwFlags & SLDF_HAS_EXP_SZ))
1576     {
1577         /* Search for a target environment block */
1578         LPEXP_SZ_LINK pInfo;
1579         pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_LINK_SIG);
1580         if (pInfo && (pInfo->cbSize == sizeof(*pInfo)))
1581             lstrcpynW(buffer, pInfo->szwTarget, cchMaxPath);
1582     }
1583 
1584     /* For diagnostics purposes only... */
1585     // NOTE: SLGP_UNCPRIORITY is unsupported
1586     fFlags &= ~(SLGP_RAWPATH | SLGP_SHORTPATH);
1587     if (fFlags) FIXME("(%p): Unsupported flags %lu\n", this, fFlags);
1588 
1589     /* Copy the data back to the user */
1590     if (*buffer)
1591         lstrcpynW(pszFile, buffer, cchMaxPath);
1592 
1593     return (*buffer ? S_OK : S_FALSE);
1594 }
1595 
1596 HRESULT STDMETHODCALLTYPE CShellLink::GetDescription(LPWSTR pszName, INT cchMaxName)
1597 {
1598     TRACE("(%p)->(%p len=%u)\n", this, pszName, cchMaxName);
1599 
1600     *pszName = 0;
1601     if (m_sDescription)
1602         lstrcpynW(pszName, m_sDescription, cchMaxName);
1603 
1604     return S_OK;
1605 }
1606 
1607 HRESULT STDMETHODCALLTYPE CShellLink::SetDescription(LPCWSTR pszName)
1608 {
1609     TRACE("(%p)->(desc=%s)\n", this, debugstr_w(pszName));
1610 
1611     HeapFree(GetProcessHeap(), 0, m_sDescription);
1612     m_sDescription = NULL;
1613 
1614     if (pszName)
1615     {
1616         m_sDescription = strdupW(pszName);
1617         if (!m_sDescription)
1618             return E_OUTOFMEMORY;
1619     }
1620     m_bDirty = TRUE;
1621 
1622     return S_OK;
1623 }
1624 
1625 HRESULT STDMETHODCALLTYPE CShellLink::GetWorkingDirectory(LPWSTR pszDir, INT cchMaxPath)
1626 {
1627     TRACE("(%p)->(%p len %u)\n", this, pszDir, cchMaxPath);
1628 
1629     if (cchMaxPath)
1630         *pszDir = 0;
1631 
1632     if (m_sWorkDir)
1633         lstrcpynW(pszDir, m_sWorkDir, cchMaxPath);
1634 
1635     return S_OK;
1636 }
1637 
1638 HRESULT STDMETHODCALLTYPE CShellLink::SetWorkingDirectory(LPCWSTR pszDir)
1639 {
1640     TRACE("(%p)->(dir=%s)\n", this, debugstr_w(pszDir));
1641 
1642     HeapFree(GetProcessHeap(), 0, m_sWorkDir);
1643     m_sWorkDir = NULL;
1644 
1645     if (pszDir)
1646     {
1647         m_sWorkDir = strdupW(pszDir);
1648         if (!m_sWorkDir)
1649             return E_OUTOFMEMORY;
1650     }
1651     m_bDirty = TRUE;
1652 
1653     return S_OK;
1654 }
1655 
1656 HRESULT STDMETHODCALLTYPE CShellLink::GetArguments(LPWSTR pszArgs, INT cchMaxPath)
1657 {
1658     TRACE("(%p)->(%p len=%u)\n", this, pszArgs, cchMaxPath);
1659 
1660     if (cchMaxPath)
1661         *pszArgs = 0;
1662 
1663     if (m_sArgs)
1664         lstrcpynW(pszArgs, m_sArgs, cchMaxPath);
1665 
1666     return S_OK;
1667 }
1668 
1669 HRESULT STDMETHODCALLTYPE CShellLink::SetArguments(LPCWSTR pszArgs)
1670 {
1671     TRACE("(%p)->(args=%s)\n", this, debugstr_w(pszArgs));
1672 
1673     HeapFree(GetProcessHeap(), 0, m_sArgs);
1674     m_sArgs = NULL;
1675 
1676     if (pszArgs)
1677     {
1678         m_sArgs = strdupW(pszArgs);
1679         if (!m_sArgs)
1680             return E_OUTOFMEMORY;
1681     }
1682     m_bDirty = TRUE;
1683 
1684     return S_OK;
1685 }
1686 
1687 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(LPWSTR pszIconPath, INT cchIconPath, INT *piIcon)
1688 {
1689     TRACE("(%p)->(%p len=%u iicon=%p)\n", this, pszIconPath, cchIconPath, piIcon);
1690 
1691     if (cchIconPath)
1692         *pszIconPath = 0;
1693 
1694     *piIcon = 0;
1695 
1696     /* Update the original icon path location */
1697     if (m_Header.dwFlags & SLDF_HAS_EXP_ICON_SZ)
1698     {
1699         WCHAR szPath[MAX_PATH];
1700 
1701         /* Search for an icon environment block */
1702         LPEXP_SZ_LINK pInfo;
1703         pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_ICON_SIG);
1704         if (pInfo && (pInfo->cbSize == sizeof(*pInfo)))
1705         {
1706             SHExpandEnvironmentStringsW(pInfo->szwTarget, szPath, _countof(szPath));
1707 
1708             m_Header.dwFlags &= ~SLDF_HAS_ICONLOCATION;
1709             HeapFree(GetProcessHeap(), 0, m_sIcoPath);
1710 
1711             m_sIcoPath = strdupW(szPath);
1712             if (!m_sIcoPath)
1713                 return E_OUTOFMEMORY;
1714 
1715             m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
1716 
1717             m_bDirty = TRUE;
1718         }
1719     }
1720 
1721     *piIcon = m_Header.nIconIndex;
1722 
1723     if (m_sIcoPath)
1724         lstrcpynW(pszIconPath, m_sIcoPath, cchIconPath);
1725 
1726     return S_OK;
1727 }
1728 
1729 static HRESULT SHELL_PidlGetIconLocationW(PCIDLIST_ABSOLUTE pidl,
1730         UINT uFlags, PWSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1731 {
1732     LPCITEMIDLIST pidlLast;
1733     CComPtr<IShellFolder> psf;
1734 
1735     HRESULT hr = SHBindToParent(pidl, IID_PPV_ARG(IShellFolder, &psf), &pidlLast);
1736     if (FAILED_UNEXPECTEDLY(hr))
1737         return hr;
1738 
1739     CComPtr<IExtractIconW> pei;
1740     hr = psf->GetUIObjectOf(0, 1, &pidlLast, IID_NULL_PPV_ARG(IExtractIconW, &pei));
1741     if (FAILED_UNEXPECTEDLY(hr))
1742         return hr;
1743 
1744     hr = pei->GetIconLocation(uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1745     if (FAILED_UNEXPECTEDLY(hr))
1746         return hr;
1747 
1748     return S_OK;
1749 }
1750 
1751 HRESULT STDMETHODCALLTYPE CShellLink::GetIconLocation(UINT uFlags, PWSTR pszIconFile, UINT cchMax, int *piIndex, UINT *pwFlags)
1752 {
1753     HRESULT hr;
1754 
1755     pszIconFile[0] = UNICODE_NULL;
1756 
1757     /*
1758      * It is possible for a shell link to point to another shell link,
1759      * and in particular there is the possibility to point to itself.
1760      * Now, suppose we ask such a link to retrieve its associated icon.
1761      * This function would be called, and due to COM would be called again
1762      * recursively. To solve this issue, we forbid calling GetIconLocation()
1763      * with GIL_FORSHORTCUT set in uFlags, as done by Windows (shown by tests).
1764      */
1765     if (uFlags & GIL_FORSHORTCUT)
1766         return E_INVALIDARG;
1767 
1768     /*
1769      * Now, we set GIL_FORSHORTCUT so that: i) we allow the icon extractor
1770      * of the target to give us a suited icon, and ii) we protect ourselves
1771      * against recursive call.
1772      */
1773     uFlags |= GIL_FORSHORTCUT;
1774 
1775     if (uFlags & GIL_DEFAULTICON)
1776         return S_FALSE;
1777 
1778     hr = GetIconLocation(pszIconFile, cchMax, piIndex);
1779     if (FAILED(hr) || pszIconFile[0] == UNICODE_NULL)
1780     {
1781         hr = SHELL_PidlGetIconLocationW(m_pPidl, uFlags, pszIconFile, cchMax, piIndex, pwFlags);
1782     }
1783     else
1784     {
1785         *pwFlags = GIL_NOTFILENAME | GIL_PERCLASS;
1786     }
1787 
1788     return hr;
1789 }
1790 
1791 HRESULT STDMETHODCALLTYPE
1792 CShellLink::Extract(PCWSTR pszFile, UINT nIconIndex, HICON *phiconLarge, HICON *phiconSmall, UINT nIconSize)
1793 {
1794     HRESULT hr = NOERROR;
1795     UINT cxyLarge = LOWORD(nIconSize), cxySmall = HIWORD(nIconSize);
1796 
1797     if (phiconLarge)
1798     {
1799         *phiconLarge = NULL;
1800         PrivateExtractIconsW(pszFile, nIconIndex, cxyLarge, cxyLarge, phiconLarge, NULL, 1, 0);
1801 
1802         if (*phiconLarge == NULL)
1803             hr = S_FALSE;
1804     }
1805 
1806     if (phiconSmall)
1807     {
1808         *phiconSmall = NULL;
1809         PrivateExtractIconsW(pszFile, nIconIndex, cxySmall, cxySmall, phiconSmall, NULL, 1, 0);
1810 
1811         if (*phiconSmall == NULL)
1812             hr = S_FALSE;
1813     }
1814 
1815     if (hr == S_FALSE)
1816     {
1817         if (phiconLarge && *phiconLarge)
1818         {
1819             DestroyIcon(*phiconLarge);
1820             *phiconLarge = NULL;
1821         }
1822         if (phiconSmall && *phiconSmall)
1823         {
1824             DestroyIcon(*phiconSmall);
1825             *phiconSmall = NULL;
1826         }
1827     }
1828 
1829     return hr;
1830 }
1831 
1832 #if 0
1833 /* Extends the functionality of PathUnExpandEnvStringsW */
1834 BOOL PathFullyUnExpandEnvStringsW(
1835     _In_  LPCWSTR pszPath,
1836     _Out_ LPWSTR  pszBuf,
1837     _In_  UINT    cchBuf)
1838 {
1839     BOOL Ret = FALSE; // Set to TRUE as soon as PathUnExpandEnvStrings starts unexpanding.
1840     BOOL res;
1841     LPCWSTR p;
1842 
1843     // *pszBuf = L'\0';
1844     while (*pszPath && cchBuf > 0)
1845     {
1846         /* Attempt unexpanding the path */
1847         res = PathUnExpandEnvStringsW(pszPath, pszBuf, cchBuf);
1848         if (!res)
1849         {
1850             /* The unexpansion failed. Try to find a path delimiter. */
1851             p = wcspbrk(pszPath, L" /\\:*?\"<>|%");
1852             if (!p) /* None found, we will copy the remaining path */
1853                 p = pszPath + wcslen(pszPath);
1854             else    /* Found one, we will copy the delimiter and skip it */
1855                 ++p;
1856             /* If we overflow, we cannot unexpand more, so return FALSE */
1857             if (p - pszPath >= cchBuf)
1858                 return FALSE; // *pszBuf = L'\0';
1859 
1860             /* Copy the untouched portion of path up to the delimiter, included */
1861             wcsncpy(pszBuf, pszPath, p - pszPath);
1862             pszBuf[p - pszPath] = L'\0'; // NULL-terminate
1863 
1864             /* Advance the pointers and decrease the remaining buffer size */
1865             cchBuf -= (p - pszPath);
1866             pszBuf += (p - pszPath);
1867             pszPath += (p - pszPath);
1868         }
1869         else
1870         {
1871             /*
1872              * The unexpansion succeeded. Skip the unexpanded part by trying
1873              * to find where the original path and the unexpanded string
1874              * become different.
1875              * NOTE: An alternative(?) would be to stop also at the last
1876              * path delimiter encountered in the loop (i.e. would be the
1877              * first path delimiter in the strings).
1878              */
1879             LPWSTR q;
1880 
1881             /*
1882              * The algorithm starts at the end of the strings and loops back
1883              * while the characters are equal, until it finds a discrepancy.
1884              */
1885             p = pszPath + wcslen(pszPath);
1886             q = pszBuf + wcslen(pszBuf); // This wcslen should be < cchBuf
1887             while ((*p == *q) && (p > pszPath) && (q > pszBuf))
1888             {
1889                 --p; --q;
1890             }
1891             /* Skip discrepancy */
1892             ++p; ++q;
1893 
1894             /* Advance the pointers and decrease the remaining buffer size */
1895             cchBuf -= (q - pszBuf);
1896             pszBuf = q;
1897             pszPath = p;
1898 
1899             Ret = TRUE;
1900         }
1901     }
1902 
1903     return Ret;
1904 }
1905 #endif
1906 
1907 HRESULT STDMETHODCALLTYPE CShellLink::SetIconLocation(LPCWSTR pszIconPath, INT iIcon)
1908 {
1909     HRESULT hr = E_FAIL;
1910     WCHAR szIconPath[MAX_PATH];
1911 
1912     TRACE("(%p)->(path=%s iicon=%u)\n", this, debugstr_w(pszIconPath), iIcon);
1913 
1914     if (pszIconPath)
1915     {
1916         /*
1917          * Check whether the user-given file path contains unexpanded
1918          * environment variables. If so, create a target environment block.
1919          * Note that in this block we will store the user-given path.
1920          * It will contain the unexpanded environment variables, but
1921          * it can also contain already expanded path that the user does
1922          * not want to see them unexpanded (e.g. so that they always
1923          * refer to the same place even if the would-be corresponding
1924          * environment variable could change).
1925          */
1926 #ifdef ICON_LINK_WINDOWS_COMPAT
1927         /* Try to fully unexpand the icon path */
1928         // if (PathFullyUnExpandEnvStringsW(pszIconPath, szIconPath, _countof(szIconPath)))
1929         BOOL bSuccess = PathUnExpandEnvStringsW(pszIconPath, szIconPath, _countof(szIconPath));
1930         if (bSuccess && wcscmp(pszIconPath, szIconPath) != 0)
1931 #else
1932         /*
1933          * In some situations, described in http://stackoverflow.com/questions/2976489/ishelllinkseticonlocation-translates-my-icon-path-into-program-files-which-i
1934          * the result of PathUnExpandEnvStringsW() could be wrong, and instead
1935          * one would have to store the actual provided icon location path, while
1936          * creating an icon environment block ONLY if that path already contains
1937          * environment variables. This is what the present case is trying to implement.
1938          */
1939         SHExpandEnvironmentStringsW(pszIconPath, szIconPath, _countof(szIconPath));
1940         if (wcscmp(pszIconPath, szIconPath) != 0)
1941 #endif
1942         {
1943             /*
1944              * The user-given file path contains unexpanded environment
1945              * variables, so we need an icon environment block.
1946              */
1947             EXP_SZ_LINK buffer;
1948             LPEXP_SZ_LINK pInfo;
1949 
1950 #ifdef ICON_LINK_WINDOWS_COMPAT
1951             /* Make pszIconPath point to the unexpanded path */
1952             LPCWSTR pszOrgIconPath = pszIconPath;
1953             pszIconPath = szIconPath;
1954 #endif
1955             pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_ICON_SIG);
1956             if (pInfo)
1957             {
1958                 /* Make sure that the size of the structure is valid */
1959                 if (pInfo->cbSize != sizeof(*pInfo))
1960                 {
1961                     ERR("Ooops. This structure is not as expected...\n");
1962 
1963                     /* Invalid structure, remove it altogether */
1964                     m_Header.dwFlags &= ~SLDF_HAS_EXP_ICON_SZ;
1965                     RemoveDataBlock(EXP_SZ_ICON_SIG);
1966 
1967                     /* Reset the pointer and go use the static buffer */
1968                     pInfo = NULL;
1969                 }
1970             }
1971             if (!pInfo)
1972             {
1973                 /* Use the static buffer */
1974                 pInfo = &buffer;
1975                 buffer.cbSize = sizeof(buffer);
1976                 buffer.dwSignature = EXP_SZ_ICON_SIG;
1977             }
1978 
1979             lstrcpynW(pInfo->szwTarget, pszIconPath, _countof(pInfo->szwTarget));
1980             WideCharToMultiByte(CP_ACP, 0, pszIconPath, -1,
1981                                 pInfo->szTarget, _countof(pInfo->szTarget), NULL, NULL);
1982 
1983             hr = S_OK;
1984             if (pInfo == &buffer)
1985                 hr = AddDataBlock(pInfo);
1986             if (hr == S_OK)
1987                 m_Header.dwFlags |= SLDF_HAS_EXP_ICON_SZ;
1988 
1989 #ifdef ICON_LINK_WINDOWS_COMPAT
1990             /* Set pszIconPath back to the original one */
1991             pszIconPath = pszOrgIconPath;
1992 #else
1993             /* Now, make pszIconPath point to the expanded path */
1994             pszIconPath = szIconPath;
1995 #endif
1996         }
1997         else
1998         {
1999             /*
2000              * The user-given file path does not contain unexpanded environment
2001              * variables, so we need to remove any icon environment block.
2002              */
2003             m_Header.dwFlags &= ~SLDF_HAS_EXP_ICON_SZ;
2004             RemoveDataBlock(EXP_SZ_ICON_SIG);
2005 
2006             /* pszIconPath points to the user path */
2007         }
2008     }
2009 
2010 #ifdef ICON_LINK_WINDOWS_COMPAT
2011     /* Store the original icon path location (may contain unexpanded environment strings) */
2012 #endif
2013     if (pszIconPath)
2014     {
2015         m_Header.dwFlags &= ~SLDF_HAS_ICONLOCATION;
2016         HeapFree(GetProcessHeap(), 0, m_sIcoPath);
2017 
2018         m_sIcoPath = strdupW(pszIconPath);
2019         if (!m_sIcoPath)
2020             return E_OUTOFMEMORY;
2021 
2022         m_Header.dwFlags |= SLDF_HAS_ICONLOCATION;
2023     }
2024 
2025     hr = S_OK;
2026 
2027     m_Header.nIconIndex = iIcon;
2028     m_bDirty = TRUE;
2029 
2030     return hr;
2031 }
2032 
2033 HRESULT STDMETHODCALLTYPE CShellLink::SetRelativePath(LPCWSTR pszPathRel, DWORD dwReserved)
2034 {
2035     TRACE("(%p)->(path=%s %x)\n", this, debugstr_w(pszPathRel), dwReserved);
2036 
2037     HeapFree(GetProcessHeap(), 0, m_sPathRel);
2038     m_sPathRel = NULL;
2039 
2040     if (pszPathRel)
2041     {
2042         m_sPathRel = strdupW(pszPathRel);
2043         if (!m_sPathRel)
2044             return E_OUTOFMEMORY;
2045     }
2046     m_bDirty = TRUE;
2047 
2048     return ShellLink_UpdatePath(m_sPathRel, m_sPath, m_sWorkDir, &m_sPath);
2049 }
2050 
2051 static LPWSTR GetAdvertisedArg(LPCWSTR str)
2052 {
2053     if (!str)
2054         return NULL;
2055 
2056     LPCWSTR p = wcschr(str, L':');
2057     if (!p)
2058         return NULL;
2059 
2060     DWORD len = p - str;
2061     LPWSTR ret = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, sizeof(WCHAR) * (len + 1));
2062     if (!ret)
2063         return ret;
2064 
2065     memcpy(ret, str, sizeof(WCHAR)*len);
2066     ret[len] = 0;
2067     return ret;
2068 }
2069 
2070 HRESULT CShellLink::WriteAdvertiseInfo(LPCWSTR string, DWORD dwSig)
2071 {
2072     EXP_DARWIN_LINK buffer;
2073     LPEXP_DARWIN_LINK pInfo;
2074 
2075     if (   (dwSig != EXP_DARWIN_ID_SIG)
2076 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2077         && (dwSig != EXP_LOGO3_ID_SIG)
2078 #endif
2079         )
2080     {
2081         return E_INVALIDARG;
2082     }
2083 
2084     if (!string)
2085         return S_FALSE;
2086 
2087     pInfo = (LPEXP_DARWIN_LINK)SHFindDataBlock(m_pDBList, dwSig);
2088     if (pInfo)
2089     {
2090         /* Make sure that the size of the structure is valid */
2091         if (pInfo->dbh.cbSize != sizeof(*pInfo))
2092         {
2093             ERR("Ooops. This structure is not as expected...\n");
2094 
2095             /* Invalid structure, remove it altogether */
2096             if (dwSig == EXP_DARWIN_ID_SIG)
2097                 m_Header.dwFlags &= ~SLDF_HAS_DARWINID;
2098 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2099             else if (dwSig == EXP_LOGO3_ID_SIG)
2100                 m_Header.dwFlags &= ~SLDF_HAS_LOGO3ID;
2101 #endif
2102             RemoveDataBlock(dwSig);
2103 
2104             /* Reset the pointer and go use the static buffer */
2105             pInfo = NULL;
2106         }
2107     }
2108     if (!pInfo)
2109     {
2110         /* Use the static buffer */
2111         pInfo = &buffer;
2112         buffer.dbh.cbSize = sizeof(buffer);
2113         buffer.dbh.dwSignature = dwSig;
2114     }
2115 
2116     lstrcpynW(pInfo->szwDarwinID, string, _countof(pInfo->szwDarwinID));
2117     WideCharToMultiByte(CP_ACP, 0, string, -1,
2118                         pInfo->szDarwinID, _countof(pInfo->szDarwinID), NULL, NULL);
2119 
2120     HRESULT hr = S_OK;
2121     if (pInfo == &buffer)
2122         hr = AddDataBlock(pInfo);
2123     if (hr == S_OK)
2124     {
2125         if (dwSig == EXP_DARWIN_ID_SIG)
2126             m_Header.dwFlags |= SLDF_HAS_DARWINID;
2127 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2128         else if (dwSig == EXP_LOGO3_ID_SIG)
2129             m_Header.dwFlags |= SLDF_HAS_LOGO3ID;
2130 #endif
2131     }
2132 
2133     return hr;
2134 }
2135 
2136 HRESULT CShellLink::SetAdvertiseInfo(LPCWSTR str)
2137 {
2138     HRESULT hr;
2139     LPCWSTR szComponent = NULL, szProduct = NULL, p;
2140     INT len;
2141     GUID guid;
2142     WCHAR szGuid[38+1];
2143 
2144     /**/sProduct = sComponent = NULL;/**/
2145 
2146     while (str[0])
2147     {
2148         /* each segment must start with two colons */
2149         if (str[0] != ':' || str[1] != ':')
2150             return E_FAIL;
2151 
2152         /* the last segment is just two colons */
2153         if (!str[2])
2154             break;
2155         str += 2;
2156 
2157         /* there must be a colon straight after a guid */
2158         p = wcschr(str, L':');
2159         if (!p)
2160             return E_FAIL;
2161         len = p - str;
2162         if (len != 38)
2163             return E_FAIL;
2164 
2165         /* get the guid, and check if it's validly formatted */
2166         memcpy(szGuid, str, sizeof(WCHAR)*len);
2167         szGuid[len] = 0;
2168 
2169         hr = CLSIDFromString(szGuid, &guid);
2170         if (hr != S_OK)
2171             return hr;
2172         str = p + 1;
2173 
2174         /* match it up to a guid that we care about */
2175         if (IsEqualGUID(guid, SHELL32_AdvtShortcutComponent) && !szComponent)
2176             szComponent = str; /* Darwin */
2177         else if (IsEqualGUID(guid, SHELL32_AdvtShortcutProduct) && !szProduct)
2178             szProduct = str;   /* Logo3  */
2179         else
2180             return E_FAIL;
2181 
2182         /* skip to the next field */
2183         str = wcschr(str, L':');
2184         if (!str)
2185             return E_FAIL;
2186     }
2187 
2188     /* we have to have a component for an advertised shortcut */
2189     if (!szComponent)
2190         return E_FAIL;
2191 
2192     szComponent = GetAdvertisedArg(szComponent);
2193     szProduct = GetAdvertisedArg(szProduct);
2194 
2195     hr = WriteAdvertiseInfo(szComponent, EXP_DARWIN_ID_SIG);
2196     // if (FAILED(hr))
2197         // return hr;
2198 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2199     hr = WriteAdvertiseInfo(szProduct, EXP_LOGO3_ID_SIG);
2200     // if (FAILED(hr))
2201         // return hr;
2202 #endif
2203 
2204     HeapFree(GetProcessHeap(), 0, (PVOID)szComponent);
2205     HeapFree(GetProcessHeap(), 0, (PVOID)szProduct);
2206 
2207     if (TRACE_ON(shell))
2208     {
2209         GetAdvertiseInfo(&sComponent, EXP_DARWIN_ID_SIG);
2210         TRACE("Component = %s\n", debugstr_w(sComponent));
2211 #if (NTDDI_VERSION < NTDDI_LONGHORN)
2212         GetAdvertiseInfo(&sProduct, EXP_LOGO3_ID_SIG);
2213         TRACE("Product = %s\n", debugstr_w(sProduct));
2214 #endif
2215     }
2216 
2217     return S_OK;
2218 }
2219 
2220 HRESULT CShellLink::SetTargetFromPIDLOrPath(LPCITEMIDLIST pidl, LPCWSTR pszFile)
2221 {
2222     HRESULT hr = S_OK;
2223     LPITEMIDLIST pidlNew = NULL;
2224     WCHAR szPath[MAX_PATH];
2225 
2226     /*
2227      * Not both 'pidl' and 'pszFile' should be set.
2228      * But either one or both can be NULL.
2229      */
2230     if (pidl && pszFile)
2231         return E_FAIL;
2232 
2233     if (pidl)
2234     {
2235         /* Clone the PIDL */
2236         pidlNew = ILClone(pidl);
2237         if (!pidlNew)
2238             return E_FAIL;
2239     }
2240     else if (pszFile)
2241     {
2242         /* Build a PIDL for this path target */
2243         hr = SHILCreateFromPathW(pszFile, &pidlNew, NULL);
2244         if (FAILED(hr))
2245         {
2246             /* This failed, try to resolve the path, then create a simple PIDL */
2247 
2248             StringCchCopyW(szPath, _countof(szPath), pszFile);
2249             PathResolveW(szPath, NULL, PRF_TRYPROGRAMEXTENSIONS);
2250 
2251             if (PathIsFileSpecW(szPath))
2252             {
2253                 hr = E_INVALIDARG;
2254                 szPath[0] = 0;
2255             }
2256             else
2257             {
2258                 hr = S_OK;
2259                 pidlNew = SHSimpleIDListFromPathW(szPath);
2260                 // NOTE: Don't make it failed here even if pidlNew was NULL.
2261                 // We don't fail on purpose even if SHSimpleIDListFromPathW returns NULL.
2262                 // This behaviour has been verified with tests.
2263             }
2264         }
2265     }
2266     // else if (!pidl && !pszFile) { pidlNew = NULL; hr = S_OK; }
2267 
2268     ILFree(m_pPidl);
2269     m_pPidl = pidlNew;
2270 
2271     if (!pszFile)
2272     {
2273         if (SHGetPathFromIDListW(pidlNew, szPath))
2274             pszFile = szPath;
2275     }
2276 
2277     // TODO: Fully update link info, tracker, file attribs...
2278 
2279     // if (pszFile)
2280     if (!pszFile)
2281     {
2282         *szPath = L'\0';
2283         pszFile = szPath;
2284     }
2285 
2286     /* Update the cached path (for link info) */
2287     ShellLink_GetVolumeInfo(pszFile, &volume);
2288 
2289     if (m_sPath)
2290         HeapFree(GetProcessHeap(), 0, m_sPath);
2291 
2292     m_sPath = strdupW(pszFile);
2293     if (!m_sPath)
2294         return E_OUTOFMEMORY;
2295 
2296     m_bDirty = TRUE;
2297     return hr;
2298 }
2299 
2300 HRESULT STDMETHODCALLTYPE CShellLink::SetPath(LPCWSTR pszFile)
2301 {
2302     LPWSTR unquoted = NULL;
2303     HRESULT hr = S_OK;
2304 
2305     TRACE("(%p)->(path=%s)\n", this, debugstr_w(pszFile));
2306 
2307     if (!pszFile)
2308         return E_INVALIDARG;
2309 
2310     /*
2311      * Allow upgrading Logo3 shortcuts (m_Header.dwFlags & SLDF_HAS_LOGO3ID),
2312      * but forbid upgrading Darwin ones.
2313      */
2314     if (m_Header.dwFlags & SLDF_HAS_DARWINID)
2315         return S_FALSE;
2316 
2317     /* quotes at the ends of the string are stripped */
2318     SIZE_T len = wcslen(pszFile);
2319     if (pszFile[0] == L'"' && pszFile[len-1] == L'"')
2320     {
2321         unquoted = strdupW(pszFile);
2322         PathUnquoteSpacesW(unquoted);
2323         pszFile = unquoted;
2324     }
2325 
2326     /* any other quote marks are invalid */
2327     if (wcschr(pszFile, L'"'))
2328     {
2329         hr = S_FALSE;
2330         goto end;
2331     }
2332 
2333     /* Clear the cached path */
2334     HeapFree(GetProcessHeap(), 0, m_sPath);
2335     m_sPath = NULL;
2336 
2337     /* Check for an advertised target (Logo3 or Darwin) */
2338     if (SetAdvertiseInfo(pszFile) != S_OK)
2339     {
2340         /* This is not an advertised target, but a regular path */
2341         WCHAR szPath[MAX_PATH];
2342 
2343         /*
2344          * Check whether the user-given file path contains unexpanded
2345          * environment variables. If so, create a target environment block.
2346          * Note that in this block we will store the user-given path.
2347          * It will contain the unexpanded environment variables, but
2348          * it can also contain already expanded path that the user does
2349          * not want to see them unexpanded (e.g. so that they always
2350          * refer to the same place even if the would-be corresponding
2351          * environment variable could change).
2352          */
2353         if (*pszFile)
2354             SHExpandEnvironmentStringsW(pszFile, szPath, _countof(szPath));
2355         else
2356             *szPath = L'\0';
2357 
2358         if (*pszFile && (wcscmp(pszFile, szPath) != 0))
2359         {
2360             /*
2361              * The user-given file path contains unexpanded environment
2362              * variables, so we need a target environment block.
2363              */
2364             EXP_SZ_LINK buffer;
2365             LPEXP_SZ_LINK pInfo;
2366 
2367             pInfo = (LPEXP_SZ_LINK)SHFindDataBlock(m_pDBList, EXP_SZ_LINK_SIG);
2368             if (pInfo)
2369             {
2370                 /* Make sure that the size of the structure is valid */
2371                 if (pInfo->cbSize != sizeof(*pInfo))
2372                 {
2373                     ERR("Ooops. This structure is not as expected...\n");
2374 
2375                     /* Invalid structure, remove it altogether */
2376                     m_Header.dwFlags &= ~SLDF_HAS_EXP_SZ;
2377                     RemoveDataBlock(EXP_SZ_LINK_SIG);
2378 
2379                     /* Reset the pointer and go use the static buffer */
2380                     pInfo = NULL;
2381                 }
2382             }
2383             if (!pInfo)
2384             {
2385                 /* Use the static buffer */
2386                 pInfo = &buffer;
2387                 buffer.cbSize = sizeof(buffer);
2388                 buffer.dwSignature = EXP_SZ_LINK_SIG;
2389             }
2390 
2391             lstrcpynW(pInfo->szwTarget, pszFile, _countof(pInfo->szwTarget));
2392             WideCharToMultiByte(CP_ACP, 0, pszFile, -1,
2393                                 pInfo->szTarget, _countof(pInfo->szTarget), NULL, NULL);
2394 
2395             hr = S_OK;
2396             if (pInfo == &buffer)
2397                 hr = AddDataBlock(pInfo);
2398             if (hr == S_OK)
2399                 m_Header.dwFlags |= SLDF_HAS_EXP_SZ;
2400 
2401             /* Now, make pszFile point to the expanded path */
2402             pszFile = szPath;
2403         }
2404         else
2405         {
2406             /*
2407              * The user-given file path does not contain unexpanded environment
2408              * variables, so we need to remove any target environment block.
2409              */
2410             m_Header.dwFlags &= ~SLDF_HAS_EXP_SZ;
2411             RemoveDataBlock(EXP_SZ_LINK_SIG);
2412 
2413             /* pszFile points to the user path */
2414         }
2415 
2416         /* Set the target */
2417         hr = SetTargetFromPIDLOrPath(NULL, pszFile);
2418     }
2419 
2420     m_bDirty = TRUE;
2421 
2422 end:
2423     HeapFree(GetProcessHeap(), 0, unquoted);
2424     return hr;
2425 }
2426 
2427 HRESULT STDMETHODCALLTYPE CShellLink::AddDataBlock(void* pDataBlock)
2428 {
2429     if (SHAddDataBlock(&m_pDBList, (DATABLOCK_HEADER*)pDataBlock))
2430     {
2431         m_bDirty = TRUE;
2432         return S_OK;
2433     }
2434     return S_FALSE;
2435 }
2436 
2437 HRESULT STDMETHODCALLTYPE CShellLink::CopyDataBlock(DWORD dwSig, void** ppDataBlock)
2438 {
2439     DATABLOCK_HEADER* pBlock;
2440     PVOID pDataBlock;
2441 
2442     TRACE("%p %08x %p\n", this, dwSig, ppDataBlock);
2443 
2444     *ppDataBlock = NULL;
2445 
2446     pBlock = SHFindDataBlock(m_pDBList, dwSig);
2447     if (!pBlock)
2448     {
2449         ERR("unknown datablock %08x (not found)\n", dwSig);
2450         return E_FAIL;
2451     }
2452 
2453     pDataBlock = LocalAlloc(LMEM_ZEROINIT, pBlock->cbSize);
2454     if (!pDataBlock)
2455         return E_OUTOFMEMORY;
2456 
2457     CopyMemory(pDataBlock, pBlock, pBlock->cbSize);
2458 
2459     *ppDataBlock = pDataBlock;
2460     return S_OK;
2461 }
2462 
2463 HRESULT STDMETHODCALLTYPE CShellLink::RemoveDataBlock(DWORD dwSig)
2464 {
2465     if (SHRemoveDataBlock(&m_pDBList, dwSig))
2466     {
2467         m_bDirty = TRUE;
2468         return S_OK;
2469     }
2470     return S_FALSE;
2471 }
2472 
2473 HRESULT STDMETHODCALLTYPE CShellLink::GetFlags(DWORD *pdwFlags)
2474 {
2475     TRACE("%p %p\n", this, pdwFlags);
2476     *pdwFlags = m_Header.dwFlags;
2477     return S_OK;
2478 }
2479 
2480 HRESULT STDMETHODCALLTYPE CShellLink::SetFlags(DWORD dwFlags)
2481 {
2482 #if 0 // FIXME!
2483     m_Header.dwFlags = dwFlags;
2484     m_bDirty = TRUE;
2485     return S_OK;
2486 #else
2487     FIXME("\n");
2488     return E_NOTIMPL;
2489 #endif
2490 }
2491 
2492 /**************************************************************************
2493  * CShellLink implementation of IShellExtInit::Initialize()
2494  *
2495  * Loads the shelllink from the dataobject the shell is pointing to.
2496  */
2497 HRESULT STDMETHODCALLTYPE CShellLink::Initialize(PCIDLIST_ABSOLUTE pidlFolder, IDataObject *pdtobj, HKEY hkeyProgID)
2498 {
2499     TRACE("%p %p %p %p\n", this, pidlFolder, pdtobj, hkeyProgID);
2500 
2501     if (!pdtobj)
2502         return E_FAIL;
2503 
2504     FORMATETC format;
2505     format.cfFormat = CF_HDROP;
2506     format.ptd = NULL;
2507     format.dwAspect = DVASPECT_CONTENT;
2508     format.lindex = -1;
2509     format.tymed = TYMED_HGLOBAL;
2510 
2511     STGMEDIUM stgm;
2512     HRESULT hr = pdtobj->GetData(&format, &stgm);
2513     if (FAILED(hr))
2514         return hr;
2515 
2516     UINT count = DragQueryFileW((HDROP)stgm.hGlobal, -1, NULL, 0);
2517     if (count == 1)
2518     {
2519         count = DragQueryFileW((HDROP)stgm.hGlobal, 0, NULL, 0);
2520         count++;
2521         LPWSTR path = (LPWSTR)HeapAlloc(GetProcessHeap(), 0, count * sizeof(WCHAR));
2522         if (path)
2523         {
2524             count = DragQueryFileW((HDROP)stgm.hGlobal, 0, path, count);
2525             hr = Load(path, 0);
2526             HeapFree(GetProcessHeap(), 0, path);
2527         }
2528     }
2529     ReleaseStgMedium(&stgm);
2530 
2531     return S_OK;
2532 }
2533 
2534 HRESULT STDMETHODCALLTYPE CShellLink::QueryContextMenu(HMENU hMenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
2535 {
2536     INT id = 0;
2537 
2538     m_idCmdFirst = idCmdFirst;
2539 
2540     TRACE("%p %p %u %u %u %u\n", this,
2541           hMenu, indexMenu, idCmdFirst, idCmdLast, uFlags);
2542 
2543     if (!hMenu)
2544         return E_INVALIDARG;
2545 
2546     CStringW strOpen(MAKEINTRESOURCEW(IDS_OPEN_VERB));
2547     CStringW strOpenFileLoc(MAKEINTRESOURCEW(IDS_OPENFILELOCATION));
2548 
2549     MENUITEMINFOW mii;
2550     ZeroMemory(&mii, sizeof(mii));
2551     mii.cbSize = sizeof(mii);
2552     mii.fMask = MIIM_TYPE | MIIM_ID | MIIM_STATE;
2553     mii.dwTypeData = strOpen.GetBuffer();
2554     mii.cch = wcslen(mii.dwTypeData);
2555     mii.wID = idCmdFirst + id++;
2556     mii.fState = MFS_DEFAULT | MFS_ENABLED;
2557     mii.fType = MFT_STRING;
2558     if (!InsertMenuItemW(hMenu, indexMenu++, TRUE, &mii))
2559         return E_FAIL;
2560 
2561     mii.fMask = MIIM_TYPE | MIIM_ID | MIIM_STATE;
2562     mii.dwTypeData = strOpenFileLoc.GetBuffer();
2563     mii.cch = wcslen(mii.dwTypeData);
2564     mii.wID = idCmdFirst + id++;
2565     mii.fState = MFS_ENABLED;
2566     mii.fType = MFT_STRING;
2567     if (!InsertMenuItemW(hMenu, indexMenu++, TRUE, &mii))
2568         return E_FAIL;
2569 
2570     UNREFERENCED_PARAMETER(indexMenu);
2571 
2572     return MAKE_HRESULT(SEVERITY_SUCCESS, 0, id);
2573 }
2574 
2575 HRESULT CShellLink::DoOpenFileLocation()
2576 {
2577     WCHAR szParams[MAX_PATH + 64];
2578     StringCbPrintfW(szParams, sizeof(szParams), L"/select,%s", m_sPath);
2579 
2580     INT_PTR ret;
2581     ret = reinterpret_cast<INT_PTR>(ShellExecuteW(NULL, NULL, L"explorer.exe", szParams,
2582                                                   NULL, m_Header.nShowCommand));
2583     if (ret <= 32)
2584     {
2585         ERR("ret: %08lX\n", ret);
2586         return E_FAIL;
2587     }
2588 
2589     return S_OK;
2590 }
2591 
2592 HRESULT STDMETHODCALLTYPE CShellLink::InvokeCommand(LPCMINVOKECOMMANDINFO lpici)
2593 {
2594     TRACE("%p %p\n", this, lpici);
2595 
2596     if (lpici->cbSize < sizeof(CMINVOKECOMMANDINFO))
2597         return E_INVALIDARG;
2598 
2599     // NOTE: We could use lpici->hwnd (certainly in case lpici->fMask doesn't contain CMIC_MASK_FLAG_NO_UI)
2600     // as the parent window handle... ?
2601     /* FIXME: get using interface set from IObjectWithSite?? */
2602     // NOTE: We might need an extended version of Resolve that provides us with paths...
2603     HRESULT hr = Resolve(lpici->hwnd, (lpici->fMask & CMIC_MASK_FLAG_NO_UI) ? SLR_NO_UI : 0);
2604     if (FAILED(hr))
2605     {
2606         TRACE("failed to resolve component error 0x%08x\n", hr);
2607         return hr;
2608     }
2609 
2610     UINT idCmd = LOWORD(lpici->lpVerb);
2611     TRACE("idCmd: %d\n", idCmd);
2612 
2613     switch (idCmd)
2614     {
2615     case IDCMD_OPEN:
2616         return DoOpen(lpici);
2617     case IDCMD_OPENFILELOCATION:
2618         return DoOpenFileLocation();
2619     default:
2620         return E_NOTIMPL;
2621     }
2622 }
2623 
2624 HRESULT CShellLink::DoOpen(LPCMINVOKECOMMANDINFO lpici)
2625 {
2626     LPCMINVOKECOMMANDINFOEX iciex = (LPCMINVOKECOMMANDINFOEX)lpici;
2627     const BOOL unicode = IsUnicode(*lpici);
2628 
2629     CStringW args;
2630     if (m_sArgs)
2631         args = m_sArgs;
2632 
2633     if (unicode)
2634     {
2635         if (!StrIsNullOrEmpty(iciex->lpParametersW))
2636         {
2637             args += L' ';
2638             args += iciex->lpParametersW;
2639         }
2640     }
2641     else
2642     {
2643         CComHeapPtr<WCHAR> pszParams;
2644         if (!StrIsNullOrEmpty(lpici->lpParameters) && __SHCloneStrAtoW(&pszParams, lpici->lpParameters))
2645         {
2646             args += L' ';
2647             args += pszParams;
2648         }
2649     }
2650 
2651     WCHAR dir[MAX_PATH];
2652     SHELLEXECUTEINFOW sei = { sizeof(sei) };
2653     sei.fMask = SEE_MASK_HASLINKNAME | SEE_MASK_UNICODE | SEE_MASK_DOENVSUBST |
2654                (lpici->fMask & (SEE_MASK_NOASYNC | SEE_MASK_ASYNCOK | SEE_MASK_FLAG_NO_UI));
2655     sei.lpDirectory = m_sWorkDir;
2656     if (m_pPidl)
2657     {
2658         sei.lpIDList = m_pPidl;
2659         sei.fMask |= SEE_MASK_INVOKEIDLIST;
2660     }
2661     else
2662     {
2663         sei.lpFile = m_sPath;
2664         if (!(m_Header.dwFlags & SLDF_HAS_EXP_SZ))
2665         {
2666             sei.fMask &= ~SEE_MASK_DOENVSUBST; // The link does not want to expand lpFile
2667             if (m_sWorkDir && ExpandEnvironmentStringsW(m_sWorkDir, dir, _countof(dir)) <= _countof(dir))
2668                 sei.lpDirectory = dir;
2669         }
2670     }
2671     sei.lpParameters = args;
2672     sei.lpClass = m_sLinkPath;
2673     sei.nShow = m_Header.nShowCommand;
2674     if (lpici->nShow != SW_SHOWNORMAL && lpici->nShow != SW_SHOW)
2675         sei.nShow = lpici->nShow; // Allow invoker to override .lnk show mode
2676 
2677     // Use the invoker specified working directory if the link did not specify one
2678     if (StrIsNullOrEmpty(sei.lpDirectory) || !PathIsDirectoryW(sei.lpDirectory))
2679     {
2680         LPCSTR pszDirA = lpici->lpDirectory;
2681         if (unicode && !StrIsNullOrEmpty(iciex->lpDirectoryW))
2682             sei.lpDirectory = iciex->lpDirectoryW;
2683         else if (pszDirA && SHAnsiToUnicode(pszDirA, dir, _countof(dir)))
2684             sei.lpDirectory = dir;
2685     }
2686     return (ShellExecuteExW(&sei) ? S_OK : E_FAIL);
2687 }
2688 
2689 HRESULT STDMETHODCALLTYPE CShellLink::GetCommandString(UINT_PTR idCmd, UINT uType, UINT* pwReserved, LPSTR pszName, UINT cchMax)
2690 {
2691     FIXME("%p %lu %u %p %p %u\n", this, idCmd, uType, pwReserved, pszName, cchMax);
2692     return E_NOTIMPL;
2693 }
2694 
2695 INT_PTR CALLBACK ExtendedShortcutProc(HWND hwndDlg, UINT uMsg,
2696                                       WPARAM wParam, LPARAM lParam)
2697 {
2698     switch(uMsg)
2699     {
2700         case WM_INITDIALOG:
2701             if (lParam)
2702             {
2703                 HWND hDlgCtrl = GetDlgItem(hwndDlg, IDC_SHORTEX_RUN_DIFFERENT);
2704                 SendMessage(hDlgCtrl, BM_SETCHECK, BST_CHECKED, 0);
2705             }
2706             return TRUE;
2707         case WM_COMMAND:
2708         {
2709             HWND hDlgCtrl = GetDlgItem(hwndDlg, IDC_SHORTEX_RUN_DIFFERENT);
2710             if (LOWORD(wParam) == IDOK)
2711             {
2712                 if (SendMessage(hDlgCtrl, BM_GETCHECK, 0, 0) == BST_CHECKED)
2713                     EndDialog(hwndDlg, 1);
2714                 else
2715                     EndDialog(hwndDlg, 0);
2716             }
2717             else if (LOWORD(wParam) == IDCANCEL)
2718             {
2719                 EndDialog(hwndDlg, -1);
2720             }
2721             else if (LOWORD(wParam) == IDC_SHORTEX_RUN_DIFFERENT)
2722             {
2723                 if (SendMessage(hDlgCtrl, BM_GETCHECK, 0, 0) == BST_CHECKED)
2724                     SendMessage(hDlgCtrl, BM_SETCHECK, BST_UNCHECKED, 0);
2725                 else
2726                     SendMessage(hDlgCtrl, BM_SETCHECK, BST_CHECKED, 0);
2727             }
2728         }
2729     }
2730     return FALSE;
2731 }
2732 
2733 static void GetTypeDescriptionByPath(PCWSTR pszFullPath, DWORD fAttributes, PWSTR szBuf, UINT cchBuf)
2734 {
2735     if (fAttributes == INVALID_FILE_ATTRIBUTES && !PathFileExistsAndAttributesW(pszFullPath, &fAttributes))
2736         fAttributes = 0;
2737 
2738     SHFILEINFOW fi;
2739     if (!SHGetFileInfoW(pszFullPath, fAttributes, &fi, sizeof(fi), SHGFI_TYPENAME | SHGFI_USEFILEATTRIBUTES))
2740     {
2741         ERR("SHGetFileInfoW failed for %ls (%lu)\n", pszFullPath, GetLastError());
2742         fi.szTypeName[0] = UNICODE_NULL;
2743     }
2744 
2745     BOOL fFolder = (fAttributes & FILE_ATTRIBUTE_DIRECTORY);
2746     LPCWSTR pwszExt = fFolder ? L"" : PathFindExtensionW(pszFullPath);
2747     if (pwszExt[0])
2748     {
2749         if (!fi.szTypeName[0])
2750             StringCchPrintfW(szBuf, cchBuf, L"%s", pwszExt + 1);
2751         else
2752             StringCchPrintfW(szBuf, cchBuf, L"%s (%s)", fi.szTypeName, pwszExt);
2753     }
2754     else
2755     {
2756         StringCchPrintfW(szBuf, cchBuf, L"%s", fi.szTypeName);
2757     }
2758 }
2759 
2760 BOOL CShellLink::OnInitDialog(HWND hwndDlg, HWND hwndFocus, LPARAM lParam)
2761 {
2762     TRACE("CShellLink::OnInitDialog(hwnd %p hwndFocus %p lParam %p)\n", hwndDlg, hwndFocus, lParam);
2763 
2764     Resolve(0, SLR_NO_UI | SLR_NOUPDATE | SLR_NOSEARCH | SLR_NOTRACK);
2765 
2766     TRACE("m_sArgs: %S sComponent: %S m_sDescription: %S m_sIcoPath: %S m_sPath: %S m_sPathRel: %S sProduct: %S m_sWorkDir: %S\n", m_sArgs, sComponent, m_sDescription,
2767           m_sIcoPath, m_sPath, m_sPathRel, sProduct, m_sWorkDir);
2768 
2769     m_bInInit = TRUE;
2770     UINT darwin = m_Header.dwFlags & (SLDF_HAS_DARWINID);
2771 
2772     /* Get file information */
2773     // FIXME! FIXME! Shouldn't we use m_sIcoPath, m_Header.nIconIndex instead???
2774     SHFILEINFOW fi;
2775     if (!SHGetFileInfoW(m_sLinkPath, 0, &fi, sizeof(fi), SHGFI_TYPENAME | SHGFI_ICON))
2776     {
2777         ERR("SHGetFileInfoW failed for %ls (%lu)\n", m_sLinkPath, GetLastError());
2778         fi.szTypeName[0] = L'\0';
2779         fi.hIcon = NULL;
2780     }
2781 
2782     if (fi.hIcon)
2783     {
2784         if (m_hIcon)
2785             DestroyIcon(m_hIcon);
2786         m_hIcon = fi.hIcon;
2787         SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_ICON, STM_SETICON, (WPARAM)m_hIcon, 0);
2788     }
2789     else
2790         ERR("ExtractIconW failed %ls %u\n", m_sIcoPath, m_Header.nIconIndex);
2791 
2792     if (!SHGetFileInfoW(m_sLinkPath, 0, &fi, sizeof(fi), SHGFI_DISPLAYNAME))
2793         fi.szDisplayName[0] = UNICODE_NULL;
2794     SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TEXT, fi.szDisplayName);
2795 
2796     /* Target type */
2797     if (m_sPath)
2798     {
2799         WCHAR buf[MAX_PATH];
2800         GetTypeDescriptionByPath(m_sPath, m_Header.dwFileAttributes, buf, _countof(buf));
2801         SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TYPE_EDIT, buf);
2802     }
2803 
2804     /* Target location */
2805     if (m_sPath)
2806     {
2807         WCHAR target[MAX_PATH];
2808         StringCchCopyW(target, _countof(target), m_sPath);
2809         PathRemoveFileSpecW(target);
2810         SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_LOCATION_EDIT, PathFindFileNameW(target));
2811     }
2812 
2813     /* Target path */
2814     if (m_sPath)
2815     {
2816         WCHAR newpath[MAX_PATH * 2];
2817         newpath[0] = UNICODE_NULL;
2818         if (wcschr(m_sPath, ' '))
2819             StringCchPrintfExW(newpath, _countof(newpath), NULL, NULL, 0, L"\"%ls\"", m_sPath);
2820         else
2821             StringCchCopyExW(newpath, _countof(newpath), m_sPath, NULL, NULL, 0);
2822 
2823         if (m_sArgs && m_sArgs[0])
2824         {
2825             StringCchCatW(newpath, _countof(newpath), L" ");
2826             StringCchCatW(newpath, _countof(newpath), m_sArgs);
2827         }
2828         SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TARGET_TEXT, newpath);
2829     }
2830 
2831     /* Working dir */
2832     if (m_sWorkDir)
2833         SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_START_IN_EDIT, m_sWorkDir);
2834 
2835     /* Description */
2836     if (m_sDescription)
2837         SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_COMMENT_EDIT, m_sDescription);
2838 
2839     /* Hot key */
2840     SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_KEY_HOTKEY, HKM_SETHOTKEY, m_Header.wHotKey, 0);
2841 
2842     /* Run */
2843     const WORD runstrings[] = { IDS_SHORTCUT_RUN_NORMAL, IDS_SHORTCUT_RUN_MIN, IDS_SHORTCUT_RUN_MAX };
2844     const DWORD runshowcmd[] = { SW_SHOWNORMAL, SW_SHOWMINNOACTIVE, SW_SHOWMAXIMIZED };
2845     HWND hRunCombo = GetDlgItem(hwndDlg, IDC_SHORTCUT_RUN_COMBO);
2846     for (UINT i = 0; i < _countof(runstrings); ++i)
2847     {
2848         WCHAR buf[MAX_PATH];
2849         if (!LoadStringW(shell32_hInstance, runstrings[i], buf, _countof(buf)))
2850             break;
2851 
2852         int index = SendMessageW(hRunCombo, CB_ADDSTRING, 0, (LPARAM)buf);
2853         if (index < 0)
2854             continue;
2855         SendMessageW(hRunCombo, CB_SETITEMDATA, index, runshowcmd[i]);
2856         if (!i || m_Header.nShowCommand == runshowcmd[i])
2857             SendMessageW(hRunCombo, CB_SETCURSEL, index, 0);
2858     }
2859 
2860     BOOL disablecontrols = FALSE;
2861     if (darwin)
2862     {
2863         disablecontrols = TRUE;
2864         EnableWindow(GetDlgItem(hwndDlg, IDC_SHORTCUT_FIND), FALSE);
2865         EnableWindow(GetDlgItem(hwndDlg, IDC_SHORTCUT_CHANGE_ICON), FALSE);
2866     }
2867     else
2868     {
2869         WCHAR path[MAX_PATH * 2];
2870         path[0] = UNICODE_NULL;
2871         HRESULT hr = GetPath(path, _countof(path), NULL, SLGP_RAWPATH);
2872         if (FAILED(hr))
2873             hr = GetPath(path, _countof(path), NULL, 0);
2874 #if DBG
2875         if (GetKeyState(VK_CONTROL) < 0) // Allow inspection of PIDL parsing path
2876         {
2877             hr = S_OK;
2878             path[0] = UNICODE_NULL;
2879         }
2880 #endif
2881         if (hr != S_OK)
2882         {
2883             disablecontrols = TRUE;
2884             LPITEMIDLIST pidl;
2885             if (GetIDList(&pidl) == S_OK)
2886             {
2887                 path[0] = UNICODE_NULL;
2888                 SHGetNameAndFlagsW(pidl, SHGDN_FORADDRESSBAR | SHGDN_FORPARSING, path, _countof(path), NULL);
2889                 SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TYPE_EDIT, path);
2890                 SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TARGET_TEXT, path);
2891                 ILRemoveLastID(pidl);
2892                 path[0] = UNICODE_NULL;
2893                 SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_START_IN_EDIT, path);
2894                 SHGetNameAndFlagsW(pidl, SHGDN_NORMAL, path, _countof(path), NULL);
2895                 SetDlgItemTextW(hwndDlg, IDC_SHORTCUT_LOCATION_EDIT, path);
2896                 ILFree(pidl);
2897             }
2898             EnableWindow(GetDlgItem(hwndDlg, IDC_SHORTCUT_ADVANCED), FALSE);
2899             EnableWindow(GetDlgItem(hwndDlg, IDC_SHORTCUT_START_IN_EDIT), FALSE);
2900         }
2901         else
2902         {
2903             ASSERT(FAILED(hr) || !(path[0] == ':' && path[1] == ':' && path[2] == '{'));
2904         }
2905     }
2906 
2907     HWND hWndTarget = GetDlgItem(hwndDlg, IDC_SHORTCUT_TARGET_TEXT);
2908     EnableWindow(hWndTarget, !disablecontrols);
2909     PostMessage(hWndTarget, EM_SETSEL, 0, -1); // Fix caret bug when first opening the tab
2910 
2911     /* auto-completion */
2912     SHAutoComplete(hWndTarget, SHACF_DEFAULT);
2913     SHAutoComplete(GetDlgItem(hwndDlg, IDC_SHORTCUT_START_IN_EDIT), SHACF_DEFAULT);
2914 
2915     m_bInInit = FALSE;
2916 
2917     return TRUE;
2918 }
2919 
2920 void CShellLink::OnCommand(HWND hwndDlg, int id, HWND hwndCtl, UINT codeNotify)
2921 {
2922     switch (id)
2923     {
2924         case IDC_SHORTCUT_FIND:
2925             SHOpenFolderAndSelectItems(m_pPidl, 0, NULL, 0);
2926             return;
2927 
2928         case IDC_SHORTCUT_CHANGE_ICON:
2929         {
2930             WCHAR wszPath[MAX_PATH] = L"";
2931 
2932             if (m_sIcoPath)
2933                 wcscpy(wszPath, m_sIcoPath);
2934             else
2935                 FindExecutableW(m_sPath, NULL, wszPath);
2936 
2937             INT IconIndex = m_Header.nIconIndex;
2938             if (PickIconDlg(hwndDlg, wszPath, _countof(wszPath), &IconIndex))
2939             {
2940                 SetIconLocation(wszPath, IconIndex);
2941                 PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
2942 
2943                 HICON hIconLarge = CreateShortcutIcon(wszPath, IconIndex);
2944                 if (hIconLarge)
2945                 {
2946                     if (m_hIcon)
2947                         DestroyIcon(m_hIcon);
2948                     m_hIcon = hIconLarge;
2949                     SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_ICON, STM_SETICON, (WPARAM)m_hIcon, 0);
2950                 }
2951             }
2952             return;
2953         }
2954 
2955         case IDC_SHORTCUT_ADVANCED:
2956         {
2957             INT_PTR result = DialogBoxParamW(shell32_hInstance, MAKEINTRESOURCEW(IDD_SHORTCUT_EXTENDED_PROPERTIES), hwndDlg, ExtendedShortcutProc, (LPARAM)m_bRunAs);
2958             if (result == 1 || result == 0)
2959             {
2960                 if (m_bRunAs != result)
2961                 {
2962                     PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
2963                 }
2964 
2965                 m_bRunAs = result;
2966             }
2967             return;
2968         }
2969     }
2970     if (codeNotify == EN_CHANGE || codeNotify == CBN_SELCHANGE)
2971     {
2972         if (!m_bInInit)
2973             PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
2974     }
2975 }
2976 
2977 LRESULT CShellLink::OnNotify(HWND hwndDlg, int idFrom, LPNMHDR pnmhdr)
2978 {
2979     WCHAR wszBuf[MAX_PATH];
2980     LPPSHNOTIFY lppsn = (LPPSHNOTIFY)pnmhdr;
2981 
2982     if (lppsn->hdr.code == PSN_APPLY)
2983     {
2984         /* set working directory */
2985         GetDlgItemTextW(hwndDlg, IDC_SHORTCUT_START_IN_EDIT, wszBuf, _countof(wszBuf));
2986         SetWorkingDirectory(wszBuf);
2987 
2988         /* set link destination */
2989         GetDlgItemTextW(hwndDlg, IDC_SHORTCUT_TARGET_TEXT, wszBuf, _countof(wszBuf));
2990         LPWSTR lpszArgs = NULL;
2991         LPWSTR unquoted = strdupW(wszBuf);
2992         StrTrimW(unquoted, L" ");
2993 
2994         if (!PathFileExistsW(unquoted))
2995         {
2996             lpszArgs = PathGetArgsW(unquoted);
2997             PathRemoveArgsW(unquoted);
2998             StrTrimW(lpszArgs, L" ");
2999         }
3000         if (unquoted[0] == '"' && unquoted[wcslen(unquoted) - 1] == '"')
3001             PathUnquoteSpacesW(unquoted);
3002 
3003         WCHAR *pwszExt = PathFindExtensionW(unquoted);
3004         if (!_wcsicmp(pwszExt, L".lnk"))
3005         {
3006             // FIXME load localized error msg
3007             MessageBoxW(hwndDlg, L"You cannot create a link to a shortcut", L"Error", MB_ICONERROR);
3008             SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
3009             return TRUE;
3010         }
3011 
3012         if (!PathFileExistsW(unquoted))
3013         {
3014             // FIXME load localized error msg
3015             MessageBoxW(hwndDlg, L"The specified file name in the target box is invalid", L"Error", MB_ICONERROR);
3016             SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
3017             return TRUE;
3018         }
3019 
3020         SetPath(unquoted);
3021         if (lpszArgs)
3022             SetArguments(lpszArgs);
3023         else
3024             SetArguments(L"\0");
3025 
3026         HeapFree(GetProcessHeap(), 0, unquoted);
3027 
3028         m_Header.wHotKey = (WORD)SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_KEY_HOTKEY, HKM_GETHOTKEY, 0, 0);
3029 
3030         int index = (int)SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_RUN_COMBO, CB_GETCURSEL, 0, 0);
3031         if (index != CB_ERR)
3032         {
3033             m_Header.nShowCommand = (UINT)SendDlgItemMessageW(hwndDlg, IDC_SHORTCUT_RUN_COMBO, CB_GETITEMDATA, index, 0);
3034         }
3035 
3036         TRACE("This %p m_sLinkPath %S\n", this, m_sLinkPath);
3037         Save(m_sLinkPath, TRUE);
3038         SHChangeNotify(SHCNE_UPDATEITEM, SHCNF_PATHW, m_sLinkPath, NULL);
3039         SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
3040         return TRUE;
3041     }
3042     return FALSE;
3043 }
3044 
3045 void CShellLink::OnDestroy(HWND hwndDlg)
3046 {
3047     if (m_hIcon)
3048     {
3049         DestroyIcon(m_hIcon);
3050         m_hIcon = NULL;
3051     }
3052 }
3053 
3054 /**************************************************************************
3055  * SH_ShellLinkDlgProc
3056  *
3057  * dialog proc of the shortcut property dialog
3058  */
3059 
3060 INT_PTR CALLBACK
3061 CShellLink::SH_ShellLinkDlgProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
3062 {
3063     LPPROPSHEETPAGEW ppsp;
3064     CShellLink *pThis = reinterpret_cast<CShellLink *>(GetWindowLongPtr(hwndDlg, DWLP_USER));
3065 
3066     switch (uMsg)
3067     {
3068         case WM_INITDIALOG:
3069             ppsp = (LPPROPSHEETPAGEW)lParam;
3070             if (ppsp == NULL)
3071                 break;
3072 
3073             pThis = reinterpret_cast<CShellLink *>(ppsp->lParam);
3074             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pThis);
3075             return pThis->OnInitDialog(hwndDlg, (HWND)(wParam), lParam);
3076 
3077         case WM_NOTIFY:
3078             return pThis->OnNotify(hwndDlg, (int)wParam, (NMHDR *)lParam);
3079 
3080         case WM_COMMAND:
3081             pThis->OnCommand(hwndDlg, LOWORD(wParam), (HWND)lParam, HIWORD(wParam));
3082             break;
3083 
3084         case WM_DESTROY:
3085             pThis->OnDestroy(hwndDlg);
3086             break;
3087 
3088         default:
3089             break;
3090     }
3091 
3092     return FALSE;
3093 }
3094 
3095 /**************************************************************************
3096  * ShellLink_IShellPropSheetExt interface
3097  */
3098 
3099 HRESULT STDMETHODCALLTYPE CShellLink::AddPages(LPFNADDPROPSHEETPAGE pfnAddPage, LPARAM lParam)
3100 {
3101     HPROPSHEETPAGE hPage = SH_CreatePropertySheetPageEx(IDD_SHORTCUT_PROPERTIES, SH_ShellLinkDlgProc,
3102                                                         (LPARAM)this, NULL, &PropSheetPageLifetimeCallback<CShellLink>);
3103     HRESULT hr = AddPropSheetPage(hPage, pfnAddPage, lParam);
3104     if (FAILED_UNEXPECTEDLY(hr))
3105         return hr;
3106     else
3107         AddRef(); // For PropSheetPageLifetimeCallback
3108     enum { CShellLink_PageIndex_Shortcut = 0 };
3109     return 1 + CShellLink_PageIndex_Shortcut; // Make this page the default (one-based)
3110 }
3111 
3112 HRESULT STDMETHODCALLTYPE CShellLink::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE pfnReplacePage, LPARAM lParam)
3113 {
3114     TRACE("(%p) (uPageID %u, pfnReplacePage %p lParam %p\n", this, uPageID, pfnReplacePage, lParam);
3115     return E_NOTIMPL;
3116 }
3117 
3118 HRESULT STDMETHODCALLTYPE CShellLink::SetSite(IUnknown *punk)
3119 {
3120     TRACE("%p %p\n", this, punk);
3121 
3122     m_site = punk;
3123 
3124     return S_OK;
3125 }
3126 
3127 HRESULT STDMETHODCALLTYPE CShellLink::GetSite(REFIID iid, void ** ppvSite)
3128 {
3129     TRACE("%p %s %p\n", this, debugstr_guid(&iid), ppvSite);
3130 
3131     if (m_site == NULL)
3132         return E_FAIL;
3133 
3134     return m_site->QueryInterface(iid, ppvSite);
3135 }
3136 
3137 HRESULT STDMETHODCALLTYPE CShellLink::DragEnter(IDataObject *pDataObject,
3138     DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
3139 {
3140     TRACE("(%p)->(DataObject=%p)\n", this, pDataObject);
3141 
3142     if (*pdwEffect == DROPEFFECT_NONE)
3143         return S_OK;
3144 
3145     LPCITEMIDLIST pidlLast;
3146     CComPtr<IShellFolder> psf;
3147 
3148     HRESULT hr = SHBindToParent(m_pPidl, IID_PPV_ARG(IShellFolder, &psf), &pidlLast);
3149 
3150     if (SUCCEEDED(hr))
3151     {
3152         hr = psf->GetUIObjectOf(0, 1, &pidlLast, IID_NULL_PPV_ARG(IDropTarget, &m_DropTarget));
3153 
3154         if (SUCCEEDED(hr))
3155             hr = m_DropTarget->DragEnter(pDataObject, dwKeyState, pt, pdwEffect);
3156         else
3157             *pdwEffect = DROPEFFECT_NONE;
3158     }
3159     else
3160         *pdwEffect = DROPEFFECT_NONE;
3161 
3162     return S_OK;
3163 }
3164 
3165 HRESULT STDMETHODCALLTYPE CShellLink::DragOver(DWORD dwKeyState, POINTL pt,
3166     DWORD *pdwEffect)
3167 {
3168     TRACE("(%p)\n", this);
3169     HRESULT hr = S_OK;
3170     if (m_DropTarget)
3171         hr = m_DropTarget->DragOver(dwKeyState, pt, pdwEffect);
3172     return hr;
3173 }
3174 
3175 HRESULT STDMETHODCALLTYPE CShellLink::DragLeave()
3176 {
3177     TRACE("(%p)\n", this);
3178     HRESULT hr = S_OK;
3179     if (m_DropTarget)
3180     {
3181         hr = m_DropTarget->DragLeave();
3182         m_DropTarget.Release();
3183     }
3184 
3185     return hr;
3186 }
3187 
3188 HRESULT STDMETHODCALLTYPE CShellLink::Drop(IDataObject *pDataObject,
3189     DWORD dwKeyState, POINTL pt, DWORD *pdwEffect)
3190 {
3191     TRACE("(%p)\n", this);
3192     HRESULT hr = S_OK;
3193     if (m_DropTarget)
3194         hr = m_DropTarget->Drop(pDataObject, dwKeyState, pt, pdwEffect);
3195 
3196     return hr;
3197 }
3198 
3199 /**************************************************************************
3200  *      IShellLink_ConstructFromFile
3201  */
3202 HRESULT WINAPI IShellLink_ConstructFromPath(WCHAR *path, REFIID riid, LPVOID *ppv)
3203 {
3204     CComPtr<IPersistFile> ppf;
3205     HRESULT hr = CShellLink::_CreatorClass::CreateInstance(NULL, IID_PPV_ARG(IPersistFile, &ppf));
3206     if (FAILED(hr))
3207         return hr;
3208 
3209     hr = ppf->Load(path, 0);
3210     if (FAILED(hr))
3211         return hr;
3212 
3213     return ppf->QueryInterface(riid, ppv);
3214 }
3215 
3216 HRESULT WINAPI IShellLink_ConstructFromFile(IShellFolder * psf, LPCITEMIDLIST pidl, REFIID riid, LPVOID *ppv)
3217 {
3218     WCHAR path[MAX_PATH];
3219     if (!ILGetDisplayNameExW(psf, pidl, path, 0))
3220         return E_FAIL;
3221 
3222     return IShellLink_ConstructFromPath(path, riid, ppv);
3223 }
3224 
3225 HICON CShellLink::CreateShortcutIcon(LPCWSTR wszIconPath, INT IconIndex)
3226 {
3227     const INT cx = GetSystemMetrics(SM_CXICON), cy = GetSystemMetrics(SM_CYICON);
3228     const COLORREF crMask = GetSysColor(COLOR_3DFACE);
3229     WCHAR wszLnkIcon[MAX_PATH];
3230     int lnk_idx;
3231     HDC hDC;
3232     HIMAGELIST himl = ImageList_Create(cx, cy, ILC_COLOR32 | ILC_MASK, 1, 1);
3233     HICON hIcon = NULL, hNewIcon = NULL, hShortcut;
3234 
3235     if (HLM_GetIconW(IDI_SHELL_SHORTCUT - 1, wszLnkIcon, _countof(wszLnkIcon), &lnk_idx))
3236     {
3237         ::ExtractIconExW(wszLnkIcon, lnk_idx, &hShortcut, NULL, 1);
3238     }
3239     else
3240     {
3241         hShortcut = (HICON)LoadImageW(shell32_hInstance, MAKEINTRESOURCE(IDI_SHELL_SHORTCUT),
3242                                       IMAGE_ICON, cx, cy, 0);
3243     }
3244 
3245     ::ExtractIconExW(wszIconPath, IconIndex, &hIcon, NULL, 1);
3246     if (!hIcon || !hShortcut || !himl)
3247         goto cleanup;
3248 
3249     hDC = CreateCompatibleDC(NULL);
3250     if (hDC)
3251     {
3252         // create 32bpp bitmap
3253         BITMAPINFO bi;
3254         ZeroMemory(&bi, sizeof(bi));
3255         bi.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
3256         bi.bmiHeader.biWidth = cx;
3257         bi.bmiHeader.biHeight = cy;
3258         bi.bmiHeader.biPlanes = 1;
3259         bi.bmiHeader.biBitCount = 32;
3260         LPVOID pvBits;
3261         HBITMAP hbm = CreateDIBSection(hDC, &bi, DIB_RGB_COLORS, &pvBits, NULL, 0);
3262         if (hbm)
3263         {
3264             // draw the icon image
3265             HGDIOBJ hbmOld = SelectObject(hDC, hbm);
3266             {
3267                 HBRUSH hbr = CreateSolidBrush(crMask);
3268                 RECT rc = { 0, 0, cx, cy };
3269                 FillRect(hDC, &rc, hbr);
3270                 DeleteObject(hbr);
3271 
3272                 DrawIconEx(hDC, 0, 0, hIcon, cx, cy, 0, NULL, DI_NORMAL);
3273                 DrawIconEx(hDC, 0, 0, hShortcut, cx, cy, 0, NULL, DI_NORMAL);
3274             }
3275             SelectObject(hDC, hbmOld);
3276 
3277             INT iAdded = ImageList_AddMasked(himl, hbm, crMask);
3278             hNewIcon = ImageList_GetIcon(himl, iAdded, ILD_NORMAL | ILD_TRANSPARENT);
3279 
3280             DeleteObject(hbm);
3281         }
3282         DeleteDC(hDC);
3283     }
3284 
3285 cleanup:
3286     if (hIcon)
3287         DestroyIcon(hIcon);
3288     if (hShortcut)
3289         DestroyIcon(hShortcut);
3290     if (himl)
3291         ImageList_Destroy(himl);
3292 
3293     return hNewIcon;
3294 }
3295