/* Unit test suite for static controls. * * Copyright 2007 Google (Mikolaj Zalewski) * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include "windef.h" #include "winbase.h" #include "wingdi.h" #define OEMRESOURCE #include "winuser.h" #include "commctrl.h" #include "resources.h" #include "uxtheme.h" #include "wine/test.h" #include "v6util.h" #define TODO_COUNT 1 #define CTRL_ID 1995 static HWND hMainWnd; static int g_nReceivedColorStatic; static HRESULT (WINAPI *pEnableThemeDialogTexture)(HWND, DWORD); static HTHEME (WINAPI *pGetWindowTheme)(HWND); static BOOL (WINAPI *pIsThemeActive)(void); static BOOL (WINAPI *pIsThemeDialogTextureEnabled)(HWND); /* try to make sure pending X events have been processed before continuing */ static void flush_events(void) { MSG msg; int diff = 200; int min_timeout = 100; DWORD time = GetTickCount() + diff; while (diff > 0) { if (MsgWaitForMultipleObjects( 0, NULL, FALSE, min_timeout, QS_ALLINPUT ) == WAIT_TIMEOUT) break; while (PeekMessageA( &msg, 0, 0, 0, PM_REMOVE )) DispatchMessageA( &msg ); diff = time - GetTickCount(); } } static HWND create_static(DWORD style) { return CreateWindowA(WC_STATICA, "Test", WS_VISIBLE|WS_CHILD|style, 5, 5, 100, 100, hMainWnd, (HMENU)CTRL_ID, NULL, 0); } static LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam) { switch (msg) { case WM_CTLCOLORSTATIC: { HDC hdc = (HDC)wparam; HRGN hrgn = CreateRectRgn(0, 0, 1, 1); ok(GetClipRgn(hdc, hrgn) == 1, "Static controls during a WM_CTLCOLORSTATIC must have a clipping region\n"); DeleteObject(hrgn); g_nReceivedColorStatic++; return (LRESULT) GetStockObject(BLACK_BRUSH); } break; } return DefWindowProcA(hwnd, msg, wparam, lparam); } static void test_updates(int style, int flags) { HWND hStatic = create_static(style); RECT r1 = {20, 20, 30, 30}; int exp; flush_events(); g_nReceivedColorStatic = 0; /* during each update parent WndProc will test the WM_CTLCOLORSTATIC message */ InvalidateRect(hMainWnd, NULL, FALSE); UpdateWindow(hMainWnd); InvalidateRect(hMainWnd, &r1, FALSE); UpdateWindow(hMainWnd); InvalidateRect(hStatic, &r1, FALSE); UpdateWindow(hStatic); InvalidateRect(hStatic, NULL, FALSE); UpdateWindow(hStatic); if ((style & SS_TYPEMASK) == SS_BITMAP) { HDC hdc = GetDC(hStatic); COLORREF colour = GetPixel(hdc, 10, 10); todo_wine ok(colour == 0, "Unexpected pixel color.\n"); ReleaseDC(hStatic, hdc); } if (style != SS_ETCHEDHORZ && style != SS_ETCHEDVERT) exp = 4; else exp = 1; /* SS_ETCHED* seems to send WM_CTLCOLORSTATIC only sometimes */ if (flags & TODO_COUNT) todo_wine ok(g_nReceivedColorStatic == exp, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); else if ((style & SS_TYPEMASK) == SS_ICON || (style & SS_TYPEMASK) == SS_BITMAP) ok(g_nReceivedColorStatic == exp, "Unexpected %u got %u\n", exp, g_nReceivedColorStatic); else ok(g_nReceivedColorStatic == exp, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); DestroyWindow(hStatic); } static void test_set_text(void) { HWND hStatic = create_static(SS_SIMPLE); char buffA[10]; GetWindowTextA(hStatic, buffA, sizeof(buffA)); ok(!strcmp(buffA, "Test"), "got wrong text %s\n", buffA); SetWindowTextA(hStatic, NULL); GetWindowTextA(hStatic, buffA, sizeof(buffA)); ok(buffA[0] == 0, "got wrong text %s\n", buffA); DestroyWindow(hStatic); } static void remove_alpha(HBITMAP hbitmap) { BITMAP bm; BYTE *ptr; int i; GetObjectW(hbitmap, sizeof(bm), &bm); ok(bm.bmBits != NULL, "bmBits is NULL\n"); for (i = 0, ptr = bm.bmBits; i < bm.bmWidth * bm.bmHeight; i++, ptr += 4) ptr[3] = 0; } static void test_image(HBITMAP image, BOOL is_dib, BOOL is_premult, BOOL is_alpha) { BITMAP bm; HDC hdc; BITMAPINFO info; BYTE bits[4]; GetObjectW(image, sizeof(bm), &bm); ok(bm.bmWidth == 1, "got %d\n", bm.bmWidth); ok(bm.bmHeight == 1, "got %d\n", bm.bmHeight); ok(bm.bmBitsPixel == 32, "got %d\n", bm.bmBitsPixel); if (is_dib) { ok(bm.bmBits != NULL, "bmBits is NULL\n"); memcpy(bits, bm.bmBits, 4); if (is_premult) todo_wine ok(bits[0] == 0x05 && bits[1] == 0x09 && bits[2] == 0x0e && bits[3] == 0x44, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); else if (is_alpha) ok(bits[0] == 0x11 && bits[1] == 0x22 && bits[2] == 0x33 && bits[3] == 0x44, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); else ok(bits[0] == 0x11 && bits[1] == 0x22 && bits[2] == 0x33 && bits[3] == 0, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); } else ok(bm.bmBits == NULL, "bmBits is not NULL\n"); info.bmiHeader.biSize = sizeof(BITMAPINFOHEADER); info.bmiHeader.biWidth = bm.bmWidth; info.bmiHeader.biHeight = bm.bmHeight; info.bmiHeader.biPlanes = 1; info.bmiHeader.biBitCount = 32; info.bmiHeader.biCompression = BI_RGB; info.bmiHeader.biSizeImage = 4; info.bmiHeader.biXPelsPerMeter = 0; info.bmiHeader.biYPelsPerMeter = 0; info.bmiHeader.biClrUsed = 0; info.bmiHeader.biClrImportant = 0; hdc = CreateCompatibleDC(0); GetDIBits(hdc, image, 0, bm.bmHeight, bits, &info, DIB_RGB_COLORS); DeleteDC(hdc); if (is_premult) todo_wine ok(bits[0] == 0x05 && bits[1] == 0x09 && bits[2] == 0x0e && bits[3] == 0x44, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); else if (is_alpha) ok(bits[0] == 0x11 && bits[1] == 0x22 && bits[2] == 0x33 && bits[3] == 0x44, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); else ok(bits[0] == 0x11 && bits[1] == 0x22 && bits[2] == 0x33 && bits[3] == 0, "bits: %02x %02x %02x %02x\n", bits[0], bits[1], bits[2], bits[3]); } static void test_set_image(void) { HWND hwnd = create_static(SS_BITMAP); HBITMAP bmp1, bmp2, image; image = LoadImageW(GetModuleHandleW(NULL), MAKEINTRESOURCEW(IDB_BITMAP_1x1_32BPP), IMAGE_BITMAP, 0, 0, 0); ok(image != NULL, "LoadImage failed\n"); test_image(image, FALSE, FALSE, TRUE); bmp1 = (HBITMAP)SendMessageW(hwnd, STM_GETIMAGE, IMAGE_BITMAP, 0); ok(bmp1 == NULL, "got not NULL\n"); bmp1 = (HBITMAP)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)image); ok(bmp1 == NULL, "got not NULL\n"); bmp1 = (HBITMAP)SendMessageW(hwnd, STM_GETIMAGE, IMAGE_BITMAP, 0); ok(bmp1 != NULL, "got NULL\n"); ok(bmp1 != image, "bmp == image\n"); test_image(bmp1, TRUE, TRUE, TRUE); bmp2 = (HBITMAP)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)image); ok(bmp2 != NULL, "got NULL\n"); ok(bmp2 != image, "bmp == image\n"); ok(bmp1 == bmp2, "bmp1 != bmp2\n"); test_image(bmp2, TRUE, TRUE, TRUE); DeleteObject(image); image = LoadImageW(GetModuleHandleW(NULL), MAKEINTRESOURCEW(IDB_BITMAP_1x1_32BPP), IMAGE_BITMAP, 0, 0, LR_CREATEDIBSECTION); ok(image != NULL, "LoadImage failed\n"); test_image(image, TRUE, FALSE, TRUE); remove_alpha(image); test_image(image, TRUE, FALSE, FALSE); bmp1 = (HBITMAP)SendMessageW(hwnd, STM_GETIMAGE, IMAGE_BITMAP, 0); ok(bmp1 != NULL, "got NULL\n"); ok(bmp1 != image, "bmp == image\n"); test_image(bmp1, TRUE, TRUE, TRUE); bmp2 = (HBITMAP)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)image); ok(bmp2 != NULL, "got NULL\n"); ok(bmp2 != image, "bmp == image\n"); ok(bmp1 == bmp2, "bmp1 != bmp2\n"); test_image(bmp1, TRUE, TRUE, FALSE); bmp2 = (HBITMAP)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)image); ok(bmp2 != NULL, "got NULL\n"); ok(bmp2 == image, "bmp1 != image\n"); test_image(bmp2, TRUE, FALSE, FALSE); DestroyWindow(hwnd); test_image(image, TRUE, FALSE, FALSE); DeleteObject(image); } static void test_STM_SETIMAGE(void) { DWORD type; HWND hwnd; HICON icon, old_image; HBITMAP bmp; HENHMETAFILE emf; HDC dc; icon = LoadIconW(0, (LPCWSTR)IDI_APPLICATION); bmp = LoadBitmapW(0, (LPCWSTR)OBM_CLOSE); dc = CreateEnhMetaFileW(0, NULL, NULL, NULL); LineTo(dc, 1, 1); emf = CloseEnhMetaFile(dc); DeleteDC(dc); for (type = SS_LEFT; type < SS_ETCHEDFRAME; type++) { winetest_push_context("%u", type); hwnd = create_static(type); ok(hwnd != 0, "failed to create static type %#x\n", type); /* set icon */ g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_ICON, (LPARAM)icon); ok(!old_image, "got %p\n", old_image); if (type == SS_ICON) ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); else ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_ICON, (LPARAM)icon); if (type == SS_ICON) { ok(old_image != 0, "got %p\n", old_image); ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } else { ok(!old_image, "got %p\n", old_image); ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETICON, (WPARAM)icon, 0); if (type == SS_ICON) { ok(old_image != 0, "got %p\n", old_image); ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } else { ok(!old_image, "got %p\n", old_image); ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } /* set bitmap */ g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)bmp); ok(!old_image, "got %p\n", old_image); if (type == SS_BITMAP) ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); else ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_BITMAP, (LPARAM)bmp); if (type == SS_BITMAP) { ok(old_image != 0, "got %p\n", old_image); ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } else { ok(!old_image, "got %p\n", old_image); ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } /* set EMF */ g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_ENHMETAFILE, (LPARAM)emf); ok(!old_image, "got %p\n", old_image); if (type == SS_ENHMETAFILE) ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); else ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); g_nReceivedColorStatic = 0; old_image = (HICON)SendMessageW(hwnd, STM_SETIMAGE, IMAGE_ENHMETAFILE, (LPARAM)emf); if (type == SS_ENHMETAFILE) { ok(old_image != 0, "got %p\n", old_image); ok(g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } else { ok(!old_image, "got %p\n", old_image); ok(!g_nReceivedColorStatic, "Unexpected WM_CTLCOLORSTATIC value %d\n", g_nReceivedColorStatic); } DestroyWindow(hwnd); winetest_pop_context(); } DestroyIcon(icon); DeleteObject(bmp); DeleteEnhMetaFile(emf); } static INT_PTR CALLBACK test_WM_CTLCOLORSTATIC_proc(HWND hwnd, UINT msg, WPARAM wp, LPARAM lp) { static HWND child; switch (msg) { case WM_INITDIALOG: child = CreateWindowA(WC_STATICA, "child", WS_CHILD | WS_VISIBLE, 1, 2, 50, 50, hwnd, (HMENU)100, 0, NULL); ok(child != NULL, "CreateWindowA failed, error %d.\n", GetLastError()); return FALSE; case WM_CLOSE: DestroyWindow(child); return TRUE; default: return FALSE; } } static void test_WM_CTLCOLORSTATIC(void) { HWND parent, dialog, child; COLORREF color, old_color; HDC child_hdc, dialog_hdc; BOOL ret, todo = FALSE; int mode, old_mode; HBRUSH brush; HRESULT hr; POINT org; struct { DLGTEMPLATE tmplate; WORD menu; WORD class; WORD title; } temp = {{0}}; parent = CreateWindowA(WC_STATICA, "parent", WS_POPUP | WS_VISIBLE, 100, 100, 200, 200, 0, 0, 0, NULL); ok(parent != NULL, "CreateWindowA failed, error %d.\n", GetLastError()); temp.tmplate.style = WS_CHILD | WS_VISIBLE; temp.tmplate.cx = 80; temp.tmplate.cy = 80; dialog = CreateDialogIndirectParamA(NULL, &temp.tmplate, parent, test_WM_CTLCOLORSTATIC_proc, 0); ok(dialog != NULL, "CreateDialogIndirectParamA failed, error %d.\n", GetLastError()); child = GetDlgItem(dialog, 100); ok(child != NULL, "Failed to get child static control, error %d.\n", GetLastError()); todo = !!pGetWindowTheme(dialog); dialog_hdc = GetDC(dialog); child_hdc = GetDC(child); PatBlt(dialog_hdc, 0, 0, 80, 80, BLACKNESS); old_mode = SetBkMode(child_hdc, OPAQUE); ok(old_mode != 0, "SetBkMode failed.\n"); old_color = SetBkColor(child_hdc, 0xaa5511); ok(old_color != CLR_INVALID, "SetBkColor failed.\n"); ret = pIsThemeDialogTextureEnabled(dialog); ok(ret, "Expected theme dialog texture supported.\n"); todo_wine_if(todo) ok(pGetWindowTheme(dialog) == NULL, "Expected NULL theme handle.\n"); brush = (HBRUSH)SendMessageW(dialog, WM_CTLCOLORSTATIC, (WPARAM)child_hdc, (LPARAM)child); todo_wine_if(todo) ok(brush == GetSysColorBrush(COLOR_BTNFACE), "Expected brush %p, got %p.\n", GetSysColorBrush(COLOR_BTNFACE), brush); color = SetBkColor(child_hdc, old_color); ok(color == GetSysColor(COLOR_BTNFACE), "Expected background color %#x, got %#x.\n", GetSysColor(COLOR_BTNFACE), color); mode = SetBkMode(child_hdc, old_mode); todo_wine_if(todo) ok(mode == OPAQUE, "Expected mode %#x, got %#x.\n", OPAQUE, mode); color = GetPixel(dialog_hdc, 40, 40); ok(color == 0, "Expected pixel %#x, got %#x.\n", 0, color); /* Test that EnableThemeDialogTexture() doesn't make WM_CTLCOLORSTATIC return a different brush */ hr = pEnableThemeDialogTexture(dialog, ETDT_DISABLE); ok(hr == S_OK, "EnableThemeDialogTexture failed, hr %#x.\n", hr); ret = pIsThemeDialogTextureEnabled(dialog); ok(!ret, "Expected theme dialog texture disabled.\n"); hr = pEnableThemeDialogTexture(dialog, ETDT_ENABLETAB); ok(hr == S_OK, "EnableThemeDialogTexture failed, hr %#x.\n", hr); ret = pIsThemeDialogTextureEnabled(dialog); ok(ret, "Expected theme dialog texture enabled.\n"); brush = (HBRUSH)SendMessageW(dialog, WM_CTLCOLORSTATIC, (WPARAM)child_hdc, (LPARAM)child); todo_wine_if(pIsThemeActive()) ok(brush == GetSysColorBrush(COLOR_BTNFACE), "Expected brush %p, got %p.\n", GetSysColorBrush(COLOR_BTNFACE), brush); /* Test that WM_CTLCOLORSTATIC doesn't change brush origin */ ret = GetBrushOrgEx(child_hdc, &org); ok(ret, "GetBrushOrgEx failed, error %u.\n", GetLastError()); ok(org.x == 0 && org.y == 0, "Expected (0,0), got %s.\n", wine_dbgstr_point(&org)); ReleaseDC(child, child_hdc); ReleaseDC(dialog, dialog_hdc); EndDialog(dialog, 0); DestroyWindow(parent); } static void init_functions(void) { HMODULE uxtheme = LoadLibraryA("uxtheme.dll"); #define X(f) p##f = (void *)GetProcAddress(uxtheme, #f); X(EnableThemeDialogTexture) X(GetWindowTheme) X(IsThemeActive) X(IsThemeDialogTextureEnabled) #undef X } START_TEST(static) { static const char classname[] = "testclass"; WNDCLASSEXA wndclass; ULONG_PTR ctx_cookie; HANDLE hCtx; if (!load_v6_module(&ctx_cookie, &hCtx)) return; init_functions(); wndclass.cbSize = sizeof(wndclass); wndclass.style = CS_HREDRAW | CS_VREDRAW; wndclass.lpfnWndProc = WndProc; wndclass.cbClsExtra = 0; wndclass.cbWndExtra = 0; wndclass.hInstance = GetModuleHandleA(NULL); wndclass.hIcon = LoadIconA(NULL, (LPCSTR)IDI_APPLICATION); wndclass.hIconSm = LoadIconA(NULL, (LPCSTR)IDI_APPLICATION); wndclass.hCursor = LoadCursorA(NULL, (LPCSTR)IDC_ARROW); wndclass.hbrBackground = GetStockObject(WHITE_BRUSH); wndclass.lpszClassName = classname; wndclass.lpszMenuName = NULL; RegisterClassExA(&wndclass); hMainWnd = CreateWindowA(classname, "Test", WS_OVERLAPPEDWINDOW, 0, 0, 500, 500, NULL, NULL, GetModuleHandleA(NULL), NULL); ShowWindow(hMainWnd, SW_SHOW); test_updates(0, 0); test_updates(SS_SIMPLE, 0); test_updates(SS_ICON, 0); test_updates(SS_BITMAP, 0); test_updates(SS_BITMAP | SS_CENTERIMAGE, 0); test_updates(SS_BLACKRECT, TODO_COUNT); test_updates(SS_WHITERECT, TODO_COUNT); test_updates(SS_ETCHEDHORZ, TODO_COUNT); test_updates(SS_ETCHEDVERT, TODO_COUNT); test_set_text(); test_set_image(); test_STM_SETIMAGE(); test_WM_CTLCOLORSTATIC(); DestroyWindow(hMainWnd); unload_v6_module(ctx_cookie, hCtx); }