ddraw/tests: Rewrite LimitTest().
[wine.git] / dlls / taskschd / folder_collection.c
blob0b2b513f271b5c19091d77dc08c01d43a30677e5
1 /*
2 * Copyright 2014 Dmitry Timoshkov
4 * This library is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
9 * This library is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with this library; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
19 #include <stdarg.h>
21 #define COBJMACROS
23 #include "windef.h"
24 #include "winbase.h"
25 #include "objbase.h"
26 #include "taskschd.h"
27 #include "schrpc.h"
28 #include "taskschd_private.h"
30 #include "wine/unicode.h"
31 #include "wine/debug.h"
33 WINE_DEFAULT_DEBUG_CHANNEL(taskschd);
35 typedef struct
37 ITaskFolderCollection ITaskFolderCollection_iface;
38 LONG ref;
39 WCHAR *path;
40 TASK_NAMES list;
41 DWORD count;
42 } TaskFolderCollection;
44 static HRESULT NewEnum_create(TaskFolderCollection *folders, IUnknown **obj);
46 static inline TaskFolderCollection *impl_from_ITaskFolderCollection(ITaskFolderCollection *iface)
48 return CONTAINING_RECORD(iface, TaskFolderCollection, ITaskFolderCollection_iface);
51 static ULONG WINAPI folders_AddRef(ITaskFolderCollection *iface)
53 TaskFolderCollection *folders = impl_from_ITaskFolderCollection(iface);
54 return InterlockedIncrement(&folders->ref);
57 static void free_list(LPWSTR *list, DWORD count)
59 LONG i;
61 for (i = 0; i < count; i++)
62 MIDL_user_free(list[i]);
64 MIDL_user_free(list);
67 static ULONG WINAPI folders_Release(ITaskFolderCollection *iface)
69 TaskFolderCollection *folders = impl_from_ITaskFolderCollection(iface);
70 LONG ref = InterlockedDecrement(&folders->ref);
72 if (!ref)
74 TRACE("destroying %p\n", iface);
75 free_list(folders->list, folders->count);
76 heap_free(folders->path);
77 heap_free(folders);
80 return ref;
83 static HRESULT WINAPI folders_QueryInterface(ITaskFolderCollection *iface, REFIID riid, void **obj)
85 if (!riid || !obj) return E_INVALIDARG;
87 TRACE("%p,%s,%p\n", iface, debugstr_guid(riid), obj);
89 if (IsEqualGUID(riid, &IID_ITaskFolderCollection) ||
90 IsEqualGUID(riid, &IID_IDispatch) ||
91 IsEqualGUID(riid, &IID_IUnknown))
93 ITaskFolderCollection_AddRef(iface);
94 *obj = iface;
95 return S_OK;
98 FIXME("interface %s is not implemented\n", debugstr_guid(riid));
99 *obj = NULL;
100 return E_NOINTERFACE;
103 static HRESULT WINAPI folders_GetTypeInfoCount(ITaskFolderCollection *iface, UINT *count)
105 FIXME("%p,%p: stub\n", iface, count);
106 return E_NOTIMPL;
109 static HRESULT WINAPI folders_GetTypeInfo(ITaskFolderCollection *iface, UINT index, LCID lcid, ITypeInfo **info)
111 FIXME("%p,%u,%u,%p: stub\n", iface, index, lcid, info);
112 return E_NOTIMPL;
115 static HRESULT WINAPI folders_GetIDsOfNames(ITaskFolderCollection *iface, REFIID riid, LPOLESTR *names,
116 UINT count, LCID lcid, DISPID *dispid)
118 FIXME("%p,%s,%p,%u,%u,%p: stub\n", iface, debugstr_guid(riid), names, count, lcid, dispid);
119 return E_NOTIMPL;
122 static HRESULT WINAPI folders_Invoke(ITaskFolderCollection *iface, DISPID dispid, REFIID riid, LCID lcid, WORD flags,
123 DISPPARAMS *params, VARIANT *result, EXCEPINFO *excepinfo, UINT *argerr)
125 FIXME("%p,%d,%s,%04x,%04x,%p,%p,%p,%p: stub\n", iface, dispid, debugstr_guid(riid), lcid, flags,
126 params, result, excepinfo, argerr);
127 return E_NOTIMPL;
130 static HRESULT WINAPI folders_get_Count(ITaskFolderCollection *iface, LONG *count)
132 TaskFolderCollection *folders = impl_from_ITaskFolderCollection(iface);
134 TRACE("%p,%p\n", iface, count);
136 if (!count) return E_POINTER;
138 *count = folders->count;
140 return S_OK;
143 static LONG get_var_int(const VARIANT *var)
145 switch(V_VT(var))
147 case VT_I1:
148 case VT_UI1:
149 return V_UI1(var);
151 case VT_I2:
152 case VT_UI2:
153 return V_UI2(var);
155 case VT_I4:
156 case VT_UI4:
157 return V_UI4(var);
159 case VT_I8:
160 case VT_UI8:
161 return V_UI8(var);
163 case VT_INT:
164 case VT_UINT:
165 return V_UINT(var);
167 default:
168 FIXME("unsupported variant type %d\n", V_VT(var));
169 return 0;
173 static HRESULT WINAPI folders_get_Item(ITaskFolderCollection *iface, VARIANT index, ITaskFolder **folder)
175 TaskFolderCollection *folders = impl_from_ITaskFolderCollection(iface);
176 LONG idx;
178 TRACE("%p,%s,%p\n", iface, debugstr_variant(&index), folder);
180 if (!folder) return E_POINTER;
182 if (V_VT(&index) == VT_BSTR)
183 return TaskFolder_create(folders->path, V_BSTR(&index), folder, FALSE);
185 idx = get_var_int(&index);
186 /* collections are 1 based */
187 if (idx < 1 || idx > folders->count)
188 return E_INVALIDARG;
190 return TaskFolder_create(folders->path, folders->list[idx - 1], folder, FALSE);
193 static HRESULT WINAPI folders_get__NewEnum(ITaskFolderCollection *iface, IUnknown **penum)
195 TaskFolderCollection *folders = impl_from_ITaskFolderCollection(iface);
197 TRACE("%p,%p\n", iface, penum);
199 if (!penum) return E_POINTER;
201 return NewEnum_create(folders, penum);
204 static const ITaskFolderCollectionVtbl TaskFolderCollection_vtbl =
206 folders_QueryInterface,
207 folders_AddRef,
208 folders_Release,
209 folders_GetTypeInfoCount,
210 folders_GetTypeInfo,
211 folders_GetIDsOfNames,
212 folders_Invoke,
213 folders_get_Count,
214 folders_get_Item,
215 folders_get__NewEnum
218 HRESULT TaskFolderCollection_create(const WCHAR *path, ITaskFolderCollection **obj)
220 TaskFolderCollection *folders;
221 HRESULT hr;
222 TASK_NAMES list;
223 DWORD start_index, count;
225 start_index = 0;
226 list = NULL;
227 hr = SchRpcEnumFolders(path, 0, &start_index, 0, &count, &list);
228 if (hr != S_OK) return hr;
230 folders = heap_alloc(sizeof(*folders));
231 if (!folders)
233 free_list(list, count);
234 return E_OUTOFMEMORY;
237 folders->ITaskFolderCollection_iface.lpVtbl = &TaskFolderCollection_vtbl;
238 folders->ref = 1;
239 if (!(folders->path = heap_strdupW(path)))
241 heap_free(folders);
242 free_list(list, count);
243 return E_OUTOFMEMORY;
245 folders->count = count;
246 folders->list = list;
247 *obj = &folders->ITaskFolderCollection_iface;
249 TRACE("created %p\n", *obj);
251 return S_OK;
254 typedef struct
256 IEnumVARIANT IEnumVARIANT_iface;
257 LONG ref, pos;
258 TaskFolderCollection *folders;
259 } EnumVARIANT;
261 static inline EnumVARIANT *impl_from_IEnumVARIANT(IEnumVARIANT *iface)
263 return CONTAINING_RECORD(iface, EnumVARIANT, IEnumVARIANT_iface);
266 static HRESULT WINAPI enumvar_QueryInterface(IEnumVARIANT *iface, REFIID riid, void **obj)
268 if (!riid || !obj) return E_INVALIDARG;
270 TRACE("%p,%s,%p\n", iface, debugstr_guid(riid), obj);
272 if (IsEqualGUID(riid, &IID_IEnumVARIANT) ||
273 IsEqualGUID(riid, &IID_IUnknown))
275 IEnumVARIANT_AddRef(iface);
276 *obj = iface;
277 return S_OK;
280 FIXME("interface %s is not implemented\n", debugstr_guid(riid));
281 *obj = NULL;
282 return E_NOINTERFACE;
285 static ULONG WINAPI enumvar_AddRef(IEnumVARIANT *iface)
287 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
288 return InterlockedIncrement(&enumvar->ref);
291 static ULONG WINAPI enumvar_Release(IEnumVARIANT *iface)
293 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
294 LONG ref = InterlockedDecrement(&enumvar->ref);
296 if (!ref)
298 TRACE("destroying %p\n", iface);
299 ITaskFolderCollection_Release(&enumvar->folders->ITaskFolderCollection_iface);
300 heap_free(enumvar);
303 return ref;
306 static HRESULT WINAPI enumvar_Next(IEnumVARIANT *iface, ULONG celt, VARIANT *var, ULONG *fetched)
308 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
309 LONG i;
311 TRACE("%p,%u,%p,%p\n", iface, celt, var, fetched);
313 for (i = 0; i < celt && enumvar->pos < enumvar->folders->count; i++)
315 ITaskFolder *folder;
316 HRESULT hr;
318 hr = TaskFolder_create(enumvar->folders->path, enumvar->folders->list[enumvar->pos++], &folder, FALSE);
319 if (hr) return hr;
321 if (!var)
323 ITaskFolder_Release(folder);
324 return E_POINTER;
327 V_VT(&var[i]) = VT_DISPATCH;
328 V_DISPATCH(&var[i]) = (IDispatch *)folder;
331 if (fetched) *fetched = i;
333 return i == celt ? S_OK : S_FALSE;
336 static HRESULT WINAPI enumvar_Skip(IEnumVARIANT *iface, ULONG celt)
338 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
340 TRACE("%p,%u\n", iface, celt);
342 enumvar->pos += celt;
344 if (enumvar->pos > enumvar->folders->count)
346 enumvar->pos = enumvar->folders->count;
347 return S_FALSE;
350 return S_OK;
353 static HRESULT WINAPI enumvar_Reset(IEnumVARIANT *iface)
355 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
357 TRACE("%p\n", iface);
359 enumvar->pos = 0;
361 return S_OK;
364 static HRESULT WINAPI enumvar_Clone(IEnumVARIANT *iface, IEnumVARIANT **penum)
366 EnumVARIANT *enumvar = impl_from_IEnumVARIANT(iface);
368 TRACE("%p,%p\n", iface, penum);
370 return NewEnum_create(enumvar->folders, (IUnknown **)penum);
373 static const struct IEnumVARIANTVtbl EnumVARIANT_vtbl =
375 enumvar_QueryInterface,
376 enumvar_AddRef,
377 enumvar_Release,
378 enumvar_Next,
379 enumvar_Skip,
380 enumvar_Reset,
381 enumvar_Clone
384 static HRESULT NewEnum_create(TaskFolderCollection *folders, IUnknown **obj)
386 EnumVARIANT *enumvar;
388 enumvar = heap_alloc(sizeof(*enumvar));
389 if (!enumvar) return E_OUTOFMEMORY;
391 enumvar->IEnumVARIANT_iface.lpVtbl = &EnumVARIANT_vtbl;
392 enumvar->ref = 1;
393 enumvar->pos = 0;
394 enumvar->folders = folders;
395 ITaskFolderCollection_AddRef(&folders->ITaskFolderCollection_iface);
397 *obj = (IUnknown *)&enumvar->IEnumVARIANT_iface;
399 TRACE("created %p\n", *obj);
401 return S_OK;