3 * reactos/subsys/csrss/win32csr/desktopbg.c
5 * Desktop background window functions
7 * ReactOS Operating System
15 #define DESKTOP_WINDOW_ATOM 32769
17 #define PM_SHOW_DESKTOP 1
18 #define PM_HIDE_DESKTOP 2
20 typedef struct tagDTBG_THREAD_DATA
25 } DTBG_THREAD_DATA
, *PDTBG_THREAD_DATA
;
27 typedef struct tagPRIVATE_NOTIFY_DESKTOP
30 struct /* PM_SHOW_DESKTOP */
35 } PRIVATE_NOTIFY_DESKTOP
, *PPRIVATE_NOTIFY_DESKTOP
;
37 static BOOL BgInitialized
= FALSE
;
38 static HWND VisibleDesktopWindow
= NULL
;
43 DtbgWindowProc(HWND Wnd
,
53 PaintDesktop((HDC
)wParam
);
57 if (BeginPaint(Wnd
, &PS
))
62 return (LRESULT
)SetCursor(LoadCursorW(0, (LPCWSTR
)IDC_ARROW
));
71 case WM_DISPLAYCHANGE
:
72 MoveWindow(Wnd
, 0, 0, LOWORD(lParam
), HIWORD(lParam
), TRUE
);
77 PPRIVATE_NOTIFY_DESKTOP nmh
= (PPRIVATE_NOTIFY_DESKTOP
)lParam
;
79 /* Use WM_NOTIFY for private messages since
80 * it can't be sent between processes!
82 switch (nmh
->hdr
.code
)
88 Result
= !SetWindowPos(Wnd
, NULL
, 0, 0,
89 nmh
->ShowDesktop
.Width
,
90 nmh
->ShowDesktop
.Height
,
91 SWP_NOACTIVATE
| SWP_NOZORDER
|
95 VisibleDesktopWindow
= Wnd
;
103 Result
= !SetWindowPos(Wnd
, NULL
, 0, 0, 0, 0,
104 SWP_NOACTIVATE
| SWP_NOZORDER
|
105 SWP_NOMOVE
| SWP_NOSIZE
| SWP_HIDEWINDOW
);
108 VisibleDesktopWindow
= NULL
;
113 DPRINT("Unknown notification code 0x%x sent to the desktop window!\n",
120 return DefWindowProcW(Wnd
, Msg
, wParam
, lParam
);
135 * Create the desktop window class
137 Class
.cbSize
= sizeof(WNDCLASSEXW
);
138 Class
.style
= CS_GLOBALCLASS
;
139 Class
.lpfnWndProc
= DtbgWindowProc
;
140 Class
.cbClsExtra
= 0;
141 Class
.cbWndExtra
= 0;
142 Class
.hInstance
= (HINSTANCE
)GetModuleHandleW(NULL
);
144 Class
.hCursor
= NULL
;
145 Class
.hbrBackground
= GetSysColorBrush(COLOR_BACKGROUND
);
146 Class
.lpszMenuName
= NULL
;
147 Class
.lpszClassName
= (LPCWSTR
)DESKTOP_WINDOW_ATOM
;
148 ClassAtom
= RegisterClassExW(&Class
);
150 if (ClassAtom
== INVALID_ATOM
)
152 DPRINT1("Unable to register desktop background class (error %d)\n",
158 VisibleDesktopWindow
= NULL
;
166 DtbgDesktopThread(PVOID Data
)
170 PDTBG_THREAD_DATA ThreadData
= (PDTBG_THREAD_DATA
)Data
;
172 if (!SetThreadDesktop(ThreadData
->Desktop
))
174 DPRINT1("Failed to set thread desktop\n");
175 ThreadData
->Status
= STATUS_UNSUCCESSFUL
;
176 SetEvent(ThreadData
->Event
);
180 BackgroundWnd
= CreateWindowW((LPCWSTR
)DESKTOP_WINDOW_ATOM
,
182 WS_POPUP
| WS_CLIPCHILDREN
,
185 (HINSTANCE
)GetModuleHandleW(NULL
),
188 if (NULL
== BackgroundWnd
)
190 DPRINT1("Failed to create desktop background window\n");
191 ThreadData
->Status
= STATUS_UNSUCCESSFUL
;
192 SetEvent(ThreadData
->Event
);
196 ThreadData
->Status
= STATUS_SUCCESS
;
197 SetEvent(ThreadData
->Event
);
199 while (GetMessageW(&msg
, NULL
, 0, 0))
201 TranslateMessage(&msg
);
202 DispatchMessageW(&msg
);
208 CSR_API(CsrCreateDesktop
)
210 DTBG_THREAD_DATA ThreadData
;
213 DPRINT("CsrCreateDesktop\n");
215 Request
->Header
.u1
.s1
.TotalLength
= sizeof(CSR_API_MESSAGE
);
216 Request
->Header
.u1
.s1
.DataLength
= sizeof(CSR_API_MESSAGE
) - sizeof(PORT_MESSAGE
);
220 BgInitialized
= TRUE
;
223 return STATUS_UNSUCCESSFUL
;
227 * The desktop handle we got from win32k is in
228 * the scope of CSRSS so we can just use it.
230 ThreadData
.Desktop
= Request
->Data
.CreateDesktopRequest
.DesktopHandle
;
232 ThreadData
.Event
= CreateEventW(NULL
, FALSE
, FALSE
, NULL
);
234 if (NULL
== ThreadData
.Event
)
236 DPRINT1("Failed to create event (error %d)\n", GetLastError());
237 return STATUS_UNSUCCESSFUL
;
240 ThreadHandle
= CreateThread(NULL
,
247 if (NULL
== ThreadHandle
)
249 CloseHandle(ThreadData
.Event
);
250 DPRINT1("Failed to create desktop window thread.\n");
251 return STATUS_UNSUCCESSFUL
;
254 CloseHandle(ThreadHandle
);
256 WaitForSingleObject(ThreadData
.Event
, INFINITE
);
257 CloseHandle(ThreadData
.Event
);
259 return ThreadData
.Status
;
262 CSR_API(CsrShowDesktop
)
264 PRIVATE_NOTIFY_DESKTOP nmh
;
265 DPRINT("CsrShowDesktop\n");
267 Request
->Header
.u1
.s1
.TotalLength
= sizeof(CSR_API_MESSAGE
);
268 Request
->Header
.u1
.s1
.DataLength
= sizeof(CSR_API_MESSAGE
) - sizeof(PORT_MESSAGE
);
270 nmh
.hdr
.hwndFrom
= Request
->Data
.ShowDesktopRequest
.DesktopWindow
;
272 nmh
.hdr
.code
= PM_SHOW_DESKTOP
;
274 nmh
.ShowDesktop
.Width
= (int)Request
->Data
.ShowDesktopRequest
.Width
;
275 nmh
.ShowDesktop
.Height
= (int)Request
->Data
.ShowDesktopRequest
.Height
;
277 if (SendMessageW(Request
->Data
.ShowDesktopRequest
.DesktopWindow
,
279 (WPARAM
)nmh
.hdr
.hwndFrom
,
282 return STATUS_UNSUCCESSFUL
;
285 return STATUS_SUCCESS
;
288 CSR_API(CsrHideDesktop
)
290 PRIVATE_NOTIFY_DESKTOP nmh
;
291 DPRINT("CsrHideDesktop\n");
293 Request
->Header
.u1
.s1
.TotalLength
= sizeof(CSR_API_MESSAGE
);
294 Request
->Header
.u1
.s1
.DataLength
= sizeof(CSR_API_MESSAGE
) - sizeof(PORT_MESSAGE
);
296 nmh
.hdr
.hwndFrom
= Request
->Data
.ShowDesktopRequest
.DesktopWindow
;
298 nmh
.hdr
.code
= PM_HIDE_DESKTOP
;
300 if (SendMessageW(Request
->Data
.ShowDesktopRequest
.DesktopWindow
,
302 (WPARAM
)nmh
.hdr
.hwndFrom
,
305 return STATUS_UNSUCCESSFUL
;
308 return STATUS_SUCCESS
;
313 DtbgIsDesktopVisible(VOID
)
315 if (VisibleDesktopWindow
!= NULL
&&
316 !IsWindowVisible(VisibleDesktopWindow
))
318 VisibleDesktopWindow
= NULL
;
321 return VisibleDesktopWindow
!= NULL
;