2 * Shell Instance Objects - Add hot water and stir until dissolved.
4 * Copyright 2005 Michael Jung
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
21 /* 'Shell Instance Objects' allow you to add a node to the shell namespace
22 * (typically a shortcut to some location in the filesystem), just by setting
23 * some registry entries. This feature was introduced with win2k. Please
24 * search for 'Shell Instance Objects' on MSDN to get more information. */
29 #define COM_NO_WINDOWS_H
39 #include "wine/unicode.h"
40 #include "wine/debug.h"
42 WINE_DEFAULT_DEBUG_CHANNEL(shdocvw
);
44 #define ADJUST_THIS(c,m,p) ((c*)(((long)p)-(long)&(((c*)0)->lp##m##Vtbl)))
45 #define STATIC_CAST(i,p) ((i*)&p->lp##i##Vtbl)
46 #define CHARS_IN_GUID 39
48 /******************************************************************************
51 * Gives access to a registry key's values via the IPropertyBag interface.
53 typedef struct _RegistryPropertyBag
{
54 const IPropertyBagVtbl
*lpIPropertyBagVtbl
;
56 HKEY m_hInitPropertyBagKey
;
57 } RegistryPropertyBag
;
59 static void RegistryPropertyBag_Destroy(RegistryPropertyBag
*This
) {
60 TRACE("This=%p)\n", This
);
62 RegCloseKey(This
->m_hInitPropertyBagKey
);
66 static HRESULT WINAPI
RegistryPropertyBag_IPropertyBag_QueryInterface(IPropertyBag
*iface
,
67 REFIID riid
, void **ppv
)
69 RegistryPropertyBag
*This
= ADJUST_THIS(RegistryPropertyBag
, IPropertyBag
, iface
);
71 TRACE("(iface=%p, riid=%s, ppv=%p)\n", iface
, debugstr_guid(riid
), ppv
);
76 if (IsEqualIID(&IID_IUnknown
, riid
) || IsEqualIID(&IID_IPropertyBag
, riid
)) {
77 *ppv
= STATIC_CAST(IPropertyBag
, This
);
83 IUnknown_AddRef((IUnknown
*)*ppv
);
87 static ULONG WINAPI
RegistryPropertyBag_IPropertyBag_AddRef(IPropertyBag
*iface
) {
88 RegistryPropertyBag
*This
= ADJUST_THIS(RegistryPropertyBag
, IPropertyBag
, iface
);
91 TRACE("(iface=%p)\n", iface
);
93 cRef
= InterlockedIncrement(&This
->m_cRef
);
101 static ULONG WINAPI
RegistryPropertyBag_IPropertyBag_Release(IPropertyBag
*iface
) {
102 RegistryPropertyBag
*This
= ADJUST_THIS(RegistryPropertyBag
, IPropertyBag
, iface
);
105 TRACE("(iface=%p)\n", iface
);
107 cRef
= InterlockedDecrement(&This
->m_cRef
);
110 RegistryPropertyBag_Destroy(This
);
111 SHDOCVW_UnlockModule();
117 static HRESULT WINAPI
RegistryPropertyBag_IPropertyBag_Read(IPropertyBag
*iface
,
118 LPCOLESTR pwszPropName
, VARIANT
*pVar
, IErrorLog
*pErrorLog
)
120 RegistryPropertyBag
*This
= ADJUST_THIS(RegistryPropertyBag
, IPropertyBag
, iface
);
122 DWORD dwType
, cbData
;
124 VARTYPE vtDst
= V_VT(pVar
);
127 TRACE("(iface=%p, pwszPropName=%s, pVar=%p, pErrorLog=%p)\n", iface
, debugstr_w(pwszPropName
),
130 res
= RegQueryValueExW(This
->m_hInitPropertyBagKey
, pwszPropName
, NULL
, &dwType
, NULL
, &cbData
);
131 if (res
!= ERROR_SUCCESS
)
134 pwszValue
= shdocvw_alloc(cbData
);
136 return E_OUTOFMEMORY
;
138 res
= RegQueryValueExW(This
->m_hInitPropertyBagKey
, pwszPropName
, NULL
, &dwType
,
139 (LPBYTE
)pwszValue
, &cbData
);
140 if (res
!= ERROR_SUCCESS
) {
141 shdocvw_free(pwszValue
);
145 V_VT(pVar
) = VT_BSTR
;
146 V_BSTR(pVar
) = SysAllocString(pwszValue
);
147 shdocvw_free(pwszValue
);
149 if (vtDst
!= VT_BSTR
) {
150 hr
= VariantChangeTypeEx(pVar
, pVar
, LOCALE_SYSTEM_DEFAULT
, 0, vtDst
);
152 SysFreeString(V_BSTR(pVar
));
158 static HRESULT WINAPI
RegistryPropertyBag_IPropertyBag_Write(IPropertyBag
*iface
,
159 LPCOLESTR pwszPropName
, VARIANT
*pVar
)
161 FIXME("(iface=%p, pwszPropName=%s, pVar=%p) stub\n", iface
, debugstr_w(pwszPropName
), pVar
);
165 static const IPropertyBagVtbl RegistryPropertyBag_IPropertyBagVtbl
= {
166 RegistryPropertyBag_IPropertyBag_QueryInterface
,
167 RegistryPropertyBag_IPropertyBag_AddRef
,
168 RegistryPropertyBag_IPropertyBag_Release
,
169 RegistryPropertyBag_IPropertyBag_Read
,
170 RegistryPropertyBag_IPropertyBag_Write
173 HRESULT
RegistryPropertyBag_Constructor(HKEY hInitPropertyBagKey
, REFIID riid
, LPVOID
*ppvObject
) {
175 RegistryPropertyBag
*pRegistryPropertyBag
;
177 TRACE("(hInitPropertyBagKey=%p, riid=%s, ppvObject=%p)\n", hInitPropertyBagKey
,
178 debugstr_guid(riid
), ppvObject
);
180 pRegistryPropertyBag
= shdocvw_alloc(sizeof(RegistryPropertyBag
));
181 if (pRegistryPropertyBag
) {
182 pRegistryPropertyBag
->lpIPropertyBagVtbl
= &RegistryPropertyBag_IPropertyBagVtbl
;
183 pRegistryPropertyBag
->m_cRef
= 0;
184 pRegistryPropertyBag
->m_hInitPropertyBagKey
= hInitPropertyBagKey
;
186 /* The clasping AddRef/Release is for the case that QueryInterface fails, which will result
187 * in a reference count of 0 in the Release call, which will result in object destruction.*/
188 IPropertyBag_AddRef(STATIC_CAST(IPropertyBag
, pRegistryPropertyBag
));
189 hr
= IPropertyBag_QueryInterface(STATIC_CAST(IPropertyBag
, pRegistryPropertyBag
), riid
, ppvObject
);
190 IPropertyBag_Release(STATIC_CAST(IPropertyBag
, pRegistryPropertyBag
));
196 /******************************************************************************
197 * InstanceObjectFactory
198 * Builds Instance Objects and asks them to initialize themselves based on the
199 * values of a PropertyBag.
201 typedef struct _InstanceObjectFactory
{
202 const IClassFactoryVtbl
*lpIClassFactoryVtbl
;
204 CLSID m_clsidInstance
; /* CLSID of the objects to create. */
205 IPropertyBag
*m_pPropertyBag
; /* PropertyBag to initialize those objects. */
206 } InstanceObjectFactory
;
208 static void InstanceObjectFactory_Destroy(InstanceObjectFactory
*This
) {
209 IPropertyBag_Release(This
->m_pPropertyBag
);
213 static HRESULT WINAPI
InstanceObjectFactory_IClassFactory_QueryInterface(IClassFactory
*iface
,
214 REFIID riid
, LPVOID
* ppv
)
216 InstanceObjectFactory
*This
= ADJUST_THIS(InstanceObjectFactory
, IClassFactory
, iface
);
218 TRACE("iface=%p, riid=%s, ppv=%p)\n", iface
, debugstr_guid(riid
), ppv
);
223 if (IsEqualIID(&IID_IUnknown
, riid
) || IsEqualIID(&IID_IClassFactory
, riid
)) {
224 *ppv
= STATIC_CAST(IClassFactory
, This
);
227 return E_NOINTERFACE
;
230 IUnknown_AddRef((IUnknown
*)*ppv
);
234 static ULONG WINAPI
InstanceObjectFactory_IClassFactory_AddRef(IClassFactory
*iface
)
236 InstanceObjectFactory
*This
= ADJUST_THIS(InstanceObjectFactory
, IClassFactory
, iface
);
239 TRACE("(iface=%p)\n", iface
);
241 cRef
= InterlockedIncrement(&This
->m_cRef
);
244 IClassFactory_LockServer(iface
, TRUE
);
249 static ULONG WINAPI
InstanceObjectFactory_IClassFactory_Release(IClassFactory
*iface
)
251 InstanceObjectFactory
*This
= ADJUST_THIS(InstanceObjectFactory
, IClassFactory
, iface
);
254 TRACE("(iface=%p)\n", iface
);
256 cRef
= InterlockedDecrement(&This
->m_cRef
);
259 IClassFactory_LockServer(iface
, FALSE
);
260 InstanceObjectFactory_Destroy(This
);
266 static HRESULT WINAPI
InstanceObjectFactory_IClassFactory_CreateInstance(IClassFactory
*iface
,
267 IUnknown
*pUnkOuter
, REFIID riid
, LPVOID
*ppvObj
)
269 InstanceObjectFactory
*This
= ADJUST_THIS(InstanceObjectFactory
, IClassFactory
, iface
);
270 IPersistPropertyBag
*pPersistPropertyBag
;
273 TRACE("(pUnkOuter=%p, riid=%s, ppvObj=%p)\n", pUnkOuter
, debugstr_guid(riid
), ppvObj
);
275 hr
= CoCreateInstance(&This
->m_clsidInstance
, NULL
, CLSCTX_INPROC_SERVER
,
276 &IID_IPersistPropertyBag
, (LPVOID
*)&pPersistPropertyBag
);
278 TRACE("Failed to create instance of %s. hr = %08lx\n",
279 debugstr_guid(&This
->m_clsidInstance
), hr
);
283 hr
= IPersistPropertyBag_Load(pPersistPropertyBag
, This
->m_pPropertyBag
, NULL
);
285 TRACE("Failed to initialize object from ProperyBag: hr = %08lx\n", hr
);
286 IPersistPropertyBag_Release(pPersistPropertyBag
);
290 hr
= IPersistPropertyBag_QueryInterface(pPersistPropertyBag
, riid
, ppvObj
);
291 IPersistPropertyBag_Release(pPersistPropertyBag
);
296 static HRESULT WINAPI
InstanceObjectFactory_IClassFactory_LockServer(IClassFactory
*iface
,
299 TRACE("(iface=%p, fLock=%d) stub\n", iface
, fLock
);
302 SHDOCVW_LockModule();
304 SHDOCVW_UnlockModule();
309 static const IClassFactoryVtbl InstanceObjectFactory_IClassFactoryVtbl
= {
310 InstanceObjectFactory_IClassFactory_QueryInterface
,
311 InstanceObjectFactory_IClassFactory_AddRef
,
312 InstanceObjectFactory_IClassFactory_Release
,
313 InstanceObjectFactory_IClassFactory_CreateInstance
,
314 InstanceObjectFactory_IClassFactory_LockServer
317 HRESULT
InstanceObjectFactory_Constructor(REFCLSID rclsid
, IPropertyBag
*pPropertyBag
, REFIID riid
,
320 InstanceObjectFactory
*pInstanceObjectFactory
;
323 TRACE("(RegistryPropertyBag=%p, riid=%s, ppvObject=%p)\n", pPropertyBag
,
324 debugstr_guid(riid
), ppvObject
);
326 pInstanceObjectFactory
= shdocvw_alloc(sizeof(InstanceObjectFactory
));
327 if (pInstanceObjectFactory
) {
328 pInstanceObjectFactory
->lpIClassFactoryVtbl
= &InstanceObjectFactory_IClassFactoryVtbl
;
329 pInstanceObjectFactory
->m_cRef
= 0;
330 memcpy(&pInstanceObjectFactory
->m_clsidInstance
, rclsid
, sizeof(CLSID
));
331 pInstanceObjectFactory
->m_pPropertyBag
= pPropertyBag
;
332 IPropertyBag_AddRef(pPropertyBag
);
334 IClassFactory_AddRef(STATIC_CAST(IClassFactory
, pInstanceObjectFactory
));
335 hr
= IClassFactory_QueryInterface(STATIC_CAST(IClassFactory
, pInstanceObjectFactory
),
337 IClassFactory_Release(STATIC_CAST(IClassFactory
, pInstanceObjectFactory
));
343 /******************************************************************************
344 * SHDOCVW_GetShellInstanceObjectClassObject [Internal]
346 * Figure if there is a 'Shell Instance Object' conformant registry entry for
347 * the given CLSID and if so create and return a corresponding ClassObject.
350 * rclsid [I] CLSID of the 'Shell Instance Object'.
351 * riid [I] Desired interface. Only IClassFactory supported.
352 * ppvClassObj [O] The corresponding ClassObject.
356 * Failure: CLASS_E_CLASSNOTAVAILABLE
358 HRESULT
SHDOCVW_GetShellInstanceObjectClassObject(REFCLSID rclsid
, REFIID riid
,
361 WCHAR wszInstanceKey
[] = { 'C','L','S','I','D','\\','{','0','0','0','0','0','0','0','0','-',
362 '0','0','0','0','-','0','0','0','0','-','0','0','0','0','-','0','0','0','0','0','0','0','0',
363 '0','0','0','0','}','\\','I','n','s','t','a','n','c','e', 0 };
364 const WCHAR wszCLSID
[] = { 'C','L','S','I','D',0 };
365 const WCHAR wszInitPropertyBag
[] =
366 { 'I','n','i','t','P','r','o','p','e','r','t','y','B','a','g',0 };
367 WCHAR wszCLSIDInstance
[CHARS_IN_GUID
];
369 HKEY hInstanceKey
, hInitPropertyBagKey
;
370 DWORD dwType
, cbBytes
= sizeof(wszCLSIDInstance
);
371 IPropertyBag
*pInitPropertyBag
;
375 TRACE("(rclsid=%s, riid=%s, ppvClassObject=%p)\n", debugstr_guid(rclsid
), debugstr_guid(riid
),
378 /* Figure if there is an 'Instance' subkey for the given CLSID and aquire a handle. */
379 if (!StringFromGUID2(rclsid
, wszInstanceKey
+ 6, CHARS_IN_GUID
) ||
380 !(wszInstanceKey
[5+CHARS_IN_GUID
]='\\') || /* Repair the null-termination. */
381 ERROR_SUCCESS
!= RegOpenKeyExW(HKEY_CLASSES_ROOT
, wszInstanceKey
, 0, KEY_READ
, &hInstanceKey
))
383 /* If there is no 'Instance' subkey, then it's not a Shell Instance Object. */
384 return CLASS_E_CLASSNOTAVAILABLE
;
387 if (RegQueryValueExW(hInstanceKey
, wszCLSID
, NULL
, &dwType
, (LPBYTE
)wszCLSIDInstance
, &cbBytes
)
388 != ERROR_SUCCESS
|| FAILED(CLSIDFromString(wszCLSIDInstance
, &clsidInstance
)))
390 /* 'Instance' should have a 'CLSID' value with a well-formed clsid-string. */
391 FIXME("Failed to infer instance CLSID! %s\n", debugstr_w(wszCLSIDInstance
));
392 RegCloseKey(hInstanceKey
);
393 return CLASS_E_CLASSNOTAVAILABLE
;
396 /* Try to open the 'InitPropertyBag' subkey. */
397 res
= RegOpenKeyExW(hInstanceKey
, wszInitPropertyBag
, 0, KEY_READ
, &hInitPropertyBagKey
);
398 RegCloseKey(hInstanceKey
);
399 if (res
!= ERROR_SUCCESS
) {
400 /* Besides 'InitPropertyBag's, shell instance objects might be initialized by streams.
401 * So this case might not be an error. */
402 TRACE("No InitPropertyBag key found!\n");
403 return CLASS_E_CLASSNOTAVAILABLE
;
406 /* If the construction succeeds, the new RegistryPropertyBag is responsible for closing
407 * hInitProperyBagKey. */
408 hr
= RegistryPropertyBag_Constructor(hInitPropertyBagKey
, &IID_IPropertyBag
,
409 (LPVOID
*)&pInitPropertyBag
);
411 RegCloseKey(hInitPropertyBagKey
);
415 /* Construct an Instance Object Factory, which creates objects of class 'clsidInstance'
416 * and asks them to initialize themselves with the help of the 'pInitiPropertyBag' */
417 hr
= InstanceObjectFactory_Constructor(&clsidInstance
, pInitPropertyBag
, riid
, ppvClassObj
);
418 IPropertyBag_Release(pInitPropertyBag
); /* The factory will hold a reference the bag. */