2004-11-24 19:09:20 +01:00
|
|
|
/*
|
|
|
|
* Unit test suite for brushes
|
|
|
|
*
|
|
|
|
* Copyright 2004 Kevin Koltzau
|
|
|
|
*
|
|
|
|
* 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
|
2006-05-18 14:49:52 +02:00
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
2004-11-24 19:09:20 +01:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "wingdi.h"
|
|
|
|
|
|
|
|
#include "wine/test.h"
|
|
|
|
|
|
|
|
typedef struct _STOCK_BRUSH {
|
|
|
|
COLORREF color;
|
|
|
|
int stockobj;
|
2005-02-14 12:03:36 +01:00
|
|
|
const char *name;
|
2004-11-24 19:09:20 +01:00
|
|
|
} STOCK_BRUSH;
|
|
|
|
|
2005-06-20 16:18:03 +02:00
|
|
|
static void test_solidbrush(void)
|
2004-11-24 19:09:20 +01:00
|
|
|
{
|
|
|
|
static const STOCK_BRUSH stock[] = {
|
|
|
|
{RGB(255,255,255), WHITE_BRUSH, "white"},
|
|
|
|
{RGB(192,192,192), LTGRAY_BRUSH, "ltgray"},
|
|
|
|
{RGB(128,128,128), GRAY_BRUSH, "gray"},
|
|
|
|
{RGB(0,0,0), BLACK_BRUSH, "black"},
|
|
|
|
{RGB(0,0,255), -1, "blue"}
|
|
|
|
};
|
|
|
|
HBRUSH solidBrush;
|
|
|
|
HBRUSH stockBrush;
|
|
|
|
LOGBRUSH br;
|
|
|
|
size_t i;
|
2004-12-27 18:26:37 +01:00
|
|
|
INT ret;
|
2004-11-24 19:09:20 +01:00
|
|
|
|
|
|
|
for(i=0; i<sizeof(stock)/sizeof(stock[0]); i++) {
|
|
|
|
solidBrush = CreateSolidBrush(stock[i].color);
|
|
|
|
|
|
|
|
if(stock[i].stockobj != -1) {
|
2008-10-25 23:39:38 +02:00
|
|
|
stockBrush = GetStockObject(stock[i].stockobj);
|
2008-09-28 18:15:49 +02:00
|
|
|
ok(stockBrush!=solidBrush ||
|
|
|
|
broken(stockBrush==solidBrush), /* win9x does return stock object */
|
|
|
|
"Stock %s brush equals solid %s brush\n", stock[i].name, stock[i].name);
|
2004-11-24 19:09:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
stockBrush = NULL;
|
2006-07-13 23:06:17 +02:00
|
|
|
memset(&br, 0, sizeof(br));
|
2004-12-27 18:26:37 +01:00
|
|
|
ret = GetObject(solidBrush, sizeof(br), &br);
|
2006-10-12 22:56:45 +02:00
|
|
|
ok( ret !=0, "GetObject on solid %s brush failed, error=%d\n", stock[i].name, GetLastError());
|
2004-11-24 19:09:20 +01:00
|
|
|
ok(br.lbStyle==BS_SOLID, "%s brush has wrong style, got %d expected %d\n", stock[i].name, br.lbStyle, BS_SOLID);
|
2006-10-12 22:56:45 +02:00
|
|
|
ok(br.lbColor==stock[i].color, "%s brush has wrong color, got 0x%08x expected 0x%08x\n", stock[i].name, br.lbColor, stock[i].color);
|
2004-11-24 19:09:20 +01:00
|
|
|
|
|
|
|
if(stockBrush) {
|
|
|
|
/* Sanity check, make sure the colors being compared do in fact have a stock brush */
|
2004-12-27 18:26:37 +01:00
|
|
|
ret = GetObject(stockBrush, sizeof(br), &br);
|
2006-10-12 22:56:45 +02:00
|
|
|
ok( ret !=0, "GetObject on stock %s brush failed, error=%d\n", stock[i].name, GetLastError());
|
|
|
|
ok(br.lbColor==stock[i].color, "stock %s brush unexpected color, got 0x%08x expected 0x%08x\n", stock[i].name, br.lbColor, stock[i].color);
|
2004-11-24 19:09:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
DeleteObject(solidBrush);
|
2008-09-28 18:15:49 +02:00
|
|
|
ret = GetObject(solidBrush, sizeof(br), &br);
|
|
|
|
ok(ret==0 ||
|
|
|
|
broken(ret!=0), /* win9x */
|
|
|
|
"GetObject succeeded on a deleted %s brush\n", stock[i].name);
|
2004-11-24 19:09:20 +01:00
|
|
|
}
|
|
|
|
}
|
2011-11-01 14:07:05 +01:00
|
|
|
|
|
|
|
static void test_pattern_brush(void)
|
|
|
|
{
|
|
|
|
char buffer[sizeof(BITMAPINFOHEADER) + 2 * sizeof(RGBQUAD) + 32 * 32 / 8];
|
|
|
|
BITMAPINFO *info = (BITMAPINFO *)buffer;
|
|
|
|
HBRUSH brush;
|
|
|
|
HBITMAP bitmap;
|
|
|
|
LOGBRUSH br;
|
|
|
|
INT ret;
|
|
|
|
void *bits;
|
|
|
|
DIBSECTION dib;
|
|
|
|
HGLOBAL mem;
|
|
|
|
|
|
|
|
bitmap = CreateBitmap( 20, 20, 1, 1, NULL );
|
|
|
|
ok( bitmap != NULL, "CreateBitmap failed\n" );
|
|
|
|
brush = CreatePatternBrush( bitmap );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_PATTERN, "wrong style %u\n", br.lbStyle );
|
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( (HBITMAP)br.lbHatch == bitmap, "wrong handle %p/%p\n", (HBITMAP)br.lbHatch, bitmap );
|
2011-11-01 14:07:05 +01:00
|
|
|
DeleteObject( brush );
|
|
|
|
|
|
|
|
br.lbStyle = BS_PATTERN8X8;
|
|
|
|
br.lbColor = 0x12345;
|
|
|
|
br.lbHatch = (ULONG_PTR)bitmap;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_PATTERN, "wrong style %u\n", br.lbStyle );
|
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( (HBITMAP)br.lbHatch == bitmap, "wrong handle %p/%p\n", (HBITMAP)br.lbHatch, bitmap );
|
2011-11-01 14:07:05 +01:00
|
|
|
ret = GetObjectW( bitmap, sizeof(dib), &dib );
|
|
|
|
ok( ret == sizeof(dib.dsBm), "wrong size %u\n", ret );
|
|
|
|
DeleteObject( bitmap );
|
|
|
|
ret = GetObjectW( bitmap, sizeof(dib), &dib );
|
|
|
|
ok( ret == 0, "wrong size %u\n", ret );
|
|
|
|
DeleteObject( brush );
|
|
|
|
|
|
|
|
memset( info, 0, sizeof(buffer) );
|
|
|
|
info->bmiHeader.biSize = sizeof(info->bmiHeader);
|
|
|
|
info->bmiHeader.biHeight = 32;
|
|
|
|
info->bmiHeader.biWidth = 32;
|
|
|
|
info->bmiHeader.biBitCount = 1;
|
|
|
|
info->bmiHeader.biPlanes = 1;
|
|
|
|
info->bmiHeader.biCompression = BI_RGB;
|
|
|
|
bitmap = CreateDIBSection( 0, info, DIB_RGB_COLORS, (void**)&bits, NULL, 0 );
|
|
|
|
ok( bitmap != NULL, "CreateDIBSection failed\n" );
|
|
|
|
|
|
|
|
/* MSDN says a DIB section is not allowed, but it works fine */
|
|
|
|
brush = CreatePatternBrush( bitmap );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_PATTERN, "wrong style %u\n", br.lbStyle );
|
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( (HBITMAP)br.lbHatch == bitmap, "wrong handle %p/%p\n", (HBITMAP)br.lbHatch, bitmap );
|
2011-11-01 14:07:05 +01:00
|
|
|
ret = GetObjectW( bitmap, sizeof(dib), &dib );
|
|
|
|
ok( ret == sizeof(dib), "wrong size %u\n", ret );
|
|
|
|
DeleteObject( brush );
|
|
|
|
DeleteObject( bitmap );
|
|
|
|
|
|
|
|
brush = CreateDIBPatternBrushPt( info, DIB_RGB_COLORS );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_DIBPATTERN, "wrong style %u\n", br.lbStyle );
|
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( (BITMAPINFO *)br.lbHatch == info || broken(!br.lbHatch), /* nt4 */
|
2011-11-01 14:07:05 +01:00
|
|
|
"wrong handle %p/%p\n", (BITMAPINFO *)br.lbHatch, info );
|
|
|
|
DeleteObject( brush );
|
|
|
|
|
|
|
|
br.lbStyle = BS_DIBPATTERNPT;
|
|
|
|
br.lbColor = DIB_PAL_COLORS;
|
|
|
|
br.lbHatch = (ULONG_PTR)info;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_DIBPATTERN, "wrong style %u\n", br.lbStyle );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
|
|
|
ok( (BITMAPINFO *)br.lbHatch == info || broken(!br.lbHatch), /* nt4 */
|
2011-11-01 14:07:05 +01:00
|
|
|
"wrong handle %p/%p\n", (BITMAPINFO *)br.lbHatch, info );
|
|
|
|
|
|
|
|
mem = GlobalAlloc( GMEM_MOVEABLE, sizeof(buffer) );
|
|
|
|
memcpy( GlobalLock( mem ), buffer, sizeof(buffer) );
|
|
|
|
|
|
|
|
br.lbStyle = BS_DIBPATTERN;
|
|
|
|
br.lbColor = DIB_PAL_COLORS;
|
|
|
|
br.lbHatch = (ULONG_PTR)mem;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( brush != NULL, "CreatePatternBrush failed\n" );
|
|
|
|
memset( &br, 0x55, sizeof(br) );
|
|
|
|
ret = GetObjectW( brush, sizeof(br), &br );
|
|
|
|
ok( ret == sizeof(br), "wrong size %u\n", ret );
|
|
|
|
ok( br.lbStyle == BS_DIBPATTERN, "wrong style %u\n", br.lbStyle );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( br.lbColor == 0, "wrong color %u\n", br.lbColor );
|
2011-11-01 14:07:05 +01:00
|
|
|
ok( (HGLOBAL)br.lbHatch != mem, "wrong handle %p/%p\n", (HGLOBAL)br.lbHatch, mem );
|
|
|
|
bits = GlobalLock( mem );
|
2011-11-03 20:18:59 +01:00
|
|
|
ok( (HGLOBAL)br.lbHatch == bits || broken(!br.lbHatch), /* nt4 */
|
2011-11-01 14:07:05 +01:00
|
|
|
"wrong handle %p/%p\n", (HGLOBAL)br.lbHatch, bits );
|
|
|
|
ret = GlobalFlags( mem );
|
|
|
|
ok( ret == 2, "wrong flags %x\n", ret );
|
|
|
|
DeleteObject( brush );
|
|
|
|
ret = GlobalFlags( mem );
|
|
|
|
ok( ret == 2, "wrong flags %x\n", ret );
|
|
|
|
|
2011-11-02 12:22:44 +01:00
|
|
|
info->bmiHeader.biBitCount = 8;
|
|
|
|
info->bmiHeader.biCompression = BI_RLE8;
|
|
|
|
brush = CreateDIBPatternBrushPt( info, DIB_RGB_COLORS );
|
|
|
|
ok( !brush, "CreateDIBPatternBrushPt succeeded\n" );
|
|
|
|
|
|
|
|
info->bmiHeader.biBitCount = 4;
|
|
|
|
info->bmiHeader.biCompression = BI_RLE4;
|
|
|
|
brush = CreateDIBPatternBrushPt( info, DIB_RGB_COLORS );
|
|
|
|
ok( !brush, "CreateDIBPatternBrushPt succeeded\n" );
|
|
|
|
|
2011-11-01 14:07:05 +01:00
|
|
|
br.lbStyle = BS_DIBPATTERN8X8;
|
|
|
|
br.lbColor = DIB_RGB_COLORS;
|
|
|
|
br.lbHatch = (ULONG_PTR)mem;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( !brush, "CreatePatternBrush succeeded\n" );
|
|
|
|
|
|
|
|
br.lbStyle = BS_MONOPATTERN;
|
|
|
|
br.lbColor = DIB_RGB_COLORS;
|
|
|
|
br.lbHatch = (ULONG_PTR)mem;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( !brush, "CreatePatternBrush succeeded\n" );
|
|
|
|
|
|
|
|
br.lbStyle = BS_INDEXED;
|
|
|
|
br.lbColor = DIB_RGB_COLORS;
|
|
|
|
br.lbHatch = (ULONG_PTR)mem;
|
|
|
|
brush = CreateBrushIndirect( &br );
|
|
|
|
ok( !brush, "CreatePatternBrush succeeded\n" );
|
|
|
|
|
|
|
|
GlobalFree( mem );
|
|
|
|
}
|
|
|
|
|
2011-11-02 11:33:00 +01:00
|
|
|
static void test_palette_brush(void)
|
|
|
|
{
|
|
|
|
char buffer[sizeof(BITMAPINFOHEADER) + 256 * sizeof(RGBQUAD) + 16 * 16];
|
|
|
|
BITMAPINFO *info = (BITMAPINFO *)buffer;
|
|
|
|
WORD *indices = (WORD *)info->bmiColors;
|
|
|
|
char pal_buffer[sizeof(LOGPALETTE) + 256 * sizeof(PALETTEENTRY)];
|
|
|
|
LOGPALETTE *pal = (LOGPALETTE *)pal_buffer;
|
|
|
|
HDC hdc = CreateCompatibleDC( 0 );
|
|
|
|
DWORD *dib_bits;
|
|
|
|
HBITMAP dib;
|
|
|
|
HBRUSH brush;
|
|
|
|
int i;
|
|
|
|
HPALETTE palette, palette2;
|
|
|
|
|
|
|
|
memset( info, 0, sizeof(*info) );
|
|
|
|
info->bmiHeader.biSize = sizeof(info->bmiHeader);
|
|
|
|
info->bmiHeader.biWidth = 16;
|
|
|
|
info->bmiHeader.biHeight = 16;
|
|
|
|
info->bmiHeader.biPlanes = 1;
|
|
|
|
info->bmiHeader.biBitCount = 32;
|
|
|
|
info->bmiHeader.biCompression = BI_RGB;
|
|
|
|
dib = CreateDIBSection( NULL, info, DIB_RGB_COLORS, (void**)&dib_bits, NULL, 0 );
|
|
|
|
ok( dib != NULL, "CreateDIBSection failed\n" );
|
|
|
|
|
|
|
|
info->bmiHeader.biBitCount = 8;
|
|
|
|
for (i = 0; i < 256; i++) indices[i] = 255 - i;
|
|
|
|
for (i = 0; i < 256; i++) ((BYTE *)(indices + 256))[i] = i;
|
|
|
|
brush = CreateDIBPatternBrushPt( info, DIB_PAL_COLORS );
|
|
|
|
ok( brush != NULL, "CreateDIBPatternBrushPt failed\n" );
|
|
|
|
|
|
|
|
pal->palVersion = 0x300;
|
|
|
|
pal->palNumEntries = 256;
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
pal->palPalEntry[i].peRed = i * 2;
|
|
|
|
pal->palPalEntry[i].peGreen = i * 2;
|
|
|
|
pal->palPalEntry[i].peBlue = i * 2;
|
|
|
|
pal->palPalEntry[i].peFlags = 0;
|
|
|
|
}
|
|
|
|
palette = CreatePalette( pal );
|
|
|
|
|
|
|
|
ok( SelectObject( hdc, dib ) != NULL, "SelectObject failed\n" );
|
|
|
|
ok( SelectPalette( hdc, palette, 0 ) != NULL, "SelectPalette failed\n" );
|
|
|
|
ok( SelectObject( hdc, brush ) != NULL, "SelectObject failed\n" );
|
|
|
|
memset( dib_bits, 0xaa, 16 * 16 * 4 );
|
|
|
|
PatBlt( hdc, 0, 0, 16, 16, PATCOPY );
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
DWORD expect = (pal->palPalEntry[255 - i].peRed << 16 |
|
|
|
|
pal->palPalEntry[255 - i].peGreen << 8 |
|
|
|
|
pal->palPalEntry[255 - i].peBlue);
|
|
|
|
ok( dib_bits[i] == expect, "wrong bits %x/%x at %u,%u\n", dib_bits[i], expect, i % 16, i / 16 );
|
|
|
|
}
|
|
|
|
|
|
|
|
for (i = 0; i < 256; i++) pal->palPalEntry[i].peRed = i * 3;
|
|
|
|
palette2 = CreatePalette( pal );
|
|
|
|
ok( SelectPalette( hdc, palette2, 0 ) != NULL, "SelectPalette failed\n" );
|
|
|
|
memset( dib_bits, 0xaa, 16 * 16 * 4 );
|
|
|
|
PatBlt( hdc, 0, 0, 16, 16, PATCOPY );
|
|
|
|
for (i = 0; i < 256; i++)
|
|
|
|
{
|
|
|
|
DWORD expect = (pal->palPalEntry[255 - i].peRed << 16 |
|
|
|
|
pal->palPalEntry[255 - i].peGreen << 8 |
|
|
|
|
pal->palPalEntry[255 - i].peBlue);
|
2011-11-09 05:50:27 +01:00
|
|
|
ok( dib_bits[i] == expect, "wrong bits %x/%x at %u,%u\n", dib_bits[i], expect, i % 16, i / 16 );
|
2011-11-02 11:33:00 +01:00
|
|
|
}
|
|
|
|
DeleteDC( hdc );
|
|
|
|
DeleteObject( dib );
|
|
|
|
DeleteObject( brush );
|
|
|
|
DeleteObject( palette );
|
|
|
|
DeleteObject( palette2 );
|
|
|
|
}
|
|
|
|
|
2004-11-24 19:09:20 +01:00
|
|
|
START_TEST(brush)
|
|
|
|
{
|
|
|
|
test_solidbrush();
|
2011-11-01 14:07:05 +01:00
|
|
|
test_pattern_brush();
|
2011-11-02 11:33:00 +01:00
|
|
|
test_palette_brush();
|
2004-11-24 19:09:20 +01:00
|
|
|
}
|