2019 lines
58 KiB
C
2019 lines
58 KiB
C
/*
|
|
* Unit test suite for metafiles
|
|
*
|
|
* Copyright (C) 2011 Vincent Povirk 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 <math.h>
|
|
|
|
#include "objbase.h"
|
|
#include "gdiplus.h"
|
|
#include "wine/test.h"
|
|
|
|
#define expect(expected, got) ok(got == expected, "Expected %.8x, got %.8x\n", expected, got)
|
|
#define expectf_(expected, got, precision) ok(fabs((expected) - (got)) <= (precision), "Expected %f, got %f\n", (expected), (got))
|
|
#define expectf(expected, got) expectf_((expected), (got), 0.001)
|
|
|
|
static BOOL save_metafiles;
|
|
static BOOL load_metafiles;
|
|
|
|
typedef struct emfplus_record
|
|
{
|
|
BOOL todo;
|
|
ULONG record_type;
|
|
BOOL playback_todo;
|
|
} emfplus_record;
|
|
|
|
typedef struct emfplus_check_state
|
|
{
|
|
const char *desc;
|
|
int count;
|
|
const struct emfplus_record *expected;
|
|
GpMetafile *metafile;
|
|
} emfplus_check_state;
|
|
|
|
static void check_record(int count, const char *desc, const struct emfplus_record *expected, const struct emfplus_record *actual)
|
|
{
|
|
todo_wine_if (expected->todo)
|
|
ok(expected->record_type == actual->record_type,
|
|
"%s.%i: Expected record type 0x%x, got 0x%x\n", desc, count,
|
|
expected->record_type, actual->record_type);
|
|
}
|
|
|
|
typedef struct EmfPlusRecordHeader
|
|
{
|
|
WORD Type;
|
|
WORD Flags;
|
|
DWORD Size;
|
|
DWORD DataSize;
|
|
} EmfPlusRecordHeader;
|
|
|
|
static int CALLBACK enum_emf_proc(HDC hDC, HANDLETABLE *lpHTable, const ENHMETARECORD *lpEMFR,
|
|
int nObj, LPARAM lpData)
|
|
{
|
|
emfplus_check_state *state = (emfplus_check_state*)lpData;
|
|
emfplus_record actual;
|
|
|
|
if (lpEMFR->iType == EMR_GDICOMMENT)
|
|
{
|
|
const EMRGDICOMMENT *comment = (const EMRGDICOMMENT*)lpEMFR;
|
|
|
|
if (comment->cbData >= 4 && memcmp(comment->Data, "EMF+", 4) == 0)
|
|
{
|
|
int offset = 4;
|
|
|
|
while (offset + sizeof(EmfPlusRecordHeader) <= comment->cbData)
|
|
{
|
|
const EmfPlusRecordHeader *record = (const EmfPlusRecordHeader*)&comment->Data[offset];
|
|
|
|
ok(record->Size == record->DataSize + sizeof(EmfPlusRecordHeader),
|
|
"%s: EMF+ record datasize %u and size %u mismatch\n", state->desc, record->DataSize, record->Size);
|
|
|
|
ok(offset + record->DataSize <= comment->cbData,
|
|
"%s: EMF+ record truncated\n", state->desc);
|
|
|
|
if (offset + record->DataSize > comment->cbData)
|
|
return 0;
|
|
|
|
if (state->expected[state->count].record_type)
|
|
{
|
|
actual.todo = FALSE;
|
|
actual.record_type = record->Type;
|
|
|
|
check_record(state->count, state->desc, &state->expected[state->count], &actual);
|
|
|
|
state->count++;
|
|
}
|
|
else
|
|
{
|
|
ok(0, "%s: Unexpected EMF+ 0x%x record\n", state->desc, record->Type);
|
|
}
|
|
|
|
offset += record->Size;
|
|
}
|
|
|
|
ok(offset == comment->cbData, "%s: truncated EMF+ record data?\n", state->desc);
|
|
|
|
return 1;
|
|
}
|
|
}
|
|
|
|
if (state->expected[state->count].record_type)
|
|
{
|
|
actual.todo = FALSE;
|
|
actual.record_type = lpEMFR->iType;
|
|
|
|
check_record(state->count, state->desc, &state->expected[state->count], &actual);
|
|
|
|
state->count++;
|
|
}
|
|
else
|
|
{
|
|
ok(0, "%s: Unexpected EMF 0x%x record\n", state->desc, lpEMFR->iType);
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
static void check_emfplus(HENHMETAFILE hemf, const emfplus_record *expected, const char *desc)
|
|
{
|
|
emfplus_check_state state;
|
|
|
|
state.desc = desc;
|
|
state.count = 0;
|
|
state.expected = expected;
|
|
|
|
EnumEnhMetaFile(0, hemf, enum_emf_proc, &state, NULL);
|
|
|
|
todo_wine_if (expected[state.count].todo)
|
|
ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
|
}
|
|
|
|
static BOOL CALLBACK enum_metafile_proc(EmfPlusRecordType record_type, unsigned int flags,
|
|
unsigned int dataSize, const unsigned char *pStr, void *userdata)
|
|
{
|
|
emfplus_check_state *state = (emfplus_check_state*)userdata;
|
|
emfplus_record actual;
|
|
|
|
actual.todo = FALSE;
|
|
actual.record_type = record_type;
|
|
|
|
if (dataSize == 0)
|
|
ok(pStr == NULL, "non-NULL pStr\n");
|
|
|
|
if (state->expected[state->count].record_type)
|
|
{
|
|
check_record(state->count, state->desc, &state->expected[state->count], &actual);
|
|
|
|
state->count++;
|
|
}
|
|
else
|
|
{
|
|
ok(0, "%s: Unexpected EMF 0x%x record\n", state->desc, record_type);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void check_metafile(GpMetafile *metafile, const emfplus_record *expected, const char *desc,
|
|
const GpPointF *dst_points, const GpRectF *src_rect, Unit src_unit)
|
|
{
|
|
GpStatus stat;
|
|
HDC hdc;
|
|
GpGraphics *graphics;
|
|
emfplus_check_state state;
|
|
|
|
state.desc = desc;
|
|
state.count = 0;
|
|
state.expected = expected;
|
|
state.metafile = metafile;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipCreateFromHDC(hdc, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipEnumerateMetafileSrcRectDestPoints(graphics, metafile, dst_points,
|
|
3, src_rect, src_unit, enum_metafile_proc, &state, NULL);
|
|
expect(Ok, stat);
|
|
|
|
todo_wine_if (expected[state.count].todo)
|
|
ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
|
|
|
GdipDeleteGraphics(graphics);
|
|
|
|
DeleteDC(hdc);
|
|
}
|
|
|
|
static BOOL CALLBACK play_metafile_proc(EmfPlusRecordType record_type, unsigned int flags,
|
|
unsigned int dataSize, const unsigned char *pStr, void *userdata)
|
|
{
|
|
emfplus_check_state *state = (emfplus_check_state*)userdata;
|
|
GpStatus stat;
|
|
|
|
stat = GdipPlayMetafileRecord(state->metafile, record_type, flags, dataSize, pStr);
|
|
|
|
if (state->expected[state->count].record_type)
|
|
{
|
|
todo_wine_if (state->expected[state->count].playback_todo)
|
|
ok(stat == Ok, "%s.%i: GdipPlayMetafileRecord failed with stat %i\n", state->desc, state->count, stat);
|
|
state->count++;
|
|
}
|
|
else
|
|
{
|
|
todo_wine_if (state->expected[state->count].playback_todo)
|
|
ok(0, "%s: too many records\n", state->desc);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void play_metafile(GpMetafile *metafile, GpGraphics *graphics, const emfplus_record *expected,
|
|
const char *desc, const GpPointF *dst_points, const GpRectF *src_rect, Unit src_unit)
|
|
{
|
|
GpStatus stat;
|
|
emfplus_check_state state;
|
|
|
|
state.desc = desc;
|
|
state.count = 0;
|
|
state.expected = expected;
|
|
state.metafile = metafile;
|
|
|
|
stat = GdipEnumerateMetafileSrcRectDestPoints(graphics, metafile, dst_points,
|
|
3, src_rect, src_unit, play_metafile_proc, &state, NULL);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
/* When 'save' or 'load' is specified on the command line, save or
|
|
* load the specified filename. */
|
|
static void sync_metafile(GpMetafile **metafile, const char *filename)
|
|
{
|
|
GpStatus stat;
|
|
if (save_metafiles)
|
|
{
|
|
GpMetafile *clone;
|
|
HENHMETAFILE hemf;
|
|
|
|
stat = GdipCloneImage((GpImage*)*metafile, (GpImage**)&clone);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(clone, &hemf);
|
|
expect(Ok, stat);
|
|
|
|
DeleteEnhMetaFile(CopyEnhMetaFileA(hemf, filename));
|
|
|
|
DeleteEnhMetaFile(hemf);
|
|
|
|
stat = GdipDisposeImage((GpImage*)clone);
|
|
expect(Ok, stat);
|
|
}
|
|
else if (load_metafiles)
|
|
{
|
|
HENHMETAFILE hemf;
|
|
|
|
stat = GdipDisposeImage((GpImage*)*metafile);
|
|
expect(Ok, stat);
|
|
*metafile = NULL;
|
|
|
|
hemf = GetEnhMetaFileA(filename);
|
|
ok(hemf != NULL, "%s could not be opened\n", filename);
|
|
|
|
stat = GdipCreateMetafileFromEmf(hemf, TRUE, metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
}
|
|
|
|
static const emfplus_record empty_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_empty(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
GpRectF bounds;
|
|
GpUnit unit;
|
|
REAL xres, yres;
|
|
HENHMETAFILE hemf, dummy;
|
|
MetafileHeader header;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(NULL, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, MetafileTypeInvalid, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, MetafileTypeWmf, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, MetafileTypeWmfPlaceable, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, MetafileTypeEmfPlusDual+1, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, NULL);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, empty_records, "empty metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "empty.emf");
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
expectf_(100.0, bounds.Width, 0.05);
|
|
expectf_(100.0, bounds.Height, 0.05);
|
|
expect(UnitPixel, unit);
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &xres);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &yres);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &dummy);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
check_emfplus(hemf, empty_records, "empty emf");
|
|
|
|
memset(&header, 0xaa, sizeof(header));
|
|
stat = GdipGetMetafileHeaderFromEmf(hemf, &header);
|
|
expect(Ok, stat);
|
|
expect(MetafileTypeEmfPlusOnly, header.Type);
|
|
expect(U(header).EmfHeader.nBytes, header.Size);
|
|
ok(header.Version == 0xdbc01001 || header.Version == 0xdbc01002, "Unexpected version %x\n", header.Version);
|
|
expect(1, header.EmfPlusFlags); /* reference device was display, not printer */
|
|
expectf(xres, header.DpiX);
|
|
expectf(xres, U(header).EmfHeader.szlDevice.cx / (REAL)U(header).EmfHeader.szlMillimeters.cx * 25.4);
|
|
expectf(yres, header.DpiY);
|
|
expectf(yres, U(header).EmfHeader.szlDevice.cy / (REAL)U(header).EmfHeader.szlMillimeters.cy * 25.4);
|
|
expect(0, header.X);
|
|
expect(0, header.Y);
|
|
expect(100, header.Width);
|
|
expect(100, header.Height);
|
|
expect(28, header.EmfPlusHeaderSize);
|
|
expect(96, header.LogicalDpiX);
|
|
expect(96, header.LogicalDpiX);
|
|
expect(EMR_HEADER, U(header).EmfHeader.iType);
|
|
expect(0, U(header).EmfHeader.rclBounds.left);
|
|
expect(0, U(header).EmfHeader.rclBounds.top);
|
|
expect(-1, U(header).EmfHeader.rclBounds.right);
|
|
expect(-1, U(header).EmfHeader.rclBounds.bottom);
|
|
expect(0, U(header).EmfHeader.rclFrame.left);
|
|
expect(0, U(header).EmfHeader.rclFrame.top);
|
|
expectf_(100.0, U(header).EmfHeader.rclFrame.right * xres / 2540.0, 2.0);
|
|
expectf_(100.0, U(header).EmfHeader.rclFrame.bottom * yres / 2540.0, 2.0);
|
|
|
|
stat = GdipCreateMetafileFromEmf(hemf, TRUE, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
expectf_(100.0, bounds.Width, 0.05);
|
|
expectf_(100.0, bounds.Height, 0.05);
|
|
expect(UnitPixel, unit);
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &xres);
|
|
expect(Ok, stat);
|
|
expectf(header.DpiX, xres);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &yres);
|
|
expect(Ok, stat);
|
|
expectf(header.DpiY, yres);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record getdc_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeGetDC},
|
|
{0, EMR_CREATEBRUSHINDIRECT},
|
|
{0, EMR_SELECTOBJECT},
|
|
{0, EMR_RECTANGLE},
|
|
{0, EMR_SELECTOBJECT},
|
|
{0, EMR_DELETEOBJECT},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_getdc(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc, metafile_dc;
|
|
HENHMETAFILE hemf;
|
|
BOOL ret;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const GpPointF dst_points_half[3] = {{0.0,0.0},{50.0,0.0},{0.0,50.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
HBRUSH hbrush, holdbrush;
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetDC(graphics, &metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
if (stat != Ok)
|
|
{
|
|
GdipDeleteGraphics(graphics);
|
|
GdipDisposeImage((GpImage*)metafile);
|
|
return;
|
|
}
|
|
|
|
hbrush = CreateSolidBrush(0xff0000);
|
|
|
|
holdbrush = SelectObject(metafile_dc, hbrush);
|
|
|
|
Rectangle(metafile_dc, 25, 25, 75, 75);
|
|
|
|
SelectObject(metafile_dc, holdbrush);
|
|
|
|
DeleteObject(hbrush);
|
|
|
|
stat = GdipReleaseDC(graphics, metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, getdc_records, "getdc metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "getdc.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, getdc_records, "getdc playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapSetPixel(bitmap, 50, 50, 0);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, getdc_records, "getdc playback", dst_points_half, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapSetPixel(bitmap, 15, 15, 0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDrawImagePointsRect(graphics, (GpImage*)metafile, dst_points, 3,
|
|
0.0, 0.0, 100.0, 100.0, UnitPixel, NULL, NULL, NULL);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
check_emfplus(hemf, getdc_records, "getdc emf");
|
|
|
|
ret = DeleteEnhMetaFile(hemf);
|
|
ok(ret != 0, "Failed to delete enhmetafile %p\n", hemf);
|
|
}
|
|
|
|
static const emfplus_record emfonly_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EMR_CREATEBRUSHINDIRECT},
|
|
{0, EMR_SELECTOBJECT},
|
|
{0, EMR_RECTANGLE},
|
|
{0, EMR_SELECTOBJECT},
|
|
{0, EMR_DELETEOBJECT},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_emfonly(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpImage *clone;
|
|
GpGraphics *graphics;
|
|
HDC hdc, metafile_dc;
|
|
GpRectF bounds;
|
|
GpUnit unit;
|
|
REAL xres, yres;
|
|
HENHMETAFILE hemf;
|
|
MetafileHeader header;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
HBRUSH hbrush, holdbrush;
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetDC(graphics, &metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
if (stat != Ok)
|
|
{
|
|
GdipDeleteGraphics(graphics);
|
|
GdipDisposeImage((GpImage*)metafile);
|
|
return;
|
|
}
|
|
|
|
hbrush = CreateSolidBrush(0xff0000);
|
|
|
|
holdbrush = SelectObject(metafile_dc, hbrush);
|
|
|
|
Rectangle(metafile_dc, 25, 25, 75, 75);
|
|
|
|
SelectObject(metafile_dc, holdbrush);
|
|
|
|
DeleteObject(hbrush);
|
|
|
|
stat = GdipReleaseDC(graphics, metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, emfonly_records, "emfonly metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "emfonly.emf");
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
expectf_(100.0, bounds.Width, 0.05);
|
|
expectf_(100.0, bounds.Height, 0.05);
|
|
expect(UnitPixel, unit);
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &xres);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &yres);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, emfonly_records, "emfonly playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapSetPixel(bitmap, 50, 50, 0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDrawImagePointsRect(graphics, (GpImage*)metafile, dst_points, 3,
|
|
0.0, 0.0, 100.0, 100.0, UnitPixel, NULL, NULL, NULL);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipCloneImage((GpImage*)metafile, &clone);
|
|
expect(Ok, stat);
|
|
|
|
if (stat == Ok)
|
|
{
|
|
stat = GdipBitmapSetPixel(bitmap, 50, 50, 0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDrawImagePointsRect(graphics, clone, dst_points, 3,
|
|
0.0, 0.0, 100.0, 100.0, UnitPixel, NULL, NULL, NULL);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
GdipDisposeImage(clone);
|
|
}
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
check_emfplus(hemf, emfonly_records, "emfonly emf");
|
|
|
|
memset(&header, 0xaa, sizeof(header));
|
|
stat = GdipGetMetafileHeaderFromEmf(hemf, &header);
|
|
expect(Ok, stat);
|
|
expect(MetafileTypeEmf, header.Type);
|
|
expect(U(header).EmfHeader.nBytes, header.Size);
|
|
expect(0x10000, header.Version);
|
|
expect(0, header.EmfPlusFlags);
|
|
expectf(xres, header.DpiX);
|
|
expectf(xres, U(header).EmfHeader.szlDevice.cx / (REAL)U(header).EmfHeader.szlMillimeters.cx * 25.4);
|
|
expectf(yres, header.DpiY);
|
|
expectf(yres, U(header).EmfHeader.szlDevice.cy / (REAL)U(header).EmfHeader.szlMillimeters.cy * 25.4);
|
|
expect(0, header.X);
|
|
expect(0, header.Y);
|
|
expect(100, header.Width);
|
|
expect(100, header.Height);
|
|
expect(0, header.EmfPlusHeaderSize);
|
|
expect(0, header.LogicalDpiX);
|
|
expect(0, header.LogicalDpiX);
|
|
expect(EMR_HEADER, U(header).EmfHeader.iType);
|
|
expect(25, U(header).EmfHeader.rclBounds.left);
|
|
expect(25, U(header).EmfHeader.rclBounds.top);
|
|
expect(74, U(header).EmfHeader.rclBounds.right);
|
|
expect(74, U(header).EmfHeader.rclBounds.bottom);
|
|
expect(0, U(header).EmfHeader.rclFrame.left);
|
|
expect(0, U(header).EmfHeader.rclFrame.top);
|
|
expectf_(100.0, U(header).EmfHeader.rclFrame.right * xres / 2540.0, 2.0);
|
|
expectf_(100.0, U(header).EmfHeader.rclFrame.bottom * yres / 2540.0, 2.0);
|
|
|
|
stat = GdipCreateMetafileFromEmf(hemf, TRUE, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
expectf_(100.0, bounds.Width, 0.05);
|
|
expectf_(100.0, bounds.Height, 0.05);
|
|
expect(UnitPixel, unit);
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &xres);
|
|
expect(Ok, stat);
|
|
expectf(header.DpiX, xres);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &yres);
|
|
expect(Ok, stat);
|
|
expectf(header.DpiY, yres);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record fillrect_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_fillrect(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
HENHMETAFILE hemf;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const GpPointF dst_points_half[3] = {{0.0,0.0},{50.0,0.0},{0.0,50.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
GpBrush *brush;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 25, 25, 75, 75);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, fillrect_records, "fillrect metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "fillrect.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, fillrect_records, "fillrect playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapSetPixel(bitmap, 50, 50, 0);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, fillrect_records, "fillrect playback", dst_points_half, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapSetPixel(bitmap, 15, 15, 0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDrawImagePointsRect(graphics, (GpImage*)metafile, dst_points, 3,
|
|
0.0, 0.0, 100.0, 100.0, UnitPixel, NULL, NULL, NULL);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record clear_emf_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeClear},
|
|
{1, EMR_SAVEDC},
|
|
{1, EMR_SETICMMODE},
|
|
{1, EMR_BITBLT},
|
|
{1, EMR_RESTOREDC},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_clear(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
HENHMETAFILE hemf;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{10.0,10.0},{20.0,10.0},{10.0,20.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGraphicsClear(graphics, 0xffffff00);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
sync_metafile(&metafile, "clear.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(30, 30, 0, PixelFormat32bppRGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDrawImagePointsRect(graphics, (GpImage*)metafile, dst_points, 3,
|
|
0.0, 0.0, 100.0, 100.0, UnitPixel, NULL, NULL, NULL);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 5, 5, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff000000, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 15, 15, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffffff00, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 25, 25, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff000000, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
check_emfplus(hemf, clear_emf_records, "clear emf");
|
|
|
|
DeleteEnhMetaFile(hemf);
|
|
}
|
|
|
|
static void test_nullframerect(void) {
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc, metafile_dc;
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpBrush *brush;
|
|
HBRUSH hbrush, holdbrush;
|
|
GpRectF bounds;
|
|
GpUnit unit;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, NULL, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
ok(bounds.Width == 1.0 || broken(bounds.Width == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Width);
|
|
ok(bounds.Height == 1.0 || broken(bounds.Height == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Height);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 25, 25, 75, 75);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
ok(bounds.Width == 1.0 || broken(bounds.Width == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Width);
|
|
ok(bounds.Height == 1.0 || broken(bounds.Height == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Height);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf_(25.0, bounds.X, 0.05);
|
|
expectf_(25.0, bounds.Y, 0.05);
|
|
expectf_(75.0, bounds.Width, 0.05);
|
|
expectf_(75.0, bounds.Height, 0.05);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, NULL, MetafileFrameUnitMillimeter, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetDC(graphics, &metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
if (stat != Ok)
|
|
{
|
|
GdipDeleteGraphics(graphics);
|
|
GdipDisposeImage((GpImage*)metafile);
|
|
return;
|
|
}
|
|
|
|
hbrush = CreateSolidBrush(0xff0000);
|
|
|
|
holdbrush = SelectObject(metafile_dc, hbrush);
|
|
|
|
Rectangle(metafile_dc, 25, 25, 75, 75);
|
|
|
|
SelectObject(metafile_dc, holdbrush);
|
|
|
|
DeleteObject(hbrush);
|
|
|
|
stat = GdipReleaseDC(graphics, metafile_dc);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf_(25.0, bounds.X, 0.05);
|
|
expectf_(25.0, bounds.Y, 0.05);
|
|
todo_wine expectf_(50.0, bounds.Width, 0.05);
|
|
todo_wine expectf_(50.0, bounds.Height, 0.05);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record pagetransform_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSetPageTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSetPageTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSetPageTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSetPageTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_pagetransform(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
static const GpRectF frame = {0.0, 0.0, 5.0, 5.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
GpBrush *brush;
|
|
GpUnit unit;
|
|
REAL scale, dpix, dpiy;
|
|
UINT width, height;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitInch, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &dpix);
|
|
todo_wine expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &dpiy);
|
|
todo_wine expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageWidth((GpImage*)metafile, &width);
|
|
todo_wine expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageHeight((GpImage*)metafile, &height);
|
|
todo_wine expect(InvalidParameter, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
/* initial scale */
|
|
stat = GdipGetPageUnit(graphics, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitDisplay, unit);
|
|
|
|
stat = GdipGetPageScale(graphics, &scale);
|
|
expect(Ok, stat);
|
|
expectf(1.0, scale);
|
|
|
|
stat = GdipGetDpiX(graphics, &dpix);
|
|
expect(Ok, stat);
|
|
expectf(96.0, dpix);
|
|
|
|
stat = GdipGetDpiY(graphics, &dpiy);
|
|
expect(Ok, stat);
|
|
expectf(96.0, dpiy);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 1, 2, 1, 1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* page unit = pixels */
|
|
stat = GdipSetPageUnit(graphics, UnitPixel);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetPageUnit(graphics, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff00ff00, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 0, 1, 1, 1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* page scale = 3, unit = pixels */
|
|
stat = GdipSetPageScale(graphics, 3.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetPageScale(graphics, &scale);
|
|
expect(Ok, stat);
|
|
expectf(3.0, scale);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff00ffff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 0, 1, 2, 2);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* page scale = 3, unit = inches */
|
|
stat = GdipSetPageUnit(graphics, UnitInch);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetPageUnit(graphics, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitInch, unit);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffff0000, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0/96.0, 0, 1, 1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* page scale = 3, unit = display */
|
|
stat = GdipSetPageUnit(graphics, UnitDisplay);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetPageUnit(graphics, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitDisplay, unit);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffff00ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 3, 3, 2, 2);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, pagetransform_records, "pagetransform metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "pagetransform.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, pagetransform_records, "pagetransform playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 30, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 10, 30, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff00ff00, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 20, 80, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff00ffff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 80, 20, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffff0000, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 80, 80, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffff00ff, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record worldtransform_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeScaleWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeResetWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeMultiplyWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeRotateWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSetWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeTranslateWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_worldtransform(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
static const GpRectF frame = {0.0, 0.0, 5.0, 5.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpBitmap *bitmap;
|
|
ARGB color;
|
|
GpBrush *brush;
|
|
GpMatrix *transform;
|
|
BOOL identity;
|
|
REAL elements[6];
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipCreateMatrix(&transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
/* initial transform */
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipIsMatrixIdentity(transform, &identity);
|
|
expect(Ok, stat);
|
|
expect(TRUE, identity);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangleI(graphics, brush, 0, 0, 1, 1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* scale transform */
|
|
stat = GdipScaleWorldTransform(graphics, 2.0, 4.0, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetMatrixElements(transform, elements);
|
|
expect(Ok, stat);
|
|
expectf(2.0, elements[0]);
|
|
expectf(0.0, elements[1]);
|
|
expectf(0.0, elements[2]);
|
|
expectf(4.0, elements[3]);
|
|
expectf(0.0, elements[4]);
|
|
expectf(0.0, elements[5]);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff00ff00, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 0.5, 0.5, 0.5, 0.25);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* reset transform */
|
|
stat = GdipResetWorldTransform(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipIsMatrixIdentity(transform, &identity);
|
|
expect(Ok, stat);
|
|
expect(TRUE, identity);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff00ffff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0, 0.0, 1.0, 1.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* multiply transform */
|
|
stat = GdipSetMatrixElements(transform, 2.0, 0.0, 0.0, 1.0, 0.0, 0.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipMultiplyWorldTransform(graphics, transform, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetMatrixElements(transform, elements);
|
|
expect(Ok, stat);
|
|
expectf(2.0, elements[0]);
|
|
expectf(0.0, elements[1]);
|
|
expectf(0.0, elements[2]);
|
|
expectf(1.0, elements[3]);
|
|
expectf(0.0, elements[4]);
|
|
expectf(0.0, elements[5]);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffff0000, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0, 1.0, 0.5, 1.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* rotate transform */
|
|
stat = GdipRotateWorldTransform(graphics, 90.0, MatrixOrderAppend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetMatrixElements(transform, elements);
|
|
expect(Ok, stat);
|
|
expectf(0.0, elements[0]);
|
|
expectf(2.0, elements[1]);
|
|
expectf(-1.0, elements[2]);
|
|
expectf(0.0, elements[3]);
|
|
expectf(0.0, elements[4]);
|
|
expectf(0.0, elements[5]);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffff00ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0, -1.0, 0.5, 1.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* set transform */
|
|
stat = GdipSetMatrixElements(transform, 1.0, 0.0, 0.0, 3.0, 0.0, 0.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipSetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetMatrixElements(transform, elements);
|
|
expect(Ok, stat);
|
|
expectf(1.0, elements[0]);
|
|
expectf(0.0, elements[1]);
|
|
expectf(0.0, elements[2]);
|
|
expectf(3.0, elements[3]);
|
|
expectf(0.0, elements[4]);
|
|
expectf(0.0, elements[5]);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffffff00, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0, 1.0, 1.0, 1.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
/* translate transform */
|
|
stat = GdipTranslateWorldTransform(graphics, -1.0, 0.0, MatrixOrderAppend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetWorldTransform(graphics, transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetMatrixElements(transform, elements);
|
|
expect(Ok, stat);
|
|
expectf(1.0, elements[0]);
|
|
expectf(0.0, elements[1]);
|
|
expectf(0.0, elements[2]);
|
|
expectf(3.0, elements[3]);
|
|
expectf(-1.0, elements[4]);
|
|
expectf(0.0, elements[5]);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xffffffff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 1.0, 1.0, 1.0, 1.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteMatrix(transform);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, worldtransform_records, "worldtransform metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "worldtransform.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, worldtransform_records, "worldtransform playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 80, 80, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 10, 10, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 30, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff00ff00, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 30, 10, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff00ffff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 50, 30, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffff0000, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 10, 50, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffff00ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 30, 90, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffffff00, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 10, 90, &color);
|
|
expect(Ok, stat);
|
|
expect(0xffffffff, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static void test_converttoemfplus(void)
|
|
{
|
|
GpStatus (WINAPI *pGdipConvertToEmfPlus)( const GpGraphics *graphics, GpMetafile *metafile, BOOL *succ,
|
|
EmfType emfType, const WCHAR *description, GpMetafile **outmetafile);
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpStatus stat;
|
|
GpMetafile *metafile, *metafile2 = NULL, *emhmeta;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
BOOL succ;
|
|
HMODULE mod = GetModuleHandleA("gdiplus.dll");
|
|
|
|
pGdipConvertToEmfPlus = (void*)GetProcAddress( mod, "GdipConvertToEmfPlus");
|
|
if(!pGdipConvertToEmfPlus)
|
|
{
|
|
/* GdipConvertToEmfPlus was introduced in Windows Vista. */
|
|
win_skip("GDIPlus version 1.1 not available\n");
|
|
return;
|
|
}
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, MetafileTypeEmf, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &emhmeta);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
/* Invalid Parameters */
|
|
stat = pGdipConvertToEmfPlus(NULL, metafile, &succ, EmfTypeEmfPlusOnly, description, &metafile2);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = pGdipConvertToEmfPlus(graphics, NULL, &succ, EmfTypeEmfPlusOnly, description, &metafile2);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = pGdipConvertToEmfPlus(graphics, metafile, &succ, EmfTypeEmfPlusOnly, description, NULL);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = pGdipConvertToEmfPlus(graphics, metafile, NULL, MetafileTypeInvalid, NULL, &metafile2);
|
|
expect(InvalidParameter, stat);
|
|
|
|
stat = pGdipConvertToEmfPlus(graphics, metafile, NULL, MetafileTypeEmfPlusDual+1, NULL, &metafile2);
|
|
expect(InvalidParameter, stat);
|
|
|
|
/* If we are already an Enhanced Metafile then the conversion fails. */
|
|
stat = pGdipConvertToEmfPlus(graphics, emhmeta, NULL, EmfTypeEmfPlusOnly, NULL, &metafile2);
|
|
todo_wine expect(InvalidParameter, stat);
|
|
|
|
stat = pGdipConvertToEmfPlus(graphics, metafile, NULL, EmfTypeEmfPlusOnly, NULL, &metafile2);
|
|
todo_wine expect(Ok, stat);
|
|
if(metafile2)
|
|
GdipDisposeImage((GpImage*)metafile2);
|
|
|
|
succ = FALSE;
|
|
stat = pGdipConvertToEmfPlus(graphics, metafile, &succ, EmfTypeEmfPlusOnly, NULL, &metafile2);
|
|
todo_wine expect(Ok, stat);
|
|
if(metafile2)
|
|
GdipDisposeImage((GpImage*)metafile2);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)emhmeta);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static void test_frameunit(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
HDC hdc;
|
|
static const GpRectF frame = {0.0, 0.0, 5.0, 5.0};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GpUnit unit;
|
|
REAL dpix, dpiy;
|
|
GpRectF bounds;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitInch, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
ok(bounds.Width == 1.0 || broken(bounds.Width == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Width);
|
|
ok(bounds.Height == 1.0 || broken(bounds.Height == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Height);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
ok(bounds.Width == 1.0 || broken(bounds.Width == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Width);
|
|
ok(bounds.Height == 1.0 || broken(bounds.Height == 0.0) /* xp sp1 */,
|
|
"expected 1.0, got %f\n", bounds.Height);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageHorizontalResolution((GpImage*)metafile, &dpix);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageVerticalResolution((GpImage*)metafile, &dpiy);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageBounds((GpImage*)metafile, &bounds, &unit);
|
|
expect(Ok, stat);
|
|
expect(UnitPixel, unit);
|
|
expectf(0.0, bounds.X);
|
|
expectf(0.0, bounds.Y);
|
|
expectf_(5.0 * dpix, bounds.Width, 1.0);
|
|
expectf_(5.0 * dpiy, bounds.Height, 1.0);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record container_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeBeginContainerNoParams},
|
|
{0, EmfPlusRecordTypeScaleWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeEndContainer},
|
|
{0, EmfPlusRecordTypeScaleWorldTransform},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeSave},
|
|
{0, EmfPlusRecordTypeRestore},
|
|
{0, EmfPlusRecordTypeScaleWorldTransform},
|
|
{0, EmfPlusRecordTypeBeginContainerNoParams},
|
|
{0, EmfPlusRecordTypeScaleWorldTransform},
|
|
{0, EmfPlusRecordTypeBeginContainerNoParams},
|
|
{0, EmfPlusRecordTypeEndContainer},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeBeginContainerNoParams},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_containers(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
GpBitmap *bitmap;
|
|
GpBrush *brush;
|
|
ARGB color;
|
|
HDC hdc;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GraphicsContainer state1, state2;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
/* Normal usage */
|
|
stat = GdipBeginContainer2(graphics, &state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipScaleWorldTransform(graphics, 2.0, 2.0, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff000000, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 5.0, 5.0, 5.0, 5.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipEndContainer(graphics, state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipScaleWorldTransform(graphics, 1.0, 1.0, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 5.0, 5.0, 5.0, 5.0);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipSaveGraphics(graphics, &state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipRestoreGraphics(graphics, state1);
|
|
expect(Ok, stat);
|
|
|
|
/* Popping two states at once */
|
|
stat = GdipScaleWorldTransform(graphics, 2.0, 2.0, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBeginContainer2(graphics, &state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipScaleWorldTransform(graphics, 4.0, 4.0, MatrixOrderPrepend);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipBeginContainer2(graphics, &state2);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipEndContainer(graphics, state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff00ff00, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 20.0, 20.0, 5.0, 5.0);
|
|
expect(Ok, stat);
|
|
|
|
/* Restoring an invalid state seems to break the graphics object? */
|
|
if (0) {
|
|
stat = GdipEndContainer(graphics, state1);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
/* Ending metafile with a state open */
|
|
stat = GdipBeginContainer2(graphics, &state1);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, container_records, "container metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "container.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, container_records, "container playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 80, 80, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 12, 12, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff000000, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 8, 8, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 42, 42, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff00ff00, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
static const emfplus_record clipping_records[] = {
|
|
{0, EMR_HEADER},
|
|
{0, EmfPlusRecordTypeHeader},
|
|
{0, EmfPlusRecordTypeSave},
|
|
{0, EmfPlusRecordTypeSetClipRect},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeRestore},
|
|
{0, EmfPlusRecordTypeSetClipRect},
|
|
{0, EmfPlusRecordTypeFillRects},
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
{0, EMR_EOF},
|
|
{0}
|
|
};
|
|
|
|
static void test_clipping(void)
|
|
{
|
|
GpStatus stat;
|
|
GpMetafile *metafile;
|
|
GpGraphics *graphics;
|
|
GpBitmap *bitmap;
|
|
GpBrush *brush;
|
|
ARGB color;
|
|
HDC hdc;
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
GraphicsState state;
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
|
expect(Ok, stat);
|
|
|
|
DeleteDC(hdc);
|
|
|
|
if (stat != Ok)
|
|
return;
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)metafile, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipSaveGraphics(graphics, &state);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipSetClipRect(graphics, 30, 30, 10, 10, CombineModeReplace);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff000000, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 0, 0, 100, 100);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipRestoreGraphics(graphics, state);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipSetClipRect(graphics, 30, 30, 10, 10, CombineModeXor);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipCreateSolidFill((ARGB)0xff0000ff, (GpSolidFill**)&brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipFillRectangle(graphics, brush, 30, 30, 20, 10);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteBrush(brush);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
check_metafile(metafile, clipping_records, "clipping metafile", dst_points, &frame, UnitPixel);
|
|
|
|
sync_metafile(&metafile, "clipping.emf");
|
|
|
|
stat = GdipCreateBitmapFromScan0(100, 100, 0, PixelFormat32bppARGB, NULL, &bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipGetImageGraphicsContext((GpImage*)bitmap, &graphics);
|
|
expect(Ok, stat);
|
|
|
|
play_metafile(metafile, graphics, clipping_records, "clipping playback", dst_points, &frame, UnitPixel);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 80, 80, &color);
|
|
expect(Ok, stat);
|
|
expect(0, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 35, 35, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff000000, color);
|
|
|
|
stat = GdipBitmapGetPixel(bitmap, 45, 35, &color);
|
|
expect(Ok, stat);
|
|
expect(0xff0000ff, color);
|
|
|
|
stat = GdipDeleteGraphics(graphics);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)bitmap);
|
|
expect(Ok, stat);
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
expect(Ok, stat);
|
|
}
|
|
|
|
START_TEST(metafile)
|
|
{
|
|
struct GdiplusStartupInput gdiplusStartupInput;
|
|
ULONG_PTR gdiplusToken;
|
|
int myARGC;
|
|
char **myARGV;
|
|
|
|
gdiplusStartupInput.GdiplusVersion = 1;
|
|
gdiplusStartupInput.DebugEventCallback = NULL;
|
|
gdiplusStartupInput.SuppressBackgroundThread = 0;
|
|
gdiplusStartupInput.SuppressExternalCodecs = 0;
|
|
|
|
GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL);
|
|
|
|
myARGC = winetest_get_mainargs( &myARGV );
|
|
|
|
if (myARGC >= 3)
|
|
{
|
|
if (!strcmp(myARGV[2], "save"))
|
|
save_metafiles = TRUE;
|
|
else if (!strcmp(myARGV[2], "load"))
|
|
load_metafiles = TRUE;
|
|
}
|
|
|
|
test_empty();
|
|
test_getdc();
|
|
test_emfonly();
|
|
test_fillrect();
|
|
test_clear();
|
|
test_nullframerect();
|
|
test_pagetransform();
|
|
test_worldtransform();
|
|
test_converttoemfplus();
|
|
test_frameunit();
|
|
test_containers();
|
|
test_clipping();
|
|
|
|
GdiplusShutdown(gdiplusToken);
|
|
}
|