| /* | 
 |  * Enhanced metafile functions | 
 |  * Copyright 1998 Douglas Ridgway | 
 |  *           1999 Huw D M Davies | 
 |  * | 
 |  * This library is free software; you can redistribute it and/or | 
 |  * modify it under the terms of the GNU Lesser General Public | 
 |  * License as published by the Free Software Foundation; either | 
 |  * version 2.1 of the License, or (at your option) any later version. | 
 |  * | 
 |  * This library 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 | 
 |  * Lesser General Public License for more details. | 
 |  * | 
 |  * You should have received a copy of the GNU Lesser General Public | 
 |  * License along with this library; if not, write to the Free Software | 
 |  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA | 
 |  * | 
 |  * NOTES: | 
 |  * | 
 |  * The enhanced format consists of the following elements: | 
 |  * | 
 |  *    A header | 
 |  *    A table of handles to GDI objects | 
 |  *    An array of metafile records | 
 |  *    A private palette | 
 |  * | 
 |  * | 
 |  *  The standard format consists of a header and an array of metafile records. | 
 |  * | 
 |  */ | 
 |  | 
 | #include "config.h" | 
 | #include "wine/port.h" | 
 |  | 
 | #include <stdarg.h> | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include <assert.h> | 
 | #include "windef.h" | 
 | #include "winbase.h" | 
 | #include "wingdi.h" | 
 | #include "winnls.h" | 
 | #include "winerror.h" | 
 | #include "gdi_private.h" | 
 | #include "wine/debug.h" | 
 |  | 
 | WINE_DEFAULT_DEBUG_CHANNEL(enhmetafile); | 
 |  | 
 | typedef struct | 
 | { | 
 |     GDIOBJHDR      header; | 
 |     ENHMETAHEADER  *emh; | 
 |     BOOL           on_disk;   /* true if metafile is on disk */ | 
 | } ENHMETAFILEOBJ; | 
 |  | 
 | static const struct emr_name { | 
 |     DWORD type; | 
 |     const char *name; | 
 | } emr_names[] = { | 
 | #define X(p) {p, #p} | 
 | X(EMR_HEADER), | 
 | X(EMR_POLYBEZIER), | 
 | X(EMR_POLYGON), | 
 | X(EMR_POLYLINE), | 
 | X(EMR_POLYBEZIERTO), | 
 | X(EMR_POLYLINETO), | 
 | X(EMR_POLYPOLYLINE), | 
 | X(EMR_POLYPOLYGON), | 
 | X(EMR_SETWINDOWEXTEX), | 
 | X(EMR_SETWINDOWORGEX), | 
 | X(EMR_SETVIEWPORTEXTEX), | 
 | X(EMR_SETVIEWPORTORGEX), | 
 | X(EMR_SETBRUSHORGEX), | 
 | X(EMR_EOF), | 
 | X(EMR_SETPIXELV), | 
 | X(EMR_SETMAPPERFLAGS), | 
 | X(EMR_SETMAPMODE), | 
 | X(EMR_SETBKMODE), | 
 | X(EMR_SETPOLYFILLMODE), | 
 | X(EMR_SETROP2), | 
 | X(EMR_SETSTRETCHBLTMODE), | 
 | X(EMR_SETTEXTALIGN), | 
 | X(EMR_SETCOLORADJUSTMENT), | 
 | X(EMR_SETTEXTCOLOR), | 
 | X(EMR_SETBKCOLOR), | 
 | X(EMR_OFFSETCLIPRGN), | 
 | X(EMR_MOVETOEX), | 
 | X(EMR_SETMETARGN), | 
 | X(EMR_EXCLUDECLIPRECT), | 
 | X(EMR_INTERSECTCLIPRECT), | 
 | X(EMR_SCALEVIEWPORTEXTEX), | 
 | X(EMR_SCALEWINDOWEXTEX), | 
 | X(EMR_SAVEDC), | 
 | X(EMR_RESTOREDC), | 
 | X(EMR_SETWORLDTRANSFORM), | 
 | X(EMR_MODIFYWORLDTRANSFORM), | 
 | X(EMR_SELECTOBJECT), | 
 | X(EMR_CREATEPEN), | 
 | X(EMR_CREATEBRUSHINDIRECT), | 
 | X(EMR_DELETEOBJECT), | 
 | X(EMR_ANGLEARC), | 
 | X(EMR_ELLIPSE), | 
 | X(EMR_RECTANGLE), | 
 | X(EMR_ROUNDRECT), | 
 | X(EMR_ARC), | 
 | X(EMR_CHORD), | 
 | X(EMR_PIE), | 
 | X(EMR_SELECTPALETTE), | 
 | X(EMR_CREATEPALETTE), | 
 | X(EMR_SETPALETTEENTRIES), | 
 | X(EMR_RESIZEPALETTE), | 
 | X(EMR_REALIZEPALETTE), | 
 | X(EMR_EXTFLOODFILL), | 
 | X(EMR_LINETO), | 
 | X(EMR_ARCTO), | 
 | X(EMR_POLYDRAW), | 
 | X(EMR_SETARCDIRECTION), | 
 | X(EMR_SETMITERLIMIT), | 
 | X(EMR_BEGINPATH), | 
 | X(EMR_ENDPATH), | 
 | X(EMR_CLOSEFIGURE), | 
 | X(EMR_FILLPATH), | 
 | X(EMR_STROKEANDFILLPATH), | 
 | X(EMR_STROKEPATH), | 
 | X(EMR_FLATTENPATH), | 
 | X(EMR_WIDENPATH), | 
 | X(EMR_SELECTCLIPPATH), | 
 | X(EMR_ABORTPATH), | 
 | X(EMR_GDICOMMENT), | 
 | X(EMR_FILLRGN), | 
 | X(EMR_FRAMERGN), | 
 | X(EMR_INVERTRGN), | 
 | X(EMR_PAINTRGN), | 
 | X(EMR_EXTSELECTCLIPRGN), | 
 | X(EMR_BITBLT), | 
 | X(EMR_STRETCHBLT), | 
 | X(EMR_MASKBLT), | 
 | X(EMR_PLGBLT), | 
 | X(EMR_SETDIBITSTODEVICE), | 
 | X(EMR_STRETCHDIBITS), | 
 | X(EMR_EXTCREATEFONTINDIRECTW), | 
 | X(EMR_EXTTEXTOUTA), | 
 | X(EMR_EXTTEXTOUTW), | 
 | X(EMR_POLYBEZIER16), | 
 | X(EMR_POLYGON16), | 
 | X(EMR_POLYLINE16), | 
 | X(EMR_POLYBEZIERTO16), | 
 | X(EMR_POLYLINETO16), | 
 | X(EMR_POLYPOLYLINE16), | 
 | X(EMR_POLYPOLYGON16), | 
 | X(EMR_POLYDRAW16), | 
 | X(EMR_CREATEMONOBRUSH), | 
 | X(EMR_CREATEDIBPATTERNBRUSHPT), | 
 | X(EMR_EXTCREATEPEN), | 
 | X(EMR_POLYTEXTOUTA), | 
 | X(EMR_POLYTEXTOUTW), | 
 | X(EMR_SETICMMODE), | 
 | X(EMR_CREATECOLORSPACE), | 
 | X(EMR_SETCOLORSPACE), | 
 | X(EMR_DELETECOLORSPACE), | 
 | X(EMR_GLSRECORD), | 
 | X(EMR_GLSBOUNDEDRECORD), | 
 | X(EMR_PIXELFORMAT), | 
 | X(EMR_DRAWESCAPE), | 
 | X(EMR_EXTESCAPE), | 
 | X(EMR_STARTDOC), | 
 | X(EMR_SMALLTEXTOUT), | 
 | X(EMR_FORCEUFIMAPPING), | 
 | X(EMR_NAMEDESCAPE), | 
 | X(EMR_COLORCORRECTPALETTE), | 
 | X(EMR_SETICMPROFILEA), | 
 | X(EMR_SETICMPROFILEW), | 
 | X(EMR_ALPHABLEND), | 
 | X(EMR_SETLAYOUT), | 
 | X(EMR_TRANSPARENTBLT), | 
 | X(EMR_RESERVED_117), | 
 | X(EMR_GRADIENTFILL), | 
 | X(EMR_SETLINKEDUFI), | 
 | X(EMR_SETTEXTJUSTIFICATION), | 
 | X(EMR_COLORMATCHTOTARGETW), | 
 | X(EMR_CREATECOLORSPACEW) | 
 | #undef X | 
 | }; | 
 |  | 
 | /**************************************************************************** | 
 |  *         get_emr_name | 
 |  */ | 
 | static const char *get_emr_name(DWORD type) | 
 | { | 
 |     unsigned int i; | 
 |     for(i = 0; i < sizeof(emr_names) / sizeof(emr_names[0]); i++) | 
 |         if(type == emr_names[i].type) return emr_names[i].name; | 
 |     TRACE("Unknown record type %d\n", type); | 
 |    return NULL; | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *          is_dib_monochrome | 
 |  * | 
 |  * Returns whether a DIB can be converted to a monochrome DDB. | 
 |  * | 
 |  * A DIB can be converted if its color table contains only black and | 
 |  * white. Black must be the first color in the color table. | 
 |  * | 
 |  * Note : If the first color in the color table is white followed by | 
 |  *        black, we can't convert it to a monochrome DDB with | 
 |  *        SetDIBits, because black and white would be inverted. | 
 |  */ | 
 | static inline BOOL is_dib_monochrome( const BITMAPINFO* info ) | 
 | { | 
 |     if (info->bmiHeader.biBitCount != 1) return FALSE; | 
 |  | 
 |     if (info->bmiHeader.biSize == sizeof(BITMAPCOREHEADER)) | 
 |     { | 
 |         const RGBTRIPLE *rgb = ((const BITMAPCOREINFO *) info)->bmciColors; | 
 |  | 
 |         /* Check if the first color is black */ | 
 |         if ((rgb->rgbtRed == 0) && (rgb->rgbtGreen == 0) && (rgb->rgbtBlue == 0)) | 
 |         { | 
 |             rgb++; | 
 |             /* Check if the second color is white */ | 
 |             return ((rgb->rgbtRed == 0xff) && (rgb->rgbtGreen == 0xff) | 
 |                  && (rgb->rgbtBlue == 0xff)); | 
 |         } | 
 |         else return FALSE; | 
 |     } | 
 |     else  /* assume BITMAPINFOHEADER */ | 
 |     { | 
 |         const RGBQUAD *rgb = info->bmiColors; | 
 |  | 
 |         /* Check if the first color is black */ | 
 |         if ((rgb->rgbRed == 0) && (rgb->rgbGreen == 0) && | 
 |             (rgb->rgbBlue == 0) && (rgb->rgbReserved == 0)) | 
 |         { | 
 |             rgb++; | 
 |  | 
 |             /* Check if the second color is white */ | 
 |             return ((rgb->rgbRed == 0xff) && (rgb->rgbGreen == 0xff) | 
 |                  && (rgb->rgbBlue == 0xff) && (rgb->rgbReserved == 0)); | 
 |         } | 
 |         else return FALSE; | 
 |     } | 
 | } | 
 |  | 
 | /**************************************************************************** | 
 |  *          EMF_Create_HENHMETAFILE | 
 |  */ | 
 | HENHMETAFILE EMF_Create_HENHMETAFILE(ENHMETAHEADER *emh, BOOL on_disk ) | 
 | { | 
 |     HENHMETAFILE hmf; | 
 |     ENHMETAFILEOBJ *metaObj; | 
 |  | 
 |     if (emh->iType != EMR_HEADER) | 
 |     { | 
 |         SetLastError(ERROR_INVALID_DATA); | 
 |         return 0; | 
 |     } | 
 |     if (emh->dSignature != ENHMETA_SIGNATURE || | 
 |         (emh->nBytes & 3)) /* refuse to load unaligned EMF as Windows does */ | 
 |     { | 
 |         WARN("Invalid emf header type 0x%08x sig 0x%08x.\n", | 
 |              emh->iType, emh->dSignature); | 
 |         return 0; | 
 |     } | 
 |  | 
 |     if (!(metaObj = HeapAlloc( GetProcessHeap(), 0, sizeof(*metaObj) ))) return 0; | 
 |  | 
 |     metaObj->emh = emh; | 
 |     metaObj->on_disk = on_disk; | 
 |  | 
 |     if (!(hmf = alloc_gdi_handle( &metaObj->header, OBJ_ENHMETAFILE, NULL ))) | 
 |         HeapFree( GetProcessHeap(), 0, metaObj ); | 
 |     return hmf; | 
 | } | 
 |  | 
 | /**************************************************************************** | 
 |  *          EMF_Delete_HENHMETAFILE | 
 |  */ | 
 | static BOOL EMF_Delete_HENHMETAFILE( HENHMETAFILE hmf ) | 
 | { | 
 |     ENHMETAFILEOBJ *metaObj = free_gdi_handle( hmf ); | 
 |  | 
 |     if(!metaObj) return FALSE; | 
 |  | 
 |     if(metaObj->on_disk) | 
 |         UnmapViewOfFile( metaObj->emh ); | 
 |     else | 
 |         HeapFree( GetProcessHeap(), 0, metaObj->emh ); | 
 |     return HeapFree( GetProcessHeap(), 0, metaObj ); | 
 | } | 
 |  | 
 | /****************************************************************** | 
 |  *         EMF_GetEnhMetaHeader | 
 |  * | 
 |  * Returns ptr to ENHMETAHEADER associated with HENHMETAFILE | 
 |  */ | 
 | static ENHMETAHEADER *EMF_GetEnhMetaHeader( HENHMETAFILE hmf ) | 
 | { | 
 |     ENHMETAHEADER *ret = NULL; | 
 |     ENHMETAFILEOBJ *metaObj = GDI_GetObjPtr( hmf, OBJ_ENHMETAFILE ); | 
 |     TRACE("hmf %p -> enhmetaObj %p\n", hmf, metaObj); | 
 |     if (metaObj) | 
 |     { | 
 |         ret = metaObj->emh; | 
 |         GDI_ReleaseObj( hmf ); | 
 |     } | 
 |     return ret; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *         EMF_GetEnhMetaFile | 
 |  * | 
 |  */ | 
 | static HENHMETAFILE EMF_GetEnhMetaFile( HANDLE hFile ) | 
 | { | 
 |     ENHMETAHEADER *emh; | 
 |     HANDLE hMapping; | 
 |     HENHMETAFILE hemf; | 
 |  | 
 |     hMapping = CreateFileMappingA( hFile, NULL, PAGE_READONLY, 0, 0, NULL ); | 
 |     emh = MapViewOfFile( hMapping, FILE_MAP_READ, 0, 0, 0 ); | 
 |     CloseHandle( hMapping ); | 
 |  | 
 |     if (!emh) return 0; | 
 |  | 
 |     hemf = EMF_Create_HENHMETAFILE( emh, TRUE ); | 
 |     if (!hemf) | 
 |         UnmapViewOfFile( emh ); | 
 |     return hemf; | 
 | } | 
 |  | 
 |  | 
 | /***************************************************************************** | 
 |  *          GetEnhMetaFileA (GDI32.@) | 
 |  * | 
 |  * | 
 |  */ | 
 | HENHMETAFILE WINAPI GetEnhMetaFileA( | 
 | 	     LPCSTR lpszMetaFile  /* [in] filename of enhanced metafile */ | 
 |     ) | 
 | { | 
 |     HENHMETAFILE hmf; | 
 |     HANDLE hFile; | 
 |  | 
 |     hFile = CreateFileA(lpszMetaFile, GENERIC_READ, FILE_SHARE_READ, 0, | 
 | 			OPEN_EXISTING, 0, 0); | 
 |     if (hFile == INVALID_HANDLE_VALUE) { | 
 |         WARN("could not open %s\n", lpszMetaFile); | 
 | 	return 0; | 
 |     } | 
 |     hmf = EMF_GetEnhMetaFile( hFile ); | 
 |     CloseHandle( hFile ); | 
 |     return hmf; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *          GetEnhMetaFileW  (GDI32.@) | 
 |  */ | 
 | HENHMETAFILE WINAPI GetEnhMetaFileW( | 
 |              LPCWSTR lpszMetaFile)  /* [in] filename of enhanced metafile */ | 
 | { | 
 |     HENHMETAFILE hmf; | 
 |     HANDLE hFile; | 
 |  | 
 |     hFile = CreateFileW(lpszMetaFile, GENERIC_READ, FILE_SHARE_READ, 0, | 
 | 			OPEN_EXISTING, 0, 0); | 
 |     if (hFile == INVALID_HANDLE_VALUE) { | 
 |         WARN("could not open %s\n", debugstr_w(lpszMetaFile)); | 
 | 	return 0; | 
 |     } | 
 |     hmf = EMF_GetEnhMetaFile( hFile ); | 
 |     CloseHandle( hFile ); | 
 |     return hmf; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *        GetEnhMetaFileHeader  (GDI32.@) | 
 |  * | 
 |  * Retrieves the record containing the header for the specified | 
 |  * enhanced-format metafile. | 
 |  * | 
 |  * RETURNS | 
 |  *  If buf is NULL, returns the size of buffer required. | 
 |  *  Otherwise, copy up to bufsize bytes of enhanced metafile header into | 
 |  *  buf. | 
 |  */ | 
 | UINT WINAPI GetEnhMetaFileHeader( | 
 |        HENHMETAFILE hmf,   /* [in] enhanced metafile */ | 
 |        UINT bufsize,       /* [in] size of buffer */ | 
 |        LPENHMETAHEADER buf /* [out] buffer */ | 
 |     ) | 
 | { | 
 |     LPENHMETAHEADER emh; | 
 |     UINT size; | 
 |  | 
 |     emh = EMF_GetEnhMetaHeader(hmf); | 
 |     if(!emh) return FALSE; | 
 |     size = emh->nSize; | 
 |     if (!buf) return size; | 
 |     size = min(size, bufsize); | 
 |     memmove(buf, emh, size); | 
 |     return size; | 
 | } | 
 |  | 
 |  | 
 | /***************************************************************************** | 
 |  *          GetEnhMetaFileDescriptionA  (GDI32.@) | 
 |  * | 
 |  * See GetEnhMetaFileDescriptionW. | 
 |  */ | 
 | UINT WINAPI GetEnhMetaFileDescriptionA( | 
 |        HENHMETAFILE hmf, /* [in] enhanced metafile */ | 
 |        UINT size,        /* [in] size of buf */ | 
 |        LPSTR buf         /* [out] buffer to receive description */ | 
 |     ) | 
 | { | 
 |      LPENHMETAHEADER emh = EMF_GetEnhMetaHeader(hmf); | 
 |      DWORD len; | 
 |      WCHAR *descrW; | 
 |  | 
 |      if(!emh) return FALSE; | 
 |      if(emh->nDescription == 0 || emh->offDescription == 0) return 0; | 
 |      descrW = (WCHAR *) ((char *) emh + emh->offDescription); | 
 |      len = WideCharToMultiByte( CP_ACP, 0, descrW, emh->nDescription, NULL, 0, NULL, NULL ); | 
 |  | 
 |      if (!buf || !size ) return len; | 
 |  | 
 |      len = min( size, len ); | 
 |      WideCharToMultiByte( CP_ACP, 0, descrW, emh->nDescription, buf, len, NULL, NULL ); | 
 |      return len; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *          GetEnhMetaFileDescriptionW  (GDI32.@) | 
 |  * | 
 |  *  Copies the description string of an enhanced metafile into a buffer | 
 |  *  _buf_. | 
 |  * | 
 |  * RETURNS | 
 |  *  If _buf_ is NULL, returns size of _buf_ required. Otherwise, returns | 
 |  *  number of characters copied. | 
 |  */ | 
 | UINT WINAPI GetEnhMetaFileDescriptionW( | 
 |        HENHMETAFILE hmf, /* [in] enhanced metafile */ | 
 |        UINT size,        /* [in] size of buf */ | 
 |        LPWSTR buf        /* [out] buffer to receive description */ | 
 |     ) | 
 | { | 
 |      LPENHMETAHEADER emh = EMF_GetEnhMetaHeader(hmf); | 
 |  | 
 |      if(!emh) return FALSE; | 
 |      if(emh->nDescription == 0 || emh->offDescription == 0) return 0; | 
 |      if (!buf || !size ) return emh->nDescription; | 
 |  | 
 |      memmove(buf, (char *) emh + emh->offDescription, min(size,emh->nDescription)*sizeof(WCHAR)); | 
 |      return min(size, emh->nDescription); | 
 | } | 
 |  | 
 | /**************************************************************************** | 
 |  *    SetEnhMetaFileBits (GDI32.@) | 
 |  * | 
 |  *  Creates an enhanced metafile by copying _bufsize_ bytes from _buf_. | 
 |  */ | 
 | HENHMETAFILE WINAPI SetEnhMetaFileBits(UINT bufsize, const BYTE *buf) | 
 | { | 
 |     ENHMETAHEADER *emh = HeapAlloc( GetProcessHeap(), 0, bufsize ); | 
 |     memmove(emh, buf, bufsize); | 
 |     return EMF_Create_HENHMETAFILE( emh, FALSE ); | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *  GetEnhMetaFileBits (GDI32.@) | 
 |  * | 
 |  */ | 
 | UINT WINAPI GetEnhMetaFileBits( | 
 |     HENHMETAFILE hmf, | 
 |     UINT bufsize, | 
 |     LPBYTE buf | 
 | ) | 
 | { | 
 |     LPENHMETAHEADER emh = EMF_GetEnhMetaHeader( hmf ); | 
 |     UINT size; | 
 |  | 
 |     if(!emh) return 0; | 
 |  | 
 |     size = emh->nBytes; | 
 |     if( buf == NULL ) return size; | 
 |  | 
 |     size = min( size, bufsize ); | 
 |     memmove(buf, emh, size); | 
 |     return size; | 
 | } | 
 |  | 
 | typedef struct EMF_dc_state | 
 | { | 
 |     INT   mode; | 
 |     XFORM world_transform; | 
 |     INT   wndOrgX; | 
 |     INT   wndOrgY; | 
 |     INT   wndExtX; | 
 |     INT   wndExtY; | 
 |     INT   vportOrgX; | 
 |     INT   vportOrgY; | 
 |     INT   vportExtX; | 
 |     INT   vportExtY; | 
 |     struct EMF_dc_state *next; | 
 | } EMF_dc_state; | 
 |  | 
 | typedef struct enum_emh_data | 
 | { | 
 |     XFORM init_transform; | 
 |     EMF_dc_state state; | 
 |     INT save_level; | 
 |     EMF_dc_state *saved_state; | 
 | } enum_emh_data; | 
 |  | 
 | #define ENUM_GET_PRIVATE_DATA(ht) \ | 
 |     ((enum_emh_data*)(((unsigned char*)(ht))-sizeof (enum_emh_data))) | 
 |  | 
 | #define WIDTH(rect) ( (rect).right - (rect).left ) | 
 | #define HEIGHT(rect) ( (rect).bottom - (rect).top ) | 
 |  | 
 | #define IS_WIN9X() (GetVersion()&0x80000000) | 
 |  | 
 | static void EMF_Update_MF_Xform(HDC hdc, const enum_emh_data *info) | 
 | { | 
 |     XFORM mapping_mode_trans, final_trans; | 
 |     double scaleX, scaleY; | 
 |  | 
 |     scaleX = (double)info->state.vportExtX / (double)info->state.wndExtX; | 
 |     scaleY = (double)info->state.vportExtY / (double)info->state.wndExtY; | 
 |     mapping_mode_trans.eM11 = scaleX; | 
 |     mapping_mode_trans.eM12 = 0.0; | 
 |     mapping_mode_trans.eM21 = 0.0; | 
 |     mapping_mode_trans.eM22 = scaleY; | 
 |     mapping_mode_trans.eDx  = (double)info->state.vportOrgX - scaleX * (double)info->state.wndOrgX; | 
 |     mapping_mode_trans.eDy  = (double)info->state.vportOrgY - scaleY * (double)info->state.wndOrgY; | 
 |  | 
 |     CombineTransform(&final_trans, &info->state.world_transform, &mapping_mode_trans); | 
 |     CombineTransform(&final_trans, &final_trans, &info->init_transform); | 
 |   | 
 |     if (!SetWorldTransform(hdc, &final_trans)) | 
 |     { | 
 |         ERR("World transform failed!\n"); | 
 |     } | 
 | } | 
 |  | 
 | static void EMF_RestoreDC( enum_emh_data *info, INT level ) | 
 | { | 
 |     if (abs(level) > info->save_level || level == 0) return; | 
 |  | 
 |     if (level < 0) level = info->save_level + level + 1; | 
 |  | 
 |     while (info->save_level >= level) | 
 |     { | 
 |         EMF_dc_state *state = info->saved_state; | 
 |         info->saved_state = state->next; | 
 |         state->next = NULL; | 
 |         if (--info->save_level < level) | 
 |             info->state = *state; | 
 |         HeapFree( GetProcessHeap(), 0, state ); | 
 |     } | 
 | } | 
 |  | 
 | static void EMF_SaveDC( enum_emh_data *info ) | 
 | { | 
 |     EMF_dc_state *state = HeapAlloc( GetProcessHeap(), 0, sizeof(*state)); | 
 |     if (state) | 
 |     { | 
 |         *state = info->state; | 
 |         state->next = info->saved_state; | 
 |         info->saved_state = state; | 
 |         info->save_level++; | 
 |         TRACE("save_level %d\n", info->save_level); | 
 |     } | 
 | } | 
 |  | 
 | static void EMF_SetMapMode(HDC hdc, enum_emh_data *info) | 
 | { | 
 |     INT horzSize = GetDeviceCaps( hdc, HORZSIZE ); | 
 |     INT vertSize = GetDeviceCaps( hdc, VERTSIZE ); | 
 |     INT horzRes  = GetDeviceCaps( hdc, HORZRES ); | 
 |     INT vertRes  = GetDeviceCaps( hdc, VERTRES ); | 
 |  | 
 |     TRACE("%d\n", info->state.mode); | 
 |  | 
 |     switch(info->state.mode) | 
 |     { | 
 |     case MM_TEXT: | 
 |         info->state.wndExtX   = 1; | 
 |         info->state.wndExtY   = 1; | 
 |         info->state.vportExtX = 1; | 
 |         info->state.vportExtY = 1; | 
 |         break; | 
 |     case MM_LOMETRIC: | 
 |     case MM_ISOTROPIC: | 
 |         info->state.wndExtX   = horzSize * 10; | 
 |         info->state.wndExtY   = vertSize * 10; | 
 |         info->state.vportExtX = horzRes; | 
 |         info->state.vportExtY = -vertRes; | 
 |         break; | 
 |     case MM_HIMETRIC: | 
 |         info->state.wndExtX   = horzSize * 100; | 
 |         info->state.wndExtY   = vertSize * 100; | 
 |         info->state.vportExtX = horzRes; | 
 |         info->state.vportExtY = -vertRes; | 
 |         break; | 
 |     case MM_LOENGLISH: | 
 |         info->state.wndExtX   = MulDiv(1000, horzSize, 254); | 
 |         info->state.wndExtY   = MulDiv(1000, vertSize, 254); | 
 |         info->state.vportExtX = horzRes; | 
 |         info->state.vportExtY = -vertRes; | 
 |         break; | 
 |     case MM_HIENGLISH: | 
 |         info->state.wndExtX   = MulDiv(10000, horzSize, 254); | 
 |         info->state.wndExtY   = MulDiv(10000, vertSize, 254); | 
 |         info->state.vportExtX = horzRes; | 
 |         info->state.vportExtY = -vertRes; | 
 |         break; | 
 |     case MM_TWIPS: | 
 |         info->state.wndExtX   = MulDiv(14400, horzSize, 254); | 
 |         info->state.wndExtY   = MulDiv(14400, vertSize, 254); | 
 |         info->state.vportExtX = horzRes; | 
 |         info->state.vportExtY = -vertRes; | 
 |         break; | 
 |     case MM_ANISOTROPIC: | 
 |         break; | 
 |     default: | 
 |         return; | 
 |     } | 
 | } | 
 |  | 
 | /*********************************************************************** | 
 |  *           EMF_FixIsotropic | 
 |  * | 
 |  * Fix viewport extensions for isotropic mode. | 
 |  */ | 
 |  | 
 | static void EMF_FixIsotropic(HDC hdc, enum_emh_data *info) | 
 | { | 
 |     double xdim = fabs((double)info->state.vportExtX * GetDeviceCaps( hdc, HORZSIZE ) / | 
 |                   (GetDeviceCaps( hdc, HORZRES ) * info->state.wndExtX)); | 
 |     double ydim = fabs((double)info->state.vportExtY * GetDeviceCaps( hdc, VERTSIZE ) / | 
 |                   (GetDeviceCaps( hdc, VERTRES ) * info->state.wndExtY)); | 
 |  | 
 |     if (xdim > ydim) | 
 |     { | 
 |         INT mincx = (info->state.vportExtX >= 0) ? 1 : -1; | 
 |         info->state.vportExtX = floor(info->state.vportExtX * ydim / xdim + 0.5); | 
 |         if (!info->state.vportExtX) info->state.vportExtX = mincx; | 
 |     } | 
 |     else | 
 |     { | 
 |         INT mincy = (info->state.vportExtY >= 0) ? 1 : -1; | 
 |         info->state.vportExtY = floor(info->state.vportExtY * xdim / ydim + 0.5); | 
 |         if (!info->state.vportExtY) info->state.vportExtY = mincy; | 
 |     } | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *       emr_produces_output | 
 |  * | 
 |  * Returns TRUE if the record type writes something to the dc.  Used by | 
 |  * PlayEnhMetaFileRecord to determine whether it needs to update the | 
 |  * dc's xform when in win9x mode. | 
 |  * | 
 |  * FIXME: need to test which records should be here. | 
 |  */ | 
 | static BOOL emr_produces_output(int type) | 
 | { | 
 |     switch(type) { | 
 |     case EMR_POLYBEZIER: | 
 |     case EMR_POLYGON: | 
 |     case EMR_POLYLINE: | 
 |     case EMR_POLYBEZIERTO: | 
 |     case EMR_POLYLINETO: | 
 |     case EMR_POLYPOLYLINE: | 
 |     case EMR_POLYPOLYGON: | 
 |     case EMR_SETPIXELV: | 
 |     case EMR_MOVETOEX: | 
 |     case EMR_EXCLUDECLIPRECT: | 
 |     case EMR_INTERSECTCLIPRECT: | 
 |     case EMR_SELECTOBJECT: | 
 |     case EMR_ANGLEARC: | 
 |     case EMR_ELLIPSE: | 
 |     case EMR_RECTANGLE: | 
 |     case EMR_ROUNDRECT: | 
 |     case EMR_ARC: | 
 |     case EMR_CHORD: | 
 |     case EMR_PIE: | 
 |     case EMR_EXTFLOODFILL: | 
 |     case EMR_LINETO: | 
 |     case EMR_ARCTO: | 
 |     case EMR_POLYDRAW: | 
 |     case EMR_GDICOMMENT: | 
 |     case EMR_FILLRGN: | 
 |     case EMR_FRAMERGN: | 
 |     case EMR_INVERTRGN: | 
 |     case EMR_PAINTRGN: | 
 |     case EMR_BITBLT: | 
 |     case EMR_STRETCHBLT: | 
 |     case EMR_MASKBLT: | 
 |     case EMR_PLGBLT: | 
 |     case EMR_SETDIBITSTODEVICE: | 
 |     case EMR_STRETCHDIBITS: | 
 |     case EMR_EXTTEXTOUTA: | 
 |     case EMR_EXTTEXTOUTW: | 
 |     case EMR_POLYBEZIER16: | 
 |     case EMR_POLYGON16: | 
 |     case EMR_POLYLINE16: | 
 |     case EMR_POLYBEZIERTO16: | 
 |     case EMR_POLYLINETO16: | 
 |     case EMR_POLYPOLYLINE16: | 
 |     case EMR_POLYPOLYGON16: | 
 |     case EMR_POLYDRAW16: | 
 |     case EMR_POLYTEXTOUTA: | 
 |     case EMR_POLYTEXTOUTW: | 
 |     case EMR_SMALLTEXTOUT: | 
 |     case EMR_ALPHABLEND: | 
 |     case EMR_TRANSPARENTBLT: | 
 |         return TRUE; | 
 |     default: | 
 |         return FALSE; | 
 |     } | 
 | } | 
 |  | 
 |  | 
 | /***************************************************************************** | 
 |  *           PlayEnhMetaFileRecord  (GDI32.@) | 
 |  * | 
 |  *  Render a single enhanced metafile record in the device context hdc. | 
 |  * | 
 |  *  RETURNS | 
 |  *    TRUE (non zero) on success, FALSE on error. | 
 |  *  BUGS | 
 |  *    Many unimplemented records. | 
 |  *    No error handling on record play failures (ie checking return codes) | 
 |  * | 
 |  * NOTES | 
 |  *    WinNT actually updates the current world transform in this function | 
 |  *     whereas Win9x does not. | 
 |  */ | 
 | BOOL WINAPI PlayEnhMetaFileRecord( | 
 |      HDC hdc,                   /* [in] device context in which to render EMF record */ | 
 |      LPHANDLETABLE handletable, /* [in] array of handles to be used in rendering record */ | 
 |      const ENHMETARECORD *mr,   /* [in] EMF record to render */ | 
 |      UINT handles               /* [in] size of handle array */ | 
 |      ) | 
 | { | 
 |   int type; | 
 |   RECT tmprc; | 
 |   enum_emh_data *info = ENUM_GET_PRIVATE_DATA(handletable); | 
 |  | 
 |   TRACE("hdc = %p, handletable = %p, record = %p, numHandles = %d\n", | 
 |         hdc, handletable, mr, handles); | 
 |   if (!mr) return FALSE; | 
 |  | 
 |   type = mr->iType; | 
 |  | 
 |   TRACE("record %s\n", get_emr_name(type)); | 
 |   switch(type) | 
 |     { | 
 |     case EMR_HEADER: | 
 |       break; | 
 |     case EMR_EOF: | 
 |       break; | 
 |     case EMR_GDICOMMENT: | 
 |       { | 
 |         const EMRGDICOMMENT *lpGdiComment = (const EMRGDICOMMENT *)mr; | 
 |         /* In an enhanced metafile, there can be both public and private GDI comments */ | 
 |         GdiComment( hdc, lpGdiComment->cbData, lpGdiComment->Data ); | 
 |         break; | 
 |       } | 
 |     case EMR_SETMAPMODE: | 
 |       { | 
 |         const EMRSETMAPMODE *pSetMapMode = (const EMRSETMAPMODE *)mr; | 
 |  | 
 |         if (info->state.mode == pSetMapMode->iMode && | 
 |             (info->state.mode == MM_ISOTROPIC || info->state.mode == MM_ANISOTROPIC)) | 
 |             break; | 
 |         info->state.mode = pSetMapMode->iMode; | 
 |         EMF_SetMapMode(hdc, info); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETBKMODE: | 
 |       { | 
 |         const EMRSETBKMODE *pSetBkMode = (const EMRSETBKMODE *)mr; | 
 | 	SetBkMode(hdc, pSetBkMode->iMode); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETBKCOLOR: | 
 |       { | 
 |         const EMRSETBKCOLOR *pSetBkColor = (const EMRSETBKCOLOR *)mr; | 
 | 	SetBkColor(hdc, pSetBkColor->crColor); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETPOLYFILLMODE: | 
 |       { | 
 |         const EMRSETPOLYFILLMODE *pSetPolyFillMode = (const EMRSETPOLYFILLMODE *)mr; | 
 | 	SetPolyFillMode(hdc, pSetPolyFillMode->iMode); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETROP2: | 
 |       { | 
 |         const EMRSETROP2 *pSetROP2 = (const EMRSETROP2 *)mr; | 
 | 	SetROP2(hdc, pSetROP2->iMode); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETSTRETCHBLTMODE: | 
 |       { | 
 | 	const EMRSETSTRETCHBLTMODE *pSetStretchBltMode = (const EMRSETSTRETCHBLTMODE *)mr; | 
 | 	SetStretchBltMode(hdc, pSetStretchBltMode->iMode); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETTEXTALIGN: | 
 |       { | 
 | 	const EMRSETTEXTALIGN *pSetTextAlign = (const EMRSETTEXTALIGN *)mr; | 
 | 	SetTextAlign(hdc, pSetTextAlign->iMode); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETTEXTCOLOR: | 
 |       { | 
 | 	const EMRSETTEXTCOLOR *pSetTextColor = (const EMRSETTEXTCOLOR *)mr; | 
 | 	SetTextColor(hdc, pSetTextColor->crColor); | 
 | 	break; | 
 |       } | 
 |     case EMR_SAVEDC: | 
 |       { | 
 |         if (SaveDC( hdc )) | 
 |             EMF_SaveDC( info ); | 
 | 	break; | 
 |       } | 
 |     case EMR_RESTOREDC: | 
 |       { | 
 | 	const EMRRESTOREDC *pRestoreDC = (const EMRRESTOREDC *)mr; | 
 |         TRACE("EMR_RESTORE: %d\n", pRestoreDC->iRelative); | 
 |         if (RestoreDC( hdc, pRestoreDC->iRelative )) | 
 |             EMF_RestoreDC( info, pRestoreDC->iRelative ); | 
 | 	break; | 
 |       } | 
 |     case EMR_INTERSECTCLIPRECT: | 
 |       { | 
 | 	const EMRINTERSECTCLIPRECT *pClipRect = (const EMRINTERSECTCLIPRECT *)mr; | 
 |         TRACE("EMR_INTERSECTCLIPRECT: rect %d,%d - %d, %d\n", | 
 |               pClipRect->rclClip.left, pClipRect->rclClip.top, | 
 |               pClipRect->rclClip.right, pClipRect->rclClip.bottom); | 
 |         IntersectClipRect(hdc, pClipRect->rclClip.left, pClipRect->rclClip.top, | 
 |                           pClipRect->rclClip.right, pClipRect->rclClip.bottom); | 
 | 	break; | 
 |       } | 
 |     case EMR_SELECTOBJECT: | 
 |       { | 
 | 	const EMRSELECTOBJECT *pSelectObject = (const EMRSELECTOBJECT *)mr; | 
 | 	if( pSelectObject->ihObject & 0x80000000 ) { | 
 | 	  /* High order bit is set - it's a stock object | 
 | 	   * Strip the high bit to get the index. | 
 | 	   * See MSDN article Q142319 | 
 | 	   */ | 
 | 	  SelectObject( hdc, GetStockObject( pSelectObject->ihObject & | 
 | 					     0x7fffffff ) ); | 
 | 	} else { | 
 | 	  /* High order bit wasn't set - not a stock object | 
 | 	   */ | 
 | 	      SelectObject( hdc, | 
 | 			(handletable->objectHandle)[pSelectObject->ihObject] ); | 
 | 	} | 
 | 	break; | 
 |       } | 
 |     case EMR_DELETEOBJECT: | 
 |       { | 
 | 	const EMRDELETEOBJECT *pDeleteObject = (const EMRDELETEOBJECT *)mr; | 
 | 	DeleteObject( (handletable->objectHandle)[pDeleteObject->ihObject]); | 
 | 	(handletable->objectHandle)[pDeleteObject->ihObject] = 0; | 
 | 	break; | 
 |       } | 
 |     case EMR_SETWINDOWORGEX: | 
 |       { | 
 |     	const EMRSETWINDOWORGEX *pSetWindowOrgEx = (const EMRSETWINDOWORGEX *)mr; | 
 |  | 
 |         info->state.wndOrgX = pSetWindowOrgEx->ptlOrigin.x; | 
 |         info->state.wndOrgY = pSetWindowOrgEx->ptlOrigin.y; | 
 |  | 
 |         TRACE("SetWindowOrgEx: %d,%d\n", info->state.wndOrgX, info->state.wndOrgY); | 
 |         break; | 
 |       } | 
 |     case EMR_SETWINDOWEXTEX: | 
 |       { | 
 | 	const EMRSETWINDOWEXTEX *pSetWindowExtEx = (const EMRSETWINDOWEXTEX *)mr; | 
 | 	 | 
 |         if (info->state.mode != MM_ISOTROPIC && info->state.mode != MM_ANISOTROPIC) | 
 | 	    break; | 
 |         info->state.wndExtX = pSetWindowExtEx->szlExtent.cx; | 
 |         info->state.wndExtY = pSetWindowExtEx->szlExtent.cy; | 
 |         if (info->state.mode == MM_ISOTROPIC) | 
 |             EMF_FixIsotropic(hdc, info); | 
 |  | 
 |         TRACE("SetWindowExtEx: %d,%d\n",info->state.wndExtX, info->state.wndExtY); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETVIEWPORTORGEX: | 
 |       { | 
 | 	const EMRSETVIEWPORTORGEX *pSetViewportOrgEx = (const EMRSETVIEWPORTORGEX *)mr; | 
 |  | 
 |         info->state.vportOrgX = pSetViewportOrgEx->ptlOrigin.x; | 
 |         info->state.vportOrgY = pSetViewportOrgEx->ptlOrigin.y; | 
 |         TRACE("SetViewportOrgEx: %d,%d\n", info->state.vportOrgX, info->state.vportOrgY); | 
 | 	break; | 
 |       } | 
 |     case EMR_SETVIEWPORTEXTEX: | 
 |       { | 
 | 	const EMRSETVIEWPORTEXTEX *pSetViewportExtEx = (const EMRSETVIEWPORTEXTEX *)mr; | 
 |  | 
 |         if (info->state.mode != MM_ISOTROPIC && info->state.mode != MM_ANISOTROPIC) | 
 | 	    break; | 
 |         info->state.vportExtX = pSetViewportExtEx->szlExtent.cx; | 
 |         info->state.vportExtY = pSetViewportExtEx->szlExtent.cy; | 
 |         if (info->state.mode == MM_ISOTROPIC) | 
 |             EMF_FixIsotropic(hdc, info); | 
 |         TRACE("SetViewportExtEx: %d,%d\n", info->state.vportExtX, info->state.vportExtY); | 
 | 	break; | 
 |       } | 
 |     case EMR_CREATEPEN: | 
 |       { | 
 | 	const EMRCREATEPEN *pCreatePen = (const EMRCREATEPEN *)mr; | 
 | 	(handletable->objectHandle)[pCreatePen->ihPen] = | 
 | 	  CreatePenIndirect(&pCreatePen->lopn); | 
 | 	break; | 
 |       } | 
 |     case EMR_EXTCREATEPEN: | 
 |       { | 
 | 	const EMREXTCREATEPEN *pPen = (const EMREXTCREATEPEN *)mr; | 
 | 	LOGBRUSH lb; | 
 | 	lb.lbStyle = pPen->elp.elpBrushStyle; | 
 | 	lb.lbColor = pPen->elp.elpColor; | 
 | 	lb.lbHatch = pPen->elp.elpHatch; | 
 |  | 
 | 	if(pPen->offBmi || pPen->offBits) | 
 | 	  FIXME("EMR_EXTCREATEPEN: Need to copy brush bitmap\n"); | 
 |  | 
 | 	(handletable->objectHandle)[pPen->ihPen] = | 
 | 	  ExtCreatePen(pPen->elp.elpPenStyle, pPen->elp.elpWidth, &lb, | 
 | 		       pPen->elp.elpNumEntries, pPen->elp.elpStyleEntry); | 
 | 	break; | 
 |       } | 
 |     case EMR_CREATEBRUSHINDIRECT: | 
 |       { | 
 | 	const EMRCREATEBRUSHINDIRECT *pBrush = (const EMRCREATEBRUSHINDIRECT *)mr; | 
 |         LOGBRUSH brush; | 
 |         brush.lbStyle = pBrush->lb.lbStyle; | 
 |         brush.lbColor = pBrush->lb.lbColor; | 
 |         brush.lbHatch = pBrush->lb.lbHatch; | 
 |         (handletable->objectHandle)[pBrush->ihBrush] = CreateBrushIndirect(&brush); | 
 | 	break; | 
 |       } | 
 |     case EMR_EXTCREATEFONTINDIRECTW: | 
 |       { | 
 | 	const EMREXTCREATEFONTINDIRECTW *pFont = (const EMREXTCREATEFONTINDIRECTW *)mr; | 
 | 	(handletable->objectHandle)[pFont->ihFont] = | 
 | 	  CreateFontIndirectW(&pFont->elfw.elfLogFont); | 
 | 	break; | 
 |       } | 
 |     case EMR_MOVETOEX: | 
 |       { | 
 | 	const EMRMOVETOEX *pMoveToEx = (const EMRMOVETOEX *)mr; | 
 | 	MoveToEx(hdc, pMoveToEx->ptl.x, pMoveToEx->ptl.y, NULL); | 
 | 	break; | 
 |       } | 
 |     case EMR_LINETO: | 
 |       { | 
 | 	const EMRLINETO *pLineTo = (const EMRLINETO *)mr; | 
 |         LineTo(hdc, pLineTo->ptl.x, pLineTo->ptl.y); | 
 | 	break; | 
 |       } | 
 |     case EMR_RECTANGLE: | 
 |       { | 
 | 	const EMRRECTANGLE *pRect = (const EMRRECTANGLE *)mr; | 
 | 	Rectangle(hdc, pRect->rclBox.left, pRect->rclBox.top, | 
 | 		  pRect->rclBox.right, pRect->rclBox.bottom); | 
 | 	break; | 
 |       } | 
 |     case EMR_ELLIPSE: | 
 |       { | 
 | 	const EMRELLIPSE *pEllipse = (const EMRELLIPSE *)mr; | 
 | 	Ellipse(hdc, pEllipse->rclBox.left, pEllipse->rclBox.top, | 
 | 		pEllipse->rclBox.right, pEllipse->rclBox.bottom); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYGON16: | 
 |       { | 
 | 	const EMRPOLYGON16 *pPoly = (const EMRPOLYGON16 *)mr; | 
 | 	/* Shouldn't use Polygon16 since pPoly->cpts is DWORD */ | 
 | 	POINT *pts = HeapAlloc( GetProcessHeap(), 0, | 
 | 				pPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPoly->cpts; i++) | 
 | 	{ | 
 | 	    pts[i].x = pPoly->apts[i].x; | 
 | 	    pts[i].y = pPoly->apts[i].y; | 
 | 	} | 
 | 	Polygon(hdc, pts, pPoly->cpts); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYLINE16: | 
 |       { | 
 | 	const EMRPOLYLINE16 *pPoly = (const EMRPOLYLINE16 *)mr; | 
 | 	/* Shouldn't use Polyline16 since pPoly->cpts is DWORD */ | 
 | 	POINT *pts = HeapAlloc( GetProcessHeap(), 0, | 
 | 				pPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPoly->cpts; i++) | 
 | 	{ | 
 | 	    pts[i].x = pPoly->apts[i].x; | 
 | 	    pts[i].y = pPoly->apts[i].y; | 
 | 	} | 
 | 	Polyline(hdc, pts, pPoly->cpts); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYLINETO16: | 
 |       { | 
 | 	const EMRPOLYLINETO16 *pPoly = (const EMRPOLYLINETO16 *)mr; | 
 | 	/* Shouldn't use PolylineTo16 since pPoly->cpts is DWORD */ | 
 | 	POINT *pts = HeapAlloc( GetProcessHeap(), 0, | 
 | 				pPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPoly->cpts; i++) | 
 | 	{ | 
 | 	    pts[i].x = pPoly->apts[i].x; | 
 | 	    pts[i].y = pPoly->apts[i].y; | 
 | 	} | 
 | 	PolylineTo(hdc, pts, pPoly->cpts); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYBEZIER16: | 
 |       { | 
 | 	const EMRPOLYBEZIER16 *pPoly = (const EMRPOLYBEZIER16 *)mr; | 
 | 	/* Shouldn't use PolyBezier16 since pPoly->cpts is DWORD */ | 
 | 	POINT *pts = HeapAlloc( GetProcessHeap(), 0, | 
 | 				pPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPoly->cpts; i++) | 
 | 	{ | 
 | 	    pts[i].x = pPoly->apts[i].x; | 
 | 	    pts[i].y = pPoly->apts[i].y; | 
 | 	} | 
 | 	PolyBezier(hdc, pts, pPoly->cpts); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYBEZIERTO16: | 
 |       { | 
 | 	const EMRPOLYBEZIERTO16 *pPoly = (const EMRPOLYBEZIERTO16 *)mr; | 
 | 	/* Shouldn't use PolyBezierTo16 since pPoly->cpts is DWORD */ | 
 | 	POINT *pts = HeapAlloc( GetProcessHeap(), 0, | 
 | 				pPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPoly->cpts; i++) | 
 | 	{ | 
 | 	    pts[i].x = pPoly->apts[i].x; | 
 | 	    pts[i].y = pPoly->apts[i].y; | 
 | 	} | 
 | 	PolyBezierTo(hdc, pts, pPoly->cpts); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYPOLYGON16: | 
 |       { | 
 |         const EMRPOLYPOLYGON16 *pPolyPoly = (const EMRPOLYPOLYGON16 *)mr; | 
 | 	/* NB POINTS array doesn't start at pPolyPoly->apts it's actually | 
 | 	   pPolyPoly->aPolyCounts + pPolyPoly->nPolys */ | 
 |  | 
 |         POINT16 *pts16 = (POINT16 *)(pPolyPoly->aPolyCounts + pPolyPoly->nPolys); | 
 |         POINT *pts = HeapAlloc( GetProcessHeap(), 0, pPolyPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPolyPoly->cpts; i++) | 
 |         { | 
 |             pts[i].x = pts16[i].x; | 
 |             pts[i].y = pts16[i].y; | 
 |         } | 
 | 	PolyPolygon(hdc, pts, (INT*)pPolyPoly->aPolyCounts, pPolyPoly->nPolys); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |     case EMR_POLYPOLYLINE16: | 
 |       { | 
 |         const EMRPOLYPOLYLINE16 *pPolyPoly = (const EMRPOLYPOLYLINE16 *)mr; | 
 | 	/* NB POINTS array doesn't start at pPolyPoly->apts it's actually | 
 | 	   pPolyPoly->aPolyCounts + pPolyPoly->nPolys */ | 
 |  | 
 |         POINT16 *pts16 = (POINT16 *)(pPolyPoly->aPolyCounts + pPolyPoly->nPolys); | 
 |         POINT *pts = HeapAlloc( GetProcessHeap(), 0, pPolyPoly->cpts * sizeof(POINT) ); | 
 | 	DWORD i; | 
 | 	for(i = 0; i < pPolyPoly->cpts; i++) | 
 |         { | 
 |             pts[i].x = pts16[i].x; | 
 |             pts[i].y = pts16[i].y; | 
 |         } | 
 | 	PolyPolyline(hdc, pts, pPolyPoly->aPolyCounts, pPolyPoly->nPolys); | 
 | 	HeapFree( GetProcessHeap(), 0, pts ); | 
 | 	break; | 
 |       } | 
 |  | 
 |     case EMR_STRETCHDIBITS: | 
 |       { | 
 | 	const EMRSTRETCHDIBITS *pStretchDIBits = (const EMRSTRETCHDIBITS *)mr; | 
 |  | 
 | 	StretchDIBits(hdc, | 
 | 		      pStretchDIBits->xDest, | 
 | 		      pStretchDIBits->yDest, | 
 | 		      pStretchDIBits->cxDest, | 
 | 		      pStretchDIBits->cyDest, | 
 | 		      pStretchDIBits->xSrc, | 
 | 		      pStretchDIBits->ySrc, | 
 | 		      pStretchDIBits->cxSrc, | 
 | 		      pStretchDIBits->cySrc, | 
 | 		      (const BYTE *)mr + pStretchDIBits->offBitsSrc, | 
 | 		      (const BITMAPINFO *)((const BYTE *)mr + pStretchDIBits->offBmiSrc), | 
 | 		      pStretchDIBits->iUsageSrc, | 
 | 		      pStretchDIBits->dwRop); | 
 | 	break; | 
 |       } | 
 |  | 
 |     case EMR_EXTTEXTOUTA: | 
 |     { | 
 | 	const EMREXTTEXTOUTA *pExtTextOutA = (const EMREXTTEXTOUTA *)mr; | 
 | 	RECT rc; | 
 |         const INT *dx = NULL; | 
 |  | 
 | 	rc.left = pExtTextOutA->emrtext.rcl.left; | 
 | 	rc.top = pExtTextOutA->emrtext.rcl.top; | 
 | 	rc.right = pExtTextOutA->emrtext.rcl.right; | 
 | 	rc.bottom = pExtTextOutA->emrtext.rcl.bottom; | 
 |         TRACE("EMR_EXTTEXTOUTA: x,y = %d, %d. rect = %d, %d - %d, %d. flags %08x\n", | 
 |               pExtTextOutA->emrtext.ptlReference.x, pExtTextOutA->emrtext.ptlReference.y, | 
 |               rc.left, rc.top, rc.right, rc.bottom, pExtTextOutA->emrtext.fOptions); | 
 |  | 
 |         /* Linux version of pstoedit produces EMFs with offDx set to 0. | 
 |          * These files can be enumerated and played under Win98 just | 
 |          * fine, but at least Win2k chokes on them. | 
 |          */ | 
 |         if (pExtTextOutA->emrtext.offDx) | 
 |             dx = (const INT *)((const BYTE *)mr + pExtTextOutA->emrtext.offDx); | 
 |  | 
 | 	ExtTextOutA(hdc, pExtTextOutA->emrtext.ptlReference.x, pExtTextOutA->emrtext.ptlReference.y, | 
 | 	    pExtTextOutA->emrtext.fOptions, &rc, | 
 | 	    (LPCSTR)((const BYTE *)mr + pExtTextOutA->emrtext.offString), pExtTextOutA->emrtext.nChars, | 
 | 	    dx); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_EXTTEXTOUTW: | 
 |     { | 
 | 	const EMREXTTEXTOUTW *pExtTextOutW = (const EMREXTTEXTOUTW *)mr; | 
 | 	RECT rc; | 
 |         const INT *dx = NULL; | 
 |  | 
 | 	rc.left = pExtTextOutW->emrtext.rcl.left; | 
 | 	rc.top = pExtTextOutW->emrtext.rcl.top; | 
 | 	rc.right = pExtTextOutW->emrtext.rcl.right; | 
 | 	rc.bottom = pExtTextOutW->emrtext.rcl.bottom; | 
 |         TRACE("EMR_EXTTEXTOUTW: x,y = %d, %d.  rect = %d, %d - %d, %d. flags %08x\n", | 
 |               pExtTextOutW->emrtext.ptlReference.x, pExtTextOutW->emrtext.ptlReference.y, | 
 |               rc.left, rc.top, rc.right, rc.bottom, pExtTextOutW->emrtext.fOptions); | 
 |  | 
 |         /* Linux version of pstoedit produces EMFs with offDx set to 0. | 
 |          * These files can be enumerated and played under Win98 just | 
 |          * fine, but at least Win2k chokes on them. | 
 |          */ | 
 |         if (pExtTextOutW->emrtext.offDx) | 
 |             dx = (const INT *)((const BYTE *)mr + pExtTextOutW->emrtext.offDx); | 
 |  | 
 | 	ExtTextOutW(hdc, pExtTextOutW->emrtext.ptlReference.x, pExtTextOutW->emrtext.ptlReference.y, | 
 | 	    pExtTextOutW->emrtext.fOptions, &rc, | 
 | 	    (LPCWSTR)((const BYTE *)mr + pExtTextOutW->emrtext.offString), pExtTextOutW->emrtext.nChars, | 
 | 	    dx); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_CREATEPALETTE: | 
 |       { | 
 | 	const EMRCREATEPALETTE *lpCreatePal = (const EMRCREATEPALETTE *)mr; | 
 |  | 
 | 	(handletable->objectHandle)[ lpCreatePal->ihPal ] = | 
 | 		CreatePalette( &lpCreatePal->lgpl ); | 
 |  | 
 | 	break; | 
 |       } | 
 |  | 
 |     case EMR_SELECTPALETTE: | 
 |       { | 
 | 	const EMRSELECTPALETTE *lpSelectPal = (const EMRSELECTPALETTE *)mr; | 
 |  | 
 | 	if( lpSelectPal->ihPal & 0x80000000 ) { | 
 | 		SelectPalette( hdc, GetStockObject(lpSelectPal->ihPal & 0x7fffffff), TRUE); | 
 | 	} else { | 
 | 		SelectPalette( hdc, (handletable->objectHandle)[lpSelectPal->ihPal], TRUE); | 
 | 	} | 
 | 	break; | 
 |       } | 
 |  | 
 |     case EMR_REALIZEPALETTE: | 
 |       { | 
 | 	RealizePalette( hdc ); | 
 | 	break; | 
 |       } | 
 |  | 
 |     case EMR_EXTSELECTCLIPRGN: | 
 |       { | 
 | 	const EMREXTSELECTCLIPRGN *lpRgn = (const EMREXTSELECTCLIPRGN *)mr; | 
 | 	HRGN hRgn = 0; | 
 |  | 
 |         if (mr->nSize >= sizeof(*lpRgn) + sizeof(RGNDATAHEADER)) | 
 |             hRgn = ExtCreateRegion( &info->init_transform, 0, (RGNDATA *)lpRgn->RgnData ); | 
 |  | 
 | 	ExtSelectClipRgn(hdc, hRgn, (INT)(lpRgn->iMode)); | 
 | 	/* ExtSelectClipRgn created a copy of the region */ | 
 | 	DeleteObject(hRgn); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETMETARGN: | 
 |       { | 
 |         SetMetaRgn( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETWORLDTRANSFORM: | 
 |       { | 
 |         const EMRSETWORLDTRANSFORM *lpXfrm = (const EMRSETWORLDTRANSFORM *)mr; | 
 |         info->state.world_transform = lpXfrm->xform; | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYBEZIER: | 
 |       { | 
 |         const EMRPOLYBEZIER *lpPolyBez = (const EMRPOLYBEZIER *)mr; | 
 |         PolyBezier(hdc, (const POINT*)lpPolyBez->aptl, (UINT)lpPolyBez->cptl); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYGON: | 
 |       { | 
 |         const EMRPOLYGON *lpPoly = (const EMRPOLYGON *)mr; | 
 |         Polygon( hdc, (const POINT*)lpPoly->aptl, (UINT)lpPoly->cptl ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYLINE: | 
 |       { | 
 |         const EMRPOLYLINE *lpPolyLine = (const EMRPOLYLINE *)mr; | 
 |         Polyline(hdc, (const POINT*)lpPolyLine->aptl, (UINT)lpPolyLine->cptl); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYBEZIERTO: | 
 |       { | 
 |         const EMRPOLYBEZIERTO *lpPolyBezierTo = (const EMRPOLYBEZIERTO *)mr; | 
 |         PolyBezierTo( hdc, (const POINT*)lpPolyBezierTo->aptl, | 
 | 		      (UINT)lpPolyBezierTo->cptl ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYLINETO: | 
 |       { | 
 |         const EMRPOLYLINETO *lpPolyLineTo = (const EMRPOLYLINETO *)mr; | 
 |         PolylineTo( hdc, (const POINT*)lpPolyLineTo->aptl, | 
 | 		    (UINT)lpPolyLineTo->cptl ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYPOLYLINE: | 
 |       { | 
 |         const EMRPOLYPOLYLINE *pPolyPolyline = (const EMRPOLYPOLYLINE *)mr; | 
 | 	/* NB Points at pPolyPolyline->aPolyCounts + pPolyPolyline->nPolys */ | 
 |  | 
 |         PolyPolyline(hdc, (LPPOINT)(pPolyPolyline->aPolyCounts + | 
 | 				    pPolyPolyline->nPolys), | 
 | 		     pPolyPolyline->aPolyCounts, | 
 | 		     pPolyPolyline->nPolys ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYPOLYGON: | 
 |       { | 
 |         const EMRPOLYPOLYGON *pPolyPolygon = (const EMRPOLYPOLYGON *)mr; | 
 |  | 
 | 	/* NB Points at pPolyPolygon->aPolyCounts + pPolyPolygon->nPolys */ | 
 |  | 
 |         PolyPolygon(hdc, (LPPOINT)(pPolyPolygon->aPolyCounts + | 
 | 				   pPolyPolygon->nPolys), | 
 | 		    (INT*)pPolyPolygon->aPolyCounts, pPolyPolygon->nPolys ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETBRUSHORGEX: | 
 |       { | 
 |         const EMRSETBRUSHORGEX *lpSetBrushOrgEx = (const EMRSETBRUSHORGEX *)mr; | 
 |  | 
 |         SetBrushOrgEx( hdc, | 
 |                        (INT)lpSetBrushOrgEx->ptlOrigin.x, | 
 |                        (INT)lpSetBrushOrgEx->ptlOrigin.y, | 
 |                        NULL ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETPIXELV: | 
 |       { | 
 |         const EMRSETPIXELV *lpSetPixelV = (const EMRSETPIXELV *)mr; | 
 |  | 
 |         SetPixelV( hdc, | 
 |                    (INT)lpSetPixelV->ptlPixel.x, | 
 |                    (INT)lpSetPixelV->ptlPixel.y, | 
 |                    lpSetPixelV->crColor ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETMAPPERFLAGS: | 
 |       { | 
 |         const EMRSETMAPPERFLAGS *lpSetMapperFlags = (const EMRSETMAPPERFLAGS *)mr; | 
 |  | 
 |         SetMapperFlags( hdc, lpSetMapperFlags->dwFlags ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETCOLORADJUSTMENT: | 
 |       { | 
 |         const EMRSETCOLORADJUSTMENT *lpSetColorAdjust = (const EMRSETCOLORADJUSTMENT *)mr; | 
 |  | 
 |         SetColorAdjustment( hdc, &lpSetColorAdjust->ColorAdjustment ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_OFFSETCLIPRGN: | 
 |       { | 
 |         const EMROFFSETCLIPRGN *lpOffsetClipRgn = (const EMROFFSETCLIPRGN *)mr; | 
 |  | 
 |         OffsetClipRgn( hdc, | 
 |                        (INT)lpOffsetClipRgn->ptlOffset.x, | 
 |                        (INT)lpOffsetClipRgn->ptlOffset.y ); | 
 |         FIXME("OffsetClipRgn\n"); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_EXCLUDECLIPRECT: | 
 |       { | 
 |         const EMREXCLUDECLIPRECT *lpExcludeClipRect = (const EMREXCLUDECLIPRECT *)mr; | 
 |  | 
 |         ExcludeClipRect( hdc, | 
 |                          lpExcludeClipRect->rclClip.left, | 
 |                          lpExcludeClipRect->rclClip.top, | 
 |                          lpExcludeClipRect->rclClip.right, | 
 |                          lpExcludeClipRect->rclClip.bottom  ); | 
 |         FIXME("ExcludeClipRect\n"); | 
 |  | 
 |          break; | 
 |       } | 
 |  | 
 |     case EMR_SCALEVIEWPORTEXTEX: | 
 |       { | 
 |         const EMRSCALEVIEWPORTEXTEX *lpScaleViewportExtEx = (const EMRSCALEVIEWPORTEXTEX *)mr; | 
 |  | 
 |         if ((info->state.mode != MM_ISOTROPIC) && (info->state.mode != MM_ANISOTROPIC)) | 
 | 	    break; | 
 |         if (!lpScaleViewportExtEx->xNum || !lpScaleViewportExtEx->xDenom ||  | 
 |             !lpScaleViewportExtEx->yNum || !lpScaleViewportExtEx->yDenom) | 
 |             break; | 
 |         info->state.vportExtX = MulDiv(info->state.vportExtX, lpScaleViewportExtEx->xNum, | 
 |                                  lpScaleViewportExtEx->xDenom); | 
 |         info->state.vportExtY = MulDiv(info->state.vportExtY, lpScaleViewportExtEx->yNum, | 
 |                                  lpScaleViewportExtEx->yDenom); | 
 |         if (info->state.vportExtX == 0) info->state.vportExtX = 1; | 
 |         if (info->state.vportExtY == 0) info->state.vportExtY = 1; | 
 |         if (info->state.mode == MM_ISOTROPIC) | 
 |             EMF_FixIsotropic(hdc, info); | 
 |  | 
 |         TRACE("EMRSCALEVIEWPORTEXTEX %d/%d %d/%d\n", | 
 |              lpScaleViewportExtEx->xNum,lpScaleViewportExtEx->xDenom, | 
 |              lpScaleViewportExtEx->yNum,lpScaleViewportExtEx->yDenom); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SCALEWINDOWEXTEX: | 
 |       { | 
 |         const EMRSCALEWINDOWEXTEX *lpScaleWindowExtEx = (const EMRSCALEWINDOWEXTEX *)mr; | 
 |  | 
 |         if ((info->state.mode != MM_ISOTROPIC) && (info->state.mode != MM_ANISOTROPIC)) | 
 | 	    break; | 
 |         if (!lpScaleWindowExtEx->xNum || !lpScaleWindowExtEx->xDenom ||  | 
 |             !lpScaleWindowExtEx->xNum || !lpScaleWindowExtEx->yDenom) | 
 |             break; | 
 |         info->state.wndExtX = MulDiv(info->state.wndExtX, lpScaleWindowExtEx->xNum, | 
 |                                lpScaleWindowExtEx->xDenom); | 
 |         info->state.wndExtY = MulDiv(info->state.wndExtY, lpScaleWindowExtEx->yNum, | 
 |                                lpScaleWindowExtEx->yDenom); | 
 |         if (info->state.wndExtX == 0) info->state.wndExtX = 1; | 
 |         if (info->state.wndExtY == 0) info->state.wndExtY = 1; | 
 |         if (info->state.mode == MM_ISOTROPIC) | 
 |             EMF_FixIsotropic(hdc, info); | 
 |  | 
 |         TRACE("EMRSCALEWINDOWEXTEX %d/%d %d/%d\n", | 
 |              lpScaleWindowExtEx->xNum,lpScaleWindowExtEx->xDenom, | 
 |              lpScaleWindowExtEx->yNum,lpScaleWindowExtEx->yDenom); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_MODIFYWORLDTRANSFORM: | 
 |       { | 
 |         const EMRMODIFYWORLDTRANSFORM *lpModifyWorldTrans = (const EMRMODIFYWORLDTRANSFORM *)mr; | 
 |  | 
 |         switch(lpModifyWorldTrans->iMode) { | 
 |         case MWT_IDENTITY: | 
 |             info->state.world_transform.eM11 = info->state.world_transform.eM22 = 1; | 
 |             info->state.world_transform.eM12 = info->state.world_transform.eM21 = 0; | 
 |             info->state.world_transform.eDx  = info->state.world_transform.eDy  = 0; | 
 |             break; | 
 |         case MWT_LEFTMULTIPLY: | 
 |             CombineTransform(&info->state.world_transform, &lpModifyWorldTrans->xform, | 
 |                              &info->state.world_transform); | 
 |             break; | 
 |         case MWT_RIGHTMULTIPLY: | 
 |             CombineTransform(&info->state.world_transform, &info->state.world_transform, | 
 |                              &lpModifyWorldTrans->xform); | 
 |             break; | 
 |         default: | 
 |             FIXME("Unknown imode %d\n", lpModifyWorldTrans->iMode); | 
 |             break; | 
 |         } | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_ANGLEARC: | 
 |       { | 
 |         const EMRANGLEARC *lpAngleArc = (const EMRANGLEARC *)mr; | 
 |  | 
 |         AngleArc( hdc, | 
 |                  (INT)lpAngleArc->ptlCenter.x, (INT)lpAngleArc->ptlCenter.y, | 
 |                  lpAngleArc->nRadius, lpAngleArc->eStartAngle, | 
 |                  lpAngleArc->eSweepAngle ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_ROUNDRECT: | 
 |       { | 
 |         const EMRROUNDRECT *lpRoundRect = (const EMRROUNDRECT *)mr; | 
 |  | 
 |         RoundRect( hdc, | 
 |                    lpRoundRect->rclBox.left, | 
 |                    lpRoundRect->rclBox.top, | 
 |                    lpRoundRect->rclBox.right, | 
 |                    lpRoundRect->rclBox.bottom, | 
 |                    lpRoundRect->szlCorner.cx, | 
 |                    lpRoundRect->szlCorner.cy ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_ARC: | 
 |       { | 
 |         const EMRARC *lpArc = (const EMRARC *)mr; | 
 |  | 
 |         Arc( hdc, | 
 |              (INT)lpArc->rclBox.left, | 
 |              (INT)lpArc->rclBox.top, | 
 |              (INT)lpArc->rclBox.right, | 
 |              (INT)lpArc->rclBox.bottom, | 
 |              (INT)lpArc->ptlStart.x, | 
 |              (INT)lpArc->ptlStart.y, | 
 |              (INT)lpArc->ptlEnd.x, | 
 |              (INT)lpArc->ptlEnd.y ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_CHORD: | 
 |       { | 
 |         const EMRCHORD *lpChord = (const EMRCHORD *)mr; | 
 |  | 
 |         Chord( hdc, | 
 |              (INT)lpChord->rclBox.left, | 
 |              (INT)lpChord->rclBox.top, | 
 |              (INT)lpChord->rclBox.right, | 
 |              (INT)lpChord->rclBox.bottom, | 
 |              (INT)lpChord->ptlStart.x, | 
 |              (INT)lpChord->ptlStart.y, | 
 |              (INT)lpChord->ptlEnd.x, | 
 |              (INT)lpChord->ptlEnd.y ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_PIE: | 
 |       { | 
 |         const EMRPIE *lpPie = (const EMRPIE *)mr; | 
 |  | 
 |         Pie( hdc, | 
 |              (INT)lpPie->rclBox.left, | 
 |              (INT)lpPie->rclBox.top, | 
 |              (INT)lpPie->rclBox.right, | 
 |              (INT)lpPie->rclBox.bottom, | 
 |              (INT)lpPie->ptlStart.x, | 
 |              (INT)lpPie->ptlStart.y, | 
 |              (INT)lpPie->ptlEnd.x, | 
 |              (INT)lpPie->ptlEnd.y ); | 
 |  | 
 |        break; | 
 |       } | 
 |  | 
 |     case EMR_ARCTO: | 
 |       { | 
 |         const EMRARC *lpArcTo = (const EMRARC *)mr; | 
 |  | 
 |         ArcTo( hdc, | 
 |                (INT)lpArcTo->rclBox.left, | 
 |                (INT)lpArcTo->rclBox.top, | 
 |                (INT)lpArcTo->rclBox.right, | 
 |                (INT)lpArcTo->rclBox.bottom, | 
 |                (INT)lpArcTo->ptlStart.x, | 
 |                (INT)lpArcTo->ptlStart.y, | 
 |                (INT)lpArcTo->ptlEnd.x, | 
 |                (INT)lpArcTo->ptlEnd.y ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_EXTFLOODFILL: | 
 |       { | 
 |         const EMREXTFLOODFILL *lpExtFloodFill = (const EMREXTFLOODFILL *)mr; | 
 |  | 
 |         ExtFloodFill( hdc, | 
 |                       (INT)lpExtFloodFill->ptlStart.x, | 
 |                       (INT)lpExtFloodFill->ptlStart.y, | 
 |                       lpExtFloodFill->crColor, | 
 |                       (UINT)lpExtFloodFill->iMode ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_POLYDRAW: | 
 |       { | 
 |         const EMRPOLYDRAW *lpPolyDraw = (const EMRPOLYDRAW *)mr; | 
 |         PolyDraw( hdc, | 
 |                   (const POINT*)lpPolyDraw->aptl, | 
 |                   lpPolyDraw->abTypes, | 
 |                   (INT)lpPolyDraw->cptl ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETARCDIRECTION: | 
 |       { | 
 |         const EMRSETARCDIRECTION *lpSetArcDirection = (const EMRSETARCDIRECTION *)mr; | 
 |         SetArcDirection( hdc, (INT)lpSetArcDirection->iArcDirection ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETMITERLIMIT: | 
 |       { | 
 |         const EMRSETMITERLIMIT *lpSetMiterLimit = (const EMRSETMITERLIMIT *)mr; | 
 |         SetMiterLimit( hdc, lpSetMiterLimit->eMiterLimit, NULL ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_BEGINPATH: | 
 |       { | 
 |         BeginPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_ENDPATH: | 
 |       { | 
 |         EndPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_CLOSEFIGURE: | 
 |       { | 
 |         CloseFigure( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_FILLPATH: | 
 |       { | 
 |         /*const EMRFILLPATH lpFillPath = (const EMRFILLPATH *)mr;*/ | 
 |         FillPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_STROKEANDFILLPATH: | 
 |       { | 
 |         /*const EMRSTROKEANDFILLPATH lpStrokeAndFillPath = (const EMRSTROKEANDFILLPATH *)mr;*/ | 
 |         StrokeAndFillPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_STROKEPATH: | 
 |       { | 
 |         /*const EMRSTROKEPATH lpStrokePath = (const EMRSTROKEPATH *)mr;*/ | 
 |         StrokePath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_FLATTENPATH: | 
 |       { | 
 |         FlattenPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_WIDENPATH: | 
 |       { | 
 |         WidenPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SELECTCLIPPATH: | 
 |       { | 
 |         const EMRSELECTCLIPPATH *lpSelectClipPath = (const EMRSELECTCLIPPATH *)mr; | 
 |         SelectClipPath( hdc, (INT)lpSelectClipPath->iMode ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_ABORTPATH: | 
 |       { | 
 |         AbortPath( hdc ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_CREATECOLORSPACE: | 
 |       { | 
 |         PEMRCREATECOLORSPACE lpCreateColorSpace = (PEMRCREATECOLORSPACE)mr; | 
 |         (handletable->objectHandle)[lpCreateColorSpace->ihCS] = | 
 |            CreateColorSpaceA( &lpCreateColorSpace->lcs ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETCOLORSPACE: | 
 |       { | 
 |         const EMRSETCOLORSPACE *lpSetColorSpace = (const EMRSETCOLORSPACE *)mr; | 
 |         SetColorSpace( hdc, | 
 |                        (handletable->objectHandle)[lpSetColorSpace->ihCS] ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_DELETECOLORSPACE: | 
 |       { | 
 |         const EMRDELETECOLORSPACE *lpDeleteColorSpace = (const EMRDELETECOLORSPACE *)mr; | 
 |         DeleteColorSpace( (handletable->objectHandle)[lpDeleteColorSpace->ihCS] ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETICMMODE: | 
 |       { | 
 |         const EMRSETICMMODE *lpSetICMMode = (const EMRSETICMMODE *)mr; | 
 |         SetICMMode( hdc, (INT)lpSetICMMode->iMode ); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_PIXELFORMAT: | 
 |       { | 
 |         INT iPixelFormat; | 
 |         const EMRPIXELFORMAT *lpPixelFormat = (const EMRPIXELFORMAT *)mr; | 
 |  | 
 |         iPixelFormat = ChoosePixelFormat( hdc, &lpPixelFormat->pfd ); | 
 |         SetPixelFormat( hdc, iPixelFormat, &lpPixelFormat->pfd ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_SETPALETTEENTRIES: | 
 |       { | 
 |         const EMRSETPALETTEENTRIES *lpSetPaletteEntries = (const EMRSETPALETTEENTRIES *)mr; | 
 |  | 
 |         SetPaletteEntries( (handletable->objectHandle)[lpSetPaletteEntries->ihPal], | 
 |                            (UINT)lpSetPaletteEntries->iStart, | 
 |                            (UINT)lpSetPaletteEntries->cEntries, | 
 |                            lpSetPaletteEntries->aPalEntries ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_RESIZEPALETTE: | 
 |       { | 
 |         const EMRRESIZEPALETTE *lpResizePalette = (const EMRRESIZEPALETTE *)mr; | 
 |  | 
 |         ResizePalette( (handletable->objectHandle)[lpResizePalette->ihPal], | 
 |                        (UINT)lpResizePalette->cEntries ); | 
 |  | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_CREATEDIBPATTERNBRUSHPT: | 
 |       { | 
 |         const EMRCREATEDIBPATTERNBRUSHPT *lpCreate = (const EMRCREATEDIBPATTERNBRUSHPT *)mr; | 
 |         LPVOID lpPackedStruct; | 
 |  | 
 |         /* Check that offsets and data are contained within the record | 
 |          * (including checking for wrap-arounds). | 
 |          */ | 
 |         if (    lpCreate->offBmi  + lpCreate->cbBmi  > mr->nSize | 
 |              || lpCreate->offBits + lpCreate->cbBits > mr->nSize | 
 |              || lpCreate->offBmi  + lpCreate->cbBmi  < lpCreate->offBmi | 
 |              || lpCreate->offBits + lpCreate->cbBits < lpCreate->offBits ) | 
 |         { | 
 |             ERR("Invalid EMR_CREATEDIBPATTERNBRUSHPT record\n"); | 
 |             break; | 
 |         } | 
 |  | 
 |         /* This is a BITMAPINFO struct followed directly by bitmap bits */ | 
 |         lpPackedStruct = HeapAlloc( GetProcessHeap(), 0, | 
 |                                     lpCreate->cbBmi + lpCreate->cbBits ); | 
 |         if(!lpPackedStruct) | 
 |         { | 
 |             SetLastError(ERROR_NOT_ENOUGH_MEMORY); | 
 |             break; | 
 |         } | 
 |  | 
 |         /* Now pack this structure */ | 
 |         memcpy( lpPackedStruct, | 
 |                 ((const BYTE *)lpCreate) + lpCreate->offBmi, | 
 |                 lpCreate->cbBmi ); | 
 |         memcpy( ((BYTE*)lpPackedStruct) + lpCreate->cbBmi, | 
 |                 ((const BYTE *)lpCreate) + lpCreate->offBits, | 
 |                 lpCreate->cbBits ); | 
 |  | 
 |         (handletable->objectHandle)[lpCreate->ihBrush] = | 
 |            CreateDIBPatternBrushPt( lpPackedStruct, | 
 |                                     (UINT)lpCreate->iUsage ); | 
 |  | 
 |         HeapFree(GetProcessHeap(), 0, lpPackedStruct); | 
 |         break; | 
 |       } | 
 |  | 
 |     case EMR_CREATEMONOBRUSH: | 
 |     { | 
 |         const EMRCREATEMONOBRUSH *pCreateMonoBrush = (const EMRCREATEMONOBRUSH *)mr; | 
 |         const BITMAPINFO *pbi = (const BITMAPINFO *)((const BYTE *)mr + pCreateMonoBrush->offBmi); | 
 |         HBITMAP hBmp; | 
 |  | 
 |         /* Need to check if the bitmap is monochrome, and if the | 
 |            two colors are really black and white */ | 
 |         if (pCreateMonoBrush->iUsage == DIB_PAL_MONO) | 
 |         { | 
 |             BITMAP bm; | 
 |  | 
 |             /* Undocumented iUsage indicates a mono bitmap with no palette table, | 
 |              * aligned to 32 rather than 16 bits. | 
 |              */ | 
 |             bm.bmType = 0; | 
 |             bm.bmWidth = pbi->bmiHeader.biWidth; | 
 |             bm.bmHeight = abs(pbi->bmiHeader.biHeight); | 
 |             bm.bmWidthBytes = 4 * ((pbi->bmiHeader.biWidth + 31) / 32); | 
 |             bm.bmPlanes = pbi->bmiHeader.biPlanes; | 
 |             bm.bmBitsPixel = pbi->bmiHeader.biBitCount; | 
 |             bm.bmBits = (BYTE *)mr + pCreateMonoBrush->offBits; | 
 |             hBmp = CreateBitmapIndirect(&bm); | 
 |         } | 
 |         else if (is_dib_monochrome(pbi)) | 
 |         { | 
 |           /* Top-down DIBs have a negative height */ | 
 |           LONG height = pbi->bmiHeader.biHeight; | 
 |  | 
 |           hBmp = CreateBitmap(pbi->bmiHeader.biWidth, abs(height), 1, 1, NULL); | 
 |           SetDIBits(hdc, hBmp, 0, pbi->bmiHeader.biHeight, | 
 |               (const BYTE *)mr + pCreateMonoBrush->offBits, pbi, pCreateMonoBrush->iUsage); | 
 |         } | 
 |         else | 
 |         { | 
 |             hBmp = CreateDIBitmap(hdc, (const BITMAPINFOHEADER *)pbi, CBM_INIT, | 
 |               (const BYTE *)mr + pCreateMonoBrush->offBits, pbi, pCreateMonoBrush->iUsage); | 
 |         } | 
 |  | 
 | 	(handletable->objectHandle)[pCreateMonoBrush->ihBrush] = CreatePatternBrush(hBmp); | 
 |  | 
 | 	/* CreatePatternBrush created a copy of the bitmap */ | 
 | 	DeleteObject(hBmp); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_BITBLT: | 
 |     { | 
 | 	const EMRBITBLT *pBitBlt = (const EMRBITBLT *)mr; | 
 |  | 
 |         if(pBitBlt->offBmiSrc == 0) { /* Record is a PatBlt */ | 
 |             PatBlt(hdc, pBitBlt->xDest, pBitBlt->yDest, pBitBlt->cxDest, pBitBlt->cyDest, | 
 |                    pBitBlt->dwRop); | 
 |         } else { /* BitBlt */ | 
 |             HDC hdcSrc = CreateCompatibleDC(hdc); | 
 |             HBRUSH hBrush, hBrushOld; | 
 |             HBITMAP hBmp = 0, hBmpOld = 0; | 
 |             const BITMAPINFO *pbi = (const BITMAPINFO *)((const BYTE *)mr + pBitBlt->offBmiSrc); | 
 |  | 
 |             SetWorldTransform(hdcSrc, &pBitBlt->xformSrc); | 
 |  | 
 |             hBrush = CreateSolidBrush(pBitBlt->crBkColorSrc); | 
 |             hBrushOld = SelectObject(hdcSrc, hBrush); | 
 |             PatBlt(hdcSrc, pBitBlt->rclBounds.left, pBitBlt->rclBounds.top, | 
 |                    pBitBlt->rclBounds.right - pBitBlt->rclBounds.left, | 
 |                    pBitBlt->rclBounds.bottom - pBitBlt->rclBounds.top, PATCOPY); | 
 |             SelectObject(hdcSrc, hBrushOld); | 
 |             DeleteObject(hBrush); | 
 |  | 
 |             hBmp = CreateDIBitmap(hdc, (const BITMAPINFOHEADER *)pbi, CBM_INIT, | 
 |                                   (const BYTE *)mr + pBitBlt->offBitsSrc, pbi, pBitBlt->iUsageSrc); | 
 |             hBmpOld = SelectObject(hdcSrc, hBmp); | 
 |  | 
 |             BitBlt(hdc, pBitBlt->xDest, pBitBlt->yDest, pBitBlt->cxDest, pBitBlt->cyDest, | 
 |                    hdcSrc, pBitBlt->xSrc, pBitBlt->ySrc, pBitBlt->dwRop); | 
 |  | 
 |             SelectObject(hdcSrc, hBmpOld); | 
 |             DeleteObject(hBmp); | 
 |             DeleteDC(hdcSrc); | 
 |         } | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_STRETCHBLT: | 
 |     { | 
 | 	const EMRSTRETCHBLT *pStretchBlt = (const EMRSTRETCHBLT *)mr; | 
 |  | 
 |         TRACE("EMR_STRETCHBLT: %d, %d %dx%d -> %d, %d %dx%d. rop %08x offBitsSrc %d\n", | 
 | 	       pStretchBlt->xSrc, pStretchBlt->ySrc, pStretchBlt->cxSrc, pStretchBlt->cySrc, | 
 | 	       pStretchBlt->xDest, pStretchBlt->yDest, pStretchBlt->cxDest, pStretchBlt->cyDest, | 
 | 	       pStretchBlt->dwRop, pStretchBlt->offBitsSrc); | 
 |  | 
 |         if(pStretchBlt->offBmiSrc == 0) { /* Record is a PatBlt */ | 
 |             PatBlt(hdc, pStretchBlt->xDest, pStretchBlt->yDest, pStretchBlt->cxDest, pStretchBlt->cyDest, | 
 |                    pStretchBlt->dwRop); | 
 |         } else { /* StretchBlt */ | 
 |             HDC hdcSrc = CreateCompatibleDC(hdc); | 
 |             HBRUSH hBrush, hBrushOld; | 
 |             HBITMAP hBmp = 0, hBmpOld = 0; | 
 |             const BITMAPINFO *pbi = (const BITMAPINFO *)((const BYTE *)mr + pStretchBlt->offBmiSrc); | 
 |  | 
 |             SetWorldTransform(hdcSrc, &pStretchBlt->xformSrc); | 
 |  | 
 |             hBrush = CreateSolidBrush(pStretchBlt->crBkColorSrc); | 
 |             hBrushOld = SelectObject(hdcSrc, hBrush); | 
 |             PatBlt(hdcSrc, pStretchBlt->rclBounds.left, pStretchBlt->rclBounds.top, | 
 |                    pStretchBlt->rclBounds.right - pStretchBlt->rclBounds.left, | 
 |                    pStretchBlt->rclBounds.bottom - pStretchBlt->rclBounds.top, PATCOPY); | 
 |             SelectObject(hdcSrc, hBrushOld); | 
 |             DeleteObject(hBrush); | 
 |  | 
 |             hBmp = CreateDIBitmap(hdc, (const BITMAPINFOHEADER *)pbi, CBM_INIT, | 
 |                                   (const BYTE *)mr + pStretchBlt->offBitsSrc, pbi, pStretchBlt->iUsageSrc); | 
 |             hBmpOld = SelectObject(hdcSrc, hBmp); | 
 |  | 
 |             StretchBlt(hdc, pStretchBlt->xDest, pStretchBlt->yDest, pStretchBlt->cxDest, pStretchBlt->cyDest, | 
 |                        hdcSrc, pStretchBlt->xSrc, pStretchBlt->ySrc, pStretchBlt->cxSrc, pStretchBlt->cySrc, | 
 |                        pStretchBlt->dwRop); | 
 |  | 
 |             SelectObject(hdcSrc, hBmpOld); | 
 |             DeleteObject(hBmp); | 
 |             DeleteDC(hdcSrc); | 
 |         } | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_ALPHABLEND: | 
 |     { | 
 | 	const EMRALPHABLEND *pAlphaBlend = (const EMRALPHABLEND *)mr; | 
 |  | 
 |         TRACE("EMR_ALPHABLEND: %d, %d %dx%d -> %d, %d %dx%d. blendfn %08x offBitsSrc %d\n", | 
 | 	       pAlphaBlend->xSrc, pAlphaBlend->ySrc, pAlphaBlend->cxSrc, pAlphaBlend->cySrc, | 
 | 	       pAlphaBlend->xDest, pAlphaBlend->yDest, pAlphaBlend->cxDest, pAlphaBlend->cyDest, | 
 | 	       pAlphaBlend->dwRop, pAlphaBlend->offBitsSrc); | 
 |  | 
 |         if(pAlphaBlend->offBmiSrc == 0) { | 
 |             FIXME("EMR_ALPHABLEND: offBmiSrc == 0\n"); | 
 |         } else { | 
 |             HDC hdcSrc = CreateCompatibleDC(hdc); | 
 |             HBITMAP hBmp = 0, hBmpOld = 0; | 
 |             const BITMAPINFO *pbi = (const BITMAPINFO *)((const BYTE *)mr + pAlphaBlend->offBmiSrc); | 
 |             BLENDFUNCTION blendfn; | 
 |             void *bits; | 
 |  | 
 |             SetWorldTransform(hdcSrc, &pAlphaBlend->xformSrc); | 
 |  | 
 |             hBmp = CreateDIBSection(hdc, pbi, pAlphaBlend->iUsageSrc, &bits, NULL, 0); | 
 |             memcpy(bits, (const BYTE *)mr + pAlphaBlend->offBitsSrc, pAlphaBlend->cbBitsSrc); | 
 |             hBmpOld = SelectObject(hdcSrc, hBmp); | 
 |  | 
 |             blendfn.BlendOp             = (pAlphaBlend->dwRop >> 24) & 0xff; | 
 |             blendfn.BlendFlags          = (pAlphaBlend->dwRop >> 16) & 0xff; | 
 |             blendfn.SourceConstantAlpha = (pAlphaBlend->dwRop >>  8) & 0xff; | 
 |             blendfn.AlphaFormat         = (pAlphaBlend->dwRop) & 0xff; | 
 |  | 
 |             GdiAlphaBlend(hdc, pAlphaBlend->xDest, pAlphaBlend->yDest, pAlphaBlend->cxDest, pAlphaBlend->cyDest, | 
 |                        hdcSrc, pAlphaBlend->xSrc, pAlphaBlend->ySrc, pAlphaBlend->cxSrc, pAlphaBlend->cySrc, | 
 |                        blendfn); | 
 |  | 
 |             SelectObject(hdcSrc, hBmpOld); | 
 |             DeleteObject(hBmp); | 
 |             DeleteDC(hdcSrc); | 
 |         } | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_MASKBLT: | 
 |     { | 
 | 	const EMRMASKBLT *pMaskBlt = (const EMRMASKBLT *)mr; | 
 | 	HDC hdcSrc = CreateCompatibleDC(hdc); | 
 | 	HBRUSH hBrush, hBrushOld; | 
 | 	HBITMAP hBmp, hBmpOld, hBmpMask; | 
 | 	const BITMAPINFO *pbi; | 
 |  | 
 | 	SetWorldTransform(hdcSrc, &pMaskBlt->xformSrc); | 
 |  | 
 | 	hBrush = CreateSolidBrush(pMaskBlt->crBkColorSrc); | 
 | 	hBrushOld = SelectObject(hdcSrc, hBrush); | 
 | 	PatBlt(hdcSrc, pMaskBlt->rclBounds.left, pMaskBlt->rclBounds.top, | 
 | 	       pMaskBlt->rclBounds.right - pMaskBlt->rclBounds.left, | 
 | 	       pMaskBlt->rclBounds.bottom - pMaskBlt->rclBounds.top, PATCOPY); | 
 | 	SelectObject(hdcSrc, hBrushOld); | 
 | 	DeleteObject(hBrush); | 
 |  | 
 | 	pbi = (const BITMAPINFO *)((const BYTE *)mr + pMaskBlt->offBmiMask); | 
 | 	hBmpMask = CreateBitmap(pbi->bmiHeader.biWidth, pbi->bmiHeader.biHeight, | 
 | 	             1, 1, NULL); | 
 | 	SetDIBits(hdc, hBmpMask, 0, pbi->bmiHeader.biHeight, | 
 | 	  (const BYTE *)mr + pMaskBlt->offBitsMask, pbi, pMaskBlt->iUsageMask); | 
 |  | 
 | 	pbi = (const BITMAPINFO *)((const BYTE *)mr + pMaskBlt->offBmiSrc); | 
 | 	hBmp = CreateDIBitmap(hdc, (const BITMAPINFOHEADER *)pbi, CBM_INIT, | 
 | 			      (const BYTE *)mr + pMaskBlt->offBitsSrc, pbi, pMaskBlt->iUsageSrc); | 
 | 	hBmpOld = SelectObject(hdcSrc, hBmp); | 
 | 	MaskBlt(hdc, | 
 | 		pMaskBlt->xDest, | 
 | 	        pMaskBlt->yDest, | 
 | 	        pMaskBlt->cxDest, | 
 | 	        pMaskBlt->cyDest, | 
 | 	        hdcSrc, | 
 | 	        pMaskBlt->xSrc, | 
 | 	        pMaskBlt->ySrc, | 
 | 	        hBmpMask, | 
 | 		pMaskBlt->xMask, | 
 | 		pMaskBlt->yMask, | 
 | 	        pMaskBlt->dwRop); | 
 | 	SelectObject(hdcSrc, hBmpOld); | 
 | 	DeleteObject(hBmp); | 
 | 	DeleteObject(hBmpMask); | 
 | 	DeleteDC(hdcSrc); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_PLGBLT: | 
 |     { | 
 | 	const EMRPLGBLT *pPlgBlt = (const EMRPLGBLT *)mr; | 
 | 	HDC hdcSrc = CreateCompatibleDC(hdc); | 
 | 	HBRUSH hBrush, hBrushOld; | 
 | 	HBITMAP hBmp, hBmpOld, hBmpMask; | 
 | 	const BITMAPINFO *pbi; | 
 | 	POINT pts[3]; | 
 |  | 
 | 	SetWorldTransform(hdcSrc, &pPlgBlt->xformSrc); | 
 |  | 
 | 	pts[0].x = pPlgBlt->aptlDest[0].x; pts[0].y = pPlgBlt->aptlDest[0].y; | 
 | 	pts[1].x = pPlgBlt->aptlDest[1].x; pts[1].y = pPlgBlt->aptlDest[1].y; | 
 | 	pts[2].x = pPlgBlt->aptlDest[2].x; pts[2].y = pPlgBlt->aptlDest[2].y; | 
 |  | 
 | 	hBrush = CreateSolidBrush(pPlgBlt->crBkColorSrc); | 
 | 	hBrushOld = SelectObject(hdcSrc, hBrush); | 
 | 	PatBlt(hdcSrc, pPlgBlt->rclBounds.left, pPlgBlt->rclBounds.top, | 
 | 	       pPlgBlt->rclBounds.right - pPlgBlt->rclBounds.left, | 
 | 	       pPlgBlt->rclBounds.bottom - pPlgBlt->rclBounds.top, PATCOPY); | 
 | 	SelectObject(hdcSrc, hBrushOld); | 
 | 	DeleteObject(hBrush); | 
 |  | 
 | 	pbi = (const BITMAPINFO *)((const BYTE *)mr + pPlgBlt->offBmiMask); | 
 | 	hBmpMask = CreateBitmap(pbi->bmiHeader.biWidth, pbi->bmiHeader.biHeight, | 
 | 	             1, 1, NULL); | 
 | 	SetDIBits(hdc, hBmpMask, 0, pbi->bmiHeader.biHeight, | 
 | 	  (const BYTE *)mr + pPlgBlt->offBitsMask, pbi, pPlgBlt->iUsageMask); | 
 |  | 
 | 	pbi = (const BITMAPINFO *)((const BYTE *)mr + pPlgBlt->offBmiSrc); | 
 | 	hBmp = CreateDIBitmap(hdc, (const BITMAPINFOHEADER *)pbi, CBM_INIT, | 
 | 			      (const BYTE *)mr + pPlgBlt->offBitsSrc, pbi, pPlgBlt->iUsageSrc); | 
 | 	hBmpOld = SelectObject(hdcSrc, hBmp); | 
 | 	PlgBlt(hdc, | 
 | 	       pts, | 
 | 	       hdcSrc, | 
 | 	       pPlgBlt->xSrc, | 
 | 	       pPlgBlt->ySrc, | 
 | 	       pPlgBlt->cxSrc, | 
 | 	       pPlgBlt->cySrc, | 
 | 	       hBmpMask, | 
 | 	       pPlgBlt->xMask, | 
 | 	       pPlgBlt->yMask); | 
 | 	SelectObject(hdcSrc, hBmpOld); | 
 | 	DeleteObject(hBmp); | 
 | 	DeleteObject(hBmpMask); | 
 | 	DeleteDC(hdcSrc); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_SETDIBITSTODEVICE: | 
 |     { | 
 | 	const EMRSETDIBITSTODEVICE *pSetDIBitsToDevice = (const EMRSETDIBITSTODEVICE *)mr; | 
 |  | 
 | 	SetDIBitsToDevice(hdc, | 
 | 			  pSetDIBitsToDevice->xDest, | 
 | 			  pSetDIBitsToDevice->yDest, | 
 | 			  pSetDIBitsToDevice->cxSrc, | 
 | 			  pSetDIBitsToDevice->cySrc, | 
 | 			  pSetDIBitsToDevice->xSrc, | 
 | 			  pSetDIBitsToDevice->ySrc, | 
 | 			  pSetDIBitsToDevice->iStartScan, | 
 | 			  pSetDIBitsToDevice->cScans, | 
 | 			  (const BYTE *)mr + pSetDIBitsToDevice->offBitsSrc, | 
 | 			  (const BITMAPINFO *)((const BYTE *)mr + pSetDIBitsToDevice->offBmiSrc), | 
 | 			  pSetDIBitsToDevice->iUsageSrc); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_POLYTEXTOUTA: | 
 |     { | 
 | 	const EMRPOLYTEXTOUTA *pPolyTextOutA = (const EMRPOLYTEXTOUTA *)mr; | 
 | 	POLYTEXTA *polytextA = HeapAlloc(GetProcessHeap(), 0, pPolyTextOutA->cStrings * sizeof(POLYTEXTA)); | 
 | 	LONG i; | 
 | 	XFORM xform, xformOld; | 
 | 	int gModeOld; | 
 |  | 
 | 	gModeOld = SetGraphicsMode(hdc, pPolyTextOutA->iGraphicsMode); | 
 | 	GetWorldTransform(hdc, &xformOld); | 
 |  | 
 | 	xform.eM11 = pPolyTextOutA->exScale; | 
 | 	xform.eM12 = 0.0; | 
 | 	xform.eM21 = 0.0; | 
 | 	xform.eM22 = pPolyTextOutA->eyScale; | 
 | 	xform.eDx = 0.0; | 
 | 	xform.eDy = 0.0; | 
 | 	SetWorldTransform(hdc, &xform); | 
 |  | 
 | 	/* Set up POLYTEXTA structures */ | 
 | 	for(i = 0; i < pPolyTextOutA->cStrings; i++) | 
 | 	{ | 
 | 	    polytextA[i].x = pPolyTextOutA->aemrtext[i].ptlReference.x; | 
 | 	    polytextA[i].y = pPolyTextOutA->aemrtext[i].ptlReference.y; | 
 | 	    polytextA[i].n = pPolyTextOutA->aemrtext[i].nChars; | 
 | 	    polytextA[i].lpstr = (LPCSTR)((const BYTE *)mr + pPolyTextOutA->aemrtext[i].offString); | 
 | 	    polytextA[i].uiFlags = pPolyTextOutA->aemrtext[i].fOptions; | 
 | 	    polytextA[i].rcl.left = pPolyTextOutA->aemrtext[i].rcl.left; | 
 | 	    polytextA[i].rcl.right = pPolyTextOutA->aemrtext[i].rcl.right; | 
 | 	    polytextA[i].rcl.top = pPolyTextOutA->aemrtext[i].rcl.top; | 
 | 	    polytextA[i].rcl.bottom = pPolyTextOutA->aemrtext[i].rcl.bottom; | 
 | 	    polytextA[i].pdx = (int *)((BYTE *)mr + pPolyTextOutA->aemrtext[i].offDx); | 
 | 	} | 
 | 	PolyTextOutA(hdc, polytextA, pPolyTextOutA->cStrings); | 
 | 	HeapFree(GetProcessHeap(), 0, polytextA); | 
 |  | 
 | 	SetWorldTransform(hdc, &xformOld); | 
 | 	SetGraphicsMode(hdc, gModeOld); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_POLYTEXTOUTW: | 
 |     { | 
 | 	const EMRPOLYTEXTOUTW *pPolyTextOutW = (const EMRPOLYTEXTOUTW *)mr; | 
 | 	POLYTEXTW *polytextW = HeapAlloc(GetProcessHeap(), 0, pPolyTextOutW->cStrings * sizeof(POLYTEXTW)); | 
 | 	LONG i; | 
 | 	XFORM xform, xformOld; | 
 | 	int gModeOld; | 
 |  | 
 | 	gModeOld = SetGraphicsMode(hdc, pPolyTextOutW->iGraphicsMode); | 
 | 	GetWorldTransform(hdc, &xformOld); | 
 |  | 
 | 	xform.eM11 = pPolyTextOutW->exScale; | 
 | 	xform.eM12 = 0.0; | 
 | 	xform.eM21 = 0.0; | 
 | 	xform.eM22 = pPolyTextOutW->eyScale; | 
 | 	xform.eDx = 0.0; | 
 | 	xform.eDy = 0.0; | 
 | 	SetWorldTransform(hdc, &xform); | 
 |  | 
 | 	/* Set up POLYTEXTW structures */ | 
 | 	for(i = 0; i < pPolyTextOutW->cStrings; i++) | 
 | 	{ | 
 | 	    polytextW[i].x = pPolyTextOutW->aemrtext[i].ptlReference.x; | 
 | 	    polytextW[i].y = pPolyTextOutW->aemrtext[i].ptlReference.y; | 
 | 	    polytextW[i].n = pPolyTextOutW->aemrtext[i].nChars; | 
 | 	    polytextW[i].lpstr = (LPCWSTR)((const BYTE *)mr + pPolyTextOutW->aemrtext[i].offString); | 
 | 	    polytextW[i].uiFlags = pPolyTextOutW->aemrtext[i].fOptions; | 
 | 	    polytextW[i].rcl.left = pPolyTextOutW->aemrtext[i].rcl.left; | 
 | 	    polytextW[i].rcl.right = pPolyTextOutW->aemrtext[i].rcl.right; | 
 | 	    polytextW[i].rcl.top = pPolyTextOutW->aemrtext[i].rcl.top; | 
 | 	    polytextW[i].rcl.bottom = pPolyTextOutW->aemrtext[i].rcl.bottom; | 
 | 	    polytextW[i].pdx = (int *)((BYTE *)mr + pPolyTextOutW->aemrtext[i].offDx); | 
 | 	} | 
 | 	PolyTextOutW(hdc, polytextW, pPolyTextOutW->cStrings); | 
 | 	HeapFree(GetProcessHeap(), 0, polytextW); | 
 |  | 
 | 	SetWorldTransform(hdc, &xformOld); | 
 | 	SetGraphicsMode(hdc, gModeOld); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_FILLRGN: | 
 |     { | 
 | 	const EMRFILLRGN *pFillRgn = (const EMRFILLRGN *)mr; | 
 | 	HRGN hRgn = ExtCreateRegion(NULL, pFillRgn->cbRgnData, (RGNDATA *)pFillRgn->RgnData); | 
 | 	FillRgn(hdc, | 
 | 		hRgn, | 
 | 		(handletable->objectHandle)[pFillRgn->ihBrush]); | 
 | 	DeleteObject(hRgn); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_FRAMERGN: | 
 |     { | 
 | 	const EMRFRAMERGN *pFrameRgn = (const EMRFRAMERGN *)mr; | 
 | 	HRGN hRgn = ExtCreateRegion(NULL, pFrameRgn->cbRgnData, (RGNDATA *)pFrameRgn->RgnData); | 
 | 	FrameRgn(hdc, | 
 | 		 hRgn, | 
 | 		 (handletable->objectHandle)[pFrameRgn->ihBrush], | 
 | 		 pFrameRgn->szlStroke.cx, | 
 | 		 pFrameRgn->szlStroke.cy); | 
 | 	DeleteObject(hRgn); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_INVERTRGN: | 
 |     { | 
 | 	const EMRINVERTRGN *pInvertRgn = (const EMRINVERTRGN *)mr; | 
 | 	HRGN hRgn = ExtCreateRegion(NULL, pInvertRgn->cbRgnData, (RGNDATA *)pInvertRgn->RgnData); | 
 | 	InvertRgn(hdc, hRgn); | 
 | 	DeleteObject(hRgn); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_PAINTRGN: | 
 |     { | 
 | 	const EMRPAINTRGN *pPaintRgn = (const EMRPAINTRGN *)mr; | 
 | 	HRGN hRgn = ExtCreateRegion(NULL, pPaintRgn->cbRgnData, (RGNDATA *)pPaintRgn->RgnData); | 
 | 	PaintRgn(hdc, hRgn); | 
 | 	DeleteObject(hRgn); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_SETTEXTJUSTIFICATION: | 
 |     { | 
 | 	const EMRSETTEXTJUSTIFICATION *pSetTextJust = (const EMRSETTEXTJUSTIFICATION *)mr; | 
 | 	SetTextJustification(hdc, pSetTextJust->nBreakExtra, pSetTextJust->nBreakCount); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_SETLAYOUT: | 
 |     { | 
 | 	const EMRSETLAYOUT *pSetLayout = (const EMRSETLAYOUT *)mr; | 
 | 	SetLayout(hdc, pSetLayout->iMode); | 
 | 	break; | 
 |     } | 
 |  | 
 |     case EMR_POLYDRAW16: | 
 |     case EMR_GLSRECORD: | 
 |     case EMR_GLSBOUNDEDRECORD: | 
 | 	case EMR_DRAWESCAPE : | 
 | 	case EMR_EXTESCAPE: | 
 | 	case EMR_STARTDOC: | 
 | 	case EMR_SMALLTEXTOUT: | 
 | 	case EMR_FORCEUFIMAPPING: | 
 | 	case EMR_NAMEDESCAPE: | 
 | 	case EMR_COLORCORRECTPALETTE: | 
 | 	case EMR_SETICMPROFILEA: | 
 | 	case EMR_SETICMPROFILEW: | 
 | 	case EMR_TRANSPARENTBLT: | 
 | 	case EMR_GRADIENTFILL: | 
 | 	case EMR_SETLINKEDUFI: | 
 | 	case EMR_COLORMATCHTOTARGETW: | 
 | 	case EMR_CREATECOLORSPACEW: | 
 |  | 
 |     default: | 
 |       /* From docs: If PlayEnhMetaFileRecord doesn't recognize a | 
 |                     record then ignore and return TRUE. */ | 
 |       FIXME("type %d is unimplemented\n", type); | 
 |       break; | 
 |     } | 
 |   tmprc.left = tmprc.top = 0; | 
 |   tmprc.right = tmprc.bottom = 1000; | 
 |   LPtoDP(hdc, (POINT*)&tmprc, 2); | 
 |   TRACE("L:0,0 - 1000,1000 -> D:%d,%d - %d,%d\n", tmprc.left, | 
 | 	tmprc.top, tmprc.right, tmprc.bottom); | 
 |  | 
 |   return TRUE; | 
 | } | 
 |  | 
 |  | 
 | /***************************************************************************** | 
 |  * | 
 |  *        EnumEnhMetaFile  (GDI32.@) | 
 |  * | 
 |  *  Walk an enhanced metafile, calling a user-specified function _EnhMetaFunc_ | 
 |  *  for each | 
 |  *  record. Returns when either every record has been used or | 
 |  *  when _EnhMetaFunc_ returns FALSE. | 
 |  * | 
 |  * | 
 |  * RETURNS | 
 |  *  TRUE if every record is used, FALSE if any invocation of _EnhMetaFunc_ | 
 |  *  returns FALSE. | 
 |  * | 
 |  * BUGS | 
 |  *   Ignores rect. | 
 |  * | 
 |  * NOTES | 
 |  *   This function behaves differently in Win9x and WinNT. | 
 |  * | 
 |  *   In WinNT, the DC's world transform is updated as the EMF changes | 
 |  *    the Window/Viewport Extent and Origin or it's world transform. | 
 |  *    The actual Window/Viewport Extent and Origin are left untouched. | 
 |  * | 
 |  *   In Win9x, the DC is left untouched, and PlayEnhMetaFileRecord | 
 |  *    updates the scaling itself but only just before a record that | 
 |  *    writes anything to the DC. | 
 |  * | 
 |  *   I'm not sure where the data (enum_emh_data) is stored in either | 
 |  *    version. For this implementation, it is stored before the handle | 
 |  *    table, but it could be stored in the DC, in the EMF handle or in | 
 |  *    TLS. | 
 |  *             MJM  5 Oct 2002 | 
 |  */ | 
 | BOOL WINAPI EnumEnhMetaFile( | 
 |      HDC hdc,                /* [in] device context to pass to _EnhMetaFunc_ */ | 
 |      HENHMETAFILE hmf,       /* [in] EMF to walk */ | 
 |      ENHMFENUMPROC callback, /* [in] callback function */ | 
 |      LPVOID data,            /* [in] optional data for callback function */ | 
 |      const RECT *lpRect      /* [in] bounding rectangle for rendered metafile */ | 
 |     ) | 
 | { | 
 |     BOOL ret; | 
 |     ENHMETAHEADER *emh; | 
 |     ENHMETARECORD *emr; | 
 |     DWORD offset; | 
 |     UINT i; | 
 |     HANDLETABLE *ht; | 
 |     INT savedMode = 0; | 
 |     XFORM savedXform; | 
 |     HPEN hPen = NULL; | 
 |     HBRUSH hBrush = NULL; | 
 |     HFONT hFont = NULL; | 
 |     HRGN hRgn = NULL; | 
 |     enum_emh_data *info; | 
 |     SIZE vp_size, win_size; | 
 |     POINT vp_org, win_org; | 
 |     INT mapMode = MM_TEXT, old_align = 0, old_rop2 = 0, old_arcdir = 0, old_polyfill = 0, old_stretchblt = 0; | 
 |     COLORREF old_text_color = 0, old_bk_color = 0; | 
 |  | 
 |     if(!lpRect && hdc) | 
 |     { | 
 | 	SetLastError(ERROR_INVALID_PARAMETER); | 
 | 	return FALSE; | 
 |     } | 
 |  | 
 |     emh = EMF_GetEnhMetaHeader(hmf); | 
 |     if(!emh) { | 
 |         SetLastError(ERROR_INVALID_HANDLE); | 
 |         return FALSE; | 
 |     } | 
 |  | 
 |     info = HeapAlloc( GetProcessHeap(), 0, | 
 | 		    sizeof (enum_emh_data) + sizeof(HANDLETABLE) * emh->nHandles ); | 
 |     if(!info) | 
 |     { | 
 | 	SetLastError(ERROR_NOT_ENOUGH_MEMORY); | 
 | 	return FALSE; | 
 |     } | 
 |     info->state.wndOrgX = 0; | 
 |     info->state.wndOrgY = 0; | 
 |     info->state.wndExtX = 1; | 
 |     info->state.wndExtY = 1; | 
 |     info->state.vportOrgX = 0; | 
 |     info->state.vportOrgY = 0; | 
 |     info->state.vportExtX = 1; | 
 |     info->state.vportExtY = 1; | 
 |     info->state.world_transform.eM11 = info->state.world_transform.eM22 = 1; | 
 |     info->state.world_transform.eM12 = info->state.world_transform.eM21 = 0; | 
 |     info->state.world_transform.eDx  = info->state.world_transform.eDy =  0; | 
 |  | 
 |     info->state.next = NULL; | 
 |     info->save_level = 0; | 
 |     info->saved_state = NULL; | 
 |  | 
 |     ht = (HANDLETABLE*) &info[1]; | 
 |     ht->objectHandle[0] = hmf; | 
 |     for(i = 1; i < emh->nHandles; i++) | 
 |         ht->objectHandle[i] = NULL; | 
 |  | 
 |     if(hdc) | 
 |     { | 
 | 	savedMode = SetGraphicsMode(hdc, GM_ADVANCED); | 
 | 	GetWorldTransform(hdc, &savedXform); | 
 |         GetViewportExtEx(hdc, &vp_size); | 
 |         GetWindowExtEx(hdc, &win_size); | 
 |         GetViewportOrgEx(hdc, &vp_org); | 
 |         GetWindowOrgEx(hdc, &win_org); | 
 |         mapMode = GetMapMode(hdc); | 
 |  | 
 | 	/* save DC */ | 
 | 	hPen = GetCurrentObject(hdc, OBJ_PEN); | 
 | 	hBrush = GetCurrentObject(hdc, OBJ_BRUSH); | 
 | 	hFont = GetCurrentObject(hdc, OBJ_FONT); | 
 |  | 
 |         hRgn = CreateRectRgn(0, 0, 0, 0); | 
 |         if (!GetClipRgn(hdc, hRgn)) | 
 |         { | 
 |             DeleteObject(hRgn); | 
 |             hRgn = 0; | 
 |         } | 
 |  | 
 |         old_text_color = SetTextColor(hdc, RGB(0,0,0)); | 
 |         old_bk_color = SetBkColor(hdc, RGB(0xff, 0xff, 0xff)); | 
 |         old_align = SetTextAlign(hdc, 0); | 
 |         old_rop2 = SetROP2(hdc, R2_COPYPEN); | 
 |         old_arcdir = SetArcDirection(hdc, AD_COUNTERCLOCKWISE); | 
 |         old_polyfill = SetPolyFillMode(hdc, ALTERNATE); | 
 |         old_stretchblt = SetStretchBltMode(hdc, BLACKONWHITE); | 
 |     } | 
 |  | 
 |     info->state.mode = MM_TEXT; | 
 |  | 
 |     if ( IS_WIN9X() ) | 
 |     { | 
 |         /* Win95 leaves the vp/win ext/org info alone */ | 
 |         info->init_transform.eM11 = 1.0; | 
 |         info->init_transform.eM12 = 0.0; | 
 |         info->init_transform.eM21 = 0.0; | 
 |         info->init_transform.eM22 = 1.0; | 
 |         info->init_transform.eDx  = 0.0; | 
 |         info->init_transform.eDy  = 0.0; | 
 |     } | 
 |     else | 
 |     { | 
 |         /* WinNT combines the vp/win ext/org info into a transform */ | 
 |         double xscale, yscale; | 
 |         xscale = (double)vp_size.cx / (double)win_size.cx; | 
 |         yscale = (double)vp_size.cy / (double)win_size.cy; | 
 |         info->init_transform.eM11 = xscale; | 
 |         info->init_transform.eM12 = 0.0; | 
 |         info->init_transform.eM21 = 0.0; | 
 |         info->init_transform.eM22 = yscale; | 
 |         info->init_transform.eDx  = (double)vp_org.x - xscale * (double)win_org.x; | 
 |         info->init_transform.eDy  = (double)vp_org.y - yscale * (double)win_org.y; | 
 |  | 
 |         CombineTransform(&info->init_transform, &savedXform, &info->init_transform); | 
 |     } | 
 |  | 
 |     if ( lpRect && WIDTH(emh->rclFrame) && HEIGHT(emh->rclFrame) ) | 
 |     { | 
 |         double xSrcPixSize, ySrcPixSize, xscale, yscale; | 
 |         XFORM xform; | 
 |  | 
 |         TRACE("rect: %d,%d - %d,%d. rclFrame: %d,%d - %d,%d\n", | 
 |            lpRect->left, lpRect->top, lpRect->right, lpRect->bottom, | 
 |            emh->rclFrame.left, emh->rclFrame.top, emh->rclFrame.right, | 
 |            emh->rclFrame.bottom); | 
 |  | 
 |         xSrcPixSize = (double) emh->szlMillimeters.cx / emh->szlDevice.cx; | 
 |         ySrcPixSize = (double) emh->szlMillimeters.cy / emh->szlDevice.cy; | 
 |         xscale = (double) WIDTH(*lpRect) * 100.0 / | 
 |                  WIDTH(emh->rclFrame) * xSrcPixSize; | 
 |         yscale = (double) HEIGHT(*lpRect) * 100.0 / | 
 |                  HEIGHT(emh->rclFrame) * ySrcPixSize; | 
 |         TRACE("xscale = %f, yscale = %f\n", xscale, yscale); | 
 |  | 
 |         xform.eM11 = xscale; | 
 |         xform.eM12 = 0; | 
 |         xform.eM21 = 0; | 
 |         xform.eM22 = yscale; | 
 |         xform.eDx = (double) lpRect->left - (double) WIDTH(*lpRect) / WIDTH(emh->rclFrame) * emh->rclFrame.left; | 
 |         xform.eDy = (double) lpRect->top - (double) HEIGHT(*lpRect) / HEIGHT(emh->rclFrame) * emh->rclFrame.top; | 
 |  | 
 |         CombineTransform(&info->init_transform, &xform, &info->init_transform); | 
 |     } | 
 |  | 
 |     /* WinNT resets the current vp/win org/ext */ | 
 |     if ( !IS_WIN9X() && hdc ) | 
 |     { | 
 |         SetMapMode(hdc, MM_TEXT); | 
 |         SetWindowOrgEx(hdc, 0, 0, NULL); | 
 |         SetViewportOrgEx(hdc, 0, 0, NULL); | 
 |         EMF_Update_MF_Xform(hdc, info); | 
 |     } | 
 |  | 
 |     ret = TRUE; | 
 |     offset = 0; | 
 |     while(ret && offset < emh->nBytes) | 
 |     { | 
 | 	emr = (ENHMETARECORD *)((char *)emh + offset); | 
 |  | 
 |         /* In Win9x mode we update the xform if the record will produce output */ | 
 |         if (hdc && IS_WIN9X() && emr_produces_output(emr->iType)) | 
 |             EMF_Update_MF_Xform(hdc, info); | 
 |  | 
 | 	TRACE("Calling EnumFunc with record %s, size %d\n", get_emr_name(emr->iType), emr->nSize); | 
 | 	ret = (*callback)(hdc, ht, emr, emh->nHandles, (LPARAM)data); | 
 | 	offset += emr->nSize; | 
 |  | 
 |         /* WinNT - update the transform (win9x updates when the next graphics | 
 |            output record is played). */ | 
 |         if (hdc && !IS_WIN9X()) | 
 |             EMF_Update_MF_Xform(hdc, info); | 
 |     } | 
 |  | 
 |     if (hdc) | 
 |     { | 
 |         SetStretchBltMode(hdc, old_stretchblt); | 
 |         SetPolyFillMode(hdc, old_polyfill); | 
 |         SetArcDirection(hdc, old_arcdir); | 
 |         SetROP2(hdc, old_rop2); | 
 |         SetTextAlign(hdc, old_align); | 
 |         SetBkColor(hdc, old_bk_color); | 
 |         SetTextColor(hdc, old_text_color); | 
 |  | 
 | 	/* restore DC */ | 
 | 	SelectObject(hdc, hBrush); | 
 | 	SelectObject(hdc, hPen); | 
 | 	SelectObject(hdc, hFont); | 
 |         ExtSelectClipRgn(hdc, hRgn, RGN_COPY); | 
 |         DeleteObject(hRgn); | 
 |  | 
 | 	SetWorldTransform(hdc, &savedXform); | 
 | 	if (savedMode) | 
 | 	    SetGraphicsMode(hdc, savedMode); | 
 |         SetMapMode(hdc, mapMode); | 
 |         SetWindowOrgEx(hdc, win_org.x, win_org.y, NULL); | 
 |         SetWindowExtEx(hdc, win_size.cx, win_size.cy, NULL); | 
 |         SetViewportOrgEx(hdc, vp_org.x, vp_org.y, NULL); | 
 |         SetViewportExtEx(hdc, vp_size.cx, vp_size.cy, NULL); | 
 |     } | 
 |  | 
 |     for(i = 1; i < emh->nHandles; i++) /* Don't delete element 0 (hmf) */ | 
 |         if( (ht->objectHandle)[i] ) | 
 | 	    DeleteObject( (ht->objectHandle)[i] ); | 
 |  | 
 |     while (info->saved_state) | 
 |     { | 
 |         EMF_dc_state *state = info->saved_state; | 
 |         info->saved_state = info->saved_state->next; | 
 |         HeapFree( GetProcessHeap(), 0, state ); | 
 |     } | 
 |     HeapFree( GetProcessHeap(), 0, info ); | 
 |     return ret; | 
 | } | 
 |  | 
 | static INT CALLBACK EMF_PlayEnhMetaFileCallback(HDC hdc, HANDLETABLE *ht, | 
 | 						const ENHMETARECORD *emr, | 
 | 						INT handles, LPARAM data) | 
 | { | 
 |     return PlayEnhMetaFileRecord(hdc, ht, emr, handles); | 
 | } | 
 |  | 
 | /************************************************************************** | 
 |  *    PlayEnhMetaFile  (GDI32.@) | 
 |  * | 
 |  *    Renders an enhanced metafile into a specified rectangle *lpRect | 
 |  *    in device context hdc. | 
 |  * | 
 |  * RETURNS | 
 |  *    Success: TRUE | 
 |  *    Failure: FALSE | 
 |  */ | 
 | BOOL WINAPI PlayEnhMetaFile( | 
 |        HDC hdc,           /* [in] DC to render into */ | 
 |        HENHMETAFILE hmf,  /* [in] metafile to render */ | 
 |        const RECT *lpRect /* [in] rectangle to place metafile inside */ | 
 |       ) | 
 | { | 
 |     return EnumEnhMetaFile(hdc, hmf, EMF_PlayEnhMetaFileCallback, NULL, | 
 | 			   lpRect); | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *  DeleteEnhMetaFile (GDI32.@) | 
 |  * | 
 |  *  Deletes an enhanced metafile and frees the associated storage. | 
 |  */ | 
 | BOOL WINAPI DeleteEnhMetaFile(HENHMETAFILE hmf) | 
 | { | 
 |     return EMF_Delete_HENHMETAFILE( hmf ); | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *  CopyEnhMetaFileA (GDI32.@) | 
 |  * | 
 |  * Duplicate an enhanced metafile. | 
 |  * | 
 |  * | 
 |  */ | 
 | HENHMETAFILE WINAPI CopyEnhMetaFileA( | 
 |     HENHMETAFILE hmfSrc, | 
 |     LPCSTR file) | 
 | { | 
 |     ENHMETAHEADER *emrSrc = EMF_GetEnhMetaHeader( hmfSrc ), *emrDst; | 
 |     HENHMETAFILE hmfDst; | 
 |  | 
 |     if(!emrSrc) return FALSE; | 
 |     if (!file) { | 
 |         emrDst = HeapAlloc( GetProcessHeap(), 0, emrSrc->nBytes ); | 
 | 	memcpy( emrDst, emrSrc, emrSrc->nBytes ); | 
 | 	hmfDst = EMF_Create_HENHMETAFILE( emrDst, FALSE ); | 
 |     } else { | 
 |         HANDLE hFile; | 
 |         DWORD w; | 
 |         hFile = CreateFileA( file, GENERIC_WRITE | GENERIC_READ, 0, | 
 | 			     NULL, CREATE_ALWAYS, 0, 0); | 
 | 	WriteFile( hFile, emrSrc, emrSrc->nBytes, &w, NULL); | 
 | 	CloseHandle( hFile ); | 
 | 	/* Reopen file for reading only, so that apps can share | 
 | 	   read access to the file while hmf is still valid */ | 
 |         hFile = CreateFileA( file, GENERIC_READ, FILE_SHARE_READ, | 
 | 			     NULL, OPEN_EXISTING, 0, 0); | 
 | 	if(hFile == INVALID_HANDLE_VALUE) { | 
 | 	    ERR("Can't reopen emf for reading\n"); | 
 | 	    return 0; | 
 | 	} | 
 | 	hmfDst = EMF_GetEnhMetaFile( hFile ); | 
 |         CloseHandle( hFile ); | 
 |     } | 
 |     return hmfDst; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *  CopyEnhMetaFileW (GDI32.@) | 
 |  * | 
 |  * See CopyEnhMetaFileA. | 
 |  * | 
 |  * | 
 |  */ | 
 | HENHMETAFILE WINAPI CopyEnhMetaFileW( | 
 |     HENHMETAFILE hmfSrc, | 
 |     LPCWSTR file) | 
 | { | 
 |     ENHMETAHEADER *emrSrc = EMF_GetEnhMetaHeader( hmfSrc ), *emrDst; | 
 |     HENHMETAFILE hmfDst; | 
 |  | 
 |     if(!emrSrc) return FALSE; | 
 |     if (!file) { | 
 |         emrDst = HeapAlloc( GetProcessHeap(), 0, emrSrc->nBytes ); | 
 | 	memcpy( emrDst, emrSrc, emrSrc->nBytes ); | 
 | 	hmfDst = EMF_Create_HENHMETAFILE( emrDst, FALSE ); | 
 |     } else { | 
 |         HANDLE hFile; | 
 |         DWORD w; | 
 |         hFile = CreateFileW( file, GENERIC_WRITE | GENERIC_READ, 0, | 
 | 			     NULL, CREATE_ALWAYS, 0, 0); | 
 | 	WriteFile( hFile, emrSrc, emrSrc->nBytes, &w, NULL); | 
 | 	CloseHandle( hFile ); | 
 | 	/* Reopen file for reading only, so that apps can share | 
 | 	   read access to the file while hmf is still valid */ | 
 |         hFile = CreateFileW( file, GENERIC_READ, FILE_SHARE_READ, | 
 | 			     NULL, OPEN_EXISTING, 0, 0); | 
 | 	if(hFile == INVALID_HANDLE_VALUE) { | 
 | 	    ERR("Can't reopen emf for reading\n"); | 
 | 	    return 0; | 
 | 	} | 
 | 	hmfDst = EMF_GetEnhMetaFile( hFile ); | 
 |         CloseHandle( hFile ); | 
 |     } | 
 |     return hmfDst; | 
 | } | 
 |  | 
 |  | 
 | /* Struct to be used to be passed in the LPVOID parameter for cbEnhPaletteCopy */ | 
 | typedef struct tagEMF_PaletteCopy | 
 | { | 
 |    UINT cEntries; | 
 |    LPPALETTEENTRY lpPe; | 
 | } EMF_PaletteCopy; | 
 |  | 
 | /*************************************************************** | 
 |  * Find the EMR_EOF record and then use it to find the | 
 |  * palette entries for this enhanced metafile. | 
 |  * The lpData is actually a pointer to an EMF_PaletteCopy struct | 
 |  * which contains the max number of elements to copy and where | 
 |  * to copy them to. | 
 |  * | 
 |  * NOTE: To be used by GetEnhMetaFilePaletteEntries only! | 
 |  */ | 
 | static INT CALLBACK cbEnhPaletteCopy( HDC a, | 
 |                                HANDLETABLE *b, | 
 |                                const ENHMETARECORD *lpEMR, | 
 |                                INT c, | 
 |                                LPARAM lpData ) | 
 | { | 
 |  | 
 |   if ( lpEMR->iType == EMR_EOF ) | 
 |   { | 
 |     const EMREOF *lpEof = (const EMREOF *)lpEMR; | 
 |     EMF_PaletteCopy* info = (EMF_PaletteCopy*)lpData; | 
 |     DWORD dwNumPalToCopy = min( lpEof->nPalEntries, info->cEntries ); | 
 |  | 
 |     TRACE( "copying 0x%08x palettes\n", dwNumPalToCopy ); | 
 |  | 
 |     memcpy( info->lpPe, (LPCSTR)lpEof + lpEof->offPalEntries, | 
 |             sizeof( *(info->lpPe) ) * dwNumPalToCopy ); | 
 |  | 
 |     /* Update the passed data as a return code */ | 
 |     info->lpPe     = NULL; /* Palettes were copied! */ | 
 |     info->cEntries = dwNumPalToCopy; | 
 |  | 
 |     return FALSE; /* That's all we need */ | 
 |   } | 
 |  | 
 |   return TRUE; | 
 | } | 
 |  | 
 | /***************************************************************************** | 
 |  *  GetEnhMetaFilePaletteEntries (GDI32.@) | 
 |  * | 
 |  *  Copy the palette and report size | 
 |  * | 
 |  *  BUGS: Error codes (SetLastError) are not set on failures | 
 |  */ | 
 | UINT WINAPI GetEnhMetaFilePaletteEntries( HENHMETAFILE hEmf, | 
 | 					  UINT cEntries, | 
 | 					  LPPALETTEENTRY lpPe ) | 
 | { | 
 |   ENHMETAHEADER* enhHeader = EMF_GetEnhMetaHeader( hEmf ); | 
 |   EMF_PaletteCopy infoForCallBack; | 
 |  | 
 |   TRACE( "(%p,%d,%p)\n", hEmf, cEntries, lpPe ); | 
 |  | 
 |   if (!enhHeader) return 0; | 
 |  | 
 |   /* First check if there are any palettes associated with | 
 |      this metafile. */ | 
 |   if ( enhHeader->nPalEntries == 0 ) return 0; | 
 |  | 
 |   /* Is the user requesting the number of palettes? */ | 
 |   if ( lpPe == NULL ) return enhHeader->nPalEntries; | 
 |  | 
 |   /* Copy cEntries worth of PALETTEENTRY structs into the buffer */ | 
 |   infoForCallBack.cEntries = cEntries; | 
 |   infoForCallBack.lpPe     = lpPe; | 
 |  | 
 |   if ( !EnumEnhMetaFile( 0, hEmf, cbEnhPaletteCopy, | 
 |                          &infoForCallBack, 0 ) ) | 
 |       return GDI_ERROR; | 
 |  | 
 |   /* Verify that the callback executed correctly */ | 
 |   if ( infoForCallBack.lpPe != NULL ) | 
 |   { | 
 |      /* Callback proc had error! */ | 
 |      ERR( "cbEnhPaletteCopy didn't execute correctly\n" ); | 
 |      return GDI_ERROR; | 
 |   } | 
 |  | 
 |   return infoForCallBack.cEntries; | 
 | } | 
 |  | 
 | typedef struct gdi_mf_comment | 
 | { | 
 |     DWORD ident; | 
 |     DWORD iComment; | 
 |     DWORD nVersion; | 
 |     DWORD nChecksum; | 
 |     DWORD fFlags; | 
 |     DWORD cbWinMetaFile; | 
 | } gdi_mf_comment; | 
 |  | 
 | /****************************************************************** | 
 |  *         SetWinMetaFileBits   (GDI32.@) | 
 |  * | 
 |  *         Translate from old style to new style. | 
 |  * | 
 |  */ | 
 | HENHMETAFILE WINAPI SetWinMetaFileBits(UINT cbBuffer, | 
 | 					   CONST BYTE *lpbBuffer, | 
 | 					   HDC hdcRef, | 
 | 					   CONST METAFILEPICT *lpmfp | 
 | 					   ) | 
 | { | 
 |     static const WCHAR szDisplayW[] = { 'D','I','S','P','L','A','Y','\0' }; | 
 |     HMETAFILE hmf = NULL; | 
 |     HENHMETAFILE ret = NULL; | 
 |     HDC hdc = NULL, hdcdisp = NULL; | 
 |     RECT rc, *prcFrame = NULL; | 
 |     LONG mm, xExt, yExt; | 
 |     INT horzsize, vertsize, horzres, vertres; | 
 |  | 
 |     TRACE("(%d, %p, %p, %p)\n", cbBuffer, lpbBuffer, hdcRef, lpmfp); | 
 |  | 
 |     hmf = SetMetaFileBitsEx(cbBuffer, lpbBuffer); | 
 |     if(!hmf) | 
 |     { | 
 |         WARN("SetMetaFileBitsEx failed\n"); | 
 |         return NULL; | 
 |     } | 
 |  | 
 |     if(!hdcRef) | 
 |         hdcRef = hdcdisp = CreateDCW(szDisplayW, NULL, NULL, NULL); | 
 |  | 
 |     if (lpmfp) | 
 |     { | 
 |         TRACE("mm = %d %dx%d\n", lpmfp->mm, lpmfp->xExt, lpmfp->yExt); | 
 |  | 
 |         mm = lpmfp->mm; | 
 |         xExt = lpmfp->xExt; | 
 |         yExt = lpmfp->yExt; | 
 |     } | 
 |     else | 
 |     { | 
 |         TRACE("lpmfp == NULL\n"); | 
 |  | 
 |         /* Use the whole device surface */ | 
 |         mm = MM_ANISOTROPIC; | 
 |         xExt = 0; | 
 |         yExt = 0; | 
 |     } | 
 |  | 
 |     if (mm == MM_ISOTROPIC || mm == MM_ANISOTROPIC) | 
 |     { | 
 |         if (xExt < 0 || yExt < 0) | 
 |         { | 
 |           /* Use the whole device surface */ | 
 |           xExt = 0; | 
 |           yExt = 0; | 
 |         } | 
 |  | 
 |         /* Use the x and y extents as the frame box */ | 
 |         if (xExt && yExt) | 
 |         { | 
 |             rc.left = rc.top = 0; | 
 |             rc.right = xExt; | 
 |             rc.bottom = yExt; | 
 |             prcFrame = &rc; | 
 |         } | 
 |     } | 
 |  | 
 |     if(!(hdc = CreateEnhMetaFileW(hdcRef, NULL, prcFrame, NULL))) | 
 |     { | 
 |         ERR("CreateEnhMetaFile failed\n"); | 
 |         goto end; | 
 |     } | 
 |  | 
 |     /* | 
 |      * Write the original METAFILE into the enhanced metafile. | 
 |      * It is encapsulated in a GDICOMMENT_WINDOWS_METAFILE record. | 
 |      */ | 
 |     if (mm != MM_TEXT) | 
 |     { | 
 |         gdi_mf_comment *mfcomment; | 
 |         UINT mfcomment_size; | 
 |  | 
 |         mfcomment_size = sizeof (gdi_mf_comment) + cbBuffer; | 
 |         mfcomment = HeapAlloc(GetProcessHeap(), 0, mfcomment_size); | 
 |         if (mfcomment) | 
 |         { | 
 |             mfcomment->ident = GDICOMMENT_IDENTIFIER; | 
 |             mfcomment->iComment = GDICOMMENT_WINDOWS_METAFILE; | 
 |             mfcomment->nVersion = 0x00000300; | 
 |             mfcomment->nChecksum = 0; /* FIXME */ | 
 |             mfcomment->fFlags = 0; | 
 |             mfcomment->cbWinMetaFile = cbBuffer; | 
 |             memcpy(&mfcomment[1], lpbBuffer, cbBuffer); | 
 |             GdiComment(hdc, mfcomment_size, (BYTE*) mfcomment); | 
 |             HeapFree(GetProcessHeap(), 0, mfcomment); | 
 |         } | 
 |         SetMapMode(hdc, mm); | 
 |     } | 
 |  | 
 |  | 
 |     horzsize = GetDeviceCaps(hdcRef, HORZSIZE); | 
 |     vertsize = GetDeviceCaps(hdcRef, VERTSIZE); | 
 |     horzres = GetDeviceCaps(hdcRef, HORZRES); | 
 |     vertres = GetDeviceCaps(hdcRef, VERTRES); | 
 |  | 
 |     if (!xExt || !yExt) | 
 |     { | 
 |         /* Use the whole device surface */ | 
 |        xExt = horzres; | 
 |        yExt = vertres; | 
 |     } | 
 |     else | 
 |     { | 
 |         xExt = MulDiv(xExt, horzres, 100 * horzsize); | 
 |         yExt = MulDiv(yExt, vertres, 100 * vertsize); | 
 |     } | 
 |  | 
 |     /* set the initial viewport:window ratio as 1:1 */ | 
 |     SetViewportExtEx(hdc, xExt, yExt, NULL); | 
 |     SetWindowExtEx(hdc,   xExt, yExt, NULL); | 
 |  | 
 |     PlayMetaFile(hdc, hmf); | 
 |  | 
 |     ret = CloseEnhMetaFile(hdc); | 
 | end: | 
 |     if (hdcdisp) DeleteDC(hdcdisp); | 
 |     DeleteMetaFile(hmf); | 
 |     return ret; | 
 | } |