1950 lines
49 KiB
C
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);
|
||
|
}
|