mirror of
https://bitbucket.org/chromiumembedded/cef
synced 2025-01-31 11:35:19 +01:00
a12c2ab3e1
The PDF loading documentation in extension_system.cc has be updated to describe the new code paths. To support delivery of input events to the mime handler renderer process it is now necessary to route events via the correct RWHV interface. For Aura-based platforms (Windows/Linux) this means RWHVAura::On*Event and for macOS this means RWHVMac::RouteOrProcess*Event. Since Aura uses UI event types these have become the source of truth on Aura-based platforms with conversion to Web event types when needed (primarily for OSR). This change also adds a timeout for CefProcessHostMsg_GetNewBrowserInfo to avoid a hung renderer process if the guest WebContents route is not registered via CefMimeHandlerViewGuestDelegate::OnGuestDetached as expected prior to CefBrowserInfoManager::OnGetNewBrowserInfo being called. This timeout can be disabled for testing purposes by passing the `--disable-new-browser-info-timeout` command-line flag. The `--disable-features=MimeHandlerViewInCrossProcessFrame` command-line flag can be used for a limited time to restore the previous implementation based on BrowserPlugin. That implementation will be deleted starting with the 3897 branch update. Known issues: - ExecuteJavaScript calls on the frame hosting the PDF extension will not be routed to the mime handler renderer process. - The PDF extension will not load successfully if blocked by ChromePluginPlaceholder and then manually continued via the "Run this plugin" context menu option (see https://crbug.com/533069#c41).
320 lines
10 KiB
C++
320 lines
10 KiB
C++
// Copyright 2015 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/native/browser_platform_delegate_native_linux.h"
|
|
|
|
#include <sys/sysinfo.h>
|
|
|
|
#include "libcef/browser/browser_host_impl.h"
|
|
#include "libcef/browser/context.h"
|
|
#include "libcef/browser/native/menu_runner_linux.h"
|
|
#include "libcef/browser/native/window_delegate_view.h"
|
|
#include "libcef/browser/thread_util.h"
|
|
|
|
#include "base/no_destructor.h"
|
|
#include "content/browser/renderer_host/render_widget_host_impl.h"
|
|
#include "content/public/browser/native_web_keyboard_event.h"
|
|
#include "content/public/browser/render_view_host.h"
|
|
#include "third_party/blink/public/mojom/renderer_preferences.mojom.h"
|
|
#include "ui/events/keycodes/dom/dom_key.h"
|
|
#include "ui/events/keycodes/dom/keycode_converter.h"
|
|
#include "ui/events/keycodes/keyboard_code_conversion_x.h"
|
|
#include "ui/events/keycodes/keyboard_code_conversion_xkb.h"
|
|
#include "ui/events/keycodes/keysym_to_unicode.h"
|
|
#include "ui/gfx/font_render_params.h"
|
|
#include "ui/views/widget/widget.h"
|
|
|
|
#if defined(USE_X11)
|
|
#include "libcef/browser/native/window_x11.h"
|
|
#include "ui/views/widget/desktop_aura/desktop_window_tree_host_x11.h"
|
|
#endif
|
|
|
|
namespace {
|
|
|
|
// Returns the number of seconds since system boot.
|
|
long GetSystemUptime() {
|
|
struct sysinfo info;
|
|
if (sysinfo(&info) == 0)
|
|
return info.uptime;
|
|
return 0;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
CefBrowserPlatformDelegateNativeLinux::CefBrowserPlatformDelegateNativeLinux(
|
|
const CefWindowInfo& window_info,
|
|
SkColor background_color,
|
|
bool use_external_begin_frame)
|
|
: CefBrowserPlatformDelegateNativeAura(window_info,
|
|
background_color,
|
|
false,
|
|
use_external_begin_frame),
|
|
host_window_created_(false),
|
|
window_widget_(nullptr) {}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::BrowserDestroyed(
|
|
CefBrowserHostImpl* browser) {
|
|
CefBrowserPlatformDelegate::BrowserDestroyed(browser);
|
|
|
|
if (host_window_created_) {
|
|
// Release the reference added in CreateHostWindow().
|
|
browser->Release();
|
|
}
|
|
}
|
|
|
|
bool CefBrowserPlatformDelegateNativeLinux::CreateHostWindow() {
|
|
DCHECK(!window_widget_);
|
|
|
|
if (window_info_.width == 0)
|
|
window_info_.width = 800;
|
|
if (window_info_.height == 0)
|
|
window_info_.height = 600;
|
|
|
|
gfx::Rect rect(window_info_.x, window_info_.y, window_info_.width,
|
|
window_info_.height);
|
|
|
|
#if defined(USE_X11)
|
|
DCHECK(!window_x11_);
|
|
// Create a new window object. It will delete itself when the associated X11
|
|
// window is destroyed.
|
|
window_x11_ =
|
|
new CefWindowX11(browser_, window_info_.parent_window, rect,
|
|
CefString(&window_info_.window_name).ToString());
|
|
window_info_.window = window_x11_->xwindow();
|
|
|
|
host_window_created_ = true;
|
|
|
|
// Add a reference that will be released in BrowserDestroyed().
|
|
browser_->AddRef();
|
|
|
|
CefWindowDelegateView* delegate_view = new CefWindowDelegateView(
|
|
GetBackgroundColor(), window_x11_->TopLevelAlwaysOnTop(),
|
|
GetBoundsChangedCallback());
|
|
delegate_view->Init(window_info_.window, browser_->web_contents(),
|
|
gfx::Rect(gfx::Point(), rect.size()));
|
|
|
|
window_widget_ = delegate_view->GetWidget();
|
|
window_widget_->Show();
|
|
|
|
window_x11_->Show();
|
|
#endif // defined(USE_X11)
|
|
|
|
// As an additional requirement on Linux, we must set the colors for the
|
|
// render widgets in webkit.
|
|
blink::mojom::RendererPreferences* prefs =
|
|
browser_->web_contents()->GetMutableRendererPrefs();
|
|
prefs->focus_ring_color = SkColorSetARGB(255, 229, 151, 0);
|
|
|
|
prefs->active_selection_bg_color = SkColorSetRGB(30, 144, 255);
|
|
prefs->active_selection_fg_color = SK_ColorWHITE;
|
|
prefs->inactive_selection_bg_color = SkColorSetRGB(200, 200, 200);
|
|
prefs->inactive_selection_fg_color = SkColorSetRGB(50, 50, 50);
|
|
|
|
// Set font-related attributes.
|
|
static const base::NoDestructor<gfx::FontRenderParams> params(
|
|
gfx::GetFontRenderParams(gfx::FontRenderParamsQuery(), nullptr));
|
|
prefs->should_antialias_text = params->antialiasing;
|
|
prefs->use_subpixel_positioning = params->subpixel_positioning;
|
|
prefs->hinting = params->hinting;
|
|
prefs->use_autohinter = params->autohinter;
|
|
prefs->use_bitmaps = params->use_bitmaps;
|
|
prefs->subpixel_rendering = params->subpixel_rendering;
|
|
|
|
browser_->web_contents()->SyncRendererPrefs();
|
|
|
|
return true;
|
|
}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::CloseHostWindow() {
|
|
#if defined(USE_X11)
|
|
if (window_x11_)
|
|
window_x11_->Close();
|
|
#endif
|
|
}
|
|
|
|
CefWindowHandle CefBrowserPlatformDelegateNativeLinux::GetHostWindowHandle()
|
|
const {
|
|
if (windowless_handler_)
|
|
return windowless_handler_->GetParentWindowHandle();
|
|
return window_info_.window;
|
|
}
|
|
|
|
views::Widget* CefBrowserPlatformDelegateNativeLinux::GetWindowWidget() const {
|
|
return window_widget_;
|
|
}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::SendFocusEvent(bool setFocus) {
|
|
if (!setFocus)
|
|
return;
|
|
|
|
if (browser_->web_contents()) {
|
|
// Give logical focus to the RenderWidgetHostViewAura in the views
|
|
// hierarchy. This does not change the native keyboard focus.
|
|
browser_->web_contents()->Focus();
|
|
}
|
|
|
|
#if defined(USE_X11)
|
|
if (window_x11_) {
|
|
// Give native focus to the DesktopNativeWidgetAura for the root window.
|
|
// Needs to be done via the ::Window so that keyboard focus is assigned
|
|
// correctly.
|
|
window_x11_->Focus();
|
|
}
|
|
#endif // defined(USE_X11)
|
|
}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::NotifyMoveOrResizeStarted() {
|
|
// Call the parent method to dismiss any existing popups.
|
|
CefBrowserPlatformDelegate::NotifyMoveOrResizeStarted();
|
|
|
|
#if defined(USE_X11)
|
|
if (!window_x11_)
|
|
return;
|
|
|
|
views::DesktopWindowTreeHostX11* tree_host = window_x11_->GetHost();
|
|
if (!tree_host)
|
|
return;
|
|
|
|
// Explicitly set the screen bounds so that WindowTreeHost::*Screen()
|
|
// methods return the correct results.
|
|
const gfx::Rect& bounds = window_x11_->GetBoundsInScreen();
|
|
tree_host->set_screen_bounds(bounds);
|
|
|
|
// Send updated screen rectangle information to the renderer process so that
|
|
// popups are displayed in the correct location.
|
|
content::RenderWidgetHostImpl::From(
|
|
browser_->web_contents()->GetRenderViewHost()->GetWidget())
|
|
->SendScreenRects();
|
|
#endif // defined(USE_X11)
|
|
}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::SizeTo(int width, int height) {
|
|
#if defined(USE_X11)
|
|
if (window_x11_) {
|
|
window_x11_->SetBounds(
|
|
gfx::Rect(window_x11_->bounds().origin(), gfx::Size(width, height)));
|
|
}
|
|
#endif // defined(USE_X11)
|
|
}
|
|
|
|
gfx::Point CefBrowserPlatformDelegateNativeLinux::GetScreenPoint(
|
|
const gfx::Point& view) const {
|
|
if (windowless_handler_)
|
|
return windowless_handler_->GetParentScreenPoint(view);
|
|
|
|
#if defined(USE_X11)
|
|
if (!window_x11_)
|
|
return view;
|
|
|
|
// We can't use aura::Window::GetBoundsInScreen on Linux because it will
|
|
// return bounds from DesktopWindowTreeHostX11 which in our case is relative
|
|
// to the parent window instead of the root window (screen).
|
|
const gfx::Rect& bounds_in_screen = window_x11_->GetBoundsInScreen();
|
|
return gfx::Point(bounds_in_screen.x() + view.x(),
|
|
bounds_in_screen.y() + view.y());
|
|
#endif // defined(USE_X11)
|
|
return gfx::Point();
|
|
}
|
|
|
|
void CefBrowserPlatformDelegateNativeLinux::ViewText(const std::string& text) {
|
|
char buff[] = "/tmp/CEFSourceXXXXXX";
|
|
int fd = mkstemp(buff);
|
|
|
|
if (fd == -1)
|
|
return;
|
|
|
|
FILE* srcOutput = fdopen(fd, "w+");
|
|
if (!srcOutput)
|
|
return;
|
|
|
|
if (fputs(text.c_str(), srcOutput) < 0) {
|
|
fclose(srcOutput);
|
|
return;
|
|
}
|
|
|
|
fclose(srcOutput);
|
|
|
|
std::string newName(buff);
|
|
newName.append(".txt");
|
|
if (rename(buff, newName.c_str()) != 0)
|
|
return;
|
|
|
|
std::string openCommand("xdg-open ");
|
|
openCommand += newName;
|
|
|
|
int result = system(openCommand.c_str());
|
|
ALLOW_UNUSED_LOCAL(result);
|
|
}
|
|
|
|
bool CefBrowserPlatformDelegateNativeLinux::HandleKeyboardEvent(
|
|
const content::NativeWebKeyboardEvent& event) {
|
|
// TODO(cef): Is something required here to handle shortcut keys?
|
|
return false;
|
|
}
|
|
|
|
// static
|
|
void CefBrowserPlatformDelegate::HandleExternalProtocol(const GURL& url) {}
|
|
|
|
CefEventHandle CefBrowserPlatformDelegateNativeLinux::GetEventHandle(
|
|
const content::NativeWebKeyboardEvent& event) const {
|
|
if (!event.os_event)
|
|
return nullptr;
|
|
return const_cast<CefEventHandle>(
|
|
static_cast<CefEventHandle>(event.os_event->native_event()));
|
|
}
|
|
|
|
std::unique_ptr<CefMenuRunner>
|
|
CefBrowserPlatformDelegateNativeLinux::CreateMenuRunner() {
|
|
return base::WrapUnique(new CefMenuRunnerLinux);
|
|
}
|
|
|
|
gfx::Point CefBrowserPlatformDelegateNativeLinux::GetDialogPosition(
|
|
const gfx::Size& size) {
|
|
const gfx::Size& max_size = GetMaximumDialogSize();
|
|
return gfx::Point((max_size.width() - size.width()) / 2,
|
|
(max_size.height() - size.height()) / 2);
|
|
}
|
|
|
|
gfx::Size CefBrowserPlatformDelegateNativeLinux::GetMaximumDialogSize() {
|
|
return GetWindowWidget()->GetWindowBoundsInScreen().size();
|
|
}
|
|
|
|
ui::KeyEvent CefBrowserPlatformDelegateNativeLinux::TranslateUiKeyEvent(
|
|
const CefKeyEvent& key_event) const {
|
|
int flags = TranslateUiEventModifiers(key_event.modifiers);
|
|
ui::KeyboardCode key_code =
|
|
static_cast<ui::KeyboardCode>(key_event.windows_key_code);
|
|
ui::DomCode dom_code =
|
|
ui::KeycodeConverter::NativeKeycodeToDomCode(key_event.native_key_code);
|
|
int keysym = ui::XKeysymForWindowsKeyCode(
|
|
key_code, !!(key_event.modifiers & EVENTFLAG_SHIFT_DOWN));
|
|
base::char16 character = ui::GetUnicodeCharacterFromXKeySym(keysym);
|
|
base::TimeTicks time_stamp = GetEventTimeStamp();
|
|
|
|
if (key_event.type == KEYEVENT_CHAR) {
|
|
return ui::KeyEvent(character, key_code, dom_code, flags, time_stamp);
|
|
}
|
|
|
|
ui::EventType type = ui::ET_UNKNOWN;
|
|
switch (key_event.type) {
|
|
case KEYEVENT_RAWKEYDOWN:
|
|
case KEYEVENT_KEYDOWN:
|
|
type = ui::ET_KEY_PRESSED;
|
|
break;
|
|
case KEYEVENT_KEYUP:
|
|
type = ui::ET_KEY_RELEASED;
|
|
break;
|
|
default:
|
|
NOTREACHED();
|
|
}
|
|
|
|
ui::DomKey dom_key = ui::XKeySymToDomKey(keysym, character);
|
|
return ui::KeyEvent(type, key_code, dom_code, flags, dom_key, time_stamp);
|
|
}
|
|
|
|
base::TimeTicks CefBrowserPlatformDelegateNativeLinux::GetEventTimeStamp()
|
|
const {
|
|
return base::TimeTicks() + base::TimeDelta::FromSeconds(GetSystemUptime());
|
|
}
|