1274 lines
41 KiB
C
1274 lines
41 KiB
C
/*
|
|
* DLGBEGIN.C -
|
|
|
|
* Copyright (c) 1985 - 1999, Microsoft Corporation
|
|
|
|
* Dialog Initialization Routines
|
|
|
|
* ??-???-???? mikeke Ported from Win 3.0 sources
|
|
* 12-Feb-1991 mikeke Added Revalidation code
|
|
*/
|
|
|
|
#include "precomp.h"
|
|
#pragma hdrstop
|
|
|
|
BOOL ValidateCallback(HANDLE h);
|
|
|
|
CONST WCHAR szEDITCLASS[] = TEXT("Edit");
|
|
|
|
/*
|
|
* Fixed, hard coded literal for Dialog hacks
|
|
*/
|
|
const WCHAR gwszShellFont[] = L"MS Shell Dlg";
|
|
const WCHAR gwszShellFont2[] = L"MS Shell Dlg 2";
|
|
|
|
|
|
#ifdef USE_MIRRORING
|
|
#define MIRRORED_HWND(hwnd) (GetWindowLong(hwnd, GWL_EXSTYLE) & WS_EX_LAYOUTRTL)
|
|
#endif
|
|
|
|
/*
|
|
* Avoid sign extending 16 bit CW2_USEDEFAULT. We need this because the
|
|
* dialog resource template uses SHORT fields to store the coordinates
|
|
* but CreateWindow wants INT values.
|
|
|
|
* History:
|
|
* 12/04/96 GerardoB Created
|
|
*/
|
|
__inline int DefShortToInt(short s)
|
|
{
|
|
if (s == (short)CW2_USEDEFAULT) {
|
|
return (int)(DWORD)(WORD)CW2_USEDEFAULT;
|
|
} else {
|
|
return (int)s;
|
|
}
|
|
}
|
|
|
|
|
|
PBYTE SkipSz(UTCHAR* lpsz)
|
|
{
|
|
if (*lpsz == 0xFF)
|
|
return (PBYTE)lpsz + 4;
|
|
|
|
while (*lpsz++ != 0);
|
|
|
|
return (PBYTE)lpsz;
|
|
}
|
|
|
|
|
|
PBYTE WordSkipSz(UTCHAR* lpsz)
|
|
{
|
|
PBYTE pb = SkipSz(lpsz);
|
|
return NextWordBoundary(pb);
|
|
}
|
|
|
|
|
|
PBYTE DWordSkipSz(UTCHAR* lpsz)
|
|
{
|
|
PBYTE pb = SkipSz(lpsz);
|
|
return NextDWordBoundary(pb);
|
|
}
|
|
|
|
|
|
/*
|
|
* If this is a low res device, we need to check if the
|
|
* font we're creating is smaller than the system font.
|
|
*/
|
|
__inline BOOLEAN IsFontNotGood(LPWSTR szTempBuffer, LPCWSTR lpStrSubst, TEXTMETRIC* ptm)
|
|
{
|
|
// For FarEast version, we will allow the font smaller than system font.
|
|
return _wcsicmp(szTempBuffer, lpStrSubst) ||
|
|
(!IS_ANY_DBCS_CHARSET(ptm->tmCharSet) &&
|
|
(SYSMET(CXICON) < 32 || SYSMET(CYICON) < 32) &&
|
|
ptm->tmHeight < gpsi->cySysFontChar);
|
|
}
|
|
|
|
|
|
// GetCharsetEnumProc()
|
|
// This gets the best asian font for a dialog box.
|
|
// 1996-Sep-11 hideyukn Port from Win95-FE
|
|
int CALLBACK GetCharsetEnumProc(
|
|
LPLOGFONT lpLogFont,
|
|
LPTEXTMETRIC lptm,
|
|
DWORD nType,
|
|
LPARAM lpData)
|
|
{
|
|
UNREFERENCED_PARAMETER(lptm);
|
|
UNREFERENCED_PARAMETER(nType);
|
|
|
|
// Use other than FIXED pitch sysfont when the face name isn't specified.
|
|
|
|
if ((lpLogFont->lfPitchAndFamily & 3) == FIXED_PITCH) {
|
|
if (!lstrcmpi(lpLogFont->lfFaceName, L"System") || !lstrcmpi(lpLogFont->lfFaceName, L"@System"))
|
|
return TRUE; // try to get another system font metric
|
|
}
|
|
|
|
((LPLOGFONT)lpData)->lfCharSet = lpLogFont->lfCharSet;
|
|
((LPLOGFONT)lpData)->lfPitchAndFamily = lpLogFont->lfPitchAndFamily;
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* Get a character set based on System's ANSI CODEPAGE
|
|
*/
|
|
UINT GetACPCharSet()
|
|
{
|
|
static UINT charset = (UINT)~0;
|
|
CHARSETINFO csInfo;
|
|
|
|
if (charset != (UINT)~0) {
|
|
return charset;
|
|
}
|
|
|
|
// Sundown: In the TCI_SRCCODEPAGE case, the GetACP() return value is zero-extended.
|
|
if (!TranslateCharsetInfo((DWORD*)UIntToPtr(GetACP()), &csInfo, TCI_SRCCODEPAGE)) {
|
|
return DEFAULT_CHARSET;
|
|
}
|
|
charset = csInfo.ciCharset;
|
|
UserAssert(charset != (UINT)~0);
|
|
return csInfo.ciCharset;
|
|
}
|
|
|
|
#if 0
|
|
BYTE GetCharsetFromResourceLang(LCID lcid)
|
|
{
|
|
CHARSETINFO csInfo;
|
|
|
|
if (!TranslateCharsetInfo((DWORD*)lcid, &csInfo, TCI_SRCLOCALE))
|
|
return DEFAULT_CHARSET;
|
|
return csInfo.ciCharset;
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* CreateDlgFont()
|
|
* Create the dialog font described at the given location in a resource
|
|
*/
|
|
#define GET_DESKTOP_CHARSET() (GetTextCharset(hdcDlg))
|
|
|
|
// Reserved Dlg resource version number
|
|
#define DLGRSC_VER_NT5COMPAT_RESERVE 10
|
|
|
|
BOOL FixupDlgLogFont(
|
|
HDC hdcDlg,
|
|
LPLOGFONT lpLogFont,
|
|
LPDLGTEMPLATE2 lpdt,
|
|
BOOLEAN fUseShellFont2,
|
|
BOOLEAN* pfWillTryDefaultCharset)
|
|
{
|
|
switch (lpdt->wDlgVer) {
|
|
case 0:
|
|
// DIALOG statement, which only has a facename.
|
|
// The new applications are not supposed to use DIALOG statement,
|
|
// they should use DIALOGEX instead.
|
|
lpLogFont->lfWeight = FW_BOLD;
|
|
if (!fUseShellFont2) {
|
|
lpLogFont->lfCharSet = (BYTE)GET_DESKTOP_CHARSET();
|
|
*pfWillTryDefaultCharset = TRUE;
|
|
}
|
|
break;
|
|
case 1:
|
|
// Win4 compatible DLG template
|
|
if (!fUseShellFont2) {
|
|
if (IS_DBCS_ENABLED()) {
|
|
if (lpLogFont->lfCharSet == ANSI_CHARSET) {
|
|
// When resource compiler generates dialog resource data
|
|
// from DIALOGEX template, it can specify 'charset'. but
|
|
// optional, if it is not specified, it will be filled
|
|
// with 0 (ANSI charset). But, on localized version,
|
|
// User might guess the default will be localized-charset
|
|
|
|
// [Dialog Resource File]
|
|
|
|
// DIALOGEX ...
|
|
// ...
|
|
// FONT pointsize, typeface, [weight], [italic], [charset]
|
|
|
|
// #100182
|
|
// Bogus hack:
|
|
// Some FE fonts started to have ANSI_CHARSET so the first attept would succeed.
|
|
// We should enumerate the charset from the beginning.
|
|
lpLogFont->lfCharSet = DEFAULT_CHARSET;
|
|
RIPMSG0(RIP_VERBOSE, "No proper CharSet information in DIALOGEX");
|
|
}
|
|
}
|
|
/*
|
|
* Note: Dialog resource version 2.0 or later has not been supported on
|
|
* Windows 95 and 98. As of Apr/98 we decided not to deploy this new feature
|
|
* in the standard tools. Still, NT 5 supports this new feature, preparing
|
|
* the future transition to do the right thing.
|
|
*/
|
|
else if (lpLogFont->lfCharSet == ANSI_CHARSET) {
|
|
// If the first attempt fail, we'll enumerate the charset for the given facename
|
|
*pfWillTryDefaultCharset = TRUE;
|
|
} else if (lpLogFont->lfCharSet == DEFAULT_CHARSET) {
|
|
lpLogFont->lfCharSet = (BYTE)GET_DESKTOP_CHARSET();
|
|
}
|
|
}
|
|
break;
|
|
default:
|
|
if (lpdt->wDlgVer <= DLGRSC_VER_NT5COMPAT_RESERVE) {
|
|
// we do nothing for the new resource compiler (>= 2.0),
|
|
// since this version of dialogs are guarunteed to have
|
|
// the proper character set for the dialog font.
|
|
} else {
|
|
RIPMSG1(RIP_WARNING, "Version %d resource is not supported.", lpdt->wDlgVer);
|
|
return FALSE;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
VOID FixupDlgFaceName(LPLOGFONT lpLogFont, BOOLEAN fUseShellFont, BOOLEAN fUseShellFont2, LPCWSTR lpStrSubst)
|
|
{
|
|
if (fUseShellFont2) {
|
|
// OK, we use "MS Shell Dlg 2" as a face name.
|
|
wcsncpycch(lpLogFont->lfFaceName, gwszShellFont2, ARRAY_SIZE(gwszShellFont2));
|
|
} else {
|
|
// Otherwise, get the face name from the dialog template.
|
|
wcsncpycch(lpLogFont->lfFaceName, lpStrSubst, sizeof(lpLogFont->lfFaceName) / sizeof(WCHAR));
|
|
}
|
|
|
|
// "MS Shell Dlg" and "MS Shell Dlg2" should have native character set ---
|
|
if (fUseShellFont || fUseShellFont2) {
|
|
lpLogFont->lfCharSet = (BYTE)GetACPCharSet();
|
|
}
|
|
}
|
|
|
|
|
|
HFONT CreateDlgFont(HDC hdcDlg, LPWORD FAR* lplpstr, LPDLGTEMPLATE2 lpdt, DWORD dwExpWinVer, UINT fSCDLGFlags)
|
|
{
|
|
LOGFONT LogFont;
|
|
int fontheight, fheight;
|
|
HFONT hOldFont, hFont;
|
|
WCHAR szTempBuffer[LF_FACESIZE];
|
|
LPCWSTR lpStrSubst;
|
|
TEXTMETRIC tm;
|
|
// Font hacks
|
|
BOOLEAN fDeleteFont;
|
|
BOOLEAN fWillTryDefaultCharset = FALSE;
|
|
BOOLEAN fUseShellFont, fUseShellFont2;
|
|
|
|
UNREFERENCED_PARAMETER(dwExpWinVer);
|
|
UNREFERENCED_PARAMETER(fSCDLGFlags);
|
|
|
|
fheight = fontheight = (SHORT)(*((WORD*)*lplpstr)++);
|
|
|
|
if (fontheight == 0x7FFF) {
|
|
// a 0x7FFF height is our special code meaning use the message box font
|
|
GetObject(gpsi->hMsgFont, sizeof(LOGFONT), &LogFont);
|
|
return CreateFontIndirect(&LogFont);
|
|
}
|
|
|
|
// The dialog template contains a font description! Use it.
|
|
|
|
// Fill the LogFont with default values
|
|
RtlZeroMemory(&LogFont, sizeof(LOGFONT));
|
|
|
|
fontheight = -MultDiv(fontheight, gpsi->dmLogPixels, 72);
|
|
LogFont.lfHeight = fontheight;
|
|
|
|
if (lpdt->wDlgVer) {
|
|
// If it's DIALOGEX, additional info should be read from the template.
|
|
LogFont.lfWeight = *((WORD FAR*) * lplpstr)++;
|
|
LogFont.lfItalic = *((BYTE FAR*) * lplpstr)++;
|
|
LogFont.lfCharSet = *((BYTE FAR*) * lplpstr)++;
|
|
}
|
|
|
|
// Per shell team request, the dialog who has DS_SETFONT
|
|
// *and* DS_FIXEDSYS (=> DS_SHELLFONT2) should have a font
|
|
// "MS Shell Dlg 2".
|
|
lpStrSubst = *lplpstr;
|
|
|
|
// Set the pointer to the next item.
|
|
*lplpstr = (WORD*)DWordSkipSz(*lplpstr);
|
|
fUseShellFont = _wcsicmp(lpStrSubst, gwszShellFont) == 0;
|
|
|
|
// Later shell team request again, to use "Dlg 2" font only
|
|
// when facename in the dialog template is "MS Shell Dlg".
|
|
|
|
fUseShellFont2 = fUseShellFont &&
|
|
(lpdt->style & DS_SHELLFONT) == DS_SHELLFONT && Is400Compat(dwExpWinVer) && lpdt->wDlgVer != 0;
|
|
|
|
if (fUseShellFont2) {
|
|
TAGMSG0(DBGTAG_IMM, "CreateDlgFont: fUseShellFont2=TRUE");
|
|
}
|
|
|
|
// Prepare the font character set.
|
|
if (!FixupDlgLogFont(hdcDlg, &LogFont, lpdt, fUseShellFont2, &fWillTryDefaultCharset)) {
|
|
return NULL;
|
|
}
|
|
|
|
// Prepare the font facename.
|
|
FixupDlgFaceName(&LogFont, fUseShellFont, fUseShellFont2, lpStrSubst);
|
|
|
|
if (lpdt->wDlgVer < 2 && lpdt->style & DS_3DLOOK)
|
|
LogFont.lfWeight = FW_NORMAL;
|
|
|
|
TryDefaultCharset:
|
|
if (LogFont.lfCharSet == DEFAULT_CHARSET) {
|
|
// Get character set for given facename.
|
|
EnumFonts(hdcDlg, LogFont.lfFaceName, (FONTENUMPROC)GetCharsetEnumProc, (LPARAM)(&LogFont));
|
|
|
|
// We already tried default charset.
|
|
fWillTryDefaultCharset = FALSE;
|
|
}
|
|
|
|
// [Windows 3.1 FarEast version did this...]
|
|
|
|
// Use FW_NORMAL as default for DIALOG template. For DIALOGEX
|
|
// template, we need to respect the value in the template.
|
|
|
|
if ((!(lpdt->wDlgVer)) && // not DIALOGEX template ?
|
|
(IS_ANY_DBCS_CHARSET(LogFont.lfCharSet)) && // any FarEast font ?
|
|
(LogFont.lfWeight != FW_NORMAL)) { // already FW_NORMAL ?
|
|
// Set weight to FW_NORMAL.
|
|
LogFont.lfWeight = FW_NORMAL;
|
|
}
|
|
|
|
if (!(hFont = CreateFontIndirect((LPLOGFONT)&LogFont)))
|
|
return(NULL);
|
|
|
|
fDeleteFont = FALSE;
|
|
|
|
if ((hOldFont = SelectFont(hdcDlg, hFont)) == NULL) {
|
|
fDeleteFont = TRUE;
|
|
} else {
|
|
// If this dialog has DS_SHELLFONT style, or the font is
|
|
// "MS Shell Dlg", we don't judge the font integrity,
|
|
// for they have been given the ACP based character set.
|
|
if (!fUseShellFont) {
|
|
if (!GetTextMetrics(hdcDlg, &tm)) {
|
|
RIPMSG0(RIP_WARNING, "CreateDlgFont: GetTextMetrics failed");
|
|
fDeleteFont = TRUE;
|
|
} else {
|
|
GetTextFaceAliasW(hdcDlg, sizeof(szTempBuffer) / sizeof(WCHAR), szTempBuffer);
|
|
|
|
// If this is a low res device, we need to check if the
|
|
// font we're creating is smaller than the system font.
|
|
// If so, just use the system font.
|
|
if (IsFontNotGood(szTempBuffer, lpStrSubst, &tm)) {
|
|
// Couldn't find a font with the height or facename
|
|
// the app wanted so use the system font instead. Note
|
|
// that we need to make sure the app knows it is
|
|
// getting the system font via the WM_SETFONT message
|
|
// so we still need to act as if a new font is being
|
|
// sent to the dialog box.
|
|
fDeleteFont = TRUE;
|
|
}
|
|
}
|
|
}
|
|
|
|
UserAssert(hOldFont != NULL);
|
|
SelectFont(hdcDlg, hOldFont);
|
|
}
|
|
|
|
if (fDeleteFont) {
|
|
DeleteFont(hFont);
|
|
|
|
// Font is deleted, Prepare for reTry...
|
|
hFont = NULL;
|
|
}
|
|
|
|
// Font hack:
|
|
// 1. We fail to create font.
|
|
// 2. We did *NOT* try to enumerate charset, yet.
|
|
// 3. We want to try to enumerate charset
|
|
|
|
// if all of answer is 'Yes', we will try...
|
|
if (hFont == NULL && fWillTryDefaultCharset) {
|
|
// Try DEFAULT_CHARSET.
|
|
LogFont.lfCharSet = DEFAULT_CHARSET;
|
|
goto TryDefaultCharset;
|
|
}
|
|
|
|
return hFont;
|
|
}
|
|
|
|
#undef GET_DESKTOP_CHARSET
|
|
|
|
#define CD_VISIBLE 0x01
|
|
#define CD_GLOBALEDIT 0x02
|
|
#define CD_USERFONT 0x04
|
|
#define CD_SETFOREGROUND 0x08
|
|
#define CD_USEDEFAULTX 0x10
|
|
#define CD_USEDEFAULTCX 0x20
|
|
|
|
|
|
/*
|
|
* GetDialogMonitor
|
|
* Gets the monitor a dialog should be created on.
|
|
* Params:
|
|
* hwndOwner - the owner of the dialog. May be NULL.
|
|
* History:
|
|
* 10-Oct-1996 adams Created.
|
|
*/
|
|
PMONITOR GetDialogMonitor(HWND hwndOwner, DWORD dwFlags)
|
|
{
|
|
PMONITOR pMonitor;
|
|
PWND pwnd;
|
|
HWND hwndForeground;
|
|
DWORD pid;
|
|
|
|
UserAssert(dwFlags == MONITOR_DEFAULTTONULL || dwFlags == MONITOR_DEFAULTTOPRIMARY);
|
|
|
|
pMonitor = NULL;
|
|
if (hwndOwner) {
|
|
pwnd = ValidateHwnd(hwndOwner);
|
|
if (pwnd && GETFNID(pwnd) != FNID_DESKTOP) {
|
|
pMonitor = _MonitorFromWindow(pwnd, MONITOR_DEFAULTTOPRIMARY);
|
|
}
|
|
} else {
|
|
/*
|
|
* HACK! They passed in no owner and are creating a top level
|
|
* dialog window. Does this process own the foreground window?
|
|
* If so, pin to that window's monitor. That way 16-bit apps
|
|
* will work mostly as expected, and old multithreaded dudes just
|
|
* might too. Especially the shell, for whom many system UI pieces
|
|
* pop up random dialogs inside of API calls.
|
|
*/
|
|
|
|
hwndForeground = NtUserGetForegroundWindow();
|
|
if (hwndForeground) {
|
|
GetWindowThreadProcessId(hwndForeground, &pid);
|
|
if (pid == HandleToUlong(NtCurrentTeb()->ClientId.UniqueProcess)) {
|
|
pwnd = ValidateHwnd(hwndForeground);
|
|
if (pwnd) {
|
|
pMonitor = _MonitorFromWindow(pwnd, MONITOR_DEFAULTTOPRIMARY);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!pMonitor && dwFlags == MONITOR_DEFAULTTOPRIMARY) {
|
|
pMonitor = GetPrimaryMonitor();
|
|
}
|
|
|
|
return pMonitor;
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
* InternalCreateDialog
|
|
|
|
* Creates a dialog from a template. Uses passed in menu if there is one,
|
|
* destroys menu if creation failed. Server portion of
|
|
* CreateDialogIndirectParam.
|
|
|
|
* WARNING: This function cannot create any windows before creating the dialog
|
|
* window. Otherwise, MFC apps will break because their hook assumes
|
|
* the dialog is the first window to be created.
|
|
|
|
* History:
|
|
* 04-10-91 ScottLu
|
|
* 04-17-91 Mikehar Win31 Merge
|
|
*/
|
|
|
|
HWND InternalCreateDialog(
|
|
HANDLE hmod,
|
|
LPDLGTEMPLATE lpdt,
|
|
DWORD cb,
|
|
HWND hwndOwner,
|
|
DLGPROC lpfnDialog,
|
|
LPARAM lParam,
|
|
UINT fSCDLGFlags)
|
|
{
|
|
HWND hwnd;
|
|
HWND hwnd2;
|
|
PWND pwnd;
|
|
HWND hwndNewFocus;
|
|
HWND hwndEditFirst = NULL;
|
|
RECT rc;
|
|
WORD w;
|
|
UTCHAR* lpszMenu,
|
|
* lpszClass,
|
|
* lpszText,
|
|
* lpCreateParams,
|
|
* lpStr;
|
|
int cxChar,
|
|
cyChar;
|
|
BOOL f40Compat;
|
|
HFONT hNewFont = NULL;
|
|
HFONT hOldFont;
|
|
LPDLGITEMTEMPLATE lpdit;
|
|
HMENU hMenu;
|
|
BOOL fSuccess;
|
|
BOOL fWowWindow;
|
|
HANDLE hmodCreate;
|
|
LPBYTE lpCreateParamsData;
|
|
DLGTEMPLATE2 dt;
|
|
DLGITEMTEMPLATE2 dit;
|
|
DWORD dwExpWinVer;
|
|
DWORD dsStyleOld;
|
|
DWORD bFlags = 0;
|
|
HDC hdcDlg;
|
|
LARGE_STRING strClassName;
|
|
PLARGE_STRING pstrClassName;
|
|
LARGE_STRING strWindowName;
|
|
PMONITOR pMonitor;
|
|
|
|
UNREFERENCED_PARAMETER(cb);
|
|
|
|
ConnectIfNecessary();
|
|
|
|
UserAssert(!(fSCDLGFlags & ~(SCDLG_CLIENT | SCDLG_ANSI | SCDLG_NOREVALIDATE | SCDLG_16BIT))); // These are the only valid flags
|
|
|
|
/*
|
|
* Is this a Win4 extended dialog?
|
|
*/
|
|
if (((LPDLGTEMPLATE2)lpdt)->wSignature == 0xffff) {
|
|
|
|
UserAssert(((LPDLGTEMPLATE2)lpdt)->wDlgVer <= DLGRSC_VER_NT5COMPAT_RESERVE);
|
|
RtlCopyMemory(&dt, lpdt, sizeof dt);
|
|
} else {
|
|
dt.wDlgVer = 0;
|
|
dt.wSignature = 0;
|
|
dt.dwHelpID = 0;
|
|
dt.dwExStyle = lpdt->dwExtendedStyle;
|
|
dt.style = lpdt->style;
|
|
dt.cDlgItems = lpdt->cdit;
|
|
dt.x = lpdt->x;
|
|
dt.y = lpdt->y;
|
|
dt.cx = lpdt->cx;
|
|
dt.cy = lpdt->cy;
|
|
}
|
|
|
|
/*
|
|
* If this is called from wow code, then the loword of hmod != 0.
|
|
* In this case, allow any DS_ style bits that were passed in win3.1
|
|
* to be legal in win32. Case in point: 16 bit quark xpress passes the
|
|
* same bit as the win32 style DS_SETFOREGROUND. Also, VC++ sample
|
|
* "scribble" does the same thing.
|
|
|
|
* For win32 apps test the DS_SETFOREGROUND bit; wow apps are not set
|
|
* foreground (this is the new NT semantics)
|
|
* We have to let no "valid" bits through because apps depend on them
|
|
* bug 5232.
|
|
*/
|
|
dsStyleOld = LOWORD(dt.style);
|
|
|
|
/*
|
|
* If the app is Win4 or greater, require correct dialog style bits.
|
|
* Prevents conflicts with new bits introduced in Chicago
|
|
*/
|
|
dwExpWinVer = GETEXPWINVER(hmod);
|
|
|
|
if (f40Compat = Is400Compat(dwExpWinVer)) {
|
|
dt.style &= (DS_VALID40 | 0xffff0000);
|
|
|
|
// For old applications:
|
|
// If DS_COMMONDIALOG isn't set, don't touch DS_3DLOOK style
|
|
// bit. If it's there, it stays there. If not, not. That way old
|
|
// apps which pass in their own templates, not commdlg's, don't get
|
|
// forced 3D.
|
|
// If DS_COMMONDIALOG is there, remove DS_3DLOOK.
|
|
|
|
// For new applications:
|
|
// Force 3D always.
|
|
|
|
if (GETAPPVER() < VER40) {
|
|
if (dt.style & DS_COMMONDIALOG) {
|
|
dt.style &= ~DS_3DLOOK;
|
|
dsStyleOld &= ~DS_3DLOOK;
|
|
}
|
|
} else {
|
|
dt.style |= DS_3DLOOK;
|
|
dsStyleOld |= DS_3DLOOK;
|
|
}
|
|
} else {
|
|
#if DBG
|
|
if (dt.style != (dt.style & (DS_VALID31 | DS_3DLOOK | 0xffff0000))) {
|
|
RIPMSG1(RIP_WARNING, "CreateDialog: stripping invalid bits %lX", dt.style);
|
|
}
|
|
#endif
|
|
|
|
/*
|
|
* Don't strip off bits for old apps, they depend on this. Especially 16 bit MFC apps!
|
|
* dt.dwStyle &= (DS_VALID31 | 0xffff0000);
|
|
*/
|
|
}
|
|
|
|
if (LOWORD((ULONG_PTR)hmod) == 0) {
|
|
if (dt.style & DS_SETFOREGROUND)
|
|
bFlags |= CD_SETFOREGROUND;
|
|
}
|
|
|
|
if (dsStyleOld != LOWORD(dt.style)) {
|
|
RIPMSG1(f40Compat ? RIP_ERROR : RIP_WARNING,
|
|
"Bad dialog style bits (%x) - please remove.",
|
|
LOWORD(dt.style));
|
|
// Fail new apps that pass in bogus bits!
|
|
|
|
if (f40Compat) {
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
if (dt.style & DS_MODALFRAME) {
|
|
dt.dwExStyle |= WS_EX_DLGMODALFRAME | WS_EX_WINDOWEDGE;
|
|
}
|
|
|
|
if ((dt.style & DS_CONTEXTHELP) && f40Compat) {
|
|
dt.dwExStyle |= WS_EX_CONTEXTHELP;
|
|
}
|
|
|
|
if (dt.style & DS_CONTROL) {
|
|
// Captions and system menus aren't allowed on "control" dialogs.
|
|
// And strip DS_SYSMODAL.
|
|
dt.style &= ~(WS_CAPTION | WS_SYSMENU | DS_SYSMODAL);
|
|
} else if (dt.style & WS_DLGFRAME) {
|
|
// Add on window edge same way that CreateWindowEx() will
|
|
dt.dwExStyle |= WS_EX_WINDOWEDGE;
|
|
}
|
|
|
|
if (dt.style & DS_SYSMODAL) {
|
|
dt.dwExStyle |= WS_EX_TOPMOST;
|
|
}
|
|
|
|
if (!(dt.style & WS_CHILD) || (dt.style & DS_CONTROL)) {
|
|
// only a control parent if it's not a child dialog or if it's
|
|
// explicitly marked as a recursive dialog
|
|
dt.dwExStyle |= WS_EX_CONTROLPARENT;
|
|
}
|
|
|
|
if (dt.x == (short)CW2_USEDEFAULT) {
|
|
bFlags |= CD_USEDEFAULTX;
|
|
dt.x = 0;
|
|
}
|
|
|
|
if (dt.cx == (short)CW2_USEDEFAULT) {
|
|
bFlags |= CD_USEDEFAULTCX;
|
|
dt.cx = 0;
|
|
} else if (dt.cx < 0) {
|
|
dt.cx = 0;
|
|
}
|
|
|
|
if (dt.cy < 0) {
|
|
dt.cy = 0;
|
|
}
|
|
|
|
// If there's a menu name string, load it.
|
|
lpszMenu = (LPWSTR)(((PBYTE)(lpdt)) + (dt.wDlgVer ? sizeof(DLGTEMPLATE2) : sizeof(DLGTEMPLATE)));
|
|
|
|
/*
|
|
* If the menu id is expressed as an ordinal and not a string,
|
|
* skip all 4 bytes to get to the class string.
|
|
*/
|
|
w = *(LPWORD)lpszMenu;
|
|
|
|
/*
|
|
* If there's a menu name string, load it.
|
|
*/
|
|
if (w != 0) {
|
|
if ((hMenu = LoadMenu(hmod, (w == 0xFFFF) ?
|
|
MAKEINTRESOURCE(*(WORD*)((PBYTE)lpszMenu + 2)) : lpszMenu)) == NULL) {
|
|
RIPMSG0(RIP_WARNING, "ServerCreateDialog() failed: couldn't load menu");
|
|
goto DeleteFontAndMenuAndFail;
|
|
}
|
|
} else {
|
|
hMenu = NULL;
|
|
}
|
|
|
|
if (w == 0xFFFF) {
|
|
lpszClass = (LPWSTR)((LPBYTE)lpszMenu + 4);
|
|
} else {
|
|
lpszClass = (UTCHAR*)WordSkipSz(lpszMenu);
|
|
}
|
|
|
|
lpszText = (UTCHAR*)WordSkipSz(lpszClass);
|
|
lpStr = (UTCHAR*)WordSkipSz(lpszText);
|
|
|
|
hdcDlg = CreateCompatibleDC(NULL);
|
|
if (hdcDlg == NULL)
|
|
goto DeleteFontAndMenuAndFail;
|
|
|
|
if (dt.style & DS_SETFONT) {
|
|
hNewFont = CreateDlgFont(hdcDlg, &lpStr, &dt, dwExpWinVer, fSCDLGFlags);
|
|
bFlags |= CD_USERFONT;
|
|
lpdit = (LPDLGITEMTEMPLATE)NextDWordBoundary(lpStr);
|
|
} else if (Is400Compat(dwExpWinVer) && (dt.style & DS_FIXEDSYS)) {
|
|
// B#2078 -- WISH for fixed width system font in dialog. We need
|
|
// to tell the dialog that it's using a font different from the
|
|
// standard system font, so set CD_USERFONT bit.
|
|
|
|
// We need the 400 compat. check for CorelDraw, since they use
|
|
// this style bit for their own purposes.
|
|
|
|
hNewFont = GetStockObject(SYSTEM_FIXED_FONT);
|
|
bFlags |= CD_USERFONT;
|
|
lpdit = (LPDLGITEMTEMPLATE)NextDWordBoundary(lpStr);
|
|
} else {
|
|
lpdit = (LPDLGITEMTEMPLATE)NextDWordBoundary(lpStr);
|
|
}
|
|
|
|
/*
|
|
* If the application requested a particular font and for some
|
|
* reason we couldn't find it, we just use the system font. BUT we
|
|
* need to make sure we tell him he gets the system font. Dialogs
|
|
* which never request a particular font get the system font and we
|
|
* don't bother telling them this (via the WM_SETFONT message).
|
|
*/
|
|
|
|
// Is it anything other than the default system font? If we can't get
|
|
// enough memory to select in the new font specified, just use the system font.
|
|
if (hNewFont && (hOldFont = SelectFont(hdcDlg, hNewFont))) {
|
|
// Get the ave character width and height to be used
|
|
cxChar = GdiGetCharDimensions(hdcDlg, NULL, &cyChar);
|
|
|
|
SelectFont(hdcDlg, hOldFont);
|
|
if (cxChar == 0) {
|
|
RIPMSG0(RIP_WARNING, "InternalCreateDialog: GdiGetCharDimensions failed");
|
|
goto UseSysFontMetrics;
|
|
}
|
|
} else {
|
|
if (hNewFont || (bFlags & CD_USERFONT))
|
|
hNewFont = ghFontSys;
|
|
|
|
UseSysFontMetrics:
|
|
cxChar = gpsi->cxSysFontChar;
|
|
cyChar = gpsi->cySysFontChar;
|
|
}
|
|
DeleteDC(hdcDlg);
|
|
|
|
if (dt.style & WS_VISIBLE) {
|
|
bFlags |= CD_VISIBLE;
|
|
dt.style &= ~WS_VISIBLE;
|
|
}
|
|
|
|
if (!(dt.style & DS_LOCALEDIT)) {
|
|
bFlags |= CD_GLOBALEDIT;
|
|
}
|
|
|
|
/* Figure out dimensions of real window
|
|
*/
|
|
rc.left = rc.top = 0;
|
|
rc.right = XPixFromXDU(dt.cx, cxChar);
|
|
rc.bottom = YPixFromYDU(dt.cy, cyChar);
|
|
|
|
_AdjustWindowRectEx(&rc, dt.style, w, dt.dwExStyle);
|
|
|
|
dt.cx = (SHORT)(rc.right - rc.left);
|
|
dt.cy = (SHORT)(rc.bottom - rc.top);
|
|
|
|
if ((dt.style & DS_CENTERMOUSE) && SYSMET(MOUSEPRESENT) && f40Compat) {
|
|
pMonitor = _MonitorFromPoint(gpsi->ptCursor, MONITOR_DEFAULTTONULL);
|
|
UserAssert(pMonitor);
|
|
*((LPPOINT)&rc.left) = gpsi->ptCursor;
|
|
rc.left -= (dt.cx / 2);
|
|
rc.top -= (dt.cy / 2);
|
|
} else {
|
|
BOOL fNoDialogMonitor;
|
|
|
|
pMonitor = GetDialogMonitor(hwndOwner, MONITOR_DEFAULTTONULL);
|
|
fNoDialogMonitor = pMonitor ? FALSE : TRUE;
|
|
if (!pMonitor) {
|
|
pMonitor = GetPrimaryMonitor();
|
|
}
|
|
|
|
if ((dt.style & (DS_CENTER | DS_CENTERMOUSE)) && f40Compat) {
|
|
/*
|
|
* Center to the work area of the owner monitor.
|
|
*/
|
|
rc.left = (pMonitor->rcWork.left + pMonitor->rcWork.right - dt.cx) / 2;
|
|
rc.top = (pMonitor->rcWork.top + pMonitor->rcWork.bottom - dt.cy) / 2;
|
|
} else {
|
|
rc.left = XPixFromXDU(dt.x, cxChar);
|
|
rc.top = YPixFromYDU(dt.y, cyChar);
|
|
|
|
if (!(dt.style & DS_ABSALIGN) && hwndOwner) {
|
|
/*
|
|
* Offset relative coordinates to the owner window. If it is
|
|
* a child window, there is nothing to do.
|
|
*/
|
|
if ((HIWORD(dt.style) & MaskWF(WFTYPEMASK)) != MaskWF(WFCHILD)) {
|
|
//This is will considre rc.left form the right hand side of the owner window if it a mirrored one.
|
|
ClientToScreen(hwndOwner, (LPPOINT)&rc.left);
|
|
|
|
#ifdef USE_MIRRORING
|
|
//It is not chiled then do Visual ClientToScreen
|
|
//i.e. rc.left it is form the left hand side of the owner window
|
|
if (MIRRORED_HWND(hwndOwner)) {
|
|
rc.left -= dt.cx;
|
|
}
|
|
#endif
|
|
|
|
}
|
|
} else {
|
|
/*
|
|
* Position the dialog in screen coordinates. If the dialog's
|
|
* owner is on a different monitor than specified in the
|
|
* template, move the dialog to the owner window. If the owner
|
|
* doesn't exist, then use the monitor from the dialog's template.
|
|
*/
|
|
|
|
PMONITOR pMonitorTemplate;
|
|
RECT rcTemplate;
|
|
|
|
rcTemplate.left = rc.left;
|
|
rcTemplate.top = rc.top;
|
|
rcTemplate.right = rc.left + dt.cx;
|
|
rcTemplate.bottom = rc.top + dt.cy;
|
|
|
|
pMonitorTemplate = _MonitorFromRect(&rcTemplate, MONITOR_DEFAULTTOPRIMARY);
|
|
if (fNoDialogMonitor) {
|
|
pMonitor = pMonitorTemplate;
|
|
} else if (pMonitorTemplate != pMonitor) {
|
|
rc.left += pMonitor->rcMonitor.left - pMonitorTemplate->rcMonitor.left;
|
|
rc.top += pMonitor->rcMonitor.top - pMonitorTemplate->rcMonitor.top;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
rc.right = rc.left + dt.cx;
|
|
rc.bottom = rc.top + dt.cy;
|
|
|
|
// If the right or bottom coordinate has overflowed, then pin it back to
|
|
// a valid rectangle. Likely to happen if a minimized window is the owner of the dialog.
|
|
if (rc.left > rc.right || rc.top > rc.bottom) {
|
|
OffsetRect(&rc, -dt.cx, -dt.cy);
|
|
}
|
|
|
|
// Need to do this for ALL dialogs, not just top-level, since we used to in 3.1.
|
|
|
|
// Clip top level dialogs within working area
|
|
// Start child dialogs at least at (0, 0)
|
|
RepositionRect(pMonitor, &rc, dt.style, dt.dwExStyle);
|
|
|
|
dt.x = (SHORT)((bFlags & CD_USEDEFAULTX) ? CW2_USEDEFAULT : rc.left);
|
|
dt.y = (SHORT)(rc.top);
|
|
dt.cx = (SHORT)((bFlags & CD_USEDEFAULTCX) ? CW2_USEDEFAULT : rc.right - rc.left);
|
|
dt.cy = (SHORT)(rc.bottom - rc.top);
|
|
|
|
if (*lpszClass != 0) {
|
|
if (IS_PTR(lpszClass)) {
|
|
RtlInitLargeUnicodeString((PLARGE_UNICODE_STRING)&strClassName, lpszClass, (UINT)-1);
|
|
pstrClassName = &strClassName;
|
|
} else {
|
|
pstrClassName = (PLARGE_STRING)lpszClass;
|
|
}
|
|
} else {
|
|
pstrClassName = (PLARGE_STRING)DIALOGCLASS;
|
|
}
|
|
|
|
RtlInitLargeUnicodeString((PLARGE_UNICODE_STRING)&strWindowName, lpszText, (UINT)-1);
|
|
|
|
UserAssert((dt.dwExStyle & WS_EX_MDICHILD) == 0);
|
|
hwnd = NtUserCreateWindowEx(
|
|
dt.dwExStyle | ((fSCDLGFlags & SCDLG_ANSI) ? WS_EX_ANSICREATOR : 0),
|
|
pstrClassName,
|
|
&strWindowName,
|
|
dt.style,
|
|
DefShortToInt(dt.x),
|
|
dt.y,
|
|
DefShortToInt(dt.cx),
|
|
dt.cy,
|
|
hwndOwner,
|
|
hMenu,
|
|
hmod,
|
|
(LPVOID)NULL,
|
|
dwExpWinVer);
|
|
if (hwnd == NULL) {
|
|
RIPMSG0(RIP_WARNING, "CreateDialog() failed: couldn't create window");
|
|
DeleteFontAndMenuAndFail:
|
|
if (hMenu != NULL)
|
|
NtUserDestroyMenu(hMenu);
|
|
/*
|
|
* Only delete the font if we didn't grab it
|
|
* from the dialog font cache.
|
|
*/
|
|
if ((hNewFont != NULL)) {
|
|
DeleteObject(hNewFont);
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
pwnd = ValidateHwnd(hwnd);
|
|
|
|
// tell WOW the hDlg of the Window just created BEFORE they get any messages
|
|
// at WOW32!w32win16wndprocex
|
|
if (fSCDLGFlags & SCDLG_16BIT) {
|
|
TellWOWThehDlg(hwnd);
|
|
}
|
|
|
|
/*
|
|
* Before anything happens with this window, we need to mark it as a
|
|
* dialog window!!!! So do that.
|
|
*/
|
|
if (pwnd == NULL || !ValidateDialogPwnd(pwnd))
|
|
goto DeleteFontAndMenuAndFail;
|
|
|
|
if (dt.dwHelpID) {
|
|
NtUserSetWindowContextHelpId(hwnd, dt.dwHelpID);
|
|
}
|
|
|
|
/*
|
|
* Set up the system menu on this dialog box if it has one.
|
|
*/
|
|
if (TestWF(pwnd, WFSYSMENU)) {
|
|
/*
|
|
* For a modal dialog box with a frame and caption, we want to
|
|
* delete the unselectable items from the system menu.
|
|
*/
|
|
UserAssert(HIBYTE(WFSIZEBOX) == HIBYTE(WFMINBOX));
|
|
UserAssert(HIBYTE(WFMINBOX) == HIBYTE(WFMAXBOX));
|
|
if (!TestWF(pwnd, WFSIZEBOX | WFMINBOX | WFMAXBOX)) {
|
|
NtUserCallHwndLock(hwnd, SFI_XXXSETDIALOGSYSTEMMENU);
|
|
} else {
|
|
/*
|
|
* We have to give this dialog its own copy of the system menu
|
|
* in case it modifies the menu.
|
|
*/
|
|
NtUserGetSystemMenu(hwnd, FALSE);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Set fDisabled to FALSE so EndDialog will Enable if dialog is ended
|
|
* before returning to DialogBox (or if modeless).
|
|
*/
|
|
PDLG(pwnd)->fDisabled = FALSE;
|
|
|
|
PDLG(pwnd)->cxChar = cxChar;
|
|
PDLG(pwnd)->cyChar = cyChar;
|
|
PDLG(pwnd)->lpfnDlg = lpfnDialog;
|
|
PDLG(pwnd)->fEnd = FALSE;
|
|
PDLG(pwnd)->result = IDOK;
|
|
|
|
/*
|
|
* Need to remember Unicode status.
|
|
*/
|
|
if (fSCDLGFlags & SCDLG_ANSI) {
|
|
PDLG(pwnd)->flags |= DLGF_ANSI;
|
|
}
|
|
|
|
/*
|
|
* Have to do a callback here for WOW apps. WOW needs what's in lParam
|
|
* before the dialog gets any messages.
|
|
*/
|
|
|
|
/*
|
|
* If the app is a Wow app then the Lo Word of the hInstance is the
|
|
* 16-bit hInstance. Set the lParam, which no-one should look at
|
|
* but the app, to the 16 bit value
|
|
*/
|
|
if (LOWORD((ULONG_PTR)hmod) != 0) {
|
|
fWowWindow = TRUE;
|
|
} else {
|
|
fWowWindow = FALSE;
|
|
}
|
|
|
|
/*
|
|
* If a user defined font is used, save the handle so that we can delete
|
|
* it when the dialog is destroyed.
|
|
*/
|
|
if (bFlags & CD_USERFONT) {
|
|
PDLG(pwnd)->hUserFont = hNewFont;
|
|
|
|
if (lpfnDialog != NULL) {
|
|
/*
|
|
* Tell the dialog that it will be using this font...
|
|
*/
|
|
SendMessageWorker(pwnd, WM_SETFONT, (WPARAM)hNewFont, 0L, FALSE);
|
|
}
|
|
}
|
|
|
|
if (!dt.wDlgVer) {
|
|
dit.dwHelpID = 0;
|
|
}
|
|
|
|
/*
|
|
* Loop through the dialog controls, doing a CreateWindowEx() for each of
|
|
* them.
|
|
*/
|
|
while (dt.cDlgItems-- != 0) {
|
|
DWORD dwExpWinVer2;
|
|
|
|
if (dt.wDlgVer) {
|
|
RtlCopyMemory(&dit, lpdit, sizeof dit);
|
|
} else {
|
|
dit.dwHelpID = 0;
|
|
dit.dwExStyle = lpdit->dwExtendedStyle;
|
|
dit.style = lpdit->style;
|
|
dit.x = lpdit->x;
|
|
dit.y = lpdit->y;
|
|
dit.cx = lpdit->cx;
|
|
dit.cy = lpdit->cy;
|
|
dit.dwID = lpdit->id;
|
|
}
|
|
|
|
dit.x = XPixFromXDU(dit.x, cxChar);
|
|
dit.y = YPixFromYDU(dit.y, cyChar);
|
|
dit.cx = XPixFromXDU(dit.cx, cxChar);
|
|
dit.cy = YPixFromYDU(dit.cy, cyChar);
|
|
|
|
lpszClass = (LPWSTR)(((PBYTE)(lpdit)) + (dt.wDlgVer ? sizeof(DLGITEMTEMPLATE2) : sizeof(DLGITEMTEMPLATE)));
|
|
|
|
/*
|
|
* If the first WORD is 0xFFFF the second word is the encoded class name index.
|
|
* Use it to look up the class name string.
|
|
*/
|
|
if (*(LPWORD)lpszClass == 0xFFFF) {
|
|
lpszText = lpszClass + 2;
|
|
lpszClass = (LPWSTR)(gpsi->atomSysClass[*(((LPWORD)lpszClass) + 1) & ~CODEBIT]);
|
|
} else {
|
|
lpszText = (UTCHAR*)SkipSz(lpszClass);
|
|
}
|
|
lpszText = (UTCHAR*)NextWordBoundary(lpszText); // UINT align lpszText
|
|
|
|
dit.dwExStyle |= WS_EX_NOPARENTNOTIFY;
|
|
|
|
|
|
// Replace flat borders with 3D ones for DS_3DLOOK dialogs
|
|
// We test the WINDOW style, not the template style now. This is so
|
|
// that 4.0 apps--who get 3D stuff automatically--can turn it off on
|
|
// create if they want.
|
|
|
|
// HACK!
|
|
// Treat DS_3DLOOK combos like they have a WS_EX_CLIENTEDGE. Why
|
|
// should we have to draw the borders of a combobox ourselves?
|
|
// We can't do the same thing for WS_BORDER though becaues of
|
|
// PC Fools--they use the presence of WS_BORDER to distinguish between lists and combos.
|
|
|
|
if (TestWF(pwnd, DF3DLOOK)) {
|
|
if ((dit.style & WS_BORDER) || (lpszClass == MAKEINTRESOURCE(gpsi->atomSysClass[ICLS_COMBOBOX]))) {
|
|
dit.style &= ~WS_BORDER;
|
|
dit.dwExStyle |= WS_EX_CLIENTEDGE;
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Get pointer to additional data. lpszText can point to an encoded
|
|
* ordinal number for some controls (e.g. static icon control) so
|
|
* we check for that here.
|
|
*/
|
|
if (*(LPWORD)lpszText == 0xFFFF) {
|
|
lpCreateParams = (LPWSTR)((PBYTE)lpszText + 4);
|
|
strWindowName.Buffer = lpszText;
|
|
strWindowName.Length = 4;
|
|
strWindowName.MaximumLength = 4;
|
|
strWindowName.bAnsi = FALSE;
|
|
} else {
|
|
lpCreateParams = (LPWSTR)((PBYTE)WordSkipSz(lpszText));
|
|
RtlInitLargeUnicodeString((PLARGE_UNICODE_STRING)&strWindowName, lpszText, (UINT)-1);
|
|
}
|
|
|
|
/*
|
|
* If control is edit control and caller wants global storage
|
|
* of edit text, allocate object in WOW and pass instance
|
|
* handle to CreateWindowEx().
|
|
*/
|
|
if (fWowWindow && (bFlags & CD_GLOBALEDIT) &&
|
|
((!IS_PTR(lpszClass) && PTR_TO_ID(lpszClass) == (ATOM)(gpsi->atomSysClass[ICLS_EDIT])) ||
|
|
(IS_PTR(lpszClass) && (wcscmp(lpszClass, szEDITCLASS) == 0)))) {
|
|
/*
|
|
* Allocate only one global object (first time we see editctl.)
|
|
*/
|
|
if (!(PDLG(pwnd)->hData)) {
|
|
PDLG(pwnd)->hData = GetEditDS();
|
|
if (!(PDLG(pwnd)->hData))
|
|
goto NoCreate;
|
|
}
|
|
|
|
hmodCreate = PDLG(pwnd)->hData;
|
|
dwExpWinVer2 = GETEXPWINVER(hmodCreate);
|
|
} else {
|
|
hmodCreate = hmod;
|
|
dwExpWinVer2 = dwExpWinVer;
|
|
}
|
|
|
|
#if DBG
|
|
if ((dit.dwExStyle & WS_EX_ANSICREATOR) != 0) {
|
|
RIPMSG1(RIP_WARNING, "Bad WS_EX_ style 0x%x for a control in the dialog", dit.dwExStyle);
|
|
}
|
|
#endif // DBG
|
|
|
|
/*
|
|
* Get pointer to additional data.
|
|
|
|
* For WOW, instead of pointing lpCreateParams at the CreateParams
|
|
* data, set lpCreateParams to whatever DWORD is stored in the 32-bit
|
|
* DLGTEMPLATE's CreateParams. WOW has already made sure that that
|
|
* 32-bit value is indeed a 16:16 pointer to the CreateParams in the 16-bit DLGTEMPLATE.
|
|
*/
|
|
|
|
if (*lpCreateParams) {
|
|
lpCreateParamsData = (LPBYTE)lpCreateParams;
|
|
if (fWowWindow || fSCDLGFlags & SCDLG_16BIT) {
|
|
lpCreateParamsData =
|
|
(LPBYTE)ULongToPtr(*(UNALIGNED DWORD*) /* Sundown WOW: zero-extension */
|
|
(lpCreateParamsData + sizeof(WORD)));
|
|
}
|
|
} else {
|
|
lpCreateParamsData = NULL;
|
|
}
|
|
|
|
/*
|
|
* If the dialog template specifies a menu ID then TestwndChild(pwnd)
|
|
* must be TRUE or CreateWindowEx will think the ID is an hMenu rather
|
|
* than an ID (in a dialog template you'll never have an hMenu).
|
|
* However for compatibility reasons we let it go if the ID = 0.
|
|
*/
|
|
if (dit.dwID) {
|
|
/*
|
|
* This makes TestwndChild(pwnd) on this window return TRUE.
|
|
*/
|
|
dit.style |= WS_CHILD;
|
|
dit.style &= ~WS_POPUP;
|
|
}
|
|
|
|
if (IS_PTR(lpszClass)) {
|
|
RtlInitLargeUnicodeString((PLARGE_UNICODE_STRING)&strClassName, lpszClass, (UINT)-1);
|
|
pstrClassName = &strClassName;
|
|
} else {
|
|
pstrClassName = (PLARGE_STRING)lpszClass;
|
|
}
|
|
UserAssert((dit.dwExStyle & WS_EX_MDICHILD) == 0);
|
|
|
|
hwnd2 = NtUserCreateWindowEx(
|
|
dit.dwExStyle | ((fSCDLGFlags & SCDLG_ANSI) ? WS_EX_ANSICREATOR : 0),
|
|
pstrClassName,
|
|
&strWindowName,
|
|
dit.style,
|
|
DefShortToInt(dit.x),
|
|
dit.y,
|
|
DefShortToInt(dit.cx),
|
|
dit.cy,
|
|
hwnd,
|
|
(HMENU)LongToHandle(dit.dwID),
|
|
hmodCreate,
|
|
lpCreateParamsData,
|
|
dwExpWinVer2);
|
|
if (hwnd2 == NULL) {
|
|
NoCreate:
|
|
/*
|
|
* Couldn't create the window -- return NULL.
|
|
*/
|
|
if (!TestWF(pwnd, DFNOFAILCREATE)) {
|
|
RIPMSG0(RIP_WARNING, "CreateDialog() failed: couldn't create control");
|
|
NtUserDestroyWindow(hwnd);
|
|
return NULL;
|
|
}
|
|
} else {
|
|
if (dit.dwHelpID) {
|
|
NtUserSetWindowContextHelpId(hwnd2, dit.dwHelpID);
|
|
}
|
|
|
|
/*
|
|
* If it is a not a default system font, set the font for all the
|
|
* child windows of the dialogbox.
|
|
*/
|
|
if (hNewFont != NULL) {
|
|
SendMessage(hwnd2, WM_SETFONT, (WPARAM)hNewFont, 0L);
|
|
}
|
|
|
|
/*
|
|
* Result gets ID of last (hopefully only) defpushbutton.
|
|
*/
|
|
if (SendMessage(hwnd2, WM_GETDLGCODE, 0, 0L) & DLGC_DEFPUSHBUTTON) {
|
|
PDLG(pwnd)->result = dit.dwID;
|
|
}
|
|
|
|
}
|
|
/*
|
|
* Point at next item template
|
|
*/
|
|
lpdit = (LPDLGITEMTEMPLATE)NextDWordBoundary((LPBYTE)(lpCreateParams + 1) + *lpCreateParams);
|
|
}
|
|
|
|
if (!TestWF(pwnd, DFCONTROL)) {
|
|
PWND pwndT = _GetNextDlgTabItem(pwnd, NULL, FALSE);
|
|
hwndEditFirst = HW(pwndT);
|
|
}
|
|
|
|
if (lpfnDialog != NULL) {
|
|
fSuccess = (BOOL)SendMessageWorker(pwnd, WM_INITDIALOG, (WPARAM)hwndEditFirst, lParam, FALSE);
|
|
|
|
// Make sure the window didn't get nuked during WM_INITDIALOG
|
|
if (!RevalidateHwnd(hwnd)) {
|
|
goto CreateDialogReturn;
|
|
}
|
|
if (fSuccess && !PDLG(pwnd)->fEnd) {
|
|
// To remove the two-default-push-buttons problem, we must make
|
|
// sure CheckDefPushButton() will remove default from other push
|
|
// buttons. This happens only if hwndEditFirst != hwndNewFocus;
|
|
// So, we make it NULL here. This breaks Designer's install
|
|
// program(which can't take a DM_GETDEFID. So, we do a version check here.
|
|
if (!TestWF(pwnd, DFCONTROL)) {
|
|
PWND pwndT;
|
|
if (!IsWindow(hwndEditFirst) || TestWF(pwnd, WFWIN40COMPAT))
|
|
hwndEditFirst = NULL;
|
|
|
|
// They could have disabled hwndEditFirst during WM_INITDIALOG.
|
|
// So, let use obtain the First Tab again.
|
|
pwndT = _GetNextDlgTabItem(pwnd, NULL, FALSE);
|
|
if (hwndNewFocus = HW(pwndT)) {
|
|
DlgSetFocus(hwndNewFocus);
|
|
}
|
|
|
|
xxxCheckDefPushButton(pwnd, hwndEditFirst, hwndNewFocus);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (!IsWindow(hwnd)) {
|
|
// Omnis7 relies on a nonzero return even though they nuked this
|
|
// dialog during processing of the WM_INITDIALOG message
|
|
// -- jeffbog -- 2/24/95 -- Win95B B#12368
|
|
if (GETAPPVER() < VER40) {
|
|
return(hwnd);
|
|
}
|
|
|
|
return(NULL);
|
|
}
|
|
|
|
/*
|
|
* UISTATE: if keyboard indicators are on and this is a topmost dialog
|
|
* set the internal bit.
|
|
*/
|
|
if (TEST_KbdCuesPUSIF) {
|
|
/*
|
|
* If property page, UISTATE bits were copied from parent when I was created
|
|
* Top level dialogs act as containers and initialize their state based on
|
|
* the type of the last input event, after sending UIS_INITIALIZE
|
|
*/
|
|
if (!TestwndChild(pwnd)) {
|
|
SendMessageWorker(pwnd, WM_CHANGEUISTATE, MAKEWPARAM(UIS_INITIALIZE, 0), 0, FALSE);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* Bring this dialog into the foreground
|
|
* if DS_SETFOREGROUND is set.
|
|
*/
|
|
if (bFlags & CD_SETFOREGROUND) {
|
|
NtUserSetForegroundWindow(hwnd);
|
|
if (!IsWindow(hwnd)) {
|
|
hwnd = NULL;
|
|
goto CreateDialogReturn;
|
|
}
|
|
}
|
|
|
|
if ((bFlags & CD_VISIBLE) && !PDLG(pwnd)->fEnd && (!TestWF(pwnd, WFVISIBLE))) {
|
|
NtUserShowWindow(hwnd, SHOW_OPENWINDOW);
|
|
UpdateWindow(hwnd);
|
|
}
|
|
|
|
CreateDialogReturn:
|
|
|
|
/*
|
|
* 17609 Gupta's SQLWin deletes the window before CreateDialog returns
|
|
* but still expects non-zero return value from CreateDialog so we will
|
|
* do like win 3.1 and not revalidate for 16 bit apps
|
|
*/
|
|
if (!(fSCDLGFlags & SCDLG_NOREVALIDATE) && !RevalidateHwnd(hwnd)) {
|
|
hwnd = NULL;
|
|
}
|
|
|
|
return hwnd;
|
|
}
|