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

616 lines
15 KiB
C

/****************************** Module Header ******************************\
* Module Name: mngray.c
* Copyright (c) 1985 - 1999, Microsoft Corporation
* This module contains the DrawState API
* History:
* 01-05-94 FritzS Ported from Chicago
*/
#include "precomp.h"
#pragma hdrstop
#define PATOR 0x00FA0089L
#define SRCSTENCIL 0x00B8074AL
#define SRCINVSTENCIL 0x00E20746L
#define BC_INVERT 0x00000001
void BltColor(HDC hdc, HBRUSH hbr, HDC hdcSrce,int xO, int yO,
int cx, int cy, int xO1, int yO1, UINT uBltFlags);
/*
* BitBltSysBmp()
* From Chicago -- client *only* for now.
*/
BOOL FAR BitBltSysBmp(HDC hdc, int x, int y, UINT i)
{
POEMBITMAPINFO pOem = gpsi->oembmi + i;
return(NtUserBitBltSysBmp(hdc, x, y, pOem->cx, pOem->cy, pOem->x, pOem->y, SRCCOPY));
}
/*
* DrawState()
* Generic state drawing routine. Does simple drawing into same DC if
* normal state; uses offscreen bitmap otherwise.
* We do drawing for these simple types ourselves:
* (1) Text
* lData is string pointer.
* wData is string length
* (2) Icon
* LOWORD(lData) is hIcon
* (3) Bitmap
* LOWORD(lData) is hBitmap
* (4) Glyph (internal)
* LOWORD(lData) is OBI_ value, one of
* OBI_CHECKMARK
* OBI_BULLET
* OBI_MENUARROW
* right now
* Other types are required to draw via the callback function, and are
* allowed to stick whatever they want in lData and wData.
* We apply the following effects onto the image:
* (1) Normal (nothing)
* (2) Default (drop shadow)
* (3) Union (gray string dither)
* (4) Disabled (embossed)
* Note that we do NOT stretch anything. We just clip.
* FritzS note -- this is client-side *only*. Similar code is in server\mngray.c
*/
BOOL DrawStateW(
HDC hdcDraw,
HBRUSH hbrFore,
DRAWSTATEPROC qfnCallBack,
LPARAM lData,
WPARAM wData,
int x,
int y,
int cx,
int cy,
UINT uFlags)
{
HFONT hFont;
HFONT hFontSave = NULL;
HDC hdcT;
HBITMAP hbmpT;
BOOL fResult = FALSE;
DWORD dwPSMFlags;
POINT ptOrg;
int oldAlign;
#ifdef USE_MIRRORING
DWORD dwLayout=0;
#endif
if (ghdcGray == NULL)
return FALSE;
RtlEnterCriticalSection(&gcsHdc);
/*
* These require monochrome conversion
* Enforce monochrome: embossed doesn't look great with 2 color displays
*/
if ((uFlags & DSS_DISABLED) &&
(gpsi->BitCount == 1 || SYSMET(SLOWMACHINE))) {
uFlags &= ~DSS_DISABLED;
uFlags |= DSS_UNION;
}
if (uFlags & (DSS_DISABLED | DSS_DEFAULT | DSS_UNION))
uFlags |= DSS_MONO;
/*
* Get drawing sizes etc. AND VALIDATE.
*/
switch (uFlags & DST_TYPEMASK) {
case DST_GLYPH:
/*
* LOWORD(lData) is OBI_ value.
*/
if (LOWORD(lData) >= (WORD)OBI_COUNT) {
goto CDS_Leave;
}
if (!cx) {
cx = gpsi->oembmi[LOWORD(lData)].cx;
}
if (!cy) {
cy = gpsi->oembmi[LOWORD(lData)].cy;
}
break;
case DST_BITMAP:
/*
* LOWORD(lData) is hbmp.
*/
if (GetObjectType((HGDIOBJ)lData) != OBJ_BITMAP) {
goto CDS_Leave;
}
if (!cx || !cy) {
BITMAP bmp;
GetObjectW((HGDIOBJ)lData, sizeof(BITMAP), &bmp);
if (!cx)
cx = bmp.bmWidth;
if (!cy)
cy = bmp.bmHeight;
}
break;
case DST_ICON:
/*
* lData is hicon.
*/
if (!cx || !cy) {
int cx1 = 0;
int cy1 = 0;
NtUserGetIconSize((HICON)lData, 0, &cx1, &cy1);
if (!cx)
cx = cx1;
if (!cy)
cy = cy1 / 2; // icons are double height in NT
}
break;
case DST_TEXT:
/*
* lData is LPSTR
* NOTE THAT WE DO NOT VALIDATE lData, DUE TO COMPATIBILITY
* WITH GRAYSTRING(). THIS _SHOULD_ FAULT IF YOU PASS IN NULL.
* wData is cch.
*/
if (!wData)
wData = wcslen((LPWSTR)lData);
if (!cx || !cy) {
SIZE size;
/*
* Make sure we use right dc w/ right font.
*/
GetTextExtentPointW(hdcDraw, (LPWSTR)lData, (INT)wData, &size);
if (!cx)
cx = size.cx;
if (!cy)
cy = size.cy;
}
/*
* Now, pretend we're complex if qfnCallBack is supplied AND
* we're supporting GrayString().
*/
#if 0 // This will get turned on if/when we change GrayString to tie
// into DrawState.
// FritzS
if ((uFlags & DST_GRAYSTRING) && SELECTOROF(qfnCallBack)) {
uFlags &= ~DST_TYPEMASK;
uFlags |= DST_COMPLEX;
}
#endif
break;
case DST_PREFIXTEXT:
if (lData == 0) {
RIPMSG0(RIP_ERROR, "DrawState: NULL DST_PREFIXTEXT string");
goto CDS_Leave;
}
if (!wData)
wData = wcslen((LPWSTR)lData);
if (!cx || !cy) {
SIZE size;
PSMGetTextExtent(hdcDraw, (LPWSTR)lData, (int)wData, &size);
if (!cx)
cx = size.cx;
if (!cy)
cy = size.cy;
}
/*
* Add on height for prefix
*/
cy += (2 * SYSMET(CYBORDER));
break;
case DST_COMPLEX:
#if 0
if (!SELECTOROF(qfnCallBack)) {
DebugErr(DBF_ERROR, "DrawState: invalid callback for DST_COMPLEX");
goto CDS_Leave;
}
#endif
break;
default:
RIPMSG0(RIP_ERROR, "DrawState: invalid DST_ type");
goto CDS_Leave;
}
/*
* Optimize: nothing to draw
* Have to call callback if GRAYSTRING for compatibility.
*/
if ((!cx || !cy)
// && !(uFlags & DST_GRAYSTRING)
) {
fResult = TRUE;
goto CDS_Leave;
}
/*
* Setup drawing dc
*/
if (uFlags & DSS_MONO) {
hdcT = ghdcGray;
#ifdef USE_MIRRORING
/*
* First turn off mirroring on hdcGray if any.
*/
SetLayoutWidth(hdcT, -1, 0);
/*
* Set the ghdcGray layout to be equal to the screen hdcDraw layout.
*/
dwLayout = GetLayout(hdcDraw);
if (dwLayout != GDI_ERROR) {
SetLayoutWidth(hdcT, cx, dwLayout);
}
#endif
/*
* Is our scratch bitmap big enough? We need potentially
* cx+1 by cy pixels for default etc.
*/
if ((gcxGray < cx + 1) || (gcyGray < cy)) {
if (hbmpT = CreateBitmap(max(gcxGray, cx + 1), max(gcyGray, cy), 1, 1, 0L)) {
HBITMAP hbmGray;
hbmGray = SelectObject(ghdcGray, hbmpT);
DeleteObject(hbmGray);
gcxGray = max(gcxGray, cx + 1);
gcyGray = max(gcyGray, cy);
} else {
cx = gcxGray - 1;
cy = gcyGray;
}
}
PatBlt(ghdcGray, 0, 0, gcxGray, gcyGray, WHITENESS);
SetTextCharacterExtra(ghdcGray, GetTextCharacterExtra(hdcDraw));
oldAlign = GetTextAlign(hdcT);
SetTextAlign(hdcT, (oldAlign & ~(TA_RTLREADING |TA_CENTER |TA_RIGHT))
| (GetTextAlign(hdcDraw) & (TA_RTLREADING |TA_CENTER |TA_RIGHT)));
/*
* Setup font
*/
if ((uFlags & DST_TYPEMASK) <= DST_TEXTMAX) {
if (GetCurrentObject(hdcDraw, OBJ_FONT) != ghFontSys) {
hFont = SelectObject(hdcDraw, ghFontSys);
SelectObject(hdcDraw, hFont);
hFontSave = SelectObject(ghdcGray, hFont);
}
}
} else {
hdcT = hdcDraw;
/*
* Adjust viewport
*/
GetViewportOrgEx(hdcT, &ptOrg);
SetViewportOrgEx(hdcT, ptOrg.x + x, ptOrg.y + y, NULL);
}
/*
* Now, draw original image
*/
fResult = TRUE;
switch (uFlags & DST_TYPEMASK) {
case DST_GLYPH:
/*
* Blt w/ current brush in hdcT
*/
BitBltSysBmp(hdcT, 0, 0, LOWORD(lData));
break;
case DST_BITMAP:
/*
* Draw the bitmap. If mono, it'll use the colors set up
* in the dc.
*/
// RtlEnterCriticalSection(&gcsHdcBits2);
UserAssert(GetBkColor(ghdcBits2) == RGB(255, 255, 255));
UserAssert(GetTextColor(ghdcBits2) == RGB(0, 0, 0));
hbmpT = SelectObject(ghdcBits2, (HBITMAP)lData);
BitBlt(hdcT, 0, 0, cx, cy, ghdcBits2, 0, 0, SRCCOPY);
SelectObject(ghdcBits2, hbmpT);
// RtlLeaveCriticalSection(&gcsHdcBits2);
break;
case DST_ICON:
/*
* Draw the icon.
*/
DrawIconEx(hdcT, 0, 0, (HICON)lData, 0, 0, 0, NULL, DI_NORMAL);
break;
case DST_PREFIXTEXT:
if (uFlags & DSS_HIDEPREFIX) {
dwPSMFlags = DT_HIDEPREFIX;
} else if (uFlags & DSS_PREFIXONLY) {
dwPSMFlags = DT_PREFIXONLY;
} else {
dwPSMFlags = 0;
}
PSMTextOut(hdcT, 0, 0, (LPWSTR)lData, (int)wData, dwPSMFlags);
break;
case DST_TEXT:
fResult = TextOutW(hdcT, 0, 0, (LPWSTR)lData, (int)wData);
break;
default:
fResult = (qfnCallBack)(hdcT, lData, wData, cx, cy);
/*
* The callbacks could have altered the attributes of ghdcGray
*/
if (hdcT == ghdcGray) {
SetBkColor(ghdcGray, RGB(255, 255, 255));
SetTextColor(ghdcGray, RGB(0, 0, 0));
SelectObject(ghdcGray, GetStockObject(BLACK_BRUSH));
SetBkMode(ghdcGray, OPAQUE);
}
break;
}
/*
* Clean up
*/
if (uFlags & DSS_MONO) {
/*
* Reset font
*/
if (hFontSave)
SelectObject(hdcT, hFontSave);
SetTextAlign(hdcT, oldAlign);
} else {
/*
* Reset DC.
*/
SetViewportOrgEx(hdcT, ptOrg.x, ptOrg.y, NULL);
goto CDS_Leave;
}
/*
* UNION state
* Dither over image
* We want white pixels to stay white, in either dest or pattern.
*/
if (uFlags & DSS_UNION) {
POLYPATBLT PolyData;
PolyData.x = 0;
PolyData.y = 0;
PolyData.cx = cx;
PolyData.cx = cy;
PolyData.BrClr.hbr = gpsi->hbrGray;
PolyPatBlt(ghdcGray, PATOR, &PolyData, 1, PPB_BRUSH);
}
/*
* DISABLED state
* Emboss
* Draw over-1/down-1 in hilight color, and in same position in shadow.
* DEFAULT state
* Drop shadow
* Draw over-1/down-1 in shadow color, and in same position in foreground
* Draw offset down in shadow color,
*/
if (uFlags & DSS_DISABLED) {
BltColor(hdcDraw,
SYSHBR(3DHILIGHT),
ghdcGray,
x + 1,
y + 1,
cx,
cy,
0,
0,
BC_INVERT);
BltColor(hdcDraw,
SYSHBR(3DSHADOW),
ghdcGray,
x,
y,
cx,
cy,
0,
0,
BC_INVERT);
} else if (uFlags & DSS_DEFAULT) {
BltColor(hdcDraw,
SYSHBR(3DSHADOW),
ghdcGray,
x+1,
y+1,
cx,
cy,
0,
0,
BC_INVERT);
goto DrawNormal;
} else {
DrawNormal:
BltColor(hdcDraw, hbrFore, ghdcGray, x, y, cx, cy, 0, 0, BC_INVERT);
}
CDS_Leave:
#ifdef USE_MIRRORING
if (uFlags & DSS_MONO) {
/*
* Set the ghdcGray layout to 0, it is a public DC.
*/
SetLayoutWidth(hdcT, -1, 0);
}
#endif
RtlLeaveCriticalSection(&gcsHdc);
return fResult;
}
BOOL DrawStateA(HDC hDC, HBRUSH hBrush, DRAWSTATEPROC func,
LPARAM lParam, WPARAM wParam, int x, int y, int cx, int cy, UINT wFlags) {
LPARAM lpwstr = lParam;
BOOL bRet;
BOOL bFree;
if (((wFlags & DST_TYPEMASK) == DST_TEXT) ||
((wFlags & DST_TYPEMASK) == DST_PREFIXTEXT)) {
bFree = TRUE;
if ((wParam = MBToWCS((LPSTR)lParam, wParam ? (int)wParam : USER_AWCONV_COUNTSTRINGSZ, &(LPWSTR)lpwstr, -1, TRUE)) == 0)
return FALSE;
} else {
bFree = FALSE;
}
bRet = DrawStateW(hDC, hBrush, func, lpwstr, wParam, x, y, cx, cy, wFlags);
if (bFree) {
UserLocalFree((HANDLE)lpwstr);
}
return bRet;
}
/*
* BltColor
* History:
*/
void BltColor(
HDC hdc,
HBRUSH hbr,
HDC hdcSrce,
int xO,
int yO,
int cx,
int cy,
int xO1,
int yO1,
UINT uBltFlags
)
{
HBRUSH hbrSave;
HBRUSH hbrNew = NULL;
DWORD textColorSave;
DWORD bkColorSave;
if (hbr == (HBRUSH)NULL) {
LOGBRUSH lb;
lb.lbStyle = BS_SOLID;
lb.lbColor = GetSysColor(COLOR_WINDOWTEXT);
hbrNew = hbr = CreateBrushIndirect(&lb);
}
/*
* Set the Text and Background colors so that bltColor handles the
* background of buttons (and other bitmaps) properly.
* Save the HDC's old Text and Background colors. This causes problems with
* Omega (and probably other apps) when calling GrayString which uses this
* routine...
*/
textColorSave = SetTextColor(hdc, 0x00000000L);
bkColorSave = SetBkColor(hdc, 0x00FFFFFFL);
hbrSave = SelectObject(hdc, hbr);
BitBlt(hdc, xO, yO, cx, cy, hdcSrce,
xO1, yO1, ((uBltFlags & BC_INVERT) ? 0xB8074AL : 0xE20746L));
//xO1, yO1, (fInvert ? 0xB80000 : 0xE20000));
SelectObject(hdc, hbrSave);
/*
* Restore saved colors
*/
SetTextColor(hdc, textColorSave);
SetBkColor(hdc, bkColorSave);
if (hbrNew) {
DeleteObject(hbrNew);
}
}