[SHELL32_WINETEST] Sync with Wine Staging 3.3. CORE-14434
[reactos.git] / modules / rostests / winetests / shell32 / autocomplete.c
1 /*
2 * Tests for autocomplete
3 *
4 * Copyright 2008 Jan de Mooij
5 *
6 * This library is free software; you can redistribute it and/or
7 * modify it under the terms of the GNU Lesser General Public
8 * License as published by the Free Software Foundation; either
9 * version 2.1 of the License, or (at your option) any later version.
10 *
11 * This library is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
14 * Lesser General Public License for more details.
15 *
16 * You should have received a copy of the GNU Lesser General Public
17 * License along with this library; if not, write to the Free Software
18 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 */
20
21 #define COBJMACROS
22
23 #include <stdarg.h>
24
25 #include "windows.h"
26 #include "shobjidl.h"
27 #include "shlguid.h"
28 #include "initguid.h"
29 #include "shldisp.h"
30
31 #include "wine/heap.h"
32 #include "wine/test.h"
33
34 static HWND hMainWnd, hEdit;
35 static HINSTANCE hinst;
36 static int killfocus_count;
37
38 static void test_invalid_init(void)
39 {
40 HRESULT hr;
41 IAutoComplete *ac;
42 IUnknown *acSource;
43 HWND edit_control;
44
45 /* AutoComplete instance */
46 hr = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER,
47 &IID_IAutoComplete, (void **)&ac);
48 if (hr == REGDB_E_CLASSNOTREG)
49 {
50 win_skip("CLSID_AutoComplete is not registered\n");
51 return;
52 }
53 ok(hr == S_OK, "no IID_IAutoComplete (0x%08x)\n", hr);
54
55 /* AutoComplete source */
56 hr = CoCreateInstance(&CLSID_ACLMulti, NULL, CLSCTX_INPROC_SERVER,
57 &IID_IACList, (void **)&acSource);
58 if (hr == REGDB_E_CLASSNOTREG)
59 {
60 win_skip("CLSID_ACLMulti is not registered\n");
61 IAutoComplete_Release(ac);
62 return;
63 }
64 ok(hr == S_OK, "no IID_IACList (0x%08x)\n", hr);
65
66 edit_control = CreateWindowExA(0, "EDIT", "Some text", 0, 10, 10, 300, 300,
67 hMainWnd, NULL, hinst, NULL);
68 ok(edit_control != NULL, "Can't create edit control\n");
69
70 /* The refcount of acSource would be incremented on older Windows. */
71 hr = IAutoComplete_Init(ac, NULL, acSource, NULL, NULL);
72 ok(hr == E_INVALIDARG ||
73 broken(hr == S_OK), /* Win2k/XP/Win2k3 */
74 "Init returned 0x%08x\n", hr);
75 if (hr == E_INVALIDARG)
76 {
77 LONG ref;
78
79 IUnknown_AddRef(acSource);
80 ref = IUnknown_Release(acSource);
81 ok(ref == 1, "Expected AutoComplete source refcount to be 1, got %d\n", ref);
82 }
83
84 if (0)
85 {
86 /* Older Windows versions never check the window handle, while newer
87 * versions only check for NULL. Subsequent attempts to initialize the
88 * object after this call succeeds would fail, because initialization
89 * state is determined by whether a non-NULL window handle is stored. */
90 hr = IAutoComplete_Init(ac, (HWND)0xdeadbeef, acSource, NULL, NULL);
91 ok(hr == S_OK, "Init returned 0x%08x\n", hr);
92
93 /* Tests crash on older Windows. */
94 hr = IAutoComplete_Init(ac, NULL, NULL, NULL, NULL);
95 ok(hr == E_INVALIDARG, "Init returned 0x%08x\n", hr);
96
97 hr = IAutoComplete_Init(ac, edit_control, NULL, NULL, NULL);
98 ok(hr == E_INVALIDARG, "Init returned 0x%08x\n", hr);
99 }
100
101 /* bind to edit control */
102 hr = IAutoComplete_Init(ac, edit_control, acSource, NULL, NULL);
103 ok(hr == S_OK, "Init returned 0x%08x\n", hr);
104
105 /* try invalid parameters after successful initialization .*/
106 hr = IAutoComplete_Init(ac, NULL, NULL, NULL, NULL);
107 ok(hr == E_INVALIDARG ||
108 hr == E_FAIL, /* Win2k/XP/Win2k3 */
109 "Init returned 0x%08x\n", hr);
110
111 hr = IAutoComplete_Init(ac, NULL, acSource, NULL, NULL);
112 ok(hr == E_INVALIDARG ||
113 hr == E_FAIL, /* Win2k/XP/Win2k3 */
114 "Init returned 0x%08x\n", hr);
115
116 hr = IAutoComplete_Init(ac, edit_control, NULL, NULL, NULL);
117 ok(hr == E_INVALIDARG ||
118 hr == E_FAIL, /* Win2k/XP/Win2k3 */
119 "Init returned 0x%08x\n", hr);
120
121 /* try initializing twice on the same control */
122 hr = IAutoComplete_Init(ac, edit_control, acSource, NULL, NULL);
123 ok(hr == E_FAIL, "Init returned 0x%08x\n", hr);
124
125 /* try initializing with a different control */
126 hr = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
127 ok(hr == E_FAIL, "Init returned 0x%08x\n", hr);
128
129 DestroyWindow(edit_control);
130
131 /* try initializing with a different control after
132 * destroying the original initialization control */
133 hr = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
134 ok(hr == E_UNEXPECTED ||
135 hr == E_FAIL, /* Win2k/XP/Win2k3 */
136 "Init returned 0x%08x\n", hr);
137
138 IUnknown_Release(acSource);
139 IAutoComplete_Release(ac);
140 }
141 static IAutoComplete *test_init(void)
142 {
143 HRESULT r;
144 IAutoComplete *ac;
145 IUnknown *acSource;
146 LONG_PTR user_data;
147
148 /* AutoComplete instance */
149 r = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER,
150 &IID_IAutoComplete, (LPVOID*)&ac);
151 if (r == REGDB_E_CLASSNOTREG)
152 {
153 win_skip("CLSID_AutoComplete is not registered\n");
154 return NULL;
155 }
156 ok(r == S_OK, "no IID_IAutoComplete (0x%08x)\n", r);
157
158 /* AutoComplete source */
159 r = CoCreateInstance(&CLSID_ACLMulti, NULL, CLSCTX_INPROC_SERVER,
160 &IID_IACList, (LPVOID*)&acSource);
161 if (r == REGDB_E_CLASSNOTREG)
162 {
163 win_skip("CLSID_ACLMulti is not registered\n");
164 IAutoComplete_Release(ac);
165 return NULL;
166 }
167 ok(r == S_OK, "no IID_IACList (0x%08x)\n", r);
168
169 user_data = GetWindowLongPtrA(hEdit, GWLP_USERDATA);
170 ok(user_data == 0, "Expected the edit control user data to be zero\n");
171
172 /* bind to edit control */
173 r = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
174 ok(r == S_OK, "Init returned 0x%08x\n", r);
175
176 user_data = GetWindowLongPtrA(hEdit, GWLP_USERDATA);
177 ok(user_data == 0, "Expected the edit control user data to be zero\n");
178
179 IUnknown_Release(acSource);
180
181 return ac;
182 }
183
184 static void test_killfocus(void)
185 {
186 /* Test if WM_KILLFOCUS messages are handled properly by checking if
187 * the parent receives an EN_KILLFOCUS message. */
188 SetFocus(hEdit);
189 killfocus_count = 0;
190 SetFocus(0);
191 ok(killfocus_count == 1, "Expected one EN_KILLFOCUS message, got: %d\n", killfocus_count);
192 }
193
194 static LRESULT CALLBACK MyWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
195 {
196 switch(msg) {
197 case WM_CREATE:
198 /* create edit control */
199 hEdit = CreateWindowExA(0, "EDIT", "Some text", 0, 10, 10, 300, 300,
200 hWnd, NULL, hinst, NULL);
201 ok(hEdit != NULL, "Can't create edit control\n");
202 break;
203 case WM_COMMAND:
204 if(HIWORD(wParam) == EN_KILLFOCUS)
205 killfocus_count++;
206 break;
207 }
208 return DefWindowProcA(hWnd, msg, wParam, lParam);
209 }
210
211 static void createMainWnd(void)
212 {
213 WNDCLASSA wc;
214 wc.style = CS_HREDRAW | CS_VREDRAW;
215 wc.cbClsExtra = 0;
216 wc.cbWndExtra = 0;
217 wc.hInstance = GetModuleHandleA(NULL);
218 wc.hIcon = NULL;
219 wc.hCursor = LoadCursorA(NULL, (LPSTR)IDC_IBEAM);
220 wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
221 wc.lpszMenuName = NULL;
222 wc.lpszClassName = "MyTestWnd";
223 wc.lpfnWndProc = MyWndProc;
224 RegisterClassA(&wc);
225
226 hMainWnd = CreateWindowExA(0, "MyTestWnd", "Blah", WS_OVERLAPPEDWINDOW,
227 CW_USEDEFAULT, CW_USEDEFAULT, 130, 105, NULL, NULL, GetModuleHandleA(NULL), 0);
228 }
229
230 struct string_enumerator
231 {
232 IEnumString IEnumString_iface;
233 LONG ref;
234 WCHAR **data;
235 int data_len;
236 int cur;
237 };
238
239 static struct string_enumerator *impl_from_IEnumString(IEnumString *iface)
240 {
241 return CONTAINING_RECORD(iface, struct string_enumerator, IEnumString_iface);
242 }
243
244 static HRESULT WINAPI string_enumerator_QueryInterface(IEnumString *iface, REFIID riid, void **ppv)
245 {
246 if (IsEqualGUID(riid, &IID_IEnumString) || IsEqualGUID(riid, &IID_IUnknown))
247 {
248 IUnknown_AddRef(iface);
249 *ppv = iface;
250 return S_OK;
251 }
252
253 *ppv = NULL;
254 return E_NOINTERFACE;
255 }
256
257 static ULONG WINAPI string_enumerator_AddRef(IEnumString *iface)
258 {
259 struct string_enumerator *this = impl_from_IEnumString(iface);
260
261 ULONG ref = InterlockedIncrement(&this->ref);
262
263 return ref;
264 }
265
266 static ULONG WINAPI string_enumerator_Release(IEnumString *iface)
267 {
268 struct string_enumerator *this = impl_from_IEnumString(iface);
269
270 ULONG ref = InterlockedDecrement(&this->ref);
271
272 if (!ref)
273 heap_free(this);
274
275 return ref;
276 }
277
278 static HRESULT WINAPI string_enumerator_Next(IEnumString *iface, ULONG num, LPOLESTR *strings, ULONG *num_returned)
279 {
280 struct string_enumerator *this = impl_from_IEnumString(iface);
281 int i, len;
282
283 *num_returned = 0;
284 for (i = 0; i < num; i++)
285 {
286 if (this->cur >= this->data_len)
287 return S_FALSE;
288
289 len = lstrlenW(this->data[this->cur]) + 1;
290
291 strings[i] = CoTaskMemAlloc(len * sizeof(WCHAR));
292 memcpy(strings[i], this->data[this->cur], len * sizeof(WCHAR));
293
294 (*num_returned)++;
295 this->cur++;
296 }
297
298 return S_OK;
299 }
300
301 static HRESULT WINAPI string_enumerator_Reset(IEnumString *iface)
302 {
303 struct string_enumerator *this = impl_from_IEnumString(iface);
304
305 this->cur = 0;
306
307 return S_OK;
308 }
309
310 static HRESULT WINAPI string_enumerator_Skip(IEnumString *iface, ULONG num)
311 {
312 struct string_enumerator *this = impl_from_IEnumString(iface);
313
314 this->cur += num;
315
316 return S_OK;
317 }
318
319 static HRESULT WINAPI string_enumerator_Clone(IEnumString *iface, IEnumString **out)
320 {
321 *out = NULL;
322 return E_NOTIMPL;
323 }
324
325 static IEnumStringVtbl string_enumerator_vtlb =
326 {
327 string_enumerator_QueryInterface,
328 string_enumerator_AddRef,
329 string_enumerator_Release,
330 string_enumerator_Next,
331 string_enumerator_Skip,
332 string_enumerator_Reset,
333 string_enumerator_Clone
334 };
335
336 static HRESULT string_enumerator_create(void **ppv, WCHAR **suggestions, int count)
337 {
338 struct string_enumerator *object;
339
340 object = heap_alloc_zero(sizeof(*object));
341 object->IEnumString_iface.lpVtbl = &string_enumerator_vtlb;
342 object->ref = 1;
343 object->data = suggestions;
344 object->data_len = count;
345 object->cur = 0;
346
347 *ppv = &object->IEnumString_iface;
348
349 return S_OK;
350 }
351
352 static void test_custom_source(void)
353 {
354 static WCHAR str_alpha[] = {'t','e','s','t','1',0};
355 static WCHAR str_alpha2[] = {'t','e','s','t','2',0};
356 static WCHAR str_beta[] = {'a','u','t','o',' ','c','o','m','p','l','e','t','e',0};
357 static WCHAR *suggestions[] = { str_alpha, str_alpha2, str_beta };
358 IUnknown *enumerator;
359 IAutoComplete2 *autocomplete;
360 HWND hwnd_edit;
361 WCHAR buffer[20];
362 HRESULT hr;
363 MSG msg;
364
365 ShowWindow(hMainWnd, SW_SHOW);
366
367 hwnd_edit = CreateWindowA("Edit", "", WS_OVERLAPPED | WS_VISIBLE | WS_CHILD | WS_BORDER, 50, 5, 200, 20, hMainWnd, 0, NULL, 0);
368
369 hr = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER, &IID_IAutoComplete2, (void**)&autocomplete);
370 ok(hr == S_OK, "CoCreateInstance failed: %x\n", hr);
371
372 string_enumerator_create((void**)&enumerator, suggestions, sizeof(suggestions) / sizeof(*suggestions));
373
374 hr = IAutoComplete2_SetOptions(autocomplete, ACO_AUTOSUGGEST | ACO_AUTOAPPEND);
375 ok(hr == S_OK, "IAutoComplete2_SetOptions failed: %x\n", hr);
376 hr = IAutoComplete2_Init(autocomplete, hwnd_edit, enumerator, NULL, NULL);
377 ok(hr == S_OK, "IAutoComplete_Init failed: %x\n", hr);
378
379 SendMessageW(hwnd_edit, WM_CHAR, 'a', 1);
380 /* Send a keyup message since wine doesn't handle WM_CHAR yet */
381 SendMessageW(hwnd_edit, WM_KEYUP, 'u', 1);
382 Sleep(100);
383 while (PeekMessageA(&msg, 0, 0, 0, PM_REMOVE))
384 {
385 TranslateMessage(&msg);
386 DispatchMessageA(&msg);
387 }
388 SendMessageW(hwnd_edit, WM_GETTEXT, sizeof(buffer) / sizeof(*buffer), (LPARAM)buffer);
389 ok(lstrcmpW(str_beta, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str_beta), wine_dbgstr_w(buffer));
390
391 ShowWindow(hMainWnd, SW_HIDE);
392 DestroyWindow(hwnd_edit);
393 }
394
395 START_TEST(autocomplete)
396 {
397 HRESULT r;
398 MSG msg;
399 IAutoComplete* ac;
400
401 r = CoInitialize(NULL);
402 ok(r == S_OK, "CoInitialize failed (0x%08x). Tests aborted.\n", r);
403 if (r != S_OK)
404 return;
405
406 createMainWnd();
407 ok(hMainWnd != NULL, "Failed to create parent window. Tests aborted.\n");
408 if (!hMainWnd) return;
409
410 test_invalid_init();
411 ac = test_init();
412 if (!ac)
413 goto cleanup;
414 test_killfocus();
415
416 test_custom_source();
417
418 PostQuitMessage(0);
419 while(GetMessageA(&msg,0,0,0)) {
420 TranslateMessage(&msg);
421 DispatchMessageA(&msg);
422 }
423
424 IAutoComplete_Release(ac);
425
426 cleanup:
427 DestroyWindow(hEdit);
428 DestroyWindow(hMainWnd);
429
430 CoUninitialize();
431 }