windows-nt/Source/XPSP1/NT/shell/comctl32/v6/button.c
2020-09-26 16:20:57 +08:00

3475 lines
92 KiB
C

#include "ctlspriv.h"
#pragma hdrstop
#include "usrctl32.h"
#include "button.h"
//
// 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
#define CBR_RADIOBUTTON 6
#define Button_IsThemed(pbutn) ((pbutn)->hTheme && (pbutn)->hImage == NULL)
//---------------------------------------------------------------------------//
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
{
LPTSTR lpsz; // Text string
PBUTN pbutn; // Button data
WORD wFlags; // Alignment flags
} BTNDATA, *LPBTNDATA;
//---- to support multiple themes in a single process, move these into PBUTN ----
static SIZE sizeCheckBox = {0};
static SIZE sizeRadioBox = {0};
//---------------------------------------------------------------------------//
//
// Forwards
//
VOID Button_DrawPush(PBUTN pbutn, HDC hdc, UINT pbfPush);
VOID GetCheckBoxSize(HDC hdc, PBUTN pbutn, BOOL fCheckBox, LPSIZE psize);
WORD GetAlignment(PBUTN pbutn);
VOID Button_CalcRect(PBUTN pbutn, HDC hdc, LPRECT lprc, int iCode, UINT uFlags);
VOID Button_MultiExtent(WORD wFlags, HDC hdc, LPRECT lprcMax, LPTSTR lpsz, int cch, PINT pcx, PINT pcy);
__inline UINT IsPushButton(PBUTN pbutn);
__inline ULONG GetButtonType(ULONG ulWinStyle);
//---------------------------------------------------------------------------//
//
// InitButtonClass() - Registers the control's window class
//
BOOL InitButtonClass(HINSTANCE hInstance)
{
WNDCLASS wc;
wc.lpfnWndProc = Button_WndProc;
wc.lpszClassName = WC_BUTTON;
wc.style = CS_GLOBALCLASS | CS_PARENTDC | CS_DBLCLKS | CS_VREDRAW | CS_HREDRAW;
wc.cbClsExtra = 0;
wc.cbWndExtra = sizeof(PBUTN);
wc.hInstance = hInstance;
wc.hIcon = NULL;
wc.hCursor = LoadCursor(NULL, IDC_ARROW);
wc.hbrBackground = NULL;
wc.lpszMenuName = NULL;
if (!RegisterClass(&wc) && !GetClassInfo(hInstance, WC_BUTTON, &wc))
return FALSE;
return TRUE;
}
//---------------------------------------------------------------------------//
//
// Button_GetThemeIds() - Gets the associated iPartId and iStateId needed for
// the theme manager APIs for the button control passed
// in pbutn.
//
HRESULT Button_GetThemeIds(PBUTN pbutn, LPINT piPartId, LPINT piStateId)
{
if ( piPartId )
{
ULONG ulStyle = GET_STYLE(pbutn);
if (IsPushButton(pbutn))
{
*piPartId = BP_PUSHBUTTON;
}
else
{
switch (GetButtonType(ulStyle))
{
case BS_CHECKBOX:
case BS_AUTOCHECKBOX:
case BS_3STATE:
case BS_AUTO3STATE:
*piPartId = BP_CHECKBOX;
break;
case BS_RADIOBUTTON:
case BS_AUTORADIOBUTTON:
*piPartId = BP_RADIOBUTTON;
break;
case BS_GROUPBOX:
*piPartId = BP_GROUPBOX;
break;
case BS_OWNERDRAW:
//
// don't do anything with owerdrawn buttons
//
return E_FAIL;
default:
TraceMsg(TF_STANDARD, "What kind of buttonType is this, %#.2x", GetButtonType(ulStyle));
*piPartId = BP_PUSHBUTTON;
break;
}
}
if (piStateId)
{
switch (*piPartId)
{
case BP_PUSHBUTTON:
if ((pbutn->buttonState & BST_PUSHED) ||
((pbutn->buttonState & (BST_CHECKED|BST_HOT)) == BST_CHECKED))
{
*piStateId = PBS_PRESSED;
}
else if (!IsWindowEnabled(pbutn->ci.hwnd))
{
*piStateId = PBS_DISABLED;
}
else if (pbutn->buttonState & BST_HOT)
{
*piStateId = PBS_HOT;
}
else if (ulStyle & BS_DEFPUSHBUTTON)
{
*piStateId = PBS_DEFAULTED;
}
else
{
*piStateId = PBS_NORMAL;
}
break;
case BP_CHECKBOX:
case BP_RADIOBUTTON:
//
// NOTE (phellyar): We're relying on the order of the RADIOBUTTONSTATES and
// CHECKBOXSTATES enums in tmdefs.h to calculate the correct
// StateId. If the ordering of those enums changes, revisit
// the logic here.
// Note also that CHECKBOXSTATES is a super set of
// RADIOBUTTONSTATES which is why we're using CBS_* here.
//
if ( pbutn->buttonState & BST_CHECKED )
{
//
// button is checked
//
*piStateId = CBS_CHECKEDNORMAL;
}
else if ( pbutn->buttonState & BST_INDETERMINATE )
{
//
// button is intedeterminate
//
*piStateId = CBS_MIXEDNORMAL;
}
else
{
//
// button is unchecked
//
*piStateId = CBS_UNCHECKEDNORMAL;
}
if ( pbutn->buttonState & BST_PUSHED )
{
//
// being pressed
//
*piStateId += 2;
}
else if (!IsWindowEnabled(pbutn->ci.hwnd))
{
//
// disabled
//
*piStateId += 3;
}
else if (pbutn->buttonState & BST_HOT )
{
//
// mouse over
//
*piStateId += 1;
}
break;
case BP_GROUPBOX:
if (!IsWindowEnabled(pbutn->ci.hwnd))
{
*piStateId = GBS_DISABLED;
}
else
{
*piStateId = GBS_NORMAL;
}
break;
}
}
}
return S_OK;
}
//---------------------------------------------------------------------------//
//
// Button_GetTextFlags() - Returns the DrawTextEx flags that should be used
// when rendering text for this control, needed by
// DrawThemeText.
//
DWORD Button_GetTextFlags(PBUTN pbutn)
{
DWORD dwTextFlags = 0;
WORD wAlign = GetAlignment(pbutn);
ULONG ulStyle = GET_STYLE(pbutn);
//
// Set up text flags
//
//
// horizontal text alignment
//
switch (wAlign & HIBYTE(BS_HORZMASK))
{
case HIBYTE(BS_LEFT):
dwTextFlags |= DT_LEFT;
break;
case HIBYTE(BS_RIGHT):
dwTextFlags |= DT_RIGHT;
break;
case HIBYTE(BS_CENTER):
dwTextFlags |= DT_CENTER;
break;
}
//
// vertical text alignment
//
switch (wAlign & HIBYTE(BS_VERTMASK))
{
case HIBYTE(BS_TOP):
dwTextFlags |= DT_TOP;
break;
case HIBYTE(BS_BOTTOM):
dwTextFlags |= DT_BOTTOM;
break;
case HIBYTE(BS_VCENTER):
dwTextFlags |= DT_VCENTER;
break;
}
//
// line break
//
if (ulStyle & BS_MULTILINE)
{
dwTextFlags |= (DT_WORDBREAK | DT_EDITCONTROL);
}
else
{
dwTextFlags |= DT_SINGLELINE;
}
if (ulStyle & SS_NOPREFIX)
{
dwTextFlags |= DT_NOPREFIX;
}
//
// Draw the underscore for accelorators?
//
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIACCELHIDDEN))
{
dwTextFlags |= DT_HIDEPREFIX;
}
return dwTextFlags;
}
DWORD ButtonStateToCustomDrawState(PBUTN pbutn)
{
DWORD itemState = 0;
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIFOCUSHIDDEN))
{
itemState |= CDIS_SHOWKEYBOARDCUES;
}
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIACCELHIDDEN))
{
itemState |= CDIS_SHOWKEYBOARDCUES;
}
if (BUTTONSTATE(pbutn) & BST_FOCUS)
{
itemState |= CDIS_FOCUS;
}
if (BUTTONSTATE(pbutn) & BST_PUSHED)
{
itemState |= CDIS_SELECTED;
}
if (BUTTONSTATE(pbutn) & BST_HOT)
{
itemState |= CDIS_HOT;
}
if (!IsWindowEnabled(pbutn->ci.hwnd))
{
itemState |= CDIS_DISABLED;
}
return itemState;
}
void Button_GetImagePosition(PBUTN pbutn, RECT* prc, int* px, int* py)
{
int cx = 0;
int cy = 0;
CCGetIconSize(&pbutn->ci, pbutn->himl, &cx, &cy);
cx += pbutn->rcIcon.left + pbutn->rcIcon.right;
cy += pbutn->rcIcon.top + pbutn->rcIcon.bottom;
switch (pbutn->uAlign)
{
case BUTTON_IMAGELIST_ALIGN_RIGHT:
*px = prc->right - cx;
*py = prc->top + (RECTHEIGHT(*prc) - cy) / 2 + pbutn->rcIcon.top;
prc->right -= cx;
break;
case BUTTON_IMAGELIST_ALIGN_CENTER: // This means no text
*px = prc->left + (RECTWIDTH(*prc) - cx) / 2 + pbutn->rcIcon.left;
*py = prc->top + (RECTHEIGHT(*prc) - cy) / 2 + pbutn->rcIcon.top;
break;
case BUTTON_IMAGELIST_ALIGN_TOP:
*px = prc->left + (RECTWIDTH(*prc) - cx) / 2 + pbutn->rcIcon.left;
*py = pbutn->rcIcon.top;
prc->top += cy;
break;
case BUTTON_IMAGELIST_ALIGN_BOTTOM:
*px = (RECTWIDTH(*prc) - cx) / 2 + pbutn->rcIcon.left;
*py = prc->bottom - cy;
prc->bottom -= cy;
break;
case BUTTON_IMAGELIST_ALIGN_LEFT:
// Fall
default:
*px = prc->left + pbutn->rcIcon.left;
*py = prc->top + (RECTHEIGHT(*prc) - cy) / 2 + pbutn->rcIcon.top;
prc->left += cx;
break;
}
}
//---------------------------------------------------------------------------//
//
// Button_DrawThemed() - Renders button control according to the current
// theme.
// pbutn - the button control to render
// hdc - the hdc to draw on
// iPartId - the button part
// iStateId - the button state
//
HRESULT Button_DrawThemed(PBUTN pbutn, HDC hdc, int iPartId, int iStateId)
{
HRESULT hr;
RECT rcClient;
RECT rcContent;
RECT rcFocus;
RECT rcCheck;
DWORD dwTextFlags;
LPWSTR pszText;
int cch;
NMCUSTOMDRAW nmcd = {0};
BOOL fRadioOrCheck = (iPartId == BP_RADIOBUTTON || iPartId == BP_CHECKBOX );
//
// Render the button background
//
GetClientRect(pbutn->ci.hwnd, &rcClient);
rcCheck = rcContent = rcClient;
if ( fRadioOrCheck )
{
SIZE sizeChar;
SIZE sizeCheck;
int iCode;
//
// Compat....
//
GetTextExtentPoint32(hdc, TEXT("0"), 1, &sizeChar);
GetCheckBoxSize(hdc, pbutn, (iPartId == BP_CHECKBOX), &sizeCheck);
if (iPartId == BP_CHECKBOX)
iCode = CBR_CHECKBOX;
else
iCode = CBR_RADIOBUTTON;
Button_CalcRect(pbutn, hdc, &rcCheck, iCode, 0);
rcCheck.bottom = rcCheck.top + sizeCheck.cx;
if ((GET_STYLE(pbutn) & BS_RIGHTBUTTON) != 0)
{
rcCheck.left = rcContent.right - sizeCheck.cx;
rcContent.right = rcCheck.left - (sizeChar.cx/2);
}
else
{
rcCheck.right = rcContent.left + sizeCheck.cx;
rcContent.left = rcCheck.right + (sizeChar.cx/2);
}
//---- shrink radiobutton/checkbox button to fix client rect ----
if (RECTWIDTH(rcClient) < RECTWIDTH(rcCheck))
{
rcCheck.right = rcCheck.left + RECTWIDTH(rcClient);
}
if (RECTHEIGHT(rcClient) < RECTHEIGHT(rcCheck))
{
rcCheck.bottom = rcCheck.top + RECTHEIGHT(rcClient);
}
}
nmcd.hdc = hdc;
nmcd.rc = rcClient;
nmcd.dwItemSpec = GetWindowID(pbutn->ci.hwnd);
nmcd.uItemState = ButtonStateToCustomDrawState(pbutn);
pbutn->ci.dwCustom = CICustomDrawNotify(&pbutn->ci, CDDS_PREERASE, &nmcd);
if (!(pbutn->ci.dwCustom & CDRF_SKIPDEFAULT))
{
hr = DrawThemeBackground(pbutn->hTheme, hdc, iPartId, iStateId, &rcCheck, 0);
if (FAILED(hr))
{
TraceMsg(TF_STANDARD, "Failed to render theme background");
return hr;
}
if (pbutn->ci.dwCustom & CDRF_NOTIFYPOSTERASE)
CICustomDrawNotify(&pbutn->ci, CDDS_POSTERASE, &nmcd);
pbutn->ci.dwCustom = CICustomDrawNotify(&pbutn->ci, CDDS_PREPAINT, &nmcd);
if (!(pbutn->ci.dwCustom & CDRF_SKIPDEFAULT))
{
//
// Render the button text
//
GetThemeBackgroundContentRect(pbutn->hTheme, hdc, iPartId, iStateId, &rcContent, &rcContent);
rcFocus = rcContent;
if (pbutn->himl)
{
int x, y;
int iImage = 0;
if (ImageList_GetImageCount(pbutn->himl) > 1)
{
iImage = (iStateId - PBS_NORMAL);
}
Button_GetImagePosition(pbutn, &rcContent, &x, &y);
ImageList_Draw(pbutn->himl, iImage, hdc, x, y, ILD_TRANSPARENT | (CCDPIScale(pbutn->ci)?ILD_DPISCALE:0));
}
//
// Get the button text
//
cch = GetWindowTextLength(pbutn->ci.hwnd);
if (cch == 0)
{
//
// Nothing to draw
//
return hr;
}
pszText = UserLocalAlloc(0, sizeof(WCHAR)*(cch+1));
if (pszText == NULL)
{
TraceMsg(TF_STANDARD, "Can't allocate buffer");
return E_FAIL;
}
GetWindowTextW(pbutn->ci.hwnd, pszText, cch+1);
dwTextFlags = Button_GetTextFlags(pbutn);
if ( TESTFLAG(GET_STYLE(pbutn), BS_MULTILINE) || fRadioOrCheck )
{
int cxWidth, cyHeight;
TEXTMETRIC tm;
if ( TESTFLAG(GET_STYLE(pbutn), BS_MULTILINE) )
{
RECT rcTextExtent = rcContent;
cyHeight = DrawTextEx(hdc, pszText, cch, &rcTextExtent, dwTextFlags|DT_CALCRECT, NULL);
cxWidth = RECTWIDTH(rcTextExtent);
}
else
{
SIZE size;
LPWSTR pszStrip = UserLocalAlloc(0, sizeof(WCHAR)*(cch+1));
if ( pszStrip )
{
int cchStrip = StripAccelerators(pszText, pszStrip, TRUE);
GetTextExtentPoint32(hdc, pszStrip, cchStrip, &size);
UserLocalFree(pszStrip);
}
else
{
GetTextExtentPoint32(hdc, pszText, cch, &size);
}
cyHeight = size.cy;
cxWidth = size.cx;
}
if ( fRadioOrCheck && (cyHeight < RECTHEIGHT(rcCheck)))
{
// optimization for single line check/radios, align them with the top
// of the check no matter when the vertical alignment
rcContent.top = rcCheck.top;
}
else
{
if (dwTextFlags & DT_VCENTER)
{
rcContent.top += (RECTHEIGHT(rcContent) - cyHeight) / 2;
}
else if (dwTextFlags & DT_BOTTOM)
{
rcContent.top = rcContent.bottom - cyHeight;
}
}
if ( GetTextMetrics( hdc, &tm ) && (tm.tmInternalLeading == 0) )
{
// Far East fonts that have no leading. Leave space to prevent
// focus rect from obscuring text.
rcContent.top += g_cyBorder;
}
rcContent.bottom = rcContent.top + cyHeight;
if (dwTextFlags & DT_CENTER)
{
rcContent.left += (RECTWIDTH(rcContent) - cxWidth) / 2;
}
else if (dwTextFlags & DT_RIGHT)
{
rcContent.left = rcContent.right - cxWidth;
}
rcContent.right= rcContent.left + cxWidth;
if ( fRadioOrCheck )
{
//
// Inflate the bounding rect a litte, but contrained to
// within the client area.
//
rcFocus.top = max(rcClient.top, rcContent.top-1);
rcFocus.bottom = min(rcClient.bottom, rcContent.bottom+1);
rcFocus.left = max(rcClient.left, rcContent.left-1);
rcFocus.right = min(rcClient.right, rcContent.right+1);
}
}
hr = DrawThemeText(pbutn->hTheme, hdc, iPartId, iStateId, pszText, cch, dwTextFlags, 0, &rcContent);
if (FAILED(hr))
{
TraceMsg(TF_STANDARD, "Failed to render button text");
}
if (!TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIFOCUSHIDDEN) && (BUTTONSTATE(pbutn) & BST_FOCUS))
{
DrawFocusRect(hdc, &rcFocus);
}
UserLocalFree(pszText);
if (pbutn->ci.dwCustom & CDRF_NOTIFYPOSTPAINT)
{
CICustomDrawNotify(&pbutn->ci, CDDS_POSTPAINT, &nmcd);
}
}
}
return hr;
}
//---------------------------------------------------------------------------//
//
// Button_GetTheme() - Get a handle to the theme for this button control
//
HTHEME Button_GetTheme(PBUTN pbutn)
{
//
// Button's with predefined IDs can be
// themed differently
//
static LPWSTR szButtonClasses[] =
{
L"Button", // =0
L"Button-OK;Button", // IDOK=1
L"Button-CANCEL;Button", // IDCANCEL=2
L"Button-ABORT;Button", // IDABORT=3
L"Button-RETRY;Button", // IDRETRY=4
L"Button-IGNORE;Button", // IDIGNORE=5
L"Button-YES;Button", // IDYES=6
L"Button-NO;Button", // IDNO=7
L"Button-CLOSE;Button", // IDCLOSE=8
L"Button-HELP;Button", // IDHELP=9
L"Button-TRYAGAIN;Button", // IDTRYAGAIN=10
L"Button-CONTINUE;Button", // IDCONTINUE=11
L"Button-APPLY;Button", // IDAPPLY=12 (not yet std)
};
int iButtonId = GetWindowID(pbutn->ci.hwnd);
if (iButtonId < 0 || iButtonId >= ARRAYSIZE(szButtonClasses)) // outside range
{
iButtonId = 0;
}
EnableThemeDialogTexture(GetParent(pbutn->ci.hwnd), ETDT_ENABLE);
return OpenThemeData(pbutn->ci.hwnd, szButtonClasses[iButtonId]);
}
//---------------------------------------------------------------------------//
//
VOID GetCheckBoxSize(HDC hdc, PBUTN pbutn, BOOL fCheckBox, LPSIZE psize)
{
SIZE *psz;
if (fCheckBox)
psz = &sizeCheckBox;
else
psz = &sizeRadioBox;
if ((! psz->cx) && (! psz->cy)) // not yet calculated
{
BOOL fGotSize = FALSE;
if (pbutn->hTheme) // get themed size
{
int iPartId;
HRESULT hr;
if (fCheckBox)
iPartId = BP_CHECKBOX;
else
iPartId = BP_RADIOBUTTON;
hr = GetThemePartSize(pbutn->hTheme, hdc, iPartId, 1, NULL, TS_DRAW, psz);
if (FAILED(hr))
{
TraceMsg(TF_STANDARD, "Failed to get theme part size for checkbox/radiobutton");
}
else
{
fGotSize = TRUE;
}
}
if (! fGotSize) // get classic size (use checkbox for both)
{
HBITMAP hbmp = LoadBitmap(NULL, MAKEINTRESOURCE(OBM_CHECKBOXES));
if (hbmp != NULL)
{
BITMAP bmp;
GetObject(hbmp, sizeof(BITMAP), &bmp);
//
// Checkbox bitmap is arranged 4 over and three down. Only need to get
// the size of a single checkbox, so do the math here.
//
psz->cx = bmp.bmWidth / 4;
psz->cy = bmp.bmHeight / 3;
DeleteObject(hbmp);
}
else
{
AssertMsg(hbmp != NULL, TEXT("Unable to load checkbox bitmap"));
}
}
}
*psize = *psz;
}
//---------------------------------------------------------------------------//
//
__inline BYTE GetButtonStyle(ULONG ulWinStyle)
{
return (BYTE) LOBYTE(ulWinStyle & BS_TYPEMASK);
}
//---------------------------------------------------------------------------//
//
__inline ULONG GetButtonType(ULONG ulWinStyle)
{
return ulWinStyle & BS_TYPEMASK;
}
//---------------------------------------------------------------------------//
//
// IsPushButton()
//
// Returns non-zero if the window is a push button. Returns flags that
// are interesting if it is. These flags are
//
UINT IsPushButton(PBUTN pbutn)
{
BYTE bStyle;
UINT flags;
ULONG ulStyle = GET_STYLE(pbutn);
bStyle = GetButtonStyle(ulStyle);
flags = 0;
switch (bStyle)
{
case LOBYTE(BS_PUSHBUTTON):
flags |= PBF_PUSHABLE;
break;
case LOBYTE(BS_DEFPUSHBUTTON):
flags |= PBF_PUSHABLE | PBF_DEFAULT;
break;
default:
if (ulStyle & BS_PUSHLIKE)
{
flags |= PBF_PUSHABLE;
}
}
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(PBUTN pbutn)
{
BYTE bHorz;
BYTE bVert;
ULONG ulStyle = GET_STYLE(pbutn);
bHorz = HIBYTE(ulStyle & BS_HORZMASK);
bVert = HIBYTE(ulStyle & BS_VERTMASK);
if (!bHorz || !bVert)
{
if (IsPushButton(pbutn))
{
if (!bHorz)
{
bHorz = HIBYTE(BS_CENTER);
}
}
else
{
if (!bHorz)
{
bHorz = HIBYTE(BS_LEFT);
}
}
if (GetButtonStyle(ulStyle) == BS_GROUPBOX)
{
if (!bVert)
{
bVert = HIBYTE(BS_TOP);
}
}
else
{
if (!bVert)
{
bVert = HIBYTE(BS_VCENTER);
}
}
}
return bHorz | bVert;
}
//---------------------------------------------------------------------------//
//
// Button_SetFont()
//
// Changes button font, and decides if we can use real bold font for default
// push buttons or if we have to simulate it.
//
VOID Button_SetFont(PBUTN pbutn, HFONT hFont, BOOL fRedraw)
{
pbutn->hFont = hFont;
if (fRedraw && IsWindowVisible(pbutn->ci.hwnd))
{
InvalidateRect(pbutn->ci.hwnd, NULL, TRUE);
}
}
//---------------------------------------------------------------------------//
//
HBRUSH Button_InitDC(PBUTN pbutn, HDC hdc)
{
UINT uMsg;
BYTE bStyle;
HBRUSH hBrush;
ULONG ulStyle = GET_STYLE(pbutn);
ULONG ulStyleEx = GET_EXSTYLE(pbutn);
//
// Set BkMode before getting brush so that the app can change it to
// transparent if it wants.
//
SetBkMode(hdc, OPAQUE);
bStyle = GetButtonStyle(ulStyle);
switch (bStyle)
{
default:
if (TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT) && ((ulStyle & BS_PUSHLIKE) == 0))
{
uMsg = WM_CTLCOLORSTATIC;
break;
}
case LOBYTE(BS_PUSHBUTTON):
case LOBYTE(BS_DEFPUSHBUTTON):
case LOBYTE(BS_OWNERDRAW):
case LOBYTE(BS_USERBUTTON):
uMsg = WM_CTLCOLORBTN;
break;
}
hBrush = (HBRUSH)SendMessage(GetParent(pbutn->ci.hwnd), uMsg, (WPARAM)hdc, (LPARAM)pbutn->ci.hwnd);
//
// 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))
{
RECT rcClient;
GetClientRect(pbutn->ci.hwnd, &rcClient);
IntersectClipRect(hdc, 0, 0,
rcClient.right,
rcClient.bottom);
}
if ((ulStyleEx & WS_EX_RTLREADING) != 0)
{
SetTextAlign(hdc, TA_RTLREADING | GetTextAlign(hdc));
}
return hBrush;
}
//---------------------------------------------------------------------------//
//
HDC Button_GetDC(PBUTN pbutn, HBRUSH *phBrush)
{
HDC hdc = NULL;
if (IsWindowVisible(pbutn->ci.hwnd))
{
HBRUSH hBrush;
hdc = GetDC(pbutn->ci.hwnd);
hBrush = Button_InitDC(pbutn, hdc);
if ((phBrush != NULL) && hBrush)
{
*phBrush = hBrush;
}
}
return hdc;
}
//---------------------------------------------------------------------------//
//
VOID Button_ReleaseDC(PBUTN pbutn, HDC hdc, HBRUSH *phBrush)
{
ULONG ulStyleEx = GET_EXSTYLE(pbutn);
if ((ulStyleEx & WS_EX_RTLREADING) != 0)
{
SetTextAlign(hdc, GetTextAlign(hdc) & ~TA_RTLREADING);
}
if (pbutn->hFont)
{
SelectObject(hdc, GetStockObject(SYSTEM_FONT));
}
ReleaseDC(pbutn->ci.hwnd, hdc);
}
//---------------------------------------------------------------------------//
//
VOID Button_OwnerDraw(PBUTN pbutn, HDC hdc, UINT itemAction)
{
DRAWITEMSTRUCT drawItemStruct;
UINT itemState = 0;
int iButtonId = GetWindowID(pbutn->ci.hwnd);
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIFOCUSHIDDEN))
{
itemState |= ODS_NOFOCUSRECT;
}
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIACCELHIDDEN))
{
itemState |= ODS_NOACCEL;
}
if (TESTFLAG(BUTTONSTATE(pbutn), BST_FOCUS))
{
itemState |= ODS_FOCUS;
}
if (TESTFLAG(BUTTONSTATE(pbutn), BST_PUSHED))
{
itemState |= ODS_SELECTED;
}
if (!IsWindowEnabled(pbutn->ci.hwnd))
{
itemState |= ODS_DISABLED;
}
//
// Populate the draw item struct
//
drawItemStruct.CtlType = ODT_BUTTON;
drawItemStruct.CtlID = iButtonId;
drawItemStruct.itemAction = itemAction;
drawItemStruct.itemState = itemState;
drawItemStruct.hwndItem = pbutn->ci.hwnd;
drawItemStruct.hDC = hdc;
GetClientRect(pbutn->ci.hwnd, &drawItemStruct.rcItem);
drawItemStruct.itemData = 0L;
//
// Send a WM_DRAWITEM message to our parent
//
SendMessage(GetParent(pbutn->ci.hwnd),
WM_DRAWITEM,
(WPARAM)iButtonId,
(LPARAM)&drawItemStruct);
}
//---------------------------------------------------------------------------//
//
VOID Button_CalcRect(PBUTN pbutn, HDC hdc, LPRECT lprc, int iCode, UINT uFlags)
{
CONST TCHAR szOneChar[] = TEXT("0");
SIZE sizeExtent;
int dy;
LPTSTR lpName = NULL;
WORD wAlign;
int cxEdge, cyEdge;
int cxBorder, cyBorder;
ULONG ulStyle = GET_STYLE(pbutn);
ULONG ulStyleEx = GET_EXSTYLE(pbutn);
cxEdge = GetSystemMetrics(SM_CXEDGE);
cyEdge = GetSystemMetrics(SM_CYEDGE);
cxBorder = GetSystemMetrics(SM_CXBORDER);
cyBorder = GetSystemMetrics(SM_CYBORDER);
GetClientRect(pbutn->ci.hwnd, lprc);
wAlign = GetAlignment(pbutn);
switch (iCode)
{
case CBR_PUSHBUTTON:
//
// Subtract out raised edge all around
//
InflateRect(lprc, -cxEdge, -cyEdge);
if (uFlags & PBF_DEFAULT)
{
InflateRect(lprc, -cxBorder, -cyBorder);
}
break;
case CBR_CHECKBOX:
case CBR_RADIOBUTTON:
{
SIZE sizeChk = {0};
GetCheckBoxSize(hdc, pbutn, (iCode == CBR_CHECKBOX), &sizeChk);
switch (wAlign & HIBYTE(BS_VERTMASK))
{
case HIBYTE(BS_VCENTER):
lprc->top = (lprc->top + lprc->bottom - sizeChk.cy) / 2;
break;
case HIBYTE(BS_TOP):
case HIBYTE(BS_BOTTOM):
GetTextExtentPoint32(hdc, (LPTSTR)szOneChar, 1, &sizeExtent);
dy = sizeExtent.cy + sizeExtent.cy/4;
//
// Save vertical extent
//
sizeExtent.cx = dy;
//
// Get centered amount
//
dy = (dy - sizeChk.cy) / 2;
if ((wAlign & HIBYTE(BS_VERTMASK)) == HIBYTE(BS_TOP))
{
lprc->top += dy;
}
else
{
lprc->top = lprc->bottom - sizeExtent.cx + dy;
}
break;
}
if ((ulStyle & BS_RIGHTBUTTON) != 0)
{
lprc->left = lprc->right - sizeChk.cx;
}
else
{
lprc->right = lprc->left + sizeChk.cx;
}
break;
}
case CBR_CHECKTEXT:
{
SIZE sizeChk = {0};
GetCheckBoxSize(hdc, pbutn, TRUE, &sizeChk);
if ((ulStyle & BS_RIGHTBUTTON) != 0)
{
lprc->right -= sizeChk.cx;
//
// More spacing for 4.0 dudes
//
if (TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT))
{
GetTextExtentPoint32(hdc, szOneChar, 1, &sizeExtent);
lprc->right -= sizeExtent.cx / 2;
}
}
else
{
lprc->left += sizeChk.cx;
//
// More spacing for 4.0 dudes
//
if (TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT))
{
GetTextExtentPoint32(hdc, szOneChar, 1, &sizeExtent);
lprc->left += sizeExtent.cx / 2;
}
}
break;
}
case CBR_GROUPTEXT:
{
int cch = GetWindowTextLength(pbutn->ci.hwnd);
if (cch != 0)
{
//
// Always use WCHAR for alloc because of MBCS
//
lpName = _alloca((cch+1)*sizeof(WCHAR));
}
if (cch == 0 || lpName == NULL || GetWindowText(pbutn->ci.hwnd, lpName, cch+1) == 0)
{
SetRectEmpty(lprc);
break;
}
//
// if not themed
//
if (!Button_IsThemed(pbutn))
{
GetTextExtentPoint32(hdc, lpName, cch, &sizeExtent);
}
else
{
DWORD dwTextFlags = Button_GetTextFlags(pbutn);
RECT rcExtent;
GetThemeTextExtent(pbutn->hTheme,
hdc,
BP_GROUPBOX,
0,
lpName,
cch,
dwTextFlags,
lprc,
&rcExtent);
sizeExtent.cx = RECTWIDTH(rcExtent);
sizeExtent.cy = RECTHEIGHT(rcExtent);
}
sizeExtent.cx += GetSystemMetrics(SM_CXEDGE) * 2;
switch (wAlign & HIBYTE(BS_HORZMASK))
{
//
// BFLEFT, nothing
//
case HIBYTE(BS_LEFT):
lprc->left += (SYSFONT_CXCHAR - GetSystemMetrics(SM_CXBORDER));
lprc->right = lprc->left + (int)(sizeExtent.cx);
break;
case HIBYTE(BS_RIGHT):
lprc->right -= (SYSFONT_CXCHAR - GetSystemMetrics(SM_CXBORDER));
lprc->left = lprc->right - (int)(sizeExtent.cx);
break;
case HIBYTE(BS_CENTER):
lprc->left = (lprc->left + lprc->right - (int)(sizeExtent.cx)) / 2;
lprc->right = lprc->left + (int)(sizeExtent.cx);
break;
}
//
// Center aligned.
//
lprc->bottom = lprc->top + sizeExtent.cy + GetSystemMetrics(SM_CYEDGE);
break;
}
case CBR_GROUPFRAME:
GetTextExtentPoint32(hdc, (LPTSTR)szOneChar, 1, &sizeExtent);
lprc->top += sizeExtent.cy / 2;
break;
}
}
//---------------------------------------------------------------------------//
//
// Button_MultiExtent()
//
// Calculates button text extent, given alignment flags.
//
VOID Button_MultiExtent(WORD wFlags, HDC hdc, LPRECT lprcMax, LPTSTR lpsz, int cch, PINT pcx, PINT pcy)
{
RECT rc;
UINT dtFlags = DT_CALCRECT | DT_WORDBREAK | DT_EDITCONTROL;
CopyRect(&rc, 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(&rc, -GetSystemMetrics(SM_CXEDGE), -GetSystemMetrics(SM_CYBORDER));
if ((wFlags & LOWORD(BS_HORZMASK)) == LOWORD(BS_CENTER))
{
dtFlags |= DT_CENTER;
}
if ((wFlags & LOWORD(BS_VERTMASK)) == LOWORD(BS_VCENTER))
{
dtFlags |= DT_VCENTER;
}
DrawTextEx(hdc, lpsz, cch, &rc, dtFlags, NULL);
if (pcx)
{
*pcx = rc.right-rc.left;
}
if (pcy)
{
*pcy = rc.bottom-rc.top;
}
}
//---------------------------------------------------------------------------//
//
// Button_MultiDraw()
//
// Draws multiline button text
//
BOOL Button_MultiDraw(HDC hdc, LPBTNDATA lpbd, int cch, int cx, int cy)
{
RECT rc;
UINT dtFlags = DT_WORDBREAK | DT_EDITCONTROL;
PBUTN pbutn = lpbd->pbutn;
if (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIACCELHIDDEN))
{
dtFlags |= DT_HIDEPREFIX;
}
else if (pbutn->fPaintKbdCuesOnly)
{
dtFlags |= DT_PREFIXONLY;
}
rc.left = 0;
rc.top = 0;
rc.right = cx;
rc.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;
}
DrawTextEx(hdc, lpbd->lpsz, cch, &rc, dtFlags, NULL);
return TRUE;
}
//---------------------------------------------------------------------------//
//
BOOL Button_SetCapture(PBUTN pbutn, UINT uCodeMouse)
{
BUTTONSTATE(pbutn) |= uCodeMouse;
if (!(BUTTONSTATE(pbutn) & BST_CAPTURED))
{
SetCapture(pbutn->ci.hwnd);
BUTTONSTATE(pbutn) |= BST_CAPTURED;
//
// To prevent redundant CLICK messages, we set the INCLICK bit so
// the WM_SETFOCUS code will not do a Button_NotifyParent(BN_CLICKED).
//
BUTTONSTATE(pbutn) |= BST_INCLICK;
SetFocus(pbutn->ci.hwnd);
BUTTONSTATE(pbutn) &= ~BST_INCLICK;
}
return BUTTONSTATE(pbutn) & BST_CAPTURED;
}
//---------------------------------------------------------------------------//
//
VOID Button_NotifyParent(PBUTN pbutn, UINT uCode)
{
HWND hwndParent = GetParent(pbutn->ci.hwnd);
int iButtonId = GetWindowID(pbutn->ci.hwnd);
if ( !hwndParent )
{
hwndParent = pbutn->ci.hwnd;
}
SendMessage(hwndParent,
WM_COMMAND,
MAKELONG(iButtonId, uCode),
(LPARAM)pbutn->ci.hwnd);
}
//---------------------------------------------------------------------------//
//
VOID Button_ReleaseCapture(PBUTN pbutn, BOOL fCheck)
{
UINT uCheck;
BOOL fNotifyParent = FALSE;
ULONG ulStyle = GET_STYLE(pbutn);
if (BUTTONSTATE(pbutn) & BST_PUSHED)
{
SendMessage(pbutn->ci.hwnd, BM_SETSTATE, FALSE, 0);
if (fCheck)
{
switch (GetButtonType(ulStyle))
{
case BS_AUTOCHECKBOX:
case BS_AUTO3STATE:
uCheck = (UINT)((BUTTONSTATE(pbutn) & BST_CHECKMASK) + 1);
if (uCheck > (UINT)(GetButtonType(ulStyle) == BS_AUTO3STATE ? BST_INDETERMINATE : BST_CHECKED))
{
uCheck = BST_UNCHECKED;
}
SendMessage(pbutn->ci.hwnd, BM_SETCHECK, uCheck, 0);
break;
case BS_AUTORADIOBUTTON:
{
//
// Walk the radio buttons in the same group as us. Check ourself
// and uncheck everyone else.
//
HWND hwndNext = pbutn->ci.hwnd;
HWND hwndParent = GetParent(pbutn->ci.hwnd);
do
{
if ((UINT)SendMessage(hwndNext, WM_GETDLGCODE, 0, 0L) & DLGC_RADIOBUTTON)
{
SendMessage(hwndNext, BM_SETCHECK, hwndNext == pbutn->ci.hwnd, 0L);
}
hwndNext = GetNextDlgGroupItem(hwndParent, hwndNext, FALSE);
}
//
// Loop until we see ourself again
//
while (hwndNext != pbutn->ci.hwnd);
break;
}
}
fNotifyParent = TRUE;
}
}
if (BUTTONSTATE(pbutn) & BST_CAPTURED)
{
BUTTONSTATE(pbutn) &= ~(BST_CAPTURED | BST_MOUSE);
ReleaseCapture();
}
if (fNotifyParent)
{
//
// We have to do the notification after setting the buttonstate bits.
//
Button_NotifyParent(pbutn, BN_CLICKED);
}
}
//---------------------------------------------------------------------------//
//
// Button_DrawText()
//
// Draws text of button.
//
VOID Button_DrawText(PBUTN pbutn, HDC hdc, BOOL dbt, BOOL fDepress)
{
RECT rc;
HBRUSH hbr;
LPTSTR lpName;
int x = 0, y = 0;
int cx = 0, cy = 0;
BYTE bStyle;
UINT dsFlags;
BTNDATA bdt;
UINT pbfPush;
ULONG ulStyle = GET_STYLE(pbutn);
int cch;
bStyle = GetButtonStyle(ulStyle);
if ((bStyle == LOBYTE(BS_GROUPBOX)) && (dbt == DBT_FOCUS))
{
return;
}
pbfPush = IsPushButton(pbutn);
cch = GetWindowTextLength(pbutn->ci.hwnd);
lpName = _alloca((cch + 1) * sizeof(wchar_t));
if (lpName == NULL)
{
return;
}
GetWindowText(pbutn->ci.hwnd, lpName, cch + 1);
//
// if not themed
//
if (!Button_IsThemed(pbutn))
{
if (pbfPush)
{
Button_CalcRect(pbutn, hdc, &rc, CBR_PUSHBUTTON, pbfPush);
IntersectClipRect(hdc, rc.left, rc.top, rc.right, rc.bottom);
//
// This is because we didn't have WM_CTLCOLOR,
// CTLCOLOR_BTN actually set up the button colors. For
// old apps, CTLCOLOR_BTN needs to work like CTLCOLOR_STATIC.
//
SetBkColor(hdc, GetSysColor(COLOR_3DFACE));
SetTextColor(hdc, GetSysColor(COLOR_BTNTEXT));
hbr = GetSysColorBrush(COLOR_BTNTEXT);
}
else
{
Button_CalcRect(pbutn, 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 = GetSysColorBrush(COLOR_WINDOWTEXT);
}
}
if (pbutn->himl)
{
int x, y;
Button_GetImagePosition(pbutn, &rc, &x, &y);
if (fDepress)
{
x += GetSystemMetrics(SM_CXBORDER);
y += GetSystemMetrics(SM_CYBORDER);
}
ImageList_Draw(pbutn->himl, 0, hdc, x, y, ILD_TRANSPARENT | (CCDPIScale(pbutn->ci)?ILD_DPISCALE:0));
}
//
// Alignment
//
bdt.wFlags = GetAlignment(pbutn);
bdt.pbutn = pbutn;
//
// Bail if we have nothing to draw
//
if ((ulStyle & BS_BITMAP) != 0)
{
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 ((ulStyle & BS_ICON) != 0)
{
SIZE sizeIcon;
//
// Icon button
//
if (!pbutn->hImage)
{
return;
}
GetIconSize(pbutn->hImage, &sizeIcon);
cx = sizeIcon.cx;
cy = sizeIcon.cy;
dsFlags = DST_ICON;
UseImageForName:
lpName = (LPTSTR)pbutn->hImage;
cch = TRUE;
}
else
{
//
// Text button
//
if (cch == 0)
{
return;
}
if ((ulStyle & BS_MULTILINE) != 0)
{
bdt.lpsz = lpName;
Button_MultiExtent(bdt.wFlags, hdc, &rc, lpName, cch, &cx, &cy);
lpName = (LPTSTR)(LPBTNDATA)&bdt;
dsFlags = DST_COMPLEX;
}
else
{
SIZE size;
LPWSTR lpwstr = UserLocalAlloc(0, sizeof(WCHAR)*(cch+1));
//
// Try to get the text extent with mnemonics stripped.
//
if (lpwstr != NULL)
{
int iLen = StripAccelerators(lpName, lpwstr, TRUE);
GetTextExtentPoint32(hdc, lpwstr, iLen, &size);
UserLocalFree(lpwstr);
}
else
{
GetTextExtentPoint32(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 (TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIACCELHIDDEN))
{
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 & HIBYTE(BS_HORZMASK))
{
//
// For left & right justified, we leave a margin of CXEDGE on either
// side for eye-pleasing space.
//
case HIBYTE(BS_LEFT):
x = rc.left + GetSystemMetrics(SM_CXEDGE);
break;
case HIBYTE(BS_RIGHT):
x = rc.right - cx - GetSystemMetrics(SM_CXEDGE);
break;
default:
x = (rc.left + rc.right - cx) / 2;
break;
}
//
// Vertical
//
switch (bdt.wFlags & HIBYTE(BS_VERTMASK))
{
//
// For top & bottom justified, we leave a margin of CYBORDER on
// either side for more eye-pleasing space.
//
case HIBYTE(BS_TOP):
y = rc.top + GetSystemMetrics(SM_CYBORDER);
break;
case HIBYTE(BS_BOTTOM):
y = rc.bottom - cy - GetSystemMetrics(SM_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 += GetSystemMetrics(SM_CXBORDER);
y += GetSystemMetrics(SM_CYBORDER);
}
if ( !IsWindowEnabled(pbutn->ci.hwnd) )
{
UserAssert(HIBYTE(BS_ICON) == HIBYTE(BS_BITMAP));
if (GetSystemMetrics(SM_SLOWMACHINE) &&
((ulStyle & (BS_ICON | BS_BITMAP)) != 0) &&
(GetBkColor(hdc) != GetSysColor(COLOR_GRAYTEXT)))
{
//
// Perf && consistency with menus, statics
//
SetTextColor(hdc, GetSysColor(COLOR_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 = GetSysColorBrush(COLOR_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)Button_MultiDraw, (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.
//
Button_OwnerDraw(pbutn, hdc, ODA_FOCUS);
}
else
{
//
// Don't draw the focus if underlines are not turned on
//
if (!TESTFLAG(GET_EXSTYLE(pbutn), WS_EXP_UIFOCUSHIDDEN))
{
//
// 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(pbutn->ci.hwnd, &rcClient);
if (bStyle == LOBYTE(BS_USERBUTTON))
{
CopyRect(&rc, &rcClient);
}
else if (Button_IsThemed(pbutn))
{
//
// if themed
//
int iPartId = 0;
int iStateId = 0;
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
GetThemeBackgroundContentRect(pbutn->hTheme,
hdc,
iPartId,
iStateId,
&rcClient,
&rc);
GetThemeTextExtent(pbutn->hTheme,
hdc,
iPartId,
iStateId,
lpName,
-1,
Button_GetTextFlags(pbutn),
&rc,
&rc);
//
// Inflate the bounding rect a litte, but contrained to
// within the client area.
//
rc.top = max(rcClient.top, rc.top-1);
rc.bottom = min(rcClient.bottom, rc.bottom+1);
rc.left = max(rcClient.left, rc.left-1);
rc.right = min(rcClient.right, rc.right+1);
}
else
{
//
// Try to leave a border all around text. That causes
// focus to hug text.
//
rc.top = max(rcClient.top, y-GetSystemMetrics(SM_CYBORDER));
rc.bottom = min(rcClient.bottom, rc.top + GetSystemMetrics(SM_CYEDGE) + cy);
rc.left = max(rcClient.left, x-GetSystemMetrics(SM_CXBORDER));
rc.right = min(rcClient.right, rc.left + GetSystemMetrics(SM_CXEDGE) + cx);
}
}
else
{
InflateRect(&rc, -GetSystemMetrics(SM_CXBORDER), -GetSystemMetrics(SM_CYBORDER));
}
//
// Are back & fore colors set properly?
//
DrawFocusRect(hdc, &rc);
}
}
}
}
//---------------------------------------------------------------------------//
//
// DrawCheck()
//
VOID Button_DrawCheck(PBUTN pbutn, HDC hdc, HBRUSH hBrush)
{
//
// if not themed
//
if (!Button_IsThemed(pbutn)) // Images don't have a mask so look ugly. Need to use old painting
{
RECT rc;
UINT uFlags;
BOOL fDoubleBlt = FALSE;
ULONG ulStyle = GET_STYLE(pbutn);
SIZE sizeChk = {0};
Button_CalcRect(pbutn, hdc, &rc, CBR_CHECKBOX, 0);
uFlags = 0;
if ( BUTTONSTATE(pbutn) & BST_CHECKMASK )
{
uFlags |= DFCS_CHECKED;
}
if ( BUTTONSTATE(pbutn) & BST_PUSHED )
{
uFlags |= DFCS_PUSHED;
}
if ( !IsWindowEnabled(pbutn->ci.hwnd) )
{
uFlags |= DFCS_INACTIVE;
}
switch (GetButtonType(ulStyle))
{
case BS_AUTORADIOBUTTON:
case BS_RADIOBUTTON:
fDoubleBlt = TRUE;
uFlags |= DFCS_BUTTONRADIO;
break;
case BS_3STATE:
case BS_AUTO3STATE:
if ((BUTTONSTATE(pbutn) & BST_CHECKMASK) == BST_INDETERMINATE)
{
uFlags |= DFCS_BUTTON3STATE;
break;
}
//
// FALL THRU
//
default:
uFlags |= DFCS_BUTTONCHECK;
break;
}
if ((ulStyle & BS_FLAT) != 0)
{
uFlags |= DFCS_FLAT | DFCS_MONO;
}
GetCheckBoxSize(hdc, pbutn, TRUE, &sizeChk);
rc.right = rc.left + sizeChk.cx;
rc.bottom = rc.top + sizeChk.cy;
FillRect(hdc, &rc, hBrush);
DrawFrameControl(hdc, &rc, DFC_BUTTON, uFlags);
}
else
{
int iStateId = 0;
int iPartId = 0;
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
if ((iPartId != BP_RADIOBUTTON) && (iPartId != BP_CHECKBOX))
{
TraceMsg(TF_STANDARD, "Button_DrawCheck: Not a radio or check, iPartId = %d", iPartId);
return;
}
Button_DrawThemed(pbutn, hdc, iPartId, iStateId);
}
}
//---------------------------------------------------------------------------//
//
VOID Button_DrawNewState(PBUTN pbutn, HDC hdc, HBRUSH hbr, UINT sOld)
{
if (sOld != (UINT)(BUTTONSTATE(pbutn) & BST_PUSHED))
{
UINT pbfPush;
ULONG ulStyle = GET_STYLE(pbutn);
pbfPush = IsPushButton(pbutn);
switch (GetButtonType(ulStyle))
{
case BS_GROUPBOX:
case BS_OWNERDRAW:
break;
default:
if (!pbfPush)
{
Button_DrawCheck(pbutn, hdc, hbr);
break;
}
case BS_PUSHBUTTON:
case BS_DEFPUSHBUTTON:
case BS_PUSHBOX:
Button_DrawPush(pbutn, hdc, pbfPush);
break;
}
}
}
//---------------------------------------------------------------------------//
//
// Button_DrawPush()
//
// Draws push-like button with text
//
VOID Button_DrawPush(PBUTN pbutn, HDC hdc, UINT pbfPush)
{
//
// if not themed
//
if (!Button_IsThemed(pbutn))
{
RECT rc;
UINT uFlags = 0;
UINT uState = 0;
ULONG ulStyle = GET_STYLE(pbutn);
NMCUSTOMDRAW nmcd = {0};
//
// Always a push button if calling this function
//
uState = DFCS_BUTTONPUSH;
GetClientRect(pbutn->ci.hwnd, &rc);
nmcd.hdc = hdc;
nmcd.rc = rc;
nmcd.dwItemSpec = GetWindowID(pbutn->ci.hwnd);
nmcd.uItemState = ButtonStateToCustomDrawState(pbutn);
if (BUTTONSTATE(pbutn) & BST_PUSHED)
{
uState |= DFCS_PUSHED;
}
pbutn->ci.dwCustom = CICustomDrawNotify(&pbutn->ci, CDDS_PREERASE, &nmcd);
if (!(pbutn->ci.dwCustom & CDRF_SKIPDEFAULT))
{
if (!pbutn->fPaintKbdCuesOnly)
{
if (BUTTONSTATE(pbutn) & BST_CHECKMASK)
{
uState |= DFCS_CHECKED;
}
if (TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT))
{
uFlags = BF_SOFT;
}
if ((ulStyle & BS_FLAT) != 0)
{
uFlags |= DFCS_FLAT | DFCS_MONO;
}
if (pbfPush & PBF_DEFAULT)
{
int cxBorder = GetSystemMetrics(SM_CXBORDER);
int cyBorder = GetSystemMetrics(SM_CYBORDER);
int clFrame = 1;
int x = rc.left;
int y = rc.top;
int cxWidth = cxBorder * clFrame;
int cyWidth = cyBorder * clFrame;
int cx = rc.right - x - cxWidth;
int cy = rc.bottom - y - cyWidth;
HBRUSH hbrFill = GetSysColorBrush(COLOR_WINDOWFRAME);
HBRUSH hbrSave = SelectObject(hdc, hbrFill);
PatBlt(hdc, x, y, cxWidth, cy, PATCOPY);
PatBlt(hdc, x + cxWidth, y, cx, cyWidth, PATCOPY);
PatBlt(hdc, x, y + cy, cx, cyWidth, PATCOPY);
PatBlt(hdc, x + cx, y + cyWidth, cxWidth, cy, PATCOPY);
SelectObject(hdc, hbrSave);
InflateRect(&rc, -cxBorder, -cyBorder);
if (uState & DFCS_PUSHED)
{
uFlags |= DFCS_FLAT;
}
}
DrawFrameControl(hdc, &rc, DFC_BUTTON, uState | uFlags);
}
if (pbutn->ci.dwCustom & CDRF_NOTIFYPOSTERASE)
CICustomDrawNotify(&pbutn->ci, CDDS_POSTERASE, &nmcd);
pbutn->ci.dwCustom = CICustomDrawNotify(&pbutn->ci, CDDS_PREPAINT, &nmcd);
if (!(pbutn->ci.dwCustom & CDRF_SKIPDEFAULT))
{
Button_DrawText(pbutn, hdc, DBT_TEXT | (BUTTONSTATE(pbutn) &
BST_FOCUS ? DBT_FOCUS : 0), (uState & DFCS_PUSHED));
if (pbutn->ci.dwCustom & CDRF_NOTIFYPOSTPAINT)
CICustomDrawNotify(&pbutn->ci, CDDS_POSTPAINT, &nmcd);
}
}
}
else
{
int iStateId = 0;
int iPartId = 0;
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
if (iPartId != BP_PUSHBUTTON)
{
TraceMsg(TF_STANDARD, "Not a Pushbutton");
return;
}
Button_DrawThemed(pbutn, hdc, iPartId, iStateId);
}
}
BOOL Button_OnSetImageList(PBUTN pbutn, BUTTON_IMAGELIST* biml)
{
BOOL fRet = FALSE;
if (biml)
{
if (biml->himl)
{
pbutn->rcIcon = biml->margin;
pbutn->himl = biml->himl;
pbutn->uAlign = biml->uAlign;
fRet = TRUE;
}
}
return fRet;
}
void ApplyMarginsToRect(RECT* prcm, RECT* prc)
{
prc->left -= prcm->left;
prc->top -= prcm->top;
prc->right += prcm->right;
prc->bottom += prcm->bottom;
}
BOOL Button_OnGetIdealSize(PBUTN pbutn, PSIZE psize)
{
UINT bsWnd;
RECT rc = {0};
HBRUSH hBrush;
HDC hdc;
if (psize == NULL)
return FALSE;
GetWindowRect(pbutn->ci.hwnd, &rc);
hdc = GetDC (pbutn->ci.hwnd);
if (hdc)
{
ULONG ulStyle = GET_STYLE(pbutn);
bsWnd = GetButtonType(ulStyle);
hBrush = Button_InitDC(pbutn, hdc);
switch (bsWnd)
{
case BS_PUSHBUTTON:
case BS_DEFPUSHBUTTON:
{
PWSTR pName;
int cch = GetWindowTextLength(pbutn->ci.hwnd);
pName = _alloca((cch + 1) * sizeof(wchar_t));
if (pName)
{
RECT rcText={0};
RECT rcIcon={0};
int cx = 0, cy = 0;
int iStateId = 0;
int iPartId = 0;
GetWindowText(pbutn->ci.hwnd, pName, cch + 1);
if (Button_IsThemed(pbutn))
{
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
// First: Get the text rect
GetThemeTextExtent(pbutn->hTheme, hdc, iPartId, iStateId, pName, cch, 0, &rcText, &rcText);
ApplyMarginsToRect(&pbutn->rcText, &rcText);
rc = rcText;
// We should now have The button with text.
}
else
{
int cxWidth = 2 * GetSystemMetrics(SM_CXEDGE);
int cyWidth = 3 * GetSystemMetrics(SM_CYEDGE);
if (IsPushButton(pbutn) & PBF_DEFAULT)
{
cxWidth += 2 * GetSystemMetrics(SM_CXBORDER);
cyWidth += 2 * GetSystemMetrics(SM_CXBORDER);
}
DrawText(hdc, pName, cch, &rcText, DT_CALCRECT);
ApplyMarginsToRect(&pbutn->rcText, &rcText);
rcText.bottom += cyWidth + 1; // +1 because draw text adds a single pixel to the first char, but not the last...
rcText.right += cxWidth + 1;
}
if (pbutn->himl)
{
rc.top = rc.left = 0; // We turn this into a width not a position
CCGetIconSize(&pbutn->ci, pbutn->himl, &cx, &cy);
rcIcon.bottom = cy;
rcIcon.right = cx;
ApplyMarginsToRect(&pbutn->rcIcon, &rcIcon);
switch (pbutn->uAlign)
{
case BUTTON_IMAGELIST_ALIGN_TOP:
case BUTTON_IMAGELIST_ALIGN_BOTTOM:
rc.bottom = RECTHEIGHT(rcIcon) + RECTHEIGHT(rcText);
rc.right = max(RECTWIDTH(rcIcon), RECTWIDTH(rcText));
break;
case BUTTON_IMAGELIST_ALIGN_CENTER: // This means no text
rc.bottom = RECTHEIGHT(rcIcon);
rc.right = RECTWIDTH(rcIcon);
break;
case BUTTON_IMAGELIST_ALIGN_RIGHT:
case BUTTON_IMAGELIST_ALIGN_LEFT:
// Fall
default:
rc.right = RECTWIDTH(rcIcon) + RECTWIDTH(rcText);
rc.bottom = max(RECTHEIGHT(rcIcon), RECTHEIGHT(rcText));
break;
}
}
else
{
rc = rcText;
}
if (Button_IsThemed(pbutn))
{
GetThemeBackgroundExtent(pbutn->hTheme, hdc, iPartId, iStateId, &rc, &rc);
}
}
}
break;
}
//
// Release the font which may have been loaded by ButtonInitDC.
//
if (pbutn->hFont)
{
SelectObject(hdc, GetStockObject(SYSTEM_FONT));
}
ReleaseDC(pbutn->ci.hwnd, hdc);
}
psize->cx = RECTWIDTH(rc);
psize->cy = RECTHEIGHT(rc);
return TRUE;
}
//---------------------------------------------------------------------------//
//
VOID Button_Paint(PBUTN pbutn, HDC hdc)
{
RECT rc;
RECT rcText;
HBRUSH hBrush;
HBRUSH hBrushSave = NULL;
BOOL fDrawBackground = TRUE;
ULONG ulStyle = GET_STYLE(pbutn);
CCDBUFFER db = {0};
UINT bsWnd = GetButtonType(ulStyle);
UINT pbfPush = IsPushButton(pbutn);
BOOL fTransparent = FALSE;
int iPartId = 0;
int iStateId = 0;
GetClientRect(pbutn->ci.hwnd, &rc);
if (Button_IsThemed(pbutn) &&
(bsWnd != LOBYTE(BS_GROUPBOX)) &&
(bsWnd != LOBYTE(BS_OWNERDRAW)) &&
!pbutn->fPaintKbdCuesOnly)
{
hdc = CCBeginDoubleBuffer(hdc, &rc, &db);
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
fTransparent = CCShouldAskForBits(&pbutn->ci, pbutn->hTheme, iPartId, iStateId);
if (fTransparent)
{
fDrawBackground = (TRUE != CCSendPrint(&pbutn->ci, hdc));
}
}
hBrush = Button_InitDC(pbutn, hdc);
if ((!pbfPush || fTransparent) && !pbutn->fPaintKbdCuesOnly &&
fDrawBackground)
{
if ((bsWnd != LOBYTE(BS_OWNERDRAW)) &&
(bsWnd != LOBYTE(BS_GROUPBOX)))
{
//
// Fill the client area with the background brush
// before we begin painting.
//
FillRect(hdc, &rc, hBrush);
}
hBrushSave = SelectObject(hdc, hBrush);
}
switch (bsWnd)
{
case BS_CHECKBOX:
case BS_RADIOBUTTON:
case BS_AUTORADIOBUTTON:
case BS_3STATE:
case BS_AUTOCHECKBOX:
case BS_AUTO3STATE:
if (!pbfPush)
{
if (!Button_IsThemed(pbutn))
{
Button_DrawText(pbutn, hdc,
DBT_TEXT | (BUTTONSTATE(pbutn) & BST_FOCUS ? DBT_FOCUS : 0), FALSE);
}
if (!pbutn->fPaintKbdCuesOnly || Button_IsThemed(pbutn))
{
Button_DrawCheck(pbutn, hdc, hBrush);
}
break;
}
//
// Fall through for PUSHLIKE buttons
//
case BS_PUSHBUTTON:
case BS_DEFPUSHBUTTON:
Button_DrawPush(pbutn, hdc, pbfPush);
break;
case BS_PUSHBOX:
Button_DrawText(pbutn, hdc,
DBT_TEXT | (BUTTONSTATE(pbutn) & BST_FOCUS ? DBT_FOCUS : 0), FALSE);
Button_DrawNewState(pbutn, hdc, hBrush, 0);
break;
case BS_USERBUTTON:
// Don't support USERBUTTON in v6. This has been superceded by OWNERDRAW in win32.
break;
case BS_OWNERDRAW:
Button_OwnerDraw(pbutn, hdc, ODA_DRAWENTIRE);
break;
case BS_GROUPBOX:
Button_CalcRect(pbutn, hdc, &rcText, CBR_GROUPTEXT, 0);
//----- get theme part, state for groupbox ----
if (Button_IsThemed(pbutn))
{
Button_GetThemeIds(pbutn, &iPartId, &iStateId);
}
if (!pbutn->fPaintKbdCuesOnly)
{
UINT uFlags;
BOOL fFillMyself = TRUE;
Button_CalcRect(pbutn, hdc, &rc, CBR_GROUPFRAME, 0);
uFlags = ((ulStyle & BS_FLAT) != 0) ? BF_FLAT | BF_MONO : 0;
if (!Button_IsThemed(pbutn))
{
DrawEdge(hdc, &rc, EDGE_ETCHED, BF_RECT | uFlags);
}
else
{
DrawThemeBackground(pbutn->hTheme, hdc, iPartId, iStateId, &rc, 0);
fFillMyself = (FALSE == CCSendPrintRect(&pbutn->ci, hdc, &rcText));
}
if (fFillMyself)
{
FillRect(hdc, &rcText, hBrush);
}
}
// FillRect(hdc, &rc, hBrush);
if (!Button_IsThemed(pbutn))
{
Button_DrawText(pbutn, hdc, DBT_TEXT, FALSE);
}
else
{
DWORD dwTextFlags;
HRESULT hr;
int cch;
LPWSTR lpName;
cch = GetWindowTextLength(pbutn->ci.hwnd);
lpName = UserLocalAlloc(0, sizeof(WCHAR)*(cch+1));
if ( !lpName )
{
TraceMsg(TF_STANDARD, "Button_Paint couldn't allocate buffer");
return;
}
GetWindowTextW(pbutn->ci.hwnd, lpName, cch+1);
dwTextFlags = Button_GetTextFlags(pbutn);
//
// Button_CalcRect padded by a CXEDGE so the groupbox frame wouldn't
// be flush with the Group text
//
rcText.left += GetSystemMetrics(SM_CXEDGE);
hr = DrawThemeText(pbutn->hTheme,
hdc,
iPartId,
iStateId,
lpName,
cch,
dwTextFlags,
0,
&rcText);
if (FAILED(hr))
{
TraceMsg(TF_STANDARD, "Button_Paint failed to render groupbox text");
}
UserLocalFree(lpName);
}
break;
}
if (!pbfPush && hBrushSave)
{
SelectObject(hdc, hBrushSave);
}
//
// Release the font which may have been loaded by ButtonInitDC.
//
if (pbutn->hFont)
{
SelectObject(hdc, GetStockObject(SYSTEM_FONT));
}
CCEndDoubleBuffer(&db);
}
//---------------------------------------------------------------------------//
//
VOID Button_Repaint(PBUTN pbutn)
{
HDC hdc = Button_GetDC(pbutn, NULL);
if (hdc != NULL)
{
Button_Paint(pbutn, hdc);
Button_ReleaseDC(pbutn, hdc, NULL);
}
}
VOID Button_SetHot(PBUTN pbutn, BOOL fHot, DWORD dwReason)
{
NMBCHOTITEM nmhot = {0};
// Send a notification about the hot item change
if (fHot)
{
nmhot.dwFlags = HICF_ENTERING;
pbutn->buttonState |= BST_HOT;
}
else
{
nmhot.dwFlags = HICF_LEAVING;
pbutn->buttonState &= ~BST_HOT;
}
nmhot.dwFlags |= dwReason;
CCSendNotify(&pbutn->ci, BCN_HOTITEMCHANGE, &nmhot.hdr);
}
void Button_EraseOwnerDraw(PBUTN pbutn, HDC hdc)
{
if (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_OWNERDRAW))
{
RECT rc;
HBRUSH hbr;
//
// Handle erase background for owner draw buttons.
//
GetClientRect(pbutn->ci.hwnd, &rc);
hbr = (HBRUSH)SendMessage(GetParent(pbutn->ci.hwnd), WM_CTLCOLORBTN, (WPARAM)hdc, (LPARAM)pbutn->ci.hwnd);
FillRect(hdc, &rc, hbr);
}
}
//---------------------------------------------------------------------------//
//
// Button_WndProc
//
// WndProc for buttons, check boxes, etc.
//
LRESULT APIENTRY Button_WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
UINT wOldState;
RECT rc;
HDC hdc;
HBRUSH hbr;
PAINTSTRUCT ps;
PBUTN pbutn;
LRESULT lResult = FALSE;
//
// Get the instance data for this button control
//
pbutn = Button_GetPtr(hwnd);
if (!pbutn && uMsg != WM_NCCREATE)
{
goto CallDWP;
}
switch (uMsg)
{
case WM_NCHITTEST:
if (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_GROUPBOX))
{
lResult = (LONG)HTTRANSPARENT;
}
else
{
lResult = DefWindowProc(hwnd, uMsg, wParam, lParam);
if ( lResult == HTCLIENT && Button_IsThemed(pbutn))
{
HRESULT hr;
int iPartId = 0;
int iStateId = 0;
POINT pt;
WORD wHitTestCode;
hr = Button_GetThemeIds(pbutn, &iPartId, &iStateId);
if ( SUCCEEDED(hr) )
GetWindowRect(pbutn->ci.hwnd, &rc);
pt.x = GET_X_LPARAM(lParam);
pt.y = GET_Y_LPARAM(lParam);
hr = HitTestThemeBackground(pbutn->hTheme,
NULL,
iPartId,
iStateId,
0,
&rc,
NULL,
pt,
&wHitTestCode);
if ( SUCCEEDED(hr) && wHitTestCode == HTTRANSPARENT)
{
lResult = (LRESULT)HTTRANSPARENT;
}
}
}
break;
case WM_ERASEBKGND:
Button_EraseOwnerDraw(pbutn, (HDC)wParam);
//
// Do nothing for other buttons, but don't let DefWndProc() do it
// either. It will be erased in Button_Paint().
//
lResult = (LONG)TRUE;
break;
case WM_PRINTCLIENT:
Button_EraseOwnerDraw(pbutn, (HDC)wParam);
Button_Paint(pbutn, (HDC)wParam);
break;
case WM_CREATE:
pbutn->hTheme = Button_GetTheme(pbutn);
CIInitialize(&pbutn->ci, hwnd, (LPCREATESTRUCT)lParam);
SendMessage(hwnd, WM_CHANGEUISTATE, MAKEWPARAM(UIS_INITIALIZE, 0), 0);
break;
case WM_PAINT:
{
//
// If wParam != NULL, then this is a subclassed paint.
//
if (wParam)
{
hdc = (HDC)wParam;
}
else
{
hdc = BeginPaint(hwnd, &ps);
}
if (IsWindowVisible(pbutn->ci.hwnd))
{
Button_Paint(pbutn, hdc);
}
if (!wParam)
{
EndPaint(hwnd, &ps);
}
}
break;
case WM_SETFOCUS:
BUTTONSTATE(pbutn) |= BST_FOCUS;
if (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_OWNERDRAW))
{
HDC hdc = Button_GetDC(pbutn, NULL);
if (hdc)
{
Button_DrawText(pbutn, hdc, DBT_FOCUS, FALSE);
Button_ReleaseDC(pbutn, hdc, NULL);
}
}
else
{
InvalidateRect(pbutn->ci.hwnd, NULL, FALSE);
}
if ((GET_STYLE(pbutn)& BS_NOTIFY) != 0)
{
Button_NotifyParent(pbutn, BN_SETFOCUS);
}
if (!(BUTTONSTATE(pbutn) & BST_INCLICK))
{
switch (GetButtonType(GET_STYLE(pbutn)))
{
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_AUTORADIOBUTTON):
if (!(BUTTONSTATE(pbutn) & BST_DONTCLICK))
{
if (!(BUTTONSTATE(pbutn) & BST_CHECKMASK))
{
Button_NotifyParent(pbutn, BN_CLICKED);
}
}
break;
}
}
break;
case WM_GETDLGCODE:
lResult = DLGC_BUTTON;
switch (GetButtonType(GET_STYLE(pbutn)))
{
case LOBYTE(BS_DEFPUSHBUTTON):
lResult |= DLGC_DEFPUSHBUTTON;
break;
case LOBYTE(BS_PUSHBUTTON):
case LOBYTE(BS_PUSHBOX):
lResult |= DLGC_UNDEFPUSHBUTTON;
break;
case LOBYTE(BS_AUTORADIOBUTTON):
case LOBYTE(BS_RADIOBUTTON):
lResult |= DLGC_RADIOBUTTON;
break;
case LOBYTE(BS_GROUPBOX):
//
// remove DLGC_BUTTON
//
lResult = DLGC_STATIC;
break;
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('-'):
lResult |= DLGC_WANTCHARS;
break;
}
}
break;
}
break;
case WM_CAPTURECHANGED:
if (BUTTONSTATE(pbutn) & BST_CAPTURED)
{
//
// Unwittingly, we've been kicked out of capture,
// so undepress etc.
//
if (BUTTONSTATE(pbutn) & BST_MOUSE)
{
SendMessage(pbutn->ci.hwnd, BM_SETSTATE, FALSE, 0);
}
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.
//
SendMessage(pbutn->ci.hwnd, BM_SETSTATE, FALSE, 0);
}
Button_ReleaseCapture(pbutn, TRUE);
BUTTONSTATE(pbutn) &= ~BST_FOCUS;
if ((GET_STYLE(pbutn) & BS_NOTIFY) != 0)
{
Button_NotifyParent(pbutn, 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.
//
InvalidateRect(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 (GetButtonType(GET_STYLE(pbutn)))
{
default:
if ((GET_STYLE(pbutn) & BS_NOTIFY) == 0)
goto btnclick;
case LOBYTE(BS_USERBUTTON):
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_OWNERDRAW):
Button_NotifyParent(pbutn, BN_DOUBLECLICKED);
break;
}
break;
case WM_LBUTTONUP:
if (BUTTONSTATE(pbutn) & BST_MOUSE)
{
Button_ReleaseCapture(pbutn, TRUE);
}
break;
case WM_MOUSELEAVE:
{
//
// We should only be requesting mouseleave messages
// if we are themed but check anyway
//
if (pbutn->buttonState & BST_HOT)
{
Button_SetHot(pbutn, FALSE, HICF_MOUSE);
InvalidateRect(pbutn->ci.hwnd, NULL, TRUE);
}
}
break;
case WM_MOUSEMOVE:
{
//
// If the hot bit is not already set
//
// 300925: Can't hottrack ownerdraw buttons for app compat reasons
//
if (!TESTFLAG(pbutn->buttonState, BST_HOT) &&
GetButtonType(GET_STYLE(pbutn)) != LOBYTE(BS_OWNERDRAW))
{
TRACKMOUSEEVENT tme;
//
// Set the hot bit and request that
// we be notified when the mouse leaves
//
Button_SetHot(pbutn, TRUE, HICF_MOUSE);
tme.cbSize = sizeof(tme);
tme.dwFlags = TME_LEAVE;
tme.hwndTrack = pbutn->ci.hwnd;
tme.dwHoverTime = 0;
TrackMouseEvent(&tme);
InvalidateRect(pbutn->ci.hwnd, NULL, TRUE);
}
if (!(BUTTONSTATE(pbutn) & BST_MOUSE))
{
break;
}
}
//
// FALL THRU
//
case WM_LBUTTONDOWN:
btnclick:
if (Button_SetCapture(pbutn, BST_MOUSE))
{
POINT pt;
GetClientRect(pbutn->ci.hwnd, &rc);
POINTSTOPOINT(pt, lParam);
SendMessage(pbutn->ci.hwnd, BM_SETSTATE, PtInRect(&rc, pt), 0);
}
break;
case WM_CHAR:
if (BUTTONSTATE(pbutn) & BST_MOUSE)
{
goto CallDWP;
}
if (GetButtonType(GET_STYLE(pbutn)) != LOBYTE(BS_CHECKBOX) &&
GetButtonType(GET_STYLE(pbutn)) != LOBYTE(BS_AUTOCHECKBOX))
{
goto CallDWP;
}
switch (wParam)
{
case TEXT('+'):
case TEXT('='):
//
// we must Set the check mark on.
//
wParam = 1;
goto SetCheck;
case TEXT('-'):
//
// Set the check mark off.
//
wParam = 0;
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 (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_AUTOCHECKBOX))
{
if (Button_SetCapture(pbutn, 0))
{
SendMessage(pbutn->ci.hwnd, BM_SETCHECK, wParam, 0);
Button_ReleaseCapture(pbutn, TRUE);
}
}
Button_NotifyParent(pbutn, BN_CLICKED);
}
break;
default:
goto CallDWP;
}
break;
case BCM_GETIDEALSIZE:
return Button_OnGetIdealSize(pbutn, (PSIZE)lParam);
case BCM_SETIMAGELIST:
return Button_OnSetImageList(pbutn, (BUTTON_IMAGELIST*)lParam);
case BCM_GETIMAGELIST:
{
BUTTON_IMAGELIST* biml = (BUTTON_IMAGELIST*)lParam;
if (biml)
{
biml->himl = pbutn->himl;
biml->uAlign = pbutn->uAlign;
biml->margin = pbutn->rcIcon;
return TRUE;
}
}
break;
case BCM_SETTEXTMARGIN:
{
RECT* prc = (RECT*)lParam;
if (prc)
{
pbutn->rcText = *prc;
return TRUE;
}
}
break;
case BCM_GETTEXTMARGIN:
{
RECT* prc = (RECT*)lParam;
if (prc)
{
*prc = pbutn->rcText;
return TRUE;
}
}
break;
case BM_CLICK:
//
// Don't recurse into this code!
//
if (BUTTONSTATE(pbutn) & BST_INBMCLICK)
{
break;
}
BUTTONSTATE(pbutn) |= BST_INBMCLICK;
SendMessage(pbutn->ci.hwnd, WM_LBUTTONDOWN, 0, 0);
SendMessage(pbutn->ci.hwnd, WM_LBUTTONUP, 0, 0);
BUTTONSTATE(pbutn) &= ~BST_INBMCLICK;
//
// FALL THRU
//
case WM_KEYDOWN:
if (BUTTONSTATE(pbutn) & BST_MOUSE)
{
break;
}
if (wParam == VK_SPACE)
{
if (Button_SetCapture(pbutn, 0))
{
SendMessage(pbutn->ci.hwnd, BM_SETSTATE, TRUE, 0);
}
}
else
{
Button_ReleaseCapture(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: pbutn->ci.hwnd is history after this call!
//
Button_ReleaseCapture(pbutn, (wParam == VK_SPACE));
if (uMsg == WM_SYSKEYUP)
{
goto CallDWP;
}
break;
case BM_GETSTATE:
lResult = (LONG)BUTTONSTATE(pbutn);
break;
case BM_SETSTATE:
wOldState = (UINT)(BUTTONSTATE(pbutn) & BST_PUSHED);
if (wParam)
{
BUTTONSTATE(pbutn) |= BST_PUSHED;
}
else
{
BUTTONSTATE(pbutn) &= ~BST_PUSHED;
}
if (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_USERBUTTON))
{
Button_NotifyParent(pbutn, (UINT)(wParam ? BN_PUSHED : BN_UNPUSHED));
}
if (wOldState != (BOOL)(BUTTONSTATE(pbutn) & BST_PUSHED))
{
// Only invalidate if the state changed.
InvalidateRect(pbutn->ci.hwnd, NULL, FALSE);
NotifyWinEvent(EVENT_OBJECT_STATECHANGE, hwnd, OBJID_CLIENT, INDEXID_CONTAINER);
}
break;
case BM_GETCHECK:
lResult = (LONG)(BUTTONSTATE(pbutn) & BST_CHECKMASK);
break;
case BM_SETCHECK:
switch (GetButtonType(GET_STYLE(pbutn)))
{
case LOBYTE(BS_RADIOBUTTON):
case LOBYTE(BS_AUTORADIOBUTTON):
if (wParam)
{
SetWindowState(pbutn->ci.hwnd, WS_TABSTOP);
}
else
{
ClearWindowState(pbutn->ci.hwnd, WS_TABSTOP);
}
//
// 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 (!IsWindowVisible(pbutn->ci.hwnd))
{
break;
}
InvalidateRect(pbutn->ci.hwnd, NULL, FALSE);
NotifyWinEvent(EVENT_OBJECT_STATECHANGE, hwnd, OBJID_CLIENT, INDEXID_CONTAINER);
}
break;
}
break;
case BM_SETSTYLE:
AlterWindowStyle(hwnd, BS_TYPEMASK, (DWORD)wParam);
if (lParam)
{
InvalidateRect(hwnd, NULL, TRUE);
}
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 (GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_GROUPBOX))
{
hdc = Button_GetDC(pbutn, &hbr);
if (hdc != NULL)
{
Button_CalcRect(pbutn, hdc, &rc, CBR_GROUPTEXT, 0);
InvalidateRect(hwnd, &rc, TRUE);
FillRect(hdc, &rc, hbr);
Button_ReleaseDC(pbutn, hdc, &hbr);
}
}
lResult = DefWindowProc(hwnd, uMsg, wParam, lParam);
NotifyWinEvent(EVENT_OBJECT_NAMECHANGE, hwnd, OBJID_WINDOW, INDEXID_CONTAINER);
goto DoEnable;
case WM_ENABLE:
lResult = 0L;
DoEnable:
Button_Repaint(pbutn);
break;
case WM_SETFONT:
//
// wParam - handle to the font
// lParam - if true, redraw else don't
//
Button_SetFont(pbutn, (HFONT)wParam, (BOOL)(lParam != 0));
break;
case WM_GETFONT:
lResult = (LRESULT)pbutn->hFont;
break;
case BM_GETIMAGE:
case BM_SETIMAGE:
if (!IsValidImage(wParam, (GET_STYLE(pbutn) & BS_IMAGEMASK) != 0, IMAGE_BMMAX))
{
TraceMsg(TF_STANDARD, "UxButton: Invalid button image type");
SetLastError(ERROR_INVALID_PARAMETER);
}
else
{
HANDLE hOld = pbutn->hImage;
if (uMsg == BM_SETIMAGE)
{
pbutn->hImage = (HANDLE)lParam;
if (IsWindowVisible(pbutn->ci.hwnd))
{
InvalidateRect(hwnd, NULL, TRUE);
}
}
lResult = (LRESULT)hOld;
}
break;
case WM_NCDESTROY:
if (pbutn->hTheme)
{
CloseThemeData(pbutn->hTheme);
}
UserLocalFree(pbutn);
TraceMsg(TF_STANDARD, "BUTTON: Clearing button instance pointer.");
Button_SetPtr(hwnd, NULL);
break;
case WM_NCCREATE:
pbutn = (PBUTN)UserLocalAlloc(HEAP_ZERO_MEMORY, sizeof(BUTN));
if (pbutn)
{
//
// Success... store the instance pointer.
//
TraceMsg(TF_STANDARD, "BUTTON: Setting button instance pointer.");
Button_SetPtr(hwnd, pbutn);
pbutn->ci.hwnd = hwnd;
pbutn->pww = (PWW)GetWindowLongPtr(hwnd, GWLP_WOWWORDS);
SetRect(&pbutn->rcText, GetSystemMetrics(SM_CXEDGE) / 2, GetSystemMetrics(SM_CYEDGE) / 2,
GetSystemMetrics(SM_CXEDGE) / 2, GetSystemMetrics(SM_CYEDGE) / 2);
//
// 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 (TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN31COMPAT))
{
if ((!TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT) &&
(((LOBYTE(GET_STYLE(pbutn))) & (LOBYTE(~BS_LEFTTEXT))) == LOBYTE(BS_USERBUTTON))) ||
(TESTFLAG(GET_STATE2(pbutn), WS_S2_WIN40COMPAT) &&
(GetButtonType(GET_STYLE(pbutn)) == LOBYTE(BS_USERBUTTON))))
{
//
// BS_USERBUTTON is no longer allowed for 3.1 and beyond.
// Just turn to normal push button.
//
AlterWindowStyle(hwnd, BS_TYPEMASK, 0);
TraceMsg(TF_STANDARD, "UxButton: BS_USERBUTTON no longer supported");
}
}
if ((GET_EXSTYLE(pbutn) & WS_EX_RIGHT) != 0)
{
AlterWindowStyle(hwnd, BS_RIGHT | BS_RIGHTBUTTON, BS_RIGHT | BS_RIGHTBUTTON);
}
goto CallDWP;
}
else
{
//
// Failed... return FALSE.
//
// From a WM_NCCREATE msg, this will cause the
// CreateWindow call to fail.
//
TraceMsg(TF_STANDARD, "BUTTON: Unable to allocate button instance structure.");
lResult = FALSE;
}
break;
case WM_INPUTLANGCHANGEREQUEST:
//
// #115190
// If the window is one of controls on top of dialogbox,
// let the parent dialog handle it.
//
#if 0 // Need to expose TestwndChild()
if (TestwndChild(pbutn->ci.hwnd) && pbutn->ci.hwnd->spbutn->ci.hwndParent)
{
PWND pbutn->ci.hwndParent = REBASEPWND(pbutn->ci.hwnd, spbutn->ci.hwndParent);
if (pbutn->ci.hwndParent)
{
PCLS pclsParent = REBASEALWAYS(pbutn->ci.hwndParent, pcls);
UserAssert(pclsParent != NULL);
if (pclsParent->atomClassName == gpsi->atomSysClass[ICLS_DIALOG])
{
TraceMsg(TF_STANDARD, "UxButton: WM_INPUTLANGCHANGEREQUEST is sent to parent.");
return SendMessage(pbutn->ci.hwndParent, uMsg, wParam, lParam);
}
}
}
#endif
goto CallDWP;
case WM_UPDATEUISTATE:
DefWindowProc(hwnd, uMsg, wParam, lParam);
if (ISBSTEXTOROD(GET_STYLE(pbutn)))
{
pbutn->fPaintKbdCuesOnly = !IsUsingCleartype();
Button_Repaint(pbutn);
pbutn->fPaintKbdCuesOnly = FALSE;
}
break;
case WM_GETOBJECT:
if(lParam == OBJID_QUERYCLASSNAMEIDX)
{
lResult = MSAA_CLASSNAMEIDX_BUTTON;
}
else
{
lResult = FALSE;
}
break;
case WM_THEMECHANGED:
if ( pbutn->hTheme )
{
CloseThemeData(pbutn->hTheme);
}
//---- reset cached sizes that may change with a theme change ----
sizeCheckBox.cx = 0;
sizeCheckBox.cy = 0;
sizeRadioBox.cx = 0;
sizeRadioBox.cy = 0;
pbutn->hTheme = Button_GetTheme(pbutn);
InvalidateRect(pbutn->ci.hwnd, NULL, TRUE);
CCSendNotify(&pbutn->ci, NM_THEMECHANGED, NULL);
lResult = TRUE;
break;
default:
if (CCWndProc(&pbutn->ci, uMsg, wParam, lParam, &lResult))
return lResult;
CallDWP:
lResult = DefWindowProc(hwnd, uMsg, wParam, lParam);
}
return lResult;
}