2008-12-02 17:57:45 +01:00
|
|
|
/*
|
|
|
|
* Tests for autocomplete
|
|
|
|
*
|
|
|
|
* Copyright 2008 Jan de Mooij
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#define COBJMACROS
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include "windows.h"
|
|
|
|
#include "shobjidl.h"
|
|
|
|
#include "shlguid.h"
|
|
|
|
#include "shldisp.h"
|
2018-10-16 13:30:46 +02:00
|
|
|
#include "shlobj.h"
|
2008-12-02 17:57:45 +01:00
|
|
|
|
2018-02-20 07:38:00 +01:00
|
|
|
#include "wine/heap.h"
|
|
|
|
#include "wine/test.h"
|
|
|
|
|
2008-12-02 17:57:45 +01:00
|
|
|
static HWND hMainWnd, hEdit;
|
|
|
|
static HINSTANCE hinst;
|
|
|
|
static int killfocus_count;
|
|
|
|
|
2011-02-01 11:16:20 +01:00
|
|
|
static void test_invalid_init(void)
|
|
|
|
{
|
|
|
|
HRESULT hr;
|
|
|
|
IAutoComplete *ac;
|
|
|
|
IUnknown *acSource;
|
|
|
|
HWND edit_control;
|
|
|
|
|
|
|
|
/* AutoComplete instance */
|
|
|
|
hr = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IAutoComplete, (void **)&ac);
|
|
|
|
if (hr == REGDB_E_CLASSNOTREG)
|
|
|
|
{
|
|
|
|
win_skip("CLSID_AutoComplete is not registered\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ok(hr == S_OK, "no IID_IAutoComplete (0x%08x)\n", hr);
|
|
|
|
|
|
|
|
/* AutoComplete source */
|
|
|
|
hr = CoCreateInstance(&CLSID_ACLMulti, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IACList, (void **)&acSource);
|
|
|
|
if (hr == REGDB_E_CLASSNOTREG)
|
|
|
|
{
|
|
|
|
win_skip("CLSID_ACLMulti is not registered\n");
|
|
|
|
IAutoComplete_Release(ac);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ok(hr == S_OK, "no IID_IACList (0x%08x)\n", hr);
|
|
|
|
|
|
|
|
edit_control = CreateWindowExA(0, "EDIT", "Some text", 0, 10, 10, 300, 300,
|
|
|
|
hMainWnd, NULL, hinst, NULL);
|
|
|
|
ok(edit_control != NULL, "Can't create edit control\n");
|
|
|
|
|
|
|
|
/* The refcount of acSource would be incremented on older Windows. */
|
|
|
|
hr = IAutoComplete_Init(ac, NULL, acSource, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG ||
|
|
|
|
broken(hr == S_OK), /* Win2k/XP/Win2k3 */
|
|
|
|
"Init returned 0x%08x\n", hr);
|
|
|
|
if (hr == E_INVALIDARG)
|
|
|
|
{
|
|
|
|
LONG ref;
|
|
|
|
|
|
|
|
IUnknown_AddRef(acSource);
|
|
|
|
ref = IUnknown_Release(acSource);
|
|
|
|
ok(ref == 1, "Expected AutoComplete source refcount to be 1, got %d\n", ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (0)
|
|
|
|
{
|
|
|
|
/* Older Windows versions never check the window handle, while newer
|
|
|
|
* versions only check for NULL. Subsequent attempts to initialize the
|
|
|
|
* object after this call succeeds would fail, because initialization
|
|
|
|
* state is determined by whether a non-NULL window handle is stored. */
|
|
|
|
hr = IAutoComplete_Init(ac, (HWND)0xdeadbeef, acSource, NULL, NULL);
|
|
|
|
ok(hr == S_OK, "Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
/* Tests crash on older Windows. */
|
|
|
|
hr = IAutoComplete_Init(ac, NULL, NULL, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG, "Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IAutoComplete_Init(ac, edit_control, NULL, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG, "Init returned 0x%08x\n", hr);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* bind to edit control */
|
|
|
|
hr = IAutoComplete_Init(ac, edit_control, acSource, NULL, NULL);
|
|
|
|
ok(hr == S_OK, "Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
/* try invalid parameters after successful initialization .*/
|
|
|
|
hr = IAutoComplete_Init(ac, NULL, NULL, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG ||
|
|
|
|
hr == E_FAIL, /* Win2k/XP/Win2k3 */
|
|
|
|
"Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IAutoComplete_Init(ac, NULL, acSource, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG ||
|
|
|
|
hr == E_FAIL, /* Win2k/XP/Win2k3 */
|
|
|
|
"Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
hr = IAutoComplete_Init(ac, edit_control, NULL, NULL, NULL);
|
|
|
|
ok(hr == E_INVALIDARG ||
|
|
|
|
hr == E_FAIL, /* Win2k/XP/Win2k3 */
|
|
|
|
"Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
/* try initializing twice on the same control */
|
|
|
|
hr = IAutoComplete_Init(ac, edit_control, acSource, NULL, NULL);
|
|
|
|
ok(hr == E_FAIL, "Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
/* try initializing with a different control */
|
|
|
|
hr = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
|
|
|
|
ok(hr == E_FAIL, "Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
DestroyWindow(edit_control);
|
|
|
|
|
|
|
|
/* try initializing with a different control after
|
|
|
|
* destroying the original initialization control */
|
|
|
|
hr = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
|
|
|
|
ok(hr == E_UNEXPECTED ||
|
|
|
|
hr == E_FAIL, /* Win2k/XP/Win2k3 */
|
|
|
|
"Init returned 0x%08x\n", hr);
|
|
|
|
|
|
|
|
IUnknown_Release(acSource);
|
|
|
|
IAutoComplete_Release(ac);
|
|
|
|
}
|
2009-12-29 20:02:16 +01:00
|
|
|
static IAutoComplete *test_init(void)
|
2009-12-29 20:01:29 +01:00
|
|
|
{
|
2008-12-02 17:57:45 +01:00
|
|
|
HRESULT r;
|
2018-08-27 19:10:47 +02:00
|
|
|
IAutoComplete *ac, *ac2;
|
2008-12-02 17:57:45 +01:00
|
|
|
IUnknown *acSource;
|
2011-02-01 11:16:30 +01:00
|
|
|
LONG_PTR user_data;
|
2008-12-02 17:57:45 +01:00
|
|
|
|
|
|
|
/* AutoComplete instance */
|
|
|
|
r = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IAutoComplete, (LPVOID*)&ac);
|
2009-03-09 14:08:52 +01:00
|
|
|
if (r == REGDB_E_CLASSNOTREG)
|
|
|
|
{
|
|
|
|
win_skip("CLSID_AutoComplete is not registered\n");
|
2009-12-29 20:02:16 +01:00
|
|
|
return NULL;
|
2009-03-09 14:08:52 +01:00
|
|
|
}
|
2010-03-23 02:21:13 +01:00
|
|
|
ok(r == S_OK, "no IID_IAutoComplete (0x%08x)\n", r);
|
2008-12-02 17:57:45 +01:00
|
|
|
|
|
|
|
/* AutoComplete source */
|
|
|
|
r = CoCreateInstance(&CLSID_ACLMulti, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IACList, (LPVOID*)&acSource);
|
2009-03-09 14:08:52 +01:00
|
|
|
if (r == REGDB_E_CLASSNOTREG)
|
|
|
|
{
|
|
|
|
win_skip("CLSID_ACLMulti is not registered\n");
|
2011-02-01 11:16:02 +01:00
|
|
|
IAutoComplete_Release(ac);
|
2009-12-29 20:02:16 +01:00
|
|
|
return NULL;
|
2009-03-09 14:08:52 +01:00
|
|
|
}
|
2010-03-23 02:21:13 +01:00
|
|
|
ok(r == S_OK, "no IID_IACList (0x%08x)\n", r);
|
2008-12-02 17:57:45 +01:00
|
|
|
|
2011-02-01 11:16:30 +01:00
|
|
|
user_data = GetWindowLongPtrA(hEdit, GWLP_USERDATA);
|
|
|
|
ok(user_data == 0, "Expected the edit control user data to be zero\n");
|
|
|
|
|
2008-12-02 17:57:45 +01:00
|
|
|
/* bind to edit control */
|
|
|
|
r = IAutoComplete_Init(ac, hEdit, acSource, NULL, NULL);
|
2011-02-01 11:16:20 +01:00
|
|
|
ok(r == S_OK, "Init returned 0x%08x\n", r);
|
2009-03-09 14:08:52 +01:00
|
|
|
|
2011-02-01 11:16:30 +01:00
|
|
|
user_data = GetWindowLongPtrA(hEdit, GWLP_USERDATA);
|
|
|
|
ok(user_data == 0, "Expected the edit control user data to be zero\n");
|
|
|
|
|
2018-08-27 19:10:47 +02:00
|
|
|
/* bind a different object to the same edit control */
|
|
|
|
r = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER,
|
|
|
|
&IID_IAutoComplete, (LPVOID*)&ac2);
|
|
|
|
ok(r == S_OK, "no IID_IAutoComplete (0x%08x)\n", r);
|
|
|
|
|
|
|
|
r = IAutoComplete_Init(ac2, hEdit, acSource, NULL, NULL);
|
|
|
|
ok(r == S_OK, "Init returned 0x%08x\n", r);
|
|
|
|
IAutoComplete_Release(ac2);
|
|
|
|
|
2009-12-29 20:02:16 +01:00
|
|
|
IUnknown_Release(acSource);
|
|
|
|
|
|
|
|
return ac;
|
2008-12-02 17:57:45 +01:00
|
|
|
}
|
2009-12-29 20:01:29 +01:00
|
|
|
|
|
|
|
static void test_killfocus(void)
|
|
|
|
{
|
2008-12-02 17:57:45 +01:00
|
|
|
/* Test if WM_KILLFOCUS messages are handled properly by checking if
|
|
|
|
* the parent receives an EN_KILLFOCUS message. */
|
|
|
|
SetFocus(hEdit);
|
|
|
|
killfocus_count = 0;
|
|
|
|
SetFocus(0);
|
|
|
|
ok(killfocus_count == 1, "Expected one EN_KILLFOCUS message, got: %d\n", killfocus_count);
|
|
|
|
}
|
2009-12-29 20:01:29 +01:00
|
|
|
|
|
|
|
static LRESULT CALLBACK MyWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
2008-12-02 17:57:45 +01:00
|
|
|
switch(msg) {
|
|
|
|
case WM_CREATE:
|
|
|
|
/* create edit control */
|
2013-10-26 21:10:26 +02:00
|
|
|
hEdit = CreateWindowExA(0, "EDIT", "Some text", 0, 10, 10, 300, 300,
|
2008-12-02 17:57:45 +01:00
|
|
|
hWnd, NULL, hinst, NULL);
|
|
|
|
ok(hEdit != NULL, "Can't create edit control\n");
|
|
|
|
break;
|
|
|
|
case WM_COMMAND:
|
|
|
|
if(HIWORD(wParam) == EN_KILLFOCUS)
|
|
|
|
killfocus_count++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return DefWindowProcA(hWnd, msg, wParam, lParam);
|
|
|
|
}
|
2009-12-29 20:01:29 +01:00
|
|
|
|
|
|
|
static void createMainWnd(void)
|
|
|
|
{
|
2008-12-02 17:57:45 +01:00
|
|
|
WNDCLASSA wc;
|
|
|
|
wc.style = CS_HREDRAW | CS_VREDRAW;
|
|
|
|
wc.cbClsExtra = 0;
|
|
|
|
wc.cbWndExtra = 0;
|
|
|
|
wc.hInstance = GetModuleHandleA(NULL);
|
|
|
|
wc.hIcon = NULL;
|
2013-10-26 21:10:26 +02:00
|
|
|
wc.hCursor = LoadCursorA(NULL, (LPSTR)IDC_IBEAM);
|
2008-12-02 17:57:45 +01:00
|
|
|
wc.hbrBackground = GetSysColorBrush(COLOR_WINDOW);
|
|
|
|
wc.lpszMenuName = NULL;
|
|
|
|
wc.lpszClassName = "MyTestWnd";
|
|
|
|
wc.lpfnWndProc = MyWndProc;
|
|
|
|
RegisterClassA(&wc);
|
|
|
|
|
|
|
|
hMainWnd = CreateWindowExA(0, "MyTestWnd", "Blah", WS_OVERLAPPEDWINDOW,
|
|
|
|
CW_USEDEFAULT, CW_USEDEFAULT, 130, 105, NULL, NULL, GetModuleHandleA(NULL), 0);
|
|
|
|
}
|
2009-12-29 20:01:29 +01:00
|
|
|
|
2018-09-22 16:53:46 +02:00
|
|
|
static WNDPROC HijackerWndProc_prev;
|
|
|
|
static const WCHAR HijackerWndProc_txt[] = {'H','i','j','a','c','k','e','d',0};
|
|
|
|
static LRESULT CALLBACK HijackerWndProc(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
switch(msg) {
|
|
|
|
case WM_GETTEXT:
|
|
|
|
{
|
|
|
|
size_t len = min(wParam, ARRAY_SIZE(HijackerWndProc_txt));
|
|
|
|
memcpy((void*)lParam, HijackerWndProc_txt, len * sizeof(WCHAR));
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
case WM_GETTEXTLENGTH:
|
|
|
|
return ARRAY_SIZE(HijackerWndProc_txt) - 1;
|
|
|
|
}
|
|
|
|
return CallWindowProcW(HijackerWndProc_prev, hWnd, msg, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
|
|
|
static LRESULT CALLBACK HijackerWndProc2(HWND hWnd, UINT msg, WPARAM wParam, LPARAM lParam)
|
|
|
|
{
|
|
|
|
switch(msg) {
|
|
|
|
case EM_SETSEL:
|
|
|
|
lParam = wParam;
|
|
|
|
break;
|
|
|
|
case WM_SETTEXT:
|
|
|
|
lParam = (LPARAM)HijackerWndProc_txt;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return CallWindowProcW(HijackerWndProc_prev, hWnd, msg, wParam, lParam);
|
|
|
|
}
|
|
|
|
|
2017-11-17 19:54:44 +01:00
|
|
|
struct string_enumerator
|
|
|
|
{
|
|
|
|
IEnumString IEnumString_iface;
|
2018-10-16 13:30:46 +02:00
|
|
|
IACList IACList_iface;
|
2017-11-17 19:54:44 +01:00
|
|
|
LONG ref;
|
|
|
|
WCHAR **data;
|
|
|
|
int data_len;
|
|
|
|
int cur;
|
2018-11-02 15:50:41 +01:00
|
|
|
UINT num_resets;
|
2018-10-16 13:30:46 +02:00
|
|
|
UINT num_expand;
|
|
|
|
WCHAR last_expand[32];
|
2017-11-17 19:54:44 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
static struct string_enumerator *impl_from_IEnumString(IEnumString *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct string_enumerator, IEnumString_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI string_enumerator_QueryInterface(IEnumString *iface, REFIID riid, void **ppv)
|
|
|
|
{
|
2018-10-16 13:30:46 +02:00
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
2017-11-17 19:54:44 +01:00
|
|
|
if (IsEqualGUID(riid, &IID_IEnumString) || IsEqualGUID(riid, &IID_IUnknown))
|
2018-10-16 13:30:46 +02:00
|
|
|
*ppv = &this->IEnumString_iface;
|
|
|
|
else if (IsEqualGUID(riid, &IID_IACList))
|
|
|
|
*ppv = &this->IACList_iface;
|
|
|
|
else
|
2017-11-17 19:54:44 +01:00
|
|
|
{
|
2018-10-16 13:30:46 +02:00
|
|
|
*ppv = NULL;
|
|
|
|
return E_NOINTERFACE;
|
2017-11-17 19:54:44 +01:00
|
|
|
}
|
|
|
|
|
2018-10-16 13:30:46 +02:00
|
|
|
IUnknown_AddRef(&this->IEnumString_iface);
|
|
|
|
return S_OK;
|
2017-11-17 19:54:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI string_enumerator_AddRef(IEnumString *iface)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
|
|
|
|
|
|
|
ULONG ref = InterlockedIncrement(&this->ref);
|
|
|
|
|
|
|
|
return ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI string_enumerator_Release(IEnumString *iface)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
|
|
|
|
|
|
|
ULONG ref = InterlockedDecrement(&this->ref);
|
|
|
|
|
|
|
|
if (!ref)
|
2018-02-20 07:38:00 +01:00
|
|
|
heap_free(this);
|
2017-11-17 19:54:44 +01:00
|
|
|
|
|
|
|
return ref;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI string_enumerator_Next(IEnumString *iface, ULONG num, LPOLESTR *strings, ULONG *num_returned)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
|
|
|
int i, len;
|
|
|
|
|
|
|
|
*num_returned = 0;
|
|
|
|
for (i = 0; i < num; i++)
|
|
|
|
{
|
|
|
|
if (this->cur >= this->data_len)
|
|
|
|
return S_FALSE;
|
|
|
|
|
|
|
|
len = lstrlenW(this->data[this->cur]) + 1;
|
|
|
|
|
|
|
|
strings[i] = CoTaskMemAlloc(len * sizeof(WCHAR));
|
|
|
|
memcpy(strings[i], this->data[this->cur], len * sizeof(WCHAR));
|
|
|
|
|
|
|
|
(*num_returned)++;
|
|
|
|
this->cur++;
|
|
|
|
}
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI string_enumerator_Reset(IEnumString *iface)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
|
|
|
|
|
|
|
this->cur = 0;
|
2018-11-02 15:50:41 +01:00
|
|
|
this->num_resets++;
|
2017-11-17 19:54:44 +01:00
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI string_enumerator_Skip(IEnumString *iface, ULONG num)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IEnumString(iface);
|
|
|
|
|
|
|
|
this->cur += num;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI string_enumerator_Clone(IEnumString *iface, IEnumString **out)
|
|
|
|
{
|
|
|
|
*out = NULL;
|
|
|
|
return E_NOTIMPL;
|
|
|
|
}
|
|
|
|
|
2018-10-16 13:30:46 +02:00
|
|
|
static IEnumStringVtbl string_enumerator_vtbl =
|
2017-11-17 19:54:44 +01:00
|
|
|
{
|
|
|
|
string_enumerator_QueryInterface,
|
|
|
|
string_enumerator_AddRef,
|
|
|
|
string_enumerator_Release,
|
|
|
|
string_enumerator_Next,
|
|
|
|
string_enumerator_Skip,
|
|
|
|
string_enumerator_Reset,
|
|
|
|
string_enumerator_Clone
|
|
|
|
};
|
|
|
|
|
2018-10-16 13:30:46 +02:00
|
|
|
static struct string_enumerator *impl_from_IACList(IACList *iface)
|
|
|
|
{
|
|
|
|
return CONTAINING_RECORD(iface, struct string_enumerator, IACList_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI aclist_QueryInterface(IACList *iface, REFIID riid, void **ppv)
|
|
|
|
{
|
|
|
|
return string_enumerator_QueryInterface(&impl_from_IACList(iface)->IEnumString_iface, riid, ppv);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI aclist_AddRef(IACList *iface)
|
|
|
|
{
|
|
|
|
return string_enumerator_AddRef(&impl_from_IACList(iface)->IEnumString_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static ULONG WINAPI aclist_Release(IACList *iface)
|
|
|
|
{
|
|
|
|
return string_enumerator_Release(&impl_from_IACList(iface)->IEnumString_iface);
|
|
|
|
}
|
|
|
|
|
|
|
|
static HRESULT WINAPI aclist_Expand(IACList *iface, const WCHAR *expand)
|
|
|
|
{
|
|
|
|
struct string_enumerator *this = impl_from_IACList(iface);
|
|
|
|
|
|
|
|
/* see what we get called with and how many times,
|
|
|
|
don't actually do any expansion of the strings */
|
|
|
|
memcpy(this->last_expand, expand, min((lstrlenW(expand) + 1)*sizeof(WCHAR), sizeof(this->last_expand)));
|
|
|
|
this->last_expand[ARRAY_SIZE(this->last_expand) - 1] = '\0';
|
|
|
|
this->num_expand++;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static IACListVtbl aclist_vtbl =
|
|
|
|
{
|
|
|
|
aclist_QueryInterface,
|
|
|
|
aclist_AddRef,
|
|
|
|
aclist_Release,
|
|
|
|
aclist_Expand
|
|
|
|
};
|
|
|
|
|
2017-11-17 19:54:44 +01:00
|
|
|
static HRESULT string_enumerator_create(void **ppv, WCHAR **suggestions, int count)
|
|
|
|
{
|
|
|
|
struct string_enumerator *object;
|
|
|
|
|
2018-02-20 07:38:00 +01:00
|
|
|
object = heap_alloc_zero(sizeof(*object));
|
2018-10-16 13:30:46 +02:00
|
|
|
object->IEnumString_iface.lpVtbl = &string_enumerator_vtbl;
|
|
|
|
object->IACList_iface.lpVtbl = &aclist_vtbl;
|
2017-11-17 19:54:44 +01:00
|
|
|
object->ref = 1;
|
|
|
|
object->data = suggestions;
|
|
|
|
object->data_len = count;
|
|
|
|
object->cur = 0;
|
|
|
|
|
|
|
|
*ppv = &object->IEnumString_iface;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2018-09-22 16:53:46 +02:00
|
|
|
static void dispatch_messages(void)
|
|
|
|
{
|
|
|
|
MSG msg;
|
|
|
|
Sleep(33);
|
|
|
|
while (PeekMessageA(&msg, 0, 0, 0, PM_REMOVE))
|
|
|
|
{
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageA(&msg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-16 13:30:46 +02:00
|
|
|
static void test_aclist_expand(HWND hwnd_edit, void *enumerator)
|
|
|
|
{
|
|
|
|
struct string_enumerator *obj = (struct string_enumerator*)enumerator;
|
|
|
|
static WCHAR str1[] = {'t','e','s','t',0};
|
|
|
|
static WCHAR str1a[] = {'t','e','s','t','\\',0};
|
|
|
|
static WCHAR str2[] = {'t','e','s','t','\\','f','o','o','\\','b','a','r','\\','b','a',0};
|
|
|
|
static WCHAR str2a[] = {'t','e','s','t','\\','f','o','o','\\','b','a','r','\\',0};
|
|
|
|
static WCHAR str2b[] = {'t','e','s','t','\\','f','o','o','\\','b','a','r','\\','b','a','z','_','b','b','q','\\',0};
|
2018-11-02 15:50:41 +01:00
|
|
|
obj->num_resets = 0;
|
2018-10-16 13:30:46 +02:00
|
|
|
|
|
|
|
ok(obj->num_expand == 0, "Expected 0 expansions, got %u\n", obj->num_expand);
|
|
|
|
SendMessageW(hwnd_edit, WM_SETTEXT, 0, (LPARAM)str1);
|
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, ARRAY_SIZE(str1) - 1, ARRAY_SIZE(str1) - 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, '\\', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 1, "Expected 1 expansion, got %u\n", obj->num_expand);
|
|
|
|
ok(lstrcmpW(obj->last_expand, str1a) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str1a), wine_dbgstr_w(obj->last_expand));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
SendMessageW(hwnd_edit, WM_SETTEXT, 0, (LPARAM)str2);
|
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, ARRAY_SIZE(str2) - 1, ARRAY_SIZE(str2) - 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'z', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 2, "Expected 2 expansions, got %u\n", obj->num_expand);
|
|
|
|
ok(lstrcmpW(obj->last_expand, str2a) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str2a), wine_dbgstr_w(obj->last_expand));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 2, "Expected 2 resets, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
SetFocus(hwnd_edit);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, '_', 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'b', 1);
|
|
|
|
SetFocus(0);
|
|
|
|
SetFocus(hwnd_edit);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'b', 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'q', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 2, "Expected 2 expansions, got %u\n", obj->num_expand);
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 2, "Expected 2 resets, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, '\\', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 3, "Expected 3 expansions, got %u\n", obj->num_expand);
|
|
|
|
ok(lstrcmpW(obj->last_expand, str2b) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str2b), wine_dbgstr_w(obj->last_expand));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 3, "Expected 3 resets, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, ARRAY_SIZE(str1a) - 1, -1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'x', 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'y', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 4, "Expected 4 expansions, got %u\n", obj->num_expand);
|
|
|
|
ok(lstrcmpW(obj->last_expand, str1a) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str1a), wine_dbgstr_w(obj->last_expand));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 4, "Expected 4 resets, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, ARRAY_SIZE(str1) - 1, -1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'x', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_expand == 4, "Expected 4 expansions, got %u\n", obj->num_expand);
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 5, "Expected 5 resets, got %u\n", obj->num_resets);
|
2018-10-16 13:30:46 +02:00
|
|
|
}
|
|
|
|
|
2017-11-17 19:54:44 +01:00
|
|
|
static void test_custom_source(void)
|
|
|
|
{
|
|
|
|
static WCHAR str_alpha[] = {'t','e','s','t','1',0};
|
|
|
|
static WCHAR str_alpha2[] = {'t','e','s','t','2',0};
|
|
|
|
static WCHAR str_beta[] = {'a','u','t','o',' ','c','o','m','p','l','e','t','e',0};
|
2018-09-22 16:53:46 +02:00
|
|
|
static WCHAR str_au[] = {'a','u',0};
|
2018-11-13 12:04:06 +01:00
|
|
|
static WCHAR str_aut[] = {'a','u','t',0};
|
2017-11-17 19:54:44 +01:00
|
|
|
static WCHAR *suggestions[] = { str_alpha, str_alpha2, str_beta };
|
2018-11-02 15:50:41 +01:00
|
|
|
struct string_enumerator *obj;
|
2017-11-17 19:54:44 +01:00
|
|
|
IUnknown *enumerator;
|
|
|
|
IAutoComplete2 *autocomplete;
|
2018-11-07 16:14:16 +01:00
|
|
|
IAutoCompleteDropDown *acdropdown;
|
2017-11-17 19:54:44 +01:00
|
|
|
HWND hwnd_edit;
|
2018-11-13 12:04:06 +01:00
|
|
|
DWORD flags = 0;
|
2017-11-17 19:54:44 +01:00
|
|
|
WCHAR buffer[20];
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
ShowWindow(hMainWnd, SW_SHOW);
|
|
|
|
|
|
|
|
hwnd_edit = CreateWindowA("Edit", "", WS_OVERLAPPED | WS_VISIBLE | WS_CHILD | WS_BORDER, 50, 5, 200, 20, hMainWnd, 0, NULL, 0);
|
|
|
|
|
|
|
|
hr = CoCreateInstance(&CLSID_AutoComplete, NULL, CLSCTX_INPROC_SERVER, &IID_IAutoComplete2, (void**)&autocomplete);
|
|
|
|
ok(hr == S_OK, "CoCreateInstance failed: %x\n", hr);
|
|
|
|
|
2018-11-07 16:14:16 +01:00
|
|
|
hr = IAutoComplete2_QueryInterface(autocomplete, &IID_IAutoCompleteDropDown, (LPVOID*)&acdropdown);
|
|
|
|
ok(hr == S_OK, "No IAutoCompleteDropDown interface: %x\n", hr);
|
|
|
|
|
2018-06-15 00:04:42 +02:00
|
|
|
string_enumerator_create((void**)&enumerator, suggestions, ARRAY_SIZE(suggestions));
|
2018-11-02 15:50:41 +01:00
|
|
|
obj = (struct string_enumerator*)enumerator;
|
2017-11-17 19:54:44 +01:00
|
|
|
|
|
|
|
hr = IAutoComplete2_SetOptions(autocomplete, ACO_AUTOSUGGEST | ACO_AUTOAPPEND);
|
|
|
|
ok(hr == S_OK, "IAutoComplete2_SetOptions failed: %x\n", hr);
|
2018-11-07 16:14:16 +01:00
|
|
|
hr = IAutoCompleteDropDown_ResetEnumerator(acdropdown);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_ResetEnumerator failed: %x\n", hr);
|
2017-11-17 19:54:44 +01:00
|
|
|
hr = IAutoComplete2_Init(autocomplete, hwnd_edit, enumerator, NULL, NULL);
|
|
|
|
ok(hr == S_OK, "IAutoComplete_Init failed: %x\n", hr);
|
|
|
|
|
2018-09-22 16:53:46 +02:00
|
|
|
SetFocus(hwnd_edit);
|
2017-11-17 19:54:44 +01:00
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'a', 1);
|
2018-09-18 22:59:56 +02:00
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'u', 1);
|
2018-09-22 16:53:46 +02:00
|
|
|
dispatch_messages();
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(lstrcmpW(str_beta, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str_beta), wine_dbgstr_w(buffer));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
2018-09-22 16:53:46 +02:00
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, 0, -1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, '\b', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(buffer[0] == '\0', "Expected empty string, got %s\n", wine_dbgstr_w(buffer));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
2018-11-07 16:14:16 +01:00
|
|
|
hr = IAutoCompleteDropDown_ResetEnumerator(acdropdown);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_ResetEnumerator failed: %x\n", hr);
|
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
2018-11-02 15:50:41 +01:00
|
|
|
obj->num_resets = 0;
|
2018-09-22 16:53:46 +02:00
|
|
|
|
|
|
|
/* hijack the window procedure */
|
|
|
|
HijackerWndProc_prev = (WNDPROC)SetWindowLongPtrW(hwnd_edit, GWLP_WNDPROC, (LONG_PTR)HijackerWndProc);
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(lstrcmpW(HijackerWndProc_txt, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(HijackerWndProc_txt), wine_dbgstr_w(buffer));
|
|
|
|
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'a', 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'u', 1);
|
|
|
|
SetWindowLongPtrW(hwnd_edit, GWLP_WNDPROC, (LONG_PTR)HijackerWndProc_prev);
|
|
|
|
dispatch_messages();
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(lstrcmpW(str_au, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str_au), wine_dbgstr_w(buffer));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
2018-09-22 16:53:46 +02:00
|
|
|
SendMessageW(hwnd_edit, EM_SETSEL, 0, -1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, '\b', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(buffer[0] == '\0', "Expected empty string, got %s\n", wine_dbgstr_w(buffer));
|
2018-11-07 16:14:16 +01:00
|
|
|
hr = IAutoCompleteDropDown_ResetEnumerator(acdropdown);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_ResetEnumerator failed: %x\n", hr);
|
2018-09-22 16:53:46 +02:00
|
|
|
|
|
|
|
HijackerWndProc_prev = (WNDPROC)SetWindowLongPtrW(hwnd_edit, GWLP_WNDPROC, (LONG_PTR)HijackerWndProc2);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'a', 1);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'u', 1);
|
|
|
|
SetWindowLongPtrW(hwnd_edit, GWLP_WNDPROC, (LONG_PTR)HijackerWndProc_prev);
|
|
|
|
dispatch_messages();
|
2018-06-15 00:04:42 +02:00
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
2017-11-17 19:54:44 +01:00
|
|
|
ok(lstrcmpW(str_beta, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str_beta), wine_dbgstr_w(buffer));
|
2018-11-02 15:50:41 +01:00
|
|
|
ok(obj->num_resets == 2, "Expected 2 resets, got %u\n", obj->num_resets);
|
2018-09-22 16:53:46 +02:00
|
|
|
/* end of hijacks */
|
2017-11-17 19:54:44 +01:00
|
|
|
|
2018-11-13 12:04:06 +01:00
|
|
|
hr = IAutoCompleteDropDown_GetDropDownStatus(acdropdown, &flags, NULL);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_GetDropDownStatus failed: %x\n", hr);
|
|
|
|
ok(flags & ACDD_VISIBLE, "AutoComplete DropDown should be visible\n");
|
|
|
|
SendMessageW(hwnd_edit, WM_SETTEXT, 0, (LPARAM)str_au);
|
|
|
|
dispatch_messages();
|
|
|
|
hr = IAutoCompleteDropDown_GetDropDownStatus(acdropdown, &flags, NULL);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_GetDropDownStatus failed: %x\n", hr);
|
|
|
|
ok(!(flags & ACDD_VISIBLE), "AutoComplete DropDown should have been hidden\n");
|
|
|
|
SendMessageW(hwnd_edit, WM_SETTEXT, 0, (LPARAM)str_aut);
|
|
|
|
dispatch_messages();
|
|
|
|
hr = IAutoCompleteDropDown_GetDropDownStatus(acdropdown, &flags, NULL);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_GetDropDownStatus failed: %x\n", hr);
|
|
|
|
ok(!(flags & ACDD_VISIBLE), "AutoComplete DropDown should be hidden\n");
|
|
|
|
SendMessageW(hwnd_edit, WM_GETTEXT, ARRAY_SIZE(buffer), (LPARAM)buffer);
|
|
|
|
ok(lstrcmpW(str_aut, buffer) == 0, "Expected %s, got %s\n", wine_dbgstr_w(str_aut), wine_dbgstr_w(buffer));
|
|
|
|
|
2018-10-16 13:30:46 +02:00
|
|
|
test_aclist_expand(hwnd_edit, enumerator);
|
2018-11-07 16:14:16 +01:00
|
|
|
obj->num_resets = 0;
|
|
|
|
|
|
|
|
hr = IAutoCompleteDropDown_ResetEnumerator(acdropdown);
|
|
|
|
ok(hr == S_OK, "IAutoCompleteDropDown_ResetEnumerator failed: %x\n", hr);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'x', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
|
|
|
SendMessageW(hwnd_edit, WM_CHAR, 'x', 1);
|
|
|
|
dispatch_messages();
|
|
|
|
ok(obj->num_resets == 1, "Expected 1 reset, got %u\n", obj->num_resets);
|
|
|
|
IAutoCompleteDropDown_Release(acdropdown);
|
2018-10-16 13:30:46 +02:00
|
|
|
|
2017-11-17 19:54:44 +01:00
|
|
|
ShowWindow(hMainWnd, SW_HIDE);
|
|
|
|
DestroyWindow(hwnd_edit);
|
|
|
|
}
|
|
|
|
|
2009-12-29 20:01:29 +01:00
|
|
|
START_TEST(autocomplete)
|
|
|
|
{
|
2008-12-02 17:57:45 +01:00
|
|
|
HRESULT r;
|
|
|
|
MSG msg;
|
2009-12-29 20:02:16 +01:00
|
|
|
IAutoComplete* ac;
|
2008-12-02 17:57:45 +01:00
|
|
|
|
|
|
|
r = CoInitialize(NULL);
|
2010-03-23 02:21:13 +01:00
|
|
|
ok(r == S_OK, "CoInitialize failed (0x%08x). Tests aborted.\n", r);
|
|
|
|
if (r != S_OK)
|
2008-12-02 17:57:45 +01:00
|
|
|
return;
|
|
|
|
|
|
|
|
createMainWnd();
|
2010-02-24 16:04:05 +01:00
|
|
|
ok(hMainWnd != NULL, "Failed to create parent window. Tests aborted.\n");
|
|
|
|
if (!hMainWnd) return;
|
2008-12-02 17:57:45 +01:00
|
|
|
|
2011-02-01 11:16:20 +01:00
|
|
|
test_invalid_init();
|
2009-12-29 20:02:16 +01:00
|
|
|
ac = test_init();
|
|
|
|
if (!ac)
|
2009-03-09 14:08:52 +01:00
|
|
|
goto cleanup;
|
2008-12-02 17:57:45 +01:00
|
|
|
test_killfocus();
|
|
|
|
|
2017-11-17 19:54:44 +01:00
|
|
|
test_custom_source();
|
|
|
|
|
2008-12-02 17:57:45 +01:00
|
|
|
PostQuitMessage(0);
|
|
|
|
while(GetMessageA(&msg,0,0,0)) {
|
|
|
|
TranslateMessage(&msg);
|
|
|
|
DispatchMessageA(&msg);
|
|
|
|
}
|
|
|
|
|
2009-12-29 20:02:16 +01:00
|
|
|
IAutoComplete_Release(ac);
|
|
|
|
|
2009-03-09 14:08:52 +01:00
|
|
|
cleanup:
|
2008-12-02 17:57:45 +01:00
|
|
|
DestroyWindow(hEdit);
|
|
|
|
DestroyWindow(hMainWnd);
|
|
|
|
|
|
|
|
CoUninitialize();
|
|
|
|
}
|