mirror of
https://github.com/reactos/reactos.git
synced 2024-11-10 16:48:16 +00:00
c424146e2c
svn path=/branches/cmake-bringup/; revision=48236
1118 lines
31 KiB
C
1118 lines
31 KiB
C
/*
|
|
* ReactOS W32 Subsystem
|
|
* Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003 ReactOS Team
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with this program; if not, write to the Free Software Foundation, Inc.,
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*/
|
|
|
|
#include <win32k.h>
|
|
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#define Rsin(d) ((d) == 0.0 ? 0.0 : ((d) == 90.0 ? 1.0 : sin(d*M_PI/180.0)))
|
|
#define Rcos(d) ((d) == 0.0 ? 1.0 : ((d) == 90.0 ? 0.0 : cos(d*M_PI/180.0)))
|
|
|
|
BOOL FASTCALL IntFillEllipse( PDC dc, INT XLeft, INT YLeft, INT Width, INT Height, PBRUSH pbrush);
|
|
BOOL FASTCALL IntDrawEllipse( PDC dc, INT XLeft, INT YLeft, INT Width, INT Height, PBRUSH pbrush);
|
|
BOOL FASTCALL IntFillRoundRect( PDC dc, INT Left, INT Top, INT Right, INT Bottom, INT Wellipse, INT Hellipse, PBRUSH pbrush);
|
|
BOOL FASTCALL IntDrawRoundRect( PDC dc, INT Left, INT Top, INT Right, INT Bottom, INT Wellipse, INT Hellipse, PBRUSH pbrush);
|
|
|
|
BOOL FASTCALL
|
|
IntGdiPolygon(PDC dc,
|
|
PPOINT Points,
|
|
int Count)
|
|
{
|
|
SURFACE *psurf;
|
|
PBRUSH pbrLine, pbrFill;
|
|
BOOL ret = FALSE; // default to failure
|
|
RECTL DestRect;
|
|
int CurrentPoint;
|
|
PDC_ATTR pdcattr;
|
|
POINTL BrushOrigin;
|
|
// int Left;
|
|
// int Top;
|
|
|
|
ASSERT(dc); // caller's responsibility to pass a valid dc
|
|
|
|
if (!Points || Count < 2 )
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
//Find start x, y
|
|
Left = Points[0].x;
|
|
Top = Points[0].y;
|
|
for (CurrentPoint = 1; CurrentPoint < Count; ++CurrentPoint) {
|
|
Left = min(Left, Points[CurrentPoint].x);
|
|
Top = min(Top, Points[CurrentPoint].y);
|
|
}
|
|
*/
|
|
|
|
pdcattr = dc->pdcattr;
|
|
|
|
/* Convert to screen coordinates */
|
|
IntLPtoDP(dc, Points, Count);
|
|
for (CurrentPoint = 0; CurrentPoint < Count; CurrentPoint++)
|
|
{
|
|
Points[CurrentPoint].x += dc->ptlDCOrig.x;
|
|
Points[CurrentPoint].y += dc->ptlDCOrig.y;
|
|
}
|
|
// No need to have path here.
|
|
{
|
|
DestRect.left = Points[0].x;
|
|
DestRect.right = Points[0].x;
|
|
DestRect.top = Points[0].y;
|
|
DestRect.bottom = Points[0].y;
|
|
|
|
for (CurrentPoint = 1; CurrentPoint < Count; ++CurrentPoint)
|
|
{
|
|
DestRect.left = min(DestRect.left, Points[CurrentPoint].x);
|
|
DestRect.right = max(DestRect.right, Points[CurrentPoint].x);
|
|
DestRect.top = min(DestRect.top, Points[CurrentPoint].y);
|
|
DestRect.bottom = max(DestRect.bottom, Points[CurrentPoint].y);
|
|
}
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_FILL | DC_BRUSH_DIRTY))
|
|
DC_vUpdateFillBrush(dc);
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_LINE | DC_PEN_DIRTY))
|
|
DC_vUpdateLineBrush(dc);
|
|
|
|
/* Special locking order to avoid lock-ups */
|
|
pbrFill = dc->dclevel.pbrFill;
|
|
pbrLine = dc->dclevel.pbrLine;
|
|
psurf = dc->dclevel.pSurface;
|
|
/* FIXME - psurf can be NULL!!!! don't assert but handle this case gracefully! */
|
|
ASSERT(psurf);
|
|
|
|
/* Now fill the polygon with the current fill brush. */
|
|
if (!(pbrFill->flAttrs & GDIBRUSH_IS_NULL))
|
|
{
|
|
BrushOrigin = *((PPOINTL)&pbrFill->ptOrigin);
|
|
BrushOrigin.x += dc->ptlDCOrig.x;
|
|
BrushOrigin.y += dc->ptlDCOrig.y;
|
|
ret = IntFillPolygon (dc,
|
|
psurf,
|
|
&dc->eboFill.BrushObject,
|
|
Points,
|
|
Count,
|
|
DestRect,
|
|
&BrushOrigin);
|
|
}
|
|
|
|
// Draw the Polygon Edges with the current pen ( if not a NULL pen )
|
|
if (!(pbrLine->flAttrs & GDIBRUSH_IS_NULL))
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < Count-1; i++)
|
|
{
|
|
|
|
// DPRINT1("Polygon Making line from (%d,%d) to (%d,%d)\n",
|
|
// Points[0].x, Points[0].y,
|
|
// Points[1].x, Points[1].y );
|
|
|
|
ret = IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
Points[i].x, /* From */
|
|
Points[i].y,
|
|
Points[i+1].x, /* To */
|
|
Points[i+1].y,
|
|
&DestRect,
|
|
ROP2_TO_MIX(pdcattr->jROP2)); /* MIX */
|
|
if (!ret) break;
|
|
}
|
|
/* Close the polygon */
|
|
if (ret)
|
|
{
|
|
ret = IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
Points[Count-1].x, /* From */
|
|
Points[Count-1].y,
|
|
Points[0].x, /* To */
|
|
Points[0].y,
|
|
&DestRect,
|
|
ROP2_TO_MIX(pdcattr->jROP2)); /* MIX */
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL FASTCALL
|
|
IntGdiPolyPolygon(DC *dc,
|
|
LPPOINT Points,
|
|
PULONG PolyCounts,
|
|
int Count)
|
|
{
|
|
if (PATH_IsPathOpen(dc->dclevel))
|
|
return PATH_PolyPolygon ( dc, Points, (PINT)PolyCounts, Count);
|
|
|
|
while (--Count >=0)
|
|
{
|
|
if (!IntGdiPolygon ( dc, Points, *PolyCounts ))
|
|
return FALSE;
|
|
Points+=*PolyCounts++;
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
|
|
/******************************************************************************/
|
|
|
|
/*
|
|
* NtGdiEllipse
|
|
*
|
|
* Author
|
|
* Filip Navara
|
|
*
|
|
* Remarks
|
|
* This function uses optimized Bresenham's ellipse algorithm. It draws
|
|
* four lines of the ellipse in one pass.
|
|
*
|
|
*/
|
|
|
|
BOOL APIENTRY
|
|
NtGdiEllipse(
|
|
HDC hDC,
|
|
int Left,
|
|
int Top,
|
|
int Right,
|
|
int Bottom)
|
|
{
|
|
PDC dc;
|
|
PDC_ATTR pdcattr;
|
|
RECTL RectBounds;
|
|
PBRUSH pbrush;
|
|
BOOL ret = TRUE;
|
|
LONG PenWidth, PenOrigWidth;
|
|
LONG RadiusX, RadiusY, CenterX, CenterY;
|
|
PBRUSH pFillBrushObj;
|
|
BRUSH tmpFillBrushObj;
|
|
|
|
if ((Left == Right) || (Top == Bottom)) return TRUE;
|
|
|
|
dc = DC_LockDc(hDC);
|
|
if (dc == NULL)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
if (dc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(dc);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
return TRUE;
|
|
}
|
|
|
|
if (PATH_IsPathOpen(dc->dclevel))
|
|
{
|
|
ret = PATH_Ellipse(dc, Left, Top, Right, Bottom);
|
|
DC_UnlockDc(dc);
|
|
return ret;
|
|
}
|
|
|
|
if (Right < Left)
|
|
{
|
|
INT tmp = Right; Right = Left; Left = tmp;
|
|
}
|
|
if (Bottom < Top)
|
|
{
|
|
INT tmp = Bottom; Bottom = Top; Top = tmp;
|
|
}
|
|
|
|
pdcattr = dc->pdcattr;
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_FILL | DC_BRUSH_DIRTY))
|
|
DC_vUpdateFillBrush(dc);
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_LINE | DC_PEN_DIRTY))
|
|
DC_vUpdateLineBrush(dc);
|
|
|
|
pbrush = PEN_LockPen(pdcattr->hpen);
|
|
if (!pbrush)
|
|
{
|
|
DPRINT1("Ellipse Fail 1\n");
|
|
DC_UnlockDc(dc);
|
|
SetLastWin32Error(ERROR_INTERNAL_ERROR);
|
|
return FALSE;
|
|
}
|
|
|
|
PenOrigWidth = PenWidth = pbrush->ptPenWidth.x;
|
|
if (pbrush->ulPenStyle == PS_NULL) PenWidth = 0;
|
|
|
|
if (pbrush->ulPenStyle == PS_INSIDEFRAME)
|
|
{
|
|
if (2*PenWidth > (Right - Left)) PenWidth = (Right -Left + 1)/2;
|
|
if (2*PenWidth > (Bottom - Top)) PenWidth = (Bottom -Top + 1)/2;
|
|
Left += PenWidth / 2;
|
|
Right -= (PenWidth - 1) / 2;
|
|
Top += PenWidth / 2;
|
|
Bottom -= (PenWidth - 1) / 2;
|
|
}
|
|
|
|
if (!PenWidth) PenWidth = 1;
|
|
pbrush->ptPenWidth.x = PenWidth;
|
|
|
|
RectBounds.left = Left;
|
|
RectBounds.right = Right;
|
|
RectBounds.top = Top;
|
|
RectBounds.bottom = Bottom;
|
|
|
|
IntLPtoDP(dc, (LPPOINT)&RectBounds, 2);
|
|
|
|
RectBounds.left += dc->ptlDCOrig.x;
|
|
RectBounds.right += dc->ptlDCOrig.x;
|
|
RectBounds.top += dc->ptlDCOrig.y;
|
|
RectBounds.bottom += dc->ptlDCOrig.y;
|
|
|
|
// Setup for dynamic width and height.
|
|
RadiusX = max((RectBounds.right - RectBounds.left) / 2, 2); // Needs room
|
|
RadiusY = max((RectBounds.bottom - RectBounds.top) / 2, 2);
|
|
CenterX = (RectBounds.right + RectBounds.left) / 2;
|
|
CenterY = (RectBounds.bottom + RectBounds.top) / 2;
|
|
|
|
DPRINT("Ellipse 1: Left: %d, Top: %d, Right: %d, Bottom: %d\n",
|
|
RectBounds.left,RectBounds.top,RectBounds.right,RectBounds.bottom);
|
|
|
|
DPRINT("Ellipse 2: XLeft: %d, YLeft: %d, Width: %d, Height: %d\n",
|
|
CenterX - RadiusX, CenterY + RadiusY, RadiusX*2, RadiusY*2);
|
|
|
|
pFillBrushObj = BRUSH_LockBrush(pdcattr->hbrush);
|
|
if (NULL == pFillBrushObj)
|
|
{
|
|
DPRINT1("FillEllipse Fail\n");
|
|
SetLastWin32Error(ERROR_INTERNAL_ERROR);
|
|
ret = FALSE;
|
|
}
|
|
else
|
|
{
|
|
RtlCopyMemory(&tmpFillBrushObj, pFillBrushObj, sizeof(tmpFillBrushObj));
|
|
// tmpFillBrushObj.ptOrigin.x += RectBounds.left - Left;
|
|
// tmpFillBrushObj.ptOrigin.y += RectBounds.top - Top;
|
|
tmpFillBrushObj.ptOrigin.x += dc->ptlDCOrig.x;
|
|
tmpFillBrushObj.ptOrigin.y += dc->ptlDCOrig.y;
|
|
ret = IntFillEllipse( dc,
|
|
CenterX - RadiusX,
|
|
CenterY - RadiusY,
|
|
RadiusX*2, // Width
|
|
RadiusY*2, // Height
|
|
&tmpFillBrushObj);
|
|
BRUSH_UnlockBrush(pFillBrushObj);
|
|
}
|
|
|
|
if (ret)
|
|
ret = IntDrawEllipse( dc,
|
|
CenterX - RadiusX,
|
|
CenterY - RadiusY,
|
|
RadiusX*2, // Width
|
|
RadiusY*2, // Height
|
|
pbrush);
|
|
|
|
pbrush->ptPenWidth.x = PenOrigWidth;
|
|
PEN_UnlockPen(pbrush);
|
|
DC_UnlockDc(dc);
|
|
DPRINT("Ellipse Exit.\n");
|
|
return ret;
|
|
}
|
|
|
|
#if 0
|
|
|
|
//When the fill mode is ALTERNATE, GDI fills the area between odd-numbered and
|
|
//even-numbered polygon sides on each scan line. That is, GDI fills the area between the
|
|
//first and second side, between the third and fourth side, and so on.
|
|
|
|
//WINDING Selects winding mode (fills any region with a nonzero winding value).
|
|
//When the fill mode is WINDING, GDI fills any region that has a nonzero winding value.
|
|
//This value is defined as the number of times a pen used to draw the polygon would go around the region.
|
|
//The direction of each edge of the polygon is important.
|
|
|
|
extern BOOL FillPolygon(PDC dc,
|
|
SURFOBJ *SurfObj,
|
|
PBRUSHOBJ BrushObj,
|
|
MIX RopMode,
|
|
CONST PPOINT Points,
|
|
int Count,
|
|
RECTL BoundRect);
|
|
|
|
#endif
|
|
|
|
|
|
ULONG_PTR
|
|
APIENTRY
|
|
NtGdiPolyPolyDraw( IN HDC hDC,
|
|
IN PPOINT UnsafePoints,
|
|
IN PULONG UnsafeCounts,
|
|
IN ULONG Count,
|
|
IN INT iFunc )
|
|
{
|
|
DC *dc;
|
|
PVOID pTemp;
|
|
LPPOINT SafePoints;
|
|
PULONG SafeCounts;
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
BOOL Ret = TRUE;
|
|
INT nPoints = 0, nMaxPoints = 0, nInvalid = 0, i;
|
|
|
|
if (!UnsafePoints || !UnsafeCounts ||
|
|
Count == 0 || iFunc == 0 || iFunc > GdiPolyPolyRgn)
|
|
{
|
|
/* Windows doesn't set last error */
|
|
return FALSE;
|
|
}
|
|
|
|
_SEH2_TRY
|
|
{
|
|
ProbeForRead(UnsafePoints, Count * sizeof(POINT), 1);
|
|
ProbeForRead(UnsafeCounts, Count * sizeof(ULONG), 1);
|
|
|
|
/* Count points and validate poligons */
|
|
for (i = 0; i < Count; i++)
|
|
{
|
|
if (UnsafeCounts[i] < 2)
|
|
{
|
|
nInvalid++;
|
|
}
|
|
nPoints += UnsafeCounts[i];
|
|
nMaxPoints = max(nMaxPoints, UnsafeCounts[i]);
|
|
}
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
Status = _SEH2_GetExceptionCode();
|
|
}
|
|
_SEH2_END;
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
/* Windows doesn't set last error */
|
|
return FALSE;
|
|
}
|
|
|
|
if (nPoints == 0 || nPoints < nMaxPoints)
|
|
{
|
|
/* If all polygon counts are zero, or we have overflow,
|
|
return without setting a last error code. */
|
|
return FALSE;
|
|
}
|
|
|
|
if (nInvalid != 0)
|
|
{
|
|
/* If at least one poly count is 0 or 1, fail */
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Allocate one buffer for both counts and points */
|
|
pTemp = ExAllocatePoolWithTag(PagedPool,
|
|
Count * sizeof(ULONG) + nPoints * sizeof(POINT),
|
|
TAG_SHAPE);
|
|
if (!pTemp)
|
|
{
|
|
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
|
|
return FALSE;
|
|
}
|
|
|
|
SafeCounts = pTemp;
|
|
SafePoints = (PVOID)(SafeCounts + Count);
|
|
|
|
_SEH2_TRY
|
|
{
|
|
/* Pointers already probed! */
|
|
RtlCopyMemory(SafeCounts, UnsafeCounts, Count * sizeof(ULONG));
|
|
RtlCopyMemory(SafePoints, UnsafePoints, nPoints * sizeof(POINT));
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
Status = _SEH2_GetExceptionCode();
|
|
}
|
|
_SEH2_END;
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
ExFreePoolWithTag(pTemp, TAG_SHAPE);
|
|
return FALSE;
|
|
}
|
|
|
|
/* Special handling for GdiPolyPolyRgn */
|
|
if (iFunc == GdiPolyPolyRgn)
|
|
{
|
|
HRGN hRgn;
|
|
hRgn = IntCreatePolyPolygonRgn(SafePoints, SafeCounts, Count, (INT_PTR)hDC);
|
|
ExFreePoolWithTag(pTemp, TAG_SHAPE);
|
|
return (ULONG_PTR)hRgn;
|
|
}
|
|
|
|
dc = DC_LockDc(hDC);
|
|
if (!dc)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
ExFreePool(pTemp);
|
|
return FALSE;
|
|
}
|
|
|
|
if (dc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(dc);
|
|
ExFreePool(pTemp);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
return TRUE;
|
|
}
|
|
|
|
/* Perform the actual work */
|
|
switch (iFunc)
|
|
{
|
|
case GdiPolyPolygon:
|
|
Ret = IntGdiPolyPolygon(dc, SafePoints, SafeCounts, Count);
|
|
break;
|
|
case GdiPolyPolyLine:
|
|
Ret = IntGdiPolyPolyline(dc, SafePoints, SafeCounts, Count);
|
|
break;
|
|
case GdiPolyBezier:
|
|
Ret = IntGdiPolyBezier(dc, SafePoints, *SafeCounts);
|
|
break;
|
|
case GdiPolyLineTo:
|
|
Ret = IntGdiPolylineTo(dc, SafePoints, *SafeCounts);
|
|
break;
|
|
case GdiPolyBezierTo:
|
|
Ret = IntGdiPolyBezierTo(dc, SafePoints, *SafeCounts);
|
|
break;
|
|
default:
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
Ret = FALSE;
|
|
}
|
|
|
|
/* Cleanup and return */
|
|
DC_UnlockDc(dc);
|
|
ExFreePool(pTemp);
|
|
|
|
return (ULONG_PTR)Ret;
|
|
}
|
|
|
|
|
|
BOOL
|
|
FASTCALL
|
|
IntRectangle(PDC dc,
|
|
int LeftRect,
|
|
int TopRect,
|
|
int RightRect,
|
|
int BottomRect)
|
|
{
|
|
SURFACE *psurf = NULL;
|
|
PBRUSH pbrLine, pbrFill;
|
|
BOOL ret = FALSE; // default to failure
|
|
RECTL DestRect;
|
|
MIX Mix;
|
|
PDC_ATTR pdcattr;
|
|
POINTL BrushOrigin;
|
|
|
|
ASSERT ( dc ); // caller's responsibility to set this up
|
|
|
|
pdcattr = dc->pdcattr;
|
|
|
|
/* Do we rotate or shear? */
|
|
if (!(dc->dclevel.mxWorldToDevice.flAccel & MX_SCALE))
|
|
{
|
|
|
|
POINTL DestCoords[4];
|
|
ULONG PolyCounts = 4;
|
|
DestCoords[0].x = DestCoords[3].x = LeftRect;
|
|
DestCoords[0].y = DestCoords[1].y = TopRect;
|
|
DestCoords[1].x = DestCoords[2].x = RightRect;
|
|
DestCoords[2].y = DestCoords[3].y = BottomRect;
|
|
// Use IntGdiPolyPolygon so to support PATH.
|
|
return IntGdiPolyPolygon(dc, DestCoords, &PolyCounts, 1);
|
|
}
|
|
// Rectangle Path only.
|
|
if ( PATH_IsPathOpen(dc->dclevel) )
|
|
{
|
|
return PATH_Rectangle ( dc, LeftRect, TopRect, RightRect, BottomRect );
|
|
}
|
|
|
|
DestRect.left = LeftRect;
|
|
DestRect.right = RightRect;
|
|
DestRect.top = TopRect;
|
|
DestRect.bottom = BottomRect;
|
|
|
|
IntLPtoDP(dc, (LPPOINT)&DestRect, 2);
|
|
|
|
DestRect.left += dc->ptlDCOrig.x;
|
|
DestRect.right += dc->ptlDCOrig.x;
|
|
DestRect.top += dc->ptlDCOrig.y;
|
|
DestRect.bottom += dc->ptlDCOrig.y;
|
|
|
|
/* In GM_COMPATIBLE, don't include bottom and right edges */
|
|
if (pdcattr->iGraphicsMode == GM_COMPATIBLE)
|
|
{
|
|
DestRect.right--;
|
|
DestRect.bottom--;
|
|
}
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_FILL | DC_BRUSH_DIRTY))
|
|
DC_vUpdateFillBrush(dc);
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_LINE | DC_PEN_DIRTY))
|
|
DC_vUpdateLineBrush(dc);
|
|
|
|
pbrFill = dc->dclevel.pbrFill;
|
|
pbrLine = dc->dclevel.pbrLine;
|
|
if (!pbrLine)
|
|
{
|
|
ret = FALSE;
|
|
goto cleanup;
|
|
}
|
|
psurf = dc->dclevel.pSurface;
|
|
if (!psurf)
|
|
{
|
|
ret = FALSE;
|
|
goto cleanup;
|
|
}
|
|
|
|
if (pbrFill)
|
|
{
|
|
if (!(pbrFill->flAttrs & GDIBRUSH_IS_NULL))
|
|
{
|
|
BrushOrigin = *((PPOINTL)&pbrFill->ptOrigin);
|
|
BrushOrigin.x += dc->ptlDCOrig.x;
|
|
BrushOrigin.y += dc->ptlDCOrig.y;
|
|
ret = IntEngBitBlt(&psurf->SurfObj,
|
|
NULL,
|
|
NULL,
|
|
dc->rosdc.CombinedClip,
|
|
NULL,
|
|
&DestRect,
|
|
NULL,
|
|
NULL,
|
|
&dc->eboFill.BrushObject,
|
|
&BrushOrigin,
|
|
ROP3_TO_ROP4(PATCOPY));
|
|
}
|
|
}
|
|
|
|
// Draw the rectangle with the current pen
|
|
|
|
ret = TRUE; // change default to success
|
|
|
|
if (!(pbrLine->flAttrs & GDIBRUSH_IS_NULL))
|
|
{
|
|
Mix = ROP2_TO_MIX(pdcattr->jROP2);
|
|
ret = ret && IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
DestRect.left, DestRect.top, DestRect.right, DestRect.top,
|
|
&DestRect, // Bounding rectangle
|
|
Mix);
|
|
|
|
ret = ret && IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
DestRect.right, DestRect.top, DestRect.right, DestRect.bottom,
|
|
&DestRect, // Bounding rectangle
|
|
Mix);
|
|
|
|
ret = ret && IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
DestRect.right, DestRect.bottom, DestRect.left, DestRect.bottom,
|
|
&DestRect, // Bounding rectangle
|
|
Mix);
|
|
|
|
ret = ret && IntEngLineTo(&psurf->SurfObj,
|
|
dc->rosdc.CombinedClip,
|
|
&dc->eboLine.BrushObject,
|
|
DestRect.left, DestRect.bottom, DestRect.left, DestRect.top,
|
|
&DestRect, // Bounding rectangle
|
|
Mix);
|
|
}
|
|
|
|
cleanup:
|
|
/* Move current position in DC?
|
|
MSDN: The current position is neither used nor updated by Rectangle. */
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
APIENTRY
|
|
NtGdiRectangle(HDC hDC,
|
|
int LeftRect,
|
|
int TopRect,
|
|
int RightRect,
|
|
int BottomRect)
|
|
{
|
|
DC *dc;
|
|
BOOL ret; // default to failure
|
|
|
|
dc = DC_LockDc(hDC);
|
|
if (!dc)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
if (dc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(dc);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
return TRUE;
|
|
}
|
|
|
|
ret = IntRectangle ( dc, LeftRect, TopRect, RightRect, BottomRect );
|
|
DC_UnlockDc ( dc );
|
|
|
|
return ret;
|
|
}
|
|
|
|
|
|
BOOL
|
|
FASTCALL
|
|
IntRoundRect(
|
|
PDC dc,
|
|
int Left,
|
|
int Top,
|
|
int Right,
|
|
int Bottom,
|
|
int xCurveDiameter,
|
|
int yCurveDiameter)
|
|
{
|
|
PDC_ATTR pdcattr;
|
|
PBRUSH pbrushLine, pbrushFill;
|
|
RECTL RectBounds;
|
|
LONG PenWidth, PenOrigWidth;
|
|
BOOL ret = TRUE; // default to success
|
|
BRUSH brushTemp;
|
|
|
|
ASSERT ( dc ); // caller's responsibility to set this up
|
|
|
|
if ( PATH_IsPathOpen(dc->dclevel) )
|
|
return PATH_RoundRect ( dc, Left, Top, Right, Bottom,
|
|
xCurveDiameter, yCurveDiameter );
|
|
|
|
if ((Left == Right) || (Top == Bottom)) return TRUE;
|
|
|
|
xCurveDiameter = max(abs( xCurveDiameter ), 1);
|
|
yCurveDiameter = max(abs( yCurveDiameter ), 1);
|
|
|
|
if (Right < Left)
|
|
{
|
|
INT tmp = Right; Right = Left; Left = tmp;
|
|
}
|
|
if (Bottom < Top)
|
|
{
|
|
INT tmp = Bottom; Bottom = Top; Top = tmp;
|
|
}
|
|
|
|
pdcattr = dc->pdcattr;
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_FILL | DC_BRUSH_DIRTY))
|
|
DC_vUpdateFillBrush(dc);
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_LINE | DC_PEN_DIRTY))
|
|
DC_vUpdateLineBrush(dc);
|
|
|
|
pbrushLine = PEN_LockPen(pdcattr->hpen);
|
|
if (!pbrushLine)
|
|
{
|
|
/* Nothing to do, as we don't have a bitmap */
|
|
SetLastWin32Error(ERROR_INTERNAL_ERROR);
|
|
return FALSE;
|
|
}
|
|
|
|
PenOrigWidth = PenWidth = pbrushLine->ptPenWidth.x;
|
|
if (pbrushLine->ulPenStyle == PS_NULL) PenWidth = 0;
|
|
|
|
if (pbrushLine->ulPenStyle == PS_INSIDEFRAME)
|
|
{
|
|
if (2*PenWidth > (Right - Left)) PenWidth = (Right -Left + 1)/2;
|
|
if (2*PenWidth > (Bottom - Top)) PenWidth = (Bottom -Top + 1)/2;
|
|
Left += PenWidth / 2;
|
|
Right -= (PenWidth - 1) / 2;
|
|
Top += PenWidth / 2;
|
|
Bottom -= (PenWidth - 1) / 2;
|
|
}
|
|
|
|
if (!PenWidth) PenWidth = 1;
|
|
pbrushLine->ptPenWidth.x = PenWidth;
|
|
|
|
RectBounds.left = Left;
|
|
RectBounds.top = Top;
|
|
RectBounds.right = Right;
|
|
RectBounds.bottom = Bottom;
|
|
|
|
IntLPtoDP(dc, (LPPOINT)&RectBounds, 2);
|
|
|
|
RectBounds.left += dc->ptlDCOrig.x;
|
|
RectBounds.top += dc->ptlDCOrig.y;
|
|
RectBounds.right += dc->ptlDCOrig.x;
|
|
RectBounds.bottom += dc->ptlDCOrig.y;
|
|
|
|
pbrushFill = BRUSH_LockBrush(pdcattr->hbrush);
|
|
if (NULL == pbrushFill)
|
|
{
|
|
DPRINT1("FillRound Fail\n");
|
|
SetLastWin32Error(ERROR_INTERNAL_ERROR);
|
|
ret = FALSE;
|
|
}
|
|
else
|
|
{
|
|
RtlCopyMemory(&brushTemp, pbrushFill, sizeof(brushTemp));
|
|
brushTemp.ptOrigin.x += RectBounds.left - Left;
|
|
brushTemp.ptOrigin.y += RectBounds.top - Top;
|
|
ret = IntFillRoundRect( dc,
|
|
RectBounds.left,
|
|
RectBounds.top,
|
|
RectBounds.right,
|
|
RectBounds.bottom,
|
|
xCurveDiameter,
|
|
yCurveDiameter,
|
|
&brushTemp);
|
|
BRUSH_UnlockBrush(pbrushFill);
|
|
}
|
|
|
|
if (ret)
|
|
ret = IntDrawRoundRect( dc,
|
|
RectBounds.left,
|
|
RectBounds.top,
|
|
RectBounds.right,
|
|
RectBounds.bottom,
|
|
xCurveDiameter,
|
|
yCurveDiameter,
|
|
pbrushLine);
|
|
|
|
pbrushLine->ptPenWidth.x = PenOrigWidth;
|
|
PEN_UnlockPen(pbrushLine);
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
APIENTRY
|
|
NtGdiRoundRect(
|
|
HDC hDC,
|
|
int LeftRect,
|
|
int TopRect,
|
|
int RightRect,
|
|
int BottomRect,
|
|
int Width,
|
|
int Height)
|
|
{
|
|
DC *dc = DC_LockDc(hDC);
|
|
BOOL ret = FALSE; /* default to failure */
|
|
|
|
DPRINT("NtGdiRoundRect(0x%x,%i,%i,%i,%i,%i,%i)\n",hDC,LeftRect,TopRect,RightRect,BottomRect,Width,Height);
|
|
if ( !dc )
|
|
{
|
|
DPRINT1("NtGdiRoundRect() - hDC is invalid\n");
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
}
|
|
else if (dc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(dc);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
ret = TRUE;
|
|
}
|
|
else
|
|
{
|
|
ret = IntRoundRect ( dc, LeftRect, TopRect, RightRect, BottomRect, Width, Height );
|
|
DC_UnlockDc ( dc );
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
BOOL
|
|
NTAPI
|
|
GreGradientFill(
|
|
HDC hdc,
|
|
PTRIVERTEX pVertex,
|
|
ULONG nVertex,
|
|
PVOID pMesh,
|
|
ULONG nMesh,
|
|
ULONG ulMode)
|
|
{
|
|
PDC pdc;
|
|
SURFACE *psurf;
|
|
PPALETTE ppal;
|
|
EXLATEOBJ exlo;
|
|
RECTL rclExtent;
|
|
POINTL ptlDitherOrg;
|
|
ULONG i;
|
|
BOOL bRet;
|
|
HPALETTE hDestPalette;
|
|
|
|
/* Check parameters */
|
|
if (ulMode == GRADIENT_FILL_TRIANGLE)
|
|
{
|
|
PGRADIENT_TRIANGLE pTriangle = (PGRADIENT_TRIANGLE)pMesh;
|
|
|
|
for (i = 0; i < nMesh; i++, pTriangle++)
|
|
{
|
|
if (pTriangle->Vertex1 >= nVertex ||
|
|
pTriangle->Vertex2 >= nVertex ||
|
|
pTriangle->Vertex3 >= nVertex)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
PGRADIENT_RECT pRect = (PGRADIENT_RECT)pMesh;
|
|
for (i = 0; i < nMesh; i++, pRect++)
|
|
{
|
|
if (pRect->UpperLeft >= nVertex || pRect->LowerRight >= nVertex)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* Lock the output DC */
|
|
pdc = DC_LockDc(hdc);
|
|
if (!pdc)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
|
|
if (pdc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(pdc);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
return TRUE;
|
|
}
|
|
|
|
psurf = pdc->dclevel.pSurface;
|
|
if (!psurf)
|
|
{
|
|
/* Memory DC with no surface selected */
|
|
DC_UnlockDc(pdc);
|
|
return TRUE; // CHECKME
|
|
}
|
|
|
|
/* calculate extent */
|
|
rclExtent.left = rclExtent.right = pVertex->x;
|
|
rclExtent.top = rclExtent.bottom = pVertex->y;
|
|
for (i = 0; i < nVertex; i++)
|
|
{
|
|
rclExtent.left = min(rclExtent.left, (pVertex + i)->x);
|
|
rclExtent.right = max(rclExtent.right, (pVertex + i)->x);
|
|
rclExtent.top = min(rclExtent.top, (pVertex + i)->y);
|
|
rclExtent.bottom = max(rclExtent.bottom, (pVertex + i)->y);
|
|
}
|
|
|
|
IntLPtoDP(pdc, (LPPOINT)&rclExtent, 2);
|
|
rclExtent.left += pdc->ptlDCOrig.x;
|
|
rclExtent.right += pdc->ptlDCOrig.x;
|
|
rclExtent.top += pdc->ptlDCOrig.y;
|
|
rclExtent.bottom += pdc->ptlDCOrig.y;
|
|
|
|
ptlDitherOrg.x = ptlDitherOrg.y = 0;
|
|
IntLPtoDP(pdc, (LPPOINT)&ptlDitherOrg, 1);
|
|
ptlDitherOrg.x += pdc->ptlDCOrig.x;
|
|
ptlDitherOrg.y += pdc->ptlDCOrig.y;
|
|
|
|
hDestPalette = psurf->hDIBPalette;
|
|
if (!hDestPalette) hDestPalette = pPrimarySurface->devinfo.hpalDefault;
|
|
|
|
ppal = PALETTE_LockPalette(hDestPalette);
|
|
EXLATEOBJ_vInitialize(&exlo, &gpalRGB, ppal, 0, 0, 0);
|
|
|
|
ASSERT(pdc->rosdc.CombinedClip);
|
|
|
|
bRet = IntEngGradientFill(&psurf->SurfObj,
|
|
pdc->rosdc.CombinedClip,
|
|
&exlo.xlo,
|
|
pVertex,
|
|
nVertex,
|
|
pMesh,
|
|
nMesh,
|
|
&rclExtent,
|
|
&ptlDitherOrg,
|
|
ulMode);
|
|
|
|
EXLATEOBJ_vCleanup(&exlo);
|
|
|
|
if (ppal)
|
|
PALETTE_UnlockPalette(ppal);
|
|
|
|
DC_UnlockDc(pdc);
|
|
|
|
return bRet;
|
|
}
|
|
|
|
BOOL
|
|
APIENTRY
|
|
NtGdiGradientFill(
|
|
HDC hdc,
|
|
PTRIVERTEX pVertex,
|
|
ULONG nVertex,
|
|
PVOID pMesh,
|
|
ULONG nMesh,
|
|
ULONG ulMode)
|
|
{
|
|
BOOL bRet;
|
|
PTRIVERTEX SafeVertex;
|
|
PVOID SafeMesh;
|
|
ULONG cbVertex, cbMesh;
|
|
|
|
/* Validate parameters */
|
|
if (!pVertex || !nVertex || !pMesh || !nMesh)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
switch (ulMode)
|
|
{
|
|
case GRADIENT_FILL_RECT_H:
|
|
case GRADIENT_FILL_RECT_V:
|
|
cbMesh = nMesh * sizeof(GRADIENT_RECT);
|
|
break;
|
|
case GRADIENT_FILL_TRIANGLE:
|
|
cbMesh = nMesh * sizeof(GRADIENT_TRIANGLE);
|
|
break;
|
|
default:
|
|
SetLastWin32Error(ERROR_INVALID_PARAMETER);
|
|
return FALSE;
|
|
}
|
|
|
|
cbVertex = nVertex * sizeof(TRIVERTEX);
|
|
if (cbVertex + cbMesh <= cbVertex)
|
|
{
|
|
/* Overflow */
|
|
return FALSE;
|
|
}
|
|
|
|
/* Allocate a kernel mode buffer */
|
|
SafeVertex = ExAllocatePoolWithTag(PagedPool, cbVertex + cbMesh, TAG_SHAPE);
|
|
if (!SafeVertex)
|
|
{
|
|
SetLastWin32Error(ERROR_NOT_ENOUGH_MEMORY);
|
|
return FALSE;
|
|
}
|
|
|
|
SafeMesh = (PVOID)((ULONG_PTR)SafeVertex + cbVertex);
|
|
|
|
/* Copy the parameters to kernel mode */
|
|
_SEH2_TRY
|
|
{
|
|
ProbeForRead(pVertex, cbVertex, 1);
|
|
ProbeForRead(pMesh, cbMesh, 1);
|
|
RtlCopyMemory(SafeVertex, pVertex, cbVertex);
|
|
RtlCopyMemory(SafeMesh, pMesh, cbMesh);
|
|
}
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
|
{
|
|
ExFreePoolWithTag(SafeVertex, TAG_SHAPE);
|
|
SetLastNtError(_SEH2_GetExceptionCode());
|
|
_SEH2_YIELD(return FALSE;)
|
|
}
|
|
_SEH2_END;
|
|
|
|
/* Call the internal function */
|
|
bRet = GreGradientFill(hdc, SafeVertex, nVertex, SafeMesh, nMesh, ulMode);
|
|
|
|
/* Cleanup and return result */
|
|
ExFreePoolWithTag(SafeVertex, TAG_SHAPE);
|
|
return bRet;
|
|
}
|
|
|
|
BOOL APIENTRY
|
|
NtGdiExtFloodFill(
|
|
HDC hDC,
|
|
INT XStart,
|
|
INT YStart,
|
|
COLORREF Color,
|
|
UINT FillType)
|
|
{
|
|
PDC dc;
|
|
PDC_ATTR pdcattr;
|
|
SURFACE *psurf = NULL;
|
|
HPALETTE hpal;
|
|
PPALETTE ppal;
|
|
EXLATEOBJ exlo;
|
|
BOOL Ret = FALSE;
|
|
RECTL DestRect;
|
|
POINTL Pt;
|
|
ULONG ConvColor;
|
|
|
|
dc = DC_LockDc(hDC);
|
|
if (!dc)
|
|
{
|
|
SetLastWin32Error(ERROR_INVALID_HANDLE);
|
|
return FALSE;
|
|
}
|
|
if (dc->dctype == DC_TYPE_INFO)
|
|
{
|
|
DC_UnlockDc(dc);
|
|
/* Yes, Windows really returns TRUE in this case */
|
|
return TRUE;
|
|
}
|
|
|
|
pdcattr = dc->pdcattr;
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_FILL | DC_BRUSH_DIRTY))
|
|
DC_vUpdateFillBrush(dc);
|
|
|
|
if (pdcattr->ulDirty_ & (DIRTY_LINE | DC_PEN_DIRTY))
|
|
DC_vUpdateLineBrush(dc);
|
|
|
|
Pt.x = XStart;
|
|
Pt.y = YStart;
|
|
IntLPtoDP(dc, (LPPOINT)&Pt, 1);
|
|
|
|
Ret = NtGdiPtInRegion(dc->rosdc.hGCClipRgn, Pt.x, Pt.y);
|
|
if (Ret)
|
|
IntGdiGetRgnBox(dc->rosdc.hGCClipRgn,(LPRECT)&DestRect);
|
|
else
|
|
goto cleanup;
|
|
|
|
psurf = dc->dclevel.pSurface;
|
|
if (!psurf)
|
|
{
|
|
Ret = FALSE;
|
|
goto cleanup;
|
|
}
|
|
|
|
hpal = dc->dclevel.pSurface->hDIBPalette;
|
|
if (!hpal) hpal = pPrimarySurface->devinfo.hpalDefault;
|
|
ppal = PALETTE_ShareLockPalette(hpal);
|
|
|
|
EXLATEOBJ_vInitialize(&exlo, &gpalRGB, ppal, 0, 0xffffff, 0);
|
|
|
|
/* Only solid fills supported for now
|
|
* How to support pattern brushes and non standard surfaces (not offering dib functions):
|
|
* Version a (most likely slow): call DrvPatBlt for every pixel
|
|
* Version b: create a flood mask and let MaskBlt blit a masked brush */
|
|
ConvColor = XLATEOBJ_iXlate(&exlo.xlo, Color);
|
|
Ret = DIB_XXBPP_FloodFillSolid(&psurf->SurfObj, &dc->eboFill.BrushObject, &DestRect, &Pt, ConvColor, FillType);
|
|
|
|
EXLATEOBJ_vCleanup(&exlo);
|
|
PALETTE_ShareUnlockPalette(ppal);
|
|
|
|
cleanup:
|
|
DC_UnlockDc(dc);
|
|
return Ret;
|
|
}
|
|
|
|
/* EOF */
|