// This file is part of BOINC. // http://boinc.berkeley.edu // Copyright (C) 2014-2015 University of California // // BOINC is free software; you can redistribute it and/or modify it // under the terms of the GNU Lesser General Public License // as published by the Free Software Foundation, // either version 3 of the License, or (at your option) any later version. // // BOINC is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. // See the GNU Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public License // along with BOINC. If not, see . #define _ATL_FREE_THREADED #define _ATL_CSTRING_EXPLICIT_CONSTRUCTORS #include #include #include #include #include #include #include #include #include #include #include #include #include "win_util.h" #include "version.h" #include "boinc_api.h" #include "diagnostics.h" #include "filesys.h" #include "browser_i.h" #include "browser_win.h" #include "browserlog.h" #include "browserctrlui_win.h" #include "browserctrl_win.h" #ifdef __IDeveloperConsoleMessageReceiver_INTERNAL__ // New for IE10 #define IDM_ADDCONSOLEMESSAGERECEIVER 3800 #define IDM_REMOVECONSOLEMESSAGERECEIVER 3801 #endif CWndClassInfo& CHTMLBrowserHost::GetWndClassInfo() { static CWndClassInfo wc = { { sizeof(WNDCLASSEX), 0, StartWindowProc, 0, 0, 0, 0, 0, (HBRUSH)(COLOR_WINDOW + 1), 0, NULL, 0 }, NULL, NULL, IDC_ARROW, TRUE, 0, _T("") }; return wc; } HRESULT CHTMLBrowserHost::FinalConstruct() { return S_OK; } void CHTMLBrowserHost::FinalRelease() { HRESULT hr; // Unregister the developer console message receiver // CComPtr pBrowser; hr = QueryControl(__uuidof(IWebBrowser2), (void**)&pBrowser); if (SUCCEEDED(hr)) { CComPtr spDocDisp; CComPtr spWebBrowser; spWebBrowser = pBrowser; hr = spWebBrowser->get_Document(&spDocDisp); if (SUCCEEDED(hr)) { CComQIPtr spOleCommandTarget(spDocDisp); spOleCommandTarget->Exec( &CGID_MSHTML, IDM_REMOVECONSOLEMESSAGERECEIVER, OLECMDEXECOPT_DODEFAULT, &varConsoleCookie, NULL ); } } ReleaseAll(); } void CHTMLBrowserHost::OnFinalMessage(HWND /*hWnd*/) { } HWND CHTMLBrowserHost::Create( HWND hWndParent, _U_RECT rect, LPCTSTR szWindowName, DWORD dwStyle, DWORD dwExStyle, _U_MENUorID MenuOrID, LPVOID lpCreateParam ){ ATOM atom = GetWndClassInfo().Register(&m_pfnSuperWindowProc); if (!atom) return NULL; // Allocate the thunk structure here, where we can fail gracefully. BOOL result = m_thunk.Init(NULL,NULL); if (result == FALSE) { SetLastError(ERROR_OUTOFMEMORY); return NULL; } _AtlWinModule.AddCreateWndData(&m_thunk.cd, this); dwStyle = GetWndStyle(dwStyle); dwExStyle = GetWndExStyle(dwExStyle); // Set Caption if (szWindowName == NULL) { szWindowName = GetWndCaption(); } // Create window return CWindow::Create((LPCTSTR)atom, hWndParent, rect, szWindowName, dwStyle, dwExStyle, MenuOrID, lpCreateParam); } STDMETHODIMP CHTMLBrowserHost::CreateControlEx( LPCOLESTR lpszTricsData, HWND hWnd, IStream* pStream, IUnknown** ppUnk, REFIID iidAdvise, IUnknown* punkSink ){ HRESULT hr = CreateControlLicEx(lpszTricsData, hWnd, pStream, ppUnk, iidAdvise, punkSink, NULL); if (SUCCEEDED(hr)) { CComPtr spWebBrowser; CComPtr pBrowser; hr = QueryControl(__uuidof(IWebBrowser2), (void**)&pBrowser); if (SUCCEEDED(hr) && pBrowser != NULL) { CComVariant v; CComVariant url("about:blank"); spWebBrowser = pBrowser; spWebBrowser->Navigate2(&url, &v, &v, &v, &v); // Register to receive the developer console message receiver events // javascript: 'window.console' // CComPtr pConsole; hr = QueryInterface(__uuidof(IDeveloperConsoleMessageReceiver), (void**)&pConsole); if (SUCCEEDED(hr)) { CComPtr spDocDisp; CComVariant varListener(pConsole); spWebBrowser = pBrowser; hr = spWebBrowser->get_Document(&spDocDisp); if (SUCCEEDED(hr)) { CComQIPtr spOleCommandTarget(spDocDisp); hr = spOleCommandTarget->Exec( &CGID_MSHTML, IDM_ADDCONSOLEMESSAGERECEIVER, OLECMDEXECOPT_DODEFAULT, &varListener, &varConsoleCookie ); } } } // Register a custom interface to extend the HTML Document Object Model // javascript: 'window.external' // CComObject *pUIObject = NULL; hr = CComObject::CreateInstance(&pUIObject); if (SUCCEEDED(hr) && pUIObject != NULL) { SetExternalDispatch((IHTMLBrowserHostUI*)pUIObject); } } return hr; } LPCWSTR MapMessageLevel(DEV_CONSOLE_MESSAGE_LEVEL level) { switch(level) { case DCML_WARNING: return L"WARNING"; case DCML_ERROR: return L"ERROR"; case DCML_INFORMATIONAL: default: return L"INFO"; } } STDMETHODIMP CHTMLBrowserHost::Write( LPCWSTR source, DEV_CONSOLE_MESSAGE_LEVEL level, int messageId, LPCWSTR messageText ){ return WriteWithUrl(source, level, messageId, messageText, L""); } STDMETHODIMP CHTMLBrowserHost::WriteWithUrl( LPCWSTR source, DEV_CONSOLE_MESSAGE_LEVEL level, int messageId, LPCWSTR messageText, LPCWSTR fileUrl ){ return WriteWithUrlAndLine(source, level, messageId, messageText, fileUrl, 0); } STDMETHODIMP CHTMLBrowserHost::WriteWithUrlAndLine( LPCWSTR source, DEV_CONSOLE_MESSAGE_LEVEL level, int messageId, LPCWSTR messageText, LPCWSTR fileUrl, ULONG line ){ return WriteWithUrlLineAndColumn(source, level, messageId, messageText, fileUrl, line, 0); } STDMETHODIMP CHTMLBrowserHost::WriteWithUrlLineAndColumn( LPCWSTR source, DEV_CONSOLE_MESSAGE_LEVEL level, int messageId, LPCWSTR messageText, LPCWSTR fileUrl, ULONG line, ULONG column ){ if ((CComBSTR("DOM") == CComBSTR(source)) && (7011 == messageId)) { // We do not need to worry about forward/backward caching being disabled } else if ((CComBSTR("HTML") == CComBSTR(source)) && (CComBSTR("about:blank") == CComBSTR(fileUrl))) { // We do not need to worry about warnings and errors from a blank page } else { browserlog_msg( "Console: (%S) (%S%d) %S\n" " File: %S, Line: %d, Column: %d", MapMessageLevel(level), source, messageId, messageText, fileUrl, line, column ); } return S_OK; } STDMETHODIMP CHTMLBrowserHost::ShowMessage( HWND hwnd, LPOLESTR lpstrText, LPOLESTR lpstrCaption, DWORD dwType, LPOLESTR lpstrHelpFile, DWORD dwHelpContext, LRESULT *plResult ){ browserlog_msg( "Show Message:\n" " Caption: %S\n" " Text: %S\n", lpstrCaption, lpstrText ); return S_OK; } STDMETHODIMP CHTMLBrowserHost::ShowHelp( HWND hwnd, LPOLESTR pszHelpFile, UINT uCommand, DWORD dwData, POINT ptMouse, IDispatch *pDispatchObjectHit ){ return S_OK; }; STDMETHODIMP CHTMLBrowserHost::QueryStatus( const GUID *pguidCmdGroup, ULONG cCmds, OLECMD prgCmds[], OLECMDTEXT *pCmdText ){ return E_NOTIMPL; } STDMETHODIMP CHTMLBrowserHost::Exec( const GUID* pguidCmdGroup, DWORD nCmdID, DWORD nCmdexecopt, VARIANTARG* pvaIn, VARIANTARG* pvaOut ){ HRESULT hr = S_OK; if (pguidCmdGroup && IsEqualGUID(*pguidCmdGroup, CGID_DocHostCommandHandler)) { switch (nCmdID) { case OLECMDID_SHOWSCRIPTERROR: { // Stop running scripts on the page. (*pvaOut).vt = VT_BOOL; (*pvaOut).boolVal = VARIANT_FALSE; break; } default: hr = OLECMDERR_E_NOTSUPPORTED; break; } } else { hr = OLECMDERR_E_UNKNOWNGROUP; } return hr; }