2011-01-27 00:23:44 +01:00
|
|
|
/*
|
|
|
|
* 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 "windows.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include "gdiplus.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
|
|
|
|
#define expect(expected, got) ok(got == expected, "Expected %.8x, got %.8x\n", expected, got)
|
|
|
|
|
2011-01-27 01:50:10 +01:00
|
|
|
typedef struct emfplus_record
|
|
|
|
{
|
|
|
|
ULONG todo;
|
|
|
|
ULONG record_type;
|
|
|
|
} emfplus_record;
|
|
|
|
|
|
|
|
typedef struct emfplus_check_state
|
|
|
|
{
|
|
|
|
const char *desc;
|
|
|
|
int count;
|
|
|
|
const struct emfplus_record *expected;
|
|
|
|
} emfplus_check_state;
|
|
|
|
|
|
|
|
static void check_record(int count, const char *desc, const struct emfplus_record *expected, const struct emfplus_record *actual)
|
|
|
|
{
|
2011-05-22 01:32:23 +02:00
|
|
|
if (expected->todo)
|
|
|
|
todo_wine 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);
|
|
|
|
else
|
|
|
|
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);
|
2011-01-27 01:50:10 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
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 = 0;
|
|
|
|
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 = 0;
|
|
|
|
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);
|
|
|
|
|
2011-05-22 01:32:23 +02:00
|
|
|
if (expected[state.count].todo)
|
|
|
|
todo_wine ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
|
|
|
else
|
|
|
|
ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
2011-01-27 01:50:10 +01:00
|
|
|
}
|
|
|
|
|
2011-05-22 00:22:12 +02:00
|
|
|
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 = 0;
|
|
|
|
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;
|
|
|
|
|
|
|
|
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);
|
2011-05-22 00:48:32 +02:00
|
|
|
expect(Ok, stat);
|
2011-05-22 00:22:12 +02:00
|
|
|
|
2011-05-22 01:32:23 +02:00
|
|
|
if (expected[state.count].todo)
|
|
|
|
todo_wine ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
|
|
|
else
|
|
|
|
ok(expected[state.count].record_type == 0, "%s: Got %i records, expecting more\n", desc, state.count);
|
2011-05-22 00:22:12 +02:00
|
|
|
|
|
|
|
GdipDeleteGraphics(graphics);
|
|
|
|
|
|
|
|
DeleteDC(hdc);
|
|
|
|
}
|
|
|
|
|
2011-01-27 01:50:10 +01:00
|
|
|
static const emfplus_record empty_records[] = {
|
2011-05-22 01:36:22 +02:00
|
|
|
{0, EMR_HEADER},
|
|
|
|
{0, EmfPlusRecordTypeHeader},
|
|
|
|
{0, EmfPlusRecordTypeEndOfFile},
|
|
|
|
{0, EMR_EOF},
|
2011-01-27 01:50:10 +01:00
|
|
|
{0}
|
|
|
|
};
|
|
|
|
|
2011-01-27 00:23:44 +01:00
|
|
|
static void test_empty(void)
|
|
|
|
{
|
|
|
|
GpStatus stat;
|
|
|
|
GpMetafile *metafile;
|
|
|
|
GpGraphics *graphics;
|
|
|
|
HDC hdc;
|
|
|
|
HENHMETAFILE hemf, dummy;
|
|
|
|
BOOL ret;
|
|
|
|
static const GpRectF frame = {0.0, 0.0, 100.0, 100.0};
|
2011-05-22 00:22:12 +02:00
|
|
|
static const GpPointF dst_points[3] = {{0.0,0.0},{100.0,0.0},{0.0,100.0}};
|
2011-01-27 00:23:44 +01:00
|
|
|
static const WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
|
|
|
|
|
|
hdc = CreateCompatibleDC(0);
|
|
|
|
|
2011-04-02 00:09:23 +02:00
|
|
|
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);
|
|
|
|
|
2011-01-27 00:23:44 +01:00
|
|
|
stat = GdipRecordMetafile(hdc, EmfTypeEmfPlusOnly, &frame, MetafileFrameUnitPixel, description, &metafile);
|
2011-04-02 00:09:23 +02:00
|
|
|
expect(Ok, stat);
|
2011-01-27 00:23:44 +01:00
|
|
|
|
|
|
|
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);
|
|
|
|
|
2011-05-22 00:22:12 +02:00
|
|
|
check_metafile(metafile, empty_records, "empty metafile", dst_points, &frame, UnitPixel);
|
|
|
|
|
2011-01-27 00:23:44 +01:00
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &hemf);
|
|
|
|
expect(Ok, stat);
|
|
|
|
|
|
|
|
stat = GdipGetHemfFromMetafile(metafile, &dummy);
|
|
|
|
expect(InvalidParameter, stat);
|
|
|
|
|
|
|
|
stat = GdipDisposeImage((GpImage*)metafile);
|
|
|
|
expect(Ok, stat);
|
|
|
|
|
2011-05-22 00:22:12 +02:00
|
|
|
check_emfplus(hemf, empty_records, "empty emf");
|
2011-01-27 01:50:10 +01:00
|
|
|
|
2011-01-27 00:23:44 +01:00
|
|
|
ret = DeleteEnhMetaFile(hemf);
|
|
|
|
ok(ret != 0, "Failed to delete enhmetafile %p\n", hemf);
|
|
|
|
}
|
|
|
|
|
2011-05-22 01:32:23 +02:00
|
|
|
static const emfplus_record getdc_records[] = {
|
|
|
|
{0, EMR_HEADER},
|
|
|
|
{0, EmfPlusRecordTypeHeader},
|
|
|
|
{1, EmfPlusRecordTypeGetDC},
|
|
|
|
{1, EMR_CREATEBRUSHINDIRECT},
|
|
|
|
{1, 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 WCHAR description[] = {'w','i','n','e','t','e','s','t',0};
|
|
|
|
HBRUSH hbrush, holdbrush;
|
|
|
|
|
|
|
|
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);
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2011-01-27 00:23:44 +01:00
|
|
|
START_TEST(metafile)
|
|
|
|
{
|
|
|
|
struct GdiplusStartupInput gdiplusStartupInput;
|
|
|
|
ULONG_PTR gdiplusToken;
|
|
|
|
|
|
|
|
gdiplusStartupInput.GdiplusVersion = 1;
|
|
|
|
gdiplusStartupInput.DebugEventCallback = NULL;
|
|
|
|
gdiplusStartupInput.SuppressBackgroundThread = 0;
|
|
|
|
gdiplusStartupInput.SuppressExternalCodecs = 0;
|
|
|
|
|
|
|
|
GdiplusStartup(&gdiplusToken, &gdiplusStartupInput, NULL);
|
|
|
|
|
|
|
|
test_empty();
|
2011-05-22 01:32:23 +02:00
|
|
|
test_getdc();
|
2011-01-27 00:23:44 +01:00
|
|
|
|
|
|
|
GdiplusShutdown(gdiplusToken);
|
|
|
|
}
|