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

1950 lines
49 KiB
C

/*** Module Header *******\
* Module Name: btnctl.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* Radio Button and Check Box Handling Routines
* History:
* ??-???-???? ?????? Ported from Win 3.0 sources
* 01-Feb-1991 mikeke Added Revalidation code
* 03-Jan-1992 ianja Neutralized (ANSI/wide-character)
*/
#include "precomp.h"
#pragma hdrstop
/* ButtonCalcRect codes */
#define CBR_CLIENTRECT 0
#define CBR_CHECKBOX 1
#define CBR_CHECKTEXT 2
#define CBR_GROUPTEXT 3
#define CBR_GROUPFRAME 4
#define CBR_PUSHBUTTON 5
CONST BYTE mpStyleCbr[] = {
CBR_PUSHBUTTON, /* BS_PUSHBUTTON */
CBR_PUSHBUTTON, /* BS_DEFPUSHBUTTON */
CBR_CHECKTEXT, /* BS_CHECKBOX */
CBR_CHECKTEXT, /* BS_AUTOCHECKBOX */
CBR_CHECKTEXT, /* BS_RADIOBUTTON */
CBR_CHECKTEXT, /* BS_3STATE */
CBR_CHECKTEXT, /* BS_AUTO3STATE */
CBR_GROUPTEXT, /* BS_GROUPBOX */
CBR_CLIENTRECT, /* BS_USERBUTTON */
CBR_CHECKTEXT, /* BS_AUTORADIOBUTTON */
CBR_CLIENTRECT, /* BS_PUSHBOX */
CBR_CLIENTRECT, /* BS_OWNERDRAW */
};
#define IMAGE_BMMAX IMAGE_CURSOR+1
static CONST BYTE rgbType[IMAGE_BMMAX] = {
BS_BITMAP, // IMAGE_BITMAP
BS_ICON, // IMAGE_CURSOR
BS_ICON // IMAGE_ICON
};
#define IsValidImage(imageType, realType, max) \
((imageType < max) && (rgbType[imageType] == realType))
typedef struct tagBTNDATA {
LPWSTR lpsz; // Text string
PBUTN pbutn; // Button data
WORD wFlags; // Alignment flags
} BTNDATA, FAR * LPBTNDATA;
void xxxDrawButton(PBUTN pbutn, HDC hdc, UINT pbfPush);
LOOKASIDE ButtonLookaside;
/*
* IsPushButton()
* Returns non-zero if the window is a push button. Returns flags that
* are interesting if it is. These flags are
*/
UINT IsPushButton(
PWND pwnd)
{
BYTE bStyle;
UINT flags;
bStyle = TestWF(pwnd, BFTYPEMASK);
flags = 0;
switch (bStyle) {
case LOBYTE(BS_PUSHBUTTON):
flags |= PBF_PUSHABLE;
break;
case LOBYTE(BS_DEFPUSHBUTTON):
flags |= PBF_PUSHABLE | PBF_DEFAULT;
break;
default:
if (TestWF(pwnd, BFPUSHLIKE))
flags |= PBF_PUSHABLE;
break;
}
return(flags);
}
/*
* GetAlignment()
* Gets default alignment of button. If BS_HORZMASK and/or BS_VERTMASK
* is specified, uses those. Otherwise, uses default for button.
* It's probably a fine time to describe what alignment flags mean for
* each type of button. Note that the presence of a bitmap/icon affects
* the meaning of alignments.
* (1) Push like buttons
* With one of {bitmap, icon, text}:
* Just like you'd expect
* With one of {bitmap, icon} AND text:
* Image & text are centered as a unit; alignment means where
* the image shows up. E.G., left-aligned means the image
* on the left, text on the right.
* (2) Radio/check like buttons
* Left aligned means check/radio box is on left, then bitmap/icon
* and text follows, left justified.
* Right aligned means checkk/radio box is on right, preceded by
* text and bitmap/icon, right justified.
* Centered has no meaning.
* With one of {bitmap, icon} AND text:
* Top aligned means bitmap/icon above, text below
* Bottom aligned means text above, bitmap/icon below
* With one of {bitmap, icon, text}
* Alignments mean what you'd expect.
* (3) Group boxes
* Left aligned means text is left justified on left side
* Right aligned means text is right justified on right side
* Center aligned means text is in middle
*/
WORD GetAlignment(
PWND pwnd)
{
BYTE bHorz;
BYTE bVert;
bHorz = TestWF(pwnd, BFHORZMASK);
bVert = TestWF(pwnd, BFVERTMASK);
if (!bHorz || !bVert) {
if (IsPushButton(pwnd)) {
if (!bHorz)
bHorz = LOBYTE(BFCENTER);
} else {
if (!bHorz)
bHorz = LOBYTE(BFLEFT);
}
if (!bVert)
bVert = LOBYTE(BFVCENTER);
}
return bHorz | bVert;
}
/*
* BNSetFont()
* Changes button font, and decides if we can use real bold font for default
* push buttons or if we have to simulate it.
*/
void BNSetFont(
PBUTN pbutn,
HFONT hfn,
BOOL fRedraw)
{
PWND pwnd = pbutn->spwnd;
pbutn->hFont = hfn;
if (fRedraw && IsVisible(pwnd)) {
NtUserInvalidateRect(HWq(pwnd), NULL, TRUE);
}
}
/*
* xxxBNInitDC
* History:
*/
HBRUSH xxxBNInitDC(
PBUTN pbutn,
HDC hdc)
{
UINT wColor;
BYTE bStyle;
HBRUSH hbr;
PWND pwnd = pbutn->spwnd;
CheckLock(pwnd);
/*
* Set BkMode before getting brush so that the app can change it to
* transparent if it wants.
*/
SetBkMode(hdc, OPAQUE);
bStyle = TestWF(pwnd, BFTYPEMASK);
switch (bStyle) {
default:
if (TestWF(pwnd, WFWIN40COMPAT) && !TestWF(pwnd, BFPUSHLIKE)) {
wColor = WM_CTLCOLORSTATIC;
break;
}
case LOBYTE(BS_PUSHBUTTON):
case LOBYTE(BS_DEFPUSHBUTTON):
case LOBYTE(BS_OWNERDRAW):
case LOBYTE(BS_USERBUTTON):
wColor = WM_CTLCOLORBTN;
break;
}
hbr = GetControlBrush(HWq(pwnd), hdc, wColor);
/*
* Select in the user's font if set, and save the old font so that we can
* restore it when we release the dc.
*/
if (pbutn->hFont) {
SelectObject(hdc, pbutn->hFont);
}
/*
* Clip output to the window rect if needed.
*/
if (bStyle != LOBYTE(BS_GROUPBOX)) {
IntersectClipRect(hdc, 0, 0,
pwnd->rcClient.right - pwnd->rcClient.left,
pwnd->rcClient.bottom - pwnd->rcClient.top);
}
if (TestWF(pwnd,WEFRTLREADING))
SetTextAlign(hdc, TA_RTLREADING | GetTextAlign(hdc));
return(hbr);
}
/*
* xxxBNGetDC
* History:
*/
HDC xxxBNGetDC(
PBUTN pbutn,
HBRUSH *lphbr)
{
HDC hdc;
PWND pwnd = pbutn->spwnd;
CheckLock(pwnd);
if (IsVisible(pwnd)) {
HBRUSH hbr;
hdc = NtUserGetDC(HWq(pwnd));
hbr = xxxBNInitDC(pbutn, hdc);
if (lphbr!=NULL)
*lphbr = hbr;
return hdc;
}
return NULL;
}
/*
* BNReleaseDC
* History:
*/
void BNReleaseDC(
PBUTN pbutn,
HDC hdc)
{
PWND pwnd = pbutn->spwnd;
if (TestWF(pwnd,WEFRTLREADING))
SetTextAlign(hdc, GetTextAlign(hdc) & ~TA_RTLREADING);
if (pbutn->hFont) {
SelectObject(hdc, ghFontSys);
}
ReleaseDC(HWq(pwnd), hdc);
}
/*
* xxxBNOwnerDraw
* History:
*/
void xxxBNOwnerDraw(
PBUTN pbutn,
HDC hdc,
UINT itemAction)
{
DRAWITEMSTRUCT drawItemStruct;
TL tlpwndParent;
PWND pwnd = pbutn->spwnd;
UINT itemState = 0;
if (TestWF(pwnd, WEFPUIFOCUSHIDDEN)) {
itemState |= ODS_NOFOCUSRECT;
}
if (TestWF(pwnd, WEFPUIACCELHIDDEN)) {
itemState |= ODS_NOACCEL;
}
if (BUTTONSTATE(pbutn) & BST_FOCUS) {
itemState |= ODS_FOCUS;
}
if (BUTTONSTATE(pbutn) & BST_PUSHED) {
itemState |= ODS_SELECTED;
}
if (TestWF(pwnd, WFDISABLED))
itemState |= ODS_DISABLED;
drawItemStruct.CtlType = ODT_BUTTON;
drawItemStruct.CtlID = PtrToUlong(pwnd->spmenu);
drawItemStruct.itemAction = itemAction;
drawItemStruct.itemState = itemState;
drawItemStruct.hwndItem = HWq(pwnd);
drawItemStruct.hDC = hdc;
_GetClientRect(pwnd, &drawItemStruct.rcItem);
drawItemStruct.itemData = 0L;
/*
* Send a WM_DRAWITEM message to the parent
* IanJa: in this case pMenu is being used as the control ID
*/
ThreadLock(REBASEPWND(pwnd, spwndParent), &tlpwndParent);
SendMessage(HW(REBASEPWND(pwnd, spwndParent)), WM_DRAWITEM, (WPARAM)pwnd->spmenu,
(LPARAM)&drawItemStruct);
ThreadUnlock(&tlpwndParent);
}
/*
* CalcBtnRect
* History:
*/
void BNCalcRect(
PWND pwnd,
HDC hdc,
LPRECT lprc,
int code,
UINT pbfFlags)
{
int cch;
SIZE extent;
int dy;
LPWSTR lpName;
UINT align;
_GetClientRect(pwnd, lprc);
align = GetAlignment(pwnd);
switch (code) {
case CBR_PUSHBUTTON:
// Subtract out raised edge all around
InflateRect(lprc, -SYSMET(CXEDGE), -SYSMET(CYEDGE));
if (pbfFlags & PBF_DEFAULT)
InflateRect(lprc, -SYSMET(CXBORDER), -SYSMET(CYBORDER));
break;
case CBR_CHECKBOX:
switch (align & LOBYTE(BFVERTMASK))
{
case LOBYTE(BFVCENTER):
lprc->top = (lprc->top + lprc->bottom - gpsi->oembmi[OBI_CHECK].cy) / 2;
break;
case LOBYTE(BFTOP):
case LOBYTE(BFBOTTOM):
PSMGetTextExtent(hdc, (LPWSTR)szOneChar, 1, &extent);
dy = extent.cy + extent.cy/4;
// Save vertical extent
extent.cx = dy;
// Get centered amount
dy = (dy - gpsi->oembmi[OBI_CHECK].cy) / 2;
if ((align & LOBYTE(BFVERTMASK)) == LOBYTE(BFTOP))
lprc->top += dy;
else
lprc->top = lprc->bottom - extent.cx + dy;
break;
}
if (TestWF(pwnd, BFRIGHTBUTTON))
lprc->left = lprc->right - gpsi->oembmi[OBI_CHECK].cx;
else
lprc->right = lprc->left + gpsi->oembmi[OBI_CHECK].cx;
break;
case CBR_CHECKTEXT:
if (TestWF(pwnd, BFRIGHTBUTTON)) {
lprc->right -= gpsi->oembmi[OBI_CHECK].cx;
// More spacing for 4.0 dudes
if (TestWF(pwnd, WFWIN40COMPAT)) {
PSMGetTextExtent(hdc, szOneChar, 1, &extent);
lprc->right -= extent.cx / 2;
}
} else {
lprc->left += gpsi->oembmi[OBI_CHECK].cx;
// More spacing for 4.0 dudes
if (TestWF(pwnd, WFWIN40COMPAT)) {
PSMGetTextExtent(hdc, szOneChar, 1, &extent);
lprc->left += extent.cx / 2;
}
}
break;
case CBR_GROUPTEXT:
if (!pwnd->strName.Length)
goto EmptyRect;
lpName = REBASE(pwnd, strName.Buffer);
if (!(cch = pwnd->strName.Length / sizeof(WCHAR))) {
EmptyRect:
SetRectEmpty(lprc);
break;
}
PSMGetTextExtent(hdc, lpName, cch, &extent);
extent.cx += SYSMET(CXEDGE) * 2;
switch (align & LOBYTE(BFHORZMASK))
{
// BFLEFT, nothing
case LOBYTE(BFLEFT):
lprc->left += (gpsi->cxSysFontChar - SYSMET(CXBORDER));
lprc->right = lprc->left + (int)(extent.cx);
break;
case LOBYTE(BFRIGHT):
lprc->right -= (gpsi->cxSysFontChar - SYSMET(CXBORDER));
lprc->left = lprc->right - (int)(extent.cx);
break;
case LOBYTE(BFCENTER):
lprc->left = (lprc->left + lprc->right - (int)(extent.cx)) / 2;
lprc->right = lprc->left + (int)(extent.cx);
break;
}
// Center aligned.
lprc->bottom = lprc->top + extent.cy + SYSMET(CYEDGE);
break;
case CBR_GROUPFRAME:
PSMGetTextExtent(hdc, (LPWSTR)szOneChar, 1, &extent);
lprc->top += extent.cy / 2;
break;
}
}
/*
* BtnGetMultiExtent()
* Calculates button text extent, given alignment flags.
*/
void BNMultiExtent(
WORD wFlags,
HDC hdc,
LPRECT lprcMax,
LPWSTR lpsz,
int cch,
PINT pcx,
PINT pcy)
{
RECT rcT;
UINT dtFlags = DT_CALCRECT | DT_WORDBREAK | DT_EDITCONTROL;
CopyRect(&rcT, lprcMax);
// Note that since we're just calculating the maximum dimensions,
// left-justification and top-justification are not important.
// Also, remember to leave margins horz and vert that follow our rules
// in DrawBtnText().
InflateRect(&rcT, -SYSMET(CXEDGE), -SYSMET(CYBORDER));
if ((wFlags & LOWORD(BS_HORZMASK)) == LOWORD(BS_CENTER))
dtFlags |= DT_CENTER;
if ((wFlags & LOWORD(BS_VERTMASK)) == LOWORD(BS_VCENTER))
dtFlags |= DT_VCENTER;
DrawTextExW(hdc, lpsz, cch, &rcT, dtFlags, NULL);
if (pcx)
*pcx = rcT.right-rcT.left;
if (pcy)
*pcy = rcT.bottom-rcT.top;
}
/*
* BtnMultiDraw()
* Draws multiline button text
*/
BOOL BNMultiDraw(
HDC hdc,
LPBTNDATA lpbd,
int cch,
int cx,
int cy)
{
RECT rcT;
UINT dtFlags = DT_WORDBREAK | DT_EDITCONTROL;
PBUTN pbutn = lpbd->pbutn;
if (TestWF(pbutn->spwnd, WEFPUIACCELHIDDEN)) {
dtFlags |= DT_HIDEPREFIX;
} else if (pbutn->fPaintKbdCuesOnly){
dtFlags |= DT_PREFIXONLY;
}
rcT.left = 0;
rcT.top = 0;
rcT.right = cx;
rcT.bottom = cy;
// Horizontal alignment
UserAssert(DT_LEFT == 0);
switch (lpbd->wFlags & LOWORD(BS_HORZMASK)) {
case LOWORD(BS_CENTER):
dtFlags |= DT_CENTER;
break;
case LOWORD(BS_RIGHT):
dtFlags |= DT_RIGHT;
break;
}
// Vertical alignment
UserAssert(DT_TOP == 0);
switch (lpbd->wFlags & LOWORD(BS_VERTMASK)) {
case LOWORD(BS_VCENTER):
dtFlags |= DT_VCENTER;
break;
case LOWORD(BS_BOTTOM):
dtFlags |= DT_BOTTOM;
break;
}
DrawTextExW(hdc, lpbd->lpsz, cch, &rcT, dtFlags, NULL);
return(TRUE);
}
/*
* xxxBNSetCapture
* History:
*/
BOOL xxxBNSetCapture(
PBUTN pbutn,
UINT codeMouse)
{
PWND pwnd = pbutn->spwnd;
BUTTONSTATE(pbutn) |= codeMouse;
CheckLock(pwnd);
if (!(BUTTONSTATE(pbutn) & BST_CAPTURED)) {
NtUserSetCapture(HWq(pwnd));
BUTTONSTATE(pbutn) |= BST_CAPTURED;
/*
* To prevent redundant CLICK messages, we set the INCLICK bit so
* the WM_SETFOCUS code will not do a xxxButtonNotifyParent(BN_CLICKED).
*/
BUTTONSTATE(pbutn) |= BST_INCLICK;
NtUserSetFocus(HWq(pwnd));
BUTTONSTATE(pbutn) &= ~BST_INCLICK;
}
return(BUTTONSTATE(pbutn) & BST_CAPTURED);
}
/*
* xxxButtonNotifyParent
* History:
*/
void xxxButtonNotifyParent(
PWND pwnd,
UINT code)
{
TL tlpwndParent;
PWND pwndParent; // Parent if it exists
CheckLock(pwnd);
if (pwnd->spwndParent)
pwndParent = REBASEPWND(pwnd, spwndParent);
else
pwndParent = pwnd;
/*
* Note: A button's pwnd->spmenu is used to store the control ID
*/
ThreadLock(pwndParent, &tlpwndParent);
SendMessage(HW(pwndParent), WM_COMMAND,
MAKELONG(PTR_TO_ID(pwnd->spmenu), code), (LPARAM)HWq(pwnd));
ThreadUnlock(&tlpwndParent);
}
/*
* xxxBNReleaseCapture
* History:
*/
void xxxBNReleaseCapture(
PBUTN pbutn,
BOOL fCheck)
{
PWND pwndT;
UINT check;
BOOL fNotifyParent = FALSE;
TL tlpwndT;
PWND pwnd = pbutn->spwnd;
CheckLock(pwnd);
if (BUTTONSTATE(pbutn) & BST_PUSHED) {
SendMessageWorker(pwnd, BM_SETSTATE, FALSE, 0, FALSE);
if (fCheck) {
switch (TestWF(pwnd, BFTYPEMASK)) {
case BS_AUTOCHECKBOX:
case BS_AUTO3STATE:
check = (UINT)((BUTTONSTATE(pbutn) & BST_CHECKMASK) + 1);
if (check > (UINT)(TestWF(pwnd, BFTYPEMASK) == BS_AUTO3STATE? BST_INDETERMINATE : BST_CHECKED)) {
check = BST_UNCHECKED;
}
SendMessageWorker(pwnd, BM_SETCHECK, check, 0, FALSE);
break;
case BS_AUTORADIOBUTTON:
pwndT = pwnd;
do {
ThreadLock(pwndT, &tlpwndT);
if ((UINT)SendMessage(HW(pwndT), WM_GETDLGCODE, 0, 0L) &
DLGC_RADIOBUTTON) {
SendMessage(HW(pwndT), BM_SETCHECK, (pwnd == pwndT), 0L);
}
pwndT = _GetNextDlgGroupItem(REBASEPWND(pwndT, spwndParent),
pwndT, FALSE);
ThreadUnlock(&tlpwndT);
} while (pwndT != pwnd);
}
fNotifyParent = TRUE;
}
}
if (BUTTONSTATE(pbutn) & BST_CAPTURED) {
BUTTONSTATE(pbutn) &= ~(BST_CAPTURED | BST_MOUSE);
NtUserReleaseCapture();
}
if (fNotifyParent) {
/*
* We have to do the notification after setting the buttonstate bits.
*/
xxxButtonNotifyParent(pwnd, BN_CLICKED);
}
}
/*
* DrawBtnText()
* Draws text of button.
*/
void xxxBNDrawText(
PBUTN pbutn,
HDC hdc,
BOOL dbt,
BOOL fDepress)
{
RECT rc;
HBRUSH hbr;
int x;
int y;
int cx;
int cy;
LPWSTR lpName;
BYTE bStyle;
int cch;
UINT dsFlags;
BTNDATA bdt;
UINT pbfPush;
PWND pwnd = pbutn->spwnd;
bStyle = TestWF(pwnd, BFTYPEMASK);
if (bStyle > sizeof(mpStyleCbr)) {
RIPERR0(ERROR_INVALID_PARAMETER, RIP_WARNING, "Invalid button style");
} else if ((bStyle == LOBYTE(BS_GROUPBOX)) && (dbt == DBT_FOCUS))
return;
pbfPush = IsPushButton(pwnd);
if (pbfPush) {
BNCalcRect(pwnd, hdc, &rc, CBR_PUSHBUTTON, pbfPush);
IntersectClipRect(hdc, rc.left, rc.top, rc.right, rc.bottom);
// This is because we were too stupid to have WM_CTLCOLOR,
// CTLCOLOR_BTN actually set up the damned button colors. For
// old apps, CTLCOLOR_BTN needs to work like CTLCOLOR_STATIC.
SetBkColor(hdc, SYSRGB(3DFACE));
SetTextColor(hdc, SYSRGB(BTNTEXT));
hbr = SYSHBR(BTNTEXT);
} else {
BNCalcRect(pwnd, hdc, &rc, mpStyleCbr[bStyle], pbfPush);
// Skip stuff for ownerdraw buttons, since we aren't going to
// draw text/image.
if (bStyle == LOBYTE(BS_OWNERDRAW))
goto DrawFocus;
else
hbr = SYSHBR(WINDOWTEXT);
}
// Alignment
bdt.wFlags = GetAlignment(pwnd);
bdt.pbutn = pbutn;
// Bail if we have nothing to draw
if (TestWF(pwnd, BFBITMAP)) {
BITMAP bmp;
// Bitmap button
if (!pbutn->hImage)
return;
GetObject(pbutn->hImage, sizeof(BITMAP), &bmp);
cx = bmp.bmWidth;
cy = bmp.bmHeight;
dsFlags = DST_BITMAP;
goto UseImageForName;
} else if (TestWF(pwnd, BFICON)) {
// Icon button
if (!pbutn->hImage)
return;
NtUserGetIconSize(pbutn->hImage, 0, &cx, &cy);
cy /= 2; // The bitmap height is half because a Mask is present in NT
dsFlags = DST_ICON;
UseImageForName:
lpName = (LPWSTR)pbutn->hImage;
cch = TRUE;
} else {
// Text button
if (!pwnd->strName.Length)
return;
lpName = REBASE(pwnd, strName.Buffer);
cch = pwnd->strName.Length / sizeof(WCHAR);
if (TestWF(pwnd, BFMULTILINE)) {
bdt.lpsz = lpName;
BNMultiExtent(bdt.wFlags, hdc, &rc, lpName, cch, &cx, &cy);
lpName = (LPWSTR)(LPBTNDATA)&bdt;
dsFlags = DST_COMPLEX;
} else {
SIZE size;
PSMGetTextExtent(hdc, lpName, cch, &size);
cx = size.cx;
cy = size.cy;
/*
* If the control doesn't need underlines, set DST_HIDEPREFIX and
* also do not show the focus indicator
*/
dsFlags = DST_PREFIXTEXT;
if (TestWF(pwnd, WEFPUIACCELHIDDEN)) {
dsFlags |= DSS_HIDEPREFIX;
} else if (pbutn->fPaintKbdCuesOnly) {
dsFlags |= DSS_PREFIXONLY;
}
}
// Add on a pixel or two of vertical space to make centering
// happier. That way underline won't abut focus rect unless
// spacing is really tight.
cy++;
}
// ALIGNMENT
// Horizontal
switch (bdt.wFlags & LOBYTE(BFHORZMASK)) {
// For left & right justified, we leave a margin of CXEDGE on either
// side for eye-pleasing space.
case LOBYTE(BFLEFT):
x = rc.left + SYSMET(CXEDGE);
break;
case LOBYTE(BFRIGHT):
x = rc.right - cx - SYSMET(CXEDGE);
break;
default:
x = (rc.left + rc.right - cx) / 2;
break;
}
// Vertical
switch (bdt.wFlags & LOBYTE(BFVERTMASK)) {
// For top & bottom justified, we leave a margin of CYBORDER on
// either side for more eye-pleasing space.
case LOBYTE(BFTOP):
y = rc.top + SYSMET(CYBORDER);
break;
case LOBYTE(BFBOTTOM):
y = rc.bottom - cy - SYSMET(CYBORDER);
break;
default:
y = (rc.top + rc.bottom - cy) / 2;
break;
}
// Draw the text
if (dbt & DBT_TEXT) {
// This isn't called for USER buttons.
UserAssert(bStyle != LOBYTE(BS_USERBUTTON));
if (fDepress) {
x += SYSMET(CXBORDER);
y += SYSMET(CYBORDER);
}
if (TestWF(pwnd, WFDISABLED)) {
UserAssert(HIBYTE(BFICON) == HIBYTE(BFBITMAP));
if (SYSMET(SLOWMACHINE) &&
!TestWF(pwnd, BFICON | BFBITMAP) &&
(GetBkColor(hdc) != SYSRGB(GRAYTEXT)))
{
// Perf && consistency with menus, statics
SetTextColor(hdc, SYSRGB(GRAYTEXT));
}
else
dsFlags |= DSS_DISABLED;
}
// Use transparent mode for checked push buttons since we're going to
// fill background with dither.
if (pbfPush) {
switch (BUTTONSTATE(pbutn) & BST_CHECKMASK) {
case BST_INDETERMINATE:
hbr = SYSHBR(GRAYTEXT);
dsFlags |= DSS_MONO;
// FALL THRU
case BST_CHECKED:
// Drawing on dithered background...
SetBkMode(hdc, TRANSPARENT);
break;
}
}
// Use brush and colors currently selected into hdc when we grabbed
// color
DrawState(hdc, hbr, (DRAWSTATEPROC)BNMultiDraw, (LPARAM)lpName,
(WPARAM)cch, x, y, cx, cy,
dsFlags);
}
// Draw focus rect.
// This can get called for OWNERDRAW and USERDRAW buttons. However, only
// OWNERDRAW buttons let the owner change the drawing of the focus button.
DrawFocus:
if (dbt & DBT_FOCUS) {
if (bStyle == LOBYTE(BS_OWNERDRAW)) {
// For ownerdraw buttons, this is only called in response to a
// WM_SETFOCUS or WM_KILL FOCUS message. So, we can check the
// new state of the focus by looking at the BUTTONSTATE bits
// which are set before this procedure is called.
xxxBNOwnerDraw(pbutn, hdc, ODA_FOCUS);
} else {
// Don't draw the focus if underlines are not turned on
if (!TestWF(pwnd, WEFPUIFOCUSHIDDEN)) {
// Let focus rect always hug edge of push buttons. We already
// have the client area setup for push buttons, so we don't have
// to do anything.
if (!pbfPush) {
RECT rcClient;
_GetClientRect(pwnd, &rcClient);
if (bStyle == LOBYTE(BS_USERBUTTON))
CopyRect(&rc, &rcClient);
else {
// Try to leave a border all around text. That causes
// focus to hug text.
rc.top = max(rcClient.top, y-SYSMET(CYBORDER));
rc.bottom = min(rcClient.bottom, rc.top + SYSMET(CYEDGE) + cy);
rc.left = max(rcClient.left, x-SYSMET(CXBORDER));
rc.right = min(rcClient.right, rc.left + SYSMET(CXEDGE) + cx);
}
} else
InflateRect(&rc, -SYSMET(CXBORDER), -SYSMET(CYBORDER));
// Are back & fore colors set properly?
DrawFocusRect(hdc, &rc);
}
}
}
}
/*
* DrawCheck()
*/
void xxxButtonDrawCheck(
PBUTN pbutn,
HDC hdc,
HBRUSH hbr)
{
RECT rc;
int bm;
UINT flags;
BOOL fDoubleBlt = FALSE;
TL tlpwnd;
PWND pwnd = pbutn->spwnd;
PWND pwndParent;
BNCalcRect(pwnd, hdc, &rc, CBR_CHECKBOX, 0);
flags = 0;
if (BUTTONSTATE(pbutn) & BST_CHECKMASK)
flags |= DFCS_CHECKED;
if (BUTTONSTATE(pbutn) & BST_PUSHED)
flags |= DFCS_PUSHED;
if (TestWF(pwnd, WFDISABLED))
flags |= DFCS_INACTIVE;
bm = OBI_CHECK;
switch (TestWF(pwnd, BFTYPEMASK)) {
case BS_AUTORADIOBUTTON:
case BS_RADIOBUTTON:
fDoubleBlt = TRUE;
bm = OBI_RADIO;
flags |= DFCS_BUTTONRADIO;
break;
case BS_3STATE:
case BS_AUTO3STATE:
if ((BUTTONSTATE(pbutn) & BST_CHECKMASK) == BST_INDETERMINATE) {
bm = OBI_3STATE;
flags |= DFCS_BUTTON3STATE;
break;
}
// FALL THRU
default:
flags |= DFCS_BUTTONCHECK;
break;
}
rc.right = rc.left + gpsi->oembmi[bm].cx;
rc.bottom = rc.top + gpsi->oembmi[bm].cy;
ThreadLockAlways(pwnd->spwndParent, &tlpwnd);
pwndParent = REBASEPWND(pwnd, spwndParent);
PaintRect(HW(pwndParent), HWq(pwnd), hdc, hbr, &rc);
ThreadUnlock(&tlpwnd);
if (TestWF(pwnd, BFFLAT) && gpsi->BitCount != 1) {
flags |= DFCS_MONO | DFCS_FLAT;
DrawFrameControl(hdc, &rc, DFC_BUTTON, flags);
} else {
switch (flags & (DFCS_CHECKED | DFCS_PUSHED | DFCS_INACTIVE))
{
case 0:
break;
case DFCS_CHECKED:
bm += DOBI_CHECK;
break;
// These are mutually exclusive!
case DFCS_PUSHED:
case DFCS_INACTIVE:
bm += DOBI_DOWN; // DOBI_DOWN == DOBI_INACTIVE
break;
case DFCS_CHECKED | DFCS_PUSHED:
bm += DOBI_CHECKDOWN;
break;
case DFCS_CHECKED | DFCS_INACTIVE:
bm += DOBI_CHECKDOWN + 1;
break;
}
if (fDoubleBlt) {
// This is a diamond-shaped radio button -- Blt with a mask so that
// the exterior keeps the same color as the window's background
DWORD clrTextSave = SetTextColor(hdc, 0x00000000L);
DWORD clrBkSave = SetBkColor(hdc, 0x00FFFFFFL);
POEMBITMAPINFO pOem = gpsi->oembmi + OBI_RADIOMASK;
NtUserBitBltSysBmp(hdc, rc.left, rc.top, pOem->cx, pOem->cy,
pOem->x, pOem->y, SRCAND);
pOem = gpsi->oembmi + bm;
NtUserBitBltSysBmp(hdc, rc.left, rc.top, pOem->cx, pOem->cy,
pOem->x, pOem->y, SRCINVERT);
SetTextColor(hdc, clrTextSave);
SetBkColor(hdc, clrBkSave);
} else {
POEMBITMAPINFO pOem = gpsi->oembmi + bm;
DWORD dwROP = 0;
#ifdef USE_MIRRORING
// We do not want to mirror the check box.
if (MIRRORED_HDC(hdc)) {
dwROP = NOMIRRORBITMAP;
}
#endif
NtUserBitBltSysBmp(hdc, rc.left, rc.top, pOem->cx, pOem->cy,
pOem->x, pOem->y, SRCCOPY | dwROP);
}
}
}
/*
* xxxButtonDrawNewState
* History:
*/
void xxxButtonDrawNewState(
PBUTN pbutn,
HDC hdc,
HBRUSH hbr,
UINT sOld)
{
PWND pwnd = pbutn->spwnd;
CheckLock(pwnd);
if (sOld != (UINT)(BUTTONSTATE(pbutn) & BST_PUSHED)) {
UINT pbfPush;
pbfPush = IsPushButton(pwnd);
switch (TestWF(pwnd, BFTYPEMASK)) {
case BS_GROUPBOX:
case BS_OWNERDRAW:
break;
default:
if (!pbfPush) {
xxxButtonDrawCheck(pbutn, hdc, hbr);
break;
}
case BS_PUSHBUTTON:
case BS_DEFPUSHBUTTON:
case BS_PUSHBOX:
xxxDrawButton(pbutn, hdc, pbfPush);
break;
}
}
}
/*
* DrawButton()
* Draws push-like button with text
*/
void xxxDrawButton(
PBUTN pbutn,
HDC hdc,
UINT pbfPush)
{
RECT rc;
UINT flags = 0;
UINT state = 0;
PWND pwnd = pbutn->spwnd;
if (BUTTONSTATE(pbutn) & BST_PUSHED)
state |= DFCS_PUSHED;
if (!pbutn->fPaintKbdCuesOnly) {
if (BUTTONSTATE(pbutn) & BST_CHECKMASK)
state |= DFCS_CHECKED;
if (TestWF(pwnd, WFWIN40COMPAT))
flags = BF_SOFT;
if (TestWF(pwnd, BFFLAT))
flags |= BF_FLAT | BF_MONO;
_GetClientRect(pwnd, &rc);
if (pbfPush & PBF_DEFAULT) {
DrawFrame(hdc, &rc, 1, DF_WINDOWFRAME);
InflateRect(&rc, -SYSMET(CXBORDER), -SYSMET(CYBORDER));
if (state & DFCS_PUSHED)
flags |= BF_FLAT;
}
DrawPushButton(hdc, &rc, state, flags);
}
xxxBNDrawText(pbutn, hdc, DBT_TEXT | (BUTTONSTATE(pbutn) &
BST_FOCUS ? DBT_FOCUS : 0), (state & DFCS_PUSHED));
}
/*
* xxxBNPaint
* History:
*/
void xxxBNPaint(
PBUTN pbutn,
HDC hdc)
{
UINT bsWnd;
RECT rc;
HBRUSH hbr;
HBRUSH hbrBtnSave;
TL tlpwndParent;
UINT pbfPush;
PWND pwnd = pbutn->spwnd;
PWND pwndParent;
CheckLock(pwnd);
hbr = xxxBNInitDC(pbutn, hdc);
bsWnd = TestWF(pwnd, BFTYPEMASK);
pbfPush = IsPushButton(pwnd);
if (!pbfPush && !pbutn->fPaintKbdCuesOnly) {
_GetClientRect(pwnd, &rc);
if ((bsWnd != LOBYTE(BS_OWNERDRAW)) &&
(bsWnd != LOBYTE(BS_GROUPBOX))) {
ThreadLock(pwnd->spwndParent, &tlpwndParent);
pwndParent = REBASEPWND(pwnd, spwndParent);
PaintRect(HW(pwndParent), HWq(pwnd), hdc, hbr, &rc);
ThreadUnlock(&tlpwndParent);
}
hbrBtnSave = SelectObject(hdc, hbr);
}
switch (bsWnd) {
case BS_CHECKBOX:
case BS_RADIOBUTTON:
case BS_AUTORADIOBUTTON:
case BS_3STATE:
case BS_AUTOCHECKBOX:
case BS_AUTO3STATE:
if (!pbfPush) {
xxxBNDrawText(pbutn, hdc,
DBT_TEXT | (BUTTONSTATE(pbutn) & BST_FOCUS ? DBT_FOCUS : 0), FALSE);
if (!pbutn->fPaintKbdCuesOnly) {
xxxButtonDrawCheck(pbutn, hdc, hbr);
}
break;
}
/*
* Fall through for PUSHLIKE buttons
*/
case BS_PUSHBUTTON:
case BS_DEFPUSHBUTTON:
xxxDrawButton(pbutn, hdc, pbfPush);
break;
case BS_PUSHBOX:
xxxBNDrawText(pbutn, hdc,
DBT_TEXT | (BUTTONSTATE(pbutn) & BST_FOCUS ? DBT_FOCUS : 0), FALSE);
xxxButtonDrawNewState(pbutn, hdc, hbr, 0);
break;
case BS_USERBUTTON:
xxxButtonNotifyParent(pwnd, BN_PAINT);
if (BUTTONSTATE(pbutn) & BST_PUSHED) {
xxxButtonNotifyParent(pwnd, BN_PUSHED);
}
if (TestWF(pwnd, WFDISABLED)) {
xxxButtonNotifyParent(pwnd, BN_DISABLE);
}
if (BUTTONSTATE(pbutn) & BST_FOCUS) {
xxxBNDrawText(pbutn, hdc, DBT_FOCUS, FALSE);
}
break;
case BS_OWNERDRAW:
xxxBNOwnerDraw(pbutn, hdc, ODA_DRAWENTIRE);
break;
case BS_GROUPBOX:
if (!pbutn->fPaintKbdCuesOnly) {
BNCalcRect(pwnd, hdc, &rc, CBR_GROUPFRAME, 0);
DrawEdge(hdc, &rc, EDGE_ETCHED, BF_RECT |
(TestWF(pwnd, BFFLAT) ? BF_FLAT | BF_MONO : 0));
BNCalcRect(pwnd, hdc, &rc, CBR_GROUPTEXT, 0);
ThreadLock(pwnd->spwndParent, &tlpwndParent);
pwndParent = REBASEPWND(pwnd, spwndParent);
PaintRect(HW(pwndParent), HWq(pwnd), hdc, hbr, &rc);
ThreadUnlock(&tlpwndParent);
}
/*
* FillRect(hdc, &rc, hbrBtn);
*/
xxxBNDrawText(pbutn, hdc, DBT_TEXT, FALSE);
break;
}
if (!pbfPush)
SelectObject(hdc, hbrBtnSave);
/*
* Release the font which may have been loaded by xxxButtonInitDC.
*/
if (pbutn->hFont) {
SelectObject(hdc, ghFontSys);
}
}
/*
* RepaintButton
*/
void RepaintButton (PBUTN pbutn)
{
HDC hdc = xxxBNGetDC(pbutn, NULL);
if (hdc != NULL) {
xxxBNPaint(pbutn, hdc);
BNReleaseDC(pbutn, hdc);
}
}
/*
* ButtonWndProc
* WndProc for buttons, check boxes, etc.
* History:
*/
LRESULT APIENTRY ButtonWndProcWorker(
PWND pwnd,
UINT message,
WPARAM wParam,
LPARAM lParam,
DWORD fAnsi)
{
HWND hwnd = HWq(pwnd);
UINT bsWnd;
UINT wOldState;
RECT rc;
POINT pt;
HDC hdc;
HBRUSH hbr;
PAINTSTRUCT ps;
TL tlpwndParent;
PBUTN pbutn;
PWND pwndParent;
static BOOL fInit = TRUE;
LONG lResult;
CheckLock(pwnd);
bsWnd = TestWF(pwnd, BFTYPEMASK);
VALIDATECLASSANDSIZE(pwnd, FNID_BUTTON);
INITCONTROLLOOKASIDE(&ButtonLookaside, BUTN, spwnd, 8);
/*
* Get the pbutn for the given window now since we will use it a lot in
* various handlers. This was stored using SetWindowLong(hwnd,0,pbutn) when
* we initially created the button control.
*/
pbutn = ((PBUTNWND)pwnd)->pbutn;
switch (message) {
case WM_NCHITTEST:
if (bsWnd == LOBYTE(BS_GROUPBOX)) {
return (LONG)HTTRANSPARENT;
} else {
goto CallDWP;
}
case WM_ERASEBKGND:
if (bsWnd == LOBYTE(BS_OWNERDRAW)) {
/*
* Handle erase background for owner draw buttons.
*/
_GetClientRect(pwnd, &rc);
ThreadLock(pwnd->spwndParent, &tlpwndParent);
pwndParent = REBASEPWND(pwnd, spwndParent);
PaintRect(HW(pwndParent), hwnd, (HDC)wParam, (HBRUSH)CTLCOLOR_BTN, &rc);
ThreadUnlock(&tlpwndParent);
}
/*
* Do nothing for other buttons, but don't let DefWndProc() do it
* either. It will be erased in xxxBNPaint().
*/
return (LONG)TRUE;
case WM_PRINTCLIENT:
xxxBNPaint(pbutn, (HDC)wParam);
break;
case WM_PAINT:
/*
* If wParam != NULL, then this is a subclassed paint.
*/
if ((hdc = (HDC)wParam) == NULL)
hdc = NtUserBeginPaint(hwnd, &ps);
if (IsVisible(pwnd))
xxxBNPaint(pbutn, hdc);
if (!wParam)
NtUserEndPaint(hwnd, &ps);
break;
case WM_SETFOCUS:
BUTTONSTATE(pbutn) |= BST_FOCUS;
if ((hdc = xxxBNGetDC(pbutn, NULL)) != NULL) {
xxxBNDrawText(pbutn, hdc, DBT_FOCUS, FALSE);
BNReleaseDC(pbutn, hdc);
}
if (TestWF(pwnd, BFNOTIFY))
xxxButtonNotifyParent(pwnd, BN_SETFOCUS);
if (!(BUTTONSTATE(pbutn) & BST_INCLICK)) {
switch (bsWnd) {
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_AUTORADIOBUTTON):
if (!(BUTTONSTATE(pbutn) & BST_DONTCLICK)) {
if (!(BUTTONSTATE(pbutn) & BST_CHECKMASK)) {
xxxButtonNotifyParent(pwnd, BN_CLICKED);
}
}
break;
}
}
break;
case WM_GETDLGCODE:
switch (bsWnd) {
case LOBYTE(BS_DEFPUSHBUTTON):
wParam = DLGC_DEFPUSHBUTTON;
break;
case LOBYTE(BS_PUSHBUTTON):
case LOBYTE(BS_PUSHBOX):
wParam = DLGC_UNDEFPUSHBUTTON;
break;
case LOBYTE(BS_AUTORADIOBUTTON):
case LOBYTE(BS_RADIOBUTTON):
wParam = DLGC_RADIOBUTTON;
break;
case LOBYTE(BS_GROUPBOX):
return (LONG)DLGC_STATIC;
case LOBYTE(BS_CHECKBOX):
case LOBYTE(BS_AUTOCHECKBOX):
/*
* If this is a char that is a '=/+', or '-', we want it
*/
if (lParam && ((LPMSG)lParam)->message == WM_CHAR) {
switch (wParam) {
case TEXT('='):
case TEXT('+'):
case TEXT('-'):
wParam = DLGC_WANTCHARS;
break;
default:
wParam = 0;
}
} else {
wParam = 0;
}
break;
default:
wParam = 0;
}
return (LONG)(wParam | DLGC_BUTTON);
case WM_CAPTURECHANGED:
if (BUTTONSTATE(pbutn) & BST_CAPTURED) {
// Unwittingly, we've been kicked out of capture,
// so undepress etc.
if (BUTTONSTATE(pbutn) & BST_MOUSE)
SendMessageWorker(pwnd, BM_SETSTATE, FALSE, 0, FALSE);
BUTTONSTATE(pbutn) &= ~(BST_CAPTURED | BST_MOUSE);
}
break;
case WM_KILLFOCUS:
/*
* If we are losing the focus and we are in "capture mode", click
* the button. This allows tab and space keys to overlap for
* fast toggle of a series of buttons.
*/
if (BUTTONSTATE(pbutn) & BST_MOUSE) {
/*
* If for some reason we are killing the focus, and we have the
* mouse captured, don't notify the parent we got clicked. This
* breaks Omnis Quartz otherwise.
*/
SendMessageWorker(pwnd, BM_SETSTATE, FALSE, 0, FALSE);
}
xxxBNReleaseCapture(pbutn, TRUE);
BUTTONSTATE(pbutn) &= ~BST_FOCUS;
if ((hdc = xxxBNGetDC(pbutn, NULL)) != NULL) {
xxxBNDrawText(pbutn, hdc, DBT_FOCUS, FALSE);
BNReleaseDC(pbutn, hdc);
}
if (TestWF(pwnd, BFNOTIFY))
xxxButtonNotifyParent(pwnd, BN_KILLFOCUS);
/*
* Since the bold border around the defpushbutton is done by
* someone else, we need to invalidate the rect so that the
* focus rect is repainted properly.
*/
NtUserInvalidateRect(hwnd, NULL, FALSE);
break;
case WM_LBUTTONDBLCLK:
/*
* Double click messages are recognized for BS_RADIOBUTTON,
* BS_USERBUTTON, and BS_OWNERDRAW styles. For all other buttons,
* double click is handled like a normal button down.
*/
switch (bsWnd) {
default:
if (!TestWF(pwnd, BFNOTIFY))
goto btnclick;
case LOBYTE(BS_USERBUTTON):
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_OWNERDRAW):
xxxButtonNotifyParent(pwnd, BN_DOUBLECLICKED);
break;
}
break;
case WM_LBUTTONUP:
if (BUTTONSTATE(pbutn) & BST_MOUSE) {
xxxBNReleaseCapture(pbutn, TRUE);
}
break;
case WM_MOUSEMOVE:
if (!(BUTTONSTATE(pbutn) & BST_MOUSE)) {
break;
}
/*
*** FALL THRU **
*/
case WM_LBUTTONDOWN:
btnclick:
if (xxxBNSetCapture(pbutn, BST_MOUSE)) {
_GetClientRect(pwnd, &rc);
POINTSTOPOINT(pt, lParam);
SendMessageWorker(pwnd, BM_SETSTATE, PtInRect(&rc, pt), 0, FALSE);
}
break;
case WM_CHAR:
if (BUTTONSTATE(pbutn) & BST_MOUSE)
goto CallDWP;
if (bsWnd != LOBYTE(BS_CHECKBOX) &&
bsWnd != LOBYTE(BS_AUTOCHECKBOX))
goto CallDWP;
switch (wParam) {
case TEXT('+'):
case TEXT('='):
wParam = 1; // we must Set the check mark on.
goto SetCheck;
case TEXT('-'):
wParam = 0; // Set the check mark off.
SetCheck:
// Must notify only if the check status changes
if ((WORD)(BUTTONSTATE(pbutn) & BST_CHECKMASK) != (WORD)wParam)
{
// We must check/uncheck only if it is AUTO
if (bsWnd == LOBYTE(BS_AUTOCHECKBOX))
{
if (xxxBNSetCapture(pbutn, 0))
{
SendMessageWorker(pwnd, BM_SETCHECK, wParam, 0, FALSE);
xxxBNReleaseCapture(pbutn, TRUE);
}
}
xxxButtonNotifyParent(pwnd, BN_CLICKED);
}
break;
default:
goto CallDWP;
}
break;
case BM_CLICK:
// Don't recurse into this code!
if (BUTTONSTATE(pbutn) & BST_INBMCLICK)
break;
BUTTONSTATE(pbutn) |= BST_INBMCLICK;
SendMessageWorker(pwnd, WM_LBUTTONDOWN, 0, 0, FALSE);
SendMessageWorker(pwnd, WM_LBUTTONUP, 0, 0, FALSE);
BUTTONSTATE(pbutn) &= ~BST_INBMCLICK;
/*
*** FALL THRU **
*/
case WM_KEYDOWN:
if (BUTTONSTATE(pbutn) & BST_MOUSE)
break;
if (wParam == VK_SPACE) {
if (xxxBNSetCapture(pbutn, 0)) {
SendMessageWorker(pwnd, BM_SETSTATE, TRUE, 0, FALSE);
}
} else {
xxxBNReleaseCapture(pbutn, FALSE);
}
break;
case WM_KEYUP:
case WM_SYSKEYUP:
if (BUTTONSTATE(pbutn) & BST_MOUSE) {
goto CallDWP;
}
/*
* Don't cancel the capture mode on the up of the tab in case the
* guy is overlapping tab and space keys.
*/
if (wParam == VK_TAB) {
goto CallDWP;
}
/*
* WARNING: pwnd is history after this call!
*/
xxxBNReleaseCapture(pbutn, (wParam == VK_SPACE));
if (message == WM_SYSKEYUP) {
goto CallDWP;
}
break;
case BM_GETSTATE:
return (LONG)BUTTONSTATE(pbutn);
case BM_SETSTATE:
wOldState = (UINT)(BUTTONSTATE(pbutn) & BST_PUSHED);
if (wParam) {
BUTTONSTATE(pbutn) |= BST_PUSHED;
} else {
BUTTONSTATE(pbutn) &= ~BST_PUSHED;
}
if ((hdc = xxxBNGetDC(pbutn, &hbr)) != NULL) {
if (bsWnd == LOBYTE(BS_USERBUTTON)) {
xxxButtonNotifyParent(pwnd, (UINT)(wParam ? BN_PUSHED : BN_UNPUSHED));
} else if (bsWnd == LOBYTE(BS_OWNERDRAW)) {
if (wOldState != (UINT)(BUTTONSTATE(pbutn) & BST_PUSHED)) {
/*
* Only notify for drawing if state has changed..
*/
xxxBNOwnerDraw(pbutn, hdc, ODA_SELECT);
}
} else {
xxxButtonDrawNewState(pbutn, hdc, hbr, wOldState);
}
BNReleaseDC(pbutn, hdc);
}
if (FWINABLE() && (wOldState != (BOOL)(BUTTONSTATE(pbutn) & BST_PUSHED))) {
NotifyWinEvent(EVENT_OBJECT_STATECHANGE, hwnd, OBJID_CLIENT, INDEXID_CONTAINER);
}
break;
case BM_GETCHECK:
return (LONG)(BUTTONSTATE(pbutn) & BST_CHECKMASK);
case BM_SETCHECK:
switch (bsWnd) {
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_AUTORADIOBUTTON):
if (wParam) {
SetWindowState(pwnd, WFTABSTOP);
} else {
ClearWindowState(pwnd, WFTABSTOP);
}
/*
*** FALL THRU **
*/
case LOBYTE(BS_CHECKBOX):
case LOBYTE(BS_AUTOCHECKBOX):
if (wParam) {
wParam = 1;
}
goto CheckIt;
case LOBYTE(BS_3STATE):
case LOBYTE(BS_AUTO3STATE):
if (wParam > BST_INDETERMINATE) {
wParam = BST_INDETERMINATE;
}
CheckIt:
if ((UINT)(BUTTONSTATE(pbutn) & BST_CHECKMASK) != (UINT)wParam) {
BUTTONSTATE(pbutn) &= ~BST_CHECKMASK;
BUTTONSTATE(pbutn) |= (UINT)wParam;
if (!IsVisible(pwnd))
break;
if ((hdc = xxxBNGetDC(pbutn, &hbr)) != NULL) {
if (TestWF(pwnd, BFPUSHLIKE)) {
xxxDrawButton(pbutn, hdc, PBF_PUSHABLE);
} else {
xxxButtonDrawCheck(pbutn, hdc, hbr);
}
BNReleaseDC(pbutn, hdc);
}
if (FWINABLE())
NotifyWinEvent(EVENT_OBJECT_STATECHANGE, hwnd, OBJID_CLIENT, INDEXID_CONTAINER);
}
break;
}
break;
case BM_SETSTYLE:
NtUserAlterWindowStyle(hwnd, BS_TYPEMASK, (DWORD)wParam);
if (lParam) {
NtUserInvalidateRect(hwnd, NULL, TRUE);
}
if (FWINABLE()) {
NotifyWinEvent(EVENT_OBJECT_STATECHANGE, hwnd, OBJID_CLIENT, INDEXID_CONTAINER);
}
break;
case WM_SETTEXT:
/*
* In case the new group name is longer than the old name,
* this paints over the old name before repainting the group
* box with the new name.
*/
if (bsWnd == LOBYTE(BS_GROUPBOX)) {
hdc = xxxBNGetDC(pbutn, &hbr);
if (hdc != NULL) {
BNCalcRect(pwnd, hdc, &rc, CBR_GROUPTEXT, 0);
NtUserInvalidateRect(hwnd, &rc, TRUE);
pwndParent = REBASEPWND(pwnd, spwndParent);
ThreadLock(pwnd->spwndParent, &tlpwndParent);
PaintRect(HW(pwndParent), hwnd, hdc, hbr, &rc);
ThreadUnlock(&tlpwndParent);
BNReleaseDC(pbutn, hdc);
}
}
lResult = _DefSetText(hwnd, (LPWSTR)lParam, (BOOL)fAnsi);
if (FWINABLE()) {
NotifyWinEvent(EVENT_OBJECT_NAMECHANGE, hwnd, OBJID_WINDOW, INDEXID_CONTAINER);
}
goto DoEnable;
/*
*** FALL THRU **
*/
case WM_ENABLE:
lResult = 0L;
DoEnable:
RepaintButton(pbutn);
return lResult;
case WM_SETFONT:
/*
* wParam - handle to the font
* lParam - if true, redraw else don't
*/
BNSetFont(pbutn, (HFONT)wParam, (BOOL)(lParam != 0));
break;
case WM_GETFONT:
return (LRESULT)pbutn->hFont;
case BM_GETIMAGE:
case BM_SETIMAGE:
if (!IsValidImage(wParam, TestWF(pwnd, BFIMAGEMASK), IMAGE_BMMAX)) {
RIPERR0(ERROR_INVALID_PARAMETER, RIP_WARNING, "Invalid button image type");
} else {
HANDLE hOld = pbutn->hImage;
if (message == BM_SETIMAGE) {
pbutn->hImage = (HANDLE)lParam;
if (TestWF(pwnd, WFVISIBLE)) {
NtUserInvalidateRect(hwnd, NULL, TRUE);
}
}
return (LRESULT)hOld;
}
break;
case WM_NCDESTROY:
case WM_FINALDESTROY:
if (pbutn) {
Unlock(&pbutn->spwnd);
FreeLookasideEntry(&ButtonLookaside, pbutn);
}
NtUserSetWindowFNID(hwnd, FNID_CLEANEDUP_BIT);
break;
case WM_NCCREATE:
// Borland's OBEX has a button with style 0x98; We didn't strip
// these bits in win3.1 because we checked for 0x08.
// Stripping these bits cause a GP Fault in OBEX.
// For win3.1 guys, I use the old code to strip the style bits.
if (TestWF(pwnd, WFWIN31COMPAT)) {
if(((!TestWF(pwnd, WFWIN40COMPAT)) &&
(((LOBYTE(pwnd->style)) & (LOBYTE(~BS_LEFTTEXT))) == LOBYTE(BS_USERBUTTON))) ||
(TestWF(pwnd, WFWIN40COMPAT) &&
(bsWnd == LOBYTE(BS_USERBUTTON))))
{
// BS_USERBUTTON is no longer allowed for 3.1 and beyond.
// Just turn to normal push button.
NtUserAlterWindowStyle(hwnd, BS_TYPEMASK, 0);
RIPMSG0(RIP_WARNING, "BS_USERBUTTON no longer supported");
}
}
if (TestWF(pwnd,WEFRIGHT)) {
NtUserAlterWindowStyle(hwnd, BS_RIGHT | BS_RIGHTBUTTON, BS_RIGHT | BS_RIGHTBUTTON);
}
goto CallDWP;
case WM_INPUTLANGCHANGEREQUEST:
// #115190
// If the window is one of controls on top of dialogbox,
// let the parent dialog handle it.
if (TestwndChild(pwnd) && pwnd->spwndParent) {
PWND pwndParent = REBASEPWND(pwnd, spwndParent);
if (pwndParent) {
PCLS pclsParent = REBASEALWAYS(pwndParent, pcls);
UserAssert(pclsParent != NULL);
if (pclsParent->atomClassName == gpsi->atomSysClass[ICLS_DIALOG]) {
RIPMSG0(RIP_VERBOSE, "Button: WM_INPUTLANGCHANGEREQUEST is sent to parent.\n");
return SendMessageWorker(pwndParent, message, wParam, lParam, FALSE);
}
}
}
goto CallDWP;
case WM_UPDATEUISTATE:
{
DefWindowProcWorker(pwnd, message, wParam, lParam, fAnsi);
if (ISBSTEXTOROD(pwnd)) {
pbutn->fPaintKbdCuesOnly = TRUE;
RepaintButton(pbutn);
pbutn->fPaintKbdCuesOnly = FALSE;
}
}
break;
default:
CallDWP:
return DefWindowProcWorker(pwnd, message, wParam, lParam, fAnsi);
}
return 0L;
}
/*
*/
LRESULT WINAPI ButtonWndProcA(
HWND hwnd,
UINT message,
WPARAM wParam,
LPARAM lParam)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return (0L);
}
/*
* If the control is not interested in this message,
* pass it to DefWindowProc.
*/
if (!FWINDOWMSG(message, FNID_BUTTON))
return DefWindowProcWorker(pwnd, message, wParam, lParam, TRUE);
return ButtonWndProcWorker(pwnd, message, wParam, lParam, TRUE);
}
LRESULT WINAPI ButtonWndProcW(
HWND hwnd,
UINT message,
WPARAM wParam,
LPARAM lParam)
{
PWND pwnd;
if ((pwnd = ValidateHwnd(hwnd)) == NULL) {
return (0L);
}
/*
* If the control is not interested in this message,
* pass it to DefWindowProc.
*/
if (!FWINDOWMSG(message, FNID_BUTTON))
return DefWindowProcWorker(pwnd, message, wParam, lParam, FALSE);
return ButtonWndProcWorker(pwnd, message, wParam, lParam, FALSE);
}