xref: /reactos/dll/win32/msgina/shutdown.c (revision 7f26a396)
1 /*
2  * COPYRIGHT:       See COPYING in the top level directory
3  * PROJECT:         ReactOS msgina.dll
4  * FILE:            lib/msgina/shutdown.c
5  * PURPOSE:         Shutdown Dialog Box (GUI only)
6  * PROGRAMMERS:     Lee Schroeder (spaceseel at gmail dot com)
7  *                  Hermes Belusca-Maito (hermes.belusca@sfr.fr)
8  *                  Arnav Bhatt (arnavbhatt288 at gmail dot com)
9  */
10 
11 #include "msgina.h"
12 #include <powrprof.h>
13 #include <wingdi.h>
14 #include <windowsx.h>
15 #include <commctrl.h>
16 
17 /* Shutdown state flags */
18 #define WLX_SHUTDOWN_STATE_LOGOFF       0x01
19 #define WLX_SHUTDOWN_STATE_POWER_OFF    0x02
20 #define WLX_SHUTDOWN_STATE_REBOOT       0x04
21 // 0x08
22 #define WLX_SHUTDOWN_STATE_SLEEP        0x10
23 // 0x20
24 #define WLX_SHUTDOWN_STATE_HIBERNATE    0x40
25 // 0x80
26 
27 /* Macros for fancy shut down dialog */
28 #define FONT_POINT_SIZE                 13
29 
30 #define DARK_GREY_COLOR                 RGB(244, 244, 244)
31 #define LIGHT_GREY_COLOR                RGB(38, 38, 38)
32 
33 /* Bitmap's size for buttons */
34 #define CX_BITMAP                       33
35 #define CY_BITMAP                       33
36 
37 #define NUMBER_OF_BUTTONS               4
38 
39 /* After determining the button as well as its state paint the image strip bitmap using these predefined positions */
40 #define BUTTON_SHUTDOWN                 0
41 #define BUTTON_SHUTDOWN_PRESSED         (CY_BITMAP + BUTTON_SHUTDOWN)
42 #define BUTTON_SHUTDOWN_FOCUSED         (CY_BITMAP + BUTTON_SHUTDOWN_PRESSED)
43 #define BUTTON_REBOOT                   (CY_BITMAP + BUTTON_SHUTDOWN_FOCUSED)
44 #define BUTTON_REBOOT_PRESSED           (CY_BITMAP + BUTTON_REBOOT)
45 #define BUTTON_REBOOT_FOCUSED           (CY_BITMAP + BUTTON_REBOOT_PRESSED)
46 #define BUTTON_SLEEP                    (CY_BITMAP + BUTTON_REBOOT_FOCUSED)
47 #define BUTTON_SLEEP_PRESSED            (CY_BITMAP + BUTTON_SLEEP)
48 #define BUTTON_SLEEP_FOCUSED            (CY_BITMAP + BUTTON_SLEEP_PRESSED)
49 #define BUTTON_SLEEP_DISABLED           (CY_BITMAP + BUTTON_SLEEP_FOCUSED)
50 
51 /* For bIsButtonHot */
52 #define SHUTDOWN_BUTTON_HOT             0
53 #define REBOOT_BUTTON_HOT               1
54 #define SLEEP_BUTTON_HOT                2
55 #define HIBERNATE_BUTTON_HOT            3
56 
57 typedef struct _SHUTDOWN_DLG_CONTEXT
58 {
59     PGINA_CONTEXT pgContext;
60     HBITMAP hBitmap;
61     HBITMAP hImageStrip;
62     DWORD ShutdownDialogId;
63     DWORD ShutdownOptions;
64     HBRUSH hBrush;
65     HFONT hfFont;
66     BOOL bCloseDlg;
67     BOOL bIsSleepButtonReplaced;
68     BOOL bReasonUI;
69     BOOL bFriendlyUI;
70     BOOL bIsButtonHot[NUMBER_OF_BUTTONS];
71     BOOL bTimer;
72     UINT_PTR iTimer;
73     WNDPROC OldButtonProc;
74 } SHUTDOWN_DLG_CONTEXT, *PSHUTDOWN_DLG_CONTEXT;
75 
76 static
77 BOOL
78 GetShutdownReasonUI(VOID)
79 {
80     OSVERSIONINFOEX VersionInfo;
81     DWORD dwValue, dwSize;
82     HKEY hKey;
83     LONG lRet;
84 
85     /* Query the policy value */
86     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
87                          L"Software\\Policies\\Microsoft\\Windows NT\\Reliability",
88                          0,
89                          KEY_QUERY_VALUE,
90                          &hKey);
91     if (lRet == ERROR_SUCCESS)
92     {
93         dwValue = 0;
94         dwSize = sizeof(dwValue);
95         RegQueryValueExW(hKey,
96                          L"ShutdownReasonUI",
97                          NULL,
98                          NULL,
99                          (LPBYTE)&dwValue,
100                          &dwSize);
101         RegCloseKey(hKey);
102 
103         return (dwValue != 0) ? TRUE : FALSE;
104     }
105 
106     /* Query the machine value */
107     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
108                          L"Software\\Microsoft\\Windows\\CurrentVersion\\Reliability",
109                          0,
110                          KEY_QUERY_VALUE,
111                          &hKey);
112     if (lRet == ERROR_SUCCESS)
113     {
114         dwValue = 0;
115         dwSize = sizeof(dwValue);
116         RegQueryValueExW(hKey,
117                          L"ShutdownReasonUI",
118                          NULL,
119                          NULL,
120                          (LPBYTE)&dwValue,
121                          &dwSize);
122         RegCloseKey(hKey);
123 
124         return (dwValue != 0) ? TRUE : FALSE;
125     }
126 
127     /* Return the default value */
128     VersionInfo.dwOSVersionInfoSize = sizeof(VersionInfo);
129     if (!GetVersionEx((POSVERSIONINFO)&VersionInfo))
130         return FALSE;
131 
132     return FALSE;
133 //    return (VersionInfo.wProductType == VER_NT_WORKSTATION) ? FALSE : TRUE;
134 }
135 
136 static
137 BOOL
138 IsFriendlyUIActive(VOID)
139 {
140     DWORD dwType, dwValue, dwSize;
141     HKEY hKey;
142     LONG lRet;
143 
144     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
145                          L"SYSTEM\\CurrentControlSet\\Control\\Windows",
146                          0,
147                          KEY_QUERY_VALUE,
148                          &hKey);
149     if (lRet != ERROR_SUCCESS)
150         return FALSE;
151 
152     /* CORE-17282 First check an optional ReactOS specific override, that Windows does not check.
153        We use this to allow users pairing 'Server'-configuration with FriendlyShutdown.
154        Otherwise users would have to change CSDVersion or LogonType (side-effects AppCompat) */
155     dwValue = 0;
156     dwSize = sizeof(dwValue);
157     lRet = RegQueryValueExW(hKey,
158                             L"EnforceFriendlyShutdown",
159                             NULL,
160                             &dwType,
161                             (LPBYTE)&dwValue,
162                             &dwSize);
163 
164     if (lRet == ERROR_SUCCESS && dwType == REG_DWORD && dwValue == 0x1)
165     {
166         RegCloseKey(hKey);
167         return TRUE;
168     }
169 
170     /* Check product version number */
171     dwValue = 0;
172     dwSize = sizeof(dwValue);
173     lRet = RegQueryValueExW(hKey,
174                             L"CSDVersion",
175                             NULL,
176                             &dwType,
177                             (LPBYTE)&dwValue,
178                             &dwSize);
179     RegCloseKey(hKey);
180 
181     if (lRet != ERROR_SUCCESS || dwType != REG_DWORD || dwValue != 0x300)
182     {
183         /* Allow Friendly UI only on Workstation */
184         return FALSE;
185     }
186 
187     /* Check LogonType value */
188     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
189                          L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon",
190                          0,
191                          KEY_QUERY_VALUE,
192                          &hKey);
193     if (lRet != ERROR_SUCCESS)
194         return FALSE;
195 
196     dwValue = 0;
197     dwSize = sizeof(dwValue);
198     lRet = RegQueryValueExW(hKey,
199                             L"LogonType",
200                             NULL,
201                             &dwType,
202                             (LPBYTE)&dwValue,
203                             &dwSize);
204     RegCloseKey(hKey);
205 
206     if (lRet != ERROR_SUCCESS || dwType != REG_DWORD)
207         return FALSE;
208 
209     return (dwValue != 0);
210 }
211 
212 static
213 BOOL
214 IsDomainMember(VOID)
215 {
216     UNIMPLEMENTED;
217     return FALSE;
218 }
219 
220 static
221 BOOL
222 IsNetwareActive(VOID)
223 {
224     UNIMPLEMENTED;
225     return FALSE;
226 }
227 
228 static
229 BOOL
230 IsShowHibernateButtonActive(VOID)
231 {
232     INT_PTR lRet;
233     HKEY hKey;
234     DWORD dwValue, dwSize;
235 
236     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
237                          L"SOFTWARE\\Policies\\Microsoft\\Windows\\System\\Shutdown",
238                          0, KEY_QUERY_VALUE, &hKey);
239     if (lRet == ERROR_SUCCESS)
240     {
241         dwValue = 0;
242         dwSize = sizeof(dwValue);
243 
244         lRet = RegQueryValueExW(hKey,
245                                 L"ShowHibernateButton",
246                                 NULL, NULL,
247                                 (LPBYTE)&dwValue, &dwSize);
248         RegCloseKey(hKey);
249         if (lRet != ERROR_SUCCESS)
250         {
251             return FALSE;
252         }
253         return (dwValue != 0);
254     }
255     return FALSE;
256 }
257 
258 static
259 BOOL
260 ForceFriendlyUI(VOID)
261 {
262     DWORD dwType, dwValue, dwSize;
263     HKEY hKey;
264     LONG lRet;
265 
266     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
267                          L"SOFTWARE\\Microsoft\\Windows\\CurrentVersion\\Policies\\System",
268                          0,
269                          KEY_QUERY_VALUE,
270                          &hKey);
271     if (lRet == ERROR_SUCCESS)
272     {
273         dwValue = 0;
274         dwSize = sizeof(dwValue);
275         lRet = RegQueryValueExW(hKey,
276                                 L"ForceFriendlyUI",
277                                 NULL,
278                                 &dwType,
279                                 (LPBYTE)&dwValue,
280                                 &dwSize);
281         RegCloseKey(hKey);
282 
283         if (lRet == ERROR_SUCCESS && dwType == REG_DWORD)
284             return (dwValue != 0);
285     }
286 
287     lRet = RegOpenKeyExW(HKEY_LOCAL_MACHINE,
288                          L"SOFTWARE\\Microsoft\\Windows NT\\CurrentVersion\\Winlogon",
289                          0,
290                          KEY_QUERY_VALUE,
291                          &hKey);
292     if (lRet == ERROR_SUCCESS)
293     {
294         dwValue = 0;
295         dwSize = sizeof(dwValue);
296         lRet = RegQueryValueExW(hKey,
297                                 L"ForceFriendlyUI",
298                                 NULL,
299                                 &dwType,
300                                 (LPBYTE)&dwValue,
301                                 &dwSize);
302 
303         RegCloseKey(hKey);
304 
305         if (lRet == ERROR_SUCCESS && dwType == REG_DWORD)
306             return (dwValue != 0);
307     }
308 
309     return FALSE;
310 }
311 
312 static
313 BOOL
314 DrawIconOnOwnerDrawnButtons(
315     DRAWITEMSTRUCT* pdis,
316     PSHUTDOWN_DLG_CONTEXT pContext)
317 {
318     BOOL bRet;
319     HDC hdcMem;
320     HBITMAP hbmOld;
321     int y;
322     RECT rect;
323 
324     hdcMem = CreateCompatibleDC(pdis->hDC);
325     hbmOld = SelectObject(hdcMem, pContext->hImageStrip);
326     rect = pdis->rcItem;
327 
328     /* Check the button ID for revelant bitmap to be used */
329     switch (pdis->CtlID)
330     {
331         case IDC_BUTTON_SHUTDOWN:
332         {
333             switch (pdis->itemAction)
334             {
335                 case ODA_DRAWENTIRE:
336                 case ODA_FOCUS:
337                 case ODA_SELECT:
338                 {
339                     y = BUTTON_SHUTDOWN;
340                     if (pdis->itemState & ODS_SELECTED)
341                     {
342                         y = BUTTON_SHUTDOWN_PRESSED;
343                     }
344                     else if (pContext->bIsButtonHot[SHUTDOWN_BUTTON_HOT] || (pdis->itemState & ODS_FOCUS))
345                     {
346                         y = BUTTON_SHUTDOWN_FOCUSED;
347                     }
348                     break;
349                 }
350             }
351             break;
352         }
353 
354         case IDC_BUTTON_REBOOT:
355         {
356             switch (pdis->itemAction)
357             {
358                 case ODA_DRAWENTIRE:
359                 case ODA_FOCUS:
360                 case ODA_SELECT:
361                 {
362                     y = BUTTON_REBOOT;
363                     if (pdis->itemState & ODS_SELECTED)
364                     {
365                         y = BUTTON_REBOOT_PRESSED;
366                     }
367                     else if (pContext->bIsButtonHot[REBOOT_BUTTON_HOT] || (pdis->itemState & ODS_FOCUS))
368                     {
369                         y = BUTTON_REBOOT_FOCUSED;
370                     }
371                     break;
372                 }
373             }
374             break;
375         }
376 
377         case IDC_BUTTON_HIBERNATE:
378         case IDC_BUTTON_SLEEP:
379         {
380             switch (pdis->itemAction)
381             {
382                 case ODA_DRAWENTIRE:
383                 case ODA_FOCUS:
384                 case ODA_SELECT:
385                 {
386                     y = BUTTON_SLEEP;
387                     if (pdis->itemState & ODS_DISABLED)
388                     {
389                         y = BUTTON_SLEEP_DISABLED;
390                     }
391                     else if (pdis->itemState & ODS_SELECTED)
392                     {
393                         y = BUTTON_SLEEP_PRESSED;
394                     }
395                     else if ((pdis->CtlID == IDC_BUTTON_SLEEP && pContext->bIsButtonHot[SLEEP_BUTTON_HOT]) ||
396                              (pdis->CtlID == IDC_BUTTON_HIBERNATE && pContext->bIsButtonHot[HIBERNATE_BUTTON_HOT]) ||
397                              (pdis->itemState & ODS_FOCUS))
398                     {
399                         y = BUTTON_SLEEP_FOCUSED;
400                     }
401                     break;
402                 }
403             }
404             break;
405         }
406     }
407 
408     /* Draw it on the required button */
409     bRet = BitBlt(pdis->hDC,
410                   (rect.right - rect.left - CX_BITMAP) / 2,
411                   (rect.bottom - rect.top - CY_BITMAP) / 2,
412                   CX_BITMAP, CY_BITMAP, hdcMem, 0, y, SRCCOPY);
413 
414     SelectObject(hdcMem, hbmOld);
415     DeleteDC(hdcMem);
416 
417     return bRet;
418 }
419 
420 BOOL
421 WINAPI
422 ShellIsFriendlyUIActive(VOID)
423 {
424     BOOL bActive;
425 
426     bActive = IsFriendlyUIActive();
427 
428     if ((IsDomainMember() || IsNetwareActive()) && !ForceFriendlyUI())
429         return FALSE;
430 
431     return bActive;
432 }
433 
434 DWORD
435 GetDefaultShutdownSelState(VOID)
436 {
437     return WLX_SAS_ACTION_SHUTDOWN_POWER_OFF;
438 }
439 
440 DWORD
441 LoadShutdownSelState(VOID)
442 {
443     LONG lRet;
444     HKEY hKeyCurrentUser, hKey;
445     DWORD dwValue, dwTemp, dwSize;
446 
447     /* Default to shutdown */
448     dwValue = WLX_SAS_ACTION_SHUTDOWN_POWER_OFF;
449 
450     /* Open the current user HKCU key */
451     lRet = RegOpenCurrentUser(MAXIMUM_ALLOWED, &hKeyCurrentUser);
452     if (lRet == ERROR_SUCCESS)
453     {
454         /* Open the subkey */
455         lRet = RegOpenKeyExW(hKeyCurrentUser,
456                              L"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer",
457                              0, KEY_QUERY_VALUE, &hKey);
458         RegCloseKey(hKeyCurrentUser);
459     }
460     if (lRet != ERROR_SUCCESS)
461         return dwValue;
462 
463     /* Read the value */
464     dwSize = sizeof(dwTemp);
465     lRet = RegQueryValueExW(hKey,
466                             L"Shutdown Setting",
467                             NULL, NULL,
468                             (LPBYTE)&dwTemp, &dwSize);
469     RegCloseKey(hKey);
470 
471     if (lRet == ERROR_SUCCESS)
472     {
473         switch (dwTemp)
474         {
475             case WLX_SHUTDOWN_STATE_LOGOFF:
476                 dwValue = WLX_SAS_ACTION_LOGOFF;
477                 break;
478 
479             case WLX_SHUTDOWN_STATE_POWER_OFF:
480                 dwValue = WLX_SAS_ACTION_SHUTDOWN_POWER_OFF;
481                 break;
482 
483             case WLX_SHUTDOWN_STATE_REBOOT:
484                 dwValue = WLX_SAS_ACTION_SHUTDOWN_REBOOT;
485                 break;
486 
487             // 0x08
488 
489             case WLX_SHUTDOWN_STATE_SLEEP:
490                 dwValue = WLX_SAS_ACTION_SHUTDOWN_SLEEP;
491                 break;
492 
493             // 0x20
494 
495             case WLX_SHUTDOWN_STATE_HIBERNATE:
496                 dwValue = WLX_SAS_ACTION_SHUTDOWN_HIBERNATE;
497                 break;
498 
499             // 0x80
500         }
501     }
502 
503     return dwValue;
504 }
505 
506 static INT_PTR
507 CALLBACK
508 OwnerDrawButtonSubclass(
509     HWND hButton,
510     UINT uMsg,
511     WPARAM wParam,
512     LPARAM lParam)
513 {
514     PSHUTDOWN_DLG_CONTEXT pContext;
515     pContext = (PSHUTDOWN_DLG_CONTEXT)GetWindowLongPtrW(GetParent(hButton), GWLP_USERDATA);
516     int buttonID = GetDlgCtrlID(hButton);
517 
518     switch (uMsg)
519     {
520         case WM_MOUSEMOVE:
521         {
522             HWND hwndTarget;
523             POINT pt = {GET_X_LPARAM(lParam), GET_Y_LPARAM(lParam)};
524 
525             if (GetCapture() != hButton)
526             {
527                 SetCapture(hButton);
528                 if (buttonID == IDC_BUTTON_SHUTDOWN)
529                 {
530                     pContext->bIsButtonHot[SHUTDOWN_BUTTON_HOT] = TRUE;
531                 }
532                 else if (buttonID == IDC_BUTTON_REBOOT)
533                 {
534                     pContext->bIsButtonHot[REBOOT_BUTTON_HOT] = TRUE;
535                 }
536                 else if (buttonID == IDC_BUTTON_SLEEP)
537                 {
538                     pContext->bIsButtonHot[SLEEP_BUTTON_HOT] = TRUE;
539                 }
540                 else if (buttonID == IDC_BUTTON_HIBERNATE)
541                 {
542                     pContext->bIsButtonHot[HIBERNATE_BUTTON_HOT] = TRUE;
543                 }
544                 SetCursor(LoadCursorW(NULL, IDC_HAND));
545             }
546 
547             ClientToScreen(hButton, &pt);
548             hwndTarget = WindowFromPoint(pt);
549 
550             if (hwndTarget != hButton)
551             {
552                 ReleaseCapture();
553                 if (buttonID == IDC_BUTTON_SHUTDOWN)
554                 {
555                     pContext->bIsButtonHot[SHUTDOWN_BUTTON_HOT] = FALSE;
556                 }
557                 else if (buttonID == IDC_BUTTON_REBOOT)
558                 {
559                     pContext->bIsButtonHot[REBOOT_BUTTON_HOT] = FALSE;
560                 }
561                 else if (buttonID == IDC_BUTTON_SLEEP)
562                 {
563                     pContext->bIsButtonHot[SLEEP_BUTTON_HOT] = FALSE;
564                 }
565                 else if (buttonID == IDC_BUTTON_HIBERNATE)
566                 {
567                     pContext->bIsButtonHot[HIBERNATE_BUTTON_HOT] = FALSE;
568                 }
569             }
570             InvalidateRect(hButton, NULL, FALSE);
571             break;
572         }
573 
574         /* Whenever one of the buttons gets the keyboard focus, set it as default button */
575         case WM_SETFOCUS:
576         {
577             SendMessageW(GetParent(hButton), DM_SETDEFID, buttonID, 0);
578             break;
579         }
580 
581         /* Otherwise, set IDCANCEL as default button */
582         case WM_KILLFOCUS:
583         {
584             SendMessageW(GetParent(hButton), DM_SETDEFID, IDCANCEL, 0);
585             break;
586         }
587     }
588     return CallWindowProcW(pContext->OldButtonProc, hButton, uMsg, wParam, lParam);
589 }
590 
591 VOID
592 CreateToolTipForButtons(
593     int controlID,
594     int detailID,
595     HWND hDlg,
596     int titleID,
597     HINSTANCE hInst)
598 {
599     HWND hwndTool, hwndTip;
600     WCHAR szBuffer[256];
601     TTTOOLINFOW tool;
602 
603     hwndTool = GetDlgItem(hDlg, controlID);
604 
605     tool.cbSize = sizeof(tool);
606     tool.hwnd = hDlg;
607     tool.uFlags = TTF_IDISHWND | TTF_SUBCLASS;
608     tool.uId = (UINT_PTR)hwndTool;
609 
610     /* Create the tooltip */
611     hwndTip = CreateWindowExW(0, TOOLTIPS_CLASSW, NULL,
612                              WS_POPUP | TTS_ALWAYSTIP | TTS_BALLOON,
613                              CW_USEDEFAULT, CW_USEDEFAULT,
614                              CW_USEDEFAULT, CW_USEDEFAULT,
615                              hDlg, NULL, hInst, NULL);
616 
617     /* Associate the tooltip with the tool. */
618     LoadStringW(hInst, detailID, szBuffer, _countof(szBuffer));
619     tool.lpszText = szBuffer;
620     SendMessageW(hwndTip, TTM_ADDTOOLW, 0, (LPARAM)&tool);
621     LoadStringW(hInst, titleID, szBuffer, _countof(szBuffer));
622     SendMessageW(hwndTip, TTM_SETTITLEW, TTI_NONE, (LPARAM)szBuffer);
623     SendMessageW(hwndTip, TTM_SETMAXTIPWIDTH, 0, 250);
624 }
625 
626 VOID
627 EndFriendlyDialog(
628     HWND hDlg,
629     PSHUTDOWN_DLG_CONTEXT pContext)
630 {
631     if (pContext->bTimer)
632     {
633         KillTimer(hDlg, pContext->iTimer);
634     }
635 
636     DeleteObject(pContext->hBitmap);
637     DeleteObject(pContext->hBrush);
638     DeleteObject(pContext->hImageStrip);
639     DeleteObject(pContext->hfFont);
640 
641     /* Remove the subclass from the buttons */
642     for (int i = 0; i < NUMBER_OF_BUTTONS; i++)
643     {
644         SetWindowLongPtrW(GetDlgItem(hDlg, IDC_BUTTON_SHUTDOWN + i),
645                           GWLP_WNDPROC,
646                           (LONG_PTR)pContext->OldButtonProc);
647     }
648 }
649 
650 VOID
651 ChangeRequiredButton(
652     HWND hDlg,
653     PSHUTDOWN_DLG_CONTEXT pContext)
654 {
655     int destID = IDC_BUTTON_SLEEP;
656     int targetedID = IDC_BUTTON_HIBERNATE;
657     HWND hwndDest, hwndTarget;
658     RECT rect;
659     WCHAR szBuffer[30];
660 
661     /* If the sleep button has been already replaced earlier, bring sleep button back to its original position */
662     if (pContext->bIsSleepButtonReplaced)
663     {
664         destID = IDC_BUTTON_HIBERNATE;
665         targetedID = IDC_BUTTON_SLEEP;
666     }
667 
668     hwndDest = GetDlgItem(hDlg, destID);
669     hwndTarget = GetDlgItem(hDlg, targetedID);
670 
671     /* Get the position of the destination button */
672     GetWindowRect(hwndDest, &rect);
673 
674     /* Get the corrected translated coordinates which is relative to the client window */
675     MapWindowPoints(HWND_DESKTOP, hDlg, (LPPOINT)&rect, sizeof(RECT)/sizeof(POINT));
676 
677     /* Set the position of targeted button and hide the destination button */
678     SetWindowPos(hwndTarget,
679                  HWND_TOP,
680                  rect.left, rect.top,
681                  0, 0,
682                  SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE);
683 
684     EnableWindow(hwndDest, FALSE);
685     ShowWindow(hwndDest, SW_HIDE);
686     EnableWindow(hwndTarget, TRUE);
687     ShowWindow(hwndTarget, SW_SHOW);
688     SetFocus(hwndTarget);
689 
690     if (!pContext->bIsSleepButtonReplaced)
691     {
692         LoadStringW(pContext->pgContext->hDllInstance, IDS_SHUTDOWN_HIBERNATE, szBuffer, _countof(szBuffer));
693         SetDlgItemTextW(hDlg, IDC_SLEEP_STATIC, szBuffer);
694     }
695     else
696     {
697         LoadStringW(pContext->pgContext->hDllInstance, IDS_SHUTDOWN_SLEEP, szBuffer, _countof(szBuffer));
698         SetDlgItemTextW(hDlg, IDC_SLEEP_STATIC, szBuffer);
699     }
700 
701     InvalidateRect(hDlg, NULL, FALSE);
702 }
703 
704 VOID OnTimer(
705     HWND hDlg,
706     PSHUTDOWN_DLG_CONTEXT pContext)
707 {
708     BOOL ReplaceButton = !!(GetKeyState(VK_SHIFT) & 0x8000);
709 
710     if (ReplaceButton && !pContext->bIsSleepButtonReplaced)
711     {
712         ChangeRequiredButton(hDlg, pContext);
713         pContext->bIsSleepButtonReplaced = TRUE;
714     }
715     else if (!ReplaceButton && pContext->bIsSleepButtonReplaced)
716     {
717         ChangeRequiredButton(hDlg, pContext);
718         pContext->bIsSleepButtonReplaced = FALSE;
719     }
720 }
721 
722 VOID
723 SaveShutdownSelState(
724     IN DWORD ShutdownCode)
725 {
726     LONG lRet;
727     HKEY hKeyCurrentUser, hKey;
728     DWORD dwValue = 0;
729 
730     /* Open the current user HKCU key */
731     lRet = RegOpenCurrentUser(MAXIMUM_ALLOWED, &hKeyCurrentUser);
732     if (lRet == ERROR_SUCCESS)
733     {
734         /* Create the subkey */
735         lRet = RegCreateKeyExW(hKeyCurrentUser,
736                                L"Software\\Microsoft\\Windows\\CurrentVersion\\Explorer",
737                                0, NULL,
738                                REG_OPTION_NON_VOLATILE,
739                                KEY_SET_VALUE,
740                                NULL, &hKey, NULL);
741         RegCloseKey(hKeyCurrentUser);
742     }
743     if (lRet != ERROR_SUCCESS)
744         return;
745 
746     switch (ShutdownCode)
747     {
748         case WLX_SAS_ACTION_LOGOFF:
749             dwValue = WLX_SHUTDOWN_STATE_LOGOFF;
750             break;
751 
752         case WLX_SAS_ACTION_SHUTDOWN_POWER_OFF:
753             dwValue = WLX_SHUTDOWN_STATE_POWER_OFF;
754             break;
755 
756         case WLX_SAS_ACTION_SHUTDOWN_REBOOT:
757             dwValue = WLX_SHUTDOWN_STATE_REBOOT;
758             break;
759 
760         case WLX_SAS_ACTION_SHUTDOWN_SLEEP:
761             dwValue = WLX_SHUTDOWN_STATE_SLEEP;
762             break;
763 
764         case WLX_SAS_ACTION_SHUTDOWN_HIBERNATE:
765             dwValue = WLX_SHUTDOWN_STATE_HIBERNATE;
766             break;
767     }
768 
769     RegSetValueExW(hKey,
770                    L"Shutdown Setting",
771                    0, REG_DWORD,
772                    (LPBYTE)&dwValue, sizeof(dwValue));
773     RegCloseKey(hKey);
774 }
775 
776 DWORD
777 GetDefaultShutdownOptions(VOID)
778 {
779     return WLX_SHUTDOWN_STATE_POWER_OFF | WLX_SHUTDOWN_STATE_REBOOT;
780 }
781 
782 DWORD
783 GetAllowedShutdownOptions(VOID)
784 {
785     DWORD Options = 0;
786 
787     // FIXME: Compute those options accordings to current user's rights!
788     Options |= WLX_SHUTDOWN_STATE_LOGOFF | WLX_SHUTDOWN_STATE_POWER_OFF | WLX_SHUTDOWN_STATE_REBOOT;
789 
790     if (IsPwrSuspendAllowed())
791         Options |= WLX_SHUTDOWN_STATE_SLEEP;
792 
793     if (IsPwrHibernateAllowed())
794         Options |= WLX_SHUTDOWN_STATE_HIBERNATE;
795 
796     return Options;
797 }
798 
799 static VOID
800 UpdateShutdownDesc(
801     IN HWND hDlg,
802     IN PSHUTDOWN_DLG_CONTEXT pContext) // HINSTANCE hInstance
803 {
804     UINT DescId = 0;
805     DWORD ShutdownCode;
806     WCHAR szBuffer[256];
807 
808     ShutdownCode = SendDlgItemMessageW(hDlg, IDC_SHUTDOWN_ACTION, CB_GETCURSEL, 0, 0);
809     if (ShutdownCode == CB_ERR) // Invalid selection
810         return;
811 
812     ShutdownCode = SendDlgItemMessageW(hDlg, IDC_SHUTDOWN_ACTION, CB_GETITEMDATA, ShutdownCode, 0);
813 
814     switch (ShutdownCode)
815     {
816         case WLX_SAS_ACTION_LOGOFF:
817             DescId = IDS_SHUTDOWN_LOGOFF_DESC;
818             break;
819 
820         case WLX_SAS_ACTION_SHUTDOWN_POWER_OFF:
821             DescId = IDS_SHUTDOWN_SHUTDOWN_DESC;
822             break;
823 
824         case WLX_SAS_ACTION_SHUTDOWN_REBOOT:
825             DescId = IDS_SHUTDOWN_RESTART_DESC;
826             break;
827 
828         case WLX_SAS_ACTION_SHUTDOWN_SLEEP:
829             DescId = IDS_SHUTDOWN_SLEEP_DESC;
830             break;
831 
832         case WLX_SAS_ACTION_SHUTDOWN_HIBERNATE:
833             DescId = IDS_SHUTDOWN_HIBERNATE_DESC;
834             break;
835 
836         default:
837             break;
838     }
839 
840     LoadStringW(pContext->pgContext->hDllInstance, DescId, szBuffer, _countof(szBuffer));
841     SetDlgItemTextW(hDlg, IDC_SHUTDOWN_DESCRIPTION, szBuffer);
842 
843     if (pContext->bReasonUI)
844     {
845         EnableWindow(GetDlgItem(hDlg, IDC_REASON_PLANNED), (ShutdownCode != WLX_SAS_ACTION_LOGOFF));
846         EnableWindow(GetDlgItem(hDlg, IDC_REASON_LIST), (ShutdownCode != WLX_SAS_ACTION_LOGOFF));
847         EnableWindow(GetDlgItem(hDlg, IDC_REASON_COMMENT), (ShutdownCode != WLX_SAS_ACTION_LOGOFF));
848     }
849 }
850 
851 static VOID
852 ShutdownOnFriendlyInit(
853     IN HWND hDlg,
854     IN PSHUTDOWN_DLG_CONTEXT pContext)
855 {
856     PGINA_CONTEXT pgContext = pContext->pgContext;
857     HDC hdc;
858     LONG lfHeight;
859 
860     /* Create font for the IDC_TURN_OFF_STATIC static control */
861     hdc = GetDC(hDlg);
862     lfHeight = -MulDiv(FONT_POINT_SIZE, GetDeviceCaps(hdc, LOGPIXELSY), 72);
863     ReleaseDC(hDlg, hdc);
864     pContext->hfFont = CreateFontW(lfHeight, 0, 0, 0, FW_MEDIUM, FALSE, 0, 0, 0, 0, 0, 0, 0, L"MS Shell Dlg");
865     SendDlgItemMessageW(hDlg, IDC_TURN_OFF_STATIC, WM_SETFONT, (WPARAM)pContext->hfFont, TRUE);
866 
867     /* Create a brush for static controls for fancy shut down dialog */
868     pContext->hBrush = CreateSolidBrush(DARK_GREY_COLOR);
869 
870     /* Gather image strip */
871     pContext->hImageStrip = LoadBitmapW(pgContext->hDllInstance, MAKEINTRESOURCEW(IDB_IMAGE_STRIP));
872 
873     /* Set the boolean flags to false */
874     pContext->bIsSleepButtonReplaced = FALSE;
875     pContext->bTimer = FALSE;
876 
877     EnableWindow(GetDlgItem(hDlg, IDC_BUTTON_HIBERNATE), FALSE);
878     EnableWindow(GetDlgItem(hDlg, IDC_BUTTON_SLEEP), IsPwrSuspendAllowed());
879 
880     /* Gather old button func */
881     pContext->OldButtonProc = (WNDPROC)GetWindowLongPtrW(GetDlgItem(hDlg, IDC_BUTTON_HIBERNATE), GWLP_WNDPROC);
882 
883     /* Set bIsButtonHot to false, create tooltips for each buttons, make buttons to remember pContext and subclass the buttons */
884     for (int i = 0; i < NUMBER_OF_BUTTONS; i++)
885     {
886         pContext->bIsButtonHot[i] = FALSE;
887         SetWindowLongPtrW(GetDlgItem(hDlg, IDC_BUTTON_SHUTDOWN + i),
888                           GWLP_WNDPROC,
889                           (LONG_PTR)OwnerDrawButtonSubclass);
890         CreateToolTipForButtons(IDC_BUTTON_SHUTDOWN + i,
891                                 IDS_SHUTDOWN_SHUTDOWN_DESC + i,
892                                 hDlg, IDS_SHUTDOWN_SHUTDOWN + i,
893                                 pContext->pgContext->hDllInstance);
894     }
895 
896     if (pContext->ShutdownDialogId == IDD_SHUTDOWN_FANCY && IsPwrSuspendAllowed())
897     {
898         pContext->iTimer = SetTimer(hDlg, 0, 50, NULL);
899         pContext->bTimer = TRUE;
900     }
901 }
902 
903 static VOID
904 ShutdownOnInit(
905     IN HWND hDlg,
906     IN PSHUTDOWN_DLG_CONTEXT pContext)
907 {
908     PGINA_CONTEXT pgContext = pContext->pgContext;
909     HWND hwndList;
910     INT idx, count, i;
911     WCHAR szBuffer[256];
912     WCHAR szBuffer2[256];
913 
914     if (pContext->bFriendlyUI)
915     {
916         ShutdownOnFriendlyInit(hDlg, pContext);
917         return;
918     }
919 
920     hwndList = GetDlgItem(hDlg, IDC_SHUTDOWN_ACTION);
921 
922     /* Clear the content before it's used */
923     SendMessageW(hwndList, CB_RESETCONTENT, 0, 0);
924 
925     /* Log off */
926     if (pContext->ShutdownOptions & WLX_SHUTDOWN_STATE_LOGOFF)
927     {
928         LoadStringW(pgContext->hDllInstance, IDS_SHUTDOWN_LOGOFF, szBuffer, _countof(szBuffer));
929         StringCchPrintfW(szBuffer2, _countof(szBuffer2), szBuffer, pgContext->UserName);
930         idx = SendMessageW(hwndList, CB_ADDSTRING, 0, (LPARAM)szBuffer2);
931         if (idx != CB_ERR)
932             SendMessageW(hwndList, CB_SETITEMDATA, idx, WLX_SAS_ACTION_LOGOFF);
933     }
934 
935     /* Shut down - DEFAULT */
936     if (pContext->ShutdownOptions & WLX_SHUTDOWN_STATE_POWER_OFF)
937     {
938         LoadStringW(pgContext->hDllInstance, IDS_SHUTDOWN_SHUTDOWN, szBuffer, _countof(szBuffer));
939         idx = SendMessageW(hwndList, CB_ADDSTRING, 0, (LPARAM)szBuffer);
940         if (idx != CB_ERR)
941             SendMessageW(hwndList, CB_SETITEMDATA, idx, WLX_SAS_ACTION_SHUTDOWN_POWER_OFF);
942     }
943 
944     /* Restart */
945     if (pContext->ShutdownOptions & WLX_SHUTDOWN_STATE_REBOOT)
946     {
947         LoadStringW(pgContext->hDllInstance, IDS_SHUTDOWN_RESTART, szBuffer, _countof(szBuffer));
948         idx = SendMessageW(hwndList, CB_ADDSTRING, 0, (LPARAM)szBuffer);
949         if (idx != CB_ERR)
950             SendMessageW(hwndList, CB_SETITEMDATA, idx, WLX_SAS_ACTION_SHUTDOWN_REBOOT);
951     }
952 
953     // if (pContext->ShutdownOptions & 0x08) {}
954 
955     /* Sleep */
956     if (pContext->ShutdownOptions & WLX_SHUTDOWN_STATE_SLEEP)
957     {
958         LoadStringW(pgContext->hDllInstance, IDS_SHUTDOWN_SLEEP, szBuffer, _countof(szBuffer));
959         idx = SendMessageW(hwndList, CB_ADDSTRING, 0, (LPARAM)szBuffer);
960         if (idx != CB_ERR)
961             SendMessageW(hwndList, CB_SETITEMDATA, idx, WLX_SAS_ACTION_SHUTDOWN_SLEEP);
962     }
963 
964     // if (pContext->ShutdownOptions & 0x20) {}
965 
966     /* Hibernate */
967     if (pContext->ShutdownOptions & WLX_SHUTDOWN_STATE_HIBERNATE)
968     {
969         LoadStringW(pgContext->hDllInstance, IDS_SHUTDOWN_HIBERNATE, szBuffer, _countof(szBuffer));
970         idx = SendMessageW(hwndList, CB_ADDSTRING, 0, (LPARAM)szBuffer);
971         if (idx != CB_ERR)
972             SendMessageW(hwndList, CB_SETITEMDATA, idx, WLX_SAS_ACTION_SHUTDOWN_HIBERNATE);
973     }
974 
975     // if (pContext->ShutdownOptions & 0x80) {}
976 
977     /* Set the default shut down selection */
978     count = SendMessageW(hwndList, CB_GETCOUNT, 0, 0);
979     for (i = 0; i < count; i++)
980     {
981         if (SendMessageW(hwndList, CB_GETITEMDATA, i, 0) == pgContext->nShutdownAction)
982         {
983             SendMessageW(hwndList, CB_SETCURSEL, i, 0);
984             break;
985         }
986     }
987 
988     /* Update the choice description based on the current selection */
989     UpdateShutdownDesc(hDlg, pContext);
990 }
991 
992 static VOID
993 ShutdownOnOk(
994     IN HWND hDlg,
995     IN PGINA_CONTEXT pgContext)
996 {
997     INT idx;
998 
999     idx = SendDlgItemMessageW(hDlg,
1000                               IDC_SHUTDOWN_ACTION,
1001                               CB_GETCURSEL,
1002                               0,
1003                               0);
1004     if (idx != CB_ERR)
1005     {
1006         pgContext->nShutdownAction =
1007             SendDlgItemMessageW(hDlg,
1008                                 IDC_SHUTDOWN_ACTION,
1009                                 CB_GETITEMDATA,
1010                                 idx,
1011                                 0);
1012     }
1013 }
1014 
1015 static INT_PTR
1016 CALLBACK
1017 ShutdownDialogProc(
1018     HWND hDlg,
1019     UINT uMsg,
1020     WPARAM wParam,
1021     LPARAM lParam)
1022 {
1023     PSHUTDOWN_DLG_CONTEXT pContext;
1024 
1025     pContext = (PSHUTDOWN_DLG_CONTEXT)GetWindowLongPtrW(hDlg, GWLP_USERDATA);
1026 
1027     switch (uMsg)
1028     {
1029         case WM_INITDIALOG:
1030         {
1031             pContext = (PSHUTDOWN_DLG_CONTEXT)lParam;
1032             SetWindowLongPtrW(hDlg, GWLP_USERDATA, (LONG_PTR)pContext);
1033 
1034             ShutdownOnInit(hDlg, pContext);
1035             return TRUE;
1036         }
1037 
1038         case WM_DESTROY:
1039             if (pContext->bFriendlyUI)
1040             {
1041                 EndFriendlyDialog(hDlg, pContext);
1042             }
1043             return TRUE;
1044 
1045         case WM_ACTIVATE:
1046         {
1047             /*
1048              * If the user deactivates the shutdown dialog (it loses its focus
1049              * while the dialog is not being closed), then destroy the dialog
1050              * and cancel shutdown.
1051              */
1052             if (LOWORD(wParam) == WA_INACTIVE)
1053             {
1054                 if (!pContext->bCloseDlg)
1055                 {
1056                     pContext->bCloseDlg = TRUE;
1057                     EndDialog(hDlg, IDCANCEL);
1058                 }
1059             }
1060             return FALSE;
1061         }
1062 
1063         case WM_CLOSE:
1064             pContext->bCloseDlg = TRUE;
1065             EndDialog(hDlg, IDCANCEL);
1066             break;
1067 
1068         case WM_COMMAND:
1069             switch (LOWORD(wParam))
1070             {
1071                 case IDC_BUTTON_SHUTDOWN:
1072                     ExitWindowsEx(EWX_SHUTDOWN, SHTDN_REASON_MAJOR_OTHER);
1073                     break;
1074 
1075                 case IDC_BUTTON_REBOOT:
1076                     ExitWindowsEx(EWX_REBOOT, SHTDN_REASON_MAJOR_OTHER);
1077                     break;
1078 
1079                 case IDC_BUTTON_SLEEP:
1080                     SetSuspendState(TRUE, TRUE, TRUE);
1081                     break;
1082 
1083                 case IDOK:
1084                     ShutdownOnOk(hDlg, pContext->pgContext);
1085 
1086                 /* Fall back */
1087                 case IDCANCEL:
1088                 case IDHELP:
1089                     pContext->bCloseDlg = TRUE;
1090                     EndDialog(hDlg, LOWORD(wParam));
1091                     break;
1092 
1093                 case IDC_SHUTDOWN_ACTION:
1094                     UpdateShutdownDesc(hDlg, pContext);
1095                     break;
1096             }
1097             break;
1098 
1099         case WM_CTLCOLORSTATIC:
1100         {
1101             /* Either make background transparent or fill it with color for required static controls */
1102             HDC hdcStatic = (HDC)wParam;
1103             UINT StaticID = (UINT)GetWindowLongPtrW((HWND)lParam, GWL_ID);
1104 
1105             switch (StaticID)
1106             {
1107                 case IDC_TURN_OFF_STATIC:
1108                    SetTextColor(hdcStatic, DARK_GREY_COLOR);
1109                    SetBkMode(hdcStatic, TRANSPARENT);
1110                    return (INT_PTR)GetStockObject(HOLLOW_BRUSH);
1111 
1112                 case IDC_HIBERNATE_STATIC:
1113                 case IDC_SHUTDOWN_STATIC:
1114                 case IDC_SLEEP_STATIC:
1115                 case IDC_RESTART_STATIC:
1116                     SetTextColor(hdcStatic, LIGHT_GREY_COLOR);
1117                     SetBkMode(hdcStatic, TRANSPARENT);
1118                     return (LONG_PTR)pContext->hBrush;
1119             }
1120             return FALSE;
1121         }
1122 
1123         case WM_DRAWITEM:
1124         {
1125             /* Draw bitmaps on required buttons */
1126             DRAWITEMSTRUCT* pdis = (DRAWITEMSTRUCT*)lParam;
1127             switch (pdis->CtlID)
1128             {
1129                 case IDC_BUTTON_SHUTDOWN:
1130                 case IDC_BUTTON_REBOOT:
1131                 case IDC_BUTTON_SLEEP:
1132                 case IDC_BUTTON_HIBERNATE:
1133                     return DrawIconOnOwnerDrawnButtons(pdis, pContext);
1134             }
1135             break;
1136         }
1137 
1138         case WM_TIMER:
1139             OnTimer(hDlg, pContext);
1140             return TRUE;
1141 
1142         default:
1143             return FALSE;
1144     }
1145     return TRUE;
1146 }
1147 
1148 INT_PTR
1149 ShutdownDialog(
1150     IN HWND hwndDlg,
1151     IN DWORD ShutdownOptions,
1152     IN PGINA_CONTEXT pgContext)
1153 {
1154     INT_PTR ret;
1155     SHUTDOWN_DLG_CONTEXT Context;
1156 
1157 #if 0
1158     DWORD ShutdownOptions;
1159 
1160     // FIXME: User impersonation!!
1161     pgContext->nShutdownAction = LoadShutdownSelState();
1162     ShutdownOptions = GetAllowedShutdownOptions();
1163 #endif
1164 
1165     Context.pgContext = pgContext;
1166     Context.ShutdownOptions = ShutdownOptions;
1167     Context.ShutdownDialogId = IDD_SHUTDOWN;
1168     Context.bCloseDlg = FALSE;
1169     Context.bReasonUI = GetShutdownReasonUI();
1170     Context.bFriendlyUI = ShellIsFriendlyUIActive();
1171 
1172     if (pgContext->hWlx && pgContext->pWlxFuncs && !Context.bFriendlyUI)
1173     {
1174         ret = pgContext->pWlxFuncs->WlxDialogBoxParam(pgContext->hWlx,
1175                                                       pgContext->hDllInstance,
1176                                                       MAKEINTRESOURCEW(Context.bReasonUI ? IDD_SHUTDOWN_REASON : IDD_SHUTDOWN),
1177                                                       hwndDlg,
1178                                                       ShutdownDialogProc,
1179                                                       (LPARAM)&Context);
1180     }
1181     else
1182     {
1183         if (Context.bFriendlyUI)
1184         {
1185             if (IsShowHibernateButtonActive())
1186             {
1187                 Context.ShutdownDialogId = IDD_SHUTDOWN_FANCY_LONG;
1188             }
1189             else
1190             {
1191                 Context.ShutdownDialogId = IDD_SHUTDOWN_FANCY;
1192             }
1193         }
1194 
1195         ret = DialogBoxParamW(pgContext->hDllInstance,
1196                               MAKEINTRESOURCEW(Context.bReasonUI ? IDD_SHUTDOWN_REASON : Context.ShutdownDialogId),
1197                               hwndDlg,
1198                               ShutdownDialogProc,
1199                               (LPARAM)&Context);
1200     }
1201 
1202 #if 0
1203     // FIXME: User impersonation!!
1204     if (ret == IDOK)
1205         SaveShutdownSelState(pgContext->nShutdownAction);
1206 #endif
1207 
1208     return ret;
1209 }
1210 
1211 
1212 /*
1213  * NOTES:
1214  * - Based upon observations on the ShellShutdownDialog() function, the function doesn't actually
1215  *   do anything except show a dialog box and returning a value based upon the value chosen. That
1216  *   means that any code that calls the function has to execute the chosen action (shut down,
1217  *   restart, etc.).
1218  * - When this function is called in Windows XP, it shows the classic dialog box regardless if
1219  *   SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon\LogonType is enabled or not.
1220  * - When the Help button is pushed, it sends the same return value as IDCANCEL (0x00), but
1221  *   at the same time, it calls the help file directly from the dialog box.
1222  * - When the dialog is created, it doesn't disable all other input from the other windows.
1223  *   This is done elsewhere. When running the function ShellShutdownDialog() from XP/2K3, if the user clicks
1224  *   out of the window, it automatically closes itself.
1225  * - The parameter, lpUsername never seems to be used when calling the function from Windows XP. Either
1226  *   it was a parameter that was never used in the final version before release, or it has a use that
1227  *   is currently not known.
1228  */
1229 DWORD WINAPI
1230 ShellShutdownDialog(
1231     HWND   hParent,
1232     LPWSTR lpUsername,
1233     BOOL   bHideLogoff)
1234 {
1235     INT_PTR dlgValue;
1236     DWORD ShutdownOptions;
1237 
1238     /*
1239      * As we are called by the shell itself, don't use
1240      * the cached GINA context but use a local copy here.
1241      */
1242     GINA_CONTEXT gContext = { 0 };
1243     DWORD BufferSize;
1244 
1245     UNREFERENCED_PARAMETER(lpUsername);
1246 
1247     ShutdownOptions = GetAllowedShutdownOptions();
1248     if (bHideLogoff)
1249         ShutdownOptions &= ~WLX_SHUTDOWN_STATE_LOGOFF;
1250 
1251     /* Initialize our local GINA context */
1252     gContext.hDllInstance = hDllInstance;
1253     BufferSize = _countof(gContext.UserName);
1254     // NOTE: Only when this function is called, Win checks inside
1255     // HKLM\\Software\\Microsoft\\Windows\\CurrentVersion\\Explorer",
1256     // value "Logon User Name", and determines whether it will display
1257     // the user name.
1258     GetUserNameW(gContext.UserName, &BufferSize);
1259     gContext.nShutdownAction = LoadShutdownSelState();
1260 
1261     /* Load the shutdown dialog box */
1262     dlgValue = ShutdownDialog(hParent, ShutdownOptions, &gContext);
1263 
1264     /* Determine what to do based on user selection */
1265     if (dlgValue == IDOK)
1266     {
1267         SaveShutdownSelState(gContext.nShutdownAction);
1268 
1269         switch (gContext.nShutdownAction)
1270         {
1271             case WLX_SAS_ACTION_LOGOFF:
1272                 return WLX_SHUTDOWN_STATE_LOGOFF;
1273 
1274             case WLX_SAS_ACTION_SHUTDOWN_POWER_OFF:
1275                 return WLX_SHUTDOWN_STATE_POWER_OFF;
1276 
1277             case WLX_SAS_ACTION_SHUTDOWN_REBOOT:
1278                 return WLX_SHUTDOWN_STATE_REBOOT;
1279 
1280             // 0x08
1281 
1282             case WLX_SAS_ACTION_SHUTDOWN_SLEEP:
1283                 return WLX_SHUTDOWN_STATE_SLEEP;
1284 
1285             // 0x20
1286 
1287             case WLX_SAS_ACTION_SHUTDOWN_HIBERNATE:
1288                 return WLX_SHUTDOWN_STATE_HIBERNATE;
1289 
1290             // 0x80
1291         }
1292     }
1293     /* Help file is called directly here */
1294     else if (dlgValue == IDHELP)
1295     {
1296         FIXME("Help is not implemented yet.\n");
1297         MessageBoxW(hParent, L"Help is not implemented yet.", L"Message", MB_OK | MB_ICONEXCLAMATION);
1298     }
1299     else if (dlgValue == -1)
1300     {
1301         ERR("Failed to create dialog\n");
1302     }
1303 
1304     return 0;
1305 }
1306 
1307 /*
1308  * NOTES:
1309  * - Undocumented, called from MS shell32.dll to show the turn off dialog.
1310  * - Seems to have the same purpose as ShellShutdownDialog.
1311  */
1312 DWORD WINAPI
1313 ShellTurnOffDialog(HWND hWnd)
1314 {
1315     return ShellShutdownDialog(hWnd, NULL, FALSE);
1316 }
1317