| /* |
| * Metafile functions |
| * |
| * Copyright David W. Metcalfe, 1994 |
| * Niels de Carpentier, Albrecht Kleine, Huw Davies 1996 |
| * |
| */ |
| |
| #include <string.h> |
| #include <fcntl.h> |
| #include "wine/winbase16.h" |
| #include "metafiledrv.h" |
| #include "metafile.h" |
| #include "bitmap.h" |
| #include "file.h" |
| #include "heap.h" |
| #include "toolhelp.h" |
| #include "debug.h" |
| |
| /****************************************************************** |
| * MF_AddHandle |
| * |
| * Add a handle to an external handle table and return the index |
| */ |
| |
| static int MF_AddHandle(HANDLETABLE16 *ht, WORD htlen, HGDIOBJ16 hobj) |
| { |
| int i; |
| |
| for (i = 0; i < htlen; i++) |
| { |
| if (*(ht->objectHandle + i) == 0) |
| { |
| *(ht->objectHandle + i) = hobj; |
| return i; |
| } |
| } |
| return -1; |
| } |
| |
| |
| /****************************************************************** |
| * MF_AddHandleDC |
| * |
| * Note: this function assumes that we never delete objects. |
| * If we do someday, we'll need to maintain a table to re-use deleted |
| * handles. |
| */ |
| static int MF_AddHandleDC( DC *dc ) |
| { |
| METAFILEDRV_PDEVICE *physDev = (METAFILEDRV_PDEVICE *)dc->physDev; |
| physDev->mh->mtNoObjects++; |
| return physDev->nextHandle++; |
| } |
| |
| |
| /****************************************************************** |
| * GetMetaFile16 (GDI.124) |
| */ |
| HMETAFILE16 WINAPI GetMetaFile16( LPCSTR lpFilename ) |
| { |
| return GetMetaFile32A( lpFilename ); |
| } |
| |
| |
| /****************************************************************** |
| * GetMetaFile32A (GDI32.197) |
| * |
| * Read a metafile from a file. Returns handle to a disk-based metafile. |
| */ |
| HMETAFILE32 WINAPI GetMetaFile32A( |
| LPCSTR lpFilename |
| /* pointer to string containing filename to read */ |
| ) |
| { |
| HMETAFILE16 hmf; |
| METAHEADER *mh; |
| HFILE32 hFile; |
| DWORD size; |
| |
| TRACE(metafile,"%s\n", lpFilename); |
| |
| if (!lpFilename) |
| return 0; |
| |
| hmf = GlobalAlloc16(GMEM_MOVEABLE, MFHEADERSIZE); |
| mh = (METAHEADER *)GlobalLock16(hmf); |
| |
| if (!mh) |
| { |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| if ((hFile = _lopen32(lpFilename, OF_READ)) == HFILE_ERROR32) |
| { |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| if (_lread32(hFile, (char *)mh, MFHEADERSIZE) == HFILE_ERROR32) |
| { |
| _lclose32( hFile ); |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| size = mh->mtSize * 2; /* alloc memory for whole metafile */ |
| GlobalUnlock16(hmf); |
| hmf = GlobalReAlloc16(hmf,size,GMEM_MOVEABLE); |
| mh = (METAHEADER *)GlobalLock16(hmf); |
| |
| if (!mh) |
| { |
| _lclose32( hFile ); |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| if (_lread32(hFile, (char*)mh + mh->mtHeaderSize * 2, |
| size - mh->mtHeaderSize * 2) == HFILE_ERROR32) |
| { |
| _lclose32( hFile ); |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| _lclose32(hFile); |
| |
| if (mh->mtType != 1) |
| { |
| GlobalFree16(hmf); |
| return 0; |
| } |
| |
| GlobalUnlock16(hmf); |
| return hmf; |
| |
| } |
| |
| |
| /****************************************************************** |
| * GetMetaFile32W (GDI32.199) |
| */ |
| HMETAFILE32 WINAPI GetMetaFile32W( LPCWSTR lpFilename ) |
| { |
| LPSTR p = HEAP_strdupWtoA( GetProcessHeap(), 0, lpFilename ); |
| HMETAFILE32 ret = GetMetaFile32A( p ); |
| HeapFree( GetProcessHeap(), 0, p ); |
| return ret; |
| } |
| |
| |
| /****************************************************************** |
| * CopyMetaFile16 (GDI.151) |
| */ |
| |
| HMETAFILE16 WINAPI CopyMetaFile16( HMETAFILE16 hSrcMetaFile, LPCSTR lpFilename ) |
| { |
| return CopyMetaFile32A( hSrcMetaFile, lpFilename ); |
| } |
| |
| |
| /****************************************************************** |
| * CopyMetaFile32A (GDI32.23) |
| * |
| * Copies the metafile corresponding to hSrcMetaFile to either |
| * a disk file, if a filename is given, or to a new memory based |
| * metafile, if lpFileName is NULL. |
| * |
| * RETURNS |
| * |
| * Handle to metafile copy on success, NULL on failure. |
| * |
| * BUGS |
| * |
| * Copying to disk returns NULL even if successful. |
| */ |
| HMETAFILE32 WINAPI CopyMetaFile32A( |
| HMETAFILE32 hSrcMetaFile, /* handle of metafile to copy */ |
| LPCSTR lpFilename /* filename if copying to a file */ |
| ) { |
| HMETAFILE16 handle = 0; |
| METAHEADER *mh; |
| METAHEADER *mh2; |
| HFILE32 hFile; |
| |
| TRACE(metafile,"(%08x,%s)\n", hSrcMetaFile, lpFilename); |
| |
| mh = (METAHEADER *)GlobalLock16(hSrcMetaFile); |
| |
| if (!mh) |
| return 0; |
| |
| if (lpFilename) /* disk based metafile */ |
| { |
| int i,j; |
| hFile = _lcreat32(lpFilename, 0); |
| j=mh->mtType; |
| mh->mtType=1; /* disk file version stores 1 here */ |
| i=_lwrite32(hFile, (char *)mh, mh->mtSize * 2) ; |
| mh->mtType=j; /* restore old value [0 or 1] */ |
| _lclose32(hFile); |
| if (i == -1) |
| return 0; |
| /* FIXME: return value */ |
| } |
| else /* memory based metafile */ |
| { |
| handle = GlobalAlloc16(GMEM_MOVEABLE,mh->mtSize * 2); |
| mh2 = (METAHEADER *)GlobalLock16(handle); |
| memcpy(mh2,mh, mh->mtSize * 2); |
| GlobalUnlock16(handle); |
| } |
| |
| GlobalUnlock16(hSrcMetaFile); |
| return handle; |
| } |
| |
| |
| /****************************************************************** |
| * CopyMetaFile32W (GDI32.24) |
| */ |
| HMETAFILE32 WINAPI CopyMetaFile32W( HMETAFILE32 hSrcMetaFile, |
| LPCWSTR lpFilename ) |
| { |
| LPSTR p = HEAP_strdupWtoA( GetProcessHeap(), 0, lpFilename ); |
| HMETAFILE32 ret = CopyMetaFile32A( hSrcMetaFile, p ); |
| HeapFree( GetProcessHeap(), 0, p ); |
| return ret; |
| } |
| |
| |
| /****************************************************************** |
| * IsValidMetaFile (GDI.410) |
| * |
| * Attempts to check if a given metafile is correctly formatted. |
| * Currently, the only things verified are several properties of the |
| * header. |
| * |
| * RETURNS |
| * TRUE if hmf passes some tests for being a valid metafile, FALSE otherwise. |
| * |
| * BUGS |
| * This is not exactly what windows does, see _Undocumented_Windows_ |
| * for details. |
| */ |
| |
| BOOL16 WINAPI IsValidMetaFile(HMETAFILE16 hmf) |
| { |
| BOOL16 resu=FALSE; |
| METAHEADER *mh = (METAHEADER *)GlobalLock16(hmf); |
| if (mh) { |
| if (mh->mtType == 1 || mh->mtType == 0) |
| if (mh->mtHeaderSize == MFHEADERSIZE/sizeof(INT16)) |
| if (mh->mtVersion == MFVERSION) |
| resu=TRUE; |
| GlobalUnlock16(hmf); |
| } |
| TRACE(metafile,"IsValidMetaFile %x => %d\n",hmf,resu); |
| return resu; |
| } |
| |
| |
| /****************************************************************** |
| * PlayMetaFile16 (GDI.123) |
| * |
| */ |
| BOOL16 WINAPI PlayMetaFile16( HDC16 hdc, HMETAFILE16 hmf ) |
| { |
| return PlayMetaFile32( hdc, hmf ); |
| } |
| |
| /****************************************************************** |
| * PlayMetaFile32 (GDI32.265) |
| * |
| * Renders the metafile specified by hmf in the DC specified by |
| * hdc. Returns FALSE on failure, TRUE on success. |
| */ |
| BOOL32 WINAPI PlayMetaFile32( |
| HDC32 hdc, /* handle of DC to render in */ |
| HMETAFILE32 hmf /* handle of metafile to render */ |
| ) |
| { |
| METAHEADER *mh = (METAHEADER *)GlobalLock16(hmf); |
| METARECORD *mr; |
| HANDLETABLE16 *ht; |
| HGLOBAL16 hHT; |
| int offset = 0; |
| WORD i; |
| HPEN32 hPen; |
| HBRUSH32 hBrush; |
| HFONT32 hFont; |
| DC *dc; |
| |
| TRACE(metafile,"(%04x %04x)\n",hdc,hmf); |
| if (!mh) return FALSE; |
| |
| /* save the current pen, brush and font */ |
| if (!(dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC ))) return 0; |
| hPen = dc->w.hPen; |
| hBrush = dc->w.hBrush; |
| hFont = dc->w.hFont; |
| GDI_HEAP_UNLOCK(hdc); |
| /* create the handle table */ |
| hHT = GlobalAlloc16(GMEM_MOVEABLE|GMEM_ZEROINIT, |
| sizeof(HANDLETABLE16) * mh->mtNoObjects); |
| ht = (HANDLETABLE16 *)GlobalLock16(hHT); |
| |
| |
| /* loop through metafile playing records */ |
| offset = mh->mtHeaderSize * 2; |
| while (offset < mh->mtSize * 2) |
| { |
| mr = (METARECORD *)((char *)mh + offset); |
| TRACE(metafile,"offset=%04x,size=%08lx\n", |
| offset, mr->rdSize); |
| if (!mr->rdSize) { |
| TRACE(metafile,"Entry got size 0 at offset %d, total mf length is %ld\n", |
| offset,mh->mtSize*2); |
| break; /* would loop endlessly otherwise */ |
| } |
| offset += mr->rdSize * 2; |
| PlayMetaFileRecord16( hdc, ht, mr, mh->mtNoObjects ); |
| } |
| |
| SelectObject32(hdc, hBrush); |
| SelectObject32(hdc, hPen); |
| SelectObject32(hdc, hFont); |
| |
| /* free objects in handle table */ |
| for(i = 0; i < mh->mtNoObjects; i++) |
| if(*(ht->objectHandle + i) != 0) |
| DeleteObject32(*(ht->objectHandle + i)); |
| |
| /* free handle table */ |
| GlobalFree16(hHT); |
| |
| return TRUE; |
| } |
| |
| |
| /****************************************************************** |
| * EnumMetaFile16 (GDI.175) |
| * |
| * Loop through the metafile records in hmf, calling the user-specified |
| * function for each one, stopping when the user's function returns FALSE |
| * (which is considered to be failure) |
| * or when no records are left (which is considered to be success). |
| * |
| * RETURNS |
| * TRUE on success, FALSE on failure. |
| * |
| * HISTORY |
| * Niels de carpentier, april 1996 |
| */ |
| BOOL16 WINAPI EnumMetaFile16( |
| HDC16 hdc, |
| HMETAFILE16 hmf, |
| MFENUMPROC16 lpEnumFunc, |
| LPARAM lpData |
| ) |
| { |
| METAHEADER *mh = (METAHEADER *)GlobalLock16(hmf); |
| METARECORD *mr; |
| HANDLETABLE16 *ht; |
| HGLOBAL16 hHT; |
| SEGPTR spht; |
| int offset = 0; |
| WORD i, seg; |
| HPEN32 hPen; |
| HBRUSH32 hBrush; |
| HFONT32 hFont; |
| DC *dc; |
| BOOL16 result = TRUE; |
| |
| TRACE(metafile,"(%04x, %04x, %08lx, %08lx)\n", |
| hdc, hmf, (DWORD)lpEnumFunc, lpData); |
| |
| if (!(dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC ))) return 0; |
| hPen = dc->w.hPen; |
| hBrush = dc->w.hBrush; |
| hFont = dc->w.hFont; |
| GDI_HEAP_UNLOCK(hdc); |
| |
| /* create the handle table */ |
| |
| hHT = GlobalAlloc16(GMEM_MOVEABLE | GMEM_ZEROINIT, |
| sizeof(HANDLETABLE16) * mh->mtNoObjects); |
| spht = WIN16_GlobalLock16(hHT); |
| |
| seg = GlobalHandleToSel(hmf); |
| offset = mh->mtHeaderSize * 2; |
| |
| /* loop through metafile records */ |
| |
| while (offset < (mh->mtSize * 2)) |
| { |
| mr = (METARECORD *)((char *)mh + offset); |
| if (!lpEnumFunc( hdc, (HANDLETABLE16 *)spht, |
| (METARECORD *) PTR_SEG_OFF_TO_HUGEPTR(seg, offset), |
| mh->mtNoObjects, (LONG)lpData )) |
| { |
| result = FALSE; |
| break; |
| } |
| |
| |
| offset += (mr->rdSize * 2); |
| } |
| |
| SelectObject32(hdc, hBrush); |
| SelectObject32(hdc, hPen); |
| SelectObject32(hdc, hFont); |
| |
| ht = (HANDLETABLE16 *)GlobalLock16(hHT); |
| |
| /* free objects in handle table */ |
| for(i = 0; i < mh->mtNoObjects; i++) |
| if(*(ht->objectHandle + i) != 0) |
| DeleteObject32(*(ht->objectHandle + i)); |
| |
| /* free handle table */ |
| GlobalFree16(hHT); |
| GlobalUnlock16(hmf); |
| return result; |
| } |
| |
| BOOL32 WINAPI EnumMetaFile32( |
| HDC32 hdc, |
| HMETAFILE32 hmf, |
| MFENUMPROC32 lpEnumFunc, |
| LPARAM lpData |
| ) { |
| METAHEADER *mh = (METAHEADER *)GlobalLock16(hmf); |
| METARECORD *mr; |
| HANDLETABLE32 *ht; |
| BOOL32 result = TRUE; |
| int i, offset = 0; |
| DC *dc = (DC *) GDI_GetObjPtr( hdc, DC_MAGIC ); |
| HPEN32 hPen; |
| HBRUSH32 hBrush; |
| HFONT32 hFont; |
| |
| TRACE(metafile,"(%08x,%08x,%p,%p)\n", |
| hdc, hmf, lpEnumFunc, (void*)lpData); |
| if (!mh) return 0; |
| |
| /* save the current pen, brush and font */ |
| if (!dc) return 0; |
| hPen = dc->w.hPen; |
| hBrush = dc->w.hBrush; |
| hFont = dc->w.hFont; |
| GDI_HEAP_UNLOCK(hdc); |
| |
| |
| ht = (HANDLETABLE32 *) GlobalAlloc32(GPTR, |
| sizeof(HANDLETABLE32) * mh->mtNoObjects); |
| |
| /* loop through metafile records */ |
| offset = mh->mtHeaderSize * 2; |
| |
| while (offset < (mh->mtSize * 2)) |
| { |
| mr = (METARECORD *)((char *)mh + offset); |
| if (!lpEnumFunc( hdc, ht, mr, mh->mtNoObjects, (LONG)lpData )) |
| { |
| result = FALSE; |
| break; |
| } |
| |
| offset += (mr->rdSize * 2); |
| } |
| |
| /* restore pen, brush and font */ |
| SelectObject32(hdc, hBrush); |
| SelectObject32(hdc, hPen); |
| SelectObject32(hdc, hFont); |
| |
| /* free objects in handle table */ |
| for(i = 0; i < mh->mtNoObjects; i++) |
| if(*(ht->objectHandle + i) != 0) |
| DeleteObject32(*(ht->objectHandle + i)); |
| |
| /* free handle table */ |
| GlobalFree32((HGLOBAL32)ht); |
| GlobalUnlock16(hmf); |
| return result; |
| } |
| |
| static BOOL32 MF_Meta_CreateRegion( METARECORD *mr, HRGN32 hrgn ); |
| |
| /****************************************************************** |
| * PlayMetaFileRecord16 (GDI.176) |
| * |
| * Render a single metafile record specified by *mr in the DC hdc, while |
| * using the handle table *ht, of length nHandles, |
| * to store metafile objects. |
| * |
| * BUGS |
| * The following metafile records are unimplemented: |
| * |
| * FRAMEREGION, DRAWTEXT, SETDIBTODEV, ANIMATEPALETTE, SETPALENTRIES, |
| * RESIZEPALETTE, EXTFLOODFILL, RESETDC, STARTDOC, STARTPAGE, ENDPAGE, |
| * ABORTDOC, ENDDOC, CREATEBRUSH, CREATEBITMAPINDIRECT, and CREATEBITMAP. |
| * |
| */ |
| void WINAPI PlayMetaFileRecord16( |
| HDC16 hdc, /* DC to render metafile into */ |
| HANDLETABLE16 *ht, /* pointer to handle table for metafile objects */ |
| METARECORD *mr, /* pointer to metafile record to render */ |
| UINT16 nHandles /* size of handle table */ |
| ) { |
| short s1; |
| HANDLE16 hndl; |
| char *ptr; |
| BITMAPINFOHEADER *infohdr; |
| |
| TRACE(metafile,"(%04x %08lx %08lx %04x) function %04x\n", |
| hdc,(LONG)ht, (LONG)mr, nHandles, mr->rdFunction); |
| |
| switch (mr->rdFunction) |
| { |
| case META_EOF: |
| break; |
| |
| case META_DELETEOBJECT: |
| DeleteObject32(*(ht->objectHandle + *(mr->rdParm))); |
| *(ht->objectHandle + *(mr->rdParm)) = 0; |
| break; |
| |
| case META_SETBKCOLOR: |
| SetBkColor16(hdc, MAKELONG(*(mr->rdParm), *(mr->rdParm + 1))); |
| break; |
| |
| case META_SETBKMODE: |
| SetBkMode16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETMAPMODE: |
| SetMapMode16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETROP2: |
| SetROP216(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETRELABS: |
| SetRelAbs16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETPOLYFILLMODE: |
| SetPolyFillMode16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETSTRETCHBLTMODE: |
| SetStretchBltMode16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SETTEXTCOLOR: |
| SetTextColor16(hdc, MAKELONG(*(mr->rdParm), *(mr->rdParm + 1))); |
| break; |
| |
| case META_SETWINDOWORG: |
| SetWindowOrg(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_SETWINDOWEXT: |
| SetWindowExt(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_SETVIEWPORTORG: |
| SetViewportOrg(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_SETVIEWPORTEXT: |
| SetViewportExt(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_OFFSETWINDOWORG: |
| OffsetWindowOrg(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_SCALEWINDOWEXT: |
| ScaleWindowExt(hdc, *(mr->rdParm + 3), *(mr->rdParm + 2), |
| *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_OFFSETVIEWPORTORG: |
| OffsetViewportOrg(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_SCALEVIEWPORTEXT: |
| ScaleViewportExt(hdc, *(mr->rdParm + 3), *(mr->rdParm + 2), |
| *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_LINETO: |
| LineTo32(hdc, (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_MOVETO: |
| MoveTo(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_EXCLUDECLIPRECT: |
| ExcludeClipRect16( hdc, *(mr->rdParm + 3), *(mr->rdParm + 2), |
| *(mr->rdParm + 1), *(mr->rdParm) ); |
| break; |
| |
| case META_INTERSECTCLIPRECT: |
| IntersectClipRect16( hdc, *(mr->rdParm + 3), *(mr->rdParm + 2), |
| *(mr->rdParm + 1), *(mr->rdParm) ); |
| break; |
| |
| case META_ARC: |
| Arc32(hdc, (INT16)*(mr->rdParm + 7), (INT16)*(mr->rdParm + 6), |
| (INT16)*(mr->rdParm + 5), (INT16)*(mr->rdParm + 4), |
| (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_ELLIPSE: |
| Ellipse32(hdc, (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_FLOODFILL: |
| FloodFill32(hdc, (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| MAKELONG(*(mr->rdParm), *(mr->rdParm + 1))); |
| break; |
| |
| case META_PIE: |
| Pie32(hdc, (INT16)*(mr->rdParm + 7), (INT16)*(mr->rdParm + 6), |
| (INT16)*(mr->rdParm + 5), (INT16)*(mr->rdParm + 4), |
| (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_RECTANGLE: |
| Rectangle32(hdc, (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_ROUNDRECT: |
| RoundRect32(hdc, (INT16)*(mr->rdParm + 5), (INT16)*(mr->rdParm + 4), |
| (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_PATBLT: |
| PatBlt16(hdc, *(mr->rdParm + 5), *(mr->rdParm + 4), |
| *(mr->rdParm + 3), *(mr->rdParm + 2), |
| MAKELONG(*(mr->rdParm), *(mr->rdParm + 1))); |
| break; |
| |
| case META_SAVEDC: |
| SaveDC32(hdc); |
| break; |
| |
| case META_SETPIXEL: |
| SetPixel32(hdc, (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| MAKELONG(*(mr->rdParm), *(mr->rdParm + 1))); |
| break; |
| |
| case META_OFFSETCLIPRGN: |
| OffsetClipRgn16( hdc, *(mr->rdParm + 1), *(mr->rdParm) ); |
| break; |
| |
| case META_TEXTOUT: |
| s1 = *(mr->rdParm); |
| TextOut16(hdc, *(mr->rdParm + ((s1 + 1) >> 1) + 2), |
| *(mr->rdParm + ((s1 + 1) >> 1) + 1), |
| (char *)(mr->rdParm + 1), s1); |
| break; |
| |
| case META_POLYGON: |
| Polygon16(hdc, (LPPOINT16)(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_POLYPOLYGON: |
| PolyPolygon16(hdc, (LPPOINT16)(mr->rdParm + *(mr->rdParm) + 1), |
| (LPINT16)(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_POLYLINE: |
| Polyline16(hdc, (LPPOINT16)(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| case META_RESTOREDC: |
| RestoreDC32(hdc, (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_SELECTOBJECT: |
| SelectObject32(hdc, *(ht->objectHandle + *(mr->rdParm))); |
| break; |
| |
| case META_CHORD: |
| Chord32(hdc, (INT16)*(mr->rdParm + 7), (INT16)*(mr->rdParm + 6), |
| (INT16)*(mr->rdParm+5), (INT16)*(mr->rdParm + 4), |
| (INT16)*(mr->rdParm + 3), (INT16)*(mr->rdParm + 2), |
| (INT16)*(mr->rdParm + 1), (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_CREATEPATTERNBRUSH: |
| switch (*(mr->rdParm)) |
| { |
| case BS_PATTERN: |
| infohdr = (BITMAPINFOHEADER *)(mr->rdParm + 2); |
| MF_AddHandle(ht, nHandles, |
| CreatePatternBrush32(CreateBitmap32(infohdr->biWidth, |
| infohdr->biHeight, |
| infohdr->biPlanes, |
| infohdr->biBitCount, |
| (LPSTR)(mr->rdParm + |
| (sizeof(BITMAPINFOHEADER) / 2) + 4)))); |
| break; |
| |
| case BS_DIBPATTERN: |
| s1 = mr->rdSize * 2 - sizeof(METARECORD) - 2; |
| hndl = GlobalAlloc16(GMEM_MOVEABLE, s1); |
| ptr = GlobalLock16(hndl); |
| memcpy(ptr, mr->rdParm + 2, s1); |
| GlobalUnlock16(hndl); |
| MF_AddHandle(ht, nHandles, |
| CreateDIBPatternBrush32(hndl, *(mr->rdParm + 1))); |
| GlobalFree16(hndl); |
| } |
| break; |
| |
| case META_CREATEPENINDIRECT: |
| MF_AddHandle(ht, nHandles, |
| CreatePenIndirect16((LOGPEN16 *)(&(mr->rdParm)))); |
| break; |
| |
| case META_CREATEFONTINDIRECT: |
| MF_AddHandle(ht, nHandles, |
| CreateFontIndirect16((LOGFONT16 *)(&(mr->rdParm)))); |
| break; |
| |
| case META_CREATEBRUSHINDIRECT: |
| MF_AddHandle(ht, nHandles, |
| CreateBrushIndirect16((LOGBRUSH16 *)(&(mr->rdParm)))); |
| break; |
| |
| /* W. Magro: Some new metafile operations. Not all debugged. */ |
| case META_CREATEPALETTE: |
| MF_AddHandle(ht, nHandles, |
| CreatePalette16((LPLOGPALETTE)mr->rdParm)); |
| break; |
| |
| case META_SETTEXTALIGN: |
| SetTextAlign16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_SELECTPALETTE: |
| SelectPalette16(hdc, *(ht->objectHandle + *(mr->rdParm+1)),*(mr->rdParm)); |
| break; |
| |
| case META_SETMAPPERFLAGS: |
| SetMapperFlags16(hdc, *(mr->rdParm)); |
| break; |
| |
| case META_REALIZEPALETTE: |
| RealizePalette16(hdc); |
| break; |
| |
| case META_ESCAPE: |
| FIXME(metafile, "META_ESCAPE unimplemented.\n"); |
| break; |
| |
| /* --- Begin of fixed or new metafile operations. July 1996 ----*/ |
| case META_EXTTEXTOUT: |
| { |
| LPINT16 dxx; |
| LPSTR sot; |
| DWORD len; |
| |
| s1 = mr->rdParm[2]; /* String length */ |
| len = sizeof(METARECORD) + (((s1 + 1) >> 1) * 2) + 2 * sizeof(short) |
| + sizeof(UINT16) + (mr->rdParm[3] ? sizeof(RECT16) : 0); /* rec len without dx array */ |
| |
| sot= (LPSTR)&mr->rdParm[4]; /* start_of_text */ |
| if (mr->rdParm[3]) |
| sot+=sizeof(RECT16); /* there is a rectangle, so add offset */ |
| |
| if (mr->rdSize == len / 2) |
| dxx = NULL; /* determine if array present */ |
| else |
| if (mr->rdSize == (len + s1 * sizeof(INT16)) / 2) |
| dxx = (LPINT16)(sot+(((s1+1)>>1)*2)); |
| else |
| { |
| TRACE(metafile,"%s len: %ld\n", |
| sot,mr->rdSize); |
| WARN(metafile, |
| "Please report: PlayMetaFile/ExtTextOut len=%ld slen=%d rdSize=%ld opt=%04x\n", |
| len,s1,mr->rdSize,mr->rdParm[3]); |
| dxx = NULL; /* should't happen -- but if, we continue with NULL [for workaround] */ |
| } |
| ExtTextOut16( hdc, mr->rdParm[1], /* X position */ |
| mr->rdParm[0], /* Y position */ |
| mr->rdParm[3], /* options */ |
| mr->rdParm[3] ? (LPRECT16) &mr->rdParm[4]:NULL, /* rectangle */ |
| sot, /* string */ |
| s1, dxx); /* length, dx array */ |
| if (dxx) |
| TRACE(metafile,"%s len: %ld dx0: %d\n", |
| sot,mr->rdSize,dxx[0]); |
| } |
| break; |
| |
| case META_STRETCHDIB: |
| { |
| LPBITMAPINFO info = (LPBITMAPINFO) &(mr->rdParm[11]); |
| LPSTR bits = (LPSTR)info + DIB_BitmapInfoSize( info, mr->rdParm[2] ); |
| StretchDIBits16(hdc,mr->rdParm[10],mr->rdParm[9],mr->rdParm[8], |
| mr->rdParm[7],mr->rdParm[6],mr->rdParm[5], |
| mr->rdParm[4],mr->rdParm[3],bits,info, |
| mr->rdParm[2],MAKELONG(mr->rdParm[0],mr->rdParm[1])); |
| } |
| break; |
| |
| case META_DIBSTRETCHBLT: |
| { |
| LPBITMAPINFO info = (LPBITMAPINFO) &(mr->rdParm[10]); |
| LPSTR bits = (LPSTR)info + DIB_BitmapInfoSize( info, mr->rdParm[2] ); |
| StretchDIBits16(hdc,mr->rdParm[9],mr->rdParm[8],mr->rdParm[7], |
| mr->rdParm[6],mr->rdParm[5],mr->rdParm[4], |
| mr->rdParm[3],mr->rdParm[2],bits,info, |
| DIB_RGB_COLORS,MAKELONG(mr->rdParm[0],mr->rdParm[1])); |
| } |
| break; |
| |
| case META_STRETCHBLT: |
| { |
| HDC16 hdcSrc=CreateCompatibleDC16(hdc); |
| HBITMAP32 hbitmap=CreateBitmap32(mr->rdParm[10], /*Width */ |
| mr->rdParm[11], /*Height*/ |
| mr->rdParm[13], /*Planes*/ |
| mr->rdParm[14], /*BitsPixel*/ |
| (LPSTR)&mr->rdParm[15]); /*bits*/ |
| SelectObject32(hdcSrc,hbitmap); |
| StretchBlt16(hdc,mr->rdParm[9],mr->rdParm[8], |
| mr->rdParm[7],mr->rdParm[6], |
| hdcSrc,mr->rdParm[5],mr->rdParm[4], |
| mr->rdParm[3],mr->rdParm[2], |
| MAKELONG(mr->rdParm[0],mr->rdParm[1])); |
| DeleteDC32(hdcSrc); |
| } |
| break; |
| |
| case META_BITBLT: /* <-- not yet debugged */ |
| { |
| HDC16 hdcSrc=CreateCompatibleDC16(hdc); |
| HBITMAP32 hbitmap=CreateBitmap32(mr->rdParm[7]/*Width */, |
| mr->rdParm[8]/*Height*/, |
| mr->rdParm[10]/*Planes*/, |
| mr->rdParm[11]/*BitsPixel*/, |
| (LPSTR)&mr->rdParm[12]/*bits*/); |
| SelectObject32(hdcSrc,hbitmap); |
| BitBlt32(hdc,(INT16)mr->rdParm[6],(INT16)mr->rdParm[5], |
| (INT16)mr->rdParm[4],(INT16)mr->rdParm[3], |
| hdcSrc, (INT16)mr->rdParm[2],(INT16)mr->rdParm[1], |
| MAKELONG(0,mr->rdParm[0])); |
| DeleteDC32(hdcSrc); |
| } |
| break; |
| |
| /* --- Begin of new metafile operations. April, 1997 (ak) ----*/ |
| case META_CREATEREGION: |
| { |
| HRGN32 hrgn = CreateRectRgn32(0,0,0,0); |
| |
| MF_Meta_CreateRegion(mr, hrgn); |
| MF_AddHandle(ht, nHandles, hrgn); |
| } |
| break; |
| |
| case META_FILLREGION: |
| FillRgn16(hdc, *(ht->objectHandle + *(mr->rdParm)), |
| *(ht->objectHandle + *(mr->rdParm+1))); |
| break; |
| |
| case META_INVERTREGION: |
| InvertRgn16(hdc, *(ht->objectHandle + *(mr->rdParm))); |
| break; |
| |
| case META_PAINTREGION: |
| PaintRgn16(hdc, *(ht->objectHandle + *(mr->rdParm))); |
| break; |
| |
| case META_SELECTCLIPREGION: |
| SelectClipRgn32(hdc, *(ht->objectHandle + *(mr->rdParm))); |
| break; |
| |
| case META_DIBCREATEPATTERNBRUSH: |
| /* *(mr->rdParm) may be BS_PATTERN or BS_DIBPATTERN: but there's no difference */ |
| TRACE(metafile,"%d\n",*(mr->rdParm)); |
| s1 = mr->rdSize * 2 - sizeof(METARECORD) - 2; |
| hndl = GlobalAlloc16(GMEM_MOVEABLE, s1); |
| ptr = GlobalLock16(hndl); |
| memcpy(ptr, mr->rdParm + 2, s1); |
| GlobalUnlock16(hndl); |
| MF_AddHandle(ht, nHandles,CreateDIBPatternBrush16(hndl, *(mr->rdParm + 1))); |
| GlobalFree16(hndl); |
| break; |
| |
| case META_DIBBITBLT: |
| { |
| LPBITMAPINFO info = (LPBITMAPINFO) &(mr->rdParm[8]); |
| LPSTR bits = (LPSTR)info + DIB_BitmapInfoSize( info, mr->rdParm[0] ); |
| StretchDIBits16(hdc,mr->rdParm[7],mr->rdParm[6],mr->rdParm[5], |
| mr->rdParm[4],mr->rdParm[3],mr->rdParm[2], |
| mr->rdParm[5],mr->rdParm[4],bits,info, |
| DIB_RGB_COLORS,MAKELONG(mr->rdParm[0],mr->rdParm[1])); |
| } |
| break; |
| |
| case META_SETTEXTCHAREXTRA: |
| SetTextCharacterExtra16(hdc, (INT16)*(mr->rdParm)); |
| break; |
| |
| case META_SETTEXTJUSTIFICATION: |
| SetTextJustification32(hdc, *(mr->rdParm + 1), *(mr->rdParm)); |
| break; |
| |
| #define META_UNIMP(x) case x: FIXME(metafile, "PlayMetaFileRecord:record type "#x" not implemented.\n");break; |
| META_UNIMP(META_FRAMEREGION) |
| META_UNIMP(META_DRAWTEXT) |
| META_UNIMP(META_SETDIBTODEV) |
| META_UNIMP(META_ANIMATEPALETTE) |
| META_UNIMP(META_SETPALENTRIES) |
| META_UNIMP(META_RESIZEPALETTE) |
| META_UNIMP(META_EXTFLOODFILL) |
| META_UNIMP(META_RESETDC) |
| META_UNIMP(META_STARTDOC) |
| META_UNIMP(META_STARTPAGE) |
| META_UNIMP(META_ENDPAGE) |
| META_UNIMP(META_ABORTDOC) |
| META_UNIMP(META_ENDDOC) |
| META_UNIMP(META_CREATEBRUSH) |
| META_UNIMP(META_CREATEBITMAPINDIRECT) |
| META_UNIMP(META_CREATEBITMAP) |
| #undef META_UNIMP |
| |
| default: |
| WARN(metafile, "PlayMetaFileRecord: Unknown record type %x\n", |
| mr->rdFunction); |
| } |
| } |
| |
| |
| BOOL32 WINAPI PlayMetaFileRecord32( |
| HDC32 hdc, |
| HANDLETABLE32 *handletable, |
| METARECORD *metarecord, |
| UINT32 handles |
| ) |
| { |
| HANDLETABLE16 * ht = (void *)GlobalAlloc32(GPTR, |
| handles*sizeof(HANDLETABLE16)); |
| int i = 0; |
| TRACE(metafile, "(%08x,%p,%p,%d)\n", hdc, handletable, metarecord, handles); |
| for (i=0; i<handles; i++) |
| ht->objectHandle[i] = handletable->objectHandle[i]; |
| PlayMetaFileRecord16(hdc, ht, metarecord, handles); |
| for (i=0; i<handles; i++) |
| handletable->objectHandle[i] = ht->objectHandle[i]; |
| GlobalFree32((HGLOBAL32)ht); |
| return TRUE; |
| } |
| |
| /****************************************************************** |
| * GetMetaFileBits (GDI.159) |
| * |
| * Trade in a metafile object handle for a handle to the metafile memory. |
| * |
| */ |
| |
| HGLOBAL16 WINAPI GetMetaFileBits( |
| HMETAFILE16 hmf /* metafile handle */ |
| ) |
| { |
| TRACE(metafile,"hMem out: %04x\n", hmf); |
| return hmf; |
| } |
| |
| /****************************************************************** |
| * SetMetaFileBits (GDI.160) |
| * |
| * Trade in a metafile memory handle for a handle to a metafile object. |
| * The memory region should hold a proper metafile, otherwise |
| * problems will occur when it is used. Validity of the memory is not |
| * checked. The function is essentially just the identity function. |
| */ |
| HMETAFILE16 WINAPI SetMetaFileBits( |
| HGLOBAL16 hMem |
| /* handle to a memory region holding a metafile */ |
| ) |
| { |
| TRACE(metafile,"hmf out: %04x\n", hMem); |
| |
| return hMem; |
| } |
| |
| /****************************************************************** |
| * SetMetaFileBitsBetter (GDI.196) |
| * |
| * Trade in a metafile memory handle for a handle to a metafile object, |
| * making a cursory check (using IsValidMetaFile()) that the memory |
| * handle points to a valid metafile. |
| * |
| * RETURNS |
| * Handle to a metafile on success, NULL on failure.. |
| */ |
| HMETAFILE16 WINAPI SetMetaFileBitsBetter( HMETAFILE16 hMeta ) |
| { |
| if( IsValidMetaFile( hMeta ) ) |
| return (HMETAFILE16)GlobalReAlloc16( hMeta, 0, |
| GMEM_SHARE | GMEM_NODISCARD | GMEM_MODIFY); |
| return (HMETAFILE16)0; |
| } |
| |
| /****************************************************************** |
| * SetMetaFileBitsEx (GDI32.323) |
| * |
| * Create a metafile from raw data. No checking of the data is performed. |
| * Use _GetMetaFileBitsEx_ to get raw data from a metafile. |
| */ |
| HMETAFILE32 WINAPI SetMetaFileBitsEx( |
| UINT32 size, /* size of metafile, in bytes */ |
| const BYTE *lpData /* pointer to metafile data */ |
| ) |
| { |
| HMETAFILE32 hmf = GlobalAlloc16(GHND, size); |
| BYTE *p = GlobalLock16(hmf) ; |
| TRACE(metafile, "(%d,%p) returning %08x\n", size, lpData, hmf); |
| if (!hmf || !p) return 0; |
| memcpy(p, lpData, size); |
| GlobalUnlock16(hmf); |
| return hmf; |
| } |
| |
| /***************************************************************** |
| * GetMetaFileBitsEx (GDI32.198) Get raw metafile data |
| * |
| * Copies the data from metafile _hmf_ into the buffer _buf_. |
| * If _buf_ is zero, returns size of buffer required. Otherwise, |
| * returns number of bytes copied. |
| */ |
| UINT32 WINAPI GetMetaFileBitsEx( |
| HMETAFILE32 hmf, /* metafile */ |
| UINT32 nSize, /* size of buf */ |
| LPVOID buf /* buffer to receive raw metafile data */ |
| ) { |
| METAHEADER *h = GlobalLock16(hmf); |
| UINT32 mfSize; |
| |
| TRACE(metafile, "(%08x,%d,%p)\n", hmf, nSize, buf); |
| if (!h) return 0; /* FIXME: error code */ |
| mfSize = h->mtSize * 2; |
| if (!buf) { |
| GlobalUnlock16(hmf); |
| TRACE(metafile,"returning size %d\n", mfSize); |
| return mfSize; |
| } |
| if(mfSize > nSize) mfSize = nSize; |
| memmove(buf, h, mfSize); |
| GlobalUnlock16(hmf); |
| return mfSize; |
| } |
| |
| /****************************************************************** |
| * GetWinMetaFileBits [GDI32.241] |
| */ |
| UINT32 WINAPI GetWinMetaFileBits(HENHMETAFILE32 hemf, |
| UINT32 cbBuffer, LPBYTE lpbBuffer, |
| INT32 fnMapMode, HDC32 hdcRef) |
| { |
| FIXME(metafile, "(%d,%d,%p,%d,%d): stub\n", |
| hemf, cbBuffer, lpbBuffer, fnMapMode, hdcRef); |
| return 0; |
| } |
| |
| /****************************************************************** |
| * MF_Meta_CreateRegion |
| * |
| * Handles META_CREATEREGION for PlayMetaFileRecord(). |
| */ |
| |
| /* |
| * The layout of the record looks something like this: |
| * |
| * rdParm meaning |
| * 0 Always 0? |
| * 1 Always 6? |
| * 2 Looks like a handle? - not constant |
| * 3 0 or 1 ?? |
| * 4 Total number of bytes |
| * 5 No. of seperate bands = n [see below] |
| * 6 Largest number of x co-ords in a band |
| * 7-10 Bounding box x1 y1 x2 y2 |
| * 11-... n bands |
| * |
| * Regions are divided into bands that are uniform in the |
| * y-direction. Each band consists of pairs of on/off x-coords and is |
| * written as |
| * m y0 y1 x1 x2 x3 ... xm m |
| * into successive rdParm[]s. |
| * |
| * This is probably just a dump of the internal RGNOBJ? |
| * |
| * HDMD - 18/12/97 |
| * |
| */ |
| |
| static BOOL32 MF_Meta_CreateRegion( METARECORD *mr, HRGN32 hrgn ) |
| { |
| WORD band, pair; |
| WORD *start, *end; |
| INT16 y0, y1; |
| HRGN32 hrgn2 = CreateRectRgn32( 0, 0, 0, 0 ); |
| |
| for(band = 0, start = &(mr->rdParm[11]); band < mr->rdParm[5]; |
| band++, start = end + 1) { |
| if(*start / 2 != (*start + 1) / 2) { |
| WARN(metafile, "Delimiter not even.\n"); |
| DeleteObject32( hrgn2 ); |
| return FALSE; |
| } |
| |
| end = start + *start + 3; |
| if(end > (WORD *)mr + mr->rdSize) { |
| WARN(metafile, "End points outside record.\n"); |
| DeleteObject32( hrgn2 ); |
| return FALSE; |
| } |
| |
| if(*start != *end) { |
| WARN(metafile, "Mismatched delimiters.\n"); |
| DeleteObject32( hrgn2 ); |
| return FALSE; |
| } |
| |
| y0 = *(INT16 *)(start + 1); |
| y1 = *(INT16 *)(start + 2); |
| for(pair = 0; pair < *start / 2; pair++) { |
| SetRectRgn32( hrgn2, *(INT16 *)(start + 3 + 2*pair), y0, |
| *(INT16 *)(start + 4 + 2*pair), y1 ); |
| CombineRgn32(hrgn, hrgn, hrgn2, RGN_OR); |
| } |
| } |
| DeleteObject32( hrgn2 ); |
| return TRUE; |
| } |
| |
| |
| /****************************************************************** |
| * MF_WriteRecord |
| * |
| * Warning: this function can change the metafile handle. |
| */ |
| |
| static BOOL32 MF_WriteRecord( DC *dc, METARECORD *mr, DWORD rlen) |
| { |
| DWORD len; |
| METAHEADER *mh; |
| METAFILEDRV_PDEVICE *physDev = (METAFILEDRV_PDEVICE *)dc->physDev; |
| |
| switch(physDev->mh->mtType) |
| { |
| case METAFILE_MEMORY: |
| len = physDev->mh->mtSize * 2 + rlen; |
| mh = HeapReAlloc( SystemHeap, 0, physDev->mh, len ); |
| if (!mh) return FALSE; |
| physDev->mh = mh; |
| memcpy((WORD *)physDev->mh + physDev->mh->mtSize, mr, rlen); |
| break; |
| case METAFILE_DISK: |
| TRACE(metafile,"Writing record to disk\n"); |
| if (_lwrite32(physDev->mh->mtNoParameters, (char *)mr, rlen) == -1) |
| return FALSE; |
| break; |
| default: |
| ERR(metafile, "Unknown metafile type %d\n", physDev->mh->mtType ); |
| return FALSE; |
| } |
| |
| physDev->mh->mtSize += rlen / 2; |
| physDev->mh->mtMaxRecord = MAX(physDev->mh->mtMaxRecord, rlen / 2); |
| return TRUE; |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam0 |
| */ |
| |
| BOOL32 MF_MetaParam0(DC *dc, short func) |
| { |
| char buffer[8]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 3; |
| mr->rdFunction = func; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam1 |
| */ |
| BOOL32 MF_MetaParam1(DC *dc, short func, short param1) |
| { |
| char buffer[8]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 4; |
| mr->rdFunction = func; |
| *(mr->rdParm) = param1; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam2 |
| */ |
| BOOL32 MF_MetaParam2(DC *dc, short func, short param1, short param2) |
| { |
| char buffer[10]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 5; |
| mr->rdFunction = func; |
| *(mr->rdParm) = param2; |
| *(mr->rdParm + 1) = param1; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam4 |
| */ |
| |
| BOOL32 MF_MetaParam4(DC *dc, short func, short param1, short param2, |
| short param3, short param4) |
| { |
| char buffer[14]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 7; |
| mr->rdFunction = func; |
| *(mr->rdParm) = param4; |
| *(mr->rdParm + 1) = param3; |
| *(mr->rdParm + 2) = param2; |
| *(mr->rdParm + 3) = param1; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam6 |
| */ |
| |
| BOOL32 MF_MetaParam6(DC *dc, short func, short param1, short param2, |
| short param3, short param4, short param5, short param6) |
| { |
| char buffer[18]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 9; |
| mr->rdFunction = func; |
| *(mr->rdParm) = param6; |
| *(mr->rdParm + 1) = param5; |
| *(mr->rdParm + 2) = param4; |
| *(mr->rdParm + 3) = param3; |
| *(mr->rdParm + 4) = param2; |
| *(mr->rdParm + 5) = param1; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_MetaParam8 |
| */ |
| BOOL32 MF_MetaParam8(DC *dc, short func, short param1, short param2, |
| short param3, short param4, short param5, |
| short param6, short param7, short param8) |
| { |
| char buffer[22]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = 11; |
| mr->rdFunction = func; |
| *(mr->rdParm) = param8; |
| *(mr->rdParm + 1) = param7; |
| *(mr->rdParm + 2) = param6; |
| *(mr->rdParm + 3) = param5; |
| *(mr->rdParm + 4) = param4; |
| *(mr->rdParm + 5) = param3; |
| *(mr->rdParm + 6) = param2; |
| *(mr->rdParm + 7) = param1; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_CreateBrushIndirect |
| */ |
| |
| BOOL32 MF_CreateBrushIndirect(DC *dc, HBRUSH16 hBrush, LOGBRUSH16 *logbrush) |
| { |
| int index; |
| char buffer[sizeof(METARECORD) - 2 + sizeof(*logbrush)]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = (sizeof(METARECORD) + sizeof(*logbrush) - 2) / 2; |
| mr->rdFunction = META_CREATEBRUSHINDIRECT; |
| memcpy(&(mr->rdParm), logbrush, sizeof(*logbrush)); |
| if (!(MF_WriteRecord( dc, mr, mr->rdSize * 2))) return FALSE; |
| |
| mr->rdSize = sizeof(METARECORD) / 2; |
| mr->rdFunction = META_SELECTOBJECT; |
| |
| if ((index = MF_AddHandleDC( dc )) == -1) return FALSE; |
| *(mr->rdParm) = index; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_CreatePatternBrush |
| */ |
| |
| BOOL32 MF_CreatePatternBrush(DC *dc, HBRUSH16 hBrush, LOGBRUSH16 *logbrush) |
| { |
| DWORD len, bmSize, biSize; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| BITMAPOBJ *bmp; |
| BITMAPINFO *info; |
| BITMAPINFOHEADER *infohdr; |
| int index; |
| char buffer[sizeof(METARECORD)]; |
| |
| switch (logbrush->lbStyle) |
| { |
| case BS_PATTERN: |
| bmp = (BITMAPOBJ *)GDI_GetObjPtr((HGDIOBJ16)logbrush->lbHatch, BITMAP_MAGIC); |
| if (!bmp) return FALSE; |
| len = sizeof(METARECORD) + sizeof(BITMAPINFOHEADER) + |
| (bmp->bitmap.bmHeight * bmp->bitmap.bmWidthBytes) + 6; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| { |
| GDI_HEAP_UNLOCK((HGDIOBJ16)logbrush->lbHatch); |
| return FALSE; |
| } |
| mr = (METARECORD *)GlobalLock16(hmr); |
| memset(mr, 0, len); |
| mr->rdFunction = META_DIBCREATEPATTERNBRUSH; |
| mr->rdSize = len / 2; |
| *(mr->rdParm) = logbrush->lbStyle; |
| *(mr->rdParm + 1) = DIB_RGB_COLORS; |
| infohdr = (BITMAPINFOHEADER *)(mr->rdParm + 2); |
| infohdr->biSize = sizeof(BITMAPINFOHEADER); |
| infohdr->biWidth = bmp->bitmap.bmWidth; |
| infohdr->biHeight = bmp->bitmap.bmHeight; |
| infohdr->biPlanes = bmp->bitmap.bmPlanes; |
| infohdr->biBitCount = bmp->bitmap.bmBitsPixel; |
| memcpy(mr->rdParm + (sizeof(BITMAPINFOHEADER) / 2) + 4, |
| PTR_SEG_TO_LIN(bmp->bitmap.bmBits), |
| bmp->bitmap.bmHeight * bmp->bitmap.bmWidthBytes); |
| GDI_HEAP_UNLOCK(logbrush->lbHatch); |
| break; |
| |
| case BS_DIBPATTERN: |
| info = (BITMAPINFO *)GlobalLock16((HGLOBAL16)logbrush->lbHatch); |
| if (info->bmiHeader.biCompression) |
| bmSize = info->bmiHeader.biSizeImage; |
| else |
| bmSize = (info->bmiHeader.biWidth * info->bmiHeader.biBitCount |
| + 31) / 32 * 8 * info->bmiHeader.biHeight; |
| biSize = DIB_BitmapInfoSize(info, LOWORD(logbrush->lbColor)); |
| len = sizeof(METARECORD) + biSize + bmSize + 2; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| memset(mr, 0, len); |
| mr->rdFunction = META_DIBCREATEPATTERNBRUSH; |
| mr->rdSize = len / 2; |
| *(mr->rdParm) = logbrush->lbStyle; |
| *(mr->rdParm + 1) = LOWORD(logbrush->lbColor); |
| memcpy(mr->rdParm + 2, info, biSize + bmSize); |
| break; |
| default: |
| return FALSE; |
| } |
| if (!(MF_WriteRecord(dc, mr, len))) |
| { |
| GlobalFree16(hmr); |
| return FALSE; |
| } |
| |
| GlobalFree16(hmr); |
| |
| mr = (METARECORD *)&buffer; |
| mr->rdSize = sizeof(METARECORD) / 2; |
| mr->rdFunction = META_SELECTOBJECT; |
| |
| if ((index = MF_AddHandleDC( dc )) == -1) return FALSE; |
| *(mr->rdParm) = index; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_CreatePenIndirect |
| */ |
| |
| BOOL32 MF_CreatePenIndirect(DC *dc, HPEN16 hPen, LOGPEN16 *logpen) |
| { |
| int index; |
| char buffer[sizeof(METARECORD) - 2 + sizeof(*logpen)]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = (sizeof(METARECORD) + sizeof(*logpen) - 2) / 2; |
| mr->rdFunction = META_CREATEPENINDIRECT; |
| memcpy(&(mr->rdParm), logpen, sizeof(*logpen)); |
| if (!(MF_WriteRecord( dc, mr, mr->rdSize * 2))) return FALSE; |
| |
| mr->rdSize = sizeof(METARECORD) / 2; |
| mr->rdFunction = META_SELECTOBJECT; |
| |
| if ((index = MF_AddHandleDC( dc )) == -1) return FALSE; |
| *(mr->rdParm) = index; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_CreateFontIndirect |
| */ |
| |
| BOOL32 MF_CreateFontIndirect(DC *dc, HFONT16 hFont, LOGFONT16 *logfont) |
| { |
| int index; |
| char buffer[sizeof(METARECORD) - 2 + sizeof(LOGFONT16)]; |
| METARECORD *mr = (METARECORD *)&buffer; |
| |
| mr->rdSize = (sizeof(METARECORD) + sizeof(LOGFONT16) - 2) / 2; |
| mr->rdFunction = META_CREATEFONTINDIRECT; |
| memcpy(&(mr->rdParm), logfont, sizeof(LOGFONT16)); |
| if (!(MF_WriteRecord( dc, mr, mr->rdSize * 2))) return FALSE; |
| |
| mr->rdSize = sizeof(METARECORD) / 2; |
| mr->rdFunction = META_SELECTOBJECT; |
| |
| if ((index = MF_AddHandleDC( dc )) == -1) return FALSE; |
| *(mr->rdParm) = index; |
| return MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| } |
| |
| |
| /****************************************************************** |
| * MF_TextOut |
| */ |
| BOOL32 MF_TextOut(DC *dc, short x, short y, LPCSTR str, short count) |
| { |
| BOOL32 ret; |
| DWORD len; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| |
| len = sizeof(METARECORD) + (((count + 1) >> 1) * 2) + 4; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| memset(mr, 0, len); |
| |
| mr->rdSize = len / 2; |
| mr->rdFunction = META_TEXTOUT; |
| *(mr->rdParm) = count; |
| memcpy(mr->rdParm + 1, str, count); |
| *(mr->rdParm + ((count + 1) >> 1) + 1) = y; |
| *(mr->rdParm + ((count + 1) >> 1) + 2) = x; |
| ret = MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| GlobalFree16(hmr); |
| return ret; |
| } |
| |
| /****************************************************************** |
| * MF_ExtTextOut |
| */ |
| BOOL32 MF_ExtTextOut(DC*dc, short x, short y, UINT16 flags, const RECT16 *rect, |
| LPCSTR str, short count, const INT16 *lpDx) |
| { |
| BOOL32 ret; |
| DWORD len; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| |
| if((!flags && rect) || (flags && !rect)) |
| WARN(metafile, "Inconsistent flags and rect\n"); |
| len = sizeof(METARECORD) + (((count + 1) >> 1) * 2) + 2 * sizeof(short) |
| + sizeof(UINT16); |
| if(rect) |
| len += sizeof(RECT16); |
| if (lpDx) |
| len+=count*sizeof(INT16); |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| memset(mr, 0, len); |
| |
| mr->rdSize = len / 2; |
| mr->rdFunction = META_EXTTEXTOUT; |
| *(mr->rdParm) = y; |
| *(mr->rdParm + 1) = x; |
| *(mr->rdParm + 2) = count; |
| *(mr->rdParm + 3) = flags; |
| if (rect) memcpy(mr->rdParm + 4, rect, sizeof(RECT16)); |
| memcpy(mr->rdParm + (rect ? 8 : 4), str, count); |
| if (lpDx) |
| memcpy(mr->rdParm + (rect ? 8 : 4) + ((count + 1) >> 1),lpDx, |
| count*sizeof(INT16)); |
| ret = MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| GlobalFree16(hmr); |
| return ret; |
| } |
| |
| /****************************************************************** |
| * MF_MetaPoly - implements Polygon and Polyline |
| */ |
| BOOL32 MF_MetaPoly(DC *dc, short func, LPPOINT16 pt, short count) |
| { |
| BOOL32 ret; |
| DWORD len; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| |
| len = sizeof(METARECORD) + (count * 4); |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| memset(mr, 0, len); |
| |
| mr->rdSize = len / 2; |
| mr->rdFunction = func; |
| *(mr->rdParm) = count; |
| memcpy(mr->rdParm + 1, pt, count * 4); |
| ret = MF_WriteRecord( dc, mr, mr->rdSize * 2); |
| GlobalFree16(hmr); |
| return ret; |
| } |
| |
| |
| /****************************************************************** |
| * MF_BitBlt |
| */ |
| BOOL32 MF_BitBlt(DC *dcDest, short xDest, short yDest, short width, |
| short height, DC *dcSrc, short xSrc, short ySrc, DWORD rop) |
| { |
| BOOL32 ret; |
| DWORD len; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| BITMAP16 BM; |
| |
| GetObject16(dcSrc->w.hBitmap, sizeof(BITMAP16), &BM); |
| len = sizeof(METARECORD) + 12 * sizeof(INT16) + BM.bmWidthBytes * BM.bmHeight; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| mr->rdFunction = META_BITBLT; |
| *(mr->rdParm + 7) = BM.bmWidth; |
| *(mr->rdParm + 8) = BM.bmHeight; |
| *(mr->rdParm + 9) = BM.bmWidthBytes; |
| *(mr->rdParm +10) = BM.bmPlanes; |
| *(mr->rdParm +11) = BM.bmBitsPixel; |
| TRACE(metafile,"MF_StretchBlt->len = %ld rop=%lx \n",len,rop); |
| if (GetBitmapBits32(dcSrc->w.hBitmap,BM.bmWidthBytes * BM.bmHeight, |
| mr->rdParm +12)) |
| { |
| mr->rdSize = len / sizeof(INT16); |
| *(mr->rdParm) = HIWORD(rop); |
| *(mr->rdParm + 1) = ySrc; |
| *(mr->rdParm + 2) = xSrc; |
| *(mr->rdParm + 3) = height; |
| *(mr->rdParm + 4) = width; |
| *(mr->rdParm + 5) = yDest; |
| *(mr->rdParm + 6) = xDest; |
| ret = MF_WriteRecord( dcDest, mr, mr->rdSize * 2); |
| } |
| else |
| ret = FALSE; |
| GlobalFree16(hmr); |
| return ret; |
| } |
| |
| |
| /********************************************************************** |
| * MF_StretchBlt |
| * this function contains TWO ways for procesing StretchBlt in metafiles, |
| * decide between rdFunction values META_STRETCHBLT or META_DIBSTRETCHBLT |
| * via #define STRETCH_VIA_DIB |
| */ |
| #define STRETCH_VIA_DIB |
| #undef STRETCH_VIA_DIB |
| BOOL32 MF_StretchBlt(DC *dcDest, short xDest, short yDest, short widthDest, |
| short heightDest, DC *dcSrc, short xSrc, short ySrc, |
| short widthSrc, short heightSrc, DWORD rop) |
| { |
| BOOL32 ret; |
| DWORD len; |
| HGLOBAL16 hmr; |
| METARECORD *mr; |
| BITMAP16 BM; |
| #ifdef STRETCH_VIA_DIB |
| LPBITMAPINFOHEADER lpBMI; |
| WORD nBPP; |
| #endif |
| GetObject16(dcSrc->w.hBitmap, sizeof(BITMAP16), &BM); |
| #ifdef STRETCH_VIA_DIB |
| nBPP = BM.bmPlanes * BM.bmBitsPixel; |
| len = sizeof(METARECORD) + 10 * sizeof(INT16) |
| + sizeof(BITMAPINFOHEADER) + (nBPP != 24 ? 1 << nBPP: 0) * sizeof(RGBQUAD) |
| + ((BM.bmWidth * nBPP + 31) / 32) * 4 * BM.bmHeight; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| mr->rdFunction = META_DIBSTRETCHBLT; |
| lpBMI=(LPBITMAPINFOHEADER)(mr->rdParm+10); |
| lpBMI->biSize = sizeof(BITMAPINFOHEADER); |
| lpBMI->biWidth = BM.bmWidth; |
| lpBMI->biHeight = BM.bmHeight; |
| lpBMI->biPlanes = 1; |
| lpBMI->biBitCount = nBPP; /* 1,4,8 or 24 */ |
| lpBMI->biClrUsed = nBPP != 24 ? 1 << nBPP : 0; |
| lpBMI->biSizeImage = ((lpBMI->biWidth * nBPP + 31) / 32) * 4 * lpBMI->biHeight; |
| lpBMI->biCompression = BI_RGB; |
| lpBMI->biXPelsPerMeter = MulDiv32(GetDeviceCaps(dcSrc->hSelf,LOGPIXELSX),3937,100); |
| lpBMI->biYPelsPerMeter = MulDiv32(GetDeviceCaps(dcSrc->hSelf,LOGPIXELSY),3937,100); |
| lpBMI->biClrImportant = 0; /* 1 meter = 39.37 inch */ |
| |
| TRACE(metafile,"MF_StretchBltViaDIB->len = %ld rop=%lx PixYPM=%ld Caps=%d\n", |
| len,rop,lpBMI->biYPelsPerMeter,GetDeviceCaps(hdcSrc,LOGPIXELSY)); |
| if (GetDIBits(hdcSrc,dcSrc->w.hBitmap,0,(UINT32)lpBMI->biHeight, |
| (LPSTR)lpBMI + DIB_BitmapInfoSize( (BITMAPINFO *)lpBMI, |
| DIB_RGB_COLORS ), |
| (LPBITMAPINFO)lpBMI, DIB_RGB_COLORS)) |
| #else |
| len = sizeof(METARECORD) + 15 * sizeof(INT16) + BM.bmWidthBytes * BM.bmHeight; |
| if (!(hmr = GlobalAlloc16(GMEM_MOVEABLE, len))) |
| return FALSE; |
| mr = (METARECORD *)GlobalLock16(hmr); |
| mr->rdFunction = META_STRETCHBLT; |
| *(mr->rdParm +10) = BM.bmWidth; |
| *(mr->rdParm +11) = BM.bmHeight; |
| *(mr->rdParm +12) = BM.bmWidthBytes; |
| *(mr->rdParm +13) = BM.bmPlanes; |
| *(mr->rdParm +14) = BM.bmBitsPixel; |
| TRACE(metafile,"MF_StretchBlt->len = %ld rop=%lx \n",len,rop); |
| if (GetBitmapBits32( dcSrc->w.hBitmap, BM.bmWidthBytes * BM.bmHeight, |
| mr->rdParm +15)) |
| #endif |
| { |
| mr->rdSize = len / sizeof(INT16); |
| *(mr->rdParm) = LOWORD(rop); |
| *(mr->rdParm + 1) = HIWORD(rop); |
| *(mr->rdParm + 2) = heightSrc; |
| *(mr->rdParm + 3) = widthSrc; |
| *(mr->rdParm + 4) = ySrc; |
| *(mr->rdParm + 5) = xSrc; |
| *(mr->rdParm + 6) = heightDest; |
| *(mr->rdParm + 7) = widthDest; |
| *(mr->rdParm + 8) = yDest; |
| *(mr->rdParm + 9) = xDest; |
| ret = MF_WriteRecord( dcDest, mr, mr->rdSize * 2); |
| } |
| else |
| ret = FALSE; |
| GlobalFree16(hmr); |
| return ret; |
| } |
| |
| |
| /****************************************************************** |
| * MF_CreateRegion |
| */ |
| INT16 MF_CreateRegion(DC *dc, HRGN32 hrgn) |
| { |
| DWORD len; |
| METARECORD *mr; |
| RGNDATA *rgndata; |
| RECT32 *pCurRect, *pEndRect; |
| WORD Bands = 0, MaxBands = 0; |
| WORD *Param, *StartBand; |
| BOOL32 ret; |
| |
| len = GetRegionData32( hrgn, 0, NULL ); |
| if( !(rgndata = HeapAlloc( SystemHeap, 0, len )) ) { |
| WARN(metafile, "MF_CreateRegion: can't alloc rgndata buffer\n"); |
| return -1; |
| } |
| GetRegionData32( hrgn, len, rgndata ); |
| |
| /* Overestimate of length: |
| * Assume every rect is a separate band -> 6 WORDs per rect |
| */ |
| len = sizeof(METARECORD) + 20 + (rgndata->rdh.nCount * 12); |
| if( !(mr = HeapAlloc( SystemHeap, 0, len )) ) { |
| WARN(metafile, "MF_CreateRegion: can't alloc METARECORD buffer\n"); |
| HeapFree( SystemHeap, 0, rgndata ); |
| return -1; |
| } |
| |
| memset(mr, 0, len); |
| |
| Param = mr->rdParm + 11; |
| StartBand = NULL; |
| |
| pEndRect = (RECT32 *)rgndata->Buffer + rgndata->rdh.nCount; |
| for(pCurRect = (RECT32 *)rgndata->Buffer; pCurRect < pEndRect; pCurRect++) |
| { |
| if( StartBand && pCurRect->top == *(StartBand + 1) ) |
| { |
| *Param++ = pCurRect->left; |
| *Param++ = pCurRect->right; |
| } |
| else |
| { |
| if(StartBand) |
| { |
| *StartBand = Param - StartBand - 3; |
| *Param++ = *StartBand; |
| if(*StartBand > MaxBands) |
| MaxBands = *StartBand; |
| Bands++; |
| } |
| StartBand = Param++; |
| *Param++ = pCurRect->top; |
| *Param++ = pCurRect->bottom; |
| *Param++ = pCurRect->left; |
| *Param++ = pCurRect->right; |
| } |
| } |
| len = Param - (WORD *)mr; |
| |
| mr->rdParm[0] = 0; |
| mr->rdParm[1] = 6; |
| mr->rdParm[2] = 0x1234; |
| mr->rdParm[3] = 0; |
| mr->rdParm[4] = len * 2; |
| mr->rdParm[5] = Bands; |
| mr->rdParm[6] = MaxBands; |
| mr->rdParm[7] = rgndata->rdh.rcBound.left; |
| mr->rdParm[8] = rgndata->rdh.rcBound.top; |
| mr->rdParm[9] = rgndata->rdh.rcBound.right; |
| mr->rdParm[10] = rgndata->rdh.rcBound.bottom; |
| mr->rdFunction = META_CREATEREGION; |
| mr->rdSize = len / 2; |
| ret = MF_WriteRecord( dc, mr, mr->rdSize * 2 ); |
| HeapFree( SystemHeap, 0, mr ); |
| HeapFree( SystemHeap, 0, rgndata ); |
| if(!ret) |
| { |
| WARN(metafile, "MF_CreateRegion: MF_WriteRecord failed\n"); |
| return -1; |
| } |
| return MF_AddHandleDC( dc ); |
| } |