2020-09-30 16:53:55 +02:00

971 lines
27 KiB
C++

/**************************************************************************\
*
* Copyright (c) 1998 Microsoft Corporation
*
* Abstract:
*
* Handles the driver viewable Context class.
*
* Revision History:
*
* 12/03/1998 andrewgo
* Created it.
*
\**************************************************************************/
#include "precomp.hpp"
LONG DpContext::Uniqueness = 0xfdbc; // Used with save/restore Id's
DpContext::DpContext(
DpContext * prev
)
{
ASSERT(prev != NULL);
Id = InterlockedDecrement(&Uniqueness) << 16;
Next = NULL;
Prev = prev;
// Save bit 15 for a container flag
Id |= ((prev->Id + 1) & 0x00007FFF);
if (Id == 0) // 0 is not a valid ID
{
Id = 0x0dbc0001;
}
AntiAliasMode = prev->AntiAliasMode;
TextRenderHint = prev->TextRenderHint;
TextContrast = prev->TextContrast;
CompositingMode = prev->CompositingMode;
CompositingQuality = prev->CompositingQuality;
FilterType = prev->FilterType;
PixelOffset = prev->PixelOffset;
Hwnd = prev->Hwnd;
Hdc = prev->Hdc;
IsEmfPlusHdc = prev->IsEmfPlusHdc;
IsPrinter = prev->IsPrinter;
IsDisplay = prev->IsDisplay;
SaveDc = prev->SaveDc;
Palette = prev->Palette;
PaletteMap = prev->PaletteMap;
OriginalHFont = NULL;
CurrentHFont = NULL;
Face = NULL;
ContainerDpiX = prev->ContainerDpiX;
ContainerDpiY = prev->ContainerDpiY;
MetafileRasterizationLimitDpi = prev->MetafileRasterizationLimitDpi;
RenderingOriginX = prev->RenderingOriginX;
RenderingOriginY = prev->RenderingOriginY;
GdiLayered = FALSE;
// Does this need to be prev->IcmMode?
IcmMode = IcmModeOff;
// Clipping and Transforms handled elsewhere
}
DpContext::DpContext(
BOOL isDisplay
)
{
Id = InterlockedDecrement(&Uniqueness) << 16;
Next = NULL;
Prev = NULL;
Id |= 0x0dbc;
AntiAliasMode = 0;
TextRenderHint = TextRenderingHintSystemDefault;
TextContrast = DEFAULT_TEXT_CONTRAST;
CompositingMode = CompositingModeSourceOver;
CompositingQuality = CompositingQualityDefault;
FilterType = InterpolationModeDefaultInternal;
PixelOffset = PixelOffsetModeDefault;
Hwnd = NULL;
Hdc = NULL;
IsEmfPlusHdc = FALSE;
IsPrinter = FALSE;
IsDisplay = isDisplay;
SaveDc = 0;
PageUnit = UnitDisplay;
PageScale = 1.0f;
Palette = NULL;
PaletteMap = NULL;
OriginalHFont = NULL;
CurrentHFont = NULL;
Face = NULL;
ContainerDpiX = Globals::DesktopDpiX;
ContainerDpiY = Globals::DesktopDpiY;
GdiLayered = FALSE;
MetafileRasterizationLimitDpi = max(ContainerDpiX, ContainerDpiY);
ASSERT(MetafileRasterizationLimitDpi > 0.0f);
// Set the default rendering origin to the top left corner of the Graphics.
RenderingOriginX = 0;
RenderingOriginY = 0;
// Set the default ICM mode == ICM off.
IcmMode = IcmModeOff;
// Clipping and Transforms handled elsewhere
}
DpContext::~DpContext()
{
delete Next;
Next = NULL;
DeleteCurrentHFont();
if (Prev == NULL)
{
if (PaletteMap != NULL)
{
delete PaletteMap;
PaletteMap = NULL;
}
if (Palette != NULL)
{
GpFree(Palette);
Palette = NULL;
}
}
} // DpContext::~DpContext
/**************************************************************************\
*
* Function Description:
*
* Internal function that retrieves a clean HDC for the specified
* context (if there is one). This is intended to be used for
* internal functions that require a DC (such as when we leverage
* GDI accelerations for rendering).
*
* The DC is cleaned for the minimum amount that we can. That is,
* the caller can expect an MM_TEXT transform, copy ROP, etc.
*
* We explicitly DO NOT clean attributes that we expect any callers
* to change, such as brush color, text color, etc. (And consequently,
* callers are not expected to preserve those values.)
*
* Reset: Transform, ROP mode
*
* NOT reset: Application clipping, stretch blt mode, current brush/pen,
* foreground color, etc.
*
* Return Value:
*
* NULL if no HDC can be retrieved.
*
* History:
*
* 12/04/1998 andrewgo
* Created it.
*
\**************************************************************************/
HDC
DpContext::GetHdc(
DpBitmap *surface
)
{
HDC hdc = NULL;
// Callers MUST pass in the surface:
ASSERT(surface != NULL);
// The first thing we have to do is flush any of our pending drawing
// (because GDI certainly doesn't know how to flush it!)
surface->Flush(FlushIntentionFlush);
if (Hwnd)
{
// The Graphics was derived off an Hwnd. Use GetCleanHdc
// to get a nice clean DC (not a CS_OWNDC).
hdc = ::GetCleanHdc(Hwnd);
if(hdc)
{
// Set the appropriate ICM mode according to the context.
if(IcmMode == IcmModeOn)
{
// Force the ICM mode on.
::SetICMMode(hdc, ICM_ON);
}
else
{
// There are only 2 IcmMode flags possible. If you've added
// more you need to recode the logic that sets the IcmMode on
// the DC.
ASSERT(IcmMode==IcmModeOff);
::SetICMMode(hdc, ICM_OFF);
}
}
return hdc;
}
else if (Hdc)
{
// The Graphics was derived from a bitmap, printer, or metafile Hdc.
// First, save the application's HDC state and reset all the state
hdc = Hdc;
if (!SaveDc)
{
SaveDc = ::SaveDC(hdc);
if (!SaveDc)
{
return(NULL);
}
this->CleanTheHdc(hdc);
}
}
else if (surface->Type == DpBitmap::CreationType::GPBITMAP)
{
// The GpBitmap is accessible from the EpScanBitmap. It will
// create an HDC and GDI bitmap appropriate for interop.
EpScanBitmap *scan = static_cast<EpScanBitmap*>(surface->Scan);
hdc = scan->GetBitmap()->GetHdc();
// !!! For some reason, this hdc is NOT clean. So in metaplay.cpp
// !!! I have to call CleanTheHdc on this hdc. I don't think I should
// !!! have to do that, but without it, there is bug #121666.
}
return(hdc);
}
/**************************************************************************\
*
* Function Description:
*
* Internal function that cleans the given HDC.
*
* Reset: Transform, ROP mode
*
* NOT reset: Application clipping, stretch blt mode, current brush/pen,
* foreground color, etc.
*
* Notes:
*
* Application clipping IS reset, contrary to the above - this is bug #99338.
*
* Arguments:
*
* hdc - the HDC to clean
*
* History:
*
* ??/??/???? andrewgo
* Created it.
*
\**************************************************************************/
VOID
DpContext::CleanTheHdc(
HDC hdc
)
{
// Reset the minimum number of DC attributes possible
//
// DON'T GRATUITOUSLY ADD RESETS HERE. Read the function
// comment above, and consider resetting any additional
// DC attributes in the function that calls GetHdc().
//
// NOTE: A possible optimization for bitmap surfaces would
// be to CreateCompatibleDC a new DC, and select
// the bitmap into that.
// Set the appropriate ICM mode.
if(IcmMode == IcmModeOn)
{
// Force the ICM mode on.
::SetICMMode(hdc, ICM_ON);
}
else
{
// There are only 2 IcmMode flags possible. If you've added
// more you need to recode the logic that sets the IcmMode on
// the DC.
ASSERT(IcmMode==IcmModeOff);
// Force the ICM mode off.
::SetICMMode(hdc, ICM_OFF);
}
if (!IsEmfPlusHdc)
{
SetMapMode(hdc, MM_TEXT);
SetViewportOrgEx(hdc, 0, 0, NULL);
SetWindowOrgEx(hdc, 0, 0, NULL);
SetROP2(hdc, R2_COPYPEN);
ModifyWorldTransform(hdc, NULL, MWT_IDENTITY);
// If someone does a GpGraphics::GetHdc and sets the clipping,
// we have to be sure to unset it before using the hdc again.
SelectClipRgn(hdc, NULL);
// Do we have to do an EndPath?
}
else // it is an EMF+ HDC
{
// record a minimum of commands in the EMF+ file (if any at all)
BOOL setMapMode = (::GetMapMode(hdc) != MM_TEXT);
POINT point;
point.x = 0;
point.y = 0;
::GetViewportOrgEx(hdc, &point);
BOOL setViewportOrg = ((point.x != 0) || (point.y != 0));
point.x = 0;
point.y = 0;
::GetWindowOrgEx(hdc, &point);
BOOL setWindowOrg = ((point.x != 0) || (point.y != 0));
BOOL setROP2 = (::GetROP2(hdc) != R2_COPYPEN);
#if 0 // do NOT turn this on -- see comments below
BOOL setWorldTransform = FALSE;
// The graphics mode is never GM_ADVANCED on Win9x.
// On WinNT it gets set to GM_ADVANCED when we are playing an EMF
// into the hdc. In that case, we don't want to set the transform
// to the identity, because it will override the srcRect->destRect
// transform of the command to play the metafile, which will mess
// up GDI's transform.
// The only other way we could be in GM_ADVANCED mode is if the
// application created the EMF hdc themselves and set it to GM_ADVANCED
// before creating a graphics from the metafile HDC. That case is
// currently NOT supported, and the app shouldn't do that!
// Perhaps we should add code in the constructor to block that case.
// This test always returns FALSE on Win9x.
if (::GetGraphicsMode(hdc) == GM_ADVANCED)
{
XFORM xformIdentity;
xformIdentity.eM11 = 1.0f;
xformIdentity.eM12 = 0.0f;
xformIdentity.eM21 = 0.0f;
xformIdentity.eM22 = 1.0f;
xformIdentity.eDx = 0.0f;
xformIdentity.eDy = 0.0f;
XFORM xform;
xform.eM11 = 0.0;
if (::GetWorldTransform(hdc, &xform))
{
setWorldTransform = (GpMemcmp(&xform, &xformIdentity, sizeof(xform)) != 0);
}
else
{
setWorldTransform = TRUE;
WARNING1("GetWorldTransform failed");
}
}
#endif
RECT clipRect;
HRGN hRgnTmp = ::CreateRectRgn(0, 0, 0, 0);
BOOL setClipping = ((hRgnTmp == NULL) ||
(::GetClipRgn(hdc, hRgnTmp) != 0));
::DeleteObject(hRgnTmp);
if (setMapMode)
{
::SetMapMode(hdc, MM_TEXT);
}
if (setViewportOrg)
{
::SetViewportOrgEx(hdc, 0, 0, NULL);
}
if (setWindowOrg)
{
::SetWindowOrgEx(hdc, 0, 0, NULL);
}
if (setROP2)
{
::SetROP2(hdc, R2_COPYPEN);
}
#if 0 // do NOT turn this on -- see comments above
if (setWorldTransform)
{
::ModifyWorldTransform(hdc, NULL, MWT_IDENTITY);
}
#endif
if (setClipping)
{
::SelectClipRgn(hdc, NULL);
}
}
}
/**************************************************************************\
*
* Function Description:
*
* Releases the HDC if necessary.
*
* Return Value:
*
* History:
*
* 12/04/1998 andrewgo
* Created it.
*
\**************************************************************************/
VOID
DpContext::ReleaseHdc(
HDC hdc,
DpBitmap *surface
)
{
if (Hwnd)
{
ReleaseDC(Hwnd, hdc);
}
else if (!Hdc && surface &&
(surface->Type == DpBitmap::CreationType::GPBITMAP))
{
// The GpBitmap is accessible from the EpScanBitmap.
EpScanBitmap *scan = static_cast<EpScanBitmap*>(surface->Scan);
scan->GetBitmap()->ReleaseHdc(hdc);
}
}
// ResetHdc() restores the HDC to the state in which it was given to us.
VOID DpContext::ResetHdc(VOID)
{
if (SaveDc)
{
RestoreDC(Hdc, SaveDc);
SaveDc = 0;
}
} // DpContext::ResetHdc
/**************************************************************************\
*
* Function Description:
*
* Retrieves the appropriate transform. Implemented as a routine so that
* we can do lazy evaluation.
*
* Arguments:
*
* [OUT] worldToDevice: world to device matrix.
*
* Return Value:
*
* Ok if the device to world matrix is invertible. If this is not Ok,
* the returned matrix is the identity matrix.
*
* History:
*
* 12/04/1998 andrewgo
* Created it.
*
\**************************************************************************/
GpStatus
DpContext::GetDeviceToWorld(
GpMatrix* deviceToWorld
) const
{
GpStatus status = Ok;
if(!InverseOk)
{
if(WorldToDevice.IsInvertible())
{
DeviceToWorld = WorldToDevice;
DeviceToWorld.Invert();
InverseOk = TRUE;
}
else
{
DeviceToWorld.Reset(); // reset to identity matrix
status = GenericError;
}
}
*deviceToWorld = DeviceToWorld;
return status;
}
// The units we use for the page transform with UnitDisplay depend
// on whether the graphics is associated with a display screen. If
// it is, then we just use the dpi of the display (which is why we
// call it display units). Otherwise (e.g. a printer), we use
// 100 dpi for display units.
#define GDIP_DISPLAY_DPI 100.0f
/**************************************************************************\
*
* Function Description:
*
* Calculate the page multiplier for going from page units to device units.
* Used to concatenate the page transform with the WorldToPage transform.
*
* Arguments:
*
* NONE
*
* Return Value:
*
* NONE
*
* Created:
*
* 3/8/1999 DCurtis
*
\**************************************************************************/
VOID
DpContext::GetPageMultipliers(
REAL * pageMultiplierX,
REAL * pageMultiplierY,
GpPageUnit unit,
REAL scale
) const
{
if ((unit == UnitDisplay) && IsDisplay)
{
// The page transform is always the identity if
// we are rendering to a display, and the unit
// is UnitDisplay.
*pageMultiplierX = 1.0f;
*pageMultiplierY = 1.0f;
return;
}
REAL multiplierX;
REAL multiplierY;
switch (unit)
{
default:
ASSERT(0);
// FALLTHRU
// The units we use for the page transform with UnitDisplay depend
// on whether the graphics is associated with a display screen. If
// it is, then we just use the dpi of the display (which is why we
// call it display units). Otherwise (e.g. a printer), we use
// 100 dpi for display units.
case UnitDisplay: // Variable
// since it's not a display, use the default display dpi of 100
multiplierX = ContainerDpiX * scale / GDIP_DISPLAY_DPI;
multiplierY = ContainerDpiY * scale / GDIP_DISPLAY_DPI;
break;
case UnitPixel: // Each unit represents one device pixel.
multiplierX = scale;
multiplierY = scale;
break;
case UnitPoint: // Each unit represents a printer's point,
// or 1/72 inch.
multiplierX = ContainerDpiX * scale / 72.0f;
multiplierY = ContainerDpiY * scale / 72.0f;
break;
case UnitInch: // Each unit represents 1 inch.
multiplierX = ContainerDpiX * scale;
multiplierY = ContainerDpiY * scale;
break;
case UnitDocument: // Each unit represents 1/300 inch.
multiplierX = ContainerDpiX * scale / 300.0f;
multiplierY = ContainerDpiY * scale / 300.0f;
break;
case UnitMillimeter: // Each unit represents 1 millimeter.
// One Millimeter is 0.03937 inches
// One Inch is 25.4 millimeters
multiplierX = ContainerDpiX * scale / 25.4f;
multiplierY = ContainerDpiY * scale / 25.4f;
break;
}
*pageMultiplierX = multiplierX;
*pageMultiplierY = multiplierY;
}
/**************************************************************************\
*
* Function Description:
*
* Prepares the contexts DC for use in an ExtTextOut call for a given
* font face realization and brush.
*
* Arguments:
*
* NONE
*
* Return Value:
*
* non-NULL - prepared hdc
* NULL - faceRealization or brush could not be represented in a DC
*
* Created:
*
* 3/7/2000 DBrown
*
\**************************************************************************/
const DOUBLE PI = 3.1415926535897932384626433832795;
HDC
DpContext::GetTextOutputHdc(
const GpFaceRealization *faceRealization, // In - Font face required
GpColor color, // In - Required GdiPlus brush effect
DpBitmap *surface, // In
INT *angle // Out
)
{
ASSERT(angle);
if (Hwnd)
{
// Since GetHdc will create a new DC each time for Graphics created
// from an hWnd, we can't track the currently selected font, and
// the overhead of selecting the font and reselecting the original
// font everytime would be inefficent. Therefore don't optimise text in
// Graphics created from Hwnds.
return NULL;
}
// GDI can't handle clearTtype or our sort of anti-aliasing
if (faceRealization->RealizationMethod() != TextRenderingHintSingleBitPerPixelGridFit)
{
return NULL;
}
// If it is a private font, then we need to go through with GDI+
if (faceRealization->IsPrivate())
return NULL;
// Check whether GDI can handle the brush and font size
if (!color.IsOpaque())
{
return NULL; // GDI can only handle solid color brushes
}
if (faceRealization->GetFontFace()->IsSymbol())
{
return NULL;
}
// GDI can't handle the simulation.
if (faceRealization->Getprface()->fobj.flFontType & (FO_SIM_BOLD | FO_SIM_ITALIC | FO_SIM_ITALIC_SIDEWAYS))
{
return NULL;
}
if (surface && (surface->Type == DpBitmap::CreationType::GPBITMAP))
return NULL;
// Check whether GDI can handle the glyph transform
PointF scale;
REAL rotateRadians;
REAL shear;
PointF translate;
SplitTransform(
faceRealization->Getprface()->mxForDDI,
scale,
rotateRadians,
shear,
translate);
if ( scale.X / scale.Y < 0.999
|| scale.X / scale.Y > 1.0001)
{
return NULL; // Don't pass non 1:1 aspect ratios to GDI
}
if ( shear < -0.0001
|| shear > 0.0001)
{
return NULL; // GDI cannot handle shearing
}
// Translate rotation from radians in x-up to tenths of a degree in x-down.
*angle = GpRound(float(3600.0 - (rotateRadians * 1800.0 / PI)));
if (*angle >= 3600)
{
*angle -= 3600;
}
// under platform before NT 5.1 if there is a rotation, we need to render through GDI+
// the main reason is a bug in the TrueType rasterizer that was causing in certain fonts
// text to be rendered unhinted under 90, 180 and 270 degree rotations
if ((*angle != 0) &&
(!Globals::IsNt ||
(Globals::OsVer.dwMajorVersion < 5) ||
((Globals::OsVer.dwMajorVersion == 5) && (Globals::OsVer.dwMinorVersion < 1)) ) )
return NULL;
// Prepare hdc for ExtTextOut
HDC hdc = GetHdc(surface);
if (!hdc)
return NULL;
INT style = faceRealization->Getprface()->Face->GetFaceStyle();
// Select the font if not already selected by a previous caller
GpStatus status = Ok;
if (CurrentHFont == 0 || Face != faceRealization->Getprface()->Face
|| !FontTransform.IsEqual(&faceRealization->Getprface()->mxForDDI)
|| Style != style)
{
Face = faceRealization->Getprface()->Face;
FontTransform = faceRealization->Getprface()->mxForDDI;
Style = style;
status = UpdateCurrentHFont(
NONANTIALIASED_QUALITY,
scale,
*angle,
hdc,
FALSE); // Sideway
}
if (status == Ok)
status = SelectCurrentHFont(hdc);
if (status != Ok)
{
ReleaseHdc(hdc);
return NULL;
}
if (GetBkMode(hdc) != TRANSPARENT)
SetBkMode(hdc, TRANSPARENT);
COLORREF colorRef = color.ToCOLORREF();
SetTextColor(hdc, colorRef);
if (GetTextAlign(hdc) != TA_BASELINE)
SetTextAlign(hdc, TA_BASELINE); // !!! may need VTA_BASELINE or VTA_CENTRE for vertical?
return hdc;
}
VOID DpContext::ReleaseTextOutputHdc(HDC hdc)
{
::SelectObject(hdc, OriginalHFont);
OriginalHFont = NULL;
ReleaseHdc(hdc);
} // DpContext::ReleaseTextOutputHdc
VOID DpContext::DeleteCurrentHFont()
{
ASSERT(OriginalHFont == 0);
if (CurrentHFont)
{
::DeleteObject(CurrentHFont);
CurrentHFont = 0;
}
} // DpContext::DeleteCurrentHFont
GpStatus DpContext::UpdateCurrentHFont(
BYTE quality,
const PointF & scale,
INT angle,
HDC hdc,
BOOL sideway,
BYTE charSet
)
{
if (charSet == 0xFF)
charSet = Face->GetCharset(hdc);
DeleteCurrentHFont();
const LONG emHeight = GpRound(Face->GetDesignEmHeight() * scale.Y);
const LONG emWidth = 0;
LONG rotateDeciDegrees = angle;
const LONG weight = (Style & FontStyleBold) ? 700 : 400;
const BYTE fItalic = (Style & FontStyleItalic) ? TRUE : FALSE;
if (sideway)
{
rotateDeciDegrees -= 900;
if (rotateDeciDegrees < 0)
{
rotateDeciDegrees += 3600;
}
}
// the GP_IFIMETRICS* Face->pifi is internally created structure
// so we trust it is honestly null-terminated
const WCHAR* pwszFamilyName = (const WCHAR*)( (BYTE*)Face->pifi + Face->pifi->dpwszFamilyName );
int sizeFamilyName = wcslen(pwszFamilyName) + 1; // including terminating null
if (Globals::IsNt) {
LOGFONTW lfw = {
-emHeight,
emWidth,
rotateDeciDegrees,
rotateDeciDegrees,
weight,
fItalic,
0,
0,
charSet, // charset
OUT_TT_ONLY_PRECIS,
0,
quality,
0,
L""};
if (sideway)
{
if (sizeFamilyName + 1 > LF_FACESIZE)
return GenericError;
lfw.lfFaceName[0] = 0x0040; // @
memcpy(&lfw.lfFaceName[1], pwszFamilyName, sizeFamilyName*sizeof(WCHAR));
}
else
{
if (sizeFamilyName > LF_FACESIZE)
return GenericError;
memcpy(&lfw.lfFaceName[0], pwszFamilyName, sizeFamilyName*sizeof(WCHAR));
}
CurrentHFont = CreateFontIndirectW(&lfw);
}
else
{
// ANSI version for Win9X
LOGFONTA lfa = {
-emHeight,
emWidth,
rotateDeciDegrees,
rotateDeciDegrees,
weight,
fItalic,
0,
0,
charSet, // charset
OUT_TT_ONLY_PRECIS,
0,
quality,
0,
""};
if (sideway)
{
if (sizeFamilyName + 1 > LF_FACESIZE)
return GenericError;
lfa.lfFaceName[0] = 0x40; // @
UnicodeToAnsiStr(
pwszFamilyName,
&lfa.lfFaceName[1],
LF_FACESIZE-1
);
}
else
{
if (sizeFamilyName > LF_FACESIZE)
return GenericError;
UnicodeToAnsiStr(
pwszFamilyName,
lfa.lfFaceName,
LF_FACESIZE
);
}
CurrentHFont = CreateFontIndirectA(&lfa);
}
if (CurrentHFont == NULL)
{
return GenericError;
}
return Ok;
} // DpContext::UpdateCurrentHFont
GpStatus DpContext::SelectCurrentHFont(HDC hdc)
{
ASSERT(CurrentHFont != 0 && OriginalHFont == 0);
OriginalHFont = (HFONT)::SelectObject(hdc, CurrentHFont);
if (OriginalHFont == 0)
return GenericError;
return Ok;
} // DpContext::SelectCurrentHFont
// Used only when recording a EMF or EMF+ through GpMetafile class
VOID
DpContext::SetMetafileDownLevelRasterizationLimit(
UINT metafileRasterizationLimitDpi
)
{
if (metafileRasterizationLimitDpi > 0)
{
ASSERT(metafileRasterizationLimitDpi >= 10);
MetafileRasterizationLimitDpi = (REAL)metafileRasterizationLimitDpi;
}
else
{
MetafileRasterizationLimitDpi = max(ContainerDpiX, ContainerDpiY);
ASSERT(MetafileRasterizationLimitDpi >= 10);
}
DpContext * prev = Prev;
// The MetafileRasterizationLimitDpi cannot be different in any
// other saved context of the graphics. Update them all.
while (prev != NULL)
{
prev->MetafileRasterizationLimitDpi = MetafileRasterizationLimitDpi;
prev = prev->Prev;
}
}