urlmon: Don't create stgmed_obj for binding to object.
[wine] / dlls / user32 / tests / combo.c
1 /* Unit test suite for combo boxes.
2  *
3  * Copyright 2007 Mikolaj Zalewski
4  *
5  * This library is free software; you can redistribute it and/or
6  * modify it under the terms of the GNU Lesser General Public
7  * License as published by the Free Software Foundation; either
8  * version 2.1 of the License, or (at your option) any later version.
9  *
10  * This library is distributed in the hope that it will be useful,
11  * but WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * Lesser General Public License for more details.
14  *
15  * You should have received a copy of the GNU Lesser General Public
16  * License along with this library; if not, write to the Free Software
17  * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
18  */
19
20 #include <assert.h>
21 #include <stdarg.h>
22 #include <stdio.h>
23
24 #define STRICT
25 #define WIN32_LEAN_AND_MEAN
26 #include <windows.h>
27
28 #include "wine/test.h"
29
30 #define COMBO_ID 1995
31
32 static HWND hMainWnd;
33
34 #define expect_eq(expr, value, type, fmt); { type val = expr; ok(val == (value), #expr " expected " #fmt " got " #fmt "\n", (value), val); }
35 #define expect_rect(r, _left, _top, _right, _bottom) ok(r.left == _left && r.top == _top && \
36     r.bottom == _bottom && r.right == _right, "Invalid rect (%d,%d) (%d,%d) vs (%d,%d) (%d,%d)\n", \
37     r.left, r.top, r.right, r.bottom, _left, _top, _right, _bottom);
38
39 static HWND build_combo(DWORD style)
40 {
41     return CreateWindow("ComboBox", "Combo", WS_VISIBLE|WS_CHILD|style, 5, 5, 100, 100, hMainWnd, (HMENU)COMBO_ID, NULL, 0);
42 }
43
44 static int font_height(HFONT hFont)
45 {
46     TEXTMETRIC tm;
47     HFONT hFontOld;
48     HDC hDC;
49
50     hDC = CreateCompatibleDC(NULL);
51     hFontOld = SelectObject(hDC, hFont);
52     GetTextMetrics(hDC, &tm);
53     SelectObject(hDC, hFontOld);
54     DeleteDC(hDC);
55
56     return tm.tmHeight;
57 }
58
59 static INT CALLBACK is_font_installed_proc(const LOGFONT *elf, const TEXTMETRIC *tm, DWORD type, LPARAM lParam)
60 {
61     return 0;
62 }
63
64 static int is_font_installed(const char *name)
65 {
66     HDC hdc = GetDC(NULL);
67     BOOL ret = !EnumFontFamilies(hdc, name, is_font_installed_proc, 0);
68     ReleaseDC(NULL, hdc);
69     return ret;
70 }
71
72 static void test_setitemheight(DWORD style)
73 {
74     HWND hCombo = build_combo(style);
75     RECT r;
76     int i;
77
78     trace("Style %x\n", style);
79     GetClientRect(hCombo, &r);
80     expect_rect(r, 0, 0, 100, 24);
81     SendMessageA(hCombo, CB_GETDROPPEDCONTROLRECT, 0, (LPARAM)&r);
82     MapWindowPoints(HWND_DESKTOP, hMainWnd, (LPPOINT)&r, 2);
83     todo_wine expect_rect(r, 5, 5, 105, 105);
84
85     for (i = 1; i < 30; i++)
86     {
87         SendMessage(hCombo, CB_SETITEMHEIGHT, -1, i);
88         GetClientRect(hCombo, &r);
89         expect_eq(r.bottom - r.top, i + 6, int, "%d");
90     }
91
92     DestroyWindow(hCombo);
93 }
94
95 static void test_setfont(DWORD style)
96 {
97     HWND hCombo = build_combo(style);
98     HFONT hFont1 = CreateFont(10, 0, 0, 0, FW_DONTCARE, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH|FF_DONTCARE, "Marlett");
99     HFONT hFont2 = CreateFont(8, 0, 0, 0, FW_DONTCARE, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH|FF_DONTCARE, "Marlett");
100     RECT r;
101     int i;
102
103     trace("Style %x\n", style);
104     GetClientRect(hCombo, &r);
105     expect_rect(r, 0, 0, 100, 24);
106     SendMessageA(hCombo, CB_GETDROPPEDCONTROLRECT, 0, (LPARAM)&r);
107     MapWindowPoints(HWND_DESKTOP, hMainWnd, (LPPOINT)&r, 2);
108     todo_wine expect_rect(r, 5, 5, 105, 105);
109
110     if (!is_font_installed("Marlett"))
111     {
112         skip("Marlett font not available\n");
113         DestroyWindow(hCombo);
114         DeleteObject(hFont1);
115         DeleteObject(hFont2);
116         return;
117     }
118
119     if (font_height(hFont1) == 10 && font_height(hFont2) == 8)
120     {
121         SendMessage(hCombo, WM_SETFONT, (WPARAM)hFont1, FALSE);
122         GetClientRect(hCombo, &r);
123         expect_rect(r, 0, 0, 100, 18);
124         SendMessageA(hCombo, CB_GETDROPPEDCONTROLRECT, 0, (LPARAM)&r);
125         MapWindowPoints(HWND_DESKTOP, hMainWnd, (LPPOINT)&r, 2);
126         todo_wine expect_rect(r, 5, 5, 105, 99);
127
128         SendMessage(hCombo, WM_SETFONT, (WPARAM)hFont2, FALSE);
129         GetClientRect(hCombo, &r);
130         expect_rect(r, 0, 0, 100, 16);
131         SendMessageA(hCombo, CB_GETDROPPEDCONTROLRECT, 0, (LPARAM)&r);
132         MapWindowPoints(HWND_DESKTOP, hMainWnd, (LPPOINT)&r, 2);
133         todo_wine expect_rect(r, 5, 5, 105, 97);
134
135         SendMessage(hCombo, WM_SETFONT, (WPARAM)hFont1, FALSE);
136         GetClientRect(hCombo, &r);
137         expect_rect(r, 0, 0, 100, 18);
138         SendMessageA(hCombo, CB_GETDROPPEDCONTROLRECT, 0, (LPARAM)&r);
139         MapWindowPoints(HWND_DESKTOP, hMainWnd, (LPPOINT)&r, 2);
140         todo_wine expect_rect(r, 5, 5, 105, 99);
141     }
142     else
143         skip("Invalid Marlett font heights\n");
144
145     for (i = 1; i < 30; i++)
146     {
147         HFONT hFont = CreateFont(i, 0, 0, 0, FW_DONTCARE, FALSE, FALSE, FALSE, ANSI_CHARSET, OUT_DEFAULT_PRECIS, CLIP_DEFAULT_PRECIS, DEFAULT_QUALITY, DEFAULT_PITCH|FF_DONTCARE, "Marlett");
148         int height = font_height(hFont);
149
150         SendMessage(hCombo, WM_SETFONT, (WPARAM)hFont, FALSE);
151         GetClientRect(hCombo, &r);
152         expect_eq(r.bottom - r.top, height + 8, int, "%d");
153         SendMessage(hCombo, WM_SETFONT, 0, FALSE);
154         DeleteObject(hFont);
155     }
156
157     DestroyWindow(hCombo);
158     DeleteObject(hFont1);
159     DeleteObject(hFont2);
160 }
161
162 static LRESULT (CALLBACK *old_parent_proc)(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam);
163 static LPCSTR expected_edit_text;
164 static LPCSTR expected_list_text;
165
166 static LRESULT CALLBACK parent_wnd_proc(HWND hwnd, UINT msg, WPARAM wparam, LPARAM lparam)
167 {
168     switch (msg)
169     {
170     case WM_COMMAND:
171         switch (wparam)
172         {
173             case MAKEWPARAM(COMBO_ID, CBN_SELCHANGE):
174             {
175                 HWND hCombo = (HWND)lparam;
176                 int idx;
177                 char list[20], edit[20];
178
179                 memset(list, 0, sizeof(list));
180                 memset(edit, 0, sizeof(edit));
181
182                 idx = SendMessage(hCombo, CB_GETCURSEL, 0, 0);
183                 SendMessage(hCombo, CB_GETLBTEXT, idx, (LPARAM)list);
184                 SendMessage(hCombo, WM_GETTEXT, sizeof(edit), (LPARAM)edit);
185
186                 ok(!strcmp(edit, expected_edit_text), "edit: got %s, expected %s\n",
187                    edit, expected_edit_text);
188                 ok(!strcmp(list, expected_list_text), "list: got %s, expected %s\n",
189                    list, expected_list_text);
190             }
191             break;
192         }
193         break;
194     }
195
196     return CallWindowProc(old_parent_proc, hwnd, msg, wparam, lparam);
197 }
198
199 static void test_selection(DWORD style, const char * const text[],
200                            const int *edit, const int *list)
201 {
202     INT idx;
203     HWND hCombo;
204
205     hCombo = build_combo(style);
206
207     SendMessage(hCombo, CB_ADDSTRING, 0, (LPARAM)text[0]);
208     SendMessage(hCombo, CB_ADDSTRING, 0, (LPARAM)text[1]);
209     SendMessage(hCombo, CB_SETCURSEL, -1, 0);
210
211     old_parent_proc = (void *)SetWindowLongPtr(hMainWnd, GWLP_WNDPROC, (ULONG_PTR)parent_wnd_proc);
212
213     idx = SendMessage(hCombo, CB_GETCURSEL, 0, 0);
214     ok(idx == -1, "expected selection -1, got %d\n", idx);
215
216     expected_list_text = text[list[0]];
217     expected_edit_text = text[edit[0]];
218     SendMessage(hCombo, WM_KEYDOWN, VK_DOWN, 0);
219
220     expected_list_text = text[list[1]];
221     expected_edit_text = text[edit[1]];
222     SendMessage(hCombo, WM_KEYDOWN, VK_DOWN, 0);
223
224     expected_list_text = text[list[2]];
225     expected_edit_text = text[edit[2]];
226     SendMessage(hCombo, WM_KEYDOWN, VK_UP, 0);
227
228     SetWindowLongPtr(hMainWnd, GWLP_WNDPROC, (ULONG_PTR)old_parent_proc);
229     DestroyWindow(hCombo);
230 }
231
232 static void test_CBN_SELCHANGE(void)
233 {
234     static const char * const text[] = { "alpha", "beta", "" };
235     static const int sel_1[] = { 2, 0, 1 };
236     static const int sel_2[] = { 0, 1, 0 };
237
238     test_selection(CBS_SIMPLE, text, sel_1, sel_2);
239     test_selection(CBS_DROPDOWN, text, sel_1, sel_2);
240     test_selection(CBS_DROPDOWNLIST, text, sel_2, sel_2);
241 }
242
243 START_TEST(combo)
244 {
245     hMainWnd = CreateWindow("static", "Test", WS_OVERLAPPEDWINDOW, 10, 10, 300, 300, NULL, NULL, NULL, 0);
246     ShowWindow(hMainWnd, SW_SHOW);
247
248     test_setfont(CBS_DROPDOWN);
249     test_setfont(CBS_DROPDOWNLIST);
250     test_setitemheight(CBS_DROPDOWN);
251     test_setitemheight(CBS_DROPDOWNLIST);
252     test_CBN_SELCHANGE();
253
254     DestroyWindow(hMainWnd);
255 }