xref: /reactos/dll/win32/shell32/dialogs/drive.cpp (revision 465745b6)
1 /*
2  *                 Shell Library Functions
3  *
4  * Copyright 2005 Johannes Anderwald
5  * Copyright 2017 Katayama Hirofumi MZ
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 Street, Fifth Floor, Boston, MA  02110-1301  USA
20  */
21 
22 #include "precomp.h"
23 #include <process.h>
24 
25 WINE_DEFAULT_DEBUG_CHANNEL(shell);
26 
27 typedef struct
28 {
29     WCHAR   Drive;
30     UINT    Options;
31     UINT Result;
32     BOOL bFormattingNow;
33 } FORMAT_DRIVE_CONTEXT, *PFORMAT_DRIVE_CONTEXT;
34 
35 EXTERN_C HPSXA WINAPI SHCreatePropSheetExtArrayEx(HKEY hKey, LPCWSTR pszSubKey, UINT max_iface, IDataObject *pDataObj);
36 HPROPSHEETPAGE SH_CreatePropertySheetPage(LPCSTR resname, DLGPROC dlgproc, LPARAM lParam, LPWSTR szTitle);
37 
38 static BOOL
39 GetDefaultClusterSize(LPWSTR szFs, PDWORD pClusterSize, PULARGE_INTEGER TotalNumberOfBytes)
40 {
41     DWORD ClusterSize;
42 
43     if (!wcsicmp(szFs, L"FAT16") ||
44         !wcsicmp(szFs, L"FAT")) //REACTOS HACK
45     {
46         if (TotalNumberOfBytes->QuadPart <= (16 * 1024 * 1024))
47             ClusterSize = 2048;
48         else if (TotalNumberOfBytes->QuadPart <= (32 * 1024 * 1024))
49             ClusterSize = 512;
50         else if (TotalNumberOfBytes->QuadPart <= (64 * 1024 * 1024))
51             ClusterSize = 1024;
52         else if (TotalNumberOfBytes->QuadPart <= (128 * 1024 * 1024))
53             ClusterSize = 2048;
54         else if (TotalNumberOfBytes->QuadPart <= (256 * 1024 * 1024))
55             ClusterSize = 4096;
56         else if (TotalNumberOfBytes->QuadPart <= (512 * 1024 * 1024))
57             ClusterSize = 8192;
58         else if (TotalNumberOfBytes->QuadPart <= (1024 * 1024 * 1024))
59             ClusterSize = 16384;
60         else if (TotalNumberOfBytes->QuadPart <= (2048LL * 1024LL * 1024LL))
61             ClusterSize = 32768;
62         else if (TotalNumberOfBytes->QuadPart <= (4096LL * 1024LL * 1024LL))
63             ClusterSize = 8192;
64         else
65             return FALSE;
66     }
67     else if (!wcsicmp(szFs, L"FAT32"))
68     {
69         if (TotalNumberOfBytes->QuadPart <= (64 * 1024 * 1024))
70             ClusterSize = 512;
71         else if (TotalNumberOfBytes->QuadPart <= (128   * 1024 * 1024))
72             ClusterSize = 1024;
73         else if (TotalNumberOfBytes->QuadPart <= (256   * 1024 * 1024))
74             ClusterSize = 2048;
75         else if (TotalNumberOfBytes->QuadPart <= (8192LL  * 1024LL * 1024LL))
76             ClusterSize = 2048;
77         else if (TotalNumberOfBytes->QuadPart <= (16384LL * 1024LL * 1024LL))
78             ClusterSize = 8192;
79         else if (TotalNumberOfBytes->QuadPart <= (32768LL * 1024LL * 1024LL))
80             ClusterSize = 16384;
81         else
82             return FALSE;
83     }
84     else if (!wcsicmp(szFs, L"NTFS"))
85     {
86         if (TotalNumberOfBytes->QuadPart <= (512 * 1024 * 1024))
87             ClusterSize = 512;
88         else if (TotalNumberOfBytes->QuadPart <= (1024 * 1024 * 1024))
89             ClusterSize = 1024;
90         else if (TotalNumberOfBytes->QuadPart <= (2048LL * 1024LL * 1024LL))
91             ClusterSize = 2048;
92         else
93             ClusterSize = 2048;
94     }
95     else if (!wcsicmp(szFs, L"EXT2"))
96     {
97         // auto block size calculation
98         ClusterSize = 0;
99     }
100     else if (!wcsicmp(szFs, L"BtrFS"))
101     {
102         // auto block size calculation
103         ClusterSize = 0;
104     }
105     else
106         return FALSE;
107 
108     *pClusterSize = ClusterSize;
109     return TRUE;
110 }
111 
112 typedef struct _DRIVE_PROP_PAGE
113 {
114     LPCSTR resname;
115     DLGPROC dlgproc;
116     UINT DriveType;
117 } DRIVE_PROP_PAGE;
118 
119 HRESULT
120 SH_ShowDriveProperties(WCHAR *pwszDrive, LPCITEMIDLIST pidlFolder, PCUITEMID_CHILD_ARRAY apidl)
121 {
122     HPSXA hpsx = NULL;
123     HPROPSHEETPAGE hpsp[MAX_PROPERTY_SHEET_PAGE];
124     PROPSHEETHEADERW psh;
125     CComObject<CDrvDefExt> *pDrvDefExt = NULL;
126     WCHAR wszName[256];
127 
128     ZeroMemory(&psh, sizeof(PROPSHEETHEADERW));
129     psh.dwSize = sizeof(PROPSHEETHEADERW);
130     psh.dwFlags = 0; // FIXME: make it modeless
131     psh.hwndParent = NULL;
132     psh.nStartPage = 0;
133     psh.phpage = hpsp;
134 
135     LPITEMIDLIST completePidl = ILCombine(pidlFolder, apidl[0]);
136     if (!completePidl)
137         return E_OUTOFMEMORY;
138 
139     if (ILGetDisplayNameExW(NULL, completePidl, wszName, ILGDN_NORMAL))
140     {
141         psh.pszCaption = wszName;
142         psh.dwFlags |= PSH_PROPTITLE;
143     }
144 
145     ILFree(completePidl);
146 
147     CComPtr<IDataObject> pDataObj;
148     HRESULT hr = SHCreateDataObject(pidlFolder, 1, apidl, NULL, IID_PPV_ARG(IDataObject, &pDataObj));
149 
150     if (SUCCEEDED(hr))
151     {
152         hr = CComObject<CDrvDefExt>::CreateInstance(&pDrvDefExt);
153         if (SUCCEEDED(hr))
154         {
155             pDrvDefExt->AddRef(); // CreateInstance returns object with 0 ref count
156             hr = pDrvDefExt->Initialize(pidlFolder, pDataObj, NULL);
157             if (SUCCEEDED(hr))
158             {
159                 hr = pDrvDefExt->AddPages(AddPropSheetPageCallback, (LPARAM)&psh);
160                 if (FAILED(hr))
161                     ERR("AddPages failed\n");
162             } else
163                 ERR("Initialize failed\n");
164         }
165 
166         hpsx = SHCreatePropSheetExtArrayEx(HKEY_CLASSES_ROOT, L"Drive", MAX_PROPERTY_SHEET_PAGE, pDataObj);
167         if (hpsx)
168             SHAddFromPropSheetExtArray(hpsx, (LPFNADDPROPSHEETPAGE)AddPropSheetPageCallback, (LPARAM)&psh);
169     }
170 
171     // NOTE: Currently property sheet is modal. If we make it modeless, then it returns HWND.
172     INT_PTR ret = PropertySheetW(&psh);
173 
174     if (hpsx)
175         SHDestroyPropSheetExtArray(hpsx);
176     if (pDrvDefExt)
177         pDrvDefExt->Release();
178 
179     if (ret > 0)
180         return S_OK;
181     if (ret == 0)
182         return S_FALSE;
183     return E_FAIL;
184 }
185 
186 static VOID
187 InsertDefaultClusterSizeForFs(HWND hwndDlg, PFORMAT_DRIVE_CONTEXT pContext)
188 {
189     WCHAR wszBuf[100] = {0};
190     WCHAR szDrive[] = L"C:\\";
191     INT iSelIndex;
192     ULARGE_INTEGER FreeBytesAvailableUser, TotalNumberOfBytes;
193     DWORD ClusterSize;
194     LRESULT lIndex;
195     HWND hDlgCtrl;
196 
197     hDlgCtrl = GetDlgItem(hwndDlg, 28677);
198     iSelIndex = SendMessage(hDlgCtrl, CB_GETCURSEL, 0, 0);
199     if (iSelIndex == CB_ERR)
200         return;
201 
202     if (SendMessageW(hDlgCtrl, CB_GETLBTEXT, iSelIndex, (LPARAM)wszBuf) == CB_ERR)
203         return;
204 
205     szDrive[0] = pContext->Drive + L'A';
206 
207     if (!GetDiskFreeSpaceExW(szDrive, &FreeBytesAvailableUser, &TotalNumberOfBytes, NULL))
208         return;
209 
210     if (!wcsicmp(wszBuf, L"FAT16") ||
211         !wcsicmp(wszBuf, L"FAT")) //REACTOS HACK
212     {
213         if (!GetDefaultClusterSize(wszBuf, &ClusterSize, &TotalNumberOfBytes))
214         {
215             TRACE("FAT16 is not supported on hdd larger than 4G current %lu\n", TotalNumberOfBytes.QuadPart);
216             SendMessageW(hDlgCtrl, CB_DELETESTRING, iSelIndex, 0);
217             return;
218         }
219 
220         if (LoadStringW(shell32_hInstance, IDS_DEFAULT_CLUSTER_SIZE, wszBuf, _countof(wszBuf)))
221         {
222             hDlgCtrl = GetDlgItem(hwndDlg, 28680);
223             SendMessageW(hDlgCtrl, CB_RESETCONTENT, 0, 0);
224             lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
225             if (lIndex != CB_ERR)
226                 SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
227             SendMessageW(hDlgCtrl, CB_SETCURSEL, 0, 0);
228         }
229 
230         SendMessageW(GetDlgItem(hwndDlg, 28675), BM_SETCHECK, BST_UNCHECKED, 0);
231         EnableWindow(GetDlgItem(hwndDlg, 28675), FALSE);
232     }
233     else if (!wcsicmp(wszBuf, L"FAT32"))
234     {
235         if (!GetDefaultClusterSize(wszBuf, &ClusterSize, &TotalNumberOfBytes))
236         {
237             TRACE("FAT32 is not supported on hdd larger than 32G current %lu\n", TotalNumberOfBytes.QuadPart);
238             SendMessageW(hDlgCtrl, CB_DELETESTRING, iSelIndex, 0);
239             return;
240         }
241 
242         if (LoadStringW(shell32_hInstance, IDS_DEFAULT_CLUSTER_SIZE, wszBuf, _countof(wszBuf)))
243         {
244             hDlgCtrl = GetDlgItem(hwndDlg, 28680);
245             SendMessageW(hDlgCtrl, CB_RESETCONTENT, 0, 0);
246             lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
247             if (lIndex != CB_ERR)
248                 SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
249             SendMessageW(hDlgCtrl, CB_SETCURSEL, 0, 0);
250         }
251 
252         SendMessageW(GetDlgItem(hwndDlg, 28675), BM_SETCHECK, BST_UNCHECKED, 0);
253         EnableWindow(GetDlgItem(hwndDlg, 28675), FALSE);
254     }
255     else if (!wcsicmp(wszBuf, L"NTFS"))
256     {
257         if (!GetDefaultClusterSize(wszBuf, &ClusterSize, &TotalNumberOfBytes))
258         {
259             TRACE("NTFS is not supported on hdd larger than 2TB current %lu\n", TotalNumberOfBytes.QuadPart);
260             SendMessageW(hDlgCtrl, CB_DELETESTRING, iSelIndex, 0);
261             return;
262         }
263 
264         hDlgCtrl = GetDlgItem(hwndDlg, 28680);
265         if (LoadStringW(shell32_hInstance, IDS_DEFAULT_CLUSTER_SIZE, wszBuf, _countof(wszBuf)))
266         {
267             SendMessageW(hDlgCtrl, CB_RESETCONTENT, 0, 0);
268             lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
269             if (lIndex != CB_ERR)
270                 SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
271             SendMessageW(hDlgCtrl, CB_SETCURSEL, 0, 0);
272         }
273         ClusterSize = 512;
274         for (lIndex = 0; lIndex < 4; lIndex++)
275         {
276             TotalNumberOfBytes.QuadPart = ClusterSize;
277             if (StrFormatByteSizeW(TotalNumberOfBytes.QuadPart, wszBuf, _countof(wszBuf)))
278             {
279                 lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
280                 if (lIndex != CB_ERR)
281                     SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
282             }
283             ClusterSize *= 2;
284         }
285 
286         EnableWindow(GetDlgItem(hwndDlg, 28675), TRUE);
287     }
288     else if (!wcsicmp(wszBuf, L"EXT2"))
289     {
290         if (!GetDefaultClusterSize(wszBuf, &ClusterSize, &TotalNumberOfBytes))
291         {
292             TRACE("EXT2 is not supported on hdd larger than 32T current %lu\n", TotalNumberOfBytes.QuadPart);
293             SendMessageW(hDlgCtrl, CB_DELETESTRING, iSelIndex, 0);
294             return;
295         }
296 
297         if (LoadStringW(shell32_hInstance, IDS_DEFAULT_CLUSTER_SIZE, wszBuf, _countof(wszBuf)))
298         {
299             hDlgCtrl = GetDlgItem(hwndDlg, 28680);
300             SendMessageW(hDlgCtrl, CB_RESETCONTENT, 0, 0);
301             lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
302             if (lIndex != CB_ERR)
303                 SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
304             SendMessageW(hDlgCtrl, CB_SETCURSEL, 0, 0);
305         }
306 
307         EnableWindow(GetDlgItem(hwndDlg, 28675), TRUE);
308     }
309     else if (!wcsicmp(wszBuf, L"BtrFS"))
310     {
311         if (!GetDefaultClusterSize(wszBuf, &ClusterSize, &TotalNumberOfBytes))
312         {
313             TRACE("BtrFS is not supported on hdd larger than 16E current %lu\n", TotalNumberOfBytes.QuadPart);
314             SendMessageW(hDlgCtrl, CB_DELETESTRING, iSelIndex, 0);
315             return;
316         }
317 
318         if (LoadStringW(shell32_hInstance, IDS_DEFAULT_CLUSTER_SIZE, wszBuf, _countof(wszBuf)))
319         {
320             hDlgCtrl = GetDlgItem(hwndDlg, 28680);
321             SendMessageW(hDlgCtrl, CB_RESETCONTENT, 0, 0);
322             lIndex = SendMessageW(hDlgCtrl, CB_ADDSTRING, 0, (LPARAM)wszBuf);
323             if (lIndex != CB_ERR)
324                 SendMessageW(hDlgCtrl, CB_SETITEMDATA, lIndex, (LPARAM)ClusterSize);
325             SendMessageW(hDlgCtrl, CB_SETCURSEL, 0, 0);
326         }
327 
328         EnableWindow(GetDlgItem(hwndDlg, 28675), TRUE);
329     }
330     else
331     {
332         FIXME("unknown fs\n");
333         SendDlgItemMessageW(hwndDlg, 28680, CB_RESETCONTENT, iSelIndex, 0);
334         return;
335     }
336 }
337 
338 static VOID
339 InitializeFormatDriveDlg(HWND hwndDlg, PFORMAT_DRIVE_CONTEXT pContext)
340 {
341     WCHAR szText[120];
342     WCHAR szDrive[] = L"C:\\";
343     WCHAR szFs[30] = L"";
344     INT cchText;
345     ULARGE_INTEGER FreeBytesAvailableUser, TotalNumberOfBytes;
346     DWORD dwIndex, dwDefault;
347     UCHAR uMinor, uMajor;
348     BOOLEAN Latest;
349     HWND hwndFileSystems;
350 
351     cchText = GetWindowTextW(hwndDlg, szText, _countof(szText) - 1);
352     if (cchText < 0)
353         cchText = 0;
354     szText[cchText++] = L' ';
355     szDrive[0] = pContext->Drive + L'A';
356     if (GetVolumeInformationW(szDrive, &szText[cchText], _countof(szText) - cchText, NULL, NULL, NULL, szFs, _countof(szFs)))
357     {
358         if (szText[cchText] == UNICODE_NULL)
359         {
360             /* load default volume label */
361             cchText += LoadStringW(shell32_hInstance, IDS_DRIVE_FIXED, &szText[cchText], _countof(szText) - cchText);
362         }
363         else
364         {
365             /* set volume label */
366             SetDlgItemTextW(hwndDlg, 28679, &szText[cchText]);
367             cchText += wcslen(&szText[cchText]);
368         }
369     }
370 
371     StringCchPrintfW(szText + cchText, _countof(szText) - cchText, L" (%c:)", szDrive[0]);
372 
373     /* set window text */
374     SetWindowTextW(hwndDlg, szText);
375 
376     if (GetDiskFreeSpaceExW(szDrive, &FreeBytesAvailableUser, &TotalNumberOfBytes, NULL))
377     {
378         if (StrFormatByteSizeW(TotalNumberOfBytes.QuadPart, szText, _countof(szText)))
379         {
380             /* add drive capacity */
381             SendDlgItemMessageW(hwndDlg, 28673, CB_ADDSTRING, 0, (LPARAM)szText);
382             SendDlgItemMessageW(hwndDlg, 28673, CB_SETCURSEL, 0, (LPARAM)0);
383         }
384     }
385 
386     if (pContext->Options & SHFMT_OPT_FULL)
387     {
388         /* check quick format button */
389         SendDlgItemMessageW(hwndDlg, 28674, BM_SETCHECK, BST_CHECKED, 0);
390     }
391 
392     /* enumerate all available filesystems */
393     dwIndex = 0;
394     dwDefault = 0;
395     hwndFileSystems = GetDlgItem(hwndDlg, 28677);
396 
397     while(QueryAvailableFileSystemFormat(dwIndex, szText, &uMajor, &uMinor, &Latest))
398     {
399         if (!wcsicmp(szText, szFs))
400             dwDefault = dwIndex;
401 
402         SendMessageW(hwndFileSystems, CB_ADDSTRING, 0, (LPARAM)szText);
403         dwIndex++;
404     }
405 
406     if (!dwIndex)
407     {
408         ERR("no filesystem providers\n");
409         return;
410     }
411 
412     /* select default filesys */
413     SendMessageW(hwndFileSystems, CB_SETCURSEL, dwDefault, 0);
414     /* setup cluster combo */
415     InsertDefaultClusterSizeForFs(hwndDlg, pContext);
416 }
417 
418 static HWND FormatDrvDialog = NULL;
419 static BOOLEAN bSuccess = FALSE;
420 
421 static BOOLEAN NTAPI
422 FormatExCB(
423     IN CALLBACKCOMMAND Command,
424     IN ULONG SubAction,
425     IN PVOID ActionInfo)
426 {
427     PDWORD Progress;
428     PBOOLEAN pSuccess;
429     switch(Command)
430     {
431         case PROGRESS:
432             Progress = (PDWORD)ActionInfo;
433             SendDlgItemMessageW(FormatDrvDialog, 28678, PBM_SETPOS, (WPARAM)*Progress, 0);
434             break;
435         case DONE:
436             pSuccess = (PBOOLEAN)ActionInfo;
437             bSuccess = (*pSuccess);
438             ShellMessageBoxW(shell32_hInstance, FormatDrvDialog, MAKEINTRESOURCEW(IDS_FORMAT_COMPLETE), MAKEINTRESOURCEW(IDS_FORMAT_TITLE), MB_OK | MB_ICONINFORMATION);
439             SendDlgItemMessageW(FormatDrvDialog, 28678, PBM_SETPOS, 0, 0);
440             break;
441 
442         case VOLUMEINUSE:
443         case INSUFFICIENTRIGHTS:
444         case FSNOTSUPPORTED:
445         case CLUSTERSIZETOOSMALL:
446             bSuccess = FALSE;
447             FIXME("\n");
448             break;
449 
450         default:
451             break;
452     }
453 
454     return TRUE;
455 }
456 
457 VOID
458 FormatDrive(HWND hwndDlg, PFORMAT_DRIVE_CONTEXT pContext)
459 {
460     WCHAR szDrive[4] = { L'C', ':', '\\', 0 };
461     WCHAR szFileSys[40] = {0};
462     WCHAR szLabel[40] = {0};
463     INT iSelIndex;
464     UINT Length;
465     HWND hDlgCtrl;
466     BOOL QuickFormat;
467     DWORD ClusterSize;
468     DWORD DriveType;
469     FMIFS_MEDIA_FLAG MediaFlag = FMIFS_HARDDISK;
470 
471     /* set volume path */
472     szDrive[0] = pContext->Drive + L'A';
473 
474     /* get filesystem */
475     hDlgCtrl = GetDlgItem(hwndDlg, 28677);
476     iSelIndex = SendMessageW(hDlgCtrl, CB_GETCURSEL, 0, 0);
477     if (iSelIndex == CB_ERR)
478     {
479         FIXME("\n");
480         return;
481     }
482     Length = SendMessageW(hDlgCtrl, CB_GETLBTEXTLEN, iSelIndex, 0);
483     if ((int)Length == CB_ERR || Length + 1 > _countof(szFileSys))
484     {
485         FIXME("\n");
486         return;
487     }
488 
489     /* retrieve the file system */
490     SendMessageW(hDlgCtrl, CB_GETLBTEXT, iSelIndex, (LPARAM)szFileSys);
491     szFileSys[_countof(szFileSys)-1] = L'\0';
492 
493     /* retrieve the volume label */
494     hDlgCtrl = GetWindow(hwndDlg, 28679);
495     Length = SendMessageW(hDlgCtrl, WM_GETTEXTLENGTH, 0, 0);
496     if (Length + 1 > _countof(szLabel))
497     {
498         FIXME("\n");
499         return;
500     }
501     SendMessageW(hDlgCtrl, WM_GETTEXT, _countof(szLabel), (LPARAM)szLabel);
502     szLabel[(sizeof(szLabel)/sizeof(WCHAR))-1] = L'\0';
503 
504     /* check for quickformat */
505     if (SendDlgItemMessageW(hwndDlg, 28674, BM_GETCHECK, 0, 0) == BST_CHECKED)
506         QuickFormat = TRUE;
507     else
508         QuickFormat = FALSE;
509 
510     /* get the cluster size */
511     hDlgCtrl = GetDlgItem(hwndDlg, 28680);
512     iSelIndex = SendMessageW(hDlgCtrl, CB_GETCURSEL, 0, 0);
513     if (iSelIndex == CB_ERR)
514     {
515         FIXME("\n");
516         return;
517     }
518     ClusterSize = SendMessageW(hDlgCtrl, CB_GETITEMDATA, iSelIndex, 0);
519     if ((int)ClusterSize == CB_ERR)
520     {
521         FIXME("\n");
522         return;
523     }
524 
525     hDlgCtrl = GetDlgItem(hwndDlg, 28680);
526     SendMessageW(hDlgCtrl, PBM_SETRANGE, 0, MAKELPARAM(0, 100));
527     bSuccess = FALSE;
528 
529     /* FIXME
530      * will cause display problems
531      * when performing more than one format
532      */
533     FormatDrvDialog = hwndDlg;
534 
535     /* See if the drive is removable or not */
536     DriveType = GetDriveTypeW(szDrive);
537     switch (DriveType)
538     {
539         case DRIVE_UNKNOWN:
540         case DRIVE_REMOTE:
541         case DRIVE_CDROM:
542         case DRIVE_NO_ROOT_DIR:
543         {
544             FIXME("\n");
545             return;
546         }
547 
548         case DRIVE_REMOVABLE:
549             MediaFlag = FMIFS_FLOPPY;
550             break;
551 
552         case DRIVE_FIXED:
553         case DRIVE_RAMDISK:
554             MediaFlag = FMIFS_HARDDISK;
555             break;
556     }
557 
558     /* Format the drive */
559     FormatEx(szDrive,
560              MediaFlag,
561              szFileSys,
562              szLabel,
563              QuickFormat,
564              ClusterSize,
565              FormatExCB);
566 
567     FormatDrvDialog = NULL;
568     if (!bSuccess)
569     {
570         pContext->Result = SHFMT_ERROR;
571     }
572     else if (QuickFormat)
573     {
574         pContext->Result = SHFMT_OPT_FULL;
575     }
576     else
577     {
578         pContext->Result = FALSE;
579     }
580 }
581 
582 struct FORMAT_DRIVE_PARAMS
583 {
584     HWND hwndDlg;
585     PFORMAT_DRIVE_CONTEXT pContext;
586 };
587 
588 static unsigned __stdcall DoFormatDrive(void *args)
589 {
590     FORMAT_DRIVE_PARAMS *pParams = reinterpret_cast<FORMAT_DRIVE_PARAMS *>(args);
591     HWND hwndDlg = pParams->hwndDlg;
592     PFORMAT_DRIVE_CONTEXT pContext = pParams->pContext;
593 
594 	/* Disable controls during format */
595     HMENU hSysMenu = GetSystemMenu(hwndDlg, FALSE);
596     EnableMenuItem(hSysMenu, SC_CLOSE, MF_BYCOMMAND | MF_GRAYED);
597     EnableWindow(GetDlgItem(hwndDlg, IDOK), FALSE);
598     EnableWindow(GetDlgItem(hwndDlg, IDCANCEL), FALSE);
599     EnableWindow(GetDlgItem(hwndDlg, 28673), FALSE);
600     EnableWindow(GetDlgItem(hwndDlg, 28677), FALSE);
601     EnableWindow(GetDlgItem(hwndDlg, 28680), FALSE);
602     EnableWindow(GetDlgItem(hwndDlg, 28679), FALSE);
603     EnableWindow(GetDlgItem(hwndDlg, 28674), FALSE);
604 
605     FormatDrive(hwndDlg, pContext);
606 
607 	/* Re-enable controls after format */
608     EnableWindow(GetDlgItem(hwndDlg, IDOK), TRUE);
609     EnableWindow(GetDlgItem(hwndDlg, IDCANCEL), TRUE);
610     EnableWindow(GetDlgItem(hwndDlg, 28673), TRUE);
611     EnableWindow(GetDlgItem(hwndDlg, 28677), TRUE);
612     EnableWindow(GetDlgItem(hwndDlg, 28680), TRUE);
613     EnableWindow(GetDlgItem(hwndDlg, 28679), TRUE);
614     EnableWindow(GetDlgItem(hwndDlg, 28674), TRUE);
615     EnableMenuItem(hSysMenu, SC_CLOSE, MF_BYCOMMAND | MF_ENABLED);
616     pContext->bFormattingNow = FALSE;
617 
618     delete pParams;
619     return 0;
620 }
621 
622 static INT_PTR CALLBACK
623 FormatDriveDlg(HWND hwndDlg, UINT uMsg, WPARAM wParam, LPARAM lParam)
624 {
625     PFORMAT_DRIVE_CONTEXT pContext;
626 
627     switch(uMsg)
628     {
629         case WM_INITDIALOG:
630             InitializeFormatDriveDlg(hwndDlg, (PFORMAT_DRIVE_CONTEXT)lParam);
631             SetWindowLongPtr(hwndDlg, DWLP_USER, (LONG_PTR)lParam);
632             return TRUE;
633         case WM_COMMAND:
634             switch(LOWORD(wParam))
635             {
636                 case IDOK:
637                     pContext = (PFORMAT_DRIVE_CONTEXT)GetWindowLongPtr(hwndDlg, DWLP_USER);
638                     if (pContext->bFormattingNow)
639                         break;
640 
641                     if (ShellMessageBoxW(shell32_hInstance, hwndDlg,
642                                          MAKEINTRESOURCEW(IDS_FORMAT_WARNING),
643                                          MAKEINTRESOURCEW(IDS_FORMAT_TITLE),
644                                          MB_OKCANCEL | MB_ICONWARNING) == IDOK)
645                     {
646                         pContext->bFormattingNow = TRUE;
647 
648                         FORMAT_DRIVE_PARAMS *pParams = new FORMAT_DRIVE_PARAMS;
649                         pParams->hwndDlg = hwndDlg;
650                         pParams->pContext = pContext;
651 
652                         unsigned tid;
653                         HANDLE hThread = (HANDLE)_beginthreadex(NULL, 0, DoFormatDrive, pParams, 0, &tid);
654                         CloseHandle(hThread);
655                     }
656                     break;
657                 case IDCANCEL:
658                     pContext = (PFORMAT_DRIVE_CONTEXT)GetWindowLongPtr(hwndDlg, DWLP_USER);
659                     if (pContext->bFormattingNow)
660                         break;
661 
662                     EndDialog(hwndDlg, pContext->Result);
663                     break;
664                 case 28677: // filesystem combo
665                     if (HIWORD(wParam) == CBN_SELENDOK)
666                     {
667                         pContext = (PFORMAT_DRIVE_CONTEXT)GetWindowLongPtr(hwndDlg, DWLP_USER);
668                         if (pContext->bFormattingNow)
669                             break;
670 
671                         InsertDefaultClusterSizeForFs(hwndDlg, pContext);
672                     }
673                     break;
674             }
675     }
676     return FALSE;
677 }
678 
679 /*************************************************************************
680  *              SHFormatDrive (SHELL32.@)
681  */
682 
683 DWORD
684 WINAPI
685 SHFormatDrive(HWND hwnd, UINT drive, UINT fmtID, UINT options)
686 {
687     FORMAT_DRIVE_CONTEXT Context;
688     int result;
689 
690     TRACE("%p, 0x%08x, 0x%08x, 0x%08x - stub\n", hwnd, drive, fmtID, options);
691 
692     Context.Drive = drive;
693     Context.Options = options;
694     Context.Result = FALSE;
695     Context.bFormattingNow = FALSE;
696 
697     result = DialogBoxParamW(shell32_hInstance, MAKEINTRESOURCEW(IDD_FORMAT_DRIVE), hwnd, FormatDriveDlg, (LPARAM)&Context);
698 
699     return result;
700 }
701