advapi32: Make rpcrt4 a delayed import to work around circular dependencies with...
[wine/testsucceed.git] / dlls / shdocvw / factory.c
blob751b5a48fd693cad4360534ee08c5b35d3dfa410
1 /*
2 * Implementation of class factory for IE Web Browser
4 * Copyright 2001 John R. Sheets (for CodeWeavers)
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 #include <string.h>
22 #include <stdio.h>
24 #include "shdocvw.h"
25 #include "winreg.h"
26 #include "advpub.h"
27 #include "isguids.h"
29 #include "winver.h"
31 #include "wine/debug.h"
33 WINE_DEFAULT_DEBUG_CHANNEL(shdocvw);
35 /**********************************************************************
36 * Implement the WebBrowser class factory
38 * (Based on implementation in ddraw/main.c)
41 #define FACTORY(x) ((IClassFactory*) &(x)->lpClassFactoryVtbl)
43 typedef struct
45 /* IUnknown fields */
46 const IClassFactoryVtbl *lpClassFactoryVtbl;
47 HRESULT (*cf)(LPUNKNOWN, REFIID, LPVOID *);
48 LONG ref;
49 } IClassFactoryImpl;
52 /**********************************************************************
53 * WBCF_QueryInterface (IUnknown)
55 static HRESULT WINAPI WBCF_QueryInterface(LPCLASSFACTORY iface,
56 REFIID riid, LPVOID *ppobj)
58 TRACE("(%s %p)\n", debugstr_guid(riid), ppobj);
60 if (!ppobj)
61 return E_POINTER;
63 if(IsEqualGUID(&IID_IUnknown, riid) || IsEqualGUID(&IID_IClassFactory, riid)) {
64 *ppobj = iface;
65 IClassFactory_AddRef(iface);
66 return S_OK;
69 WARN("Not supported interface %s\n", debugstr_guid(riid));
71 *ppobj = NULL;
72 return E_NOINTERFACE;
75 /************************************************************************
76 * WBCF_AddRef (IUnknown)
78 static ULONG WINAPI WBCF_AddRef(LPCLASSFACTORY iface)
80 SHDOCVW_LockModule();
82 return 2; /* non-heap based object */
85 /************************************************************************
86 * WBCF_Release (IUnknown)
88 static ULONG WINAPI WBCF_Release(LPCLASSFACTORY iface)
90 SHDOCVW_UnlockModule();
92 return 1; /* non-heap based object */
95 /************************************************************************
96 * WBCF_CreateInstance (IClassFactory)
98 static HRESULT WINAPI WBCF_CreateInstance(LPCLASSFACTORY iface, LPUNKNOWN pOuter,
99 REFIID riid, LPVOID *ppobj)
101 IClassFactoryImpl *This = (IClassFactoryImpl *) iface;
102 return This->cf(pOuter, riid, ppobj);
105 /************************************************************************
106 * WBCF_LockServer (IClassFactory)
108 static HRESULT WINAPI WBCF_LockServer(LPCLASSFACTORY iface, BOOL dolock)
110 TRACE("(%d)\n", dolock);
112 if (dolock)
113 SHDOCVW_LockModule();
114 else
115 SHDOCVW_UnlockModule();
117 return S_OK;
120 static const IClassFactoryVtbl WBCF_Vtbl =
122 WBCF_QueryInterface,
123 WBCF_AddRef,
124 WBCF_Release,
125 WBCF_CreateInstance,
126 WBCF_LockServer
129 /*************************************************************************
130 * DllGetClassObject (SHDOCVW.@)
132 HRESULT WINAPI DllGetClassObject(REFCLSID rclsid, REFIID riid, void **ppv)
134 static IClassFactoryImpl WB1ClassFactory = {&WBCF_Vtbl, WebBrowserV1_Create};
135 static IClassFactoryImpl WB2ClassFactory = {&WBCF_Vtbl, WebBrowserV2_Create};
136 static IClassFactoryImpl CUHClassFactory = {&WBCF_Vtbl, CUrlHistory_Create};
137 static IClassFactoryImpl ISCClassFactory = {&WBCF_Vtbl, InternetShortcut_Create};
138 static IClassFactoryImpl TBLClassFactory = {&WBCF_Vtbl, TaskbarList_Create};
140 TRACE("\n");
142 if(IsEqualGUID(&CLSID_WebBrowser, rclsid))
143 return IClassFactory_QueryInterface(FACTORY(&WB2ClassFactory), riid, ppv);
145 if(IsEqualGUID(&CLSID_WebBrowser_V1, rclsid))
146 return IClassFactory_QueryInterface(FACTORY(&WB1ClassFactory), riid, ppv);
148 if(IsEqualGUID(&CLSID_CUrlHistory, rclsid))
149 return IClassFactory_QueryInterface(FACTORY(&CUHClassFactory), riid, ppv);
151 if(IsEqualGUID(&CLSID_InternetShortcut, rclsid))
152 return IClassFactory_QueryInterface(FACTORY(&ISCClassFactory), riid, ppv);
154 if(IsEqualGUID(&CLSID_TaskbarList, rclsid))
155 return IClassFactory_QueryInterface(FACTORY(&TBLClassFactory), riid, ppv);
157 /* As a last resort, figure if the CLSID belongs to a 'Shell Instance Object' */
158 return SHDOCVW_GetShellInstanceObjectClassObject(rclsid, riid, ppv);
161 HRESULT register_class_object(BOOL do_reg)
163 HRESULT hres;
165 static DWORD cookie;
166 static IClassFactoryImpl IEClassFactory = {&WBCF_Vtbl, InternetExplorer_Create};
168 if(do_reg) {
169 hres = CoRegisterClassObject(&CLSID_InternetExplorer, (IUnknown*)FACTORY(&IEClassFactory),
170 CLSCTX_SERVER, REGCLS_MULTIPLEUSE|REGCLS_SUSPENDED, &cookie);
171 if (FAILED(hres)) {
172 ERR("failed to register object %08x\n", hres);
173 return hres;
176 hres = CoResumeClassObjects();
177 if(SUCCEEDED(hres))
178 return hres;
180 ERR("failed to resume object %08x\n", hres);
183 return CoRevokeClassObject(cookie);
186 static HRESULT reg_install(LPCSTR section, STRTABLEA *strtable)
188 HRESULT (WINAPI *pRegInstall)(HMODULE hm, LPCSTR pszSection, const STRTABLEA* pstTable);
189 HMODULE hadvpack;
190 HRESULT hres;
192 static const WCHAR advpackW[] = {'a','d','v','p','a','c','k','.','d','l','l',0};
194 hadvpack = LoadLibraryW(advpackW);
195 pRegInstall = (void *)GetProcAddress(hadvpack, "RegInstall");
197 hres = pRegInstall(shdocvw_hinstance, section, strtable);
199 FreeLibrary(hadvpack);
200 return hres;
203 static const GUID CLSID_MicrosoftBrowserArchitecture =
204 {0xa5e46e3a, 0x8849, 0x11d1, {0x9d, 0x8c, 0x00, 0xc0, 0x4f, 0xc9, 0x9d, 0x61}};
205 static const GUID CLSID_MruLongList =
206 {0x53bd6b4e, 0x3780, 0x4693, {0xaf, 0xc3, 0x71, 0x61, 0xc2, 0xf3, 0xee, 0x9c}};
208 #define INF_SET_CLSID(clsid) \
209 do \
211 static CHAR name[] = "CLSID_" #clsid; \
213 pse[i].pszName = name; \
214 clsids[i++] = &CLSID_ ## clsid; \
215 } while (0)
217 static HRESULT register_server(BOOL doregister)
219 STRTABLEA strtable;
220 STRENTRYA pse[15];
221 static CLSID const *clsids[15];
222 unsigned int i = 0;
223 HRESULT hres;
225 INF_SET_CLSID(CUrlHistory);
226 INF_SET_CLSID(Internet);
227 INF_SET_CLSID(InternetExplorer);
228 INF_SET_CLSID(InternetShortcut);
229 INF_SET_CLSID(MicrosoftBrowserArchitecture);
230 INF_SET_CLSID(MruLongList);
231 INF_SET_CLSID(SearchAssistantOC);
232 INF_SET_CLSID(ShellNameSpace);
233 INF_SET_CLSID(ShellSearchAssistantOC);
234 INF_SET_CLSID(ShellShellNameSpace);
235 INF_SET_CLSID(ShellUIHelper);
236 INF_SET_CLSID(ShellWindows);
237 INF_SET_CLSID(TaskbarList);
238 INF_SET_CLSID(WebBrowser);
239 INF_SET_CLSID(WebBrowser_V1);
241 for(i = 0; i < sizeof(pse)/sizeof(pse[0]); i++) {
242 pse[i].pszValue = HeapAlloc(GetProcessHeap(), 0, 39);
243 sprintf(pse[i].pszValue, "{%08X-%04X-%04X-%02X%02X-%02X%02X%02X%02X%02X%02X}",
244 clsids[i]->Data1, clsids[i]->Data2, clsids[i]->Data3, clsids[i]->Data4[0],
245 clsids[i]->Data4[1], clsids[i]->Data4[2], clsids[i]->Data4[3], clsids[i]->Data4[4],
246 clsids[i]->Data4[5], clsids[i]->Data4[6], clsids[i]->Data4[7]);
249 strtable.cEntries = sizeof(pse)/sizeof(pse[0]);
250 strtable.pse = pse;
252 hres = reg_install(doregister ? "RegisterDll" : "UnregisterDll", &strtable);
254 for(i=0; i < sizeof(pse)/sizeof(pse[0]); i++)
255 HeapFree(GetProcessHeap(), 0, pse[i].pszValue);
257 return hres;
260 #undef INF_SET_CLSID
262 /***********************************************************************
263 * DllRegisterServer (shdocvw.@)
265 HRESULT WINAPI DllRegisterServer(void)
267 ITypeLib *typelib;
268 HRESULT hres;
270 static const WCHAR shdocvwW[] = {'s','h','d','o','c','v','w','.','d','l','l',0};
272 hres = register_server(TRUE);
273 if(FAILED(hres))
274 return hres;
276 hres = LoadTypeLibEx(shdocvwW, REGKIND_REGISTER, &typelib);
277 if(FAILED(hres)) {
278 ERR("Could not load typelib: %08x\n", hres);
279 return hres;
282 ITypeLib_Release(typelib);
284 return hres;
287 /***********************************************************************
288 * DllUnregisterServer (shdocvw.@)
290 HRESULT WINAPI DllUnregisterServer(void)
292 HRESULT hres;
294 hres = register_server(FALSE);
295 if(FAILED(hres))
296 return hres;
298 return UnRegisterTypeLib(&LIBID_SHDocVw, 1, 1, LOCALE_SYSTEM_DEFAULT, SYS_WIN32);
301 static BOOL check_native_ie(void)
303 static const WCHAR cszPath[] = {'b','r','o','w','s','e','u','i','.','d','l','l',0};
304 DWORD handle,size;
305 BOOL ret = TRUE;
307 size = GetFileVersionInfoSizeW(cszPath,&handle);
308 if (size)
310 LPVOID buf;
311 LPWSTR lpFileDescription;
312 UINT dwBytes;
313 static const WCHAR cszFD[] = {'\\','S','t','r','i','n','g','F','i','l','e','I','n','f','o','\\','0','4','0','9','0','4','e','4','\\','F','i','l','e','D','e','s','c','r','i','p','t','i','o','n',0};
314 static const WCHAR cszWine[] = {'W','i','n','e',0};
316 buf = HeapAlloc(GetProcessHeap(),0,size);
317 GetFileVersionInfoW(cszPath,0,size,buf);
319 if (VerQueryValueW(buf, cszFD, (LPVOID*)&lpFileDescription, &dwBytes) &&
320 strstrW(lpFileDescription,cszWine))
321 ret = FALSE;
323 HeapFree(GetProcessHeap(), 0, buf);
326 return ret;
329 DWORD register_iexplore(BOOL doregister)
331 HRESULT hres;
332 if (check_native_ie())
334 TRACE("Native IE detected, not doing registration\n");
335 return S_OK;
337 hres = reg_install(doregister ? "RegisterIE" : "UnregisterIE", NULL);
338 return FAILED(hres);