mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2024-12-13 10:06:28 +01:00
4fbd247231
This change adds support for: - Protocol and request handling. - Loading and navigation events. - Display and focus events. - Mouse/keyboard events. - Popup browsers. - Callbacks in the renderer process. - Misc. functionality required for ceftests. This change also adds a new CefBrowserProcessHandler::GetCookieableSchemes callback for configuring global state that will be applied to all CefCookieManagers by default. This global callback is currently required by the chrome runtime because the primary ProfileImpl is created via ChromeBrowserMainParts::PreMainMessageLoopRun (CreatePrimaryProfile) before OnContextCreated can be called. ProfileImpl will use the "C:\Users\[user]\AppData\Local\CEF\User Data\Default" directory by default (on Windows). Cookies may persist in this directory when running ceftests and may need to be manually deleted if those tests fail. Remaining work includes: - Support for client-created request contexts. - Embedding the browser in a Views hierarchy (cefclient support). - TryCloseBrowser and DoClose support. - Most of the CefSettings configuration. - DevTools protocol and window control (ShowDevTools, ExecuteDevToolsMethod). - CEF-specific WebUI pages (about, license, webui-hosts). - Context menu customization (CefContextMenuHandler). - Auto resize (SetAutoResizeEnabled). - Zoom settings (SetZoomLevel). - File dialog runner (RunFileDialog). - File and JS dialog handlers (CefDialogHandler, CefJSDialogHandler). - Extension loading (LoadExtension, etc). - Plugin loading (OnBeforePluginLoad). - Widevine loading (CefRegisterWidevineCdm). - PDF and print preview does not display. - Crash reporting is untested. - Mac: Web content loads but does not display. The following ceftests are now passing when run with the "--enable-chrome-runtime" command-line flag: CorsTest.* DisplayTest.*:-DisplayTest.AutoResize DOMTest.* DraggableRegionsTest.* ImageTest.* MessageRouterTest.* NavigationTest.* ParserTest.* RequestContextTest.*Global* RequestTest.* ResourceManagerTest.* ResourceRequestHandlerTest.* ResponseTest.* SchemeHandlerTest.* ServerTest.* StreamResourceHandlerTest.* StreamTest.* StringTest.* TaskTest.* TestServerTest.* ThreadTest.* URLRequestTest.*Global* V8Test.*:-V8Test.OnUncaughtExceptionDevTools ValuesTest.* WaitableEventTest.* XmlReaderTest.* ZipReaderTest.*
251 lines
8.1 KiB
C++
251 lines
8.1 KiB
C++
// Copyright 2016 The Chromium Embedded Framework Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style license that can be found
|
|
// in the LICENSE file.
|
|
|
|
#include "libcef/browser/views/browser_view_impl.h"
|
|
|
|
#include "libcef/browser/alloy/alloy_browser_host_impl.h"
|
|
#include "libcef/browser/browser_util.h"
|
|
#include "libcef/browser/context.h"
|
|
#include "libcef/browser/thread_util.h"
|
|
#include "libcef/browser/views/window_impl.h"
|
|
#include "libcef/features/runtime_checks.h"
|
|
|
|
#include "content/public/browser/native_web_keyboard_event.h"
|
|
#include "ui/content_accelerators/accelerator_util.h"
|
|
|
|
// static
|
|
CefRefPtr<CefBrowserView> CefBrowserView::CreateBrowserView(
|
|
CefRefPtr<CefClient> client,
|
|
const CefString& url,
|
|
const CefBrowserSettings& settings,
|
|
CefRefPtr<CefDictionaryValue> extra_info,
|
|
CefRefPtr<CefRequestContext> request_context,
|
|
CefRefPtr<CefBrowserViewDelegate> delegate) {
|
|
return CefBrowserViewImpl::Create(client, url, settings, extra_info,
|
|
request_context, delegate);
|
|
}
|
|
|
|
// static
|
|
CefRefPtr<CefBrowserView> CefBrowserView::GetForBrowser(
|
|
CefRefPtr<CefBrowser> browser) {
|
|
REQUIRE_ALLOY_RUNTIME();
|
|
CEF_REQUIRE_UIT_RETURN(nullptr);
|
|
|
|
AlloyBrowserHostImpl* browser_impl =
|
|
static_cast<AlloyBrowserHostImpl*>(browser.get());
|
|
if (browser_impl && browser_impl->IsViewsHosted())
|
|
return browser_impl->GetBrowserView();
|
|
return nullptr;
|
|
}
|
|
|
|
// static
|
|
CefRefPtr<CefBrowserViewImpl> CefBrowserViewImpl::Create(
|
|
CefRefPtr<CefClient> client,
|
|
const CefString& url,
|
|
const CefBrowserSettings& settings,
|
|
CefRefPtr<CefDictionaryValue> extra_info,
|
|
CefRefPtr<CefRequestContext> request_context,
|
|
CefRefPtr<CefBrowserViewDelegate> delegate) {
|
|
REQUIRE_ALLOY_RUNTIME();
|
|
CEF_REQUIRE_UIT_RETURN(nullptr);
|
|
|
|
CefRefPtr<CefBrowserViewImpl> browser_view = new CefBrowserViewImpl(delegate);
|
|
browser_view->SetPendingBrowserCreateParams(client, url, settings, extra_info,
|
|
request_context);
|
|
browser_view->Initialize();
|
|
browser_view->SetDefaults(settings);
|
|
return browser_view;
|
|
}
|
|
|
|
// static
|
|
CefRefPtr<CefBrowserViewImpl> CefBrowserViewImpl::CreateForPopup(
|
|
const CefBrowserSettings& settings,
|
|
CefRefPtr<CefBrowserViewDelegate> delegate) {
|
|
REQUIRE_ALLOY_RUNTIME();
|
|
CEF_REQUIRE_UIT_RETURN(nullptr);
|
|
|
|
CefRefPtr<CefBrowserViewImpl> browser_view = new CefBrowserViewImpl(delegate);
|
|
browser_view->Initialize();
|
|
browser_view->SetDefaults(settings);
|
|
return browser_view;
|
|
}
|
|
|
|
void CefBrowserViewImpl::WebContentsCreated(
|
|
content::WebContents* web_contents) {
|
|
if (root_view())
|
|
root_view()->SetWebContents(web_contents);
|
|
}
|
|
|
|
void CefBrowserViewImpl::BrowserCreated(
|
|
CefBrowserHostBase* browser,
|
|
base::RepeatingClosure on_bounds_changed) {
|
|
browser_ = browser;
|
|
on_bounds_changed_ = on_bounds_changed;
|
|
}
|
|
|
|
void CefBrowserViewImpl::BrowserDestroyed(CefBrowserHostBase* browser) {
|
|
DCHECK_EQ(browser, browser_);
|
|
browser_ = nullptr;
|
|
|
|
if (root_view())
|
|
root_view()->SetWebContents(nullptr);
|
|
}
|
|
|
|
bool CefBrowserViewImpl::HandleKeyboardEvent(
|
|
const content::NativeWebKeyboardEvent& event) {
|
|
if (!root_view())
|
|
return false;
|
|
|
|
views::FocusManager* focus_manager = root_view()->GetFocusManager();
|
|
if (!focus_manager)
|
|
return false;
|
|
|
|
if (HandleAccelerator(event, focus_manager))
|
|
return true;
|
|
|
|
// Give the CefWindowDelegate a chance to handle the event.
|
|
CefRefPtr<CefWindow> window =
|
|
view_util::GetWindowFor(root_view()->GetWidget());
|
|
CefWindowImpl* window_impl = static_cast<CefWindowImpl*>(window.get());
|
|
if (window_impl) {
|
|
CefKeyEvent cef_event;
|
|
if (browser_util::GetCefKeyEvent(event, cef_event) &&
|
|
window_impl->OnKeyEvent(cef_event)) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// Proceed with default native handling.
|
|
return unhandled_keyboard_event_handler_.HandleKeyboardEvent(event,
|
|
focus_manager);
|
|
}
|
|
|
|
CefRefPtr<CefBrowser> CefBrowserViewImpl::GetBrowser() {
|
|
CEF_REQUIRE_VALID_RETURN(nullptr);
|
|
return browser_;
|
|
}
|
|
|
|
void CefBrowserViewImpl::SetPreferAccelerators(bool prefer_accelerators) {
|
|
CEF_REQUIRE_VALID_RETURN_VOID();
|
|
if (root_view())
|
|
root_view()->set_allow_accelerators(prefer_accelerators);
|
|
}
|
|
|
|
void CefBrowserViewImpl::SetBackgroundColor(cef_color_t color) {
|
|
CEF_REQUIRE_VALID_RETURN_VOID();
|
|
ParentClass::SetBackgroundColor(color);
|
|
if (root_view())
|
|
root_view()->SetResizeBackgroundColor(color);
|
|
}
|
|
|
|
void CefBrowserViewImpl::Detach() {
|
|
ParentClass::Detach();
|
|
|
|
// root_view() will be nullptr now.
|
|
DCHECK(!root_view());
|
|
|
|
if (browser_) {
|
|
// |browser_| will disappear when WindowDestroyed() indirectly calls
|
|
// BrowserDestroyed() so keep a reference.
|
|
CefRefPtr<CefBrowserHostBase> browser = browser_;
|
|
|
|
// Force the browser to be destroyed.
|
|
static_cast<AlloyBrowserHostImpl*>(browser.get())->WindowDestroyed();
|
|
}
|
|
}
|
|
|
|
void CefBrowserViewImpl::GetDebugInfo(base::DictionaryValue* info,
|
|
bool include_children) {
|
|
ParentClass::GetDebugInfo(info, include_children);
|
|
if (browser_)
|
|
info->SetString("url", browser_->GetMainFrame()->GetURL().ToString());
|
|
}
|
|
|
|
void CefBrowserViewImpl::OnBrowserViewAdded() {
|
|
if (!browser_ && pending_browser_create_params_) {
|
|
// Top-level browsers will be created when this view is added to the views
|
|
// hierarchy.
|
|
pending_browser_create_params_->browser_view = this;
|
|
|
|
AlloyBrowserHostImpl::Create(*pending_browser_create_params_);
|
|
DCHECK(browser_);
|
|
|
|
pending_browser_create_params_.reset(nullptr);
|
|
}
|
|
}
|
|
|
|
void CefBrowserViewImpl::OnBoundsChanged() {
|
|
if (!on_bounds_changed_.is_null())
|
|
on_bounds_changed_.Run();
|
|
}
|
|
|
|
CefBrowserViewImpl::CefBrowserViewImpl(
|
|
CefRefPtr<CefBrowserViewDelegate> delegate)
|
|
: ParentClass(delegate) {}
|
|
|
|
void CefBrowserViewImpl::SetPendingBrowserCreateParams(
|
|
CefRefPtr<CefClient> client,
|
|
const CefString& url,
|
|
const CefBrowserSettings& settings,
|
|
CefRefPtr<CefDictionaryValue> extra_info,
|
|
CefRefPtr<CefRequestContext> request_context) {
|
|
DCHECK(!pending_browser_create_params_);
|
|
pending_browser_create_params_.reset(new CefBrowserCreateParams());
|
|
pending_browser_create_params_->client = client;
|
|
pending_browser_create_params_->url = GURL(url.ToString());
|
|
pending_browser_create_params_->settings = settings;
|
|
pending_browser_create_params_->extra_info = extra_info;
|
|
pending_browser_create_params_->request_context = request_context;
|
|
}
|
|
|
|
void CefBrowserViewImpl::SetDefaults(const CefBrowserSettings& settings) {
|
|
SetBackgroundColor(
|
|
CefContext::Get()->GetBackgroundColor(&settings, STATE_DISABLED));
|
|
}
|
|
|
|
CefBrowserViewView* CefBrowserViewImpl::CreateRootView() {
|
|
return new CefBrowserViewView(delegate(), this);
|
|
}
|
|
|
|
void CefBrowserViewImpl::InitializeRootView() {
|
|
static_cast<CefBrowserViewView*>(root_view())->Initialize();
|
|
}
|
|
|
|
bool CefBrowserViewImpl::HandleAccelerator(
|
|
const content::NativeWebKeyboardEvent& event,
|
|
views::FocusManager* focus_manager) {
|
|
// Previous calls to TranslateMessage can generate Char events as well as
|
|
// RawKeyDown events, even if the latter triggered an accelerator. In these
|
|
// cases, we discard the Char events.
|
|
if (event.GetType() == blink::WebInputEvent::Type::kChar &&
|
|
ignore_next_char_event_) {
|
|
ignore_next_char_event_ = false;
|
|
return true;
|
|
}
|
|
|
|
// It's necessary to reset this flag, because a RawKeyDown event may not
|
|
// always generate a Char event.
|
|
ignore_next_char_event_ = false;
|
|
|
|
if (event.GetType() == blink::WebInputEvent::Type::kRawKeyDown) {
|
|
ui::Accelerator accelerator =
|
|
ui::GetAcceleratorFromNativeWebKeyboardEvent(event);
|
|
|
|
// This is tricky: we want to set ignore_next_char_event_ if
|
|
// ProcessAccelerator returns true. But ProcessAccelerator might delete
|
|
// |this| if the accelerator is a "close tab" one. So we speculatively
|
|
// set the flag and fix it if no event was handled.
|
|
ignore_next_char_event_ = true;
|
|
|
|
if (focus_manager->ProcessAccelerator(accelerator))
|
|
return true;
|
|
|
|
// ProcessAccelerator didn't handle the accelerator, so we know both
|
|
// that |this| is still valid, and that we didn't want to set the flag.
|
|
ignore_next_char_event_ = false;
|
|
}
|
|
|
|
return false;
|
|
}
|