1 /*
2  * Provides default file shell extension
3  *
4  * Copyright 2005 Johannes Anderwald
5  * Copyright 2012 Rafal Harabien
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
20  */
21 
22 #include "precomp.h"
23 
24 #define NTOS_MODE_USER
25 #include <ndk/iofuncs.h>
26 #include <ndk/obfuncs.h>
27 
28 WINE_DEFAULT_DEBUG_CHANNEL(shell);
29 
30 EXTERN_C BOOL PathIsExeW(LPCWSTR lpszPath);
31 
32 BOOL GetPhysicalFileSize(LPCWSTR PathBuffer, PULARGE_INTEGER Size)
33 {
34     UNICODE_STRING FileName;
35     OBJECT_ATTRIBUTES ObjectAttributes;
36     IO_STATUS_BLOCK IoStatusBlock;
37     HANDLE FileHandle;
38     FILE_STANDARD_INFORMATION FileInfo;
39     NTSTATUS Status;
40 
41     if (!RtlDosPathNameToNtPathName_U(PathBuffer, &FileName, NULL, NULL))
42     {
43         ERR("RtlDosPathNameToNtPathName_U failed\n");
44         return FALSE;
45     }
46 
47     InitializeObjectAttributes(&ObjectAttributes,
48                                &FileName,
49                                OBJ_CASE_INSENSITIVE,
50                                NULL,
51                                NULL);
52     Status = NtOpenFile(&FileHandle,
53                         FILE_READ_ATTRIBUTES | SYNCHRONIZE,
54                         &ObjectAttributes,
55                         &IoStatusBlock,
56                         FILE_SHARE_READ,
57                         FILE_SYNCHRONOUS_IO_NONALERT);
58     RtlFreeUnicodeString(&FileName);
59     if (!NT_SUCCESS(Status))
60     {
61         ERR("NtOpenFile failed for %S (Status 0x%08lx)\n", PathBuffer, Status);
62         return FALSE;
63     }
64 
65     /* Query the file size */
66     Status = NtQueryInformationFile(FileHandle,
67                                     &IoStatusBlock,
68                                     &FileInfo,
69                                     sizeof(FileInfo),
70                                     FileStandardInformation);
71     NtClose(FileHandle);
72     if (!NT_SUCCESS(Status))
73     {
74         ERR("NtQueryInformationFile failed for %S (Status: %08lX)\n", PathBuffer, Status);
75         return FALSE;
76     }
77 
78     Size->QuadPart = FileInfo.AllocationSize.QuadPart;
79     return TRUE;
80 }
81 
82 BOOL CFileVersionInfo::Load(LPCWSTR pwszPath)
83 {
84     ULONG cbInfo = GetFileVersionInfoSizeW(pwszPath, NULL);
85     if (!cbInfo)
86     {
87         WARN("GetFileVersionInfoSize %ls failed\n", pwszPath);
88         return FALSE;
89     }
90 
91     m_pInfo = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, cbInfo);
92     if (!m_pInfo)
93     {
94         ERR("HeapAlloc failed bytes %x\n", cbInfo);
95         return FALSE;
96     }
97 
98     if (!GetFileVersionInfoW(pwszPath, 0, cbInfo, m_pInfo))
99     {
100         ERR("GetFileVersionInfoW failed\n");
101         return FALSE;
102     }
103 
104     LPLANGANDCODEPAGE lpLangCode;
105     UINT cBytes;
106     if (!VerQueryValueW(m_pInfo, L"\\VarFileInfo\\Translation", (LPVOID *)&lpLangCode, &cBytes) || cBytes < sizeof(LANGANDCODEPAGE))
107     {
108         ERR("VerQueryValueW failed\n");
109         return FALSE;
110     }
111 
112     /* FIXME: find language from current locale / if not available,
113      * default to english
114      * for now default to first available language
115      */
116     m_wLang = lpLangCode->wLang;
117     m_wCode = lpLangCode->wCode;
118     TRACE("Lang %hx Code %hu\n", m_wLang, m_wCode);
119 
120     return TRUE;
121 }
122 
123 LPCWSTR CFileVersionInfo::GetString(LPCWSTR pwszName)
124 {
125     if (!m_pInfo)
126         return NULL;
127 
128     WCHAR wszBuf[256];
129     swprintf(wszBuf, L"\\StringFileInfo\\%04x%04x\\%s", m_wLang, m_wCode, pwszName);
130 
131     /* Query string in version block */
132     LPCWSTR pwszResult = NULL;
133     UINT cBytes = 0;
134     if (!VerQueryValueW(m_pInfo, wszBuf, (LPVOID *)&pwszResult, &cBytes))
135         pwszResult = NULL;
136 
137     if (!pwszResult)
138     {
139         /* Try US English */
140         swprintf(wszBuf, L"\\StringFileInfo\\%04x%04x\\%s", MAKELANGID(LANG_ENGLISH, SUBLANG_ENGLISH_US), 1252, pwszName);
141         if (!VerQueryValueW(m_pInfo, wszBuf, (LPVOID *)&pwszResult, &cBytes))
142             pwszResult = NULL;
143     }
144 
145     if (!pwszResult)
146         ERR("VerQueryValueW %ls failed\n", pwszName);
147     else
148         TRACE("%ls: %ls\n", pwszName, pwszResult);
149 
150     return pwszResult;
151 }
152 
153 VS_FIXEDFILEINFO *CFileVersionInfo::GetFixedInfo()
154 {
155     if (!m_pInfo)
156         return NULL;
157 
158     VS_FIXEDFILEINFO *pInfo;
159     UINT cBytes;
160     if (!VerQueryValueW(m_pInfo, L"\\", (PVOID*)&pInfo, &cBytes))
161         return NULL;
162     return pInfo;
163 }
164 
165 LPCWSTR CFileVersionInfo::GetLangName()
166 {
167     if (!m_pInfo)
168         return NULL;
169 
170     if (!m_wszLang[0])
171     {
172         if (!VerLanguageNameW(m_wLang, m_wszLang, _countof(m_wszLang)))
173             ERR("VerLanguageNameW failed\n");
174     }
175 
176     return m_wszLang;
177 }
178 
179 UINT
180 SH_FormatInteger(LONGLONG Num, LPWSTR pwszResult, UINT cchResultMax)
181 {
182     // Print the number in uniform mode
183     WCHAR wszNumber[24];
184     swprintf(wszNumber, L"%I64u", Num);
185 
186     // Get system strings for decimal and thousand separators.
187     WCHAR wszDecimalSep[8], wszThousandSep[8];
188     GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_SDECIMAL, wszDecimalSep, _countof(wszDecimalSep));
189     GetLocaleInfoW(LOCALE_USER_DEFAULT, LOCALE_STHOUSAND, wszThousandSep, _countof(wszThousandSep));
190 
191     // Initialize format for printing the number in bytes
192     NUMBERFMTW nf;
193     ZeroMemory(&nf, sizeof(nf));
194     nf.lpDecimalSep = wszDecimalSep;
195     nf.lpThousandSep = wszThousandSep;
196 
197     // Get system string for groups separator
198     WCHAR wszGrouping[12];
199     INT cchGrouping = GetLocaleInfoW(LOCALE_USER_DEFAULT,
200                                      LOCALE_SGROUPING,
201                                      wszGrouping,
202                                      _countof(wszGrouping));
203 
204     // Convert grouping specs from string to integer
205     for (INT i = 0; i < cchGrouping; i++)
206     {
207         WCHAR wch = wszGrouping[i];
208 
209         if (wch >= L'0' && wch <= L'9')
210             nf.Grouping = nf.Grouping * 10 + (wch - L'0');
211         else if (wch != L';')
212             break;
213     }
214 
215     if ((nf.Grouping % 10) == 0)
216         nf.Grouping /= 10;
217     else
218         nf.Grouping *= 10;
219 
220     // Format the number
221     INT cchResult = GetNumberFormatW(LOCALE_USER_DEFAULT,
222                                     0,
223                                     wszNumber,
224                                     &nf,
225                                     pwszResult,
226                                     cchResultMax);
227 
228     if (!cchResult)
229         return 0;
230 
231     // GetNumberFormatW returns number of characters including UNICODE_NULL
232     return cchResult - 1;
233 }
234 
235 UINT
236 SH_FormatByteSize(LONGLONG cbSize, LPWSTR pwszResult, UINT cchResultMax)
237 {
238     /* Write formated bytes count */
239     INT cchWritten = SH_FormatInteger(cbSize, pwszResult, cchResultMax);
240     if (!cchWritten)
241         return 0;
242 
243     /* Copy " bytes" to buffer */
244     LPWSTR pwszEnd = pwszResult + cchWritten;
245     size_t cchRemaining = cchResultMax - cchWritten;
246     StringCchCopyExW(pwszEnd, cchRemaining, L" ", &pwszEnd, &cchRemaining, 0);
247     cchWritten = LoadStringW(shell32_hInstance, IDS_BYTES_FORMAT, pwszEnd, cchRemaining);
248     cchRemaining -= cchWritten;
249 
250     return cchResultMax - cchRemaining;
251 }
252 
253 /*************************************************************************
254  *
255  * SH_FormatFileSizeWithBytes
256  *
257  * Format a size in bytes to string.
258  *
259  * lpQwSize = Pointer to 64bit large integer to format
260  * pszBuf   = Buffer to fill with output string
261  * cchBuf   = size of pszBuf in characters
262  *
263  */
264 
265 LPWSTR
266 SH_FormatFileSizeWithBytes(const PULARGE_INTEGER lpQwSize, LPWSTR pwszResult, UINT cchResultMax)
267 {
268     /* Format bytes in KBs, MBs etc */
269     if (StrFormatByteSizeW(lpQwSize->QuadPart, pwszResult, cchResultMax) == NULL)
270         return NULL;
271 
272     /* If there is less bytes than 1KB, we have nothing to do */
273     if (lpQwSize->QuadPart < 1024)
274         return pwszResult;
275 
276     /* Concatenate " (" */
277     UINT cchWritten = wcslen(pwszResult);
278     LPWSTR pwszEnd = pwszResult + cchWritten;
279     size_t cchRemaining = cchResultMax - cchWritten;
280     StringCchCopyExW(pwszEnd, cchRemaining, L" (", &pwszEnd, &cchRemaining, 0);
281 
282     /* Write formated bytes count */
283     cchWritten = SH_FormatByteSize(lpQwSize->QuadPart, pwszEnd, cchRemaining);
284     pwszEnd += cchWritten;
285     cchRemaining -= cchWritten;
286 
287     /* Copy ")" to the buffer */
288     StringCchCopyW(pwszEnd, cchRemaining, L")");
289 
290     return pwszResult;
291 }
292 
293 /*************************************************************************
294  *
295  * SH_CreatePropertySheetPage [Internal]
296  *
297  * creates a property sheet page from a resource id
298  *
299  */
300 
301 HPROPSHEETPAGE
302 SH_CreatePropertySheetPage(WORD wDialogId, DLGPROC pfnDlgProc, LPARAM lParam, LPCWSTR pwszTitle)
303 {
304     PROPSHEETPAGEW Page;
305 
306     memset(&Page, 0x0, sizeof(PROPSHEETPAGEW));
307     Page.dwSize = sizeof(PROPSHEETPAGEW);
308     Page.dwFlags = PSP_DEFAULT;
309     Page.hInstance = shell32_hInstance;
310     Page.pszTemplate = MAKEINTRESOURCE(wDialogId);
311     Page.pfnDlgProc = pfnDlgProc;
312     Page.lParam = lParam;
313     Page.pszTitle = pwszTitle;
314 
315     if (pwszTitle)
316         Page.dwFlags |= PSP_USETITLE;
317 
318     return CreatePropertySheetPageW(&Page);
319 }
320 
321 VOID
322 CFileDefExt::InitOpensWithField(HWND hwndDlg)
323 {
324     WCHAR wszBuf[MAX_PATH] = L"";
325     WCHAR wszPath[MAX_PATH] = L"";
326     DWORD dwSize = sizeof(wszBuf);
327     BOOL bUnknownApp = TRUE;
328     LPCWSTR pwszExt = PathFindExtensionW(m_wszPath);
329 
330     if (RegGetValueW(HKEY_CLASSES_ROOT, pwszExt, L"", RRF_RT_REG_SZ, NULL, wszBuf, &dwSize) == ERROR_SUCCESS)
331     {
332         bUnknownApp = FALSE;
333         StringCbCatW(wszBuf, sizeof(wszBuf), L"\\shell\\open\\command");
334         dwSize = sizeof(wszPath);
335         if (RegGetValueW(HKEY_CLASSES_ROOT, wszBuf, L"", RRF_RT_REG_SZ, NULL, wszPath, &dwSize) == ERROR_SUCCESS)
336         {
337             /* Get path from command line */
338             ExpandEnvironmentStringsW(wszPath, wszBuf, _countof(wszBuf));
339             PathRemoveArgs(wszBuf);
340             PathUnquoteSpacesW(wszBuf);
341             PathSearchAndQualify(wszBuf, wszPath, _countof(wszPath));
342 
343             HICON hIcon;
344             if (ExtractIconExW(wszPath, 0, NULL, &hIcon, 1))
345             {
346                 HWND hIconCtrl = GetDlgItem(hwndDlg, 14025);
347                 HWND hDescrCtrl = GetDlgItem(hwndDlg, 14007);
348                 ShowWindow(hIconCtrl, SW_SHOW);
349                 RECT rcIcon, rcDescr;
350                 GetWindowRect(hIconCtrl, &rcIcon);
351 
352                 rcIcon.right = rcIcon.left + GetSystemMetrics(SM_CXSMICON);
353                 rcIcon.bottom = rcIcon.top + GetSystemMetrics(SM_CYSMICON);
354 
355                 MapWindowPoints(NULL, hwndDlg, (LPPOINT)&rcIcon, 2);
356                 GetWindowRect(hDescrCtrl, &rcDescr);
357                 MapWindowPoints(NULL, hwndDlg, (LPPOINT)&rcDescr, 2);
358                 INT cxOffset = rcIcon.right + 2 - rcDescr.left;
359                 SetWindowPos(hDescrCtrl, NULL,
360                              rcDescr.left + cxOffset, rcDescr.top,
361                              rcDescr.right - rcDescr.left - cxOffset, rcDescr.bottom - rcDescr.top,
362                              SWP_NOZORDER);
363                 SendMessageW(hIconCtrl, STM_SETICON, (WPARAM)hIcon, 0);
364             } else
365                 ERR("Failed to extract icon\n");
366 
367             if (PathFileExistsW(wszPath))
368             {
369                 /* Get file description */
370                 CFileVersionInfo VerInfo;
371                 VerInfo.Load(wszPath);
372                 LPCWSTR pwszDescr = VerInfo.GetString(L"FileDescription");
373                 if (pwszDescr)
374                     SetDlgItemTextW(hwndDlg, 14007, pwszDescr);
375                 else
376                 {
377                     /* File has no description - display filename */
378                     LPWSTR pwszFilename = PathFindFileNameW(wszPath);
379                     PathRemoveExtension(pwszFilename);
380                     pwszFilename[0] = towupper(pwszFilename[0]);
381                     SetDlgItemTextW(hwndDlg, 14007, pwszFilename);
382                 }
383             }
384             else
385                 bUnknownApp = TRUE;
386         } else
387             WARN("RegGetValueW %ls failed\n", wszBuf);
388     } else
389         WARN("RegGetValueW %ls failed\n", pwszExt);
390 
391     if (bUnknownApp)
392     {
393         /* Unknown application */
394         LoadStringW(shell32_hInstance, IDS_UNKNOWN_APP, wszBuf, _countof(wszBuf));
395         SetDlgItemTextW(hwndDlg, 14007, wszBuf);
396     }
397 }
398 
399 /*************************************************************************
400  *
401  * SH_FileGeneralFileType [Internal]
402  *
403  * retrieves file extension description from registry and sets it in dialog
404  *
405  * TODO: retrieve file extension default icon and load it
406  *       find executable name from registry, retrieve description from executable
407  */
408 
409 BOOL
410 CFileDefExt::InitFileType(HWND hwndDlg)
411 {
412     TRACE("path %s\n", debugstr_w(m_wszPath));
413 
414     HWND hDlgCtrl = GetDlgItem(hwndDlg, 14005);
415     if (hDlgCtrl == NULL)
416         return FALSE;
417 
418     /* Get file information */
419     SHFILEINFOW fi;
420     if (!SHGetFileInfoW(m_wszPath, 0, &fi, sizeof(fi), SHGFI_TYPENAME|SHGFI_ICON))
421     {
422         ERR("SHGetFileInfoW failed for %ls (%lu)\n", m_wszPath, GetLastError());
423         fi.szTypeName[0] = L'\0';
424         fi.hIcon = NULL;
425     }
426 
427     LPCWSTR pwszExt = PathFindExtensionW(m_wszPath);
428     if (pwszExt[0])
429     {
430         WCHAR wszBuf[256];
431 
432         if (!fi.szTypeName[0])
433         {
434             /* The file type is unknown, so default to string "FileExtension File" */
435             size_t cchRemaining = 0;
436             LPWSTR pwszEnd = NULL;
437 
438             StringCchPrintfExW(wszBuf, _countof(wszBuf), &pwszEnd, &cchRemaining, 0, L"%s ", pwszExt + 1);
439             SendMessageW(hDlgCtrl, WM_GETTEXT, (WPARAM)cchRemaining, (LPARAM)pwszEnd);
440 
441             SendMessageW(hDlgCtrl, WM_SETTEXT, (WPARAM)NULL, (LPARAM)wszBuf);
442         }
443         else
444         {
445             /* Update file type */
446             StringCbPrintfW(wszBuf, sizeof(wszBuf), L"%s (%s)", fi.szTypeName, pwszExt);
447             SendMessageW(hDlgCtrl, WM_SETTEXT, (WPARAM)NULL, (LPARAM)wszBuf);
448         }
449     }
450 
451     /* Update file icon */
452     if (fi.hIcon)
453         SendDlgItemMessageW(hwndDlg, 14000, STM_SETICON, (WPARAM)fi.hIcon, 0);
454     else
455         ERR("No icon %ls\n", m_wszPath);
456 
457     return TRUE;
458 }
459 
460 /*************************************************************************
461  *
462  * CFileDefExt::InitFilePath [Internal]
463  *
464  * sets file path string and filename string
465  *
466  */
467 
468 BOOL
469 CFileDefExt::InitFilePath(HWND hwndDlg)
470 {
471     /* Find the filename */
472     WCHAR *pwszFilename = PathFindFileNameW(m_wszPath);
473 
474     if (pwszFilename > m_wszPath)
475     {
476         /* Location field */
477         WCHAR wszLocation[MAX_PATH];
478         StringCchCopyNW(wszLocation, _countof(wszLocation), m_wszPath, pwszFilename - m_wszPath);
479         PathRemoveBackslashW(wszLocation);
480 
481         SetDlgItemTextW(hwndDlg, 14009, wszLocation);
482     }
483 
484     /* Filename field */
485     SetDlgItemTextW(hwndDlg, 14001, pwszFilename);
486 
487     return TRUE;
488 }
489 
490 /*************************************************************************
491  *
492  * CFileDefExt::GetFileTimeString [Internal]
493  *
494  * formats a given LPFILETIME struct into readable user format
495  */
496 
497 BOOL
498 CFileDefExt::GetFileTimeString(LPFILETIME lpFileTime, LPWSTR pwszResult, UINT cchResult)
499 {
500     FILETIME ft;
501     SYSTEMTIME st;
502 
503     if (!FileTimeToLocalFileTime(lpFileTime, &ft) || !FileTimeToSystemTime(&ft, &st))
504         return FALSE;
505 
506     size_t cchRemaining = cchResult;
507     LPWSTR pwszEnd = pwszResult;
508     int cchWritten = GetDateFormatW(LOCALE_USER_DEFAULT, DATE_LONGDATE, &st, NULL, pwszEnd, cchRemaining);
509     if (cchWritten)
510         --cchWritten; // GetDateFormatW returns count with terminating zero
511     else
512         ERR("GetDateFormatW failed\n");
513     cchRemaining -= cchWritten;
514     pwszEnd += cchWritten;
515 
516     StringCchCopyExW(pwszEnd, cchRemaining, L", ", &pwszEnd, &cchRemaining, 0);
517 
518     cchWritten = GetTimeFormatW(LOCALE_USER_DEFAULT, 0, &st, NULL, pwszEnd, cchRemaining);
519     if (cchWritten)
520         --cchWritten; // GetTimeFormatW returns count with terminating zero
521     else
522         ERR("GetTimeFormatW failed\n");
523     TRACE("result %s\n", debugstr_w(pwszResult));
524     return TRUE;
525 }
526 
527 /*************************************************************************
528  *
529  * CFileDefExt::InitFileAttr [Internal]
530  *
531  * retrieves file information from file and sets in dialog
532  *
533  */
534 
535 BOOL
536 CFileDefExt::InitFileAttr(HWND hwndDlg)
537 {
538     BOOL Success;
539     WIN32_FIND_DATAW FileInfo; // WIN32_FILE_ATTRIBUTE_DATA
540     WCHAR wszBuf[MAX_PATH];
541 
542     TRACE("InitFileAttr %ls\n", m_wszPath);
543 
544     /*
545      * There are situations where GetFileAttributes(Ex) can fail even if the
546      * specified path represents a file. This happens when e.g. the file is a
547      * locked system file, such as C:\pagefile.sys . In this case, the function
548      * returns INVALID_FILE_ATTRIBUTES and GetLastError returns ERROR_SHARING_VIOLATION.
549      * (this would allow us to distinguish between this failure and a failure
550      * due to the fact that the path actually refers to a directory).
551      *
552      * Because we really want to retrieve the file attributes/size/date&time,
553      * we do the following trick:
554      * - First we call GetFileAttributesEx. If it succeeds we know we have
555      *   a file or a directory, and we have retrieved its attributes.
556      * - If GetFileAttributesEx fails, we call FindFirstFile on the full path.
557      *   While we could have called FindFirstFile at first and skip GetFileAttributesEx
558      *   altogether, we do it after GetFileAttributesEx because it performs more
559      *   work to retrieve the file attributes. However it actually works even
560      *   for locked system files.
561      * - If FindFirstFile succeeds we have retrieved its attributes.
562      * - Otherwise (FindFirstFile has failed), we do not retrieve anything.
563      *
564      * The following code also relies on the fact that the first 6 members
565      * of WIN32_FIND_DATA are *exactly* the same as the WIN32_FILE_ATTRIBUTE_DATA
566      * structure. Therefore it is safe to use a single WIN32_FIND_DATA
567      * structure for both the GetFileAttributesEx and FindFirstFile calls.
568      */
569 
570     Success = GetFileAttributesExW(m_wszPath,
571                                    GetFileExInfoStandard,
572                                    (LPWIN32_FILE_ATTRIBUTE_DATA)&FileInfo);
573     if (!Success)
574     {
575         HANDLE hFind = FindFirstFileW(m_wszPath, &FileInfo);
576         Success = (hFind != INVALID_HANDLE_VALUE);
577         if (Success)
578             FindClose(hFind);
579     }
580 
581     if (Success)
582     {
583         /* Update attribute checkboxes */
584         if (FileInfo.dwFileAttributes & FILE_ATTRIBUTE_READONLY)
585             SendDlgItemMessage(hwndDlg, 14021, BM_SETCHECK, BST_CHECKED, 0);
586         if (FileInfo.dwFileAttributes & FILE_ATTRIBUTE_HIDDEN)
587             SendDlgItemMessage(hwndDlg, 14022, BM_SETCHECK, BST_CHECKED, 0);
588         if (FileInfo.dwFileAttributes & FILE_ATTRIBUTE_ARCHIVE)
589             SendDlgItemMessage(hwndDlg, 14023, BM_SETCHECK, BST_CHECKED, 0);
590 
591         /* Update creation time */
592         if (GetFileTimeString(&FileInfo.ftCreationTime, wszBuf, _countof(wszBuf)))
593             SetDlgItemTextW(hwndDlg, 14015, wszBuf);
594 
595         /* For files display last access and last write time */
596         if (!m_bDir)
597         {
598             if (GetFileTimeString(&FileInfo.ftLastAccessTime, wszBuf, _countof(wszBuf)))
599                 SetDlgItemTextW(hwndDlg, 14019, wszBuf);
600 
601             if (GetFileTimeString(&FileInfo.ftLastWriteTime, wszBuf, _countof(wszBuf)))
602                 SetDlgItemTextW(hwndDlg, 14017, wszBuf);
603 
604             /* Update size of file */
605             ULARGE_INTEGER FileSize;
606             FileSize.u.LowPart = FileInfo.nFileSizeLow;
607             FileSize.u.HighPart = FileInfo.nFileSizeHigh;
608             if (SH_FormatFileSizeWithBytes(&FileSize, wszBuf, _countof(wszBuf)))
609             {
610                 SetDlgItemTextW(hwndDlg, 14011, wszBuf);
611 
612                 // Compute file on disk. If fails, use logical size
613                 if (GetPhysicalFileSize(m_wszPath, &FileSize))
614                     SH_FormatFileSizeWithBytes(&FileSize, wszBuf, _countof(wszBuf));
615                 else
616                     ERR("Unreliable size on disk\n");
617 
618                 SetDlgItemTextW(hwndDlg, 14012, wszBuf);
619             }
620         }
621     }
622 
623     if (m_bDir)
624     {
625         /* For directories files have to be counted */
626 
627         _CountFolderAndFilesData *data = static_cast<_CountFolderAndFilesData*>(HeapAlloc(GetProcessHeap(), 0, sizeof(_CountFolderAndFilesData)));
628         data->This = this;
629         data->pwszBuf = static_cast<LPWSTR>(HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY, sizeof(WCHAR) * MAX_PATH));
630         data->hwndDlg = hwndDlg;
631         this->AddRef();
632         StringCchCopyW(data->pwszBuf, MAX_PATH, m_wszPath);
633 
634         SHCreateThread(CFileDefExt::_CountFolderAndFilesThreadProc, data, NULL, NULL);
635 
636         /* Update size field */
637         if (SH_FormatFileSizeWithBytes(&m_DirSize, wszBuf, _countof(wszBuf)))
638             SetDlgItemTextW(hwndDlg, 14011, wszBuf);
639 
640         if (SH_FormatFileSizeWithBytes(&m_DirSizeOnDisc, wszBuf, _countof(wszBuf)))
641             SetDlgItemTextW(hwndDlg, 14012, wszBuf);
642 
643         /* Display files and folders count */
644         WCHAR wszFormat[256];
645         LoadStringW(shell32_hInstance, IDS_FILE_FOLDER, wszFormat, _countof(wszFormat));
646         StringCchPrintfW(wszBuf, _countof(wszBuf), wszFormat, m_cFiles, m_cFolders);
647         SetDlgItemTextW(hwndDlg, 14027, wszBuf);
648     }
649 
650     /* Hide Advanced button. TODO: Implement advanced dialog and enable this button if filesystem supports compression or encryption */
651     ShowWindow(GetDlgItem(hwndDlg, 14028), SW_HIDE);
652 
653     return TRUE;
654 }
655 
656 /*************************************************************************
657  *
658  * CFileDefExt::InitGeneralPage [Internal]
659  *
660  * sets all file general properties in dialog
661  */
662 
663 BOOL
664 CFileDefExt::InitGeneralPage(HWND hwndDlg)
665 {
666     /* Set general text properties filename filelocation and icon */
667     InitFilePath(hwndDlg);
668 
669     /* Set file type and icon */
670     InitFileType(hwndDlg);
671 
672     /* Set open with application */
673     if (!m_bDir)
674     {
675         if (!PathIsExeW(m_wszPath))
676             InitOpensWithField(hwndDlg);
677         else
678         {
679             WCHAR wszBuf[MAX_PATH];
680             LoadStringW(shell32_hInstance, IDS_EXE_DESCRIPTION, wszBuf, _countof(wszBuf));
681             SetDlgItemTextW(hwndDlg, 14006, wszBuf);
682             ShowWindow(GetDlgItem(hwndDlg, 14024), SW_HIDE);
683 
684             /* hidden button 14024 allows to draw edit 14007 larger than defined in resources , we use edit 14009 as idol */
685             RECT rectIdol, rectToAdjust;
686             GetClientRect(GetDlgItem(hwndDlg, 14009), &rectIdol);
687             GetClientRect(GetDlgItem(hwndDlg, 14007), &rectToAdjust);
688             SetWindowPos(GetDlgItem(hwndDlg, 14007), HWND_TOP, 0, 0,
689                 rectIdol.right-rectIdol.left /* make it as wide as its idol */,
690                 rectToAdjust.bottom-rectToAdjust.top /* but keep its current height */,
691                 SWP_NOMOVE | SWP_NOZORDER );
692 
693             LPCWSTR pwszDescr = m_VerInfo.GetString(L"FileDescription");
694             if (pwszDescr)
695                 SetDlgItemTextW(hwndDlg, 14007, pwszDescr);
696             else
697             {
698                 StringCbCopyW(wszBuf, sizeof(wszBuf), PathFindFileNameW(m_wszPath));
699                 PathRemoveExtension(wszBuf);
700                 SetDlgItemTextW(hwndDlg, 14007, wszBuf);
701             }
702         }
703     }
704 
705     /* Set file created/modfied/accessed time, size and attributes */
706     InitFileAttr(hwndDlg);
707 
708     return TRUE;
709 }
710 
711 /*************************************************************************
712  *
713  * CFileDefExt::GeneralPageProc
714  *
715  * wnd proc of 'General' property sheet page
716  *
717  */
718 
719 INT_PTR CALLBACK
720 CFileDefExt::GeneralPageProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
721 {
722     CFileDefExt *pFileDefExt = reinterpret_cast<CFileDefExt *>(GetWindowLongPtr(hwndDlg, DWLP_USER));
723     switch (uMsg)
724     {
725         case WM_INITDIALOG:
726         {
727             LPPROPSHEETPAGEW ppsp = (LPPROPSHEETPAGEW)lParam;
728 
729             if (ppsp == NULL || !ppsp->lParam)
730                 break;
731 
732             TRACE("WM_INITDIALOG hwnd %p lParam %p ppsplParam %S\n", hwndDlg, lParam, ppsp->lParam);
733 
734             pFileDefExt = reinterpret_cast<CFileDefExt *>(ppsp->lParam);
735             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)pFileDefExt);
736             pFileDefExt->InitGeneralPage(hwndDlg);
737             break;
738         }
739         case WM_COMMAND:
740             if (LOWORD(wParam) == 14024) /* Opens With - Change */
741             {
742                 OPENASINFO oainfo;
743                 oainfo.pcszFile = pFileDefExt->m_wszPath;
744                 oainfo.pcszClass = NULL;
745                 oainfo.oaifInFlags = OAIF_REGISTER_EXT|OAIF_FORCE_REGISTRATION;
746                 return SUCCEEDED(SHOpenWithDialog(hwndDlg, &oainfo));
747             }
748             else if (LOWORD(wParam) == 14021 || LOWORD(wParam) == 14022 || LOWORD(wParam) == 14023) /* checkboxes */
749                 PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
750             else if (LOWORD(wParam) == 14001) /* Name */
751             {
752                 if (HIWORD(wParam) == EN_CHANGE)
753                     PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
754             }
755             break;
756         case WM_NOTIFY:
757         {
758             LPPSHNOTIFY lppsn = (LPPSHNOTIFY)lParam;
759             if (lppsn->hdr.code == PSN_APPLY)
760             {
761                 /* Update attributes first */
762                 DWORD dwAttr = GetFileAttributesW(pFileDefExt->m_wszPath);
763                 if (dwAttr)
764                 {
765                     dwAttr &= ~(FILE_ATTRIBUTE_READONLY|FILE_ATTRIBUTE_HIDDEN|FILE_ATTRIBUTE_ARCHIVE);
766 
767                     if (BST_CHECKED == SendDlgItemMessageW(hwndDlg, 14021, BM_GETCHECK, 0, 0))
768                         dwAttr |= FILE_ATTRIBUTE_READONLY;
769                     if (BST_CHECKED == SendDlgItemMessageW(hwndDlg, 14022, BM_GETCHECK, 0, 0))
770                         dwAttr |= FILE_ATTRIBUTE_HIDDEN;
771                     if (BST_CHECKED == SendDlgItemMessageW(hwndDlg, 14023, BM_GETCHECK, 0, 0))
772                         dwAttr |= FILE_ATTRIBUTE_ARCHIVE;
773 
774                     if (!SetFileAttributesW(pFileDefExt->m_wszPath, dwAttr))
775                         ERR("SetFileAttributesW failed\n");
776                 }
777 
778                 /* Update filename now */
779                 WCHAR wszBuf[MAX_PATH];
780                 StringCchCopyW(wszBuf, _countof(wszBuf), pFileDefExt->m_wszPath);
781                 LPWSTR pwszFilename = PathFindFileNameW(wszBuf);
782                 UINT cchFilenameMax = _countof(wszBuf) - (pwszFilename - wszBuf);
783                 if (GetDlgItemTextW(hwndDlg, 14001, pwszFilename, cchFilenameMax))
784                 {
785                     if (!MoveFileW(pFileDefExt->m_wszPath, wszBuf))
786                         ERR("MoveFileW failed\n");
787                 }
788 
789                 SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
790                 return TRUE;
791             }
792             break;
793         }
794         case PSM_QUERYSIBLINGS:
795         {
796             // reset icon
797             HWND hIconCtrl = GetDlgItem(hwndDlg, 14025);
798             HICON hIcon = (HICON)SendMessageW(hIconCtrl, STM_GETICON, 0, 0);
799             DestroyIcon(hIcon);
800             hIcon = NULL;
801             SendMessageW(hIconCtrl, STM_SETICON, (WPARAM)hIcon, 0);
802 
803             // refresh the page
804             pFileDefExt->InitGeneralPage(hwndDlg);
805             return FALSE;   // continue
806         }
807         default:
808             break;
809     }
810 
811     return FALSE;
812 }
813 
814 /*************************************************************************
815  *
816  * CFileDefExt::InitVersionPage [Internal]
817  *
818  * sets all file version properties in dialog
819  */
820 
821 BOOL
822 CFileDefExt::InitVersionPage(HWND hwndDlg)
823 {
824     /* Get fixed info */
825     VS_FIXEDFILEINFO *pInfo = m_VerInfo.GetFixedInfo();
826     if (pInfo)
827     {
828         WCHAR wszVersion[256];
829         swprintf(wszVersion, L"%u.%u.%u.%u", HIWORD(pInfo->dwFileVersionMS),
830                  LOWORD(pInfo->dwFileVersionMS),
831                  HIWORD(pInfo->dwFileVersionLS),
832                  LOWORD(pInfo->dwFileVersionLS));
833         TRACE("MS %x LS %x ver %s \n", pInfo->dwFileVersionMS, pInfo->dwFileVersionLS, debugstr_w(wszVersion));
834         SetDlgItemTextW(hwndDlg, 14001, wszVersion);
835     }
836 
837     /* Update labels */
838     SetVersionLabel(hwndDlg, 14003, L"FileDescription");
839     SetVersionLabel(hwndDlg, 14005, L"LegalCopyright");
840 
841     /* Add items to listbox */
842     AddVersionString(hwndDlg, L"CompanyName");
843     LPCWSTR pwszLang = m_VerInfo.GetLangName();
844     if (pwszLang)
845     {
846         HWND hDlgCtrl = GetDlgItem(hwndDlg, 14009);
847         UINT Index = SendMessageW(hDlgCtrl, LB_ADDSTRING, (WPARAM)-1, (LPARAM)L"Language");
848         SendMessageW(hDlgCtrl, LB_SETITEMDATA, (WPARAM)Index, (LPARAM)(WCHAR *)pwszLang);
849     }
850     AddVersionString(hwndDlg, L"ProductName");
851     AddVersionString(hwndDlg, L"InternalName");
852     AddVersionString(hwndDlg, L"OriginalFilename");
853     AddVersionString(hwndDlg, L"FileVersion");
854     AddVersionString(hwndDlg, L"ProductVersion");
855     AddVersionString(hwndDlg, L"Comments");
856     AddVersionString(hwndDlg, L"LegalTrademarks");
857 
858     if (pInfo && (pInfo->dwFileFlags & VS_FF_PRIVATEBUILD))
859         AddVersionString(hwndDlg, L"PrivateBuild");
860 
861     if (pInfo && (pInfo->dwFileFlags & VS_FF_SPECIALBUILD))
862         AddVersionString(hwndDlg, L"SpecialBuild");
863 
864     /* Attach file version to dialog window */
865     SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)this);
866 
867     /* Select first item */
868     HWND hDlgCtrl = GetDlgItem(hwndDlg, 14009);
869     SendMessageW(hDlgCtrl, LB_SETCURSEL, 0, 0);
870     LPCWSTR pwszText = (LPCWSTR)SendMessageW(hDlgCtrl, LB_GETITEMDATA, (WPARAM)0, (LPARAM)NULL);
871     if (pwszText && pwszText != (LPCWSTR)LB_ERR)
872         SetDlgItemTextW(hwndDlg, 14010, pwszText);
873 
874     return TRUE;
875 }
876 
877 /*************************************************************************
878  *
879  * CFileDefExt::SetVersionLabel [Internal]
880  *
881  * retrieves a version string and uses it to set label text
882  */
883 
884 BOOL
885 CFileDefExt::SetVersionLabel(HWND hwndDlg, DWORD idCtrl, LPCWSTR pwszName)
886 {
887     if (hwndDlg == NULL || pwszName == NULL)
888         return FALSE;
889 
890     LPCWSTR pwszValue = m_VerInfo.GetString(pwszName);
891     if (pwszValue)
892     {
893         /* file description property */
894         TRACE("%s :: %s\n", debugstr_w(pwszName), debugstr_w(pwszValue));
895         SetDlgItemTextW(hwndDlg, idCtrl, pwszValue);
896         return TRUE;
897     }
898 
899     return FALSE;
900 }
901 
902 /*************************************************************************
903  *
904  * CFileDefExt::AddVersionString [Internal]
905  *
906  * retrieves a version string and adds it to listbox
907  */
908 
909 BOOL
910 CFileDefExt::AddVersionString(HWND hwndDlg, LPCWSTR pwszName)
911 {
912     TRACE("pwszName %s, hwndDlg %p\n", debugstr_w(pwszName), hwndDlg);
913 
914     if (hwndDlg == NULL || pwszName == NULL)
915         return FALSE;
916 
917     LPCWSTR pwszValue = m_VerInfo.GetString(pwszName);
918     if (pwszValue)
919     {
920         /* listbox name property */
921         HWND hDlgCtrl = GetDlgItem(hwndDlg, 14009);
922         TRACE("%s :: %s\n", debugstr_w(pwszName), debugstr_w(pwszValue));
923         UINT Index = SendMessageW(hDlgCtrl, LB_ADDSTRING, (WPARAM) -1, (LPARAM)pwszName);
924         SendMessageW(hDlgCtrl, LB_SETITEMDATA, (WPARAM)Index, (LPARAM)(WCHAR *)pwszValue);
925         return TRUE;
926     }
927 
928     return FALSE;
929 }
930 
931 /*************************************************************************
932  *
933  * CFileDefExt::VersionPageProc
934  *
935  * wnd proc of 'Version' property sheet page
936  */
937 
938 INT_PTR CALLBACK
939 CFileDefExt::VersionPageProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
940 {
941     switch (uMsg)
942     {
943         case WM_INITDIALOG:
944         {
945             LPPROPSHEETPAGE ppsp = (LPPROPSHEETPAGE)lParam;
946 
947             if (ppsp == NULL || !ppsp->lParam)
948                 break;
949 
950             TRACE("WM_INITDIALOG hwnd %p lParam %p ppsplParam %x\n", hwndDlg, lParam, ppsp->lParam);
951 
952             CFileDefExt *pFileDefExt = reinterpret_cast<CFileDefExt *>(ppsp->lParam);
953             return pFileDefExt->InitVersionPage(hwndDlg);
954         }
955         case WM_COMMAND:
956             if (LOWORD(wParam) == 14009 && HIWORD(wParam) == LBN_SELCHANGE)
957             {
958                 HWND hDlgCtrl = (HWND)lParam;
959 
960                 LRESULT Index = SendMessageW(hDlgCtrl, LB_GETCURSEL, (WPARAM)NULL, (LPARAM)NULL);
961                 if (Index == LB_ERR)
962                     break;
963 
964                 LPCWSTR pwszData = (LPCWSTR)SendMessageW(hDlgCtrl, LB_GETITEMDATA, (WPARAM)Index, (LPARAM)NULL);
965                 if (pwszData == NULL)
966                     break;
967 
968                 CString str(pwszData);
969                 str.Trim();
970 
971                 TRACE("hDlgCtrl %x string %s\n", hDlgCtrl, debugstr_w(str));
972                 SetDlgItemTextW(hwndDlg, 14010, str);
973 
974                 return TRUE;
975             }
976             break;
977         case WM_DESTROY:
978             break;
979         case PSM_QUERYSIBLINGS:
980             return FALSE;   // continue
981         default:
982             break;
983     }
984 
985     return FALSE;
986 }
987 
988 /*************************************************************************/
989 /* Folder Customize */
990 
991 static const WCHAR s_szShellClassInfo[] = L".ShellClassInfo";
992 static const WCHAR s_szIconIndex[] = L"IconIndex";
993 static const WCHAR s_szIconFile[] = L"IconFile";
994 static const WCHAR s_szIconResource[] = L"IconResource";
995 
996 // IDD_FOLDER_CUSTOMIZE
997 INT_PTR CALLBACK
998 CFileDefExt::FolderCustomizePageProc(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
999 {
1000     CFileDefExt *pFileDefExt = reinterpret_cast<CFileDefExt *>(GetWindowLongPtr(hwndDlg, DWLP_USER));
1001     switch (uMsg)
1002     {
1003         case WM_INITDIALOG:
1004         {
1005             LPPROPSHEETPAGE ppsp = (LPPROPSHEETPAGE)lParam;
1006 
1007             if (ppsp == NULL || !ppsp->lParam)
1008                 break;
1009 
1010             TRACE("WM_INITDIALOG hwnd %p lParam %p ppsplParam %x\n", hwndDlg, lParam, ppsp->lParam);
1011 
1012             pFileDefExt = reinterpret_cast<CFileDefExt *>(ppsp->lParam);
1013             return pFileDefExt->InitFolderCustomizePage(hwndDlg);
1014         }
1015 
1016         case WM_COMMAND:
1017             switch (LOWORD(wParam))
1018             {
1019                 case IDC_FOLDERCUST_CHANGE_ICON:
1020                     pFileDefExt->OnFolderCustChangeIcon(hwndDlg);
1021                     break;
1022 
1023                 case IDC_FOLDERCUST_CHOOSE_PIC:
1024                     // TODO:
1025                     break;
1026 
1027                 case IDC_FOLDERCUST_RESTORE_DEFAULTS:
1028                     // TODO:
1029                     break;
1030             }
1031             break;
1032 
1033         case WM_NOTIFY:
1034         {
1035             LPPSHNOTIFY lppsn = (LPPSHNOTIFY)lParam;
1036             if (lppsn->hdr.code == PSN_APPLY)
1037             {
1038                 // apply or not
1039                 if (pFileDefExt->OnFolderCustApply(hwndDlg))
1040                 {
1041                     SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_NOERROR);
1042                 }
1043                 else
1044                 {
1045                     SetWindowLongPtr(hwndDlg, DWLP_MSGRESULT, PSNRET_INVALID_NOCHANGEPAGE);
1046                 }
1047                 return TRUE;
1048             }
1049             break;
1050         }
1051 
1052         case PSM_QUERYSIBLINGS:
1053             return FALSE;   // continue
1054 
1055         case WM_DESTROY:
1056             pFileDefExt->OnFolderCustDestroy(hwndDlg);
1057             break;
1058 
1059         default:
1060             break;
1061     }
1062 
1063     return FALSE;
1064 }
1065 
1066 // IDD_FOLDER_CUSTOMIZE WM_DESTROY
1067 void CFileDefExt::OnFolderCustDestroy(HWND hwndDlg)
1068 {
1069     ::DestroyIcon(m_hFolderIcon);
1070     m_hFolderIcon = NULL;
1071 
1072     /* Detach the object from dialog window */
1073     SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)0);
1074 }
1075 
1076 void CFileDefExt::UpdateFolderIcon(HWND hwndDlg)
1077 {
1078     // destroy icon if any
1079     if (m_hFolderIcon)
1080     {
1081         ::DestroyIcon(m_hFolderIcon);
1082         m_hFolderIcon = NULL;
1083     }
1084 
1085     // create the icon
1086     if (m_szFolderIconPath[0] == 0 && m_nFolderIconIndex == 0)
1087     {
1088         m_hFolderIcon = LoadIconW(shell32_hInstance, MAKEINTRESOURCEW(IDI_SHELL_FOLDER));
1089     }
1090     else
1091     {
1092         ExtractIconExW(m_szFolderIconPath, m_nFolderIconIndex, &m_hFolderIcon, NULL, 1);
1093     }
1094 
1095     // set icon
1096     SendDlgItemMessageW(hwndDlg, IDC_FOLDERCUST_ICON, STM_SETICON, (WPARAM)m_hFolderIcon, 0);
1097 }
1098 
1099 // IDD_FOLDER_CUSTOMIZE WM_INITDIALOG
1100 BOOL CFileDefExt::InitFolderCustomizePage(HWND hwndDlg)
1101 {
1102     /* Attach the object to dialog window */
1103     SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)this);
1104 
1105     EnableWindow(GetDlgItem(hwndDlg, IDC_FOLDERCUST_COMBOBOX), FALSE);
1106     EnableWindow(GetDlgItem(hwndDlg, IDC_FOLDERCUST_CHECKBOX), FALSE);
1107     EnableWindow(GetDlgItem(hwndDlg, IDC_FOLDERCUST_CHOOSE_PIC), FALSE);
1108     EnableWindow(GetDlgItem(hwndDlg, IDC_FOLDERCUST_RESTORE_DEFAULTS), FALSE);
1109 
1110     // build the desktop.ini file path
1111     WCHAR szIniFile[MAX_PATH];
1112     StringCchCopyW(szIniFile, _countof(szIniFile), m_wszPath);
1113     PathAppendW(szIniFile, L"desktop.ini");
1114 
1115     // desktop.ini --> m_szFolderIconPath, m_nFolderIconIndex
1116     m_szFolderIconPath[0] = 0;
1117     m_nFolderIconIndex = 0;
1118     if (GetPrivateProfileStringW(s_szShellClassInfo, s_szIconFile, NULL,
1119                                  m_szFolderIconPath, _countof(m_szFolderIconPath), szIniFile))
1120     {
1121         m_nFolderIconIndex = GetPrivateProfileIntW(s_szShellClassInfo, s_szIconIndex, 0, szIniFile);
1122     }
1123     else if (GetPrivateProfileStringW(s_szShellClassInfo, s_szIconResource, NULL,
1124                                       m_szFolderIconPath, _countof(m_szFolderIconPath), szIniFile))
1125     {
1126         m_nFolderIconIndex = PathParseIconLocationW(m_szFolderIconPath);
1127     }
1128 
1129     // update icon
1130     UpdateFolderIcon(hwndDlg);
1131 
1132     return TRUE;
1133 }
1134 
1135 // IDD_FOLDER_CUSTOMIZE IDC_FOLDERCUST_CHANGE_ICON
1136 void CFileDefExt::OnFolderCustChangeIcon(HWND hwndDlg)
1137 {
1138     WCHAR szPath[MAX_PATH];
1139     INT nIconIndex;
1140 
1141     // m_szFolderIconPath, m_nFolderIconIndex --> szPath, nIconIndex
1142     if (m_szFolderIconPath[0])
1143     {
1144         StringCchCopyW(szPath, _countof(szPath), m_szFolderIconPath);
1145         nIconIndex = m_nFolderIconIndex;
1146     }
1147     else
1148     {
1149         szPath[0] = 0;
1150         nIconIndex = 0;
1151     }
1152 
1153     // let the user choose the icon
1154     if (PickIconDlg(hwndDlg, szPath, _countof(szPath), &nIconIndex))
1155     {
1156         // changed
1157         m_bFolderIconIsSet = TRUE;
1158         PropSheet_Changed(GetParent(hwndDlg), hwndDlg);
1159 
1160         // update
1161         StringCchCopyW(m_szFolderIconPath, _countof(m_szFolderIconPath), szPath);
1162         m_nFolderIconIndex = nIconIndex;
1163         UpdateFolderIcon(hwndDlg);
1164     }
1165 }
1166 
1167 // IDD_FOLDER_CUSTOMIZE PSN_APPLY
1168 BOOL CFileDefExt::OnFolderCustApply(HWND hwndDlg)
1169 {
1170     // build the desktop.ini file path
1171     WCHAR szIniFile[MAX_PATH];
1172     StringCchCopyW(szIniFile, _countof(szIniFile), m_wszPath);
1173     PathAppendW(szIniFile, L"desktop.ini");
1174 
1175     if (m_bFolderIconIsSet)     // it is set!
1176     {
1177         DWORD attrs;
1178 
1179         // change folder attributes (-S -R)
1180         attrs = GetFileAttributesW(m_wszPath);
1181         attrs &= ~(FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_READONLY);
1182         SetFileAttributesW(m_wszPath, attrs);
1183 
1184         // change desktop.ini attributes (-S -H -R)
1185         attrs = GetFileAttributesW(szIniFile);
1186         attrs &= ~(FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_READONLY);
1187         SetFileAttributesW(szIniFile, attrs);
1188 
1189         if (m_szFolderIconPath[0])
1190         {
1191             // write IconFile and IconIndex
1192             WritePrivateProfileStringW(s_szShellClassInfo, s_szIconFile, m_szFolderIconPath, szIniFile);
1193 
1194             WCHAR szInt[32];
1195             StringCchPrintfW(szInt, _countof(szInt), L"%d", m_nFolderIconIndex);
1196             WritePrivateProfileStringW(s_szShellClassInfo, s_szIconIndex, szInt, szIniFile);
1197 
1198             // flush!
1199             WritePrivateProfileStringW(NULL, NULL, NULL, szIniFile);
1200         }
1201         else
1202         {
1203             // erase three values
1204             WritePrivateProfileStringW(s_szShellClassInfo, s_szIconFile, NULL, szIniFile);
1205             WritePrivateProfileStringW(s_szShellClassInfo, s_szIconIndex, NULL, szIniFile);
1206             WritePrivateProfileStringW(s_szShellClassInfo, s_szIconResource, NULL, szIniFile);
1207 
1208             // flush!
1209             WritePrivateProfileStringW(NULL, NULL, NULL, szIniFile);
1210         }
1211 
1212         // change desktop.ini attributes (+S +H)
1213         attrs = GetFileAttributesW(szIniFile);
1214         attrs |= FILE_ATTRIBUTE_SYSTEM | FILE_ATTRIBUTE_HIDDEN;
1215         SetFileAttributesW(szIniFile, attrs);
1216 
1217         // change folder attributes (+R)
1218         attrs = GetFileAttributesW(m_wszPath);
1219         attrs |= FILE_ATTRIBUTE_READONLY;
1220         SetFileAttributesW(m_wszPath, attrs);
1221 
1222         // notify to the siblings
1223         PropSheet_QuerySiblings(GetParent(hwndDlg), 0, 0);
1224 
1225         // done!
1226         m_bFolderIconIsSet = FALSE;
1227     }
1228 
1229     return TRUE;
1230 }
1231 
1232 /*****************************************************************************/
1233 
1234 CFileDefExt::CFileDefExt():
1235     m_bDir(FALSE), m_cFiles(0), m_cFolders(0)
1236 {
1237     m_wszPath[0] = L'\0';
1238     m_DirSize.QuadPart = 0ull;
1239     m_DirSizeOnDisc.QuadPart = 0ull;
1240 
1241     m_szFolderIconPath[0] = 0;
1242     m_nFolderIconIndex = 0;
1243     m_hFolderIcon = NULL;
1244     m_bFolderIconIsSet = FALSE;
1245 }
1246 
1247 CFileDefExt::~CFileDefExt()
1248 {
1249 
1250 }
1251 
1252 HRESULT WINAPI
1253 CFileDefExt::Initialize(PCIDLIST_ABSOLUTE pidlFolder, IDataObject *pDataObj, HKEY hkeyProgID)
1254 {
1255     FORMATETC format;
1256     STGMEDIUM stgm;
1257     HRESULT hr;
1258 
1259     TRACE("%p %p %p %p\n", this, pidlFolder, pDataObj, hkeyProgID);
1260 
1261     if (!pDataObj)
1262         return E_FAIL;
1263 
1264     format.cfFormat = CF_HDROP;
1265     format.ptd = NULL;
1266     format.dwAspect = DVASPECT_CONTENT;
1267     format.lindex = -1;
1268     format.tymed = TYMED_HGLOBAL;
1269 
1270     hr = pDataObj->GetData(&format, &stgm);
1271     if (FAILED_UNEXPECTEDLY(hr))
1272         return hr;
1273 
1274     if (!DragQueryFileW((HDROP)stgm.hGlobal, 0, m_wszPath, _countof(m_wszPath)))
1275     {
1276         ERR("DragQueryFileW failed\n");
1277         ReleaseStgMedium(&stgm);
1278         return E_FAIL;
1279     }
1280 
1281     ReleaseStgMedium(&stgm);
1282 
1283     TRACE("File properties %ls\n", m_wszPath);
1284     m_bDir = PathIsDirectoryW(m_wszPath) ? TRUE : FALSE;
1285     if (!m_bDir)
1286         m_VerInfo.Load(m_wszPath);
1287 
1288     return S_OK;
1289 }
1290 
1291 HRESULT WINAPI
1292 CFileDefExt::QueryContextMenu(HMENU hmenu, UINT indexMenu, UINT idCmdFirst, UINT idCmdLast, UINT uFlags)
1293 {
1294     UNIMPLEMENTED;
1295     return E_NOTIMPL;
1296 }
1297 
1298 HRESULT WINAPI
1299 CFileDefExt::InvokeCommand(LPCMINVOKECOMMANDINFO lpici)
1300 {
1301     UNIMPLEMENTED;
1302     return E_NOTIMPL;
1303 }
1304 
1305 HRESULT WINAPI
1306 CFileDefExt::GetCommandString(UINT_PTR idCmd, UINT uType, UINT *pwReserved, LPSTR pszName, UINT cchMax)
1307 {
1308     UNIMPLEMENTED;
1309     return E_NOTIMPL;
1310 }
1311 
1312 HRESULT WINAPI
1313 CFileDefExt::AddPages(LPFNADDPROPSHEETPAGE pfnAddPage, LPARAM lParam)
1314 {
1315     HPROPSHEETPAGE hPage;
1316     WORD wResId = m_bDir ? IDD_FOLDER_PROPERTIES : IDD_FILE_PROPERTIES;
1317 
1318     hPage = SH_CreatePropertySheetPage(wResId,
1319                                        GeneralPageProc,
1320                                        (LPARAM)this,
1321                                        NULL);
1322     if (hPage)
1323         pfnAddPage(hPage, lParam);
1324 
1325     if (!m_bDir && GetFileVersionInfoSizeW(m_wszPath, NULL))
1326     {
1327         hPage = SH_CreatePropertySheetPage(IDD_FILE_VERSION,
1328                                             VersionPageProc,
1329                                             (LPARAM)this,
1330                                             NULL);
1331         if (hPage)
1332             pfnAddPage(hPage, lParam);
1333     }
1334 
1335     if (m_bDir)
1336     {
1337         hPage = SH_CreatePropertySheetPage(IDD_FOLDER_CUSTOMIZE,
1338                                            FolderCustomizePageProc,
1339                                            (LPARAM)this,
1340                                            NULL);
1341         if (hPage)
1342             pfnAddPage(hPage, lParam);
1343     }
1344 
1345     return S_OK;
1346 }
1347 
1348 HRESULT WINAPI
1349 CFileDefExt::ReplacePage(UINT uPageID, LPFNADDPROPSHEETPAGE pfnReplacePage, LPARAM lParam)
1350 {
1351     UNIMPLEMENTED;
1352     return E_NOTIMPL;
1353 }
1354 
1355 HRESULT WINAPI
1356 CFileDefExt::SetSite(IUnknown *punk)
1357 {
1358     UNIMPLEMENTED;
1359     return E_NOTIMPL;
1360 }
1361 
1362 HRESULT WINAPI
1363 CFileDefExt::GetSite(REFIID iid, void **ppvSite)
1364 {
1365     UNIMPLEMENTED;
1366     return E_NOTIMPL;
1367 }
1368 
1369 DWORD WINAPI
1370 CFileDefExt::_CountFolderAndFilesThreadProc(LPVOID lpParameter)
1371 {
1372     _CountFolderAndFilesData *data = static_cast<_CountFolderAndFilesData*>(lpParameter);
1373     DWORD ticks = 0;
1374     data->This->CountFolderAndFiles(data->hwndDlg, data->pwszBuf, &ticks);
1375 
1376     //Release the CFileDefExt and data object holds in the copying thread.
1377     data->This->Release();
1378     HeapFree(GetProcessHeap(), 0, data->pwszBuf);
1379     HeapFree(GetProcessHeap(), 0, data);
1380 
1381     return 0;
1382 }
1383 
1384 BOOL
1385 CFileDefExt::CountFolderAndFiles(HWND hwndDlg, LPCWSTR pwszBuf, DWORD *ticks)
1386 {
1387     CString sBuf = pwszBuf;
1388     sBuf += L"\\" ;
1389     CString sSearch = sBuf;
1390     sSearch += L"*" ;
1391     CString sFileName;
1392 
1393     WIN32_FIND_DATAW wfd;
1394     HANDLE hFind = FindFirstFileW(sSearch, &wfd);
1395     if (hFind == INVALID_HANDLE_VALUE)
1396     {
1397         ERR("FindFirstFileW %ls failed\n", sSearch.GetString());
1398         return FALSE;
1399     }
1400 
1401     BOOL root = FALSE;
1402     if (*ticks == 0) {
1403         *ticks = GetTickCount();
1404         root = TRUE;
1405     }
1406 
1407     do
1408     {
1409         sFileName = sBuf;
1410         sFileName += wfd.cFileName;
1411         if (wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
1412         {
1413             /* Don't process "." and ".." items */
1414             if (!wcscmp(wfd.cFileName, L".") || !wcscmp(wfd.cFileName, L".."))
1415                 continue;
1416 
1417             ++m_cFolders;
1418 
1419             CountFolderAndFiles(hwndDlg, sFileName, ticks);
1420         }
1421         else
1422         {
1423             m_cFiles++;
1424 
1425             ULARGE_INTEGER FileSize;
1426             FileSize.u.LowPart  = wfd.nFileSizeLow;
1427             FileSize.u.HighPart = wfd.nFileSizeHigh;
1428             m_DirSize.QuadPart += FileSize.QuadPart;
1429             // Calculate size on disc
1430             if (!GetPhysicalFileSize(sFileName.GetString(), &FileSize))
1431                 ERR("GetPhysicalFileSize failed for %ls\n", sFileName.GetString());
1432             m_DirSizeOnDisc.QuadPart += FileSize.QuadPart;
1433         }
1434         if (GetTickCount() - *ticks > (DWORD) 300)
1435         {
1436             /* FIXME Using IsWindow is generally ill advised */
1437             if (IsWindow(hwndDlg))
1438             {
1439                 WCHAR wszBuf[100];
1440 
1441                 if (SH_FormatFileSizeWithBytes(&m_DirSize, wszBuf, _countof(wszBuf)))
1442                     SetDlgItemTextW(hwndDlg, 14011, wszBuf);
1443 
1444                 if (SH_FormatFileSizeWithBytes(&m_DirSizeOnDisc, wszBuf, _countof(wszBuf)))
1445                     SetDlgItemTextW(hwndDlg, 14012, wszBuf);
1446 
1447                 /* Display files and folders count */
1448                 WCHAR wszFormat[100];
1449                 LoadStringW(shell32_hInstance, IDS_FILE_FOLDER, wszFormat, _countof(wszFormat));
1450                 StringCchPrintfW(wszBuf, _countof(wszBuf), wszFormat, m_cFiles, m_cFolders);
1451                 SetDlgItemTextW(hwndDlg, 14027, wszBuf);
1452                 *ticks = GetTickCount();
1453             }
1454             else
1455                 break;
1456         }
1457     } while(FindNextFileW(hFind, &wfd));
1458 
1459     if (root && IsWindow(hwndDlg))
1460     {
1461         WCHAR wszBuf[100];
1462 
1463         if (SH_FormatFileSizeWithBytes(&m_DirSize, wszBuf, _countof(wszBuf)))
1464             SetDlgItemTextW(hwndDlg, 14011, wszBuf);
1465 
1466         if (SH_FormatFileSizeWithBytes(&m_DirSizeOnDisc, wszBuf, _countof(wszBuf)))
1467             SetDlgItemTextW(hwndDlg, 14012, wszBuf);
1468 
1469         /* Display files and folders count */
1470         WCHAR wszFormat[100];
1471         LoadStringW(shell32_hInstance, IDS_FILE_FOLDER, wszFormat, _countof(wszFormat));
1472         StringCchPrintfW(wszBuf, _countof(wszBuf), wszFormat, m_cFiles, m_cFolders);
1473         SetDlgItemTextW(hwndDlg, 14027, wszBuf);
1474     }
1475 
1476     FindClose(hFind);
1477     return TRUE;
1478 }
1479