// Copyright (c) 2014 GitHub, Inc. All rights reserved. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "atom/browser/native_window_gtk.h" #include #include #include "atom/common/draggable_region.h" #include "atom/common/options_switches.h" #include "base/environment.h" #include "base/nix/xdg_util.h" #include "chrome/browser/ui/gtk/gtk_window_util.h" #include "content/public/browser/native_web_keyboard_event.h" #include "content/public/browser/web_contents.h" #include "content/public/browser/web_contents_view.h" #include "content/public/common/renderer_preferences.h" #include "native_mate/dictionary.h" #include "ui/base/accelerators/platform_accelerator_gtk.h" #include "ui/base/models/simple_menu_model.h" #include "ui/base/x/active_window_watcher_x.h" #include "ui/base/x/x11_util.h" #include "ui/gfx/font_render_params_linux.h" #include "ui/gfx/gtk_util.h" #include "ui/gfx/image/image.h" #include "ui/gfx/rect.h" #include "ui/gfx/skia_utils_gtk.h" namespace atom { namespace { // Dividing GTK's cursor blink cycle time (in milliseconds) by this value yields // an appropriate value for content::RendererPreferences::caret_blink_interval. // This matches the logic in the WebKit GTK port. const double kGtkCursorBlinkCycleFactor = 2000.0; // Substract window border's size from window size according to current window // manager. void SubstractBorderSize(int* width, int* height) { scoped_ptr env(base::Environment::Create()); base::nix::DesktopEnvironment de(base::nix::GetDesktopEnvironment(env.get())); if (de == base::nix::DESKTOP_ENVIRONMENT_UNITY) { *width -= 2; *height -= 29; } else if (de == base::nix::DESKTOP_ENVIRONMENT_GNOME) { *width -= 2; *height -= 33; } else if (de == base::nix::DESKTOP_ENVIRONMENT_XFCE) { *width -= 6; *height -= 27; } else { *width -= 2; *height -= 29; } } content::RendererPreferencesHintingEnum GetRendererPreferencesHintingEnum( gfx::FontRenderParams::Hinting hinting) { switch (hinting) { case gfx::FontRenderParams::HINTING_NONE: return content::RENDERER_PREFERENCES_HINTING_NONE; case gfx::FontRenderParams::HINTING_SLIGHT: return content::RENDERER_PREFERENCES_HINTING_SLIGHT; case gfx::FontRenderParams::HINTING_MEDIUM: return content::RENDERER_PREFERENCES_HINTING_MEDIUM; case gfx::FontRenderParams::HINTING_FULL: return content::RENDERER_PREFERENCES_HINTING_FULL; default: NOTREACHED() << "Unhandled hinting style " << hinting; return content::RENDERER_PREFERENCES_HINTING_SYSTEM_DEFAULT; } } content::RendererPreferencesSubpixelRenderingEnum GetRendererPreferencesSubpixelRenderingEnum( gfx::FontRenderParams::SubpixelRendering subpixel_rendering) { switch (subpixel_rendering) { case gfx::FontRenderParams::SUBPIXEL_RENDERING_NONE: return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_NONE; case gfx::FontRenderParams::SUBPIXEL_RENDERING_RGB: return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_RGB; case gfx::FontRenderParams::SUBPIXEL_RENDERING_BGR: return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_BGR; case gfx::FontRenderParams::SUBPIXEL_RENDERING_VRGB: return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_VRGB; case gfx::FontRenderParams::SUBPIXEL_RENDERING_VBGR: return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_VBGR; default: NOTREACHED() << "Unhandled subpixel rendering style " << subpixel_rendering; return content::RENDERER_PREFERENCES_SUBPIXEL_RENDERING_SYSTEM_DEFAULT; } } } // namespace NativeWindowGtk::NativeWindowGtk(content::WebContents* web_contents, const mate::Dictionary& options) : NativeWindow(web_contents, options), window_(GTK_WINDOW(gtk_window_new(GTK_WINDOW_TOPLEVEL))), vbox_(gtk_vbox_new(FALSE, 0)), state_(GDK_WINDOW_STATE_WITHDRAWN), is_always_on_top_(false), is_active_(false), suppress_window_raise_(false), has_ever_been_shown_(false), frame_cursor_(NULL) { gtk_container_add(GTK_CONTAINER(window_), vbox_); gtk_container_add(GTK_CONTAINER(vbox_), GetWebContents()->GetView()->GetNativeView()); int width = 800, height = 600; options.Get(switches::kWidth, &width); options.Get(switches::kHeight, &height); bool use_content_size = false; options.Get(switches::kUseContentSize, &use_content_size); if (has_frame_ && !use_content_size) SubstractBorderSize(&width, &height); // Force a size allocation so the web page of hidden window can have correct // value of $(window).width(). GtkAllocation size = { 0, 0, width, height }; gtk_widget_show_all(vbox_); gtk_widget_size_allocate(GTK_WIDGET(window_), &size); gtk_window_util::SetWindowSize(window_, gfx::Size(width, height)); // Create the underlying gdk window. gtk_widget_realize(GTK_WIDGET(window_)); if (icon_) gtk_window_set_icon(window_, icon_->ToGdkPixbuf()); ui::ActiveWindowWatcherX::AddObserver(this); // In some (older) versions of compiz, raising top-level windows when they // are partially off-screen causes them to get snapped back on screen, not // always even on the current virtual desktop. If we are running under // compiz, suppress such raises, as they are not necessary in compiz anyway. if (ui::GuessWindowManager() == ui::WM_COMPIZ) suppress_window_raise_ = true; g_signal_connect(window_, "delete-event", G_CALLBACK(OnWindowDeleteEventThunk), this); g_signal_connect(window_, "focus-out-event", G_CALLBACK(OnFocusOutThunk), this); g_signal_connect(window_, "focus-in-event", G_CALLBACK(OnFocusInThunk), this); g_signal_connect(window_, "window-state-event", G_CALLBACK(OnWindowStateThunk), this); if (!has_frame_) { gtk_window_set_decorated(window_, false); g_signal_connect(window_, "motion-notify-event", G_CALLBACK(OnMouseMoveEventThunk), this); g_signal_connect(window_, "button-press-event", G_CALLBACK(OnButtonPressThunk), this); } SetWebKitColorStyle(); SetFontRenderering(); } NativeWindowGtk::~NativeWindowGtk() { CloseImmediately(); } void NativeWindowGtk::Close() { CloseWebContents(); } void NativeWindowGtk::CloseImmediately() { if (window_ == NULL) return; NotifyWindowClosed(); ui::ActiveWindowWatcherX::RemoveObserver(this); gtk_widget_destroy(GTK_WIDGET(window_)); window_ = NULL; } void NativeWindowGtk::Move(const gfx::Rect& pos) { gtk_window_move(window_, pos.x(), pos.y()); SetSize(pos.size()); } void NativeWindowGtk::Focus(bool focus) { if (!IsVisible()) return; if (focus) gtk_window_present(window_); else gdk_window_lower(gtk_widget_get_window(GTK_WIDGET(window_))); } bool NativeWindowGtk::IsFocused() { if (ui::ActiveWindowWatcherX::WMSupportsActivation()) return is_active_; // This still works even though we don't get the activation notification. return gtk_window_is_active(window_); } void NativeWindowGtk::Show() { has_ever_been_shown_ = true; gtk_widget_show_all(GTK_WIDGET(window_)); } void NativeWindowGtk::Hide() { gtk_widget_hide(GTK_WIDGET(window_)); } bool NativeWindowGtk::IsVisible() { return gtk_widget_get_visible(GTK_WIDGET(window_)); } void NativeWindowGtk::Maximize() { gtk_window_maximize(window_); } void NativeWindowGtk::Unmaximize() { gtk_window_unmaximize(window_); } void NativeWindowGtk::Minimize() { gtk_window_iconify(window_); } void NativeWindowGtk::Restore() { gtk_window_present(window_); } void NativeWindowGtk::SetFullscreen(bool fullscreen) { if (fullscreen) gtk_window_fullscreen(window_); else gtk_window_unfullscreen(window_); } bool NativeWindowGtk::IsFullscreen() { return state_ & GDK_WINDOW_STATE_FULLSCREEN; } void NativeWindowGtk::SetSize(const gfx::Size& size) { // When the window has not been mapped the window size does not include frame. int width = size.width(); int height = size.height(); if (has_frame_ && !has_ever_been_shown_) SubstractBorderSize(&width, &height); gtk_window_util::SetWindowSize(window_, gfx::Size(width, height)); } gfx::Size NativeWindowGtk::GetSize() { GdkWindow* gdk_window = gtk_widget_get_window(GTK_WIDGET(window_)); GdkRectangle frame_extents; gdk_window_get_frame_extents(gdk_window, &frame_extents); return gfx::Size(frame_extents.width, frame_extents.height); } void NativeWindowGtk::SetContentSize(const gfx::Size& size) { if (!has_frame_ || !has_ever_been_shown_) { gtk_window_util::SetWindowSize(window_, size); } else { gfx::Size large = GetSize(); gfx::Size small = GetContentSize(); gfx::Size target(size.width() + large.width() - small.width(), size.height() + large.height() - small.height()); gtk_window_util::SetWindowSize(window_, target); } } gfx::Size NativeWindowGtk::GetContentSize() { gint width, height; gtk_window_get_size(window_, &width, &height); return gfx::Size(width, height); } void NativeWindowGtk::SetMinimumSize(const gfx::Size& size) { minimum_size_ = size; GdkGeometry geometry = { 0 }; geometry.min_width = size.width(); geometry.min_height = size.height(); int hints = GDK_HINT_POS | GDK_HINT_MIN_SIZE; gtk_window_set_geometry_hints( window_, GTK_WIDGET(window_), &geometry, (GdkWindowHints)hints); } gfx::Size NativeWindowGtk::GetMinimumSize() { return minimum_size_; } void NativeWindowGtk::SetMaximumSize(const gfx::Size& size) { maximum_size_ = size; GdkGeometry geometry = { 0 }; geometry.max_width = size.width(); geometry.max_height = size.height(); int hints = GDK_HINT_POS | GDK_HINT_MAX_SIZE; gtk_window_set_geometry_hints( window_, GTK_WIDGET(window_), &geometry, (GdkWindowHints)hints); } gfx::Size NativeWindowGtk::GetMaximumSize() { return maximum_size_; } void NativeWindowGtk::SetResizable(bool resizable) { // Should request widget size after setting unresizable, otherwise the // window will shrink to a very small size. if (!IsResizable() || !has_ever_been_shown_) { gint width, height; gtk_window_get_size(window_, &width, &height); gtk_widget_set_size_request(GTK_WIDGET(window_), width, height); } gtk_window_set_resizable(window_, resizable); } bool NativeWindowGtk::IsResizable() { return gtk_window_get_resizable(window_); } void NativeWindowGtk::SetAlwaysOnTop(bool top) { is_always_on_top_ = top; gtk_window_set_keep_above(window_, top ? TRUE : FALSE); } bool NativeWindowGtk::IsAlwaysOnTop() { return is_always_on_top_; } void NativeWindowGtk::Center() { gtk_window_set_position(window_, GTK_WIN_POS_CENTER); } void NativeWindowGtk::SetPosition(const gfx::Point& position) { gtk_window_move(window_, position.x(), position.y()); } gfx::Point NativeWindowGtk::GetPosition() { GdkWindow* gdk_window = gtk_widget_get_window(GTK_WIDGET(window_)); GdkRectangle frame_extents; gdk_window_get_frame_extents(gdk_window, &frame_extents); return gfx::Point(frame_extents.x, frame_extents.y); } void NativeWindowGtk::SetTitle(const std::string& title) { gtk_window_set_title(window_, title.c_str()); } std::string NativeWindowGtk::GetTitle() { return gtk_window_get_title(window_); } void NativeWindowGtk::FlashFrame(bool flash) { gtk_window_set_urgency_hint(window_, flash); } void NativeWindowGtk::SetSkipTaskbar(bool skip) { gtk_window_set_skip_taskbar_hint(window_, skip); gtk_window_set_skip_pager_hint(window_, skip); } void NativeWindowGtk::SetKiosk(bool kiosk) { SetFullscreen(kiosk); } bool NativeWindowGtk::IsKiosk() { return IsFullscreen(); } gfx::NativeWindow NativeWindowGtk::GetNativeWindow() { return window_; } void NativeWindowGtk::SetMenu(ui::MenuModel* menu_model) { menu_.reset(new ::MenuGtk(this, menu_model, true)); gtk_box_pack_start(GTK_BOX(vbox_), menu_->widget(), FALSE, FALSE, 0); gtk_box_reorder_child(GTK_BOX(vbox_), menu_->widget(), 0); gtk_widget_show_all(vbox_); RegisterAccelerators(); } void NativeWindowGtk::UpdateDraggableRegions( const std::vector& regions) { // Draggable region is not supported for non-frameless window. if (has_frame_) return; draggable_region_.reset(new SkRegion); // By default, the whole window is non-draggable. We need to explicitly // include those draggable regions. for (std::vector::const_iterator iter = regions.begin(); iter != regions.end(); ++iter) { const DraggableRegion& region = *iter; draggable_region_->op( region.bounds.x(), region.bounds.y(), region.bounds.right(), region.bounds.bottom(), region.draggable ? SkRegion::kUnion_Op : SkRegion::kDifference_Op); } } void NativeWindowGtk::HandleKeyboardEvent( content::WebContents*, const content::NativeWebKeyboardEvent& event) { if (event.type == WebKit::WebInputEvent::RawKeyDown) { GdkEventKey* os_event = reinterpret_cast(event.os_event); ui::Accelerator accelerator = ui::AcceleratorForGdkKeyCodeAndModifier( os_event->keyval, static_cast(os_event->state)); accelerator_util::TriggerAcceleratorTableCommand(&accelerator_table_, accelerator); } } void NativeWindowGtk::ActiveWindowChanged(GdkWindow* active_window) { is_active_ = gtk_widget_get_window(GTK_WIDGET(window_)) == active_window; } void NativeWindowGtk::RegisterAccelerators() { DCHECK(menu_); accelerator_table_.clear(); accelerator_util::GenerateAcceleratorTable(&accelerator_table_, menu_->model()); } void NativeWindowGtk::SetWebKitColorStyle() { content::RendererPreferences* prefs = GetWebContents()->GetMutableRendererPrefs(); GtkStyle* frame_style = gtk_rc_get_style(GTK_WIDGET(window_)); prefs->focus_ring_color = gfx::GdkColorToSkColor(frame_style->bg[GTK_STATE_SELECTED]); prefs->thumb_active_color = SkColorSetRGB(244, 244, 244); prefs->thumb_inactive_color = SkColorSetRGB(234, 234, 234); prefs->track_color = SkColorSetRGB(211, 211, 211); GtkWidget* url_entry = gtk_entry_new(); GtkStyle* entry_style = gtk_rc_get_style(url_entry); prefs->active_selection_bg_color = gfx::GdkColorToSkColor(entry_style->base[GTK_STATE_SELECTED]); prefs->active_selection_fg_color = gfx::GdkColorToSkColor(entry_style->text[GTK_STATE_SELECTED]); prefs->inactive_selection_bg_color = gfx::GdkColorToSkColor(entry_style->base[GTK_STATE_ACTIVE]); prefs->inactive_selection_fg_color = gfx::GdkColorToSkColor(entry_style->text[GTK_STATE_ACTIVE]); gtk_widget_destroy(url_entry); const base::TimeDelta cursor_blink_time = gfx::GetCursorBlinkCycle(); prefs->caret_blink_interval = cursor_blink_time.InMilliseconds() ? cursor_blink_time.InMilliseconds() / kGtkCursorBlinkCycleFactor : 0; } void NativeWindowGtk::SetFontRenderering() { content::RendererPreferences* prefs = GetWebContents()->GetMutableRendererPrefs(); const gfx::FontRenderParams& params = gfx::GetDefaultWebKitFontRenderParams(); prefs->should_antialias_text = params.antialiasing; prefs->use_subpixel_positioning = params.subpixel_positioning; prefs->hinting = GetRendererPreferencesHintingEnum(params.hinting); prefs->use_autohinter = params.autohinter; prefs->use_bitmaps = params.use_bitmaps; prefs->subpixel_rendering = GetRendererPreferencesSubpixelRenderingEnum(params.subpixel_rendering); } bool NativeWindowGtk::IsMaximized() const { return state_ & GDK_WINDOW_STATE_MAXIMIZED; } bool NativeWindowGtk::GetWindowEdge(int x, int y, GdkWindowEdge* edge) { if (has_frame_) return false; if (IsMaximized() || IsFullscreen()) return false; return gtk_window_util::GetWindowEdge(GetSize(), 0, x, y, edge); } gboolean NativeWindowGtk::OnWindowDeleteEvent(GtkWidget* widget, GdkEvent* event) { Close(); return TRUE; } gboolean NativeWindowGtk::OnFocusIn(GtkWidget* window, GdkEventFocus*) { NotifyWindowFocus(); return FALSE; } gboolean NativeWindowGtk::OnFocusOut(GtkWidget* window, GdkEventFocus*) { NotifyWindowBlur(); return FALSE; } gboolean NativeWindowGtk::OnWindowState(GtkWidget* window, GdkEventWindowState* event) { state_ = event->new_window_state; return FALSE; } gboolean NativeWindowGtk::OnMouseMoveEvent(GtkWidget* widget, GdkEventMotion* event) { if (has_frame_) { // Reset the cursor. if (frame_cursor_) { frame_cursor_ = NULL; gdk_window_set_cursor(gtk_widget_get_window(GTK_WIDGET(window_)), NULL); } return FALSE; } if (!IsResizable()) return FALSE; // Update the cursor if we're on the custom frame border. GdkWindowEdge edge; bool has_hit_edge = GetWindowEdge(static_cast(event->x), static_cast(event->y), &edge); GdkCursorType new_cursor = GDK_LAST_CURSOR; if (has_hit_edge) new_cursor = gtk_window_util::GdkWindowEdgeToGdkCursorType(edge); GdkCursorType last_cursor = GDK_LAST_CURSOR; if (frame_cursor_) last_cursor = frame_cursor_->type; if (last_cursor != new_cursor) { frame_cursor_ = has_hit_edge ? gfx::GetCursor(new_cursor) : NULL; gdk_window_set_cursor(gtk_widget_get_window(GTK_WIDGET(window_)), frame_cursor_); } return FALSE; } gboolean NativeWindowGtk::OnButtonPress(GtkWidget* widget, GdkEventButton* event) { DCHECK(!has_frame_); // Make the button press coordinate relative to the browser window. int win_x, win_y; GdkWindow* gdk_window = gtk_widget_get_window(GTK_WIDGET(window_)); gdk_window_get_origin(gdk_window, &win_x, &win_y); GdkWindowEdge edge; gfx::Point point(static_cast(event->x_root - win_x), static_cast(event->y_root - win_y)); bool has_hit_edge = IsResizable() && GetWindowEdge(point.x(), point.y(), &edge); bool has_hit_titlebar = draggable_region_ && draggable_region_->contains(event->x, event->y); if (event->button == 1) { if (GDK_BUTTON_PRESS == event->type) { // Raise the window after a click on either the titlebar or the border to // match the behavior of most window managers, unless that behavior has // been suppressed. if ((has_hit_titlebar || has_hit_edge) && !suppress_window_raise_) gdk_window_raise(GTK_WIDGET(widget)->window); if (has_hit_edge) { gtk_window_begin_resize_drag(window_, edge, event->button, static_cast(event->x_root), static_cast(event->y_root), event->time); return TRUE; } else if (has_hit_titlebar) { GdkRectangle window_bounds = {0}; gdk_window_get_frame_extents(gdk_window, &window_bounds); gfx::Rect bounds(window_bounds.x, window_bounds.y, window_bounds.width, window_bounds.height); return gtk_window_util::HandleTitleBarLeftMousePress( window_, bounds, event); } } else if (GDK_2BUTTON_PRESS == event->type) { if (has_hit_titlebar && IsResizable()) { // Maximize/restore on double click. if (IsMaximized()) gtk_window_unmaximize(window_); else gtk_window_maximize(window_); return TRUE; } } } else if (event->button == 2) { if (has_hit_titlebar || has_hit_edge) gdk_window_lower(gdk_window); return TRUE; } return FALSE; } // static NativeWindow* NativeWindow::Create(content::WebContents* web_contents, const mate::Dictionary& options) { return new NativeWindowGtk(web_contents, options); } } // namespace atom