Windows2000/private/ntos/w32/ntuser/client/dlgmgr.c
2020-09-30 17:12:32 +02:00

1439 lines
38 KiB
C

/*
* DLGMGR.C -
* Copyright (c) 1985 - 1999, Microsoft Corporation
* Dialog Box Manager Routines
* ??-???-???? mikeke Ported from Win 3.0 sources
* 12-Feb-1991 mikeke Added Revalidation code
* 19-Feb-1991 JimA Added access checks
*/
#include "precomp.h"
#pragma hdrstop
#define UNICODE_MINUS_SIGN 0x2212
LOOKASIDE DialogLookaside;
BOOL ValidateCallback(HANDLE h);
#define IsInForegroundQueue(hwnd) \
(NtUserQueryWindow(hwnd, WindowIsForegroundThread) != NULL)
#define IsCurrentThreadForeground() \
((BOOL)NtUserGetThreadState(UserThreadStateIsForeground))
/*
* GetParentDialog()
* Gets top level window, not a control parent. If not a dialog, then use
* "highest" control parent guy.
* BOGUS
* Need a way to mark a window as a dialog. If it ever comes into
* DefDlgProc(), set an internal flag. Will be used by thunking and
* CallDlgProc() optimizations also!
*/
PWND GetParentDialog(PWND pwndDialog)
{
PWND pwndParent;
pwndParent = pwndDialog;
// Walk up the parent chain. We're looking for the top-most dialog
// window. Most cases, the window is a top level one. But in case of
// backup app, the window will be a child of some other window.
for (; pwndDialog; pwndDialog = REBASEPWND(pwndDialog, spwndParent)) {
if (TestWF(pwndDialog, WFDIALOGWINDOW)) {
// For old guys: If not DS_RECURSE, then stop here.
// that way old apps which try to do the nested dialog
// stuff in their old limited way don't die.
if (TestWF(pwndDialog, WEFCONTROLPARENT))
pwndParent = pwndDialog;
else if (!TestWF(pwndDialog, DFCONTROL))
break;
}
if (!TestWF(pwndDialog, WFCHILD))
break;
}
return(pwndParent);
}
/*
* xxxSaveDlgFocus
* History:
* 02-18-92 JimA Ported from Win31 sources
*/
BOOL xxxSaveDlgFocus(
PWND pwnd)
{
HWND hwndFocus = GetFocus();
CheckLock(pwnd);
if (hwndFocus != NULL && IsChild(HWq(pwnd), hwndFocus) &&
PDLG(pwnd)->hwndFocusSave == NULL) {
PDLG(pwnd)->hwndFocusSave = hwndFocus;
xxxRemoveDefaultButton(pwnd, ValidateHwnd(hwndFocus));
return TRUE;
}
return FALSE;
}
/*
* xxxRestoreDlgFocus
* History:
* 02-18-92 JimA Ported from Win31 sources
*/
// LATER
// 21-Mar-1992 mikeke
// does pwndFocusSave need to be unlocked when the dialog is destroyed?
BOOL xxxRestoreDlgFocus(
PWND pwnd)
{
HWND hwndFocus;
HWND hwndFocusSave;
BOOL fRestored = FALSE;
CheckLock(pwnd);
if (PDLG(pwnd)->hwndFocusSave && !TestWF(pwnd, WFMINIMIZED)) {
hwndFocus = GetFocus();
hwndFocusSave = PDLG(pwnd)->hwndFocusSave;
if (IsWindow(hwndFocusSave)) {
xxxCheckDefPushButton(pwnd, hwndFocus, hwndFocusSave);
fRestored = (NtUserSetFocus(hwndFocusSave) != NULL);
#ifdef FE_IME
// after calling SetFocus(), we need to re-validate
// the window. PDLG(pwnd) might be NULL. This can
// be happened in FE environment where IME window
// exist. (kkntbug #12613)
if (IS_IME_ENABLED() && !ValidateDialogPwnd(pwnd)) {
return fRestored;
}
#endif
}
PDLG(pwnd)->hwndFocusSave = NULL;
}
return fRestored;
}
/*
* DlgSetFocus
* History:
*/
void DlgSetFocus(
HWND hwnd)
{
if (((UINT)SendMessage(hwnd, WM_GETDLGCODE, 0, 0)) & DLGC_HASSETSEL) {
SendMessage(hwnd, EM_SETSEL, 0, MAXLONG);
}
NtUserSetFocus(hwnd);
}
int GetDlgCtrlID(
HWND hwnd)
{
PWND pwnd;
pwnd = ValidateHwnd(hwnd);
if (pwnd == NULL)
return 0;
return PtrToLong(pwnd->spmenu);
}
/*
* ValidateDialogPwnd
* Under Win3, DLGWINDOWEXTRA is 30 bytes. We cannot change that for 16 bit
* compatibility reasons. Problem is there is no way to tell if a given
* 16 bit window depends on byte count. If there was, this would be easy.
* The only way to tell is when a window is about to be used as a dialog
* window. This window may be of the class DIALOGCLASS, but again it may
* not!! So we keep dialog window words at 30 bytes, and allocate another
* structure for the real dialog structure fields. Problem is that this
* structure has to be created lazily! And that's what we're doing here.
* 05-21-91 ScottLu Created.
*/
BOOL ValidateDialogPwnd(
PWND pwnd)
{
static BOOL sfInit = TRUE;
PDLG pdlg;
/*
* This bit is set if we've already run through this initialization and
* have identified this window as a dialog window (able to withstand
* peeks into window words at random moments in time).
*/
if (TestWF(pwnd, WFDIALOGWINDOW))
return TRUE;
if (pwnd->cbwndExtra < DLGWINDOWEXTRA) {
RIPERR0(ERROR_WINDOW_NOT_DIALOG, RIP_VERBOSE, "");
return FALSE;
}
/*
* See if the pdlg was destroyed and this is a rogue message to be ignored
*/
if (pwnd->fnid & FNID_STATUS_BITS) {
return FALSE;
}
/*
* If the lookaside buffer has not been initialized, do it now.
*/
if (sfInit) {
if (!NT_SUCCESS(InitLookaside(&DialogLookaside, sizeof(DLG), 2))) {
return FALSE;
}
sfInit = FALSE;
}
if ((pdlg = (PDLG)AllocLookasideEntry(&DialogLookaside)) == NULL) {
return FALSE;
}
NtUserCallHwndParam(HWq(pwnd), (ULONG_PTR)pdlg, SFI_SETDIALOGPOINTER);
return TRUE;
}
/*
* CvtDec
* LATER!!! convert to itoa?
* History:
*/
void CvtDec(
int u,
LPWSTR *lplpch)
{
if (u >= 10) {
CvtDec(u / 10, lplpch);
u %= 10;
}
*(*lplpch)++ = (WCHAR)(u + '0');
}
/*
* SetDlgItemInt
* History:
*/
BOOL SetDlgItemInt(
HWND hwnd,
int item,
UINT u,
BOOL fSigned)
{
LPWSTR lpch;
WCHAR rgch[16];
lpch = rgch;
if (fSigned) {
if ((int)u < 0) {
*lpch++ = TEXT('-');
u = (UINT)(-(int)u);
}
} else {
if (u & 0x80000000) {
CvtDec(u / 10, (LPWSTR FAR *)&lpch);
u = u % 10;
}
}
CvtDec(u, (LPWSTR FAR *)&lpch);
*lpch = 0;
return SetDlgItemTextW(hwnd, item, rgch);
}
/*
* CheckDlgButton
* History:
*/
BOOL CheckDlgButton(
HWND hwnd,
int id,
UINT cmdCheck)
{
if ((hwnd = GetDlgItem(hwnd, id)) == NULL) {
return FALSE;
}
SendMessage(hwnd, BM_SETCHECK, cmdCheck, 0);
return TRUE;
}
/*
* GetDlgItemInt
* History:
*/
UINT GetDlgItemInt(
HWND hwnd,
int item,
BOOL FAR *lpfValOK,
BOOL fSigned)
{
int i, digit, ch;
int maxTens, maxUnits;
BOOL fOk, fNeg;
LPWSTR lpch;
WCHAR rgch[48];
WCHAR rgchDigits[48];
fOk = FALSE;
if (lpfValOK != NULL)
*lpfValOK = FALSE;
if (!GetDlgItemTextW(hwnd, item, rgch, sizeof(rgch) / sizeof(WCHAR) - 1))
return 0;
lpch = rgch;
/*
* Skip leading white space.
*/
while (*lpch == TEXT(' '))
lpch++;
fNeg = FALSE;
while (fSigned && ((*lpch == L'-') || (*lpch == UNICODE_MINUS_SIGN))) {
lpch++;
fNeg ^= TRUE;
}
if (fSigned) {
maxTens = INT_MAX / 10;
maxUnits = INT_MAX % 10;
} else {
maxTens = UINT_MAX / 10;
maxUnits = UINT_MAX % 10;
}
/*
* Convert all decimal digits to ASCII Unicode digits 0x0030 - 0x0039
*/
FoldStringW(MAP_FOLDDIGITS, lpch, -1, rgchDigits,
sizeof(rgchDigits) / sizeof(rgchDigits[0]));
lpch = rgchDigits;
i = 0;
while (ch = *lpch++) {
digit = ch - TEXT('0');
if (digit < 0 || digit > 9) {
break;
}
if (i >= maxTens) {
/*
* We need to special case INT_MIN as the i = -i
* would damage it
*/
if (i == maxTens) {
if (digit == maxUnits + 1 && fNeg && (*lpch) == 0) {
i = INT_MIN;
goto HaveResult;
} else if (digit > maxUnits) {
return 0;
}
} else {
return 0;
}
}
fOk = TRUE;
i = ((UINT)i * 10) + digit;
}
if (fNeg)
i = -i;
HaveResult:
if (lpfValOK != NULL)
*lpfValOK = ((ch == 0) && fOk);
return (UINT)i;
}
/*
* CheckRadioButton
* History:
*/
BOOL CheckRadioButton(
HWND hwnd,
int idFirst,
int idLast,
int id)
{
PWND pwnd, pwndDialog;
BOOL fCheckOn;
pwndDialog = ValidateHwnd(hwnd);
if (pwndDialog == NULL)
return 0;
for (pwnd = REBASE(pwndDialog, spwndChild); pwnd; pwnd = REBASE(pwnd, spwndNext)) {
if ((PtrToLong(pwnd->spmenu) >= idFirst) &&
(PtrToLong(pwnd->spmenu) <= idLast)) {
fCheckOn = (PtrToLong(pwnd->spmenu) == id);
SendMessage(PtoHq(pwnd), BM_SETCHECK, fCheckOn, 0L);
}
}
return TRUE;
}
/*
* IsDlgButtonChecked
* History:
*/
UINT IsDlgButtonChecked(
HWND hwnd,
int id)
{
if ((hwnd = GetDlgItem(hwnd, id)) != NULL) {
return (UINT)SendMessage(hwnd, BM_GETCHECK, 0, 0);
}
return FALSE;
}
/*
* DefDlgProc
* History:
*/
LRESULT DefDlgProcWorker(
PWND pwnd,
UINT message,
WPARAM wParam,
LPARAM lParam,
DWORD fAnsi)
{
HWND hwnd = HWq(pwnd);
TL tlpwndT1, tlpwndT2, tlpwndT3, tlpwndTop;
PWND pwndT;
PWND pwndT1, pwndT2, pwndT3, pwndTop;
HWND hwndT1;
LRESULT result;
BOOL fSetBit;
DLGPROC pfn;
CheckLock(pwnd);
/*
* use the Win 3.1 documented size
*/
VALIDATECLASSANDSIZE(pwnd, FNID_DIALOG);
/*
* Must do special validation here to make sure pwnd is a dialog window.
*/
if (!ValidateDialogPwnd(pwnd))
return 0;
if (((PDIALOG)pwnd)->resultWP != 0)
NtUserSetWindowLongPtr(hwnd, DWLP_MSGRESULT, 0, FALSE);
result = 0; // no dialog proc
if (message == WM_FINALDESTROY) {
goto DoCleanup;
}
if ((pfn = PDLG(pwnd)->lpfnDlg) != NULL) {
/* Bug 234292 - joejo
* Since the called window/dialog proc may have a different calling
* convention, we must wrap the call and, check esp and replace with
* a good esp when the call returns. This is what UserCallWinProc* does.
*/
result = CALLPROC_WOWCHECKPWW_DLG(pfn, hwnd, message, wParam, lParam, &(pwnd->state));
/*
* Get out if the window was destroyed in the dialog proc.
*/
if ((RevalidateHwnd(hwnd) == NULL) || (pwnd->fnid & FNID_STATUS_BITS))
return result;
}
/*
* SPECIAL CASED ... and DOCUMENTED that way !!!
* These 6, and ONLY these 6, should be hacked in this fashion.
* Anybody who needs the REAL return value to a message should
* use SetDlgMsgResult in WINDOWSX.H
*/
switch (message) {
case WM_COMPAREITEM:
case WM_VKEYTOITEM:
case WM_CHARTOITEM:
case WM_INITDIALOG:
case WM_QUERYDRAGICON:
return ((LRESULT)(DWORD)result);
case WM_CTLCOLOR:
case WM_CTLCOLORMSGBOX:
case WM_CTLCOLOREDIT:
case WM_CTLCOLORLISTBOX:
case WM_CTLCOLORBTN:
case WM_CTLCOLORDLG:
case WM_CTLCOLORSCROLLBAR:
case WM_CTLCOLORSTATIC:
// QuarkXPress doesn't like finding the WM_CTLCOLOR result in
// resultWP -- we should never be setting resultWP -- that's meant
// as a pass-thru return value -- so let's go back to doing it the
// old way -- Win95B B#21269 -- 03/13/95 -- tracysh (cr: jeffbog)
if (result)
return result;
break;
}
if (!result) {
/*
* Save the result value in case our private memory is freed
* before we return
*/
// result = PDLG(pwnd)->resultWP;
switch (message) {
case WM_CTLCOLOR:
case WM_CTLCOLORMSGBOX:
case WM_CTLCOLOREDIT:
case WM_CTLCOLORLISTBOX:
case WM_CTLCOLORBTN:
case WM_CTLCOLORDLG:
case WM_CTLCOLORSCROLLBAR:
case WM_CTLCOLORSTATIC:
{
// HACK OF DEATH:
// To get 3D colors for non 4.0 apps who use 3DLOOK,
// we temporarily add on the 4.0 compat bit, pass this
// down to DWP, and clear it.
// Use "result" var for bool saying we have to add/clear 4.0
// compat bit.
fSetBit = (TestWF(pwnd, DF3DLOOK) != 0) &&
(TestWF(pwnd, WFWIN40COMPAT) == 0);
if (fSetBit)
SetWindowState(pwnd, WFWIN40COMPAT);
result = DefWindowProcWorker(pwnd, message,
wParam, lParam, fAnsi);
if (fSetBit)
ClearWindowState(pwnd, WFWIN40COMPAT);
return result;
}
case WM_ERASEBKGND:
FillWindow(hwnd, hwnd, (HDC)wParam, (HBRUSH)CTLCOLOR_DLG);
return TRUE;
case WM_SHOWWINDOW:
/*
* If hiding the window, save the focus. If showing the window
* by means of a SW_* command and the fEnd bit is set, do not
* pass to DWP so it won't get shown.
*/
if (GetParentDialog(pwnd) == pwnd) {
if (!wParam) {
xxxSaveDlgFocus(pwnd);
} else {
if (LOWORD(lParam) != 0 && PDLG(pwnd)->fEnd)
break;
/*
* Snap the cursor to the center of the default button.
* Only do this if the current thread is in the foreground.
* The _ShowCursor() code is added to work around a
* problem with hardware cursors. If change is done
* in the same refresh cycle, the display of the cursor
* would not reflect the new position.
*/
if (TEST_PUSIF(PUSIF_SNAPTO) &&
IsInForegroundQueue(hwnd)) {
hwndT1 = GetDlgItem(hwnd, (int)PDLG(pwnd)->result);
if (hwndT1) {
RECT rc;
NtUserShowCursor(FALSE);
GetWindowRect(hwndT1, &rc);
NtUserSetCursorPos(rc.left + ((rc.right - rc.left) / 2),
rc.top + ((rc.bottom - rc.top) / 2));
NtUserShowCursor(TRUE);
}
}
}
}
goto CallDWP;
case WM_SYSCOMMAND:
if (GetParentDialog(pwnd) == pwnd) {
/*
* If hiding the window, save the focus. If showing the window
* by means of a SW_* command and the fEnd bit is set, do not
* pass to DWP so it won't get shown.
*/
if ((int)wParam == SC_MINIMIZE)
xxxSaveDlgFocus(pwnd);
}
goto CallDWP;
case WM_ACTIVATE:
pwndT1 = GetParentDialog(pwnd);
if (pwndT1 != pwnd) {
/*
* This random bit is used during key processing - bit
* 08000000 of WM_CHAR messages is set if a dialog is currently
* active.
*/
NtUserSetThreadState(wParam ? QF_DIALOGACTIVE : 0, QF_DIALOGACTIVE);
}
ThreadLock(pwndT1, &tlpwndT1);
if (wParam != 0)
xxxRestoreDlgFocus(pwndT1);
else
xxxSaveDlgFocus(pwndT1);
ThreadUnlock(&tlpwndT1);
break;
case WM_SETFOCUS:
pwndT1 = GetParentDialog(pwnd);
if (!PDLG(pwndT1)->fEnd && !xxxRestoreDlgFocus(pwndT1)) {
pwndT = _GetNextDlgTabItem(pwndT1, NULL, FALSE);
DlgSetFocus(HW(pwndT));
}
break;
case WM_CLOSE:
/*
* Make sure cancel button is not disabled before sending the
* IDCANCEL. Note that we need to do this as a message instead
* of directly calling the dlg proc so that any dialog box
* filters get this.
*/
pwndT1 = _GetDlgItem(pwnd, IDCANCEL);
if (pwndT1 && TestWF(pwndT1, WFDISABLED))
NtUserMessageBeep(0);
else
PostMessage(hwnd, WM_COMMAND, MAKELONG(IDCANCEL, BN_CLICKED),
(LPARAM)HW(pwndT1));
break;
case WM_NCDESTROY:
case WM_FINALDESTROY:
DoCleanup:
NtUserSetThreadState(0, QF_DIALOGACTIVE);
if (!(pwnd->style & DS_LOCALEDIT)) {
if (PDLG(pwnd)->hData) {
ReleaseEditDS(PDLG(pwnd)->hData);
PDLG(pwnd)->hData = NULL;
}
}
/*
* Delete the user defined font if any
*/
if (PDLG(pwnd)->hUserFont) {
DeleteObject(PDLG(pwnd)->hUserFont);
PDLG(pwnd)->hUserFont = NULL;
}
/*
* Free the dialog memory and mark this as a non-dialog window
*/
FreeLookasideEntry(&DialogLookaside, PDLG(pwnd));
NtUserCallHwndParam(hwnd, 0, SFI_SETDIALOGPOINTER);
break;
case DM_REPOSITION:
{
RECT rc;
PMONITOR pMonitor;
// DAT recorder APP sends it's own private message 0x402
// through and we mistake it to be DM_REPOSITION. To avoid
// this confusion, we do the following check.
// Fix for Bug#25747 -- 9/29/94 --
if (!TestWF(pwnd, WEFCONTROLPARENT) ||
(GETFNID(pwnd) != FNID_DESKTOP &&
GETFNID(REBASEPWND(pwnd, spwndParent)) != FNID_DESKTOP)) {
goto CallDWP;
}
CopyRect(&rc, &pwnd->rcWindow);
pMonitor = _MonitorFromRect(&rc, MONITOR_DEFAULTTOPRIMARY);
RepositionRect(pMonitor, &rc, pwnd->style, pwnd->ExStyle);
NtUserSetWindowPos(hwnd, HWND_TOP, rc.left, rc.top,
rc.right - rc.left, rc.bottom - rc.top,
SWP_NOZORDER | SWP_NOSIZE | SWP_NOACTIVATE);
}
break;
case DM_SETDEFID:
pwndT1 = GetParentDialog(pwnd);
ThreadLock(pwndT1, &tlpwndT1);
if (!(PDLG(pwndT1)->fEnd)) {
pwndT2 = NULL;
if (PDLG(pwndT1)->result != 0)
pwndT2 = _FindDlgItem(pwndT1, (int)PDLG(pwndT1)->result);
pwndT3 = NULL;
if (wParam != 0) {
pwndT3 = _GetDlgItem(pwnd, (UINT)wParam);
}
ThreadLock(pwndT2, &tlpwndT2);
ThreadLock(pwndT3, &tlpwndT3);
xxxCheckDefPushButton(pwndT1, HW(pwndT2), HW(pwndT3));
ThreadUnlock(&tlpwndT3);
ThreadUnlock(&tlpwndT2);
PDLG(pwndT1)->result = (UINT)wParam;
// if (PDLG(pwnd)->spwndFocusSave) {
// Lock(&(PDLG(pwnd)->spwndFocusSave), pwndT2);
// }
if (FWINABLE()) {
NotifyWinEvent(EVENT_OBJECT_DEFACTIONCHANGE, HW(pwndT1), OBJID_CLIENT, INDEXID_CONTAINER);
}
}
ThreadUnlock(&tlpwndT1);
return TRUE;
case DM_GETDEFID:
pwndT1 = GetParentDialog(pwnd);
if (!PDLG(pwndT1)->fEnd && PDLG(pwndT1)->result)
return(MAKELONG(PDLG(pwndT1)->result, DC_HASDEFID));
else
return 0;
break;
/*
* This message was added so that user defined controls that want
* tab keys can pass the tab off to the next/previous control in the
* dialog box. Without this, all they could do was set the focus
* which didn't do the default button stuff.
*/
case WM_NEXTDLGCTL:
pwndTop = GetParentDialog(pwnd);
ThreadLock(pwndTop, &tlpwndTop);
hwndT1 = GetFocus();
pwndT2 = ValidateHwndNoRip(hwndT1);
if (LOWORD(lParam)) {
if (pwndT2 == NULL)
pwndT2 = pwndTop;
/*
* wParam contains the pwnd of the ctl to set focus to.
*/
if ((pwndT1 = ValidateHwnd((HWND)wParam)) == NULL) {
ThreadUnlock(&tlpwndTop);
return TRUE;
}
} else {
if (pwndT2 == NULL) {
/*
* Set focus to the first tab item.
*/
pwndT1 = _GetNextDlgTabItem(pwndTop, NULL, FALSE);
pwndT2 = pwndTop;
} else {
/*
* If window with focus not a dlg ctl, ignore message.
*/
if (!_IsChild(pwndTop, pwndT2)) {
ThreadUnlock(&tlpwndTop);
return TRUE;
}
/*
* wParam = TRUE for previous, FALSE for next
*/
pwndT1 = _GetNextDlgTabItem(pwndTop, pwndT2, (wParam != 0));
/*
* If there is no next item, ignore the message.
*/
if (pwndT1 == NULL) {
ThreadUnlock(&tlpwndTop);
return TRUE;
}
}
}
ThreadLock(pwndT1, &tlpwndT1);
ThreadLock(pwndT2, &tlpwndT2);
DlgSetFocus(HW(pwndT1));
xxxCheckDefPushButton(pwndTop, HW(pwndT2), HW(pwndT1));
ThreadUnlock(&tlpwndT2);
ThreadUnlock(&tlpwndT1);
ThreadUnlock(&tlpwndTop);
return TRUE;
case WM_ENTERMENULOOP:
/*
* We need to pop up the combo box window if the user brings
* down a menu.
* ... FALL THROUGH...
*/
case WM_LBUTTONDOWN:
case WM_NCLBUTTONDOWN:
hwndT1 = GetFocus();
if (hwndT1 != NULL) {
pwndT1 = ValidateHwndNoRip(hwndT1);
if (GETFNID(pwndT1) == FNID_COMBOBOX) {
/*
* If user clicks anywhere in dialog box and a combo box (or
* the editcontrol of a combo box) has the focus, then hide
* it's listbox.
*/
ThreadLockAlways(pwndT1, &tlpwndT1);
SendMessage(HWq(pwndT1), CB_SHOWDROPDOWN, FALSE, 0);
ThreadUnlock(&tlpwndT1);
} else {
PWND pwndParent;
/*
* It's a subclassed combo box. See if the listbox and edit
* boxes exist (this is a very cheezy evaluation - what if
* these controls are subclassed too? NOTE: Not checking
* for EditWndProc: it's a client proc address.
*/
pwndParent = REBASEPWND(pwndT1, spwndParent);
if (GETFNID(pwndParent) == FNID_COMBOBOX) {
pwndT1 = pwndParent;
ThreadLock(pwndT1, &tlpwndT1);
SendMessage(HWq(pwndT1), CB_SHOWDROPDOWN, FALSE, 0);
ThreadUnlock(&tlpwndT1);
}
}
}
/*
* Always send the message off to DefWndProc
*/
goto CallDWP;
case WM_GETFONT:
return (LRESULT)PDLG(pwnd)->hUserFont;
case WM_VKEYTOITEM:
case WM_COMPAREITEM:
case WM_CHARTOITEM:
case WM_INITDIALOG:
/*
* We need to return the 0 the app may have returned for these
* items instead of calling defwindow proc.
*/
return result;
case WM_NOTIFYFORMAT:
if (lParam == NF_QUERY)
return((PDLG(pwnd)->flags & DLGF_ANSI) ? NFR_ANSI : NFR_UNICODE);
return result;
case WM_INPUTLANGCHANGEREQUEST:
if (IS_IME_ENABLED() || IS_MIDEAST_ENABLED()) {
/*
* #115190
* For dialogbox itself, buttons/static controls on top of
* dialogbox, we'll simply discard this message. B#3843-win95c
*/
break;
}
if (PDLG(pwnd)->lpfnDlg == MB_DlgProcA ||
PDLG(pwnd)->lpfnDlg == MB_DlgProcW) {
break;
}
goto CallDWP;
default:
CallDWP:
return DefWindowProcWorker(pwnd, message, wParam, lParam, fAnsi);
}
} else if ((message == WM_SHOWWINDOW) && result) {
/*
* For a visible-case we want to snap the cursor regardless of
* what was returned from the dialog-handler on the client. If
* we're going visible, snap the cursor to the dialog-button.
*/
if (GetParentDialog(pwnd) == pwnd) {
if (wParam && ((LOWORD(lParam) == 0) || !PDLG(pwnd)->fEnd)) {
/*
* Snap the cursor to the center of the default button.
* Only do this if the current thread is in the foreground.
* The _ShowCursor() code is added to work around a
* problem with hardware cursors. If change is done
* in the same refresh cycle, the display of the cursor
* would not reflect the new position.
*/
if (TEST_PUSIF(PUSIF_SNAPTO) &&
IsInForegroundQueue(hwnd)) {
hwndT1 = GetDlgItem(hwnd, (int)PDLG(pwnd)->result);
if (hwndT1) {
RECT rc;
NtUserShowCursor(FALSE);
GetWindowRect(hwndT1, &rc);
NtUserSetCursorPos(rc.left + ((rc.right - rc.left) / 2),
rc.top + ((rc.bottom - rc.top) / 2));
NtUserShowCursor(TRUE);
}
}
}
}
}
/*
* If this is still marked as a dialog window then return the real
* result. Otherwise, we've already processed the WM_NCDESTROY message
* and freed our private memory so return the stored value.
*/
if (TestWF(pwnd, WFDIALOGWINDOW))
return KERNEL_LRESULT_TO_LRESULT(((PDIALOG)pwnd)->resultWP);
else
return result;
}
/*
* DefDlgProc
* Translates the message, calls DefDlgProc on server side. DefDlgProc
* is the default WindowProc for dialogs (NOT the dialog's dialog proc)
* 04-11-91 ScottLu Created.
*/
LRESULT WINAPI DefDlgProcW(
HWND hwnd,
UINT message,
WPARAM wParam,
LPARAM lParam)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return (0L);
}
return DefDlgProcWorker(pwnd, message, wParam, lParam, FALSE);
}
LRESULT WINAPI DefDlgProcA(
HWND hwnd,
UINT message,
WPARAM wParam,
LPARAM lParam)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return (0L);
}
return DefDlgProcWorker(pwnd, message, wParam, lParam, TRUE);
}
/*
* DialogBox2
* History:
*/
INT_PTR DialogBox2(
HWND hwnd,
HWND hwndOwner,
BOOL fDisabled,
BOOL fOwnerIsActiveWindow)
{
MSG msg;
INT_PTR result;
BOOL fShown;
BOOL fWantIdleMsgs;
BOOL fSentIdleMessage = FALSE;
HWND hwndCapture;
PWND pwnd;
if (hwnd) {
pwnd = ValidateHwnd(hwnd);
} else {
pwnd = NULL;
}
CheckLock(pwnd);
if (pwnd == NULL) {
if ((hwndOwner != NULL) && !fDisabled && IsWindow(hwndOwner)) {
NtUserEnableWindow(hwndOwner, TRUE);
if (fOwnerIsActiveWindow) {
/*
* The dialog box failed but we disabled the owner in
* xxxDialogBoxIndirectParam and if it had the focus, the
* focus was set to NULL. Now, when we enable the window, it
* doesn't get the focus back if it had it previously so we
* need to correct this.
*/
NtUserSetFocus(hwndOwner);
}
}
return -1;
}
hwndCapture = GetCapture();
if (hwndCapture != NULL) {
SendMessage(hwndCapture, WM_CANCELMODE, 0, 0);
}
/*
* Set the 'parent disabled' flag for EndDialog().
* convert BOOL to definite bit 0 or 1
*/
PDLG(pwnd)->fDisabled = !!fDisabled;
fShown = TestWF(pwnd, WFVISIBLE);
/*
* Should the WM_ENTERIDLE messages be sent?
*/
fWantIdleMsgs = !(pwnd->style & DS_NOIDLEMSG);
if ((SYSMET(SLOWMACHINE) & 1) && !fShown && !PDLG(pwnd)->fEnd)
goto ShowIt;
while (PDLG(pwnd) && (!PDLG(pwnd)->fEnd)) {
if (!PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)) {
ShowIt:
if (!fShown) {
fShown = TRUE;
#ifdef SYSMODALWINDOWS
if (pwnd == gspwndSysModal) {
/*
* Make this a topmost window
*/
NtUserSetWindowPos(hwnd, HWND_TOPMOST, 0, 0, 0, 0,
SWP_NOSIZE | SWP_NOMOVE |
SWP_NOREDRAW | SWP_NOACTIVATE);
}
#endif
NtUserShowWindow(hwnd, SHOW_OPENWINDOW);
UpdateWindow(hwnd);
if (FWINABLE()) {
NotifyWinEvent(EVENT_SYSTEM_DIALOGSTART, hwnd, OBJID_WINDOW, INDEXID_CONTAINER);
}
} else {
/*
* Make sure window still exists
*/
if (hwndOwner && !IsWindow(hwndOwner))
hwndOwner = NULL;
if (hwndOwner && fWantIdleMsgs && !fSentIdleMessage) {
fSentIdleMessage = TRUE;
SendMessage(hwndOwner, WM_ENTERIDLE, MSGF_DIALOGBOX, (LPARAM)hwnd);
} else {
if ((RevalidateHwnd(hwnd) == NULL) || (pwnd->fnid & FNID_STATUS_BITS))
break;
NtUserWaitMessage();
}
}
} else {
/*
* We got a real message. Reset fSentIdleMessage so that we send
* one next time things are calm.
*/
fSentIdleMessage = FALSE;
if (msg.message == WM_QUIT) {
PostQuitMessage((int)msg.wParam);
break;
}
/*
* If pwnd is a message box, allow Ctrl-C and Ctrl-Ins
* to copy its content to the clipboard.
* Fall through in case hooking apps look for these keys.
*/
if (TestWF(pwnd, WFMSGBOX)) {
if ((msg.message == WM_CHAR && LOBYTE(msg.wParam) == 3) ||
(msg.message == WM_KEYDOWN && LOBYTE(msg.wParam) == VK_INSERT && GetKeyState(VK_CONTROL) < 0)) {
/*
* Send the WM_COPY message and let the original message fall through
* as some apps might want it
*/
SendMessage(hwnd, WM_COPY, 0, 0);
}
}
/*
* Moved the msg filter hook call to IsDialogMessage to allow
* messages to be hooked for both modal and modeless dialog
* boxes.
*/
if (!IsDialogMessage(hwnd, &msg)) {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
/*
* If we get a timer message, go ahead and show the window.
* We may continuously get timer msgs if there are zillions of
* apps running.
* If we get a syskeydown message, show the dialog box because
* the user may be bringing down a menu and we want the dialog
* box to become visible.
*/
if (!fShown && (msg.message == WM_TIMER ||
msg.message == WM_SYSTIMER || msg.message == WM_SYSKEYDOWN))
goto ShowIt;
}
if (!RevalidateHwnd(hwnd)) {
/*
* Bogus case - we've already been destroyed somehow (by app,
* GP, etc.)
*/
RIPMSG0(RIP_WARNING,
"Dialog should be dismissed with EndDialog, not DestroyWindow");
break;
}
}
if (FWINABLE()) {
NotifyWinEvent(EVENT_SYSTEM_DIALOGEND, hwnd, OBJID_WINDOW, INDEXID_CONTAINER);
}
/*
* Make sure the window still exists
*/
if (!RevalidateHwnd(hwnd)) {
return 0;
}
if (PDLG(pwnd))
result = KERNEL_INT_PTR_TO_INT_PTR(PDLG(pwnd)->result);
else
result = 0;
NtUserDestroyWindow(hwnd);
/*
* If the owner window belongs to another thread, the reactivation
* of the owner may have failed within DestroyWindow(). Therefore,
* if the current thread is in the foreground and the owner is not
* in the foreground we can safely set the foreground back
* to the owner.
*/
if (hwndOwner != NULL) {
if (IsCurrentThreadForeground() &&
!IsInForegroundQueue(hwndOwner)) {
NtUserSetForegroundWindow(hwndOwner);
}
}
return result;
}
/*
* InternalDialogBox
* Server portion of DialogBoxIndirectParam.
* 04-05-91 ScottLu Created.
*/
INT_PTR InternalDialogBox(
HANDLE hModule,
LPDLGTEMPLATE lpdt,
HWND hwndOwner,
DLGPROC pfnDialog,
LPARAM lParam,
UINT fSCDLGFlags)
{
INT_PTR i;
BOOL fDisabled;
HWND hwnd;
PWND pwndOwner;
BOOL fOwnerIsActiveWindow = FALSE;
TL tlpwndOwner;
BOOL fUnlockOwner;
UserAssert(!(fSCDLGFlags & ~(SCDLG_CLIENT | SCDLG_ANSI | SCDLG_16BIT))); // These are the only valid flags
/*
* If hwndOwner == HWNDESKTOP, change it to NULL. This way the desktop
* (and all its children) won't be disabled if the dialog is modal.
*/
if (hwndOwner && SAMEWOWHANDLE(hwndOwner, GetDesktopWindow()))
hwndOwner = NULL;
/*
* We return 0 if the ValidateHwnd fails in order to match Win 3.1
* validation layer which always returns 0 for invalid hwnds even
* if the function is spec'ed to return -1. Autocad setup bug #3615
*/
if (hwndOwner) {
if ((pwndOwner = ValidateHwnd(hwndOwner)) == NULL) {
return (0L);
}
} else {
pwndOwner = NULL;
}
CheckLock(pwndOwner);
fUnlockOwner = FALSE;
if (pwndOwner != NULL) {
/* The following fixes an AV in Corel Photo-Paint 6.0. It passes a
* 16-bit HWND in, and croaks at some point when it gets 16-bit hwnds
* back in send messages. FritzS -- fixing bug 12531
*/
hwndOwner = PtoHq(pwndOwner);
/*
* Make sure the owner is a top level window.
*/
if (TestwndChild(pwndOwner)) {
pwndOwner = GetTopLevelWindow(pwndOwner);
hwndOwner = HWq(pwndOwner);
ThreadLock(pwndOwner, &tlpwndOwner);
fUnlockOwner = TRUE;
}
/*
* Remember if window was originally disabled (so we can set
* the correct state when the dialog goes away.
*/
fDisabled = TestWF(pwndOwner, WFDISABLED);
fOwnerIsActiveWindow = (SAMEWOWHANDLE(hwndOwner, GetActiveWindow()));
/*
* Disable the window.
*/
NtUserEnableWindow(hwndOwner, FALSE);
}
/*
* Don't show cursors on a mouseless system. Put up an hour glass while
* the dialog comes up.
*/
if (SYSMET(MOUSEPRESENT)) {
NtUserSetCursor(LoadCursor(NULL, IDC_WAIT));
}
/*
* Creates the dialog. Frees the menu if this routine fails.
*/
hwnd = InternalCreateDialog(hModule, lpdt, 0, hwndOwner,
pfnDialog, lParam, fSCDLGFlags);
if (hwnd == NULL) {
/*
* The dialog creation failed. Re-enable the window, destroy the
* menu, ie., fail gracefully.
*/
if (!fDisabled && hwndOwner != NULL)
NtUserEnableWindow(hwndOwner, TRUE);
if (fUnlockOwner)
ThreadUnlock(&tlpwndOwner);
return -1;
}
i = DialogBox2(hwnd, hwndOwner, fDisabled, fOwnerIsActiveWindow);
if (fUnlockOwner)
ThreadUnlock(&tlpwndOwner);
return i;
}
/*
** RepositionRect()
** Used to ensure that toplevel dialogs are still visible within the
** desktop area after they've resized.
*/
void
RepositionRect(
PMONITOR pMonitor,
LPRECT lprc,
DWORD dwStyle,
DWORD dwExStyle)
{
LPRECT lprcClip;
int y;
UserAssert(lprc);
UserAssert(pMonitor);
if (dwStyle & WS_CHILD) {
if (dwExStyle & WS_EX_CONTROLPARENT)
return;
/*
* Old style 3.1 child dialogs--do this nonsense anyway. Keeps
* FedEx happy.
*/
pMonitor = GetPrimaryMonitor();
lprcClip = &pMonitor->rcMonitor;
} else if (dwExStyle & WS_EX_TOOLWINDOW) {
lprcClip = &pMonitor->rcMonitor;
} else {
lprcClip = &pMonitor->rcWork;
}
UserAssert(lprc);
y = lprcClip->bottom - (SYSMET(CYEDGE) * 2 + SYSMET(CYKANJIWINDOW));
if (lprc->bottom > y) {
OffsetRect(lprc, 0, y - lprc->bottom);
}
if (lprc->top < lprcClip->top) {
OffsetRect(lprc, 0, lprcClip->top - lprc->top);
}
if (lprc->right > lprcClip->right) {
OffsetRect(lprc, lprcClip->right - lprc->right, 0);
}
if (lprc->left < lprcClip->left) {
OffsetRect(lprc, lprcClip->left - lprc->left, 0);
}
}
/*
* MapDialogRect
* History:
*/
BOOL MapDialogRect(
HWND hwnd,
LPRECT lprc)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return FALSE;
}
/*
* Must do special validation here to make sure pwnd is a dialog window.
*/
if (!ValidateDialogPwnd(pwnd))
return FALSE;
lprc->left = XPixFromXDU(lprc->left, PDLG(pwnd)->cxChar);
lprc->right = XPixFromXDU(lprc->right, PDLG(pwnd)->cxChar);
lprc->top = YPixFromYDU(lprc->top, PDLG(pwnd)->cyChar);
lprc->bottom = YPixFromYDU(lprc->bottom, PDLG(pwnd)->cyChar);
return TRUE;
}