[REACTOS]
[reactos.git] / reactos / win32ss / user / user32 / misc / desktop.c
1 /*
2 * COPYRIGHT: See COPYING in the top level directory
3 * PROJECT: ReactOS user32.dll
4 * FILE: lib/user32/misc/desktop.c
5 * PURPOSE: Desktops
6 * PROGRAMMER: Casper S. Hornstrup (chorns@users.sourceforge.net)
7 * UPDATE HISTORY:
8 * 06-06-2001 CSH Created
9 */
10
11 #include <user32.h>
12
13 #include <wine/debug.h>
14 WINE_DEFAULT_DEBUG_CHANNEL(user32);
15
16 /*********************************************************************
17 * desktop class descriptor
18 */
19 #if 0 // Kept for referencing.
20 const struct builtin_class_descr DESKTOP_builtin_class =
21 {
22 WC_DESKTOP, /* name */
23 CS_DBLCLKS, /* style */
24 NULL, /* procA (winproc is Unicode only) */
25 (WNDPROC) DesktopWndProc, /* procW */
26 0, /* extra */
27 IDC_ARROW, /* cursor */
28 (HBRUSH)(COLOR_BACKGROUND+1) /* brush */
29 };
30 #endif
31
32 LRESULT
33 WINAPI
34 DesktopWndProcW(HWND Wnd,
35 UINT Msg,
36 WPARAM wParam,
37 LPARAM lParam)
38 {
39 TRACE("Desktop W Class Atom! hWnd 0x%x, Msg %d\n", Wnd, Msg);
40
41 switch(Msg)
42 {
43 case WM_ERASEBKGND:
44 case WM_NCCREATE:
45 case WM_CREATE:
46 case WM_CLOSE:
47 case WM_DISPLAYCHANGE:
48 case WM_PAINT:
49 case WM_SYSCOLORCHANGE:
50 {
51 LRESULT lResult;
52 NtUserMessageCall( Wnd, Msg, wParam, lParam, (ULONG_PTR)&lResult, FNID_DESKTOP, FALSE);
53 TRACE("Desktop lResult %d\n", lResult);
54 return lResult;
55 }
56
57 case WM_PALETTECHANGED:
58 if (Wnd == (HWND)wParam) break;
59 case WM_QUERYNEWPALETTE:
60 {
61 HDC hdc = GetWindowDC( Wnd );
62 PaintDesktop(hdc);
63 ReleaseDC( Wnd, hdc );
64 break;
65 }
66
67 case WM_SETCURSOR:
68 return (LRESULT)SetCursor(LoadCursorW(0, (LPCWSTR)IDC_ARROW));
69
70 default:
71 return DefWindowProcW(Wnd, Msg, wParam, lParam);
72 }
73 return 0;
74 }
75
76 VOID
77 WINAPI
78 LogFontA2W(LPLOGFONTW pW, CONST LOGFONTA *pA)
79 {
80 #define COPYS(f,len) MultiByteToWideChar ( CP_THREAD_ACP, 0, pA->f, len, pW->f, len )
81 #define COPYN(f) pW->f = pA->f
82
83 COPYN(lfHeight);
84 COPYN(lfWidth);
85 COPYN(lfEscapement);
86 COPYN(lfOrientation);
87 COPYN(lfWeight);
88 COPYN(lfItalic);
89 COPYN(lfUnderline);
90 COPYN(lfStrikeOut);
91 COPYN(lfCharSet);
92 COPYN(lfOutPrecision);
93 COPYN(lfClipPrecision);
94 COPYN(lfQuality);
95 COPYN(lfPitchAndFamily);
96 COPYS(lfFaceName,LF_FACESIZE);
97
98 #undef COPYN
99 #undef COPYS
100 }
101
102 VOID
103 WINAPI
104 LogFontW2A(LPLOGFONTA pA, CONST LOGFONTW *pW)
105 {
106 #define COPYS(f,len) WideCharToMultiByte ( CP_THREAD_ACP, 0, pW->f, len, pA->f, len, NULL, NULL )
107 #define COPYN(f) pA->f = pW->f
108
109 COPYN(lfHeight);
110 COPYN(lfWidth);
111 COPYN(lfEscapement);
112 COPYN(lfOrientation);
113 COPYN(lfWeight);
114 COPYN(lfItalic);
115 COPYN(lfUnderline);
116 COPYN(lfStrikeOut);
117 COPYN(lfCharSet);
118 COPYN(lfOutPrecision);
119 COPYN(lfClipPrecision);
120 COPYN(lfQuality);
121 COPYN(lfPitchAndFamily);
122 COPYS(lfFaceName,LF_FACESIZE);
123
124 #undef COPYN
125 #undef COPYS
126 }
127
128 int WINAPI
129 RealGetSystemMetrics(int nIndex)
130 {
131 GetConnected();
132 // FIXME("Global Server Data -> %x\n",gpsi);
133 if (nIndex < 0 || nIndex >= SM_CMETRICS) return 0;
134 return gpsi->aiSysMet[nIndex];
135 }
136
137 /*
138 * @implemented
139 */
140 int WINAPI
141 GetSystemMetrics(int nIndex)
142 {
143 BOOL Hook;
144 int Ret = 0;
145
146 if (!gpsi) // Fixme! Hax! Need Timos delay load support?
147 {
148 return RealGetSystemMetrics(nIndex);
149 }
150
151 LoadUserApiHook();
152
153 Hook = BeginIfHookedUserApiHook();
154
155 /* Bypass SEH and go direct. */
156 if (!Hook) return RealGetSystemMetrics(nIndex);
157
158 _SEH2_TRY
159 {
160 Ret = guah.GetSystemMetrics(nIndex);
161 }
162 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
163 {
164 }
165 _SEH2_END;
166
167 EndUserApiHook();
168
169 return Ret;
170 }
171
172 /*
173 * @unimplemented
174 */
175 BOOL WINAPI SetDeskWallpaper(LPCSTR filename)
176 {
177 return SystemParametersInfoA(SPI_SETDESKWALLPAPER,0,(PVOID)filename,TRUE);
178 }
179
180 BOOL WINAPI
181 RealSystemParametersInfoA(UINT uiAction,
182 UINT uiParam,
183 PVOID pvParam,
184 UINT fWinIni)
185 {
186 switch (uiAction)
187 {
188
189 case SPI_GETNONCLIENTMETRICS:
190 {
191 LPNONCLIENTMETRICSA pnclma = (LPNONCLIENTMETRICSA)pvParam;
192 NONCLIENTMETRICSW nclmw;
193 if(pnclma->cbSize != sizeof(NONCLIENTMETRICSA))
194 {
195 SetLastError(ERROR_INVALID_PARAMETER);
196 return FALSE;
197 }
198 nclmw.cbSize = sizeof(NONCLIENTMETRICSW);
199
200 if (!SystemParametersInfoW(uiAction, sizeof(NONCLIENTMETRICSW),
201 &nclmw, fWinIni))
202 return FALSE;
203
204 pnclma->iBorderWidth = nclmw.iBorderWidth;
205 pnclma->iScrollWidth = nclmw.iScrollWidth;
206 pnclma->iScrollHeight = nclmw.iScrollHeight;
207 pnclma->iCaptionWidth = nclmw.iCaptionWidth;
208 pnclma->iCaptionHeight = nclmw.iCaptionHeight;
209 pnclma->iSmCaptionWidth = nclmw.iSmCaptionWidth;
210 pnclma->iSmCaptionHeight = nclmw.iSmCaptionHeight;
211 pnclma->iMenuWidth = nclmw.iMenuWidth;
212 pnclma->iMenuHeight = nclmw.iMenuHeight;
213 LogFontW2A(&(pnclma->lfCaptionFont), &(nclmw.lfCaptionFont));
214 LogFontW2A(&(pnclma->lfSmCaptionFont), &(nclmw.lfSmCaptionFont));
215 LogFontW2A(&(pnclma->lfMenuFont), &(nclmw.lfMenuFont));
216 LogFontW2A(&(pnclma->lfStatusFont), &(nclmw.lfStatusFont));
217 LogFontW2A(&(pnclma->lfMessageFont), &(nclmw.lfMessageFont));
218 return TRUE;
219 }
220 case SPI_SETNONCLIENTMETRICS:
221 {
222 LPNONCLIENTMETRICSA pnclma = (LPNONCLIENTMETRICSA)pvParam;
223 NONCLIENTMETRICSW nclmw;
224 if(pnclma->cbSize != sizeof(NONCLIENTMETRICSA))
225 {
226 SetLastError(ERROR_INVALID_PARAMETER);
227 return FALSE;
228 }
229 nclmw.cbSize = sizeof(NONCLIENTMETRICSW);
230 nclmw.iBorderWidth = pnclma->iBorderWidth;
231 nclmw.iScrollWidth = pnclma->iScrollWidth;
232 nclmw.iScrollHeight = pnclma->iScrollHeight;
233 nclmw.iCaptionWidth = pnclma->iCaptionWidth;
234 nclmw.iCaptionHeight = pnclma->iCaptionHeight;
235 nclmw.iSmCaptionWidth = pnclma->iSmCaptionWidth;
236 nclmw.iSmCaptionHeight = pnclma->iSmCaptionHeight;
237 nclmw.iMenuWidth = pnclma->iMenuWidth;
238 nclmw.iMenuHeight = pnclma->iMenuHeight;
239 LogFontA2W(&(nclmw.lfCaptionFont), &(pnclma->lfCaptionFont));
240 LogFontA2W(&(nclmw.lfSmCaptionFont), &(pnclma->lfSmCaptionFont));
241 LogFontA2W(&(nclmw.lfMenuFont), &(pnclma->lfMenuFont));
242 LogFontA2W(&(nclmw.lfStatusFont), &(pnclma->lfStatusFont));
243 LogFontA2W(&(nclmw.lfMessageFont), &(pnclma->lfMessageFont));
244
245 return SystemParametersInfoW(uiAction, sizeof(NONCLIENTMETRICSW),
246 &nclmw, fWinIni);
247 }
248 case SPI_GETICONMETRICS:
249 {
250 LPICONMETRICSA picma = (LPICONMETRICSA)pvParam;
251 ICONMETRICSW icmw;
252 if(picma->cbSize != sizeof(ICONMETRICSA))
253 {
254 SetLastError(ERROR_INVALID_PARAMETER);
255 return FALSE;
256 }
257 icmw.cbSize = sizeof(ICONMETRICSW);
258 if (!SystemParametersInfoW(uiAction, sizeof(ICONMETRICSW),
259 &icmw, fWinIni))
260 return FALSE;
261
262 picma->iHorzSpacing = icmw.iHorzSpacing;
263 picma->iVertSpacing = icmw.iVertSpacing;
264 picma->iTitleWrap = icmw.iTitleWrap;
265 LogFontW2A(&(picma->lfFont), &(icmw.lfFont));
266 return TRUE;
267 }
268 case SPI_SETICONMETRICS:
269 {
270 LPICONMETRICSA picma = (LPICONMETRICSA)pvParam;
271 ICONMETRICSW icmw;
272 if(picma->cbSize != sizeof(ICONMETRICSA))
273 {
274 SetLastError(ERROR_INVALID_PARAMETER);
275 return FALSE;
276 }
277 icmw.cbSize = sizeof(ICONMETRICSW);
278 icmw.iHorzSpacing = picma->iHorzSpacing;
279 icmw.iVertSpacing = picma->iVertSpacing;
280 icmw.iTitleWrap = picma->iTitleWrap;
281 LogFontA2W(&(icmw.lfFont), &(picma->lfFont));
282
283 return SystemParametersInfoW(uiAction, sizeof(ICONMETRICSW),
284 &icmw, fWinIni);
285 }
286 case SPI_GETICONTITLELOGFONT:
287 {
288 LOGFONTW lfw;
289 if (!SystemParametersInfoW(uiAction, 0, &lfw, fWinIni))
290 return FALSE;
291 LogFontW2A(pvParam, &lfw);
292 return TRUE;
293 }
294 case SPI_SETICONTITLELOGFONT:
295 {
296 LPLOGFONTA plfa = (LPLOGFONTA)pvParam;
297 LOGFONTW lfw;
298 LogFontA2W(&lfw,plfa);
299 return SystemParametersInfoW(uiAction, 0, &lfw, fWinIni);
300 }
301 case SPI_GETDESKWALLPAPER:
302 {
303 BOOL Ret;
304 WCHAR awc[MAX_PATH];
305 UNICODE_STRING ustrWallpaper;
306 ANSI_STRING astrWallpaper;
307
308 Ret = NtUserSystemParametersInfo(SPI_GETDESKWALLPAPER, MAX_PATH, awc, fWinIni);
309 RtlInitUnicodeString(&ustrWallpaper, awc);
310 RtlUnicodeStringToAnsiString(&astrWallpaper, &ustrWallpaper, TRUE);
311
312 RtlCopyMemory(pvParam, astrWallpaper.Buffer, uiParam);
313 RtlFreeAnsiString(&astrWallpaper);
314 return Ret;
315 }
316
317 case SPI_SETDESKWALLPAPER:
318 {
319 UNICODE_STRING ustrWallpaper;
320 BOOL Ret;
321
322 if (pvParam)
323 {
324 if (!RtlCreateUnicodeStringFromAsciiz(&ustrWallpaper, pvParam))
325 {
326 ERR("RtlCreateUnicodeStringFromAsciiz failed\n");
327 return FALSE;
328 }
329 pvParam = &ustrWallpaper;
330 }
331
332 Ret = NtUserSystemParametersInfo(SPI_SETDESKWALLPAPER, uiParam, pvParam, fWinIni);
333 RtlFreeUnicodeString(&ustrWallpaper);
334 return Ret;
335 }
336 }
337 return NtUserSystemParametersInfo(uiAction, uiParam, pvParam, fWinIni);
338 }
339
340 BOOL WINAPI
341 RealSystemParametersInfoW(UINT uiAction,
342 UINT uiParam,
343 PVOID pvParam,
344 UINT fWinIni)
345 {
346 switch(uiAction)
347 {
348
349 case SPI_SETDESKWALLPAPER:
350 {
351 UNICODE_STRING ustrWallpaper;
352
353 RtlInitUnicodeString(&ustrWallpaper, pvParam);
354 return NtUserSystemParametersInfo(SPI_SETDESKWALLPAPER, uiParam, &ustrWallpaper, fWinIni);
355 }
356 }
357 return NtUserSystemParametersInfo(uiAction, uiParam, pvParam, fWinIni);
358 }
359
360
361 /*
362 * @implemented
363 */
364 BOOL WINAPI
365 SystemParametersInfoA(UINT uiAction,
366 UINT uiParam,
367 PVOID pvParam,
368 UINT fWinIni)
369 {
370 BOOL Hook, Ret = FALSE;
371
372 LoadUserApiHook();
373
374 Hook = BeginIfHookedUserApiHook();
375
376 /* Bypass SEH and go direct. */
377 if (!Hook) return RealSystemParametersInfoA(uiAction, uiParam, pvParam, fWinIni);
378
379 _SEH2_TRY
380 {
381 Ret = guah.SystemParametersInfoA(uiAction, uiParam, pvParam, fWinIni);
382 }
383 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
384 {
385 }
386 _SEH2_END;
387
388 EndUserApiHook();
389
390 return Ret;
391 }
392
393 /*
394 * @implemented
395 */
396 BOOL WINAPI
397 SystemParametersInfoW(UINT uiAction,
398 UINT uiParam,
399 PVOID pvParam,
400 UINT fWinIni)
401 {
402 BOOL Hook, Ret = FALSE;
403
404 LoadUserApiHook();
405
406 Hook = BeginIfHookedUserApiHook();
407
408 /* Bypass SEH and go direct. */
409 if (!Hook) return RealSystemParametersInfoW(uiAction, uiParam, pvParam, fWinIni);
410
411 _SEH2_TRY
412 {
413 Ret = guah.SystemParametersInfoW(uiAction, uiParam, pvParam, fWinIni);
414 }
415 _SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
416 {
417 }
418 _SEH2_END;
419
420 EndUserApiHook();
421
422 return Ret;
423 }
424
425 /*
426 * @implemented
427 */
428 HDESK WINAPI
429 CreateDesktopA(LPCSTR lpszDesktop,
430 LPCSTR lpszDevice,
431 LPDEVMODEA pDevmode,
432 DWORD dwFlags,
433 ACCESS_MASK dwDesiredAccess,
434 LPSECURITY_ATTRIBUTES lpsa)
435 {
436 UNICODE_STRING DesktopNameU;
437 HDESK hDesktop;
438 LPDEVMODEW DevmodeW = NULL;
439
440 if (lpszDesktop)
441 {
442 /* After conversion, the buffer is zero-terminated */
443 RtlCreateUnicodeStringFromAsciiz(&DesktopNameU, lpszDesktop);
444 }
445 else
446 {
447 RtlInitUnicodeString(&DesktopNameU, NULL);
448 }
449
450 if (pDevmode)
451 DevmodeW = GdiConvertToDevmodeW(pDevmode);
452
453 hDesktop = CreateDesktopW(DesktopNameU.Buffer,
454 NULL,
455 DevmodeW,
456 dwFlags,
457 dwDesiredAccess,
458 lpsa);
459
460 /* Free the string, if it was allocated */
461 if (lpszDesktop) RtlFreeUnicodeString(&DesktopNameU);
462
463 return hDesktop;
464 }
465
466
467 /*
468 * @implemented
469 */
470 HDESK WINAPI
471 CreateDesktopW(LPCWSTR lpszDesktop,
472 LPCWSTR lpszDevice,
473 LPDEVMODEW pDevmode,
474 DWORD dwFlags,
475 ACCESS_MASK dwDesiredAccess,
476 LPSECURITY_ATTRIBUTES lpsa)
477 {
478 OBJECT_ATTRIBUTES oas;
479 UNICODE_STRING DesktopName, DesktopDevice;
480 HWINSTA hWinSta;
481 HDESK hDesktop;
482 ULONG Attributes = (OBJ_OPENIF|OBJ_CASE_INSENSITIVE);
483
484 /* Retrive WinStation handle. */
485 hWinSta = NtUserGetProcessWindowStation();
486
487 /* Initialize the strings. */
488 RtlInitUnicodeString(&DesktopName, lpszDesktop);
489 RtlInitUnicodeString(&DesktopDevice, lpszDevice);
490
491 /* Check for process is inherited, set flag if set. */
492 if (lpsa && lpsa->bInheritHandle) Attributes |= OBJ_INHERIT;
493
494 /* Initialize the attributes for the desktop. */
495 InitializeObjectAttributes( &oas,
496 &DesktopName,
497 Attributes,
498 hWinSta,
499 lpsa ? lpsa->lpSecurityDescriptor : NULL);
500
501 /* Send the request and call to win32k. */
502 hDesktop = NtUserCreateDesktop( &oas,
503 &DesktopDevice,
504 pDevmode,
505 dwFlags,
506 dwDesiredAccess);
507
508 return(hDesktop);
509 }
510
511
512 /*
513 * @implemented
514 */
515 BOOL
516 WINAPI
517 EnumDesktopsA(
518 HWINSTA WindowStation,
519 DESKTOPENUMPROCA EnumFunc,
520 LPARAM Context)
521 {
522 return EnumNamesA(WindowStation, EnumFunc, Context, TRUE);
523 }
524
525
526 /*
527 * @implemented
528 */
529 BOOL
530 WINAPI
531 EnumDesktopsW(
532 HWINSTA WindowStation,
533 DESKTOPENUMPROCW EnumFunc,
534 LPARAM Context)
535 {
536 return EnumNamesW(WindowStation, EnumFunc, Context, TRUE);
537 }
538
539
540 /*
541 * @implemented
542 */
543 HDESK
544 WINAPI
545 GetThreadDesktop(
546 DWORD dwThreadId)
547 {
548 return NtUserGetThreadDesktop(dwThreadId, 0);
549 }
550
551
552 /*
553 * @implemented
554 */
555 HDESK
556 WINAPI
557 OpenDesktopA(
558 LPCSTR lpszDesktop,
559 DWORD dwFlags,
560 BOOL fInherit,
561 ACCESS_MASK dwDesiredAccess)
562 {
563 UNICODE_STRING DesktopNameU;
564 HDESK hDesktop;
565
566 if (lpszDesktop)
567 {
568 /* After conversion, the buffer is zero-terminated */
569 RtlCreateUnicodeStringFromAsciiz(&DesktopNameU, lpszDesktop);
570 }
571 else
572 {
573 RtlInitUnicodeString(&DesktopNameU, NULL);
574 }
575
576 hDesktop = OpenDesktopW(DesktopNameU.Buffer,
577 dwFlags,
578 fInherit,
579 dwDesiredAccess);
580
581 /* Free the string, if it was allocated */
582 if (lpszDesktop) RtlFreeUnicodeString(&DesktopNameU);
583
584 return hDesktop;
585 }
586
587
588 /*
589 * @implemented
590 */
591 HDESK
592 WINAPI
593 OpenDesktopW(
594 LPCWSTR lpszDesktop,
595 DWORD dwFlags,
596 BOOL fInherit,
597 ACCESS_MASK dwDesiredAccess)
598 {
599 UNICODE_STRING DesktopName;
600 OBJECT_ATTRIBUTES ObjectAttributes;
601
602 RtlInitUnicodeString(&DesktopName, lpszDesktop);
603
604 InitializeObjectAttributes(&ObjectAttributes,
605 &DesktopName,
606 OBJ_CASE_INSENSITIVE,
607 GetProcessWindowStation(),
608 0);
609
610 if( fInherit == TRUE )
611 {
612 ObjectAttributes.Attributes |= OBJ_INHERIT;
613 }
614
615 return NtUserOpenDesktop(&ObjectAttributes, dwFlags, dwDesiredAccess);
616 }
617
618
619 /*
620 * @implemented
621 */
622 BOOL WINAPI
623 SetShellWindow(HWND hwndShell)
624 {
625 return SetShellWindowEx(hwndShell, hwndShell);
626 }
627
628
629 /*
630 * @implemented
631 */
632 HWND WINAPI
633 GetShellWindow(VOID)
634 {
635 PDESKTOPINFO pdi;
636 pdi = GetThreadDesktopInfo();
637 if (pdi) return pdi->hShellWindow;
638 return NULL;
639 }
640
641
642 /* EOF */