/* * Enhanced MetaFile recording functions * * Copyright 1999 Huw D M Davies * Copyright 2016 Alexandre Julliard * Copyright 2021 Jacek Caban for CodeWeavers * * 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 */ #include #include "enhmfdrv/enhmetafiledrv.h" #include "winnls.h" #include "wine/debug.h" WINE_DEFAULT_DEBUG_CHANNEL(enhmetafile); BOOL emfdc_record( struct emf *emf, EMR *emr ) { DWORD len, size; ENHMETAHEADER *emh; TRACE( "record %d, size %d\n", emr->iType, emr->nSize ); assert( !(emr->nSize & 3) ); emf->emh->nBytes += emr->nSize; emf->emh->nRecords++; size = HeapSize( GetProcessHeap(), 0, emf->emh ); len = emf->emh->nBytes; if (len > size) { size += (size / 2) + emr->nSize; emh = HeapReAlloc( GetProcessHeap(), 0, emf->emh, size ); if (!emh) return FALSE; emf->emh = emh; } memcpy( (char *)emf->emh + emf->emh->nBytes - emr->nSize, emr, emr->nSize ); return TRUE; } void emfdc_update_bounds( struct emf *emf, RECTL *rect ) { RECTL *bounds = &emf->dc_attr->emf_bounds; RECTL vport_rect = *rect; LPtoDP( emf->dc_attr->hdc, (POINT *)&vport_rect, 2 ); /* The coordinate systems may be mirrored (LPtoDP handles points, not rectangles) */ if (vport_rect.left > vport_rect.right) { LONG temp = vport_rect.right; vport_rect.right = vport_rect.left; vport_rect.left = temp; } if (vport_rect.top > vport_rect.bottom) { LONG temp = vport_rect.bottom; vport_rect.bottom = vport_rect.top; vport_rect.top = temp; } if (bounds->left > bounds->right) { /* first bounding rectangle */ *bounds = vport_rect; } else { bounds->left = min(bounds->left, vport_rect.left); bounds->top = min(bounds->top, vport_rect.top); bounds->right = max(bounds->right, vport_rect.right); bounds->bottom = max(bounds->bottom, vport_rect.bottom); } } BOOL EMFDC_SaveDC( DC_ATTR *dc_attr ) { EMRSAVEDC emr; emr.emr.iType = EMR_SAVEDC; emr.emr.nSize = sizeof(emr); return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_RestoreDC( DC_ATTR *dc_attr, INT level ) { EMRRESTOREDC emr; if (abs(level) > dc_attr->save_level || level == 0) return FALSE; emr.emr.iType = EMR_RESTOREDC; emr.emr.nSize = sizeof(emr); if (level < 0) emr.iRelative = level; else emr.iRelative = level - dc_attr->save_level - 1; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetTextAlign( DC_ATTR *dc_attr, UINT align ) { EMRSETTEXTALIGN emr; emr.emr.iType = EMR_SETTEXTALIGN; emr.emr.nSize = sizeof(emr); emr.iMode = align; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetTextJustification( DC_ATTR *dc_attr, INT extra, INT breaks ) { EMRSETTEXTJUSTIFICATION emr; emr.emr.iType = EMR_SETTEXTJUSTIFICATION; emr.emr.nSize = sizeof(emr); emr.nBreakExtra = extra; emr.nBreakCount = breaks; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetBkMode( DC_ATTR *dc_attr, INT mode ) { EMRSETBKMODE emr; emr.emr.iType = EMR_SETBKMODE; emr.emr.nSize = sizeof(emr); emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetBkColor( DC_ATTR *dc_attr, COLORREF color ) { EMRSETBKCOLOR emr; emr.emr.iType = EMR_SETBKCOLOR; emr.emr.nSize = sizeof(emr); emr.crColor = color; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetTextColor( DC_ATTR *dc_attr, COLORREF color ) { EMRSETTEXTCOLOR emr; emr.emr.iType = EMR_SETTEXTCOLOR; emr.emr.nSize = sizeof(emr); emr.crColor = color; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetROP2( DC_ATTR *dc_attr, INT rop ) { EMRSETROP2 emr; emr.emr.iType = EMR_SETROP2; emr.emr.nSize = sizeof(emr); emr.iMode = rop; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetPolyFillMode( DC_ATTR *dc_attr, INT mode ) { EMRSETPOLYFILLMODE emr; emr.emr.iType = EMR_SETPOLYFILLMODE; emr.emr.nSize = sizeof(emr); emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetStretchBltMode( DC_ATTR *dc_attr, INT mode ) { EMRSETSTRETCHBLTMODE emr; emr.emr.iType = EMR_SETSTRETCHBLTMODE; emr.emr.nSize = sizeof(emr); emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetArcDirection( DC_ATTR *dc_attr, INT dir ) { EMRSETARCDIRECTION emr; emr.emr.iType = EMR_SETARCDIRECTION; emr.emr.nSize = sizeof(emr); emr.iArcDirection = dir; return emfdc_record( dc_attr->emf, &emr.emr ); } INT EMFDC_ExcludeClipRect( DC_ATTR *dc_attr, INT left, INT top, INT right, INT bottom ) { EMREXCLUDECLIPRECT emr; emr.emr.iType = EMR_EXCLUDECLIPRECT; emr.emr.nSize = sizeof(emr); emr.rclClip.left = left; emr.rclClip.top = top; emr.rclClip.right = right; emr.rclClip.bottom = bottom; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_IntersectClipRect( DC_ATTR *dc_attr, INT left, INT top, INT right, INT bottom) { EMRINTERSECTCLIPRECT emr; emr.emr.iType = EMR_INTERSECTCLIPRECT; emr.emr.nSize = sizeof(emr); emr.rclClip.left = left; emr.rclClip.top = top; emr.rclClip.right = right; emr.rclClip.bottom = bottom; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_OffsetClipRgn( DC_ATTR *dc_attr, INT x, INT y ) { EMROFFSETCLIPRGN emr; emr.emr.iType = EMR_OFFSETCLIPRGN; emr.emr.nSize = sizeof(emr); emr.ptlOffset.x = x; emr.ptlOffset.y = y; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_ExtSelectClipRgn( DC_ATTR *dc_attr, HRGN hrgn, INT mode ) { EMREXTSELECTCLIPRGN *emr; DWORD size, rgnsize; BOOL ret; if (!hrgn) { if (mode != RGN_COPY) return ERROR; rgnsize = 0; } else rgnsize = NtGdiGetRegionData( hrgn, 0, NULL ); size = rgnsize + offsetof(EMREXTSELECTCLIPRGN,RgnData); emr = HeapAlloc( GetProcessHeap(), 0, size ); if (rgnsize) NtGdiGetRegionData( hrgn, rgnsize, (RGNDATA *)&emr->RgnData ); emr->emr.iType = EMR_EXTSELECTCLIPRGN; emr->emr.nSize = size; emr->cbRgnData = rgnsize; emr->iMode = mode; ret = emfdc_record( dc_attr->emf, &emr->emr ); HeapFree( GetProcessHeap(), 0, emr ); return ret; } BOOL EMFDC_SetMapMode( DC_ATTR *dc_attr, INT mode ) { EMRSETMAPMODE emr; emr.emr.iType = EMR_SETMAPMODE; emr.emr.nSize = sizeof(emr); emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetViewportExtEx( DC_ATTR *dc_attr, INT cx, INT cy ) { EMRSETVIEWPORTEXTEX emr; emr.emr.iType = EMR_SETVIEWPORTEXTEX; emr.emr.nSize = sizeof(emr); emr.szlExtent.cx = cx; emr.szlExtent.cy = cy; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetWindowExtEx( DC_ATTR *dc_attr, INT cx, INT cy ) { EMRSETWINDOWEXTEX emr; emr.emr.iType = EMR_SETWINDOWEXTEX; emr.emr.nSize = sizeof(emr); emr.szlExtent.cx = cx; emr.szlExtent.cy = cy; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetViewportOrgEx( DC_ATTR *dc_attr, INT x, INT y ) { EMRSETVIEWPORTORGEX emr; emr.emr.iType = EMR_SETVIEWPORTORGEX; emr.emr.nSize = sizeof(emr); emr.ptlOrigin.x = x; emr.ptlOrigin.y = y; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetWindowOrgEx( DC_ATTR *dc_attr, INT x, INT y ) { EMRSETWINDOWORGEX emr; emr.emr.iType = EMR_SETWINDOWORGEX; emr.emr.nSize = sizeof(emr); emr.ptlOrigin.x = x; emr.ptlOrigin.y = y; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_ScaleViewportExtEx( DC_ATTR *dc_attr, INT x_num, INT x_denom, INT y_num, INT y_denom ) { EMRSCALEVIEWPORTEXTEX emr; emr.emr.iType = EMR_SCALEVIEWPORTEXTEX; emr.emr.nSize = sizeof(emr); emr.xNum = x_num; emr.xDenom = x_denom; emr.yNum = y_num; emr.yDenom = y_denom; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_ScaleWindowExtEx( DC_ATTR *dc_attr, INT x_num, INT x_denom, INT y_num, INT y_denom ) { EMRSCALEWINDOWEXTEX emr; emr.emr.iType = EMR_SCALEWINDOWEXTEX; emr.emr.nSize = sizeof(emr); emr.xNum = x_num; emr.xDenom = x_denom; emr.yNum = y_num; emr.yDenom = y_denom; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetLayout( DC_ATTR *dc_attr, DWORD layout ) { EMRSETLAYOUT emr; emr.emr.iType = EMR_SETLAYOUT; emr.emr.nSize = sizeof(emr); emr.iMode = layout; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetWorldTransform( DC_ATTR *dc_attr, const XFORM *xform ) { EMRSETWORLDTRANSFORM emr; emr.emr.iType = EMR_SETWORLDTRANSFORM; emr.emr.nSize = sizeof(emr); emr.xform = *xform; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_ModifyWorldTransform( DC_ATTR *dc_attr, const XFORM *xform, DWORD mode ) { EMRMODIFYWORLDTRANSFORM emr; emr.emr.iType = EMR_MODIFYWORLDTRANSFORM; emr.emr.nSize = sizeof(emr); if (mode == MWT_IDENTITY) { emr.xform.eM11 = 1.0f; emr.xform.eM12 = 0.0f; emr.xform.eM21 = 0.0f; emr.xform.eM22 = 1.0f; emr.xform.eDx = 0.0f; emr.xform.eDy = 0.0f; } else { emr.xform = *xform; } emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SetMapperFlags( DC_ATTR *dc_attr, DWORD flags ) { EMRSETMAPPERFLAGS emr; emr.emr.iType = EMR_SETMAPPERFLAGS; emr.emr.nSize = sizeof(emr); emr.dwFlags = flags; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_AbortPath( DC_ATTR *dc_attr ) { struct emf *emf = dc_attr->emf; EMRABORTPATH emr; emr.emr.iType = EMR_ABORTPATH; emr.emr.nSize = sizeof(emr); emf->path = FALSE; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_BeginPath( DC_ATTR *dc_attr ) { struct emf *emf = dc_attr->emf; EMRBEGINPATH emr; emr.emr.iType = EMR_BEGINPATH; emr.emr.nSize = sizeof(emr); if (!emfdc_record( emf, &emr.emr )) return FALSE; emf->path = TRUE; return TRUE; } BOOL EMFDC_CloseFigure( DC_ATTR *dc_attr ) { EMRCLOSEFIGURE emr; emr.emr.iType = EMR_CLOSEFIGURE; emr.emr.nSize = sizeof(emr); return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_EndPath( DC_ATTR *dc_attr ) { struct emf *emf = dc_attr->emf; EMRENDPATH emr; emf->path = FALSE; emr.emr.iType = EMR_ENDPATH; emr.emr.nSize = sizeof(emr); return emfdc_record( emf, &emr.emr ); } BOOL EMFDC_FlattenPath( DC_ATTR *dc_attr ) { EMRFLATTENPATH emr; emr.emr.iType = EMR_FLATTENPATH; emr.emr.nSize = sizeof(emr); return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_SelectClipPath( DC_ATTR *dc_attr, INT mode ) { EMRSELECTCLIPPATH emr; emr.emr.iType = EMR_SELECTCLIPPATH; emr.emr.nSize = sizeof(emr); emr.iMode = mode; return emfdc_record( dc_attr->emf, &emr.emr ); } BOOL EMFDC_WidenPath( DC_ATTR *dc_attr ) { EMRWIDENPATH emr; emr.emr.iType = EMR_WIDENPATH; emr.emr.nSize = sizeof(emr); return emfdc_record( dc_attr->emf, &emr.emr ); } void EMFDC_DeleteDC( DC_ATTR *dc_attr ) { struct emf *emf = dc_attr->emf; UINT index; HeapFree( GetProcessHeap(), 0, emf->emh ); for (index = 0; index < emf->handles_size; index++) if (emf->handles[index]) GDI_hdc_not_using_object( emf->handles[index], dc_attr->hdc ); HeapFree( GetProcessHeap(), 0, emf->handles ); } /********************************************************************** * CreateEnhMetaFileA (GDI32.@) */ HDC WINAPI CreateEnhMetaFileA( HDC hdc, const char *filename, const RECT *rect, const char *description ) { WCHAR *filenameW = NULL; WCHAR *descriptionW = NULL; DWORD len1, len2, total; HDC ret; if (filename) { total = MultiByteToWideChar( CP_ACP, 0, filename, -1, NULL, 0 ); filenameW = HeapAlloc( GetProcessHeap(), 0, total * sizeof(WCHAR) ); MultiByteToWideChar( CP_ACP, 0, filename, -1, filenameW, total ); } if(description) { len1 = strlen(description); len2 = strlen(description + len1 + 1); total = MultiByteToWideChar( CP_ACP, 0, description, len1 + len2 + 3, NULL, 0 ); descriptionW = HeapAlloc( GetProcessHeap(), 0, total * sizeof(WCHAR) ); MultiByteToWideChar( CP_ACP, 0, description, len1 + len2 + 3, descriptionW, total ); } ret = CreateEnhMetaFileW( hdc, filenameW, rect, descriptionW ); HeapFree( GetProcessHeap(), 0, filenameW ); HeapFree( GetProcessHeap(), 0, descriptionW ); return ret; } /********************************************************************** * CreateEnhMetaFileW (GDI32.@) */ HDC WINAPI CreateEnhMetaFileW( HDC hdc, const WCHAR *filename, const RECT *rect, const WCHAR *description ) { HDC ret; struct emf *emf; DC_ATTR *dc_attr; HANDLE file; DWORD size = 0, length = 0; TRACE( "(%p %s %s %s)\n", hdc, debugstr_w(filename), wine_dbgstr_rect(rect), debugstr_w(description) ); if (!(ret = NtGdiCreateMetafileDC( hdc ))) return 0; if (!(dc_attr = get_dc_attr( ret )) || !(emf = HeapAlloc( GetProcessHeap(), 0, sizeof(*emf) ))) { DeleteDC( ret ); return 0; } dc_attr->emf = emf; if (description) /* App name\0Title\0\0 */ { length = lstrlenW( description ); length += lstrlenW( description + length + 1 ); length += 3; length *= 2; } size = sizeof(ENHMETAHEADER) + (length + 3) / 4 * 4; if (!(emf->emh = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, size))) { DeleteDC( ret ); return 0; } emf->dc_attr = dc_attr; emf->handles = HeapAlloc( GetProcessHeap(), HEAP_ZERO_MEMORY, HANDLE_LIST_INC * sizeof(emf->handles[0]) ); emf->handles_size = HANDLE_LIST_INC; emf->cur_handles = 1; emf->hFile = 0; emf->dc_brush = 0; emf->dc_pen = 0; emf->path = FALSE; emf->emh->iType = EMR_HEADER; emf->emh->nSize = size; dc_attr->emf_bounds.left = dc_attr->emf_bounds.top = 0; dc_attr->emf_bounds.right = dc_attr->emf_bounds.bottom = -1; if (rect) { emf->emh->rclFrame.left = rect->left; emf->emh->rclFrame.top = rect->top; emf->emh->rclFrame.right = rect->right; emf->emh->rclFrame.bottom = rect->bottom; } else { /* Set this to {0,0 - -1,-1} and update it at the end */ emf->emh->rclFrame.left = emf->emh->rclFrame.top = 0; emf->emh->rclFrame.right = emf->emh->rclFrame.bottom = -1; } emf->emh->dSignature = ENHMETA_SIGNATURE; emf->emh->nVersion = 0x10000; emf->emh->nBytes = emf->emh->nSize; emf->emh->nRecords = 1; emf->emh->nHandles = 1; emf->emh->sReserved = 0; /* According to docs, this is reserved and must be 0 */ emf->emh->nDescription = length / 2; emf->emh->offDescription = length ? sizeof(ENHMETAHEADER) : 0; emf->emh->nPalEntries = 0; /* I guess this should start at 0 */ /* Size in pixels */ emf->emh->szlDevice.cx = GetDeviceCaps( ret, HORZRES ); emf->emh->szlDevice.cy = GetDeviceCaps( ret, VERTRES ); /* Size in millimeters */ emf->emh->szlMillimeters.cx = GetDeviceCaps( ret, HORZSIZE ); emf->emh->szlMillimeters.cy = GetDeviceCaps( ret, VERTSIZE ); /* Size in micrometers */ emf->emh->szlMicrometers.cx = emf->emh->szlMillimeters.cx * 1000; emf->emh->szlMicrometers.cy = emf->emh->szlMillimeters.cy * 1000; memcpy( (char *)emf->emh + sizeof(ENHMETAHEADER), description, length ); if (filename) /* disk based metafile */ { if ((file = CreateFileW( filename, GENERIC_WRITE | GENERIC_READ, 0, NULL, CREATE_ALWAYS, 0, 0)) == INVALID_HANDLE_VALUE) { DeleteDC( ret ); return 0; } emf->hFile = file; } TRACE( "returning %p\n", ret ); return ret; } /****************************************************************** * CloseEnhMetaFile (GDI32.@) */ HENHMETAFILE WINAPI CloseEnhMetaFile( HDC hdc ) { HENHMETAFILE hmf; struct emf *emf; DC_ATTR *dc_attr; EMREOF emr; HANDLE mapping = 0; TRACE("(%p)\n", hdc ); if (!(dc_attr = get_dc_attr( hdc )) || !dc_attr->emf) return 0; emf = dc_attr->emf; if (dc_attr->save_level) RestoreDC( hdc, 1 ); if (emf->dc_brush) DeleteObject( emf->dc_brush ); if (emf->dc_pen) DeleteObject( emf->dc_pen ); emr.emr.iType = EMR_EOF; emr.emr.nSize = sizeof(emr); emr.nPalEntries = 0; emr.offPalEntries = FIELD_OFFSET(EMREOF, nSizeLast); emr.nSizeLast = emr.emr.nSize; emfdc_record( emf, &emr.emr ); emf->emh->rclBounds = dc_attr->emf_bounds; /* Update rclFrame if not initialized in CreateEnhMetaFile */ if (emf->emh->rclFrame.left > emf->emh->rclFrame.right) { emf->emh->rclFrame.left = emf->emh->rclBounds.left * emf->emh->szlMillimeters.cx * 100 / emf->emh->szlDevice.cx; emf->emh->rclFrame.top = emf->emh->rclBounds.top * emf->emh->szlMillimeters.cy * 100 / emf->emh->szlDevice.cy; emf->emh->rclFrame.right = emf->emh->rclBounds.right * emf->emh->szlMillimeters.cx * 100 / emf->emh->szlDevice.cx; emf->emh->rclFrame.bottom = emf->emh->rclBounds.bottom * emf->emh->szlMillimeters.cy * 100 / emf->emh->szlDevice.cy; } if (emf->hFile) /* disk based metafile */ { if (!WriteFile( emf->hFile, emf->emh, emf->emh->nBytes, NULL, NULL )) { CloseHandle( emf->hFile ); return 0; } HeapFree( GetProcessHeap(), 0, emf->emh ); mapping = CreateFileMappingA( emf->hFile, NULL, PAGE_READONLY, 0, 0, NULL ); TRACE( "mapping = %p\n", mapping ); emf->emh = MapViewOfFile( mapping, FILE_MAP_READ, 0, 0, 0 ); TRACE( "view = %p\n", emf->emh ); CloseHandle( mapping ); CloseHandle( emf->hFile ); } hmf = EMF_Create_HENHMETAFILE( emf->emh, emf->emh->nBytes, emf->hFile != 0 ); emf->emh = NULL; /* So it won't be deleted */ DeleteDC( hdc ); return hmf; }