1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
5 #include "chrome_frame/bho_loader.h"
11 #include "chrome_frame/chrome_frame_helper_util.h"
12 #include "chrome_frame/event_hooker.h"
13 #include "chrome_tab.h" // NOLINT
16 // Describes the window class we look for.
17 const wchar_t kStatusBarWindowClass
[] = L
"msctls_statusbar32";
19 // On IE9, the status bar is disabled by default, so we look for an
20 // AsyncBoundaryLayer window instead.
21 const wchar_t kAsyncBoundaryDnWindow
[] = L
"asynclayerboundarydn\0";
23 BHOLoader::BHOLoader() : hooker_(new EventHooker()) {
26 BHOLoader::~BHOLoader() {
33 void BHOLoader::OnHookEvent(DWORD event
, HWND window
) {
34 // Step 1: Make sure that we are in a process named iexplore.exe.
35 if (IsNamedProcess(L
"iexplore.exe")) {
36 if (!IsWindowOfClass(window
, kStatusBarWindowClass
) &&
37 !IsWindowOfClass(window
, kAsyncBoundaryDnWindow
)) {
40 // We have the right sort of window, check to make sure it was created
41 // on the current thread.
42 DWORD thread_id
= GetWindowThreadProcessId(window
, NULL
);
43 _ASSERTE(thread_id
== GetCurrentThreadId());
46 // Step 2: Check to see if the window is of the right class.
47 HWND browser_hwnd
= NULL
;
48 if (IsWindowOfClass(window
, kStatusBarWindowClass
)) {
49 // For IE8 and under, IE loads BHOs in the WM_CREATE handler of the tab
50 // window approximately after it creates the status bar window. To be as
51 // close to IE as possible in our simulation on BHO loading, we watch for
52 // the status bar to be created and do our simulated BHO loading at that
54 browser_hwnd
= GetParent(window
);
55 } else if (IsWindowOfClass(window
, kAsyncBoundaryDnWindow
)) {
56 // For IE9, the status bar is disabled by default, so we look for an
57 // AsyncBoundaryWindow to be created. When we find that, look for a
58 // child window owned by the current thread named "tabwindowclass".
59 // That will be our browser window.
60 browser_hwnd
= RecurseFindWindow(NULL
, L
"tabwindowclass", NULL
,
62 GetCurrentProcessId());
63 _ASSERTE(NULL
!= browser_hwnd
);
66 if (browser_hwnd
!= NULL
) {
68 // Parent window of status bar window is the web browser window. Try to
69 // get its IWebBrowser2 interface
70 CComPtr
<IWebBrowser2
> browser
;
71 UtilGetWebBrowserObjectFromWindow(browser_hwnd
, __uuidof(browser
),
72 reinterpret_cast<void**>(&browser
));
74 if (IsSystemLevelChromeFrameInstalled()) {
75 // We're in the right place, but a system-level installation has
76 // appeared. We should leave now.
80 // Figure out if we're already in the property map.
81 wchar_t bho_clsid_as_string
[MAX_PATH
] = {0};
82 StringFromGUID2(CLSID_ChromeFrameBHO
, bho_clsid_as_string
,
83 ARRAYSIZE(bho_clsid_as_string
));
84 CComBSTR
bho_clsid_as_string_bstr(bho_clsid_as_string
);
86 CComVariant existing_bho
;
87 HRESULT hr
= browser
->GetProperty(bho_clsid_as_string_bstr
,
90 if (V_VT(&existing_bho
) != VT_DISPATCH
&&
91 V_VT(&existing_bho
) != VT_UNKNOWN
) {
93 // We have the IWebBrowser2 interface. Now create the BHO instance
94 CComPtr
<IObjectWithSite
> bho_object
;
95 hr
= bho_object
.CoCreateInstance(CLSID_ChromeFrameBHO
,
97 CLSCTX_INPROC_SERVER
);
100 if (SUCCEEDED(hr
) && bho_object
) {
102 // Initialize the BHO by calling SetSite and passing it IWebBrowser2
103 hr
= bho_object
->SetSite(browser
);
104 _ASSERTE(bho_object
);
107 // Now add the BHO to the collection of automation objects. This
108 // will ensure that BHO will be accessible from the web pages as
109 // any other BHO. Importantly, it will make sure that our BHO
110 // will be cleaned up at the right time along with other BHOs.
111 CComVariant
object_variant(bho_object
);
112 browser
->PutProperty(bho_clsid_as_string_bstr
, object_variant
);
121 bool BHOLoader::StartHook() {
122 return hooker_
->StartHook();
125 void BHOLoader::StopHook() {
129 BHOLoader
* BHOLoader::GetInstance() {
130 static BHOLoader loader
;