/* * Message boxes * * Copyright 1995 Bernd Schmidt */ #include <string.h> #include "windef.h" #include "wingdi.h" #include "wine/winbase16.h" #include "wine/winuser16.h" #include "dlgs.h" #include "heap.h" #include "user.h" #include "debugtools.h" DEFAULT_DEBUG_CHANNEL(dialog); #define MSGBOX_IDICON 1088 #define MSGBOX_IDTEXT 100 static HFONT MSGBOX_OnInit(HWND hwnd, LPMSGBOXPARAMSA lpmb) { static HFONT hFont = 0, hPrevFont = 0; RECT rect; HWND hItem; HDC hdc; int i, buttons; int bspace, bw, bh, theight, tleft, wwidth, wheight, bpos; int borheight, borwidth, iheight, ileft, iwidth, twidth, tiheight; LPCSTR lpszText; char buf[256]; if (TWEAK_WineLook >= WIN95_LOOK) { NONCLIENTMETRICSA nclm; nclm.cbSize = sizeof(NONCLIENTMETRICSA); SystemParametersInfoA (SPI_GETNONCLIENTMETRICS, 0, &nclm, 0); hFont = CreateFontIndirectA (&nclm.lfMessageFont); /* set button font */ for (i=1; i < 8; i++) SendDlgItemMessageA (hwnd, i, WM_SETFONT, (WPARAM)hFont, 0); /* set text font */ SendDlgItemMessageA (hwnd, MSGBOX_IDTEXT, WM_SETFONT, (WPARAM)hFont, 0); } if (HIWORD(lpmb->lpszCaption)) { SetWindowTextA(hwnd, lpmb->lpszCaption); } else { if (LoadStringA(lpmb->hInstance, LOWORD(lpmb->lpszCaption), buf, sizeof(buf))) SetWindowTextA(hwnd, buf); } if (HIWORD(lpmb->lpszText)) { lpszText = lpmb->lpszText; } else { lpszText = buf; if (!LoadStringA(lpmb->hInstance, LOWORD(lpmb->lpszText), buf, sizeof(buf))) *buf = 0; /* FIXME ?? */ } SetWindowTextA(GetDlgItem(hwnd, MSGBOX_IDTEXT), lpszText); /* Hide not selected buttons */ switch(lpmb->dwStyle & MB_TYPEMASK) { case MB_OK: ShowWindow(GetDlgItem(hwnd, IDCANCEL), SW_HIDE); /* fall through */ case MB_OKCANCEL: ShowWindow(GetDlgItem(hwnd, IDABORT), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDRETRY), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDIGNORE), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDYES), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDNO), SW_HIDE); break; case MB_ABORTRETRYIGNORE: ShowWindow(GetDlgItem(hwnd, IDOK), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDCANCEL), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDYES), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDNO), SW_HIDE); break; case MB_YESNO: ShowWindow(GetDlgItem(hwnd, IDCANCEL), SW_HIDE); /* fall through */ case MB_YESNOCANCEL: ShowWindow(GetDlgItem(hwnd, IDOK), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDABORT), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDRETRY), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDIGNORE), SW_HIDE); break; case MB_RETRYCANCEL: ShowWindow(GetDlgItem(hwnd, IDOK), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDABORT), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDIGNORE), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDYES), SW_HIDE); ShowWindow(GetDlgItem(hwnd, IDNO), SW_HIDE); break; } /* Set the icon */ switch(lpmb->dwStyle & MB_ICONMASK) { case MB_ICONEXCLAMATION: SendDlgItemMessageA(hwnd, stc1, STM_SETICON, LoadIconA(0, IDI_EXCLAMATIONA), 0); break; case MB_ICONQUESTION: SendDlgItemMessageA(hwnd, stc1, STM_SETICON, LoadIconA(0, IDI_QUESTIONA), 0); break; case MB_ICONASTERISK: SendDlgItemMessageA(hwnd, stc1, STM_SETICON, LoadIconA(0, IDI_ASTERISKA), 0); break; case MB_ICONHAND: SendDlgItemMessageA(hwnd, stc1, STM_SETICON, LoadIconA(0, IDI_HANDA), 0); break; default: /* By default, Windows 95/98/NT do not associate an icon to message boxes. * So wine should do the same. */ break; } /* Position everything */ GetWindowRect(hwnd, &rect); borheight = rect.bottom - rect.top; borwidth = rect.right - rect.left; GetClientRect(hwnd, &rect); borheight -= rect.bottom - rect.top; borwidth -= rect.right - rect.left; /* Get the icon height */ GetWindowRect(GetDlgItem(hwnd, MSGBOX_IDICON), &rect); MapWindowPoints(0, hwnd, (LPPOINT)&rect, 2); iheight = rect.bottom - rect.top; ileft = rect.left; iwidth = rect.right - ileft; hdc = GetDC(hwnd); if (hFont) hPrevFont = SelectObject(hdc, hFont); /* Get the number of visible buttons and their size */ bh = bw = 1; /* Minimum button sizes */ for (buttons = 0, i = 1; i < 8; i++) { hItem = GetDlgItem(hwnd, i); if (GetWindowLongA(hItem, GWL_STYLE) & WS_VISIBLE) { char buttonText[1024]; int w, h; buttons++; if (GetWindowTextA(hItem, buttonText, sizeof buttonText)) { DrawTextA( hdc, buttonText, -1, &rect, DT_LEFT | DT_EXPANDTABS | DT_CALCRECT); h = rect.bottom - rect.top; w = rect.right - rect.left; if (h > bh) bh = h; if (w > bw) bw = w ; } } } bw = max(bw, bh * 2); /* Button white space */ bh = bh * 2; bw = bw * 2; bspace = bw/3; /* Space between buttons */ /* Get the text size */ GetClientRect(GetDlgItem(hwnd, MSGBOX_IDTEXT), &rect); rect.top = rect.left = rect.bottom = 0; DrawTextA( hdc, lpszText, -1, &rect, DT_LEFT | DT_EXPANDTABS | DT_WORDBREAK | DT_CALCRECT); /* Min text width corresponds to space for the buttons */ tleft = 2 * ileft + iwidth; twidth = max((bw + bspace) * buttons + bspace - tleft, rect.right); theight = rect.bottom; if (hFont) SelectObject(hdc, hPrevFont); ReleaseDC(hItem, hdc); tiheight = 16 + max(iheight, theight); wwidth = tleft + twidth + ileft + borwidth; wheight = 8 + tiheight + bh + borheight; /* Resize the window */ SetWindowPos(hwnd, 0, 0, 0, wwidth, wheight, SWP_NOMOVE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOREDRAW); /* Position the icon */ SetWindowPos(GetDlgItem(hwnd, MSGBOX_IDICON), 0, ileft, (tiheight - iheight) / 2, 0, 0, SWP_NOSIZE | SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOREDRAW); /* Position the text */ SetWindowPos(GetDlgItem(hwnd, MSGBOX_IDTEXT), 0, tleft, (tiheight - theight) / 2, twidth, theight, SWP_NOZORDER | SWP_NOACTIVATE | SWP_NOREDRAW); /* Position the buttons */ bpos = (wwidth - (bw + bspace) * buttons + bspace) / 2; for (buttons = i = 0; i < 7; i++) { /* some arithmetic to get the right order for YesNoCancel windows */ hItem = GetDlgItem(hwnd, (i + 5) % 7 + 1); if (GetWindowLongA(hItem, GWL_STYLE) & WS_VISIBLE) { if (buttons++ == ((lpmb->dwStyle & MB_DEFMASK) >> 8)) { SetFocus(hItem); SendMessageA( hItem, BM_SETSTYLE, BS_DEFPUSHBUTTON, TRUE ); } SetWindowPos(hItem, 0, bpos, tiheight, bw, bh, SWP_NOZORDER|SWP_NOACTIVATE|SWP_NOREDRAW); bpos += bw + bspace; } } /* handle modal MessageBoxes */ if (lpmb->dwStyle & (MB_TASKMODAL|MB_SYSTEMMODAL)) { FIXME("%s modal msgbox ! Not modal yet.\n", lpmb->dwStyle & MB_TASKMODAL ? "task" : "system"); /* Probably do EnumTaskWindows etc. here for TASKMODAL * and work your way up to the top - I'm lazy (HWND_TOP) */ SetWindowPos(hwnd, HWND_TOP, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE); if (lpmb->dwStyle & MB_TASKMODAL) /* at least MB_TASKMODAL seems to imply a ShowWindow */ ShowWindow(hwnd, SW_SHOW); } if (lpmb->dwStyle & MB_APPLMODAL) FIXME("app modal msgbox ! Not modal yet.\n"); return hFont; } /************************************************************************** * MSGBOX_DlgProc * * Dialog procedure for message boxes. */ static LRESULT CALLBACK MSGBOX_DlgProc( HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam ) { static HFONT hFont; switch(message) { case WM_INITDIALOG: hFont = MSGBOX_OnInit(hwnd, (LPMSGBOXPARAMSA)lParam); return 0; case WM_COMMAND: switch (wParam) { case IDOK: case IDCANCEL: case IDABORT: case IDRETRY: case IDIGNORE: case IDYES: case IDNO: EndDialog(hwnd, wParam); if (hFont) DeleteObject(hFont); break; } default: /* Ok. Ignore all the other messages */ TRACE("Message number 0x%04x is being ignored.\n", message); break; } return 0; } /************************************************************************** * MessageBoxA (USER32.@) * * NOTES * The WARN is here to help debug erroneous MessageBoxes * Use: -debugmsg warn+dialog,+relay */ INT WINAPI MessageBoxA(HWND hWnd, LPCSTR text, LPCSTR title, UINT type) { LPVOID template; HRSRC hRes; MSGBOXPARAMSA mbox; WARN("Messagebox\n"); if(!(hRes = FindResourceA(GetModuleHandleA("USER32"), "MSGBOX", RT_DIALOGA))) return 0; if(!(template = (LPVOID)LoadResource(GetModuleHandleA("USER32"), hRes))) return 0; if (!text) text="<WINE-NULL>"; if (!title) title="Error"; mbox.lpszCaption = title; mbox.lpszText = text; mbox.dwStyle = type; return DialogBoxIndirectParamA( GetWindowLongA(hWnd,GWL_HINSTANCE), template, hWnd, (DLGPROC)MSGBOX_DlgProc, (LPARAM)&mbox ); } /************************************************************************** * MessageBoxW (USER32.@) */ INT WINAPI MessageBoxW( HWND hwnd, LPCWSTR text, LPCWSTR title, UINT type ) { LPSTR titleA = HEAP_strdupWtoA( GetProcessHeap(), 0, title ); LPSTR textA = HEAP_strdupWtoA( GetProcessHeap(), 0, text ); INT ret; WARN("Messagebox\n"); ret = MessageBoxA( hwnd, textA, titleA, type ); HeapFree( GetProcessHeap(), 0, titleA ); HeapFree( GetProcessHeap(), 0, textA ); return ret; } /************************************************************************** * MessageBoxExA (USER32.@) */ INT WINAPI MessageBoxExA( HWND hWnd, LPCSTR text, LPCSTR title, UINT type, WORD langid ) { WARN("Messagebox\n"); /* ignore language id for now */ return MessageBoxA(hWnd,text,title,type); } /************************************************************************** * MessageBoxExW (USER32.@) */ INT WINAPI MessageBoxExW( HWND hWnd, LPCWSTR text, LPCWSTR title, UINT type, WORD langid ) { WARN("Messagebox\n"); /* ignore language id for now */ return MessageBoxW(hWnd,text,title,type); } /************************************************************************** * MessageBoxIndirectA (USER32.@) */ INT WINAPI MessageBoxIndirectA( LPMSGBOXPARAMSA msgbox ) { LPVOID template; HRSRC hRes; WARN("Messagebox\n"); if(!(hRes = FindResourceA(GetModuleHandleA("USER32"), "MSGBOX", RT_DIALOGA))) return 0; if(!(template = (LPVOID)LoadResource(GetModuleHandleA("USER32"), hRes))) return 0; return DialogBoxIndirectParamA( msgbox->hInstance, template, msgbox->hwndOwner, (DLGPROC)MSGBOX_DlgProc, (LPARAM)msgbox ); } /************************************************************************** * MessageBoxIndirectW (USER32.@) */ INT WINAPI MessageBoxIndirectW( LPMSGBOXPARAMSW msgbox ) { MSGBOXPARAMSA msgboxa; memcpy(&msgboxa,msgbox,sizeof(msgboxa)); msgboxa.lpszCaption = HEAP_strdupWtoA( GetProcessHeap(), 0, msgbox->lpszCaption ); msgboxa.lpszText = HEAP_strdupWtoA( GetProcessHeap(), 0, msgbox->lpszText ); msgboxa.lpszIcon = HEAP_strdupWtoA( GetProcessHeap(), 0, msgbox->lpszIcon ); return MessageBoxIndirectA(&msgboxa); }