/* Unit tests for autocomplete * * 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 */ #define WIN32_NO_STATUS #define _INC_WINDOWS #define COM_NO_WINDOWS_H #define COBJMACROS #include //#include #include //#include #include //#include //#include //#include #define ole_ok(exp) \ { \ HRESULT res = (exp); \ if (res != S_OK) \ ok(FALSE, #exp " failed: %x\n", res); \ } static LPWSTR strdup_AtoW(LPCSTR str) { int size = MultiByteToWideChar(CP_ACP, 0, str, -1, NULL, 0); LPWSTR wstr = CoTaskMemAlloc((size + 1)*sizeof(WCHAR)); MultiByteToWideChar(CP_ACP, 0, str, -1, wstr, size+1); return wstr; } typedef struct { IEnumString IEnumString_iface; IACList IACList_iface; LONG ref; HRESULT expret; INT expcount; INT pos; INT limit; const char **data; } TestACL; extern IEnumStringVtbl TestACLVtbl; extern IACListVtbl TestACL_ACListVtbl; static inline TestACL *impl_from_IEnumString(IEnumString *iface) { return CONTAINING_RECORD(iface, TestACL, IEnumString_iface); } static TestACL *impl_from_IACList(IACList *iface) { return CONTAINING_RECORD(iface, TestACL, IACList_iface); } static TestACL *TestACL_Constructor(int limit, const char **strings) { TestACL *This = CoTaskMemAlloc(sizeof(TestACL)); ZeroMemory(This, sizeof(*This)); This->IEnumString_iface.lpVtbl = &TestACLVtbl; This->IACList_iface.lpVtbl = &TestACL_ACListVtbl; This->ref = 1; This->expret = S_OK; This->limit = limit; This->data = strings; return This; } static ULONG STDMETHODCALLTYPE TestACL_AddRef(IEnumString *iface) { TestACL *This = impl_from_IEnumString(iface); trace("ACL(%p): addref (%d)\n", This, This->ref+1); return InterlockedIncrement(&This->ref); } static ULONG STDMETHODCALLTYPE TestACL_Release(IEnumString *iface) { TestACL *This = impl_from_IEnumString(iface); ULONG res; res = InterlockedDecrement(&This->ref); trace("ACL(%p): release (%d)\n", This, res); return res; } static HRESULT STDMETHODCALLTYPE TestACL_QueryInterface(IEnumString *iface, REFIID iid, LPVOID *ppvOut) { TestACL *This = impl_from_IEnumString(iface); *ppvOut = NULL; if (IsEqualGUID(iid, &IID_IUnknown) || IsEqualGUID(iid, &IID_IEnumString)) { *ppvOut = iface; } else if (IsEqualGUID(iid, &IID_IACList)) { *ppvOut = &This->IACList_iface; } if (*ppvOut) { IEnumString_AddRef(iface); return S_OK; } if (!IsEqualGUID(iid, &IID_IEnumACString)) trace("unknown interface queried\n"); return E_NOINTERFACE; } static HRESULT STDMETHODCALLTYPE TestACL_Next(IEnumString *iface, ULONG celt, LPOLESTR *rgelt, ULONG *pceltFetched) { TestACL *This = impl_from_IEnumString(iface); ULONG i; trace("ACL(%p): read %d item(s)\n", This, celt); for (i = 0; i < celt; i++) { if (This->pos >= This->limit) break; rgelt[i] = strdup_AtoW(This->data[This->pos]); This->pos++; } if (pceltFetched) *pceltFetched = i; if (i == celt) return S_OK; return S_FALSE; } static HRESULT STDMETHODCALLTYPE TestACL_Skip(IEnumString *iface, ULONG celt) { ok(FALSE, "Unexpected call to TestACL_Skip\n"); return E_NOTIMPL; } static HRESULT STDMETHODCALLTYPE TestACL_Clone(IEnumString *iface, IEnumString **out) { ok(FALSE, "Unexpected call to TestACL_Clone\n"); return E_OUTOFMEMORY; } static HRESULT STDMETHODCALLTYPE TestACL_Reset(IEnumString *iface) { TestACL *This = impl_from_IEnumString(iface); trace("ACL(%p): Reset\n", This); This->pos = 0; return S_OK; } static HRESULT STDMETHODCALLTYPE TestACL_Expand(IACList *iface, LPCOLESTR str) { TestACL *This = impl_from_IACList(iface); trace("ACL(%p): Expand\n", This); This->expcount++; return This->expret; } IEnumStringVtbl TestACLVtbl = { TestACL_QueryInterface, TestACL_AddRef, TestACL_Release, TestACL_Next, TestACL_Skip, TestACL_Reset, TestACL_Clone }; static ULONG STDMETHODCALLTYPE TestACL_ACList_AddRef(IACList *iface) { TestACL *This = impl_from_IACList(iface); return TestACL_AddRef(&This->IEnumString_iface); } static ULONG STDMETHODCALLTYPE TestACL_ACList_Release(IACList *iface) { TestACL *This = impl_from_IACList(iface); return TestACL_Release(&This->IEnumString_iface); } static HRESULT STDMETHODCALLTYPE TestACL_ACList_QueryInterface(IACList *iface, REFIID iid, LPVOID *ppvout) { TestACL *This = impl_from_IACList(iface); return TestACL_QueryInterface(&This->IEnumString_iface, iid, ppvout); } IACListVtbl TestACL_ACListVtbl = { TestACL_ACList_QueryInterface, TestACL_ACList_AddRef, TestACL_ACList_Release, TestACL_Expand }; #define expect_str(obj, str) \ { \ ole_ok(IEnumString_Next(obj, 1, &wstr, &i)); \ ok(i == 1, "Expected i == 1, got %d\n", i); \ ok(str[0] == wstr[0], "String mismatch\n"); \ CoTaskMemFree(wstr); \ } #define expect_end(obj) \ ok(IEnumString_Next(obj, 1, &wstr, &i) == S_FALSE, "Unexpected return from Next\n"); static void test_ACLMulti(void) { const char *strings1[] = {"a", "c", "e"}; const char *strings2[] = {"a", "b", "d"}; WCHAR exp[] = {'A','B','C',0}; IEnumString *obj; IEnumACString *unk; HRESULT hr; TestACL *acl1, *acl2; IACList *acl; IObjMgr *mgr; LPWSTR wstr; LPWSTR wstrtab[15]; LPVOID tmp; ULONG ref; UINT i; hr = CoCreateInstance(&CLSID_ACLMulti, NULL, CLSCTX_INPROC, &IID_IEnumString, (void**)&obj); ok(hr == S_OK, "failed to create ACLMulti instance, 0x%08x\n", hr); if (hr != S_OK) return; hr = IEnumString_QueryInterface(obj, &IID_IACList, (void**)&acl); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IEnumString_QueryInterface(obj, &IID_IACList2, &tmp); ok(hr == E_NOINTERFACE, "got 0x%08x\n", hr); hr = IEnumString_QueryInterface(obj, &IID_IObjMgr, (void**)&mgr); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IEnumString_QueryInterface(obj, &IID_IEnumACString, (LPVOID*)&unk); if (hr == E_NOINTERFACE) todo_wine win_skip("IEnumACString is not supported, skipping tests\n"); else { ok(hr == S_OK, "QueryInterface(IID_IEnumACString) failed: %x\n", hr); if (unk != NULL) IEnumACString_Release(unk); } i = -1; hr = IEnumString_Next(obj, 1, (LPOLESTR *)&tmp, &i); ok(hr == S_FALSE, "got 0x%08x\n", hr); ok(i == 0, "Unexpected fetched value %d\n", i); hr = IEnumString_Next(obj, 44, (LPOLESTR *)&tmp, &i); ok(hr == S_FALSE, "got 0x%08x\n", hr); hr = IEnumString_Skip(obj, 1); ok(hr == E_NOTIMPL, "got 0x%08x\n", hr); hr = IEnumString_Clone(obj, (IEnumString **)&tmp); ok(hr == E_OUTOFMEMORY, "got 0x%08x\n", hr); hr = IACList_Expand(acl, exp); ok(hr == S_OK, "got 0x%08x\n", hr); acl1 = TestACL_Constructor(3, strings1); acl2 = TestACL_Constructor(3, strings2); hr = IObjMgr_Append(mgr, (IUnknown *)&acl1->IACList_iface); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IObjMgr_Append(mgr, (IUnknown *)&acl2->IACList_iface); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IObjMgr_Append(mgr, NULL); ok(hr == E_FAIL, "got 0x%08x\n", hr); expect_str(obj, "a"); expect_str(obj, "c"); expect_str(obj, "e"); expect_str(obj, "a"); expect_str(obj, "b"); expect_str(obj, "d"); expect_end(obj); hr = IEnumString_Reset(obj); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->pos == 0, "acl1 not reset\n"); ok(acl2->pos == 0, "acl2 not reset\n"); hr = IACList_Expand(acl, exp); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->expcount == 1, "expcount - expected 1, got %d\n", acl1->expcount); ok(acl2->expcount == 0 /* XP */ || acl2->expcount == 1 /* Vista */, "expcount - expected 0 or 1, got %d\n", acl2->expcount); hr = IEnumString_Next(obj, 15, wstrtab, &i); ok(hr == S_OK, "got 0x%08x\n", hr); ok(i == 1, "Expected i == 1, got %d\n", i); CoTaskMemFree(wstrtab[0]); hr = IEnumString_Next(obj, 15, wstrtab, &i); ok(hr == S_OK, "got 0x%08x\n", hr); CoTaskMemFree(wstrtab[0]); hr = IEnumString_Next(obj, 15, wstrtab, &i); ok(hr == S_OK, "got 0x%08x\n", hr); CoTaskMemFree(wstrtab[0]); hr = IEnumString_Next(obj, 15, wstrtab, &i); ok(hr == S_OK, "got 0x%08x\n", hr); CoTaskMemFree(wstrtab[0]); hr = IACList_Expand(acl, exp); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->expcount == 2, "expcount - expected 1, got %d\n", acl1->expcount); ok(acl2->expcount == 0 /* XP */ || acl2->expcount == 2 /* Vista */, "expcount - expected 0 or 2, got %d\n", acl2->expcount); acl1->expret = S_FALSE; hr = IACList_Expand(acl, exp); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->expcount == 3, "expcount - expected 1, got %d\n", acl1->expcount); ok(acl2->expcount == 1 /* XP */ || acl2->expcount == 3 /* Vista */, "expcount - expected 0 or 3, got %d\n", acl2->expcount); acl1->expret = E_NOTIMPL; hr = IACList_Expand(acl, exp); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->expcount == 4, "expcount - expected 1, got %d\n", acl1->expcount); ok(acl2->expcount == 2 /* XP */ || acl2->expcount == 4 /* Vista */, "expcount - expected 0 or 4, got %d\n", acl2->expcount); acl2->expret = E_OUTOFMEMORY; hr = IACList_Expand(acl, exp); ok(hr == E_OUTOFMEMORY, "got 0x%08x\n", hr); acl2->expret = E_FAIL; hr = IACList_Expand(acl, exp); ok(hr == E_FAIL, "got 0x%08x\n", hr); hr = IObjMgr_Remove(mgr, (IUnknown *)&acl1->IACList_iface); ok(hr == S_OK, "got 0x%08x\n", hr); ok(acl1->ref == 1, "acl1 not released\n"); expect_end(obj); IEnumString_Reset(obj); expect_str(obj, "a"); expect_str(obj, "b"); expect_str(obj, "d"); expect_end(obj); IEnumString_Release(obj); IACList_Release(acl); ref = IObjMgr_Release(mgr); ok(ref == 0, "Unexpected references\n"); ok(acl1->ref == 1, "acl1 not released\n"); ok(acl2->ref == 1, "acl2 not released\n"); CoTaskMemFree(acl1); CoTaskMemFree(acl2); } static void test_ACListISF(void) { IEnumString *enumstring; IACList *list, *list2; HRESULT hr; hr = CoCreateInstance(&CLSID_ACListISF, NULL, CLSCTX_INPROC, &IID_IACList, (void**)&list); ok(hr == S_OK, "failed to create ACListISF instance, 0x%08x\n", hr); hr = IACList_QueryInterface(list, &IID_IEnumString, (void**)&enumstring); ok(hr == S_OK, "got 0x%08x\n", hr); hr = IEnumString_QueryInterface(enumstring, &IID_IACList, (void**)&list2); ok(hr == S_OK, "got 0x%08x\n", hr); ok(list == list2, "got %p, %p\n", list, list2); IACList_Release(list2); IEnumString_Release(enumstring); IACList_Release(list); } START_TEST(autocomplete) { CoInitialize(NULL); test_ACLMulti(); test_ACListISF(); CoUninitialize(); }