Windows2003-3790/windows/advcore/gdiplus/engine/gpmf3216/polygons.c
2020-09-30 16:53:55 +02:00

177 lines
4.8 KiB
C

/*****************************************************************************
*
* polygons - Entry points for Win32 to Win 16 converter
*
* Date: 7/1/91
* Author: Jeffrey Newman (c-jeffn)
*
* Copyright 1991 Microsoft Corp
*****************************************************************************/
#include "precomp.h"
#pragma hdrstop
/***************************************************************************
* PolyPolygon - Win32 to Win16 Metafile Converter Entry Point
**************************************************************************/
BOOL WINAPI DoPolyPolygon
(
PLOCALDC pLocalDC,
PPOINTL pptl,
PDWORD pcptl,
DWORD cptl,
DWORD ccptl,
BOOL transform
)
{
BOOL b;
PWORD pcptlBuff = (PWORD) NULL;
PPOINTL pptlBuff = (PPOINTL) NULL;
PPOINTL pptlSrc, pptlDst;
DWORD i, cptlMax, cptlNeed, cptli;
// If we're recording the drawing orders for a path
// then just pass the drawing order to the helper DC.
// Do not emit any Win16 drawing orders.
if (pLocalDC->flags & RECORDING_PATH)
{
if (pfnSetVirtualResolution == NULL)
{
bXformWorkhorse(pptl, cptl, &pLocalDC->xformRWorldToRDev);
}
return(PolyPolygon(pLocalDC->hdcHelper, (LPPOINT) pptl, (LPINT) pcptl, (INT) ccptl));
}
// NOTE: There is a semantic between the Win32 PolyPolygon and
// the Win16 PolyPolygon. Win32 will close each polygon, Win16
// will not. As a result, we have to insert points as necessary
// to make the polygons closed. We cannot use multiple polygons
// to replace a single PolyPolygon because they are different if
// the polygons overlap and the polyfill mode is winding.
// If there are not verrics just return TRUE.
if (ccptl == 0)
return(TRUE) ;
b = FALSE; // assume failure
// Compute the maximum size of the temporary point array required
// to create closed PolyPolygon in win16.
cptlMax = cptl + ccptl;
// Allocate a buffer for the temporary point array.
pptlBuff = (PPOINTL) LocalAlloc(LMEM_FIXED, cptlMax * sizeof(POINTL)) ;
if (!pptlBuff)
{
PUTS("MF3216: DoPolyPolygon, LocalAlloc failed\n") ;
goto exit;
}
// Allocate a buffer for the new polycount array and make a copy
// of the old array.
pcptlBuff = (PWORD) LocalAlloc(LMEM_FIXED, ccptl * sizeof(WORD)) ;
if (!pcptlBuff)
{
PUTS("MF3216: DoPolyPolygon, LocalAlloc failed\n") ;
goto exit;
}
for (i = 0; i < ccptl; i++)
pcptlBuff[i] = (WORD) pcptl[i];
// Insert the points and update the polycount as necessary.
pptlDst = pptlBuff;
pptlSrc = pptl;
cptlNeed = cptl;
for (i = 0; i < ccptl; i++)
{
cptli = pcptl[i];
if (cptli < 2)
goto exit;
RtlCopyMemory(pptlDst, pptlSrc, cptli * sizeof(POINTL)) ;
if (pptlDst[0].x != pptlDst[cptli - 1].x
|| pptlDst[0].y != pptlDst[cptli - 1].y)
{
pptlDst[cptli] = pptlDst[0];
pptlDst++;
cptlNeed++;
pcptlBuff[i]++;
}
pptlSrc += cptli;
pptlDst += cptli;
}
// The Win16 poly record is limited to 64K points.
// Need to check this limit.
if (cptlNeed > (DWORD) (WORD) MAXWORD)
{
PUTS("MF3216: DoPolyPolygon, Too many point in poly array\n") ;
SetLastError(ERROR_NOT_ENOUGH_MEMORY) ;
goto exit ;
}
// Do the transformations.
if (transform)
{
if (!bXformRWorldToPPage(pLocalDC, pptlBuff, cptlNeed))
goto exit;
}
// Compress the POINTLs to POINTSs
vCompressPoints(pptlBuff, cptlNeed) ;
// Call the Win16 routine to emit the PolyPolygon to the metafile.
if (ccptl == 1)
{
b = bEmitWin16Poly(pLocalDC, (PPOINTS) pptlBuff, (WORD)cptlNeed, META_POLYGON);
}
else
{
b = bEmitWin16PolyPolygon(pLocalDC, (PPOINTS) pptlBuff,
pcptlBuff, (WORD) cptlNeed, (WORD) ccptl);
}
exit:
// Free the memory.
if (pptlBuff)
if (LocalFree(pptlBuff))
ASSERTGDI(FALSE, "MF3216: DoPolyPolygon, LocalFree failed");
if (pcptlBuff)
if (LocalFree(pcptlBuff))
ASSERTGDI(FALSE, "MF3216: DoPolyPolygon, LocalFree failed");
return(b) ;
}
/***************************************************************************
* SetPolyFillMode - Win32 to Win16 Metafile Converter Entry Point
**************************************************************************/
BOOL WINAPI DoSetPolyFillMode
(
PLOCALDC pLocalDC,
DWORD iPolyFillMode
)
{
BOOL b ;
// Emit the Win16 metafile drawing order.
b = bEmitWin16SetPolyFillMode(pLocalDC, LOWORD(iPolyFillMode)) ;
return(b) ;
}