Release 950620
[wine/multimedia.git] / windows / mdi.c
blob85b49b504e13691bc8bc040340ed6c2c66e519b1
1 /* MDI.C
3 * Copyright 1994, Bob Amstadt
5 * This file contains routines to support MDI features.
6 */
7 #include <stdlib.h>
8 #include <stdio.h>
9 #include <math.h>
10 #include "windows.h"
11 #include "win.h"
12 #include "nonclient.h"
13 #include "mdi.h"
14 #include "user.h"
15 #include "menu.h"
16 #include "sysmetrics.h"
17 #include "stddebug.h"
18 /* #define DEBUG_MDI */
19 #include "debug.h"
21 /**********************************************************************
22 * MDIRecreateMenuList
24 void MDIRecreateMenuList(MDICLIENTINFO *ci)
26 HLOCAL hinfo;
28 char buffer[128];
29 int id, n, index;
31 dprintf_mdi(stddeb, "MDIRecreateMenuList: hWindowMenu %0x\n",
32 ci->hWindowMenu);
34 id = ci->idFirstChild;
35 while (DeleteMenu(ci->hWindowMenu, id, MF_BYCOMMAND))
36 id++;
38 dprintf_mdi(stddeb, "MDIRecreateMenuList: id %04x, idFirstChild %04x\n",
39 id, ci->idFirstChild);
41 if (!ci->flagMenuAltered)
43 ci->flagMenuAltered = TRUE;
44 AppendMenu(ci->hWindowMenu, MF_SEPARATOR, 0, NULL);
47 id = ci->idFirstChild;
48 index = 1;
49 for (hinfo = ci->infoActiveChildren; hinfo != 0;)
51 MDICHILDINFO *chi = USER_HEAP_LIN_ADDR(hinfo);
53 n = sprintf(buffer, "%d ", index++);
54 GetWindowText(chi->hwnd, buffer + n, sizeof(buffer) - n - 1);
56 dprintf_mdi(stddeb, "MDIRecreateMenuList: id %04x, '%s'\n",
57 id, buffer);
59 AppendMenu(ci->hWindowMenu, MF_STRING, id++, buffer);
60 hinfo = chi->next;
65 /**********************************************************************
66 * MDIIconArrange
68 WORD MDIIconArrange(HWND parent)
70 return ArrangeIconicWindows(parent); /* Any reason why the */
71 /* existing icon arrange */
72 /* can't be used here? */
73 /* -DRP */
76 /**********************************************************************
77 * MDICreateChild
79 HWND MDICreateChild(WND *w, MDICLIENTINFO *ci, HWND parent, LPARAM lParam )
81 MDICREATESTRUCT *cs = (MDICREATESTRUCT *)PTR_SEG_TO_LIN(lParam);
82 HWND hwnd;
83 int spacing;
86 * Create child window
88 cs->style &= (WS_MINIMIZE | WS_MAXIMIZE | WS_HSCROLL | WS_VSCROLL);
90 /* The child windows should probably */
91 /* stagger, shouldn't they? -DRP */
92 spacing = GetSystemMetrics(SM_CYCAPTION) + GetSystemMetrics(SM_CYFRAME);
93 cs->x = ci->nActiveChildren * spacing;
94 cs->y = ci->nActiveChildren * spacing;
96 hwnd = CreateWindowEx(0, PTR_SEG_TO_LIN(cs->szClass),
97 PTR_SEG_TO_LIN(cs->szTitle),
98 WS_CHILD | WS_BORDER | WS_CAPTION | WS_CLIPSIBLINGS |
99 WS_MAXIMIZEBOX | WS_MINIMIZEBOX | WS_SYSMENU |
100 WS_THICKFRAME | WS_VISIBLE | cs->style,
101 cs->x, cs->y, cs->cx, cs->cy, parent, (HMENU) 0,
102 w->hInstance, lParam);
104 if (hwnd)
106 HANDLE h = USER_HEAP_ALLOC( sizeof(MDICHILDINFO) );
107 MDICHILDINFO *child_info = USER_HEAP_LIN_ADDR(h);
109 if (!h)
111 DestroyWindow(hwnd);
112 return 0;
115 ci->nActiveChildren++;
117 child_info->next = ci->infoActiveChildren;
118 child_info->prev = 0;
119 child_info->hwnd = hwnd;
121 if (ci->infoActiveChildren) {
122 MDICHILDINFO *nextinfo = USER_HEAP_LIN_ADDR(ci->infoActiveChildren);
123 nextinfo->prev = h;
126 ci->infoActiveChildren = h;
128 SendMessage(parent, WM_CHILDACTIVATE, 0, 0);
131 return hwnd;
134 /**********************************************************************
135 * MDIDestroyChild
137 HWND MDIDestroyChild(WND *w_parent, MDICLIENTINFO *ci, HWND parent,
138 HWND child, BOOL flagDestroy)
140 MDICHILDINFO *chi;
141 HLOCAL hinfo;
143 hinfo = ci->infoActiveChildren;
144 while (hinfo != 0) {
145 chi = (MDICHILDINFO *)USER_HEAP_LIN_ADDR(hinfo);
146 if (chi->hwnd == child) break;
147 hinfo = chi->next;
150 if (hinfo != 0)
152 if (chi->prev)
153 ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(chi->prev))->next = chi->next;
154 if (chi->next)
155 ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(chi->next))->prev = chi->prev;
156 if (ci->infoActiveChildren == hinfo)
157 ci->infoActiveChildren = chi->next;
159 ci->nActiveChildren--;
161 if (chi->hwnd == ci->hwndActiveChild)
162 SendMessage(parent, WM_CHILDACTIVATE, 0, 0);
164 USER_HEAP_FREE(hinfo);
166 if (flagDestroy)
167 DestroyWindow(child);
170 return 0;
173 /**********************************************************************
174 * MDIBringChildToTop
176 void MDIBringChildToTop(HWND parent, WORD id, WORD by_id, BOOL send_to_bottom)
178 HLOCAL hinfo;
179 MDICHILDINFO *chi;
180 MDICLIENTINFO *ci;
181 WND *w;
182 int i;
184 w = WIN_FindWndPtr(parent);
185 ci = (MDICLIENTINFO *) w->wExtra;
187 dprintf_mdi(stddeb, "MDIBringToTop: id %04x, by_id %d\n", id, by_id);
189 if (by_id)
190 id -= ci->idFirstChild;
191 if (!by_id || id < ci->nActiveChildren)
193 hinfo = ci->infoActiveChildren;
195 if (by_id)
197 for (i = 0; i < id; i++)
198 hinfo = ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(hinfo))->next;
199 chi = USER_HEAP_LIN_ADDR(hinfo);
201 else
203 while (hinfo != 0) {
204 chi = (MDICHILDINFO *)USER_HEAP_LIN_ADDR(hinfo);
205 if (chi->hwnd == id) break;
206 hinfo = chi->next;
210 if (hinfo == 0)
211 return;
213 dprintf_mdi(stddeb, "MDIBringToTop: child %04x\n", chi->hwnd);
214 if (hinfo != ci->infoActiveChildren)
216 if (ci->flagChildMaximized)
218 RECT rectOldRestore, rect;
220 w = WIN_FindWndPtr(chi->hwnd);
222 rectOldRestore = ci->rectRestore;
223 GetWindowRect(chi->hwnd, &ci->rectRestore);
225 rect.top = (ci->rectMaximize.top -
226 (w->rectClient.top - w->rectWindow.top));
227 rect.bottom = (ci->rectMaximize.bottom +
228 (w->rectWindow.bottom - w->rectClient.bottom));
229 rect.left = (ci->rectMaximize.left -
230 (w->rectClient.left - w->rectWindow.left));
231 rect.right = (ci->rectMaximize.right +
232 (w->rectWindow.right - w->rectClient.right));
233 w->dwStyle |= WS_MAXIMIZE;
234 SetWindowPos(chi->hwnd, HWND_TOP, rect.left, rect.top,
235 rect.right - rect.left + 1,
236 rect.bottom - rect.top + 1, 0);
237 SendMessage(chi->hwnd, WM_SIZE, SIZE_MAXIMIZED,
238 MAKELONG(w->rectClient.right-w->rectClient.left,
239 w->rectClient.bottom-w->rectClient.top));
241 w = WIN_FindWndPtr(ci->hwndActiveChild);
242 w->dwStyle &= ~WS_MAXIMIZE;
243 SetWindowPos(ci->hwndActiveChild, HWND_BOTTOM,
244 rectOldRestore.left, rectOldRestore.top,
245 rectOldRestore.right - rectOldRestore.left + 1,
246 rectOldRestore.bottom - rectOldRestore.top + 1,
247 SWP_NOACTIVATE |
248 (send_to_bottom ? 0 : SWP_NOZORDER));
250 else
252 SetWindowPos(chi->hwnd, HWND_TOP, 0, 0, 0, 0,
253 SWP_NOMOVE | SWP_NOSIZE );
254 if (send_to_bottom)
256 SetWindowPos(ci->hwndActiveChild, HWND_BOTTOM, 0, 0, 0, 0,
257 SWP_NOMOVE | SWP_NOSIZE | SWP_NOACTIVATE);
261 if (chi->next)
262 ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(chi->next))->prev = chi->prev;
264 if (chi->prev)
265 ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(chi->prev))->next = chi->next;
267 chi->prev = 0;
268 chi->next = ci->infoActiveChildren;
269 ((MDICHILDINFO *)USER_HEAP_LIN_ADDR(chi->next))->prev = hinfo;
270 ci->infoActiveChildren = hinfo;
272 SendMessage(parent, WM_CHILDACTIVATE, 0, 0);
275 dprintf_mdi(stddeb, "MDIBringToTop: pos %04x, hwnd %04x\n",
276 id, chi->hwnd);
280 /**********************************************************************
281 * MDIMaximizeChild
283 LONG MDIMaximizeChild(HWND parent, HWND child, MDICLIENTINFO *ci)
285 WND *w = WIN_FindWndPtr(child);
286 RECT rect;
288 MDIBringChildToTop(parent, child, FALSE, FALSE);
289 ci->rectRestore = w->rectWindow;
291 rect.top = (ci->rectMaximize.top -
292 (w->rectClient.top - w->rectWindow.top));
293 rect.bottom = (ci->rectMaximize.bottom +
294 (w->rectWindow.bottom - w->rectClient.bottom));
295 rect.left = (ci->rectMaximize.left -
296 (w->rectClient.left - w->rectWindow.left));
297 rect.right = (ci->rectMaximize.right +
298 (w->rectWindow.right - w->rectClient.right));
299 w->dwStyle |= WS_MAXIMIZE;
300 SetWindowPos(child, 0, rect.left, rect.top,
301 rect.right - rect.left + 1, rect.bottom - rect.top + 1,
302 SWP_NOACTIVATE | SWP_NOZORDER);
304 ci->flagChildMaximized = TRUE;
306 SendMessage(child, WM_SIZE, SIZE_MAXIMIZED,
307 MAKELONG(w->rectClient.right-w->rectClient.left,
308 w->rectClient.bottom-w->rectClient.top));
309 SendMessage(GetParent(parent), WM_NCPAINT, 0, 0);
311 return 0;
314 /**********************************************************************
315 * MDIRestoreChild
317 LONG MDIRestoreChild(HWND parent, MDICLIENTINFO *ci)
319 HWND child;
321 dprintf_mdi(stddeb,"restoring mdi child\n");
323 child = ci->hwndActiveChild;
324 ci->flagChildMaximized = FALSE;
326 ShowWindow(child, SW_RESTORE); /* display the window */
327 MDIBringChildToTop(parent, child, FALSE, FALSE);
328 SendMessage(GetParent(parent), WM_NCPAINT, 0, 0);
330 return 0;
333 /**********************************************************************
334 * MDIChildActivated
336 LONG MDIChildActivated(WND *w, MDICLIENTINFO *ci, HWND parent)
338 HLOCAL hinfo;
339 MDICHILDINFO *chi;
340 HWND deact_hwnd;
341 HWND act_hwnd;
342 LONG lParam;
344 dprintf_mdi(stddeb, "MDIChildActivate: top %04x\n", w->hwndChild);
346 hinfo = ci->infoActiveChildren;
347 if (hinfo)
349 chi = (MDICHILDINFO *)USER_HEAP_LIN_ADDR(hinfo);
350 deact_hwnd = ci->hwndActiveChild;
351 act_hwnd = chi->hwnd;
352 lParam = ((LONG) deact_hwnd << 16) | act_hwnd;
354 dprintf_mdi(stddeb, "MDIChildActivate: deact %04x, act %04x\n",
355 deact_hwnd, act_hwnd);
357 ci->hwndActiveChild = act_hwnd;
359 if (deact_hwnd != act_hwnd)
361 MDIRecreateMenuList(ci);
362 SendMessage(deact_hwnd, WM_NCACTIVATE, FALSE, 0);
363 SendMessage(deact_hwnd, WM_MDIACTIVATE, FALSE, lParam);
366 SendMessage(act_hwnd, WM_NCACTIVATE, TRUE, 0);
367 SendMessage(act_hwnd, WM_MDIACTIVATE, TRUE, lParam);
370 if (hinfo || ci->nActiveChildren == 0)
372 MDIRecreateMenuList(ci);
373 SendMessage(GetParent(parent), WM_NCPAINT, 0, 0);
376 return 0;
379 /**********************************************************************
380 * MDICascade
382 LONG MDICascade(HWND parent, MDICLIENTINFO *ci)
384 HLOCAL hinfo;
385 MDICHILDINFO *chi;
386 RECT rect;
387 int spacing, xsize, ysize;
388 int x, y;
390 if (ci->flagChildMaximized)
391 MDIRestoreChild(parent, ci);
393 /* If there aren't any children, don't even bother.
395 if (ci->nActiveChildren == 0)
396 return 0;
398 GetClientRect(parent, &rect);
399 spacing = GetSystemMetrics(SM_CYCAPTION) + GetSystemMetrics(SM_CYFRAME);
400 ysize = rect.bottom - 8 * spacing;
401 xsize = rect.right - 8 * spacing;
403 dprintf_mdi(stddeb,
404 "MDICascade: Client wnd at (%d,%d) - (%d,%d), spacing %d\n",
405 rect.left, rect.top, rect.right, rect.bottom, spacing);
406 dprintf_mdi(stddeb, "MDICascade: searching for last child\n");
407 hinfo = ci->infoActiveChildren;
408 while(1) {
409 chi = USER_HEAP_LIN_ADDR(hinfo);
410 if (chi->next == 0) break;
411 hinfo = chi->next;
414 dprintf_mdi(stddeb, "MDICascade: last child is %04x\n", chi->hwnd);
415 x = 0;
416 y = 0;
417 while (hinfo != 0)
419 chi = USER_HEAP_LIN_ADDR(hinfo);
420 dprintf_mdi(stddeb, "MDICascade: move %04x to (%d,%d) size [%d,%d]\n",
421 chi->hwnd, x, y, xsize, ysize);
422 if (IsIconic(chi->hwnd)) continue;
423 SetWindowPos(chi->hwnd, 0, x, y, xsize, ysize,
424 SWP_DRAWFRAME | SWP_NOACTIVATE | SWP_NOZORDER);
426 x += spacing;
427 y += spacing;
429 hinfo = chi->prev;
432 return 0;
435 /**********************************************************************
436 * MDITile
438 LONG MDITile(HWND parent, MDICLIENTINFO *ci)
440 HLOCAL hinfo;
441 MDICHILDINFO *chi;
442 RECT rect;
443 int xsize, ysize;
444 int x, y;
445 int rows, columns;
446 int r, c;
447 int i;
449 if (ci->flagChildMaximized)
450 MDIRestoreChild(parent, ci);
452 /* If there aren't any children, don't even bother.
454 if (ci->nActiveChildren == 0)
455 return 0;
457 GetClientRect(parent, &rect);
458 rows = (int) sqrt((double) ci->nActiveChildren);
459 columns = ci->nActiveChildren / rows;
460 ysize = rect.bottom / rows;
461 xsize = rect.right / columns;
463 hinfo = ci->infoActiveChildren;
464 x = 0;
465 i = 0;
466 for (c = 1; c <= columns; c++)
468 if (c == columns)
470 rows = ci->nActiveChildren - i;
471 ysize = rect.bottom / rows;
474 y = 0;
475 for (r = 1; r <= rows; r++, i++)
477 chi = (MDICHILDINFO *)USER_HEAP_LIN_ADDR(hinfo);
478 SetWindowPos(chi->hwnd, 0, x, y, xsize, ysize,
479 SWP_DRAWFRAME | SWP_NOACTIVATE | SWP_NOZORDER);
481 y += ysize;
482 hinfo = chi->next;
485 x += xsize;
489 return 0;
492 /**********************************************************************
493 * MDIHandleLButton
495 BOOL MDIHandleLButton(HWND hwndFrame, HWND hwndClient,
496 WORD wParam, LONG lParam)
498 MDICLIENTINFO *ci;
499 WND *w;
500 RECT rect;
501 WORD x;
503 w = WIN_FindWndPtr(hwndClient);
504 ci = (MDICLIENTINFO *) w->wExtra;
506 if (wParam == HTMENU && ci->flagChildMaximized)
508 x = LOWORD(lParam);
510 NC_GetInsideRect(hwndFrame, &rect);
511 if (x < rect.left + SYSMETRICS_CXSIZE)
513 SendMessage(ci->hwndActiveChild, WM_SYSCOMMAND,
514 SC_CLOSE, lParam);
515 return TRUE;
517 else if (x >= rect.right - SYSMETRICS_CXSIZE)
519 SendMessage(ci->hwndActiveChild, WM_SYSCOMMAND,
520 SC_RESTORE, lParam);
521 return TRUE;
525 return FALSE;
528 /**********************************************************************
529 * MDIPaintMaximized
531 LONG MDIPaintMaximized(HWND hwndFrame, HWND hwndClient, WORD message,
532 WORD wParam, LONG lParam)
534 static HBITMAP hbitmapClose = 0;
535 static HBITMAP hbitmapMaximized = 0;
537 MDICLIENTINFO *ci;
538 WND *w;
539 HDC hdc, hdcMem;
540 RECT rect;
541 WND *wndPtr = WIN_FindWndPtr(hwndFrame);
543 w = WIN_FindWndPtr(hwndClient);
544 ci = (MDICLIENTINFO *) w->wExtra;
546 dprintf_mdi(stddeb, "MDIPaintMaximized: frame %04x, client %04x"
547 ", max flag %d, menu %04x\n", hwndFrame, hwndClient,
548 ci->flagChildMaximized, wndPtr ? wndPtr->wIDmenu : 0);
550 if (ci->flagChildMaximized && wndPtr && wndPtr->wIDmenu != 0)
552 NC_DoNCPaint(hwndFrame, wParam, TRUE);
554 hdc = GetDCEx(hwndFrame, 0, DCX_CACHE | DCX_WINDOW);
555 if (!hdc) return 0;
557 hdcMem = CreateCompatibleDC(hdc);
559 if (hbitmapClose == 0)
561 hbitmapClose = LoadBitmap(0, MAKEINTRESOURCE(OBM_OLD_CLOSE));
562 hbitmapMaximized = LoadBitmap(0, MAKEINTRESOURCE(OBM_RESTORE));
565 dprintf_mdi(stddeb,
566 "MDIPaintMaximized: hdcMem %04x, close bitmap %04x, "
567 "maximized bitmap %04x\n",
568 hdcMem, hbitmapClose, hbitmapMaximized);
570 NC_GetInsideRect(hwndFrame, &rect);
571 rect.top += (wndPtr->dwStyle & WS_CAPTION) ? SYSMETRICS_CYSIZE + 1 : 0;
572 SelectObject(hdcMem, hbitmapClose);
573 BitBlt(hdc, rect.left, rect.top + 1,
574 SYSMETRICS_CXSIZE, SYSMETRICS_CYSIZE,
575 hdcMem, 1, 1, SRCCOPY);
577 NC_GetInsideRect(hwndFrame, &rect);
578 rect.top += (wndPtr->dwStyle & WS_CAPTION) ? SYSMETRICS_CYSIZE + 1 : 0;
579 rect.left = rect.right - SYSMETRICS_CXSIZE;
580 SelectObject(hdcMem, hbitmapMaximized);
581 BitBlt(hdc, rect.left, rect.top + 1,
582 SYSMETRICS_CXSIZE, SYSMETRICS_CYSIZE,
583 hdcMem, 1, 1, SRCCOPY);
585 NC_GetInsideRect(hwndFrame, &rect);
586 rect.top += (wndPtr->dwStyle & WS_CAPTION) ? SYSMETRICS_CYSIZE + 1 : 0;
587 rect.left += SYSMETRICS_CXSIZE;
588 rect.right -= SYSMETRICS_CXSIZE;
589 rect.bottom = rect.top + SYSMETRICS_CYMENU;
591 MENU_DrawMenuBar(hdc, &rect, hwndFrame, FALSE);
593 DeleteDC(hdcMem);
594 ReleaseDC(hwndFrame, hdc);
596 else
597 return DefWindowProc(hwndFrame, message, wParam, lParam);
599 return 0;
602 /**********************************************************************
603 * MDIClientWndProc
605 * This function is the handler for all MDI requests.
607 LONG MDIClientWndProc(HWND hwnd, WORD message, WORD wParam, LONG lParam)
609 LPCREATESTRUCT cs;
610 LPCLIENTCREATESTRUCT ccs;
611 MDICLIENTINFO *ci;
612 WND *w;
614 w = WIN_FindWndPtr(hwnd);
615 ci = (MDICLIENTINFO *) w->wExtra;
617 switch (message)
619 case WM_CHILDACTIVATE:
620 return MDIChildActivated(w, ci, hwnd);
622 case WM_CREATE:
623 cs = (LPCREATESTRUCT) PTR_SEG_TO_LIN(lParam);
624 ccs = (LPCLIENTCREATESTRUCT) PTR_SEG_TO_LIN(cs->lpCreateParams);
625 ci->hWindowMenu = ccs->hWindowMenu;
626 ci->idFirstChild = ccs->idFirstChild;
627 ci->infoActiveChildren = 0;
628 ci->flagMenuAltered = FALSE;
629 ci->flagChildMaximized = FALSE;
630 w->dwStyle |= WS_CLIPCHILDREN;
632 GetClientRect(w->hwndParent, &ci->rectMaximize);
633 MoveWindow(hwnd, 0, 0,
634 ci->rectMaximize.right, ci->rectMaximize.bottom, 1);
636 return 0;
638 case WM_MDIACTIVATE:
639 MDIBringChildToTop(hwnd, wParam, FALSE, FALSE);
640 return 0;
642 case WM_MDICASCADE:
643 return MDICascade(hwnd, ci);
645 case WM_MDICREATE:
646 return MDICreateChild(w, ci, hwnd, lParam );
648 case WM_MDIDESTROY:
649 return MDIDestroyChild(w, ci, hwnd, wParam, TRUE);
651 case WM_MDIGETACTIVE:
652 return ((LONG) ci->hwndActiveChild |
653 ((LONG) ci->flagChildMaximized << 16));
655 case WM_MDIICONARRANGE:
656 return MDIIconArrange(hwnd);
658 case WM_MDIMAXIMIZE:
659 return MDIMaximizeChild(hwnd, wParam, ci);
661 case WM_MDINEXT:
662 MDIBringChildToTop(hwnd, wParam, FALSE, TRUE);
663 break;
665 case WM_MDIRESTORE:
666 return MDIRestoreChild(hwnd, ci);
668 case WM_MDISETMENU:
669 /* return MDISetMenu(...) */
670 break;
672 case WM_MDITILE:
673 return MDITile(hwnd, ci);
675 case WM_NCACTIVATE:
676 SendMessage(ci->hwndActiveChild, message, wParam, lParam);
677 break;
679 case WM_PARENTNOTIFY:
680 if (wParam == WM_DESTROY)
681 return MDIDestroyChild(w, ci, hwnd, LOWORD(lParam), FALSE);
682 else if (wParam == WM_LBUTTONDOWN)
683 MDIBringChildToTop(hwnd, ci->hwndHitTest, FALSE, FALSE);
684 break;
686 case WM_SIZE:
687 GetClientRect(w->hwndParent, &ci->rectMaximize);
688 break;
692 return DefWindowProc(hwnd, message, wParam, lParam);
695 /**********************************************************************
696 * DefFrameProc (USER.445)
699 LONG DefFrameProc(HWND hwnd, HWND hwndMDIClient, WORD message,
700 WORD wParam, LONG lParam)
702 if (hwndMDIClient)
704 switch (message)
706 case WM_COMMAND:
707 MDIBringChildToTop(hwndMDIClient, wParam, TRUE, FALSE);
708 break;
710 case WM_NCLBUTTONDOWN:
711 if (MDIHandleLButton(hwnd, hwndMDIClient, wParam, lParam))
712 return 0;
713 break;
715 case WM_NCACTIVATE:
716 SendMessage(hwndMDIClient, message, wParam, lParam);
717 return MDIPaintMaximized(hwnd, hwndMDIClient,
718 message, wParam, lParam);
720 case WM_NCPAINT:
721 return MDIPaintMaximized(hwnd, hwndMDIClient,
722 message, wParam, lParam);
724 case WM_SETFOCUS:
725 SendMessage(hwndMDIClient, WM_SETFOCUS, wParam, lParam);
726 break;
728 case WM_SIZE:
729 MoveWindow(hwndMDIClient, 0, 0,
730 LOWORD(lParam), HIWORD(lParam), TRUE);
731 break;
735 return DefWindowProc(hwnd, message, wParam, lParam);
738 /**********************************************************************
739 * DefMDIChildProc (USER.447)
742 LONG DefMDIChildProc(HWND hwnd, WORD message, WORD wParam, LONG lParam)
744 MDICLIENTINFO *ci;
745 WND *w;
747 w = WIN_FindWndPtr(GetParent(hwnd));
748 ci = (MDICLIENTINFO *) w->wExtra;
750 switch (message)
752 case WM_NCHITTEST:
753 ci->hwndHitTest = hwnd;
754 break;
756 case WM_NCPAINT:
757 NC_DoNCPaint( hwnd, hwnd == ci->hwndActiveChild, FALSE );
758 return 0;
760 case WM_SYSCOMMAND:
761 switch (wParam)
763 case SC_MAXIMIZE:
764 return SendMessage(GetParent(hwnd), WM_MDIMAXIMIZE, hwnd, 0);
766 case SC_RESTORE:
767 return SendMessage(GetParent(hwnd), WM_MDIRESTORE, hwnd, 0);
769 break;
773 return DefWindowProc(hwnd, message, wParam, lParam);
776 /**********************************************************************
777 * TranslateMDISysAccel (USER.451)
780 BOOL TranslateMDISysAccel(HWND hwndClient, LPMSG msg)
782 return 0;
786 /***********************************************************************
787 * CalcChildScroll (USER.462)
789 void CalcChildScroll( HWND hwnd, WORD scroll )
791 RECT childRect, clientRect;
792 HWND hwndChild;
794 GetClientRect( hwnd, &clientRect );
795 SetRectEmpty( &childRect );
796 hwndChild = GetWindow( hwnd, GW_CHILD );
797 while (hwndChild)
799 WND *wndPtr = WIN_FindWndPtr( hwndChild );
800 UnionRect( &childRect, &wndPtr->rectWindow, &childRect );
801 hwndChild = wndPtr->hwndNext;
803 UnionRect( &childRect, &clientRect, &childRect );
805 if ((scroll == SB_HORZ) || (scroll == SB_BOTH))
807 SetScrollRange( hwnd, SB_HORZ, childRect.left,
808 childRect.right - clientRect.right, FALSE );
809 SetScrollPos( hwnd, SB_HORZ, clientRect.left - childRect.left, TRUE );
811 if ((scroll == SB_VERT) || (scroll == SB_BOTH))
813 SetScrollRange( hwnd, SB_VERT, childRect.top,
814 childRect.bottom - clientRect.bottom, FALSE );
815 SetScrollPos( hwnd, SB_HORZ, clientRect.top - childRect.top, TRUE );