/* * 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 #include #include #include #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 = 0; ENHMETAFILEOBJ *metaObj; if (emh->iType != EMR_HEADER || 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); SetLastError(ERROR_INVALID_DATA); return 0; } metaObj = GDI_AllocObject( sizeof(ENHMETAFILEOBJ), ENHMETAFILE_MAGIC, (HGDIOBJ *)&hmf, NULL ); if (metaObj) { metaObj->emh = emh; metaObj->on_disk = on_disk; GDI_ReleaseObj( hmf ); } return hmf; } /**************************************************************************** * EMF_Delete_HENHMETAFILE */ static BOOL EMF_Delete_HENHMETAFILE( HENHMETAFILE hmf ) { ENHMETAFILEOBJ *metaObj = (ENHMETAFILEOBJ *)GDI_GetObjPtr( hmf, ENHMETAFILE_MAGIC ); if(!metaObj) return FALSE; if(metaObj->on_disk) UnmapViewOfFile( metaObj->emh ); else HeapFree( GetProcessHeap(), 0, metaObj->emh ); return GDI_FreeObject( hmf, metaObj ); } /****************************************************************** * EMF_GetEnhMetaHeader * * Returns ptr to ENHMETAHEADER associated with HENHMETAFILE */ static ENHMETAHEADER *EMF_GetEnhMetaHeader( HENHMETAFILE hmf ) { ENHMETAHEADER *ret = NULL; ENHMETAFILEOBJ *metaObj = (ENHMETAFILEOBJ *)GDI_GetObjPtr( hmf, ENHMETAFILE_MAGIC ); 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_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; /* In Win9x mode we update the xform if the record will produce output */ if ( IS_WIN9X() && emr_produces_output(type) ) EMF_Update_MF_Xform(hdc, info); 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); if ( !IS_WIN9X() ) { /* WinNT - update the transform (win9x updates when the next graphics output record is played). */ EMF_Update_MF_Xform(hdc, info); } 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; 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; 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 the current pen, brush and font */ hPen = GetCurrentObject(hdc, OBJ_PEN); hBrush = GetCurrentObject(hdc, OBJ_BRUSH); hFont = GetCurrentObject(hdc, OBJ_FONT); 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); 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; } 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 pen, brush and font */ SelectObject(hdc, hBrush); SelectObject(hdc, hPen); SelectObject(hdc, hFont); 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( (LPVOID)info->lpPe, (LPCVOID)(((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; gdi_mf_comment *mfcomment; UINT mfcomment_size; LONG mm, xExt, yExt; 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. */ 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); } if (mm != MM_TEXT) SetMapMode(hdc, mm); if (mm == MM_ISOTROPIC || mm == MM_ANISOTROPIC) { INT horzsize, vertsize, horzres, vertres; 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; }