Use aliases for calls to ordinals.
[wine/dcerpc.git] / dlls / devenum / mediacatenum.c
blob5fc04f92eee511819b63ceb75ceeaba7161a1907
1 /*
2 * IEnumMoniker implementation for DEVENUM.dll
4 * Copyright (C) 2002 Robert Shearman
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.
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.
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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 * NOTES ON THIS FILE:
21 * - Implements IEnumMoniker interface which enumerates through moniker
22 * objects created from HKEY_CLASSES/CLSID/{DEVICE_CLSID}/Instance
25 #include "devenum_private.h"
26 #include "vfwmsgs.h"
27 #include "oleauto.h"
29 #include "wine/debug.h"
32 /* #define ICOM_THIS_From_IROTData(class, name) class* This = (class*)(((char*)name)-sizeof(void*)) */
34 WINE_DEFAULT_DEBUG_CHANNEL(devenum);
36 static ULONG WINAPI DEVENUM_IEnumMoniker_AddRef(LPENUMMONIKER iface);
37 static ULONG WINAPI DEVENUM_IMediaCatMoniker_AddRef(LPMONIKER iface);
38 static ULONG WINAPI DEVENUM_IPropertyBag_AddRef(LPPROPERTYBAG iface);
40 typedef struct
42 IPropertyBagVtbl *lpVtbl;
43 DWORD ref;
44 HKEY hkey;
45 } RegPropBagImpl;
48 static HRESULT WINAPI DEVENUM_IPropertyBag_QueryInterface(
49 LPPROPERTYBAG iface,
50 REFIID riid,
51 LPVOID *ppvObj)
53 RegPropBagImpl *This = (RegPropBagImpl *)iface;
54 TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
56 if (This == NULL || ppvObj == NULL) return E_POINTER;
58 if (IsEqualGUID(riid, &IID_IUnknown) ||
59 IsEqualGUID(riid, &IID_IPropertyBag))
61 *ppvObj = (LPVOID)iface;
62 DEVENUM_IPropertyBag_AddRef(iface);
63 return S_OK;
66 FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
67 return E_NOINTERFACE;
70 /**********************************************************************
71 * DEVENUM_IPropertyBag_AddRef (also IUnknown)
73 static ULONG WINAPI DEVENUM_IPropertyBag_AddRef(LPPROPERTYBAG iface)
75 RegPropBagImpl *This = (RegPropBagImpl *)iface;
76 TRACE("\n");
78 return InterlockedIncrement(&This->ref);
81 /**********************************************************************
82 * DEVENUM_IPropertyBag_Release (also IUnknown)
84 static ULONG WINAPI DEVENUM_IPropertyBag_Release(LPPROPERTYBAG iface)
86 RegPropBagImpl *This = (RegPropBagImpl *)iface;
87 ULONG ref;
89 TRACE("\n");
91 ref = InterlockedDecrement(&This->ref);
92 if (ref == 0) {
93 RegCloseKey(This->hkey);
94 CoTaskMemFree(This);
95 DEVENUM_UnlockModule();
97 return ref;
100 static HRESULT WINAPI DEVENUM_IPropertyBag_Read(
101 LPPROPERTYBAG iface,
102 LPCOLESTR pszPropName,
103 VARIANT* pVar,
104 IErrorLog* pErrorLog)
106 LPVOID pData = NULL;
107 LONG received;
108 DWORD type = 0;
109 RegPropBagImpl *This = (RegPropBagImpl *)iface;
110 HRESULT res = S_OK;
111 LONG reswin32;
113 TRACE("(%p)->(%s, %p, %p)\n", This, debugstr_w(pszPropName), pVar, pErrorLog);
115 if (!pszPropName || !pVar)
116 return E_POINTER;
118 reswin32 = RegQueryValueExW(This->hkey, pszPropName, NULL, NULL, NULL, &received);
119 res = HRESULT_FROM_WIN32(reswin32);
121 if (SUCCEEDED(res))
123 pData = HeapAlloc(GetProcessHeap(), 0, received);
125 /* work around a GCC bug that occurs here unless we use the reswin32 variable as well */
126 reswin32 = RegQueryValueExW(This->hkey, pszPropName, NULL, &type, pData, &received);
127 res = HRESULT_FROM_WIN32(reswin32);
130 if (SUCCEEDED(res))
132 res = E_INVALIDARG; /* assume we cannot coerce into right type */
134 TRACE("Read %ld bytes (%s)\n", received, type == REG_SZ ? debugstr_w((LPWSTR)pData) : "binary data");
136 switch (type)
138 case REG_SZ:
139 switch (V_VT(pVar))
141 case VT_LPWSTR:
142 V_UNION(pVar, bstrVal) = CoTaskMemAlloc(received);
143 memcpy(V_UNION(pVar, bstrVal), (LPWSTR)pData, received);
144 res = S_OK;
145 break;
146 case VT_EMPTY:
147 V_VT(pVar) = VT_BSTR;
148 /* fall through */
149 case VT_BSTR:
150 V_UNION(pVar, bstrVal) = SysAllocStringLen((LPWSTR)pData, received/sizeof(WCHAR) - 1);
151 res = S_OK;
152 break;
154 break;
155 case REG_DWORD:
156 TRACE("REG_DWORD: %lx\n", *(DWORD *)pData);
157 switch (V_VT(pVar))
159 case VT_EMPTY:
160 V_VT(pVar) = VT_I4;
161 /* fall through */
162 case VT_I4:
163 case VT_UI4:
164 V_UNION(pVar, ulVal) = *(DWORD *)pData;
165 res = S_OK;
166 break;
168 break;
169 case REG_BINARY:
171 SAFEARRAYBOUND bound;
172 void * pArrayElements;
173 bound.lLbound = 0;
174 bound.cElements = received;
175 TRACE("REG_BINARY: len = %ld\n", received);
176 switch (V_VT(pVar))
178 case VT_EMPTY:
179 case VT_ARRAY | VT_UI1:
180 if (!(V_UNION(pVar, parray) = SafeArrayCreate(VT_UI1, 1, &bound)))
181 res = E_OUTOFMEMORY;
182 res = S_OK;
183 break;
186 if (res == E_INVALIDARG)
187 break;
189 res = SafeArrayAccessData(V_UNION(pVar, parray), &pArrayElements);
190 if (FAILED(res))
191 break;
193 CopyMemory(pArrayElements, pData, received);
194 res = SafeArrayUnaccessData(V_UNION(pVar, parray));
195 break;
198 if (res == E_INVALIDARG)
199 FIXME("Variant type %x not supported for regtype %lx\n", V_VT(pVar), type);
202 HeapFree(GetProcessHeap(), 0, pData);
204 TRACE("<- %lx\n", res);
205 return res;
208 static HRESULT WINAPI DEVENUM_IPropertyBag_Write(
209 LPPROPERTYBAG iface,
210 LPCOLESTR pszPropName,
211 VARIANT* pVar)
213 RegPropBagImpl *This = (RegPropBagImpl *)iface;
214 LPVOID lpData = NULL;
215 DWORD cbData = 0;
216 DWORD dwType = 0;
217 HRESULT res = S_OK;
219 TRACE("(%p)->(%s, %p)\n", This, debugstr_w(pszPropName), pVar);
221 switch (V_VT(pVar))
223 case VT_BSTR:
224 TRACE("writing %s\n", debugstr_w(V_UNION(pVar, bstrVal)));
225 lpData = (LPVOID)V_UNION(pVar, bstrVal);
226 dwType = REG_SZ;
227 cbData = (lstrlenW(V_UNION(pVar, bstrVal)) + 1) * sizeof(WCHAR);
228 break;
229 case VT_I4:
230 case VT_UI4:
231 TRACE("writing %lu\n", V_UNION(pVar, ulVal));
232 lpData = (LPVOID)&V_UNION(pVar, ulVal);
233 dwType = REG_DWORD;
234 cbData = sizeof(DWORD);
235 break;
236 case VT_ARRAY | VT_UI1:
238 LONG lUbound = 0;
239 LONG lLbound = 0;
240 dwType = REG_BINARY;
241 res = SafeArrayGetLBound(V_UNION(pVar, parray), 1, &lLbound);
242 res = SafeArrayGetUBound(V_UNION(pVar, parray), 1, &lUbound);
243 cbData = (lUbound - lLbound + 1) /* * sizeof(BYTE)*/;
244 TRACE("cbData: %ld\n", cbData);
245 res = SafeArrayAccessData(V_UNION(pVar, parray), &lpData);
246 break;
248 default:
249 FIXME("Variant type %d not handled\n", V_VT(pVar));
250 return E_FAIL;
253 if (RegSetValueExW(This->hkey,
254 pszPropName, 0,
255 dwType, lpData, cbData) != ERROR_SUCCESS)
256 res = E_FAIL;
258 if (V_VT(pVar) & VT_ARRAY)
259 res = SafeArrayUnaccessData(V_UNION(pVar, parray));
261 return res;
264 static IPropertyBagVtbl IPropertyBag_Vtbl =
266 DEVENUM_IPropertyBag_QueryInterface,
267 DEVENUM_IPropertyBag_AddRef,
268 DEVENUM_IPropertyBag_Release,
269 DEVENUM_IPropertyBag_Read,
270 DEVENUM_IPropertyBag_Write
273 static HRESULT DEVENUM_IPropertyBag_Construct(HANDLE hkey, IPropertyBag **ppBag)
275 RegPropBagImpl * rpb = CoTaskMemAlloc(sizeof(RegPropBagImpl));
276 if (!rpb)
277 return E_OUTOFMEMORY;
278 rpb->lpVtbl = &IPropertyBag_Vtbl;
279 rpb->ref = 1;
280 rpb->hkey = hkey;
281 *ppBag = (IPropertyBag*)rpb;
282 DEVENUM_LockModule();
283 return S_OK;
287 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_QueryInterface(
288 LPMONIKER iface,
289 REFIID riid,
290 LPVOID *ppvObj)
292 MediaCatMoniker *This = (MediaCatMoniker *)iface;
293 TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
295 if (This == NULL || ppvObj == NULL) return E_POINTER;
297 *ppvObj = NULL;
299 if (IsEqualGUID(riid, &IID_IUnknown) ||
300 IsEqualGUID(riid, &IID_IPersist) ||
301 IsEqualGUID(riid, &IID_IPersistStream) ||
302 IsEqualGUID(riid, &IID_IMoniker))
304 *ppvObj = (LPVOID)iface;
305 DEVENUM_IMediaCatMoniker_AddRef(iface);
306 return S_OK;
309 FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
310 return E_NOINTERFACE;
313 /**********************************************************************
314 * DEVENUM_IMediaCatMoniker_AddRef (also IUnknown)
316 static ULONG WINAPI DEVENUM_IMediaCatMoniker_AddRef(LPMONIKER iface)
318 MediaCatMoniker *This = (MediaCatMoniker *)iface;
319 TRACE("\n");
321 return InterlockedIncrement(&This->ref);
324 /**********************************************************************
325 * DEVENUM_IMediaCatMoniker_Release (also IUnknown)
327 static ULONG WINAPI DEVENUM_IMediaCatMoniker_Release(LPMONIKER iface)
329 MediaCatMoniker *This = (MediaCatMoniker *)iface;
330 ULONG ref;
331 TRACE("\n");
333 ref = InterlockedDecrement(&This->ref);
334 if (ref == 0) {
335 RegCloseKey(This->hkey);
336 CoTaskMemFree(This);
337 DEVENUM_UnlockModule();
339 return ref;
342 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetClassID(
343 LPMONIKER iface,
344 CLSID* pClassID)
346 MediaCatMoniker *This = (MediaCatMoniker *)iface;
347 FIXME("(%p)->(%p)\n", This, pClassID);
349 if (pClassID == NULL)
350 return E_POINTER;
352 return E_NOTIMPL;
355 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsDirty(LPMONIKER iface)
357 FIXME("()\n");
359 return S_FALSE;
362 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Load(LPMONIKER iface, IStream* pStm)
364 FIXME("(%p)\n", pStm);
366 return E_NOTIMPL;
369 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Save(LPMONIKER iface, IStream* pStm, BOOL fClearDirty)
371 FIXME("(%p, %s)\n", pStm, fClearDirty ? "true" : "false");
373 return STG_E_CANTSAVE;
376 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetSizeMax(
377 LPMONIKER iface,
378 ULARGE_INTEGER* pcbSize)
380 FIXME("(%p)\n", pcbSize);
382 ZeroMemory(pcbSize, sizeof(*pcbSize));
384 return S_OK;
387 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToObject(
388 LPMONIKER iface,
389 IBindCtx* pbc,
390 IMoniker* pmkToLeft,
391 REFIID riidResult,
392 void** ppvResult)
394 IUnknown * pObj = NULL;
395 IPropertyBag * pProp = NULL;
396 CLSID clsID;
397 VARIANT var;
398 HRESULT res = E_FAIL;
400 MediaCatMoniker *This = (MediaCatMoniker *)iface;
402 VariantClear(&var);
404 TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riidResult), ppvResult);
406 *ppvResult = NULL;
408 if(pmkToLeft==NULL)
410 /* first activation of this class */
411 LPVOID pvptr;
412 res=IMoniker_BindToStorage(iface, NULL, NULL, &IID_IPropertyBag, &pvptr);
413 pProp = (IPropertyBag*)pvptr;
414 if (SUCCEEDED(res))
416 V_VT(&var) = VT_LPWSTR;
417 res = IPropertyBag_Read(pProp, clsid_keyname, &var, NULL);
419 if (SUCCEEDED(res))
421 res = CLSIDFromString(V_UNION(&var,bstrVal), &clsID);
422 CoTaskMemFree(V_UNION(&var, bstrVal));
424 if (SUCCEEDED(res))
426 res=CoCreateInstance(&clsID,NULL,CLSCTX_ALL,&IID_IUnknown,&pvptr);
427 pObj = (IUnknown*)pvptr;
431 if (pObj!=NULL)
433 /* get the requested interface from the loaded class */
434 res= IUnknown_QueryInterface(pObj,riidResult,ppvResult);
437 if (pProp)
439 IPropertyBag_Release(pProp);
442 TRACE("<- 0x%lx\n", res);
444 return res;
447 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToStorage(
448 LPMONIKER iface,
449 IBindCtx* pbc,
450 IMoniker* pmkToLeft,
451 REFIID riid,
452 void** ppvObj)
454 MediaCatMoniker *This = (MediaCatMoniker *)iface;
455 TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riid), ppvObj);
457 *ppvObj = NULL;
459 if (pbc || pmkToLeft)
460 return MK_E_NOSTORAGE;
462 if (IsEqualGUID(riid, &IID_IPropertyBag))
464 HANDLE hkey;
465 DuplicateHandle(GetCurrentProcess(), This->hkey, GetCurrentProcess(), &hkey, 0, 0, DUPLICATE_SAME_ACCESS);
466 return DEVENUM_IPropertyBag_Construct(hkey, (IPropertyBag**)ppvObj);
469 return MK_E_NOSTORAGE;
472 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Reduce(
473 LPMONIKER iface,
474 IBindCtx* pbc,
475 DWORD dwReduceHowFar,
476 IMoniker** ppmkToLeft,
477 IMoniker** ppmkReduced)
479 TRACE("(%p, %ld, %p, %p)\n", pbc, dwReduceHowFar, ppmkToLeft, ppmkReduced);
481 if (ppmkToLeft)
482 *ppmkToLeft = NULL;
483 *ppmkReduced = iface;
485 return MK_S_REDUCED_TO_SELF;
488 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ComposeWith(
489 LPMONIKER iface,
490 IMoniker* pmkRight,
491 BOOL fOnlyIfNotGeneric,
492 IMoniker** ppmkComposite)
494 FIXME("(%p, %s, %p): stub\n", pmkRight, fOnlyIfNotGeneric ? "true" : "false", ppmkComposite);
496 /* FIXME: use CreateGenericComposite? */
497 *ppmkComposite = NULL;
499 return E_NOTIMPL;
502 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Enum(
503 LPMONIKER iface,
504 BOOL fForward,
505 IEnumMoniker** ppenumMoniker)
507 FIXME("(%s, %p): stub\n", fForward ? "true" : "false", ppenumMoniker);
509 *ppenumMoniker = NULL;
511 return S_OK;
514 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsEqual(
515 LPMONIKER iface,
516 IMoniker* pmkOtherMoniker)
518 FIXME("(%p)\n", pmkOtherMoniker);
520 return E_NOTIMPL;
523 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Hash(
524 LPMONIKER iface,
525 DWORD* pdwHash)
527 TRACE("(%p)\n", pdwHash);
529 *pdwHash = 0;
531 return S_OK;
534 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsRunning(
535 LPMONIKER iface,
536 IBindCtx* pbc,
537 IMoniker* pmkToLeft,
538 IMoniker* pmkNewlyRunning)
540 FIXME("(%p, %p, %p)\n", pbc, pmkToLeft, pmkNewlyRunning);
542 return S_FALSE;
545 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetTimeOfLastChange(
546 LPMONIKER iface,
547 IBindCtx* pbc,
548 IMoniker* pmkToLeft,
549 FILETIME* pFileTime)
551 TRACE("(%p, %p, %p)\n", pbc, pmkToLeft, pFileTime);
553 pFileTime->dwLowDateTime = 0xFFFFFFFF;
554 pFileTime->dwHighDateTime = 0x7FFFFFFF;
556 return MK_E_UNAVAILABLE;
559 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Inverse(
560 LPMONIKER iface,
561 IMoniker** ppmk)
563 TRACE("(%p)\n", ppmk);
565 *ppmk = NULL;
567 return MK_E_NOINVERSE;
570 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_CommonPrefixWith(
571 LPMONIKER iface,
572 IMoniker* pmkOtherMoniker,
573 IMoniker** ppmkPrefix)
575 TRACE("(%p, %p)\n", pmkOtherMoniker, ppmkPrefix);
577 *ppmkPrefix = NULL;
579 return MK_E_NOPREFIX;
582 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_RelativePathTo(
583 LPMONIKER iface,
584 IMoniker* pmkOther,
585 IMoniker** ppmkRelPath)
587 TRACE("(%p, %p)\n", pmkOther, ppmkRelPath);
589 *ppmkRelPath = pmkOther;
591 return MK_S_HIM;
594 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetDisplayName(
595 LPMONIKER iface,
596 IBindCtx* pbc,
597 IMoniker* pmkToLeft,
598 LPOLESTR* ppszDisplayName)
600 MediaCatMoniker *This = (MediaCatMoniker *)iface;
601 WCHAR wszBuffer[MAX_PATH];
602 static const WCHAR wszFriendlyName[] = {'F','r','i','e','n','d','l','y','N','a','m','e',0};
603 LONG received = sizeof(wszFriendlyName);
605 TRACE("(%p, %p, %p)\n", pbc, pmkToLeft, ppszDisplayName);
607 *ppszDisplayName = NULL;
609 /* FIXME: should this be the weird stuff we have to parse in IParseDisplayName? */
610 if (RegQueryValueW(This->hkey, wszFriendlyName, wszBuffer, &received) == ERROR_SUCCESS)
612 *ppszDisplayName = CoTaskMemAlloc(received);
613 strcpyW(*ppszDisplayName, wszBuffer);
614 return S_OK;
617 return E_FAIL;
620 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ParseDisplayName(
621 LPMONIKER iface,
622 IBindCtx* pbc,
623 IMoniker* pmkToLeft,
624 LPOLESTR pszDisplayName,
625 ULONG* pchEaten,
626 IMoniker** ppmkOut)
628 FIXME("(%p, %p, %s, %p, %p)\n", pbc, pmkToLeft, debugstr_w(pszDisplayName), pchEaten, ppmkOut);
630 *pchEaten = 0;
631 *ppmkOut = NULL;
633 return MK_E_SYNTAX;
636 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsSystemMoniker(
637 LPMONIKER iface,
638 DWORD* pdwMksys)
640 TRACE("(%p)\n", pdwMksys);
642 return S_FALSE;
645 static IMonikerVtbl IMoniker_Vtbl =
647 DEVENUM_IMediaCatMoniker_QueryInterface,
648 DEVENUM_IMediaCatMoniker_AddRef,
649 DEVENUM_IMediaCatMoniker_Release,
650 DEVENUM_IMediaCatMoniker_GetClassID,
651 DEVENUM_IMediaCatMoniker_IsDirty,
652 DEVENUM_IMediaCatMoniker_Load,
653 DEVENUM_IMediaCatMoniker_Save,
654 DEVENUM_IMediaCatMoniker_GetSizeMax,
655 DEVENUM_IMediaCatMoniker_BindToObject,
656 DEVENUM_IMediaCatMoniker_BindToStorage,
657 DEVENUM_IMediaCatMoniker_Reduce,
658 DEVENUM_IMediaCatMoniker_ComposeWith,
659 DEVENUM_IMediaCatMoniker_Enum,
660 DEVENUM_IMediaCatMoniker_IsEqual,
661 DEVENUM_IMediaCatMoniker_Hash,
662 DEVENUM_IMediaCatMoniker_IsRunning,
663 DEVENUM_IMediaCatMoniker_GetTimeOfLastChange,
664 DEVENUM_IMediaCatMoniker_Inverse,
665 DEVENUM_IMediaCatMoniker_CommonPrefixWith,
666 DEVENUM_IMediaCatMoniker_RelativePathTo,
667 DEVENUM_IMediaCatMoniker_GetDisplayName,
668 DEVENUM_IMediaCatMoniker_ParseDisplayName,
669 DEVENUM_IMediaCatMoniker_IsSystemMoniker
672 MediaCatMoniker * DEVENUM_IMediaCatMoniker_Construct()
674 MediaCatMoniker * pMoniker = NULL;
675 pMoniker = CoTaskMemAlloc(sizeof(MediaCatMoniker));
676 if (!pMoniker)
677 return NULL;
679 pMoniker->lpVtbl = &IMoniker_Vtbl;
680 pMoniker->ref = 0;
681 pMoniker->hkey = NULL;
683 DEVENUM_IMediaCatMoniker_AddRef((LPMONIKER)pMoniker);
685 DEVENUM_LockModule();
687 return pMoniker;
690 /**********************************************************************
691 * DEVENUM_IEnumMoniker_QueryInterface (also IUnknown)
693 static HRESULT WINAPI DEVENUM_IEnumMoniker_QueryInterface(
694 LPENUMMONIKER iface,
695 REFIID riid,
696 LPVOID *ppvObj)
698 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
699 TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
701 if (This == NULL || ppvObj == NULL) return E_POINTER;
703 if (IsEqualGUID(riid, &IID_IUnknown) ||
704 IsEqualGUID(riid, &IID_IEnumMoniker))
706 *ppvObj = (LPVOID)iface;
707 DEVENUM_IEnumMoniker_AddRef(iface);
708 return S_OK;
711 FIXME("- no interface\n\tIID:\t%s\n", debugstr_guid(riid));
712 return E_NOINTERFACE;
715 /**********************************************************************
716 * DEVENUM_IEnumMoniker_AddRef (also IUnknown)
718 static ULONG WINAPI DEVENUM_IEnumMoniker_AddRef(LPENUMMONIKER iface)
720 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
721 ULONG ref = InterlockedIncrement(&This->ref);
723 TRACE("(%p)->() AddRef from %ld\n", iface, ref - 1);
725 return ref;
728 /**********************************************************************
729 * DEVENUM_IEnumMoniker_Release (also IUnknown)
731 static ULONG WINAPI DEVENUM_IEnumMoniker_Release(LPENUMMONIKER iface)
733 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
734 ULONG ref = InterlockedDecrement(&This->ref);
736 TRACE("(%p)->() Release from %ld\n", iface, ref + 1);
738 if (!ref)
740 RegCloseKey(This->hkey);
741 CoTaskMemFree(This);
742 DEVENUM_UnlockModule();
743 return 0;
745 return ref;
748 static HRESULT WINAPI DEVENUM_IEnumMoniker_Next(LPENUMMONIKER iface, ULONG celt, IMoniker ** rgelt, ULONG * pceltFetched)
750 WCHAR buffer[MAX_PATH + 1];
751 LONG res;
752 ULONG fetched = 0;
753 MediaCatMoniker * pMoniker;
754 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
756 TRACE("(%p)->(%ld, %p, %p)\n", iface, celt, rgelt, pceltFetched);
758 while (fetched < celt)
760 res = RegEnumKeyW(This->hkey, This->index, buffer, sizeof(buffer) / sizeof(WCHAR));
761 if (res != ERROR_SUCCESS)
763 break;
765 pMoniker = DEVENUM_IMediaCatMoniker_Construct();
766 if (!pMoniker)
767 return E_OUTOFMEMORY;
769 if (RegOpenKeyW(This->hkey, buffer, &pMoniker->hkey) != ERROR_SUCCESS)
771 DEVENUM_IMediaCatMoniker_Release((LPMONIKER)pMoniker);
772 break;
774 rgelt[fetched] = (LPMONIKER)pMoniker;
775 fetched++;
778 This->index += fetched;
780 TRACE("-- fetched %ld\n", fetched);
782 if (pceltFetched)
783 *pceltFetched = fetched;
785 if (fetched != celt)
786 return S_FALSE;
787 else
788 return S_OK;
791 static HRESULT WINAPI DEVENUM_IEnumMoniker_Skip(LPENUMMONIKER iface, ULONG celt)
793 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
795 TRACE("(%p)->(%ld)\n", iface, celt);
797 This->index += celt;
799 return S_OK;
802 static HRESULT WINAPI DEVENUM_IEnumMoniker_Reset(LPENUMMONIKER iface)
804 EnumMonikerImpl *This = (EnumMonikerImpl *)iface;
806 TRACE("(%p)->()\n", iface);
808 This->index = 0;
810 return S_OK;
813 static HRESULT WINAPI DEVENUM_IEnumMoniker_Clone(LPENUMMONIKER iface, IEnumMoniker ** ppenum)
815 FIXME("(%p)->(%p): stub\n", iface, ppenum);
817 return E_NOTIMPL;
820 /**********************************************************************
821 * IEnumMoniker_Vtbl
823 static IEnumMonikerVtbl IEnumMoniker_Vtbl =
825 DEVENUM_IEnumMoniker_QueryInterface,
826 DEVENUM_IEnumMoniker_AddRef,
827 DEVENUM_IEnumMoniker_Release,
828 DEVENUM_IEnumMoniker_Next,
829 DEVENUM_IEnumMoniker_Skip,
830 DEVENUM_IEnumMoniker_Reset,
831 DEVENUM_IEnumMoniker_Clone
834 HRESULT DEVENUM_IEnumMoniker_Construct(HKEY hkey, IEnumMoniker ** ppEnumMoniker)
836 EnumMonikerImpl * pEnumMoniker = CoTaskMemAlloc(sizeof(EnumMonikerImpl));
837 if (!pEnumMoniker)
838 return E_OUTOFMEMORY;
840 pEnumMoniker->lpVtbl = &IEnumMoniker_Vtbl;
841 pEnumMoniker->ref = 1;
842 pEnumMoniker->index = 0;
843 pEnumMoniker->hkey = hkey;
845 *ppEnumMoniker = (IEnumMoniker *)pEnumMoniker;
847 DEVENUM_LockModule();
849 return S_OK;