Release 4.0.4.
[wine.git] / dlls / devenum / mediacatenum.c
blob5fa2f008dcd0e74eb8b7676e8b81815210d1d363
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., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, 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 "oleauto.h"
27 #include "ocidl.h"
28 #include "dmoreg.h"
30 #include "wine/debug.h"
32 WINE_DEFAULT_DEBUG_CHANNEL(devenum);
34 typedef struct
36 IEnumMoniker IEnumMoniker_iface;
37 CLSID class;
38 LONG ref;
39 IEnumDMO *dmo_enum;
40 HKEY sw_key;
41 DWORD sw_index;
42 HKEY cm_key;
43 DWORD cm_index;
44 } EnumMonikerImpl;
46 typedef struct
48 IPropertyBag IPropertyBag_iface;
49 LONG ref;
50 enum device_type type;
51 union
53 WCHAR path[MAX_PATH]; /* for filters and codecs */
54 CLSID clsid; /* for DMOs */
56 } RegPropBagImpl;
59 static inline RegPropBagImpl *impl_from_IPropertyBag(IPropertyBag *iface)
61 return CONTAINING_RECORD(iface, RegPropBagImpl, IPropertyBag_iface);
64 static HRESULT WINAPI DEVENUM_IPropertyBag_QueryInterface(
65 LPPROPERTYBAG iface,
66 REFIID riid,
67 LPVOID *ppvObj)
69 RegPropBagImpl *This = impl_from_IPropertyBag(iface);
71 TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), ppvObj);
73 if (This == NULL || ppvObj == NULL) return E_POINTER;
75 if (IsEqualGUID(riid, &IID_IUnknown) ||
76 IsEqualGUID(riid, &IID_IPropertyBag))
78 *ppvObj = iface;
79 IPropertyBag_AddRef(iface);
80 return S_OK;
83 FIXME("- no interface IID: %s\n", debugstr_guid(riid));
84 return E_NOINTERFACE;
87 /**********************************************************************
88 * DEVENUM_IPropertyBag_AddRef (also IUnknown)
90 static ULONG WINAPI DEVENUM_IPropertyBag_AddRef(LPPROPERTYBAG iface)
92 RegPropBagImpl *This = impl_from_IPropertyBag(iface);
94 TRACE("(%p)->() AddRef from %d\n", iface, This->ref);
96 return InterlockedIncrement(&This->ref);
99 /**********************************************************************
100 * DEVENUM_IPropertyBag_Release (also IUnknown)
102 static ULONG WINAPI DEVENUM_IPropertyBag_Release(LPPROPERTYBAG iface)
104 RegPropBagImpl *This = impl_from_IPropertyBag(iface);
105 ULONG ref;
107 TRACE("(%p)->() ReleaseThis->ref from %d\n", iface, This->ref);
109 ref = InterlockedDecrement(&This->ref);
110 if (ref == 0) {
111 CoTaskMemFree(This);
112 DEVENUM_UnlockModule();
114 return ref;
117 static HRESULT WINAPI DEVENUM_IPropertyBag_Read(
118 LPPROPERTYBAG iface,
119 LPCOLESTR pszPropName,
120 VARIANT* pVar,
121 IErrorLog* pErrorLog)
123 static const WCHAR FriendlyNameW[] = {'F','r','i','e','n','d','l','y','N','a','m','e',0};
124 LPVOID pData = NULL;
125 DWORD received;
126 DWORD type = 0;
127 RegPropBagImpl *This = impl_from_IPropertyBag(iface);
128 HRESULT res = S_OK;
129 LONG reswin32 = ERROR_SUCCESS;
130 WCHAR name[80];
131 HKEY hkey;
133 TRACE("(%p)->(%s, %p, %p)\n", This, debugstr_w(pszPropName), pVar, pErrorLog);
135 if (!pszPropName || !pVar)
136 return E_POINTER;
138 if (This->type == DEVICE_DMO)
140 if (!strcmpW(pszPropName, FriendlyNameW))
142 res = DMOGetName(&This->clsid, name);
143 if (SUCCEEDED(res))
145 V_VT(pVar) = VT_BSTR;
146 V_BSTR(pVar) = SysAllocString(name);
148 return res;
150 return HRESULT_FROM_WIN32(ERROR_NOT_FOUND);
153 if (This->type == DEVICE_FILTER)
154 reswin32 = RegOpenKeyW(HKEY_CLASSES_ROOT, This->path, &hkey);
155 else if (This->type == DEVICE_CODEC)
156 reswin32 = RegOpenKeyW(HKEY_CURRENT_USER, This->path, &hkey);
157 res = HRESULT_FROM_WIN32(reswin32);
159 if (SUCCEEDED(res))
161 reswin32 = RegQueryValueExW(hkey, pszPropName, NULL, NULL, NULL, &received);
162 res = HRESULT_FROM_WIN32(reswin32);
165 if (SUCCEEDED(res))
167 pData = HeapAlloc(GetProcessHeap(), 0, received);
169 /* work around a GCC bug that occurs here unless we use the reswin32 variable as well */
170 reswin32 = RegQueryValueExW(hkey, pszPropName, NULL, &type, pData, &received);
171 res = HRESULT_FROM_WIN32(reswin32);
174 if (SUCCEEDED(res))
176 res = E_INVALIDARG; /* assume we cannot coerce into right type */
178 TRACE("Read %d bytes (%s)\n", received, type == REG_SZ ? debugstr_w(pData) : "binary data");
180 switch (type)
182 case REG_SZ:
183 switch (V_VT(pVar))
185 case VT_LPWSTR:
186 V_BSTR(pVar) = CoTaskMemAlloc(received);
187 memcpy(V_BSTR(pVar), pData, received);
188 res = S_OK;
189 break;
190 case VT_EMPTY:
191 V_VT(pVar) = VT_BSTR;
192 /* fall through */
193 case VT_BSTR:
194 V_BSTR(pVar) = SysAllocStringLen(pData, received/sizeof(WCHAR) - 1);
195 res = S_OK;
196 break;
198 break;
199 case REG_DWORD:
200 TRACE("REG_DWORD: %x\n", *(DWORD *)pData);
201 switch (V_VT(pVar))
203 case VT_EMPTY:
204 V_VT(pVar) = VT_I4;
205 /* fall through */
206 case VT_I4:
207 case VT_UI4:
208 V_I4(pVar) = *(DWORD *)pData;
209 res = S_OK;
210 break;
212 break;
213 case REG_BINARY:
215 SAFEARRAYBOUND bound;
216 void * pArrayElements;
217 bound.lLbound = 0;
218 bound.cElements = received;
219 TRACE("REG_BINARY: len = %d\n", received);
220 switch (V_VT(pVar))
222 case VT_EMPTY:
223 V_VT(pVar) = VT_ARRAY | VT_UI1;
224 /* fall through */
225 case VT_ARRAY | VT_UI1:
226 if (!(V_ARRAY(pVar) = SafeArrayCreate(VT_UI1, 1, &bound)))
227 res = E_OUTOFMEMORY;
228 else
229 res = S_OK;
230 break;
233 if (res == E_INVALIDARG)
234 break;
236 res = SafeArrayAccessData(V_ARRAY(pVar), &pArrayElements);
237 if (FAILED(res))
238 break;
240 CopyMemory(pArrayElements, pData, received);
241 res = SafeArrayUnaccessData(V_ARRAY(pVar));
242 break;
245 if (res == E_INVALIDARG)
246 FIXME("Variant type %x not supported for regtype %x\n", V_VT(pVar), type);
249 HeapFree(GetProcessHeap(), 0, pData);
251 RegCloseKey(hkey);
253 TRACE("<- %x\n", res);
254 return res;
257 static HRESULT WINAPI DEVENUM_IPropertyBag_Write(
258 LPPROPERTYBAG iface,
259 LPCOLESTR pszPropName,
260 VARIANT* pVar)
262 RegPropBagImpl *This = impl_from_IPropertyBag(iface);
263 LPVOID lpData = NULL;
264 DWORD cbData = 0;
265 DWORD dwType = 0;
266 HRESULT res = S_OK;
267 LONG lres = ERROR_SUCCESS;
268 HKEY hkey;
270 TRACE("(%p)->(%s, %p)\n", This, debugstr_w(pszPropName), pVar);
272 if (This->type == DEVICE_DMO)
273 return E_ACCESSDENIED;
275 switch (V_VT(pVar))
277 case VT_BSTR:
278 case VT_LPWSTR:
279 TRACE("writing %s\n", debugstr_w(V_BSTR(pVar)));
280 lpData = V_BSTR(pVar);
281 dwType = REG_SZ;
282 cbData = (lstrlenW(V_BSTR(pVar)) + 1) * sizeof(WCHAR);
283 break;
284 case VT_I4:
285 case VT_UI4:
286 TRACE("writing %u\n", V_UI4(pVar));
287 lpData = &V_UI4(pVar);
288 dwType = REG_DWORD;
289 cbData = sizeof(DWORD);
290 break;
291 case VT_ARRAY | VT_UI1:
293 LONG lUbound = 0;
294 LONG lLbound = 0;
295 dwType = REG_BINARY;
296 res = SafeArrayGetLBound(V_ARRAY(pVar), 1, &lLbound);
297 res = SafeArrayGetUBound(V_ARRAY(pVar), 1, &lUbound);
298 cbData = (lUbound - lLbound + 1) /* * sizeof(BYTE)*/;
299 TRACE("cbData: %d\n", cbData);
300 res = SafeArrayAccessData(V_ARRAY(pVar), &lpData);
301 break;
303 default:
304 FIXME("Variant type %d not handled\n", V_VT(pVar));
305 return E_FAIL;
308 if (This->type == DEVICE_FILTER)
309 lres = RegCreateKeyW(HKEY_CLASSES_ROOT, This->path, &hkey);
310 else if (This->type == DEVICE_CODEC)
311 lres = RegCreateKeyW(HKEY_CURRENT_USER, This->path, &hkey);
312 res = HRESULT_FROM_WIN32(lres);
314 if (SUCCEEDED(res))
316 lres = RegSetValueExW(hkey, pszPropName, 0, dwType, lpData, cbData);
317 res = HRESULT_FROM_WIN32(lres);
318 RegCloseKey(hkey);
321 if (V_VT(pVar) & VT_ARRAY)
322 res = SafeArrayUnaccessData(V_ARRAY(pVar));
324 return res;
327 static const IPropertyBagVtbl IPropertyBag_Vtbl =
329 DEVENUM_IPropertyBag_QueryInterface,
330 DEVENUM_IPropertyBag_AddRef,
331 DEVENUM_IPropertyBag_Release,
332 DEVENUM_IPropertyBag_Read,
333 DEVENUM_IPropertyBag_Write
336 static HRESULT create_PropertyBag(MediaCatMoniker *mon, IPropertyBag **ppBag)
338 RegPropBagImpl * rpb = CoTaskMemAlloc(sizeof(RegPropBagImpl));
339 if (!rpb)
340 return E_OUTOFMEMORY;
341 rpb->IPropertyBag_iface.lpVtbl = &IPropertyBag_Vtbl;
342 rpb->ref = 1;
343 rpb->type = mon->type;
345 if (rpb->type == DEVICE_DMO)
346 rpb->clsid = mon->clsid;
347 else if (rpb->type == DEVICE_FILTER)
349 strcpyW(rpb->path, clsidW);
350 strcatW(rpb->path, backslashW);
351 if (mon->has_class)
353 StringFromGUID2(&mon->class, rpb->path + strlenW(rpb->path), CHARS_IN_GUID);
354 strcatW(rpb->path, instanceW);
355 strcatW(rpb->path, backslashW);
357 strcatW(rpb->path, mon->name);
359 else if (rpb->type == DEVICE_CODEC)
361 strcpyW(rpb->path, wszActiveMovieKey);
362 if (mon->has_class)
364 StringFromGUID2(&mon->class, rpb->path + strlenW(rpb->path), CHARS_IN_GUID);
365 strcatW(rpb->path, backslashW);
367 strcatW(rpb->path, mon->name);
370 *ppBag = &rpb->IPropertyBag_iface;
371 DEVENUM_LockModule();
372 return S_OK;
376 static inline MediaCatMoniker *impl_from_IMoniker(IMoniker *iface)
378 return CONTAINING_RECORD(iface, MediaCatMoniker, IMoniker_iface);
381 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_QueryInterface(IMoniker *iface, REFIID riid,
382 void **ppv)
384 TRACE("\n\tIID:\t%s\n",debugstr_guid(riid));
386 if (!ppv)
387 return E_POINTER;
389 if (IsEqualGUID(riid, &IID_IUnknown) ||
390 IsEqualGUID(riid, &IID_IPersist) ||
391 IsEqualGUID(riid, &IID_IPersistStream) ||
392 IsEqualGUID(riid, &IID_IMoniker))
394 *ppv = iface;
395 IMoniker_AddRef(iface);
396 return S_OK;
399 FIXME("- no interface IID: %s\n", debugstr_guid(riid));
400 *ppv = NULL;
401 return E_NOINTERFACE;
404 static ULONG WINAPI DEVENUM_IMediaCatMoniker_AddRef(IMoniker *iface)
406 MediaCatMoniker *This = impl_from_IMoniker(iface);
407 ULONG ref = InterlockedIncrement(&This->ref);
409 TRACE("(%p) ref=%d\n", This, ref);
411 return ref;
414 static ULONG WINAPI DEVENUM_IMediaCatMoniker_Release(IMoniker *iface)
416 MediaCatMoniker *This = impl_from_IMoniker(iface);
417 ULONG ref = InterlockedDecrement(&This->ref);
419 TRACE("(%p) ref=%d\n", This, ref);
421 if (ref == 0) {
422 CoTaskMemFree(This->name);
423 CoTaskMemFree(This);
424 DEVENUM_UnlockModule();
426 return ref;
429 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetClassID(IMoniker *iface, CLSID *pClassID)
431 MediaCatMoniker *This = impl_from_IMoniker(iface);
433 TRACE("(%p)->(%p)\n", This, pClassID);
435 if (pClassID == NULL)
436 return E_INVALIDARG;
438 *pClassID = CLSID_CDeviceMoniker;
440 return S_OK;
443 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsDirty(IMoniker *iface)
445 FIXME("(%p)->(): stub\n", iface);
447 return S_FALSE;
450 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Load(IMoniker *iface, IStream *pStm)
452 FIXME("(%p)->(%p): stub\n", iface, pStm);
454 return E_NOTIMPL;
457 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Save(IMoniker *iface, IStream *pStm, BOOL fClearDirty)
459 FIXME("(%p)->(%p, %s): stub\n", iface, pStm, fClearDirty ? "true" : "false");
461 return STG_E_CANTSAVE;
464 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetSizeMax(IMoniker *iface, ULARGE_INTEGER *pcbSize)
466 FIXME("(%p)->(%p): stub\n", iface, pcbSize);
468 ZeroMemory(pcbSize, sizeof(*pcbSize));
470 return S_OK;
473 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToObject(IMoniker *iface, IBindCtx *pbc,
474 IMoniker *pmkToLeft, REFIID riidResult, void **ppvResult)
476 MediaCatMoniker *This = impl_from_IMoniker(iface);
477 IUnknown * pObj = NULL;
478 IPropertyBag * pProp = NULL;
479 CLSID clsID;
480 VARIANT var;
481 HRESULT res = E_FAIL;
483 TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riidResult), ppvResult);
485 if (!ppvResult)
486 return E_POINTER;
488 VariantInit(&var);
489 *ppvResult = NULL;
491 if(pmkToLeft==NULL)
493 /* first activation of this class */
494 LPVOID pvptr;
495 res=IMoniker_BindToStorage(iface, NULL, NULL, &IID_IPropertyBag, &pvptr);
496 pProp = pvptr;
497 if (SUCCEEDED(res))
499 V_VT(&var) = VT_LPWSTR;
500 res = IPropertyBag_Read(pProp, clsidW, &var, NULL);
502 if (SUCCEEDED(res))
504 res = CLSIDFromString(V_BSTR(&var), &clsID);
505 CoTaskMemFree(V_BSTR(&var));
507 if (SUCCEEDED(res))
509 res=CoCreateInstance(&clsID,NULL,CLSCTX_ALL,&IID_IUnknown,&pvptr);
510 pObj = pvptr;
514 if (pObj!=NULL)
516 /* get the requested interface from the loaded class */
517 res = S_OK;
518 if (pProp) {
519 HRESULT res2;
520 LPVOID ppv = NULL;
521 res2 = IUnknown_QueryInterface(pObj, &IID_IPersistPropertyBag, &ppv);
522 if (SUCCEEDED(res2)) {
523 res = IPersistPropertyBag_Load((IPersistPropertyBag *) ppv, pProp, NULL);
524 IPersistPropertyBag_Release((IPersistPropertyBag *) ppv);
527 if (SUCCEEDED(res))
528 res= IUnknown_QueryInterface(pObj,riidResult,ppvResult);
529 IUnknown_Release(pObj);
532 if (pProp)
534 IPropertyBag_Release(pProp);
537 TRACE("<- 0x%x\n", res);
539 return res;
542 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_BindToStorage(IMoniker *iface, IBindCtx *pbc,
543 IMoniker *pmkToLeft, REFIID riid, void **ppvObj)
545 MediaCatMoniker *This = impl_from_IMoniker(iface);
547 TRACE("(%p)->(%p, %p, %s, %p)\n", This, pbc, pmkToLeft, debugstr_guid(riid), ppvObj);
549 *ppvObj = NULL;
551 if (pmkToLeft)
552 return MK_E_NOSTORAGE;
554 if (pbc != NULL)
556 static DWORD reported;
557 if (!reported)
559 FIXME("ignoring IBindCtx %p\n", pbc);
560 reported++;
564 if (IsEqualGUID(riid, &IID_IPropertyBag))
566 return create_PropertyBag(This, (IPropertyBag**)ppvObj);
569 return MK_E_NOSTORAGE;
572 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Reduce(IMoniker *iface, IBindCtx *pbc,
573 DWORD dwReduceHowFar, IMoniker **ppmkToLeft, IMoniker **ppmkReduced)
575 TRACE("(%p)->(%p, %d, %p, %p)\n", iface, pbc, dwReduceHowFar, ppmkToLeft, ppmkReduced);
577 if (ppmkToLeft)
578 *ppmkToLeft = NULL;
579 *ppmkReduced = iface;
581 return MK_S_REDUCED_TO_SELF;
584 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ComposeWith(IMoniker *iface, IMoniker *pmkRight,
585 BOOL fOnlyIfNotGeneric, IMoniker **ppmkComposite)
587 FIXME("(%p)->(%p, %s, %p): stub\n", iface, pmkRight, fOnlyIfNotGeneric ? "true" : "false", ppmkComposite);
589 /* FIXME: use CreateGenericComposite? */
590 *ppmkComposite = NULL;
592 return E_NOTIMPL;
595 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Enum(IMoniker *iface, BOOL fForward,
596 IEnumMoniker **ppenumMoniker)
598 FIXME("(%p)->(%s, %p): stub\n", iface, fForward ? "true" : "false", ppenumMoniker);
600 *ppenumMoniker = NULL;
602 return S_OK;
605 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsEqual(IMoniker *iface, IMoniker *pmkOtherMoniker)
607 CLSID clsid;
608 LPOLESTR this_name, other_name;
609 IBindCtx *bind;
610 HRESULT res;
612 TRACE("(%p)->(%p)\n", iface, pmkOtherMoniker);
614 if (!pmkOtherMoniker)
615 return E_INVALIDARG;
617 IMoniker_GetClassID(pmkOtherMoniker, &clsid);
618 if (!IsEqualCLSID(&clsid, &CLSID_CDeviceMoniker))
619 return S_FALSE;
621 res = CreateBindCtx(0, &bind);
622 if (FAILED(res))
623 return res;
625 res = S_FALSE;
626 if (SUCCEEDED(IMoniker_GetDisplayName(iface, bind, NULL, &this_name)) &&
627 SUCCEEDED(IMoniker_GetDisplayName(pmkOtherMoniker, bind, NULL, &other_name)))
629 int result = lstrcmpiW(this_name, other_name);
630 CoTaskMemFree(this_name);
631 CoTaskMemFree(other_name);
632 if (!result)
633 res = S_OK;
635 IBindCtx_Release(bind);
636 return res;
639 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Hash(IMoniker *iface, DWORD *pdwHash)
641 TRACE("(%p)->(%p)\n", iface, pdwHash);
643 *pdwHash = 0;
645 return S_OK;
648 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsRunning(IMoniker *iface, IBindCtx *pbc,
649 IMoniker *pmkToLeft, IMoniker *pmkNewlyRunning)
651 FIXME("(%p)->(%p, %p, %p): stub\n", iface, pbc, pmkToLeft, pmkNewlyRunning);
653 return S_FALSE;
656 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetTimeOfLastChange(IMoniker *iface, IBindCtx *pbc,
657 IMoniker *pmkToLeft, FILETIME *pFileTime)
659 TRACE("(%p)->(%p, %p, %p)\n", iface, pbc, pmkToLeft, pFileTime);
661 pFileTime->dwLowDateTime = 0xFFFFFFFF;
662 pFileTime->dwHighDateTime = 0x7FFFFFFF;
664 return MK_E_UNAVAILABLE;
667 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_Inverse(IMoniker *iface, IMoniker **ppmk)
669 TRACE("(%p)->(%p)\n", iface, ppmk);
671 *ppmk = NULL;
673 return MK_E_NOINVERSE;
676 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_CommonPrefixWith(IMoniker *iface,
677 IMoniker *pmkOtherMoniker, IMoniker **ppmkPrefix)
679 TRACE("(%p)->(%p, %p)\n", iface, pmkOtherMoniker, ppmkPrefix);
681 *ppmkPrefix = NULL;
683 return MK_E_NOPREFIX;
686 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_RelativePathTo(IMoniker *iface, IMoniker *pmkOther,
687 IMoniker **ppmkRelPath)
689 TRACE("(%p)->(%p, %p)\n", iface, pmkOther, ppmkRelPath);
691 *ppmkRelPath = pmkOther;
693 return MK_S_HIM;
696 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_GetDisplayName(IMoniker *iface, IBindCtx *pbc,
697 IMoniker *pmkToLeft, LPOLESTR *ppszDisplayName)
699 MediaCatMoniker *This = impl_from_IMoniker(iface);
700 WCHAR *buffer;
702 TRACE("(%p)->(%p, %p, %p)\n", iface, pbc, pmkToLeft, ppszDisplayName);
704 *ppszDisplayName = NULL;
706 if (This->type == DEVICE_DMO)
708 buffer = CoTaskMemAlloc((strlenW(deviceW) + strlenW(dmoW)
709 + 2 * CHARS_IN_GUID + 1) * sizeof(WCHAR));
710 if (!buffer) return E_OUTOFMEMORY;
712 strcpyW(buffer, deviceW);
713 strcatW(buffer, dmoW);
714 StringFromGUID2(&This->clsid, buffer + strlenW(buffer), CHARS_IN_GUID);
715 StringFromGUID2(&This->class, buffer + strlenW(buffer), CHARS_IN_GUID);
717 else
719 buffer = CoTaskMemAlloc((strlenW(deviceW) + 3 + (This->has_class ? CHARS_IN_GUID : 0)
720 + strlenW(This->name) + 1) * sizeof(WCHAR));
721 if (!buffer) return E_OUTOFMEMORY;
723 strcpyW(buffer, deviceW);
724 if (This->type == DEVICE_FILTER)
725 strcatW(buffer, swW);
726 else if (This->type == DEVICE_CODEC)
727 strcatW(buffer, cmW);
729 if (This->has_class)
731 StringFromGUID2(&This->class, buffer + strlenW(buffer), CHARS_IN_GUID);
732 strcatW(buffer, backslashW);
734 strcatW(buffer, This->name);
737 *ppszDisplayName = buffer;
738 return S_OK;
741 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_ParseDisplayName(IMoniker *iface, IBindCtx *pbc,
742 IMoniker *pmkToLeft, LPOLESTR pszDisplayName, ULONG *pchEaten, IMoniker **ppmkOut)
744 FIXME("(%p)->(%p, %p, %s, %p, %p)\n", iface, pbc, pmkToLeft, debugstr_w(pszDisplayName), pchEaten, ppmkOut);
746 *pchEaten = 0;
747 *ppmkOut = NULL;
749 return MK_E_SYNTAX;
752 static HRESULT WINAPI DEVENUM_IMediaCatMoniker_IsSystemMoniker(IMoniker *iface, DWORD *pdwMksys)
754 TRACE("(%p)->(%p)\n", iface, pdwMksys);
756 return S_FALSE;
759 static const IMonikerVtbl IMoniker_Vtbl =
761 DEVENUM_IMediaCatMoniker_QueryInterface,
762 DEVENUM_IMediaCatMoniker_AddRef,
763 DEVENUM_IMediaCatMoniker_Release,
764 DEVENUM_IMediaCatMoniker_GetClassID,
765 DEVENUM_IMediaCatMoniker_IsDirty,
766 DEVENUM_IMediaCatMoniker_Load,
767 DEVENUM_IMediaCatMoniker_Save,
768 DEVENUM_IMediaCatMoniker_GetSizeMax,
769 DEVENUM_IMediaCatMoniker_BindToObject,
770 DEVENUM_IMediaCatMoniker_BindToStorage,
771 DEVENUM_IMediaCatMoniker_Reduce,
772 DEVENUM_IMediaCatMoniker_ComposeWith,
773 DEVENUM_IMediaCatMoniker_Enum,
774 DEVENUM_IMediaCatMoniker_IsEqual,
775 DEVENUM_IMediaCatMoniker_Hash,
776 DEVENUM_IMediaCatMoniker_IsRunning,
777 DEVENUM_IMediaCatMoniker_GetTimeOfLastChange,
778 DEVENUM_IMediaCatMoniker_Inverse,
779 DEVENUM_IMediaCatMoniker_CommonPrefixWith,
780 DEVENUM_IMediaCatMoniker_RelativePathTo,
781 DEVENUM_IMediaCatMoniker_GetDisplayName,
782 DEVENUM_IMediaCatMoniker_ParseDisplayName,
783 DEVENUM_IMediaCatMoniker_IsSystemMoniker
786 MediaCatMoniker * DEVENUM_IMediaCatMoniker_Construct(void)
788 MediaCatMoniker * pMoniker = NULL;
789 pMoniker = CoTaskMemAlloc(sizeof(MediaCatMoniker));
790 if (!pMoniker)
791 return NULL;
793 pMoniker->IMoniker_iface.lpVtbl = &IMoniker_Vtbl;
794 pMoniker->ref = 0;
795 pMoniker->has_class = FALSE;
796 pMoniker->name = NULL;
798 DEVENUM_IMediaCatMoniker_AddRef(&pMoniker->IMoniker_iface);
800 DEVENUM_LockModule();
802 return pMoniker;
805 static inline EnumMonikerImpl *impl_from_IEnumMoniker(IEnumMoniker *iface)
807 return CONTAINING_RECORD(iface, EnumMonikerImpl, IEnumMoniker_iface);
810 static HRESULT WINAPI DEVENUM_IEnumMoniker_QueryInterface(IEnumMoniker *iface, REFIID riid,
811 void **ppv)
813 TRACE("(%p)->(%s, %p)\n", iface, debugstr_guid(riid), ppv);
815 if (!ppv)
816 return E_POINTER;
818 if (IsEqualGUID(riid, &IID_IUnknown) ||
819 IsEqualGUID(riid, &IID_IEnumMoniker))
821 *ppv = iface;
822 IEnumMoniker_AddRef(iface);
823 return S_OK;
826 FIXME("- no interface IID: %s\n", debugstr_guid(riid));
827 *ppv = NULL;
828 return E_NOINTERFACE;
831 static ULONG WINAPI DEVENUM_IEnumMoniker_AddRef(IEnumMoniker *iface)
833 EnumMonikerImpl *This = impl_from_IEnumMoniker(iface);
834 ULONG ref = InterlockedIncrement(&This->ref);
836 TRACE("(%p) ref=%d\n", This, ref);
838 return ref;
841 static ULONG WINAPI DEVENUM_IEnumMoniker_Release(IEnumMoniker *iface)
843 EnumMonikerImpl *This = impl_from_IEnumMoniker(iface);
844 ULONG ref = InterlockedDecrement(&This->ref);
846 TRACE("(%p) ref=%d\n", This, ref);
848 if (!ref)
850 IEnumDMO_Release(This->dmo_enum);
851 RegCloseKey(This->sw_key);
852 RegCloseKey(This->cm_key);
853 CoTaskMemFree(This);
854 DEVENUM_UnlockModule();
855 return 0;
857 return ref;
860 static HRESULT WINAPI DEVENUM_IEnumMoniker_Next(IEnumMoniker *iface, ULONG celt, IMoniker **rgelt,
861 ULONG *pceltFetched)
863 EnumMonikerImpl *This = impl_from_IEnumMoniker(iface);
864 WCHAR buffer[MAX_PATH + 1];
865 LONG res;
866 ULONG fetched = 0;
867 MediaCatMoniker * pMoniker;
868 WCHAR *name;
869 CLSID clsid;
870 HRESULT hr;
871 HKEY hkey;
873 TRACE("(%p)->(%d, %p, %p)\n", iface, celt, rgelt, pceltFetched);
875 while (fetched < celt)
877 /* FIXME: try PNP devices first */
879 /* try DMOs */
880 if ((hr = IEnumDMO_Next(This->dmo_enum, 1, &clsid, &name, NULL)) == S_OK)
882 if (!(pMoniker = DEVENUM_IMediaCatMoniker_Construct()))
883 return E_OUTOFMEMORY;
885 pMoniker->type = DEVICE_DMO;
886 pMoniker->clsid = clsid;
888 StringFromGUID2(&clsid, buffer, CHARS_IN_GUID);
889 StringFromGUID2(&This->class, buffer + CHARS_IN_GUID - 1, CHARS_IN_GUID);
891 /* try DirectShow filters */
892 else if (!(res = RegEnumKeyW(This->sw_key, This->sw_index, buffer, ARRAY_SIZE(buffer))))
894 This->sw_index++;
895 if ((res = RegOpenKeyExW(This->sw_key, buffer, 0, KEY_QUERY_VALUE, &hkey)))
896 break;
898 if (!(pMoniker = DEVENUM_IMediaCatMoniker_Construct()))
899 return E_OUTOFMEMORY;
901 pMoniker->type = DEVICE_FILTER;
903 if (!(pMoniker->name = CoTaskMemAlloc((strlenW(buffer) + 1) * sizeof(WCHAR))))
905 IMoniker_Release(&pMoniker->IMoniker_iface);
906 return E_OUTOFMEMORY;
908 strcpyW(pMoniker->name, buffer);
910 /* then try codecs */
911 else if (!(res = RegEnumKeyW(This->cm_key, This->cm_index, buffer, ARRAY_SIZE(buffer))))
913 This->cm_index++;
915 if ((res = RegOpenKeyExW(This->cm_key, buffer, 0, KEY_QUERY_VALUE, &hkey)))
916 break;
918 if (!(pMoniker = DEVENUM_IMediaCatMoniker_Construct()))
919 return E_OUTOFMEMORY;
921 pMoniker->type = DEVICE_CODEC;
923 if (!(pMoniker->name = CoTaskMemAlloc((strlenW(buffer) + 1) * sizeof(WCHAR))))
925 IMoniker_Release(&pMoniker->IMoniker_iface);
926 return E_OUTOFMEMORY;
928 strcpyW(pMoniker->name, buffer);
930 else
931 break;
933 pMoniker->has_class = TRUE;
934 pMoniker->class = This->class;
936 rgelt[fetched] = &pMoniker->IMoniker_iface;
937 fetched++;
940 TRACE("-- fetched %d\n", fetched);
942 if (pceltFetched)
943 *pceltFetched = fetched;
945 if (fetched != celt)
946 return S_FALSE;
947 else
948 return S_OK;
951 static HRESULT WINAPI DEVENUM_IEnumMoniker_Skip(IEnumMoniker *iface, ULONG celt)
953 EnumMonikerImpl *This = impl_from_IEnumMoniker(iface);
955 TRACE("(%p)->(%d)\n", iface, celt);
957 while (celt--)
959 /* FIXME: try PNP devices first */
961 /* try DMOs */
962 if (IEnumDMO_Skip(This->dmo_enum, 1) == S_OK)
964 /* try DirectShow filters */
965 else if (RegEnumKeyW(This->sw_key, This->sw_index, NULL, 0) != ERROR_NO_MORE_ITEMS)
967 This->sw_index++;
969 /* then try codecs */
970 else if (RegEnumKeyW(This->cm_key, This->cm_index, NULL, 0) != ERROR_NO_MORE_ITEMS)
972 This->cm_index++;
974 else
975 return S_FALSE;
978 return S_OK;
981 static HRESULT WINAPI DEVENUM_IEnumMoniker_Reset(IEnumMoniker *iface)
983 EnumMonikerImpl *This = impl_from_IEnumMoniker(iface);
985 TRACE("(%p)->()\n", iface);
987 IEnumDMO_Reset(This->dmo_enum);
988 This->sw_index = 0;
989 This->cm_index = 0;
991 return S_OK;
994 static HRESULT WINAPI DEVENUM_IEnumMoniker_Clone(IEnumMoniker *iface, IEnumMoniker **ppenum)
996 FIXME("(%p)->(%p): stub\n", iface, ppenum);
998 return E_NOTIMPL;
1001 /**********************************************************************
1002 * IEnumMoniker_Vtbl
1004 static const IEnumMonikerVtbl IEnumMoniker_Vtbl =
1006 DEVENUM_IEnumMoniker_QueryInterface,
1007 DEVENUM_IEnumMoniker_AddRef,
1008 DEVENUM_IEnumMoniker_Release,
1009 DEVENUM_IEnumMoniker_Next,
1010 DEVENUM_IEnumMoniker_Skip,
1011 DEVENUM_IEnumMoniker_Reset,
1012 DEVENUM_IEnumMoniker_Clone
1015 HRESULT create_EnumMoniker(REFCLSID class, IEnumMoniker **ppEnumMoniker)
1017 EnumMonikerImpl * pEnumMoniker = CoTaskMemAlloc(sizeof(EnumMonikerImpl));
1018 WCHAR buffer[78];
1019 HRESULT hr;
1021 if (!pEnumMoniker)
1022 return E_OUTOFMEMORY;
1024 pEnumMoniker->IEnumMoniker_iface.lpVtbl = &IEnumMoniker_Vtbl;
1025 pEnumMoniker->ref = 1;
1026 pEnumMoniker->sw_index = 0;
1027 pEnumMoniker->cm_index = 0;
1028 pEnumMoniker->class = *class;
1030 strcpyW(buffer, clsidW);
1031 strcatW(buffer, backslashW);
1032 StringFromGUID2(class, buffer + strlenW(buffer), CHARS_IN_GUID);
1033 strcatW(buffer, instanceW);
1034 if (RegOpenKeyExW(HKEY_CLASSES_ROOT, buffer, 0, KEY_ENUMERATE_SUB_KEYS, &pEnumMoniker->sw_key))
1035 pEnumMoniker->sw_key = NULL;
1037 strcpyW(buffer, wszActiveMovieKey);
1038 StringFromGUID2(class, buffer + strlenW(buffer), CHARS_IN_GUID);
1039 if (RegOpenKeyExW(HKEY_CURRENT_USER, buffer, 0, KEY_ENUMERATE_SUB_KEYS, &pEnumMoniker->cm_key))
1040 pEnumMoniker->cm_key = NULL;
1042 hr = DMOEnum(class, 0, 0, NULL, 0, NULL, &pEnumMoniker->dmo_enum);
1043 if (FAILED(hr))
1045 IEnumMoniker_Release(&pEnumMoniker->IEnumMoniker_iface);
1046 return hr;
1049 *ppEnumMoniker = &pEnumMoniker->IEnumMoniker_iface;
1051 DEVENUM_LockModule();
1053 return S_OK;