/* Unit tests for toolbar. * * Copyright 2005 Krzysztof Foltman * Copyright 2007 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 "windef.h" #include "winbase.h" #include "wingdi.h" #include "winuser.h" #include "winnls.h" #include "winreg.h" #include "commctrl.h" #include "uxtheme.h" #include "resources.h" #include "wine/test.h" #include "v6util.h" #include "msg.h" #define PARENT_SEQ_INDEX 0 #define NUM_MSG_SEQUENCES 1 static HWND (WINAPI *pCreateToolbarEx)(HWND, DWORD, UINT, INT, HINSTANCE, UINT_PTR, const TBBUTTON *, INT, INT, INT, INT, INT, UINT); static BOOL (WINAPI *pImageList_Destroy)(HIMAGELIST); static INT (WINAPI *pImageList_GetImageCount)(HIMAGELIST); static BOOL (WINAPI *pImageList_GetIconSize)(HIMAGELIST, int *, int *); static HIMAGELIST (WINAPI *pImageList_LoadImageA)(HINSTANCE, LPCSTR, int, int, COLORREF, UINT, UINT); static HRESULT (WINAPI *pCloseThemeData)(HTHEME); static BOOL (WINAPI *pIsThemeActive)(VOID); static HTHEME (WINAPI *pGetWindowTheme)(HWND); static HTHEME (WINAPI *pOpenThemeData)(HWND, LPCWSTR); static BOOL is_theme_active; static struct msg_sequence *sequences[NUM_MSG_SEQUENCES]; static HWND hMainWnd; static BOOL g_fBlockHotItemChange; static BOOL g_fReceivedHotItemChange; static BOOL g_fExpectedHotItemOld; static BOOL g_fExpectedHotItemNew; static DWORD g_dwExpectedDispInfoMask; static BOOL g_ResetDispTextPtr; static const struct message ttgetdispinfo_parent_seq[] = { { WM_NOTIFY, sent|id, 0, 0, TBN_GETINFOTIPA }, { WM_NOTIFY, sent|id, 0, 0, TTN_GETDISPINFOA }, { 0 } }; static const struct message save_parent_seq[] = { { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, -1 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 0 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 1 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 2 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 3 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 4 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 5 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_SAVE, 6 }, { 0 } }; static const struct message restore_parent_seq[] = { { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, -1 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 0 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 1 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 2 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 3 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 4 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 5 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 6 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 7 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 8 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 9 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_RESTORE, 0xa }, { WM_NOTIFY, sent|id, 0, 0, TBN_BEGINADJUST }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 0 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 1 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 2 }, { WM_NOTIFY, sent|id|custdraw, 0, 0, TBN_GETBUTTONINFOA, 3 }, { WM_NOTIFY, sent|id, 0, 0, TBN_ENDADJUST }, { 0 } }; #define DEFINE_EXPECT(func) \ static BOOL expect_ ## func = FALSE, called_ ## func = FALSE #define CHECK_EXPECT2(func) \ do { \ ok(expect_ ##func, "unexpected call " #func "\n"); \ called_ ## func = TRUE; \ }while(0) #define CHECK_CALLED(func) \ do { \ ok(called_ ## func, "expected " #func "\n"); \ expect_ ## func = called_ ## func = FALSE; \ }while(0) #define SET_EXPECT(func) \ expect_ ## func = TRUE #define expect(EXPECTED,GOT) ok((GOT)==(EXPECTED), "Expected %d, got %d\n", (EXPECTED), (GOT)) #define check_rect(name, val, exp, ...) ok(EqualRect(&val, &exp), \ "invalid rect %s - expected %s - (" name ")\n", \ wine_dbgstr_rect(&val), wine_dbgstr_rect(&exp), __VA_ARGS__); #define compare(val, exp, format) ok((val) == (exp), #val " value " format " expected " format "\n", (val), (exp)); #define check_button_size(handle, width, height, ...) {\ LRESULT bsize = SendMessageA(handle, TB_GETBUTTONSIZE, 0, 0);\ ok(bsize == MAKELONG(width, height), "Unexpected button size - got size (%d, %d), expected (%d, %d)\n", LOWORD(bsize), HIWORD(bsize), width, height);\ } static void MakeButton(TBBUTTON *p, int idCommand, int fsStyle, int nString) { p->iBitmap = -2; p->idCommand = idCommand; p->fsState = TBSTATE_ENABLED; p->fsStyle = fsStyle; p->iString = nString; } /* 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 BOOL equal_dc(HDC hdc1, HDC hdc2, int width, int height) { int x, y; for (x = 0; x < width; ++x) { for (y = 0; y < height; ++y) { if (GetPixel(hdc1, x, y) != GetPixel(hdc2, x, y)) return FALSE; } } return TRUE; } static void *alloced_str; static LRESULT parent_wnd_notify(LPARAM lParam) { NMHDR *hdr = (NMHDR *)lParam; NMTBHOTITEM *nmhi; NMTBDISPINFOA *nmdisp; switch (hdr->code) { case TBN_HOTITEMCHANGE: nmhi = (NMTBHOTITEM *)lParam; g_fReceivedHotItemChange = TRUE; if (g_fExpectedHotItemOld != g_fExpectedHotItemNew) { compare(nmhi->idOld, g_fExpectedHotItemOld, "%d"); compare(nmhi->idNew, g_fExpectedHotItemNew, "%d"); } if (g_fBlockHotItemChange) return 1; break; case TBN_GETDISPINFOA: ok(FALSE, "TBN_GETDISPINFOA received\n"); break; case TBN_GETINFOTIPA: case TBN_GETINFOTIPW: { NMTBGETINFOTIPA *tbgit = (NMTBGETINFOTIPA*)lParam; if (g_ResetDispTextPtr) { tbgit->pszText = NULL; return 0; } break; } case TBN_GETDISPINFOW: nmdisp = (NMTBDISPINFOA *)lParam; compare(nmdisp->dwMask, g_dwExpectedDispInfoMask, "%lx"); ok(nmdisp->pszText == NULL, "pszText is not NULL\n"); break; case TBN_SAVE: { NMTBSAVE *save = (NMTBSAVE *)lParam; if (save->iItem == -1) { save->cbData = save->cbData * 2 + 11 * sizeof(DWORD); save->pData = heap_alloc( save->cbData ); save->pData[0] = 0xcafe; save->pCurrent = save->pData + 1; } else { save->pCurrent[0] = 0xcafe0000 + save->iItem; save->pCurrent++; } /* Add on 5 more pseudo buttons. */ if (save->iItem == save->cButtons - 1) { save->pCurrent[0] = 0xffffffff; save->pCurrent[1] = 0xcafe0007; save->pCurrent[2] = 0xfffffffe; save->pCurrent[3] = 0xcafe0008; save->pCurrent[4] = 0x80000000; save->pCurrent[5] = 0xcafe0009; save->pCurrent[6] = 0x7fffffff; save->pCurrent[7] = 0xcafe000a; save->pCurrent[8] = 0x100; save->pCurrent[9] = 0xcafe000b; } /* Return value is ignored */ return 1; } case TBN_RESTORE: { NMTBRESTORE *restore = (NMTBRESTORE *)lParam; if (restore->iItem == -1) { ok( restore->cButtons == 25, "got %d\n", restore->cButtons ); ok( *restore->pCurrent == 0xcafe, "got %08lx\n", *restore->pCurrent ); /* Skip the last one */ restore->cButtons = 11; restore->pCurrent++; /* BytesPerRecord is ignored */ restore->cbBytesPerRecord = 10; } else { ok( *restore->pCurrent == 0xcafe0000 + restore->iItem, "got %08lx\n", *restore->pCurrent ); if (restore->iItem < 7 || restore->iItem == 10) { ok( restore->tbButton.iBitmap == -1, "got %08x\n", restore->tbButton.iBitmap ); if (restore->iItem < 7) ok( restore->tbButton.idCommand == restore->iItem * 2 + 1, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand ); else ok( restore->tbButton.idCommand == 0x7fffffff, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand ); ok( restore->tbButton.fsState == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState ); ok( restore->tbButton.fsStyle == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsStyle ); } else { ok( restore->tbButton.iBitmap == 8, "got %08x\n", restore->tbButton.iBitmap ); ok( restore->tbButton.idCommand == 0, "%d: got %08x\n", restore->iItem, restore->tbButton.idCommand ); if (restore->iItem == 7) ok( restore->tbButton.fsState == 0, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState ); else ok( restore->tbButton.fsState == TBSTATE_HIDDEN, "%d: got %02x\n", restore->iItem, restore->tbButton.fsState ); ok( restore->tbButton.fsStyle == BTNS_SEP, "%d: got %02x\n", restore->iItem, restore->tbButton.fsStyle ); } ok( restore->tbButton.dwData == 0, "got %08Ix\n", restore->tbButton.dwData ); ok( restore->tbButton.iString == 0, "got %08Ix\n", restore->tbButton.iString ); restore->tbButton.iBitmap = 0; restore->tbButton.fsState = TBSTATE_ENABLED; restore->tbButton.fsStyle = 0; restore->tbButton.dwData = restore->iItem; if (restore->iItem == 0) { restore->tbButton.iString = (INT_PTR)heap_alloc_zero( 8 ); strcpy( (char *)restore->tbButton.iString, "foo" ); } else if (restore->iItem == 1) restore->tbButton.iString = 2; else restore->tbButton.iString = -1; restore->pCurrent++; /* Altering cButtons after the 1st call makes no difference. */ restore->cButtons--; } /* Returning non-zero from the 1st call aborts the restore, otherwise the return value is ignored. */ if (restore->iItem == -1) return 0; return 1; } case TBN_GETBUTTONINFOA: { NMTOOLBARA *tb = (NMTOOLBARA *)lParam; tb->tbButton.iBitmap = 0; tb->tbButton.fsState = 0; tb->tbButton.fsStyle = 0; tb->tbButton.dwData = 0; ok( tb->cchText == 128, "got %d\n", tb->cchText ); switch (tb->iItem) { case 0: tb->tbButton.idCommand = 7; alloced_str = heap_alloc_zero( 8 ); strcpy( alloced_str, "foo" ); tb->tbButton.iString = (INT_PTR)alloced_str; return 1; case 1: tb->tbButton.idCommand = 9; tb->tbButton.iString = 0; /* tb->pszText is ignored */ strcpy( tb->pszText, "foo" ); return 1; case 2: tb->tbButton.idCommand = 11; tb->tbButton.iString = 3; return 1; } return 0; } } return 0; } static LRESULT CALLBACK parent_wnd_proc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam) { static LONG defwndproc_counter = 0; struct message msg; LRESULT ret; msg.message = message; msg.flags = sent|wparam|lparam; if (defwndproc_counter) msg.flags |= defwinproc; msg.wParam = wParam; msg.lParam = lParam; if (message == WM_NOTIFY && lParam) { msg.id = ((NMHDR*)lParam)->code; switch (msg.id) { case TBN_SAVE: { NMTBSAVE *save = (NMTBSAVE *)lParam; msg.stage = save->iItem; } break; case TBN_RESTORE: { NMTBRESTORE *restore = (NMTBRESTORE *)lParam; msg.stage = restore->iItem; } break; case TBN_GETBUTTONINFOA: { NMTOOLBARA *tb = (NMTOOLBARA *)lParam; msg.stage = tb->iItem; } break; } } /* log system messages, except for painting */ if (message < WM_USER && message != WM_PAINT && message != WM_ERASEBKGND && message != WM_NCPAINT && message != WM_NCHITTEST && message != WM_GETTEXT && message != WM_GETICON && message != WM_DEVICECHANGE) { add_message(sequences, PARENT_SEQ_INDEX, &msg); } switch (message) { case WM_NOTIFY: return parent_wnd_notify(lParam); } defwndproc_counter++; ret = DefWindowProcA(hWnd, message, wParam, lParam); defwndproc_counter--; return ret; } static void basic_test(void) { TBBUTTON buttons[9]; HWND hToolbar; int i; for (i=0; i<9; i++) MakeButton(buttons+i, 1000+i, TBSTYLE_CHECKGROUP, 0); MakeButton(buttons+3, 1003, TBSTYLE_SEP|TBSTYLE_GROUP, 0); MakeButton(buttons+6, 1006, TBSTYLE_SEP, 0); hToolbar = pCreateToolbarEx(hMainWnd, WS_VISIBLE | WS_CLIPCHILDREN | CCS_TOP | WS_CHILD | TBSTYLE_LIST, 100, 0, NULL, 0, buttons, ARRAY_SIZE(buttons), 0, 0, 20, 16, sizeof(TBBUTTON)); ok(hToolbar != NULL, "Toolbar creation\n"); SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)"test\000"); /* test for exclusion working inside a separator-separated :-) group */ SendMessageA(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 */ ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n"); ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1001, 0), "A2 not pressed\n"); SendMessageA(hToolbar, TB_CHECKBUTTON, 1004, 1); /* press A5, release A1 */ ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 pressed\n"); ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 not pressed anymore\n"); SendMessageA(hToolbar, TB_CHECKBUTTON, 1005, 1); /* press A6, release A5 */ ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n"); ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1004, 0), "A5 not pressed anymore\n"); /* test for inter-group crosstalk, i.e. two radio groups interfering with each other */ SendMessageA(hToolbar, TB_CHECKBUTTON, 1007, 1); /* press B2 */ ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 still pressed, no inter-group crosstalk\n"); ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 still not pressed\n"); ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 pressed\n"); SendMessageA(hToolbar, TB_CHECKBUTTON, 1000, 1); /* press A1 and ensure B group didn't suffer */ ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 not pressed anymore\n"); ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n"); ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 still pressed\n"); SendMessageA(hToolbar, TB_CHECKBUTTON, 1008, 1); /* press B3, and ensure A group didn't suffer */ ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1005, 0), "A6 pressed\n"); ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1000, 0), "A1 pressed\n"); ok(!SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1007, 0), "B2 not pressed\n"); ok(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 1008, 0), "B3 pressed\n"); /* tests with invalid index */ compare(SendMessageA(hToolbar, TB_ISBUTTONCHECKED, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); compare(SendMessageA(hToolbar, TB_ISBUTTONPRESSED, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); compare(SendMessageA(hToolbar, TB_ISBUTTONENABLED, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); compare(SendMessageA(hToolbar, TB_ISBUTTONINDETERMINATE, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); compare(SendMessageA(hToolbar, TB_ISBUTTONHIGHLIGHTED, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); compare(SendMessageA(hToolbar, TB_ISBUTTONHIDDEN, 0xdeadbeef, 0), (LRESULT)-1L, "%Id"); DestroyWindow(hToolbar); } static void rebuild_toolbar_ex(HWND *hToolbar, DWORD exstyle) { if (*hToolbar) DestroyWindow(*hToolbar); *hToolbar = CreateWindowExA(exstyle, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL); ok(*hToolbar != NULL, "Toolbar creation problem\n"); ok(SendMessageA(*hToolbar, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0) == 0, "TB_BUTTONSTRUCTSIZE failed\n"); ok(SendMessageA(*hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n"); ok(SendMessageA(*hToolbar, WM_SETFONT, (WPARAM)GetStockObject(SYSTEM_FONT), 0)==1, "WM_SETFONT\n"); } static void rebuild_toolbar(HWND *hToolbar) { rebuild_toolbar_ex(hToolbar, 0); } static void rebuild_toolbar_with_buttons(HWND *hToolbar) { TBBUTTON buttons[5]; rebuild_toolbar(hToolbar); ZeroMemory(&buttons, sizeof(buttons)); buttons[0].idCommand = 1; buttons[0].fsStyle = BTNS_BUTTON; buttons[0].fsState = TBSTATE_ENABLED; buttons[0].iString = -1; buttons[1].idCommand = 3; buttons[1].fsStyle = BTNS_BUTTON; buttons[1].fsState = TBSTATE_ENABLED; buttons[1].iString = -1; buttons[2].idCommand = 5; buttons[2].fsStyle = BTNS_SEP; buttons[2].fsState = TBSTATE_ENABLED; buttons[2].iString = -1; buttons[3].idCommand = 7; buttons[3].fsStyle = BTNS_BUTTON; buttons[3].fsState = TBSTATE_ENABLED; buttons[3].iString = -1; buttons[4].idCommand = 9; buttons[4].fsStyle = BTNS_BUTTON; buttons[4].fsState = 0; /* disabled */ buttons[4].iString = -1; ok(SendMessageA(*hToolbar, TB_ADDBUTTONSA, 5, (LPARAM)buttons) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(*hToolbar, TB_AUTOSIZE, 0, 0) == 0, "TB_AUTOSIZE failed\n"); } static void add_128x15_bitmap(HWND hToolbar, int nCmds) { TBADDBITMAP bmp128; bmp128.hInst = GetModuleHandleA(NULL); bmp128.nID = IDB_BITMAP_128x15; ok(SendMessageA(hToolbar, TB_ADDBITMAP, nCmds, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); } #define CHECK_IMAGELIST(count, dx, dy) { \ int cx, cy; \ HIMAGELIST himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); \ ok(himl != NULL, "No image list\n"); \ if (himl != NULL) {\ ok(pImageList_GetImageCount(himl) == count, "Images count mismatch - %d vs %d\n", count, pImageList_GetImageCount(himl)); \ pImageList_GetIconSize(himl, &cx, &cy); \ ok(cx == dx && cy == dy, "Icon size mismatch - %dx%d vs %dx%d\n", dx, dy, cx, cy); \ } \ } static void test_add_bitmap(void) { TBADDBITMAP stdsmall, std; HWND hToolbar = NULL; TBADDBITMAP bmp128; TBADDBITMAP bmp80; TBADDBITMAP addbmp; HIMAGELIST himl; INT ret, id; /* Test default bitmaps range */ for (id = IDB_STD_SMALL_COLOR; id < IDB_HIST_LARGE_COLOR; id++) { HIMAGELIST himl; int cx, cy, count; rebuild_toolbar(&hToolbar); std.hInst = HINST_COMMCTRL; std.nID = id; ret = SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&std); ok(ret == 0, "Got %d\n", ret); himl = (HIMAGELIST)SendMessageA(hToolbar, TB_GETIMAGELIST, 0, 0); ok(himl != NULL, "Got %p\n", himl); ret = pImageList_GetIconSize(himl, &cx, &cy); ok(ret, "Got %d\n", ret); ok(cx == cy, "Got %d x %d\n", cx, cy); count = pImageList_GetImageCount(himl); /* Image count */ switch (id) { case IDB_STD_SMALL_COLOR: case IDB_STD_LARGE_COLOR: case 2: case 3: ok(count == 15, "got count %d\n", count); break; case IDB_VIEW_SMALL_COLOR: case IDB_VIEW_LARGE_COLOR: case 6: case 7: ok(count == 12, "got count %d\n", count); break; case IDB_HIST_SMALL_COLOR: case IDB_HIST_LARGE_COLOR: ok(count == 5, "got count %d\n", count); break; default: ok(0, "id %d, count %d\n", id, count); } /* Image sizes */ switch (id) { case IDB_STD_SMALL_COLOR: case 2: case IDB_VIEW_SMALL_COLOR: case 6: case IDB_HIST_SMALL_COLOR: ok(cx == 16, "got size %d\n", cx); break; case IDB_STD_LARGE_COLOR: case 3: case IDB_VIEW_LARGE_COLOR: case 7: case IDB_HIST_LARGE_COLOR: ok(cx == 24, "got size %d\n", cx); break; default: ok(0, "id %d, size %d\n", id, cx); } } /* empty 128x15 bitmap */ bmp128.hInst = GetModuleHandleA(NULL); bmp128.nID = IDB_BITMAP_128x15; /* empty 80x15 bitmap */ bmp80.hInst = GetModuleHandleA(NULL); bmp80.nID = IDB_BITMAP_80x15; /* standard bitmap - 240x15 pixels */ stdsmall.hInst = HINST_COMMCTRL; stdsmall.nID = IDB_STD_SMALL_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(8, 16, 16); /* adding more bitmaps */ ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 8, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(13, 16, 16); /* adding the same bitmap will simply return the index of the already loaded block */ ret = SendMessageA(hToolbar, TB_ADDBITMAP, 8, (LPARAM)&bmp128); ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(13, 16, 16); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80); ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(13, 16, 16); /* even if we increase the wParam */ ret = SendMessageA(hToolbar, TB_ADDBITMAP, 55, (LPARAM)&bmp80); ok(ret == 8, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(13, 16, 16); /* when the wParam is smaller than the bitmaps count but non-zero, all the bitmaps will be added*/ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 3, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(8, 16, 16); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80); ok(ret == 3, "TB_ADDBITMAP - unexpected return %d\n", ret); /* the returned value is misleading - id 8 is the id of the first icon from bmp80 */ CHECK_IMAGELIST(13, 16, 16); /* the same for negative wParam */ rebuild_toolbar(&hToolbar); ret = SendMessageA(hToolbar, TB_ADDBITMAP, -143, (LPARAM)&bmp128); ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(8, 16, 16); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80); ok(ret == -143, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(13, 16, 16); /* for zero only one bitmap will be added */ rebuild_toolbar(&hToolbar); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&bmp80); ok(ret == 0, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(1, 16, 16); /* if wParam is larger than the amount of icons, the list is grown */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(100, 16, 16); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp128); ok(ret == 100, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(200, 16, 16); /* adding built-in items - the wParam is ignored */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp80) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(5, 16, 16); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&stdsmall) == 5, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(20, 16, 16); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp128) == 20, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(28, 16, 16); /* when we increase the bitmap size, less icons will be created */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(20, 20)) == TRUE, "TB_SETBITMAPSIZE failed\n"); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(6, 20, 20); ret = SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp80); ok(ret == 1, "TB_ADDBITMAP - unexpected return %d\n", ret); CHECK_IMAGELIST(10, 20, 20); /* the icons can be resized - an UpdateWindow is needed as this probably happens during WM_PAINT */ ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(26, 8, 8); /* loading a standard bitmaps automatically resizes the icons */ ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&stdsmall) == 2, "TB_ADDBITMAP - unexpected return\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(28, 16, 16); /* two more SETBITMAPSIZE tests */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(100, 16, 16); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 100, (LPARAM)&bmp80) == 100, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(200, 16, 16); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(200, 8, 8); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(30, 30)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(200, 30, 30); rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 5, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(8, 16, 16); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 3, (LPARAM)&bmp80) == 5, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(13, 16, 16); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(30, 30)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(8, 30, 30); /* when the width or height is zero, set it to 1 */ ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(0, 0)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(208, 1, 1); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(0, 5)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(208, 1, 5); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(5, 0)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); CHECK_IMAGELIST(41, 5, 1); /* the control can add bitmaps to an existing image list */ rebuild_toolbar(&hToolbar); himl = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_80x15), 20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR); ok(himl != NULL, "failed to create imagelist\n"); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n"); CHECK_IMAGELIST(4, 20, 15); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&bmp128) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(10, 20, 15); /* however TB_SETBITMAPSIZE/add std bitmap won't change the image size (the button size does change) */ ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(8, 8)) == TRUE, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(15, 14), "%lx"); CHECK_IMAGELIST(10, 20, 15); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 0, (LPARAM)&stdsmall) == 1, "TB_SETBITMAPSIZE failed\n"); UpdateWindow(hToolbar); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(23, 22), "%lx"); CHECK_IMAGELIST(22, 20, 15); /* check standard bitmaps */ addbmp.hInst = HINST_COMMCTRL; addbmp.nID = IDB_STD_SMALL_COLOR; rebuild_toolbar(&hToolbar); pImageList_Destroy(himl); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(15, 16, 16); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(23, 22), "%lx"); addbmp.nID = IDB_STD_LARGE_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(15, 24, 24); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(31, 30), "%lx"); addbmp.nID = IDB_VIEW_SMALL_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(12, 16, 16); addbmp.nID = IDB_VIEW_LARGE_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(12, 24, 24); addbmp.nID = IDB_HIST_SMALL_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(5, 16, 16); addbmp.nID = IDB_HIST_LARGE_COLOR; rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBITMAP, 1, (LPARAM)&addbmp) == 0, "TB_ADDBITMAP - unexpected return\n"); CHECK_IMAGELIST(5, 24, 24); DestroyWindow(hToolbar); } #define CHECK_STRING_TABLE(count, tab) { \ INT _i; \ CHAR _buf[260]; \ for (_i = 0; _i < (count); _i++) {\ ret = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, _i), (LPARAM)_buf); \ ok(ret >= 0, "TB_GETSTRINGA - unexpected return %d while checking string %d\n", ret, _i); \ if (ret >= 0) \ ok(strcmp(_buf, (tab)[_i]) == 0, "Invalid string #%d - '%s' vs '%s'\n", _i, (tab)[_i], _buf); \ } \ ok(SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, (count)), (LPARAM)_buf) == -1, \ "Too many strings in table\n"); \ } static void test_add_string(void) { LPCSTR test1 = "a\0b\0"; LPCSTR test2 = "|a|b||\0"; LPCSTR ret1[] = {"a", "b"}; LPCSTR ret2[] = {"a", "b", "|a|b||"}; LPCSTR ret3[] = {"a", "b", "|a|b||", "p", "q"}; LPCSTR ret4[] = {"a", "b", "|a|b||", "p", "q", "p"}; LPCSTR ret5[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q"}; LPCSTR ret6[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q", "p", "", "q"}; LPCSTR ret7[] = {"a", "b", "|a|b||", "p", "q", "p", "p", "q", "p", "", "q", "br", "c", "d"}; HWND hToolbar = NULL; TBBUTTON button; int ret; CHAR buf[260]; rebuild_toolbar(&hToolbar); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)test1); ok(ret == 0, "TB_ADDSTRINGA - unexpected return %d\n", ret); ret = SendMessageA(hToolbar, TB_GETSTRINGA, MAKEWPARAM(260, 1), (LPARAM)buf); if (ret == 0) { win_skip("TB_GETSTRINGA needs 5.80\n"); return; } CHECK_STRING_TABLE(2, ret1); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)test2); ok(ret == 2, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(3, ret2); /* null instance handle */ ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0, IDS_TBADD1); ok(ret == -1, "TB_ADDSTRINGA - unexpected return %d\n", ret); /* invalid instance handle */ ret = SendMessageA(hToolbar, TB_ADDSTRINGA, 0xdeadbeef, IDS_TBADD1); ok(ret == -1, "TB_ADDSTRINGA - unexpected return %d\n", ret); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD1); ok(ret == 3, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(3, ret2); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD2); ok(ret == 3, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(5, ret3); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD3); ok(ret == 5, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(6, ret4); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD4); ok(ret == 6, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(8, ret5); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD5); ok(ret == 8, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(11, ret6); ret = SendMessageA(hToolbar, TB_ADDSTRINGA, (WPARAM)GetModuleHandleA(NULL), IDS_TBADD7); ok(ret == 11, "TB_ADDSTRINGA - unexpected return %d\n", ret); CHECK_STRING_TABLE(14, ret7); ZeroMemory(&button, sizeof(button)); button.iString = (UINT_PTR)"Test"; SendMessageA(hToolbar, TB_INSERTBUTTONA, 0, (LPARAM)&button); CHECK_STRING_TABLE(14, ret7); SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&button); CHECK_STRING_TABLE(14, ret7); DestroyWindow(hToolbar); } static void expect_hot_notify(int idold, int idnew) { g_fExpectedHotItemOld = idold; g_fExpectedHotItemNew = idnew; g_fReceivedHotItemChange = FALSE; } #define check_hot_notify() \ ok(g_fReceivedHotItemChange, "TBN_HOTITEMCHANGE not received\n"); \ g_fExpectedHotItemOld = g_fExpectedHotItemNew = 0; static void test_hotitem(void) { HWND hToolbar = NULL; TBBUTTONINFOA tbinfo; LRESULT ret; g_fBlockHotItemChange = FALSE; rebuild_toolbar_with_buttons(&hToolbar); /* set TBSTYLE_FLAT. comctl5 allows hot items only for such toolbars. * comctl6 doesn't have this requirement even when theme == NULL */ SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_FLAT | GetWindowLongA(hToolbar, GWL_STYLE)); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == -1, "Hot item: %Id, expected -1\n", ret); ret = SendMessageA(hToolbar, TB_SETHOTITEM, 1, 0); ok(ret == -1, "TB_SETHOTITEM returned %Id, expected -1\n", ret); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 1, "Hot item: %Id, expected 1\n", ret); ret = SendMessageA(hToolbar, TB_SETHOTITEM, 2, 0); ok(ret == 1, "TB_SETHOTITEM returned %Id, expected 1\n", ret); ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0xbeef, 0); ok(ret == 2, "TB_SETHOTITEM returned %Id, expected 2\n", ret); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 2, "Hot item: %Ix, expected 2\n", ret); ret = SendMessageA(hToolbar, TB_SETHOTITEM, -0xbeef, 0); ok(ret == 2, "TB_SETHOTITEM returned %Id, expected 2\n", ret); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == -1, "Hot item: %Ix, expected -1\n", ret); expect_hot_notify(0, 7); ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0); ok(ret == -1, "TB_SETHOTITEM returned %Id, expected -1\n", ret); check_hot_notify(); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 3, "Hot item: %Ix, expected 3\n", ret); g_fBlockHotItemChange = TRUE; ret = SendMessageA(hToolbar, TB_SETHOTITEM, 2, 0); ok(ret == 3, "TB_SETHOTITEM returned %Id, expected 3\n", ret); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 3, "Hot item: %Ix, expected 3\n", ret); g_fBlockHotItemChange = FALSE; g_fReceivedHotItemChange = FALSE; ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0xbeaf, 0); ok(ret == 3, "TB_SETHOTITEM returned %Id, expected 3\n", ret); ok(g_fReceivedHotItemChange == FALSE, "TBN_HOTITEMCHANGE received for invalid parameter\n"); g_fReceivedHotItemChange = FALSE; ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0); ok(ret == 3, "TB_SETHOTITEM returned %Id, expected 3\n", ret); ok(g_fReceivedHotItemChange == FALSE, "TBN_HOTITEMCHANGE received after a duplication\n"); expect_hot_notify(7, 0); ret = SendMessageA(hToolbar, TB_SETHOTITEM, -0xbeaf, 0); ok(ret == 3, "TB_SETHOTITEM returned %Id, expected 3\n", ret); check_hot_notify(); SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0); /* setting disabled buttons will generate a notify with the button id but no button will be hot */ expect_hot_notify(7, 9); ret = SendMessageA(hToolbar, TB_SETHOTITEM, 4, 0); ok(ret == 3, "TB_SETHOTITEM returned %Id, expected 3\n", ret); check_hot_notify(); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == -1, "Hot item: %Ix, expected -1\n", ret); /* enabling the button won't change that */ SendMessageA(hToolbar, TB_ENABLEBUTTON, 9, TRUE); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == -1, "TB_GETHOTITEM returned %Id, expected -1\n", ret); /* disabling a hot button works */ ret = SendMessageA(hToolbar, TB_SETHOTITEM, 3, 0); ok(ret == -1, "TB_SETHOTITEM returned %Id, expected -1\n", ret); g_fReceivedHotItemChange = FALSE; SendMessageA(hToolbar, TB_ENABLEBUTTON, 7, FALSE); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 3, "TB_GETHOTITEM returned %Id, expected 3\n", ret); ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n"); SendMessageA(hToolbar, TB_SETHOTITEM, 1, 0); tbinfo.cbSize = sizeof(TBBUTTONINFOA); tbinfo.dwMask = TBIF_STATE; tbinfo.fsState = 0; /* disabled */ g_fReceivedHotItemChange = FALSE; ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 1, (LPARAM)&tbinfo) == TRUE, "TB_SETBUTTONINFOA failed\n"); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == 1, "TB_GETHOTITEM returned %Id, expected 1\n", ret); ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n"); /* deleting a button unsets the hot item */ ret = SendMessageA(hToolbar, TB_SETHOTITEM, 0, 0); ok(ret == 1, "TB_SETHOTITEM returned %Id, expected 1\n", ret); g_fReceivedHotItemChange = FALSE; ret = SendMessageA(hToolbar, TB_DELETEBUTTON, 1, 0); ok(ret == TRUE, "TB_DELETEBUTTON returned %Id, expected TRUE\n", ret); ret = SendMessageA(hToolbar, TB_GETHOTITEM, 0, 0); ok(ret == -1, "TB_GETHOTITEM returned %Id, expected -1\n", ret); ok(g_fReceivedHotItemChange == FALSE, "Unexpected TBN_HOTITEMCHANGE\n"); DestroyWindow(hToolbar); } #if 0 /* use this to generate more tests*/ static void dump_sizes(HWND hToolbar) { SIZE sz; RECT r; int count = SendMessageA(hToolbar, TB_BUTTONCOUNT, 0, 0); int i; GetClientRect(hToolbar, &r); SendMessageA(hToolbar, TB_GETMAXSIZE, 0, &sz); printf(" { {%d, %d, %d, %d}, {%d, %d}, %d, {", r.left, r.top, r.right, r.bottom, sz.cx, sz.cy, count); for (i=0; iprcButtons[tbsr->nButtons], left, top, right, bottom); tbsr->nButtons++; } #define STRING0 "A" #define STRING1 "MMMMMMMMMMMMM" #define STRING2 "Tst" static tbsize_result_t *tbsize_results; #define tbsize_results_num 29 static void init_tbsize_results(void) { int fontheight = system_font_height(); int buttonwidth; tbsize_results = heap_alloc_zero(tbsize_results_num * sizeof(*tbsize_results)); tbsize_results[0] = init_tbsize_result(5, 0, 0 ,672 ,26, 100 ,22); tbsize_addbutton(&tbsize_results[0], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[0], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[0], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[0], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[0], 77, 2, 100, 24); tbsize_results[1] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22); tbsize_addbutton(&tbsize_results[1], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[1], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[1], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[1], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[1], 77, 2, 100, 24); tbsize_addbutton(&tbsize_results[1], 100, 2, 123, 24); tbsize_addbutton(&tbsize_results[1], 0, 24, 23, 46); tbsize_results[2] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22); tbsize_addbutton(&tbsize_results[2], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[2], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[2], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[2], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[2], 77, 2, 100, 24); tbsize_addbutton(&tbsize_results[2], 100, 2, 123, 24); tbsize_addbutton(&tbsize_results[2], 0, 24, 23, 46); tbsize_results[3] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22); tbsize_addbutton(&tbsize_results[3], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[3], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[3], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[3], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[3], 77, 2, 100, 24); tbsize_addbutton(&tbsize_results[3], 100, 2, 123, 24); tbsize_addbutton(&tbsize_results[3], 123, 2, 146, 24); tbsize_results[4] = init_tbsize_result(9, 0, 0, 672, 26, 192, 22); tbsize_addbutton(&tbsize_results[4], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[4], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[4], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[4], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[4], 77, 2, 100, 24); tbsize_addbutton(&tbsize_results[4], 100, 2, 123, 24); tbsize_addbutton(&tbsize_results[4], 123, 2, 146, 24); tbsize_addbutton(&tbsize_results[4], 146, 2, 169, 24); tbsize_addbutton(&tbsize_results[4], 169, 2, 192, 24); tbsize_results[5] = init_tbsize_result(39, 0, 0, 672, 92, 882, 22); tbsize_addbutton(&tbsize_results[5], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[5], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[5], 0, 2, 8, 29); tbsize_addbutton(&tbsize_results[5], 0, 29, 23, 51); tbsize_addbutton(&tbsize_results[5], 23, 29, 46, 51); tbsize_addbutton(&tbsize_results[5], 46, 29, 69, 51); tbsize_addbutton(&tbsize_results[5], 69, 29, 92, 51); tbsize_addbutton(&tbsize_results[5], 92, 29, 115, 51); tbsize_addbutton(&tbsize_results[5], 115, 29, 138, 51); tbsize_addbutton(&tbsize_results[5], 138, 29, 161, 51); tbsize_addbutton(&tbsize_results[5], 161, 29, 184, 51); tbsize_addbutton(&tbsize_results[5], 184, 29, 207, 51); tbsize_addbutton(&tbsize_results[5], 207, 29, 230, 51); tbsize_addbutton(&tbsize_results[5], 230, 29, 253, 51); tbsize_addbutton(&tbsize_results[5], 253, 29, 276, 51); tbsize_addbutton(&tbsize_results[5], 276, 29, 299, 51); tbsize_addbutton(&tbsize_results[5], 299, 29, 322, 51); tbsize_addbutton(&tbsize_results[5], 322, 29, 345, 51); tbsize_addbutton(&tbsize_results[5], 345, 29, 368, 51); tbsize_addbutton(&tbsize_results[5], 368, 29, 391, 51); tbsize_addbutton(&tbsize_results[5], 391, 29, 414, 51); tbsize_addbutton(&tbsize_results[5], 414, 29, 437, 51); tbsize_addbutton(&tbsize_results[5], 437, 29, 460, 51); tbsize_addbutton(&tbsize_results[5], 460, 29, 483, 51); tbsize_addbutton(&tbsize_results[5], 483, 29, 506, 51); tbsize_addbutton(&tbsize_results[5], 506, 29, 529, 51); tbsize_addbutton(&tbsize_results[5], 529, 29, 552, 51); tbsize_addbutton(&tbsize_results[5], 552, 29, 575, 51); tbsize_addbutton(&tbsize_results[5], 575, 29, 598, 51); tbsize_addbutton(&tbsize_results[5], 598, 29, 621, 51); tbsize_addbutton(&tbsize_results[5], 621, 29, 644, 51); tbsize_addbutton(&tbsize_results[5], 644, 29, 667, 51); tbsize_addbutton(&tbsize_results[5], 0, 51, 23, 73); tbsize_addbutton(&tbsize_results[5], 23, 51, 46, 73); tbsize_addbutton(&tbsize_results[5], 46, 51, 69, 73); tbsize_addbutton(&tbsize_results[5], 69, 51, 92, 73); tbsize_addbutton(&tbsize_results[5], 92, 51, 115, 73); tbsize_addbutton(&tbsize_results[5], 115, 51, 138, 73); tbsize_addbutton(&tbsize_results[5], 138, 51, 161, 73); tbsize_results[6] = init_tbsize_result(7, 0, 0, 48, 226, 23, 140); tbsize_addbutton(&tbsize_results[6], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[6], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[6], 46, 2, 94, 24); tbsize_addbutton(&tbsize_results[6], 94, 2, 117, 24); tbsize_addbutton(&tbsize_results[6], 117, 2, 140, 24); tbsize_addbutton(&tbsize_results[6], 140, 2, 163, 24); tbsize_addbutton(&tbsize_results[6], 0, 24, 23, 46); tbsize_results[7] = init_tbsize_result(7, 0, 0, 92, 226, 23, 140); tbsize_addbutton(&tbsize_results[7], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[7], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[7], 0, 24, 92, 32); tbsize_addbutton(&tbsize_results[7], 0, 32, 23, 54); tbsize_addbutton(&tbsize_results[7], 23, 32, 46, 54); tbsize_addbutton(&tbsize_results[7], 46, 32, 69, 54); tbsize_addbutton(&tbsize_results[7], 69, 32, 92, 54); tbsize_results[8] = init_tbsize_result(7, 0, 0, 672, 26, 194, 30); tbsize_addbutton(&tbsize_results[8], 0, 2, 31, 32); tbsize_addbutton(&tbsize_results[8], 31, 2, 62, 32); tbsize_addbutton(&tbsize_results[8], 62, 2, 70, 32); tbsize_addbutton(&tbsize_results[8], 70, 2, 101, 32); tbsize_addbutton(&tbsize_results[8], 101, 2, 132, 32); tbsize_addbutton(&tbsize_results[8], 132, 2, 163, 32); tbsize_addbutton(&tbsize_results[8], 0, 32, 31, 62); tbsize_results[9] = init_tbsize_result(7, 0, 0, 672, 64, 194, 30); tbsize_addbutton(&tbsize_results[9], 0, 2, 31, 32); tbsize_addbutton(&tbsize_results[9], 31, 2, 62, 32); tbsize_addbutton(&tbsize_results[9], 62, 2, 70, 32); tbsize_addbutton(&tbsize_results[9], 70, 2, 101, 32); tbsize_addbutton(&tbsize_results[9], 101, 2, 132, 32); tbsize_addbutton(&tbsize_results[9], 132, 2, 163, 32); tbsize_addbutton(&tbsize_results[9], 0, 32, 31, 62); tbsize_results[10] = init_tbsize_result(7, 0, 0, 672, 64, 194, 30); tbsize_addbutton(&tbsize_results[10], 0, 0, 31, 30); tbsize_addbutton(&tbsize_results[10], 31, 0, 62, 30); tbsize_addbutton(&tbsize_results[10], 62, 0, 70, 30); tbsize_addbutton(&tbsize_results[10], 70, 0, 101, 30); tbsize_addbutton(&tbsize_results[10], 101, 0, 132, 30); tbsize_addbutton(&tbsize_results[10], 132, 0, 163, 30); tbsize_addbutton(&tbsize_results[10], 0, 30, 31, 60); tbsize_results[11] = init_tbsize_result(7, 0, 0, 124, 226, 31, 188); tbsize_addbutton(&tbsize_results[11], 0, 0, 31, 30); tbsize_addbutton(&tbsize_results[11], 31, 0, 62, 30); tbsize_addbutton(&tbsize_results[11], 0, 30, 124, 38); tbsize_addbutton(&tbsize_results[11], 0, 38, 31, 68); tbsize_addbutton(&tbsize_results[11], 31, 38, 62, 68); tbsize_addbutton(&tbsize_results[11], 62, 38, 93, 68); tbsize_addbutton(&tbsize_results[11], 93, 38, 124, 68); tbsize_results[12] = init_tbsize_result(7, 0, 0, 672, 26, 146, 22); tbsize_addbutton(&tbsize_results[12], 0, 2, 23, 24); tbsize_addbutton(&tbsize_results[12], 23, 2, 46, 24); tbsize_addbutton(&tbsize_results[12], 46, 2, 54, 24); tbsize_addbutton(&tbsize_results[12], 54, 2, 77, 24); tbsize_addbutton(&tbsize_results[12], 77, 2, 100, 24); tbsize_addbutton(&tbsize_results[12], 100, 2, 123, 24); tbsize_addbutton(&tbsize_results[12], 123, 2, 146, 24); tbsize_results[13] = init_tbsize_result(7, 0, 0, 672, 26, 146, 100); tbsize_addbutton(&tbsize_results[13], 0, 0, 23, 100); tbsize_addbutton(&tbsize_results[13], 23, 0, 46, 100); tbsize_addbutton(&tbsize_results[13], 46, 0, 54, 100); tbsize_addbutton(&tbsize_results[13], 54, 0, 77, 100); tbsize_addbutton(&tbsize_results[13], 77, 0, 100, 100); tbsize_addbutton(&tbsize_results[13], 100, 0, 123, 100); tbsize_addbutton(&tbsize_results[13], 123, 0, 146, 100); tbsize_results[14] = init_tbsize_result(10, 0, 0, 672, 26, 146, 100); tbsize_addbutton(&tbsize_results[14], 0, 0, 23, 100); tbsize_addbutton(&tbsize_results[14], 23, 0, 46, 100); tbsize_addbutton(&tbsize_results[14], 46, 0, 54, 100); tbsize_addbutton(&tbsize_results[14], 54, 0, 77, 100); tbsize_addbutton(&tbsize_results[14], 77, 0, 100, 100); tbsize_addbutton(&tbsize_results[14], 100, 0, 123, 100); tbsize_addbutton(&tbsize_results[14], 123, 0, 146, 100); tbsize_addbutton(&tbsize_results[14], 146, 0, 169, 100); tbsize_addbutton(&tbsize_results[14], 169, 0, 192, 100); tbsize_addbutton(&tbsize_results[14], 192, 0, 215, 100); tbsize_results[15] = init_tbsize_result(11, 0, 0, 672, 26, 238, 39); tbsize_addbutton(&tbsize_results[15], 0, 0, 23, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 23, 0, 46, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 46, 0, 54, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 54, 0, 77, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 77, 0, 100, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 100, 0, 123, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 123, 0, 146, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 146, 0, 169, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 169, 0, 192, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 192, 0, 215, 23 + fontheight); tbsize_addbutton(&tbsize_results[15], 215, 0, 238, 23 + fontheight); tbsize_results[16] = init_tbsize_result(11, 0, 0, 672, 26, 239, 22); tbsize_addbutton(&tbsize_results[16], 0, 0, 23, 22); tbsize_addbutton(&tbsize_results[16], 23, 0, 46, 22); tbsize_addbutton(&tbsize_results[16], 46, 0, 54, 22); tbsize_addbutton(&tbsize_results[16], 54, 0, 77, 22); tbsize_addbutton(&tbsize_results[16], 77, 0, 100, 22); tbsize_addbutton(&tbsize_results[16], 100, 0, 123, 22); tbsize_addbutton(&tbsize_results[16], 123, 0, 146, 22); tbsize_addbutton(&tbsize_results[16], 146, 0, 169, 22); tbsize_addbutton(&tbsize_results[16], 169, 0, 192, 22); tbsize_addbutton(&tbsize_results[16], 192, 0, 215, 22); tbsize_addbutton(&tbsize_results[16], 215, 0, 238, 22); buttonwidth = 7 + string_width(STRING1); tbsize_results[17] = init_tbsize_result(3, 0, 0, 672, 26, 489, 39); tbsize_addbutton(&tbsize_results[17], 0, 2, buttonwidth, 25 + fontheight); tbsize_addbutton(&tbsize_results[17], buttonwidth, 2, 2*buttonwidth + 4, 25 + fontheight); tbsize_addbutton(&tbsize_results[17], 2*buttonwidth + 4, 2, 3*buttonwidth + 4, 25 + fontheight); tbsize_results[18] = init_tbsize_result(6, 0, 0, 672, 104, 978, 24); tbsize_addbutton(&tbsize_results[18], 0, 2, buttonwidth, 10 + fontheight); tbsize_addbutton(&tbsize_results[18], buttonwidth, 2, 2*buttonwidth, 10 + fontheight); tbsize_addbutton(&tbsize_results[18], 2*buttonwidth, 2, 3*buttonwidth, 10 + fontheight); tbsize_addbutton(&tbsize_results[18], 3*buttonwidth, 2, 4*buttonwidth, 10 + fontheight); tbsize_addbutton(&tbsize_results[18], 4*buttonwidth, 2, 5*buttonwidth + 4, 10 + fontheight); tbsize_addbutton(&tbsize_results[18], 5*buttonwidth + 4, 2, 5*buttonwidth + 4 + string_width(STRING2) + 11, 10 + fontheight); tbsize_results[19] = init_tbsize_result(6, 0, 0, 672, 28, 978, 38); tbsize_addbutton(&tbsize_results[19], 0, 0, buttonwidth, 22 + fontheight); tbsize_addbutton(&tbsize_results[19], buttonwidth, 0, 2*buttonwidth, 22 + fontheight); tbsize_addbutton(&tbsize_results[19], 2*buttonwidth, 0, 3*buttonwidth, 22 + fontheight); tbsize_addbutton(&tbsize_results[19], 3*buttonwidth, 0, 4*buttonwidth, 22 + fontheight); tbsize_addbutton(&tbsize_results[19], 4*buttonwidth, 0, 5*buttonwidth + 4, 22 + fontheight); tbsize_addbutton(&tbsize_results[19], 5*buttonwidth + 4, 0, 5*buttonwidth + 4 + string_width(STRING2) + 11, 22 + fontheight); tbsize_results[20] = init_tbsize_result(3, 0, 0, 672, 100, 239, 102); tbsize_addbutton(&tbsize_results[20], 0, 2, 100, 102); tbsize_addbutton(&tbsize_results[20], 100, 2, 139, 102); tbsize_addbutton(&tbsize_results[20], 139, 2, 239, 102); tbsize_results[21] = init_tbsize_result(3, 0, 0, 672, 42, 185, 40); tbsize_addbutton(&tbsize_results[21], 0, 2, 75, 40); tbsize_addbutton(&tbsize_results[21], 75, 2, 118, 40); tbsize_addbutton(&tbsize_results[21], 118, 2, 165 + string_width(STRING2), 40); tbsize_results[22] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[22], 0, 2, 47 + string_width(STRING2), 40); tbsize_results[23] = init_tbsize_result(2, 0, 0, 672, 42, 67, 41); tbsize_addbutton(&tbsize_results[23], 0, 2, 672, 25 + fontheight); tbsize_addbutton(&tbsize_results[23], 0, 25 + fontheight, 672, 48 + 2*fontheight); tbsize_results[24] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[24], 0, 2, 11 + string_width(STRING2), 24); tbsize_results[25] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[25], 0, 2, 40, 24); tbsize_results[26] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[26], 0, 2, 40, 24); tbsize_results[27] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[27], 0, 2, 40, 24); tbsize_results[28] = init_tbsize_result(1, 0, 0, 672, 42, 67, 40); tbsize_addbutton(&tbsize_results[28], 0, 2, 23, 24); } static void free_tbsize_results(void) { int i; for (i = 0; i < tbsize_results_num; i++) heap_free(tbsize_results[i].prcButtons); heap_free(tbsize_results); tbsize_results = NULL; } static int tbsize_numtests = 0; typedef struct { int test_num; int rect_index; RECT rcButton; } tbsize_alt_result_t; static tbsize_alt_result_t tbsize_alt_results[] = { { 5, 2, { 0, 24, 8, 29 } }, { 20, 1, { 100, 2, 107, 102 } }, { 20, 2, { 107, 2, 207, 102 } } }; static DWORD tbsize_alt_numtests = 0; #define check_sizes_todo(todomask) { \ RECT rc; \ int buttonCount, i, mask=(todomask); \ tbsize_result_t *res = &tbsize_results[tbsize_numtests]; \ GetClientRect(hToolbar, &rc); \ /*check_rect("client", rc, res->rcClient);*/ \ buttonCount = SendMessageA(hToolbar, TB_BUTTONCOUNT, 0, 0); \ compare(buttonCount, res->nButtons, "%d"); \ for (i=0; inButtons); i++) { \ ok(SendMessageA(hToolbar, TB_GETITEMRECT, i, (LPARAM)&rc) == 1, "TB_GETITEMRECT\n"); \ if (broken(tbsize_alt_numtests < ARRAY_SIZE(tbsize_alt_results) && \ EqualRect(&rc, &tbsize_alt_results[tbsize_alt_numtests].rcButton))) { \ win_skip("Alternate rect found\n"); \ tbsize_alt_numtests++; \ } else todo_wine_if(mask&1) \ check_rect("button = %d, tbsize_numtests = %d", rc, res->prcButtons[i], i, tbsize_numtests); \ mask >>= 1; \ } \ tbsize_numtests++; \ } #define check_sizes() check_sizes_todo(0) #endif static TBBUTTON buttons1[] = { {0, 10, TBSTATE_WRAP|TBSTATE_ENABLED, 0, {0, }, 0, -1}, {0, 11, 0, 0, {0, }, 0, -1}, }; static TBBUTTON buttons2[] = { {0, 20, TBSTATE_ENABLED, 0, {0, }, 0, -1}, {0, 21, TBSTATE_ENABLED, 0, {0, }, 0, -1}, }; static TBBUTTON buttons3[] = { {0, 30, TBSTATE_ENABLED, 0, {0, }, 0, 0}, {0, 31, TBSTATE_ENABLED, 0, {0, }, 0, 1}, {0, 32, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, 1}, {0, 33, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, (UINT_PTR)STRING2} }; static TBBUTTON buttons4[] = { {0, 40, TBSTATE_ENABLED, BTNS_AUTOSIZE, {0, }, 0, (UINT_PTR)STRING2}, {0, 41, TBSTATE_ENABLED, 0, {0, }, 0, (UINT_PTR)STRING2}, {0, 41, TBSTATE_ENABLED, BTNS_SHOWTEXT, {0, }, 0, (UINT_PTR)STRING2} }; static void test_sizes(void) { HWND hToolbar = NULL; HIMAGELIST himl, himl2; TBBUTTONINFOA tbinfo; TBBUTTON button; int style; int i; int fontheight = system_font_height(); RECT rect; init_tbsize_results(); rebuild_toolbar_with_buttons(&hToolbar); style = GetWindowLongA(hToolbar, GWL_STYLE); ok(style == (WS_CHILD|WS_VISIBLE|CCS_TOP), "Invalid style %x\n", style); check_sizes(); /* the TBSTATE_WRAP makes a second row */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1); check_sizes(); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0); check_sizes(); SendMessageA(hToolbar, TB_GETBUTTON, 5, (LPARAM)&button); ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "got %08x\n", button.fsState); /* after setting the TBSTYLE_WRAPABLE the TBSTATE_WRAP is ignored */ SetWindowLongA(hToolbar, GWL_STYLE, style|TBSTYLE_WRAPABLE); check_sizes(); SendMessageA(hToolbar, TB_GETBUTTON, 5, (LPARAM)&button); ok(button.fsState == TBSTATE_ENABLED, "got %08x\n", button.fsState); /* adding new buttons with TBSTYLE_WRAPABLE doesn't add a new row */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1); check_sizes(); /* only after adding enough buttons the bar will be wrapped on a * separator and then on the first button */ for (i=0; i<15; i++) SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1); check_sizes_todo(0x4); SendMessageA(hToolbar, TB_GETBUTTON, 31, (LPARAM)&button); ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "got %08x\n", button.fsState); SetWindowLongA(hToolbar, GWL_STYLE, style); SendMessageA(hToolbar, TB_GETBUTTON, 31, (LPARAM)&button); ok(button.fsState == TBSTATE_ENABLED, "got %08x\n", button.fsState); rebuild_toolbar_with_buttons(&hToolbar); SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1); /* setting the buttons vertical will only change the window client size */ SetWindowLongA(hToolbar, GWL_STYLE, style | CCS_VERT); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0); check_sizes_todo(0x3c); /* with a TBSTYLE_WRAPABLE a wrapping will occur on the separator */ SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_WRAPABLE | CCS_VERT); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0); check_sizes_todo(0x7c); rebuild_toolbar_with_buttons(&hToolbar); SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons1); /* a TB_SETBITMAPSIZE changes button sizes*/ SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(24, 24)); check_sizes(); /* setting a TBSTYLE_FLAT doesn't change anything - even after a TB_AUTOSIZE */ SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0); check_sizes(); /* but after a TB_SETBITMAPSIZE the top margins is changed */ SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(20, 20)); SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(24, 24)); check_sizes(); /* some vertical toolbar sizes */ SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT | TBSTYLE_WRAPABLE | CCS_VERT); check_sizes_todo(0x7c); rebuild_toolbar_with_buttons(&hToolbar); SetWindowLongA(hToolbar, GWL_STYLE, style | TBSTYLE_FLAT); /* newly added buttons will be use the previous margin */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons2); check_sizes(); /* TB_SETBUTTONSIZE can't be used to reduce the size of a button below the default */ check_button_size(hToolbar, 23, 22); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(22, 21))==1, "TB_SETBUTTONSIZE\n"); check_button_size(hToolbar, 23, 22); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(5, 100))==1, "TB_SETBUTTONSIZE\n"); check_button_size(hToolbar, 23, 100); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3))==1, "TB_SETBUTTONSIZE\n"); check_button_size(hToolbar, 23, 22); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(5, 100))==1, "TB_SETBUTTONSIZE\n"); check_button_size(hToolbar, 23, 100); check_sizes(); /* add some buttons with non-default sizes */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 2, (LPARAM)buttons2); SendMessageA(hToolbar, TB_INSERTBUTTONA, -1, (LPARAM)&buttons2[0]); check_sizes(); SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]); /* TB_ADDSTRINGA resets the size */ SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM) STRING0 "\0" STRING1 "\0"); check_button_size(hToolbar, 23, 23 + fontheight); check_sizes(); /* TB_SETBUTTONSIZE can be used to crop the text */ SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3)); check_button_size(hToolbar, 23, 22); check_sizes(); /* the default size is bitmap size + padding */ SendMessageA(hToolbar, TB_SETPADDING, 0, MAKELONG(1, 1)); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3)); check_button_size(hToolbar, 17, 17); SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(3, 3)); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(3, 3)); check_button_size(hToolbar, 4, 4); rebuild_toolbar(&hToolbar); /* sending a TB_SETBITMAPSIZE with the same sizes is enough to make the button smaller */ check_button_size(hToolbar, 23, 22); SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(16, 15)); check_button_size(hToolbar, 23, 21); /* -1 in TB_SETBITMAPSIZE is a special code meaning that the coordinate shouldn't be changed */ add_128x15_bitmap(hToolbar, 16); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(14, -1)), "TB_SETBITMAPSIZE failed\n"); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 21), "%lx"); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(-1, 12)), "TB_SETBITMAPSIZE failed\n"); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 18), "%lx"); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELONG(-1, -1)), "TB_SETBITMAPSIZE failed\n"); compare((LONG)SendMessageA(hToolbar, TB_GETBUTTONSIZE, 0, 0), MAKELONG(21, 18), "%lx"); /* check the imagelist */ InvalidateRect(hToolbar, NULL, TRUE); UpdateWindow(hToolbar); CHECK_IMAGELIST(16, 14, 12); rebuild_toolbar(&hToolbar); SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)STRING0 "\0" STRING1 "\0"); /* the height is increased after a TB_ADDSTRINGA */ check_button_size(hToolbar, 23, 23 + fontheight); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100)); /* if a string is in the pool, even adding a button without a string resets the size */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]); check_button_size(hToolbar, 23, 22); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100)); /* an BTNS_AUTOSIZE button is also considered when computing the new size */ SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]); check_button_size(hToolbar, 7 + string_width(STRING1), 23 + fontheight); SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]); check_sizes(); /* delete button doesn't change the buttons size */ SendMessageA(hToolbar, TB_DELETEBUTTON, 2, 0); SendMessageA(hToolbar, TB_DELETEBUTTON, 1, 0); check_button_size(hToolbar, 7 + string_width(STRING1), 23 + fontheight); /* TB_INSERTBUTTONAS will */ SendMessageA(hToolbar, TB_INSERTBUTTONA, 1, (LPARAM)&buttons2[0]); check_button_size(hToolbar, 23, 22); /* TB_HIDEBUTTON and TB_MOVEBUTTON doesn't force a recalc */ SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100)); ok(SendMessageA(hToolbar, TB_MOVEBUTTON, 0, 1), "TB_MOVEBUTTON failed\n"); check_button_size(hToolbar, 100, 100); ok(SendMessageA(hToolbar, TB_HIDEBUTTON, 20, TRUE), "TB_HIDEBUTTON failed\n"); check_button_size(hToolbar, 100, 100); /* however changing the hidden flag with TB_SETSTATE does */ ok(SendMessageA(hToolbar, TB_SETSTATE, 20, TBSTATE_ENABLED|TBSTATE_HIDDEN), "TB_SETSTATE failed\n"); check_button_size(hToolbar, 100, 100); ok(SendMessageA(hToolbar, TB_SETSTATE, 20, TBSTATE_ENABLED), "TB_SETSTATE failed\n"); check_button_size(hToolbar, 23, 22); /* TB_SETIMAGELIST always changes the height but the width only if necessary */ SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100)); himl = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_80x15), 20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n"); check_button_size(hToolbar, 100, 21); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(100, 100)); check_button_size(hToolbar, 100, 100); /* But there are no update when we change imagelist, and image sizes are the same */ himl2 = pImageList_LoadImageA(GetModuleHandleA(NULL), (LPCSTR)MAKEINTRESOURCE(IDB_BITMAP_128x15), 20, 2, CLR_NONE, IMAGE_BITMAP, LR_DEFAULTCOLOR); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LRESULT)himl2) == (LRESULT)himl, "TB_SETIMAGELIST failed\n"); check_button_size(hToolbar, 100, 100); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(1, 1)); check_button_size(hToolbar, 27, 21); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, 0) == (LRESULT)himl2, "TB_SETIMAGELIST failed\n"); check_button_size(hToolbar, 27, 7); SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELONG(1, 1)); check_button_size(hToolbar, 8, 7) ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n"); check_button_size(hToolbar, 27, 21) /* the text is taken into account */ SendMessageA(hToolbar, TB_ADDSTRINGA, 0, (LPARAM)STRING0 "\0" STRING1 "\0"); SendMessageA(hToolbar, TB_ADDBUTTONSA, 4, (LPARAM)buttons3); check_button_size(hToolbar, 7 + string_width(STRING1), 22 + fontheight); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, 0) == (LRESULT)himl, "TB_SETIMAGELIST failed\n"); check_button_size(hToolbar, 7 + string_width(STRING1), 8 + fontheight); /* the style change also comes into effect */ check_sizes(); SetWindowLongA(hToolbar, GWL_STYLE, GetWindowLongA(hToolbar, GWL_STYLE) | TBSTYLE_FLAT); ok(SendMessageA(hToolbar, TB_SETIMAGELIST, 0, (LPARAM)himl) == 0, "TB_SETIMAGELIST failed\n"); check_sizes_todo(0x30); /* some small problems with BTNS_AUTOSIZE button sizes */ rebuild_toolbar(&hToolbar); pImageList_Destroy(himl); pImageList_Destroy(himl2); SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]); check_button_size(hToolbar, 7 + string_width(STRING2), 23 + fontheight); SendMessageA(hToolbar, TB_DELETEBUTTON, 0, 0); check_button_size(hToolbar, 7 + string_width(STRING2), 23 + fontheight); rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[0]) == 1, "TB_ADDBUTTONSA failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); rebuild_toolbar(&hToolbar); SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_LIST | GetWindowLongA(hToolbar, GWL_STYLE)); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons2[0]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[2]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes_todo(0xff); rebuild_toolbar(&hToolbar); SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_LIST | GetWindowLongA(hToolbar, GWL_STYLE)); ok(SendMessageA(hToolbar, TB_SETBITMAPSIZE, 0, MAKELPARAM(32, 32)) == 1, "TB_SETBITMAPSIZE failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(100, 100)) == 1, "TB_SETBUTTONSIZE failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); rebuild_toolbar(&hToolbar); SetWindowLongA(hToolbar, GWL_STYLE, TBSTYLE_WRAPABLE | GetWindowLongA(hToolbar, GWL_STYLE)); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons3[3]) == 1, "TB_ADDBUTTONSA failed\n"); tbinfo.cx = 672; tbinfo.cbSize = sizeof(TBBUTTONINFOA); tbinfo.dwMask = TBIF_SIZE | TBIF_BYINDEX; if (SendMessageA(hToolbar, TB_SETBUTTONINFOA, 0, (LPARAM)&tbinfo)) { ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 1, (LPARAM)&tbinfo) != 0, "TB_SETBUTTONINFOA failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0); check_sizes(); } else /* TBIF_BYINDEX probably not supported, confirm that this was the reason for the failure */ { tbinfo.dwMask = TBIF_SIZE; ok(SendMessageA(hToolbar, TB_SETBUTTONINFOA, 33, (LPARAM)&tbinfo) != 0, "TB_SETBUTTONINFOA failed\n"); tbsize_numtests++; } /* Single BTNS_AUTOSIZE button with string. */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[0]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); /* Single non-BTNS_AUTOSIZE button with string. */ rebuild_toolbar(&hToolbar); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[1]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); /* Single non-BTNS_AUTOSIZE button with string with TBSTYLE_EX_MIXEDBUTTONS set. */ rebuild_toolbar(&hToolbar); SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[1]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); /* Single non-BTNS_AUTOSIZE, BTNS_SHOWTEXT button with string with TBSTYLE_EX_MIXEDBUTTONS set. */ rebuild_toolbar(&hToolbar); SendMessageA(hToolbar, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_MIXEDBUTTONS); ok(SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)&buttons4[2]) == 1, "TB_ADDBUTTONSA failed\n"); ok(SendMessageA(hToolbar, TB_SETBUTTONSIZE, 0, MAKELPARAM(40, 20)) == 1, "TB_SETBUTTONSIZE failed\n"); SendMessageA(hToolbar, TB_AUTOSIZE, 0, 0 ); check_sizes(); /* Toolbar with borders around client area */ rebuild_toolbar_ex(&hToolbar, WS_EX_DLGMODALFRAME); SetWindowLongA(hToolbar, GWL_STYLE, CCS_NODIVIDER | GetWindowLongA(hToolbar, GWL_STYLE)); SendMessageA(hToolbar, TB_ADDBUTTONSA, 1, (LPARAM)buttons1); check_sizes(); GetClientRect(hToolbar, &rect); ok(rect.top == 0, "rect.top = %ld\n", rect.top); ok(rect.bottom == 26, "rect.bottom = %ld\n", rect.bottom); rebuild_toolbar_ex(&hToolbar, WS_EX_DLGMODALFRAME); GetClientRect(hToolbar, &rect); ok(rect.top == 0, "rect.top = %ld\n", rect.top); ok(rect.bottom == 26, "rect.bottom = %ld\n", rect.bottom); free_tbsize_results(); DestroyWindow(hToolbar); } /* Toolbar control has two ways of reacting to a change. We call them a * relayout and recalc. A recalc forces a recompute of values like button size * and top margin (the latter in comctl32 style & TBSTYLE_TOOLTIPS) == 0, "0x%08lx\n", cs->style); ret = CallWindowProcA(oldproc, hwnd, msg, wParam, lParam); style = GetWindowLongA(hwnd, GWL_STYLE); ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08lx\n", style); /* test if toolbar-specific messages are already working before WM_CREATE */ style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0); ok(style != 0, "got %lx\n", style); ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%lx\n", style); return ret; } } return CallWindowProcA(oldproc, hwnd, msg, wParam, lParam); } static LRESULT CALLBACK cbt_hook_proc(int code, WPARAM wParam, LPARAM lParam) { if (code == HCBT_CREATEWND) { HWND hwnd = (HWND)wParam; if (!g_toolbar) { WNDPROC oldproc; CHECK_EXPECT2(g_hook_create); g_toolbar = hwnd; /* subclass */ oldproc = (WNDPROC)SetWindowLongPtrA(hwnd, GWLP_WNDPROC, (LONG_PTR)toolbar_subclass_proc); SetWindowLongPtrA(hwnd, GWLP_USERDATA, (LONG_PTR)oldproc); } return 0; } return CallNextHookEx(g_tbhook, code, wParam, lParam); } static void test_create(BOOL v6) { HWND hwnd, tooltip; DWORD style; g_tbhook = SetWindowsHookA(WH_CBT, cbt_hook_proc); SET_EXPECT(g_hook_create); SET_EXPECT(g_hook_WM_NCCREATE); SET_EXPECT(g_hook_WM_CREATE); hwnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE, 0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL); if (v6) { expect(called_g_hook_create, FALSE); expect(called_g_hook_WM_NCCREATE, FALSE); expect(called_g_hook_WM_CREATE, FALSE); } else { CHECK_CALLED(g_hook_create); CHECK_CALLED(g_hook_WM_NCCREATE); CHECK_CALLED(g_hook_WM_CREATE); } style = GetWindowLongA(hwnd, GWL_STYLE); if (v6) { ok(!(style & TBSTYLE_TOOLTIPS), "got 0x%08lx\n", style); } else { ok((style & TBSTYLE_TOOLTIPS) == TBSTYLE_TOOLTIPS, "got 0x%08lx\n", style); tooltip = (HWND)SendMessageA(hwnd, TB_GETTOOLTIPS, 0, 0); ok(tooltip != NULL, "got %p\n", tooltip); ok(GetParent(tooltip) == hMainWnd, "got %p, %p\n", hMainWnd, hwnd); } DestroyWindow(hwnd); UnhookWindowsHook(WH_CBT, cbt_hook_proc); /* TBSTYLE_TRANSPARENT */ hwnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD|WS_VISIBLE|WS_CLIPSIBLINGS|TBSTYLE_FLAT|TBSTYLE_TOOLTIPS|TBSTYLE_GROUP, 0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL); style = GetWindowLongA(hwnd, GWL_STYLE); ok((style & TBSTYLE_TRANSPARENT) == TBSTYLE_TRANSPARENT, "got 0x%08lx\n", style); style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0); ok((style & TBSTYLE_TRANSPARENT) == TBSTYLE_TRANSPARENT, "got 0x%08lx\n", style); DestroyWindow(hwnd); if (!is_theme_active) { skip("Theming is not active, skipping following tests.\n"); return; } hwnd = CreateWindowA(TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE | WS_CLIPSIBLINGS | TBSTYLE_TOOLTIPS | TBSTYLE_GROUP, 0, 0, 0, 0, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL); style = GetWindowLongA(hwnd, GWL_STYLE); ok(!(style & TBSTYLE_TRANSPARENT), "got 0x%08lx\n", style); style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0); ok(!(style & TBSTYLE_TRANSPARENT), "got 0x%08lx\n", style); DestroyWindow(hwnd); } typedef struct { DWORD mask; DWORD style; DWORD style_set; } extended_style_t; static const extended_style_t extended_style_test[] = { { TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER }, { TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_DRAWDDARROWS | TBSTYLE_EX_HIDECLIPPEDBUTTONS | TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS }, { 0, TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS }, { 0, 0, 0 }, { 0, TBSTYLE_EX_DRAWDDARROWS, TBSTYLE_EX_DRAWDDARROWS }, { 0, TBSTYLE_EX_HIDECLIPPEDBUTTONS, TBSTYLE_EX_HIDECLIPPEDBUTTONS }, { 0, 0, 0 }, { TBSTYLE_EX_HIDECLIPPEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS, 0 }, { TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_HIDECLIPPEDBUTTONS, 0 }, { TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_MIXEDBUTTONS, 0 }, { TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_MIXEDBUTTONS }, { TBSTYLE_EX_DOUBLEBUFFER | TBSTYLE_EX_MIXEDBUTTONS, TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_DOUBLEBUFFER } }; static void test_TB_GET_SET_EXTENDEDSTYLE(void) { DWORD style, oldstyle, oldstyle2; const extended_style_t *ptr; HWND hwnd = NULL; int i; rebuild_toolbar(&hwnd); SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, TBSTYLE_EX_DOUBLEBUFFER, TBSTYLE_EX_MIXEDBUTTONS); style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0); if (style == TBSTYLE_EX_MIXEDBUTTONS) { win_skip("Some extended style bits are not supported\n"); DestroyWindow(hwnd); return; } for (i = 0; i < ARRAY_SIZE(extended_style_test); i++) { ptr = &extended_style_test[i]; oldstyle2 = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0); oldstyle = SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, ptr->mask, ptr->style); ok(oldstyle == oldstyle2, "%d: got old style 0x%08lx, expected 0x%08lx\n", i, oldstyle, oldstyle2); style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0); ok(style == ptr->style_set, "%d: got style 0x%08lx, expected 0x%08lx\n", i, style, ptr->style_set); } /* Windows sets CCS_VERT when TB_GETEXTENDEDSTYLE is set */ oldstyle2 = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0); oldstyle = SendMessageA(hwnd, TB_SETEXTENDEDSTYLE, 0, TBSTYLE_EX_VERTICAL); ok(oldstyle == oldstyle2, "got old style 0x%08lx, expected 0x%08lx\n", oldstyle, oldstyle2); style = SendMessageA(hwnd, TB_GETEXTENDEDSTYLE, 0, 0); ok(style == TBSTYLE_EX_VERTICAL, "got style 0x%08lx, expected 0x%08x\n", style, TBSTYLE_EX_VERTICAL); style = SendMessageA(hwnd, TB_GETSTYLE, 0, 0); todo_wine ok(style == CCS_VERT, "got style 0x%08lx, expected CCS_VERT\n", style); DestroyWindow(hwnd); } static void test_noresize(void) { HWND wnd; int i; TBBUTTON button = {0, 10, TBSTATE_ENABLED, 0, {0, }, 0, -1}; wnd = CreateWindowExA(0, TOOLBARCLASSNAMEA, NULL, WS_CHILD | WS_VISIBLE | CCS_NORESIZE | TBSTYLE_WRAPABLE, 0, 0, 100, 20, hMainWnd, (HMENU)5, GetModuleHandleA(NULL), NULL); SendMessageA(wnd, TB_BUTTONSTRUCTSIZE, sizeof(TBBUTTON), 0); for (i=0; i<30; i++) { button.idCommand = 10 + i; SendMessageA(wnd, TB_ADDBUTTONSA, 1, (LPARAM)&button); } SendMessageA(wnd, TB_SETSTATE, 10, TBSTATE_WRAP|TBSTATE_ENABLED); /* autosize clears the wrap on button 0 */ SendMessageA(wnd, TB_AUTOSIZE, 0, 0); for (i=0; i<30; i++) { SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button); if (i % 4 == 3) ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState); else ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState); } /* changing the parent doesn't do anything */ MoveWindow(hMainWnd, 0,0, 400, 200, FALSE); for (i=0; i<30; i++) { SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button); if (i % 4 == 3) ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState); else ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState); } /* again nothing here */ SendMessageA(wnd, TB_AUTOSIZE, 0, 0); for (i=0; i<30; i++) { SendMessageA(wnd, TB_GETBUTTON, i, (LPARAM)&button); if (i % 4 == 3) ok(button.fsState == (TBSTATE_WRAP|TBSTATE_ENABLED), "%d: got %08x\n", i, button.fsState); else ok(button.fsState == TBSTATE_ENABLED, "%d: got %08x\n", i, button.fsState); } DestroyWindow(wnd); } static void test_save(void) { HWND wnd = NULL; TBSAVEPARAMSW params; LONG res; HKEY key; BYTE data[100]; DWORD size = sizeof(data), type, i, count; TBBUTTON tb; static const TBBUTTON more_btns[2] = { {0, 11, TBSTATE_HIDDEN, BTNS_BUTTON, {0}, 0, -1}, {0, 13, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0, -1} }; static const DWORD expect[] = {0xcafe, 1, 0xcafe0000, 3, 0xcafe0001, 5, 0xcafe0002, 7, 0xcafe0003, 9, 0xcafe0004, 11, 0xcafe0005, 13, 0xcafe0006, 0xffffffff, 0xcafe0007, 0xfffffffe, 0xcafe0008, 0x80000000, 0xcafe0009, 0x7fffffff, 0xcafe000a, 0x100, 0xcafe000b}; static const TBBUTTON expect_btns[] = { {0, 1, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0, 0}, {0, 3, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 1, 2}, {0, 5, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 2, 0}, {0, 7, 0, BTNS_BUTTON, {0}, 0, (INT_PTR)"foo"}, {0, 9, 0, BTNS_BUTTON, {0}, 0, 0}, {0, 11, 0, BTNS_BUTTON, {0}, 0, 3}, {0, 13, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 6, 0}, {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 7, 0}, {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 8, 0}, {0, 0, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 9, 0}, {0, 0x7fffffff, TBSTATE_ENABLED, BTNS_BUTTON, {0}, 0xa, 0}, }; params.hkr = HKEY_CURRENT_USER; params.pszSubKey = L"Software\\Wine\\WineTest"; params.pszValueName = L"toolbartest"; rebuild_toolbar_with_buttons( &wnd ); SendMessageW(wnd, TB_ADDBUTTONSW, ARRAY_SIZE(more_btns), (LPARAM)more_btns); flush_sequences(sequences, NUM_MSG_SEQUENCES); res = SendMessageW( wnd, TB_SAVERESTOREW, TRUE, (LPARAM)¶ms ); ok( res, "saving failed\n" ); ok_sequence(sequences, PARENT_SEQ_INDEX, save_parent_seq, "save", FALSE); DestroyWindow( wnd ); res = RegOpenKeyW( HKEY_CURRENT_USER, L"Software\\Wine\\WineTest", &key ); ok( !res, "got %08lx\n", res ); res = RegQueryValueExW( key, L"toolbartest", NULL, &type, data, &size ); ok( !res, "got %08lx\n", res ); ok( type == REG_BINARY, "got %08lx\n", type ); ok( size == sizeof(expect), "got %08lx\n", size ); ok( !memcmp( data, expect, size ), "mismatch\n" ); RegCloseKey( key ); wnd = NULL; rebuild_toolbar( &wnd ); flush_sequences(sequences, NUM_MSG_SEQUENCES); res = SendMessageW( wnd, TB_SAVERESTOREW, FALSE, (LPARAM)¶ms ); ok( res, "restoring failed\n" ); ok_sequence(sequences, PARENT_SEQ_INDEX, restore_parent_seq, "restore", FALSE); count = SendMessageW( wnd, TB_BUTTONCOUNT, 0, 0 ); ok( count == ARRAY_SIZE(expect_btns), "got %ld\n", count ); for (i = 0; i < count; i++) { res = SendMessageW( wnd, TB_GETBUTTON, i, (LPARAM)&tb ); ok( res, "got %ld\n", res ); ok( tb.iBitmap == expect_btns[i].iBitmap, "%ld: got %d\n", i, tb.iBitmap ); ok( tb.idCommand == expect_btns[i].idCommand, "%ld: got %d\n", i, tb.idCommand ); ok( tb.fsState == expect_btns[i].fsState, "%ld: got %02x\n", i, tb.fsState ); ok( tb.fsStyle == expect_btns[i].fsStyle, "%ld: got %02x\n", i, tb.fsStyle ); ok( tb.dwData == expect_btns[i].dwData, "%ld: got %Ix\n", i, tb.dwData ); if (IS_INTRESOURCE(expect_btns[i].iString)) ok( tb.iString == expect_btns[i].iString, "%ld: got %Ix\n", i, tb.iString ); else ok( !strcmp( (char *)tb.iString, (char *)expect_btns[i].iString ), "%ld: got %s\n", i, (char *)tb.iString ); /* In fact the ptr value set in TBN_GETBUTTONINFOA is simply copied */ if (tb.idCommand == 7) ok( tb.iString == (INT_PTR)alloced_str, "string not set\n"); } DestroyWindow( wnd ); RegOpenKeyW( HKEY_CURRENT_USER, L"Software\\Wine\\WineTest", &key ); RegDeleteValueW( key, L"toolbartest" ); RegCloseKey( key ); } static void test_drawtext_flags(void) { HWND hwnd = NULL; UINT flags; rebuild_toolbar(&hwnd); flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, 0); todo_wine ok(flags == 0, "Unexpected draw text flags %#x\n", flags); /* zero mask, flags are retained */ flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, DT_BOTTOM); todo_wine ok(flags == 0, "Unexpected draw text flags %#x\n", flags); ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n"); flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, 0, 0); todo_wine ok(flags == 0, "Unexpected draw text flags %#x\n", flags); ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n"); /* set/remove */ flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, DT_BOTTOM); todo_wine ok(flags == 0, "Unexpected draw text flags %#x\n", flags); ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n"); flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, 0); todo_wine ok(flags == DT_BOTTOM, "Unexpected draw text flags %#x\n", flags); ok(flags & DT_BOTTOM, "Expected DT_BOTTOM style, %#x\n", flags); flags = SendMessageA(hwnd, TB_SETDRAWTEXTFLAGS, DT_BOTTOM, 0); todo_wine ok(flags == 0, "Unexpected draw text flags %#x\n", flags); ok(!(flags & DT_BOTTOM), "Unexpected DT_BOTTOM style\n"); DestroyWindow(hwnd); } static void test_imagelist(void) { HIMAGELIST imagelist; HWND hwnd = NULL; int ret; rebuild_toolbar(&hwnd); imagelist = (HIMAGELIST)SendMessageA(hwnd, TB_GETIMAGELIST, 0, 0); ok(imagelist == NULL, "got %p\n", imagelist); ret = SendMessageA(hwnd, TB_SETBITMAPSIZE, 0, MAKELONG(16, 16)); ok(ret, "got %d\n", ret); imagelist = (HIMAGELIST)SendMessageA(hwnd, TB_GETIMAGELIST, 0, 0); ok(imagelist == NULL, "got %p\n", imagelist); DestroyWindow(hwnd); } static void test_visual(void) { HBITMAP mem_bitmap1, mem_bitmap2; HDC mem_dc1, mem_dc2, toolbar_dc; TBBUTTON tbbutton; int width, height; HTHEME theme; HWND toolbar; RECT rect; BOOL ret; if (!is_theme_active) { skip("Theming is not active, skipping visual tests.\n"); return; } /* Test that toolbar shouldn't use outside theme handles */ toolbar = CreateWindowA(TOOLBARCLASSNAMEA, "", WS_CHILD | WS_VISIBLE, 0, 0, 50, 50, hMainWnd, 0, 0, NULL); ok(!!toolbar, "Failed to create a toolbar window.\n"); /* Toolbar needs data for it to show */ memset(&tbbutton, 0, sizeof(tbbutton)); tbbutton.fsState = TBSTATE_ENABLED; tbbutton.iString = (INT_PTR)"test"; SendMessageA(toolbar, TB_BUTTONSTRUCTSIZE, sizeof(tbbutton), 0); ret = SendMessageA(toolbar, TB_ADDBUTTONSA, 1, (LPARAM)&tbbutton); ok(ret, "TB_ADDBUTTONSA failed.\n"); theme = pGetWindowTheme(toolbar); ok(!theme, "Expected theme not opened by window.\n"); toolbar_dc = GetDC(toolbar); GetClientRect(toolbar, &rect); width = rect.right - rect.left; height = rect.bottom - rect.top; ret = RedrawWindow(toolbar, NULL, NULL, RDW_FRAME | RDW_INVALIDATE | RDW_ERASE | RDW_ERASENOW | RDW_UPDATENOW); ok(ret, "RedrawWindow failed.\n"); flush_events(); mem_dc1 = CreateCompatibleDC(toolbar_dc); mem_bitmap1 = CreateCompatibleBitmap(toolbar_dc, width, height); SelectObject(mem_dc1, mem_bitmap1); BitBlt(mem_dc1, 0, 0, width, height, toolbar_dc, 0, 0, SRCCOPY); if (theme) pCloseThemeData(theme); theme = pOpenThemeData(toolbar, L"Rebar"); ok(!!theme, "OpenThemeData failed.\n"); ret = RedrawWindow(toolbar, NULL, NULL, RDW_FRAME | RDW_INVALIDATE | RDW_ERASE | RDW_ERASENOW | RDW_UPDATENOW); ok(ret, "RedrawWindow failed.\n"); mem_dc2 = CreateCompatibleDC(toolbar_dc); mem_bitmap2 = CreateCompatibleBitmap(toolbar_dc, width, height); SelectObject(mem_dc2, mem_bitmap2); BitBlt(mem_dc2, 0, 0, width, height, toolbar_dc, 0, 0, SRCCOPY); ret = equal_dc(mem_dc1, mem_dc2, width, height); ok(ret, "Expected same content.\n"); pCloseThemeData(theme); DeleteObject(mem_bitmap2); DeleteObject(mem_bitmap1); DeleteDC(mem_dc2); DeleteDC(mem_dc1); ReleaseDC(toolbar, toolbar_dc); DestroyWindow(toolbar); } static void init_functions(void) { HMODULE hComCtl32, hUxtheme; hComCtl32 = LoadLibraryA("comctl32.dll"); hUxtheme = LoadLibraryA("uxtheme.dll"); #define X(f) p##f = (void*)GetProcAddress(hComCtl32, #f); X(CreateToolbarEx); X(ImageList_GetIconSize); X(ImageList_GetImageCount); X(ImageList_LoadImageA); X(ImageList_Destroy); #undef X #define X(f) p##f = (void*)GetProcAddress(hUxtheme, #f) X(GetWindowTheme); X(IsThemeActive); X(OpenThemeData); X(CloseThemeData); #undef X } static void test_BTNS_SEP(void) { TBBUTTON buttons[1], button; LRESULT ret; HWND hwnd; rebuild_toolbar(&hwnd); /* 0 iBitmap */ memset(buttons, 0, sizeof(buttons)); buttons[0].idCommand = 5; buttons[0].fsStyle = BTNS_SEP; buttons[0].fsState = TBSTATE_ENABLED; buttons[0].iString = -1; ret = SendMessageA(hwnd, TB_ADDBUTTONSA, 1, (LPARAM)buttons); ok(ret == 1, "Unexpected return value.\n"); ret = SendMessageA(hwnd, TB_AUTOSIZE, 0, 0); ok(!ret, "Unexpected return value.\n"); memset(&button, 0, sizeof(button)); ret = SendMessageA(hwnd, TB_GETBUTTON, 0, (LPARAM)&button); ok(ret == 1, "Unexpected return value.\n"); todo_wine ok(button.iBitmap == 8, "Unexpected iBitmap value %d.\n", button.iBitmap); rebuild_toolbar(&hwnd); /* Nonzero iBitmap */ memset(buttons, 0, sizeof(buttons)); buttons[0].idCommand = 5; buttons[0].fsStyle = BTNS_SEP; buttons[0].fsState = TBSTATE_ENABLED; buttons[0].iString = -1; buttons[0].iBitmap = 2; ret = SendMessageA(hwnd, TB_ADDBUTTONSA, 1, (LPARAM)buttons); ok(ret == 1, "Unexpected return value.\n"); ret = SendMessageA(hwnd, TB_AUTOSIZE, 0, 0); ok(!ret, "Unexpected return value.\n"); memset(&button, 0, sizeof(button)); ret = SendMessageA(hwnd, TB_GETBUTTON, 0, (LPARAM)&button); ok(ret == 1, "Unexpected return value.\n"); ok(button.iBitmap == 2, "Unexpected iBitmap value %d.\n", button.iBitmap); rebuild_toolbar(&hwnd); /* Exceeds default width */ memset(buttons, 0, sizeof(buttons)); buttons[0].idCommand = 5; buttons[0].fsStyle = BTNS_SEP; buttons[0].fsState = TBSTATE_ENABLED; buttons[0].iString = -1; buttons[0].iBitmap = 32; ret = SendMessageA(hwnd, TB_ADDBUTTONSA, 1, (LPARAM)buttons); ok(ret == 1, "Unexpected return value.\n"); ret = SendMessageA(hwnd, TB_AUTOSIZE, 0, 0); ok(!ret, "Unexpected return value.\n"); memset(&button, 0, sizeof(button)); ret = SendMessageA(hwnd, TB_GETBUTTON, 0, (LPARAM)&button); ok(ret == 1, "Unexpected return value.\n"); ok(button.iBitmap == 32, "Unexpected iBitmap value %d.\n", button.iBitmap); rebuild_toolbar(&hwnd); /* Negative */ memset(buttons, 0, sizeof(buttons)); buttons[0].idCommand = 5; buttons[0].fsStyle = BTNS_SEP; buttons[0].fsState = TBSTATE_ENABLED; buttons[0].iString = -1; buttons[0].iBitmap = -3; ret = SendMessageA(hwnd, TB_ADDBUTTONSA, 1, (LPARAM)buttons); ok(ret == 1, "Unexpected return value.\n"); ret = SendMessageA(hwnd, TB_AUTOSIZE, 0, 0); ok(!ret, "Unexpected return value.\n"); memset(&button, 0, sizeof(button)); ret = SendMessageA(hwnd, TB_GETBUTTON, 0, (LPARAM)&button); ok(ret == 1, "Unexpected return value.\n"); todo_wine ok(button.iBitmap == 8, "Unexpected iBitmap value %d.\n", button.iBitmap); DestroyWindow(hwnd); } START_TEST(toolbar) { ULONG_PTR ctx_cookie; WNDCLASSA wc; HANDLE ctx; MSG msg; RECT rc; init_msg_sequences(sequences, NUM_MSG_SEQUENCES); init_functions(); if (pIsThemeActive) is_theme_active = pIsThemeActive(); wc.style = CS_HREDRAW | CS_VREDRAW; wc.cbClsExtra = 0; wc.cbWndExtra = 0; wc.hInstance = GetModuleHandleA(NULL); wc.hIcon = NULL; wc.hCursor = LoadCursorA(NULL, (LPCSTR)IDC_IBEAM); wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW); wc.lpszMenuName = NULL; wc.lpszClassName = "Toolbar test parent"; wc.lpfnWndProc = parent_wnd_proc; RegisterClassA(&wc); hMainWnd = CreateWindowExA(0, "Toolbar test parent", "Blah", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT, CW_USEDEFAULT, 680, 260, NULL, NULL, GetModuleHandleA(NULL), 0); GetClientRect(hMainWnd, &rc); ShowWindow(hMainWnd, SW_SHOW); basic_test(); test_add_bitmap(); test_add_string(); test_hotitem(); test_sizes(); test_recalc(); test_getbuttoninfo(); test_createtoolbarex(); test_dispinfo(); test_setrows(); test_getstring(); test_tooltip(); test_get_set_style(); test_create(FALSE); test_TB_GET_SET_EXTENDEDSTYLE(); test_noresize(); test_save(); test_drawtext_flags(); test_imagelist(); test_BTNS_SEP(); if (!load_v6_module(&ctx_cookie, &ctx)) return; test_create(TRUE); test_visual(); test_BTNS_SEP(); PostQuitMessage(0); while(GetMessageA(&msg,0,0,0)) { TranslateMessage(&msg); DispatchMessageA(&msg); } DestroyWindow(hMainWnd); unload_v6_module(ctx_cookie, ctx); }