971 lines
27 KiB
C++
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;
|
|
}
|
|
}
|