| Index: chrome/browser/ui/gtk/browser_window_gtk.cc | 
| diff --git a/chrome/browser/ui/gtk/browser_window_gtk.cc b/chrome/browser/ui/gtk/browser_window_gtk.cc | 
| deleted file mode 100644 | 
| index 9c6c24368e69eeb7b2862fad54b2d4720b8b81ac..0000000000000000000000000000000000000000 | 
| --- a/chrome/browser/ui/gtk/browser_window_gtk.cc | 
| +++ /dev/null | 
| @@ -1,2385 +0,0 @@ | 
| -// Copyright 2012 The Chromium 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 "chrome/browser/ui/gtk/browser_window_gtk.h" | 
| - | 
| -#include <gdk/gdkkeysyms.h> | 
| - | 
| -#include <algorithm> | 
| -#include <string> | 
| - | 
| -#include "base/base_paths.h" | 
| -#include "base/bind.h" | 
| -#include "base/command_line.h" | 
| -#include "base/debug/trace_event.h" | 
| -#include "base/environment.h" | 
| -#include "base/i18n/file_util_icu.h" | 
| -#include "base/logging.h" | 
| -#include "base/memory/scoped_ptr.h" | 
| -#include "base/memory/singleton.h" | 
| -#include "base/message_loop/message_loop.h" | 
| -#include "base/nix/xdg_util.h" | 
| -#include "base/path_service.h" | 
| -#include "base/prefs/pref_service.h" | 
| -#include "base/prefs/scoped_user_pref_update.h" | 
| -#include "base/strings/string_util.h" | 
| -#include "base/strings/utf_string_conversions.h" | 
| -#include "base/time/time.h" | 
| -#include "chrome/app/chrome_command_ids.h" | 
| -#include "chrome/browser/app_mode/app_mode_utils.h" | 
| -#include "chrome/browser/browser_process.h" | 
| -#include "chrome/browser/chrome_notification_types.h" | 
| -#include "chrome/browser/download/download_item_model.h" | 
| -#include "chrome/browser/extensions/tab_helper.h" | 
| -#include "chrome/browser/infobars/infobar_service.h" | 
| -#include "chrome/browser/profiles/profile.h" | 
| -#include "chrome/browser/themes/theme_properties.h" | 
| -#include "chrome/browser/ui/app_modal_dialogs/app_modal_dialog_queue.h" | 
| -#include "chrome/browser/ui/bookmarks/bookmark_tab_helper.h" | 
| -#include "chrome/browser/ui/browser.h" | 
| -#include "chrome/browser/ui/browser_command_controller.h" | 
| -#include "chrome/browser/ui/browser_commands.h" | 
| -#include "chrome/browser/ui/browser_dialogs.h" | 
| -#include "chrome/browser/ui/browser_list.h" | 
| -#include "chrome/browser/ui/browser_window_state.h" | 
| -#include "chrome/browser/ui/find_bar/find_bar_controller.h" | 
| -#include "chrome/browser/ui/find_bar/find_tab_helper.h" | 
| -#include "chrome/browser/ui/gtk/accelerators_gtk.h" | 
| -#include "chrome/browser/ui/gtk/avatar_menu_bubble_gtk.h" | 
| -#include "chrome/browser/ui/gtk/avatar_menu_button_gtk.h" | 
| -#include "chrome/browser/ui/gtk/bookmarks/bookmark_bar_gtk.h" | 
| -#include "chrome/browser/ui/gtk/browser_titlebar.h" | 
| -#include "chrome/browser/ui/gtk/browser_toolbar_gtk.h" | 
| -#include "chrome/browser/ui/gtk/create_application_shortcuts_dialog_gtk.h" | 
| -#include "chrome/browser/ui/gtk/download/download_in_progress_dialog_gtk.h" | 
| -#include "chrome/browser/ui/gtk/download/download_shelf_gtk.h" | 
| -#include "chrome/browser/ui/gtk/edit_search_engine_dialog.h" | 
| -#include "chrome/browser/ui/gtk/extensions/extension_keybinding_registry_gtk.h" | 
| -#include "chrome/browser/ui/gtk/find_bar_gtk.h" | 
| -#include "chrome/browser/ui/gtk/fullscreen_exit_bubble_gtk.h" | 
| -#include "chrome/browser/ui/gtk/global_menu_bar.h" | 
| -#include "chrome/browser/ui/gtk/gtk_theme_service.h" | 
| -#include "chrome/browser/ui/gtk/gtk_util.h" | 
| -#include "chrome/browser/ui/gtk/gtk_window_util.h" | 
| -#include "chrome/browser/ui/gtk/infobars/infobar_container_gtk.h" | 
| -#include "chrome/browser/ui/gtk/infobars/infobar_gtk.h" | 
| -#include "chrome/browser/ui/gtk/location_bar_view_gtk.h" | 
| -#include "chrome/browser/ui/gtk/nine_box.h" | 
| -#include "chrome/browser/ui/gtk/one_click_signin_bubble_gtk.h" | 
| -#include "chrome/browser/ui/gtk/password_generation_bubble_gtk.h" | 
| -#include "chrome/browser/ui/gtk/reload_button_gtk.h" | 
| -#include "chrome/browser/ui/gtk/status_bubble_gtk.h" | 
| -#include "chrome/browser/ui/gtk/tab_contents_container_gtk.h" | 
| -#include "chrome/browser/ui/gtk/tabs/tab_strip_gtk.h" | 
| -#include "chrome/browser/ui/gtk/task_manager_gtk.h" | 
| -#include "chrome/browser/ui/gtk/update_recommended_dialog.h" | 
| -#include "chrome/browser/ui/gtk/website_settings/website_settings_popup_gtk.h" | 
| -#include "chrome/browser/ui/omnibox/location_bar.h" | 
| -#include "chrome/browser/ui/omnibox/omnibox_view.h" | 
| -#include "chrome/browser/ui/tabs/tab_strip_model.h" | 
| -#include "chrome/browser/web_applications/web_app.h" | 
| -#include "chrome/common/chrome_switches.h" | 
| -#include "chrome/common/pref_names.h" | 
| -#include "components/user_prefs/pref_registry_syncable.h" | 
| -#include "content/public/browser/download_manager.h" | 
| -#include "content/public/browser/native_web_keyboard_event.h" | 
| -#include "content/public/browser/notification_service.h" | 
| -#include "content/public/browser/render_view_host.h" | 
| -#include "content/public/browser/render_widget_host_view.h" | 
| -#include "content/public/browser/web_contents.h" | 
| -#include "content/public/browser/web_contents_view.h" | 
| -#include "grit/chromium_strings.h" | 
| -#include "grit/generated_resources.h" | 
| -#include "grit/theme_resources.h" | 
| -#include "grit/ui_resources.h" | 
| -#include "ui/base/accelerators/platform_accelerator_gtk.h" | 
| -#include "ui/base/gtk/gtk_floating_container.h" | 
| -#include "ui/base/gtk/gtk_hig_constants.h" | 
| -#include "ui/base/gtk/gtk_screen_util.h" | 
| -#include "ui/base/l10n/l10n_util.h" | 
| -#include "ui/base/resource/resource_bundle.h" | 
| -#include "ui/base/x/active_window_watcher_x.h" | 
| -#include "ui/events/keycodes/keyboard_codes.h" | 
| -#include "ui/gfx/gtk_util.h" | 
| -#include "ui/gfx/image/cairo_cached_surface.h" | 
| -#include "ui/gfx/image/image.h" | 
| -#include "ui/gfx/rect.h" | 
| -#include "ui/gfx/screen.h" | 
| -#include "ui/gfx/skia_utils_gtk.h" | 
| - | 
| -using content::NativeWebKeyboardEvent; | 
| -using content::SSLStatus; | 
| -using content::WebContents; | 
| -using web_modal::WebContentsModalDialogHost; | 
| - | 
| -#error "The GTK+ port will be deleted later this week. If you are seeing this, you are trying to compile it. Please check your gyp flags for 'use_aura=0' and remove them." | 
| - | 
| -namespace { | 
| - | 
| -// The number of milliseconds between loading animation frames. | 
| -const int kLoadingAnimationFrameTimeMs = 30; | 
| - | 
| -const char* kBrowserWindowKey = "__BROWSER_WINDOW_GTK__"; | 
| - | 
| -// While resize areas on Windows are normally the same size as the window | 
| -// borders, our top area is shrunk by 1 px to make it easier to move the window | 
| -// around with our thinner top grabbable strip.  (Incidentally, our side and | 
| -// bottom resize areas don't match the frame border thickness either -- they | 
| -// span the whole nonclient area, so there's no "dead zone" for the mouse.) | 
| -const int kTopResizeAdjust = 1; | 
| -// The thickness of the shadow around the toolbar+web content area.  There are | 
| -// actually a couple pixels more that should overlap the toolbar and web | 
| -// content area, but we don't use those pixels. | 
| -const int kContentShadowThickness = 2; | 
| -// The offset to the background when the custom frame is off.  We want the | 
| -// window background to line up with the tab background regardless of whether | 
| -// we're in custom frame mode or not.  Since themes are designed with the | 
| -// custom frame in mind, we need to offset the background when the custom frame | 
| -// is off. | 
| -const int kCustomFrameBackgroundVerticalOffset = 15; | 
| - | 
| -// The timeout in milliseconds before we'll get the true window position with | 
| -// gtk_window_get_position() after the last GTK configure-event signal. | 
| -const int kDebounceTimeoutMilliseconds = 100; | 
| - | 
| -// Using gtk_window_get_position/size creates a race condition, so only use | 
| -// this to get the initial bounds.  After window creation, we pick up the | 
| -// normal bounds by connecting to the configure-event signal. | 
| -gfx::Rect GetInitialWindowBounds(GtkWindow* window) { | 
| -  gint x, y, width, height; | 
| -  gtk_window_get_position(window, &x, &y); | 
| -  gtk_window_get_size(window, &width, &height); | 
| -  return gfx::Rect(x, y, width, height); | 
| -} | 
| - | 
| -// Get the command ids of the key combinations that are not valid gtk | 
| -// accelerators. | 
| -int GetCustomCommandId(GdkEventKey* event) { | 
| -  // Filter modifier to only include accelerator modifiers. | 
| -  guint modifier = event->state & gtk_accelerator_get_default_mod_mask(); | 
| -  switch (event->keyval) { | 
| -    // Gtk doesn't allow GDK_Tab or GDK_ISO_Left_Tab to be an accelerator (see | 
| -    // gtk_accelerator_valid), so we need to handle these accelerators | 
| -    // manually. | 
| -    // Some X clients (e.g. cygwin, NX client, etc.) also send GDK_KP_Tab when | 
| -    // typing a tab key. We should also handle GDK_KP_Tab for such X clients as | 
| -    // Firefox does. | 
| -    case GDK_Tab: | 
| -    case GDK_ISO_Left_Tab: | 
| -    case GDK_KP_Tab: | 
| -      if (GDK_CONTROL_MASK == modifier) { | 
| -        return IDC_SELECT_NEXT_TAB; | 
| -      } else if ((GDK_CONTROL_MASK | GDK_SHIFT_MASK) == modifier) { | 
| -        return IDC_SELECT_PREVIOUS_TAB; | 
| -      } | 
| -      break; | 
| - | 
| -    default: | 
| -      break; | 
| -  } | 
| -  return -1; | 
| -} | 
| - | 
| -// Get the command ids of the accelerators that we don't want the native widget | 
| -// to be able to override. | 
| -int GetPreHandleCommandId(GdkEventKey* event) { | 
| -  // Filter modifier to only include accelerator modifiers. | 
| -  guint modifier = event->state & gtk_accelerator_get_default_mod_mask(); | 
| -  switch (event->keyval) { | 
| -    case GDK_Page_Down: | 
| -      if (GDK_CONTROL_MASK == modifier) { | 
| -        return IDC_SELECT_NEXT_TAB; | 
| -      } else if ((GDK_CONTROL_MASK | GDK_SHIFT_MASK) == modifier) { | 
| -        return IDC_MOVE_TAB_NEXT; | 
| -      } | 
| -      break; | 
| - | 
| -    case GDK_Page_Up: | 
| -      if (GDK_CONTROL_MASK == modifier) { | 
| -        return IDC_SELECT_PREVIOUS_TAB; | 
| -      } else if ((GDK_CONTROL_MASK | GDK_SHIFT_MASK) == modifier) { | 
| -        return IDC_MOVE_TAB_PREVIOUS; | 
| -      } | 
| -      break; | 
| - | 
| -    default: | 
| -      break; | 
| -  } | 
| -  return -1; | 
| -} | 
| - | 
| -GQuark GetBrowserWindowQuarkKey() { | 
| -  static GQuark quark = g_quark_from_static_string(kBrowserWindowKey); | 
| -  return quark; | 
| -} | 
| - | 
| -}  // namespace | 
| - | 
| -BrowserWindowGtk::BrowserWindowGtk(Browser* browser) | 
| -    :  window_(NULL), | 
| -       window_has_shown_(false), | 
| -       window_container_(NULL), | 
| -       window_vbox_(NULL), | 
| -       render_area_vbox_(NULL), | 
| -       render_area_floating_container_(NULL), | 
| -       render_area_event_box_(NULL), | 
| -       toolbar_border_(NULL), | 
| -       browser_(browser), | 
| -       state_(GDK_WINDOW_STATE_WITHDRAWN), | 
| -       devtools_window_(NULL), | 
| -       devtools_floating_container_(NULL), | 
| -       frame_cursor_(NULL), | 
| -       is_active_(false), | 
| -       show_state_after_show_(ui::SHOW_STATE_DEFAULT), | 
| -       suppress_window_raise_(false), | 
| -       accel_group_(NULL), | 
| -       is_fullscreen_(false) { | 
| -} | 
| - | 
| -BrowserWindowGtk::~BrowserWindowGtk() { | 
| -  ui::ActiveWindowWatcherX::RemoveObserver(this); | 
| - | 
| -  browser_->tab_strip_model()->RemoveObserver(this); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Init() { | 
| -  // We register first so that other views like the toolbar can use the | 
| -  // is_active() function in their ActiveWindowChanged() handlers. | 
| -  ui::ActiveWindowWatcherX::AddObserver(this); | 
| - | 
| -  use_custom_frame_pref_.Init( | 
| -      prefs::kUseCustomChromeFrame, | 
| -      browser_->profile()->GetPrefs(), | 
| -      base::Bind(&BrowserWindowGtk::OnUseCustomChromeFrameChanged, | 
| -                 base::Unretained(this))); | 
| - | 
| -  // Register to be notified of changes to the profile's avatar icon. | 
| -  if (!browser_->profile()->IsOffTheRecord()) { | 
| -    registrar_.Add(this, chrome::NOTIFICATION_PROFILE_CACHED_INFO_CHANGED, | 
| -                   content::NotificationService::AllSources()); | 
| -  } | 
| - | 
| -  // 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; | 
| - | 
| -  window_ = GTK_WINDOW(gtk_window_new(GTK_WINDOW_TOPLEVEL)); | 
| -  g_object_set_qdata(G_OBJECT(window_), GetBrowserWindowQuarkKey(), this); | 
| -  gtk_widget_add_events(GTK_WIDGET(window_), GDK_BUTTON_PRESS_MASK | | 
| -                                             GDK_POINTER_MOTION_MASK); | 
| - | 
| -  // Disable the resize gripper on Ubuntu. | 
| -  gtk_window_util::DisableResizeGrip(window_); | 
| - | 
| -  // Add this window to its own unique window group to allow for | 
| -  // window-to-parent modality. | 
| -  gtk_window_group_add_window(gtk_window_group_new(), window_); | 
| -  g_object_unref(gtk_window_get_group(window_)); | 
| - | 
| -  // Set up a custom WM_CLASS for some sorts of window types.  This allows | 
| -  // task switchers to distinguish between main browser windows and e.g | 
| -  // app windows. | 
| -  const CommandLine& command_line = *CommandLine::ForCurrentProcess(); | 
| -  if (browser_->is_app()) { | 
| -    std::string app_name = browser_->app_name(); | 
| -    if (app_name != DevToolsWindow::kDevToolsApp) { | 
| -      gtk_window_util::SetWindowCustomClass(window_, | 
| -          web_app::GetWMClassFromAppName(app_name)); | 
| -    } | 
| -  } else if (command_line.HasSwitch(switches::kUserDataDir)) { | 
| -    // Set the class name to e.g. "Chrome (/tmp/my-user-data)".  The | 
| -    // class name will show up in the alt-tab list in gnome-shell if | 
| -    // you're running a binary that doesn't have a matching .desktop | 
| -    // file. | 
| -    const std::string user_data_dir = | 
| -        command_line.GetSwitchValueNative(switches::kUserDataDir); | 
| -    gtk_window_util::SetWindowCustomClass(window_, | 
| -        std::string(gdk_get_program_class()) + " (" + user_data_dir + ")"); | 
| -  } | 
| - | 
| -  // For popups, we initialize widgets then set the window geometry, because | 
| -  // popups need the widgets inited before they can set the window size | 
| -  // properly. For other windows, we set the geometry first to prevent resize | 
| -  // flicker. | 
| -  if (browser_->is_type_popup()) { | 
| -    gtk_window_set_role(window_, "pop-up"); | 
| -    InitWidgets(); | 
| -    SetGeometryHints(); | 
| -  } else { | 
| -    gtk_window_set_role(window_, "browser"); | 
| -    SetGeometryHints(); | 
| -    InitWidgets(); | 
| -  } | 
| - | 
| -  ConnectAccelerators(); | 
| - | 
| -  // Set the initial background color of widgets. | 
| -  SetBackgroundColor(); | 
| -  HideUnsupportedWindowFeatures(); | 
| - | 
| -  if (UseCustomFrame()) { | 
| -    // Setting _GTK_HIDE_TITLEBAR_WHEN_MAXIMIZED tells gnome-shell to not force | 
| -    // fullscreen on the window when it matches the desktop size. | 
| -    ui::SetHideTitlebarWhenMaximizedProperty( | 
| -        ui::GetX11WindowFromGtkWidget(GTK_WIDGET(window_)), | 
| -        ui::HIDE_TITLEBAR_WHEN_MAXIMIZED); | 
| -  } | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnCustomFrameExpose(GtkWidget* widget, | 
| -                                               GdkEventExpose* event) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::OnCustomFrameExpose"); | 
| - | 
| -  // Draw the default background. | 
| -  cairo_t* cr = gdk_cairo_create(gtk_widget_get_window(widget)); | 
| -  gdk_cairo_rectangle(cr, &event->area); | 
| -  cairo_clip(cr); | 
| - | 
| -  if (UsingCustomPopupFrame()) { | 
| -    DrawPopupFrame(cr, widget, event); | 
| -  } else { | 
| -    DrawCustomFrame(cr, widget, event); | 
| -  } | 
| - | 
| -  DrawContentShadow(cr); | 
| - | 
| -  cairo_destroy(cr); | 
| - | 
| -  if (UseCustomFrame() && !IsMaximized()) | 
| -    DrawCustomFrameBorder(widget); | 
| - | 
| -  return FALSE;  // Allow subwidgets to paint. | 
| -} | 
| - | 
| -void BrowserWindowGtk::DrawCustomFrameBorder(GtkWidget* widget) { | 
| -  static NineBox* custom_frame_border = NULL; | 
| -  if (!custom_frame_border) { | 
| -    custom_frame_border = new NineBox(IDR_WINDOW_TOP_LEFT_CORNER, | 
| -                                      IDR_WINDOW_TOP_CENTER, | 
| -                                      IDR_WINDOW_TOP_RIGHT_CORNER, | 
| -                                      IDR_WINDOW_LEFT_SIDE, | 
| -                                      0, | 
| -                                      IDR_WINDOW_RIGHT_SIDE, | 
| -                                      IDR_WINDOW_BOTTOM_LEFT_CORNER, | 
| -                                      IDR_WINDOW_BOTTOM_CENTER, | 
| -                                      IDR_WINDOW_BOTTOM_RIGHT_CORNER); | 
| -  } | 
| -  custom_frame_border->RenderToWidget(widget); | 
| -} | 
| - | 
| -void BrowserWindowGtk::DrawContentShadow(cairo_t* cr) { | 
| -  // Draw the shadow above the toolbar. Tabs on the tabstrip will draw over us. | 
| -  ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | 
| -  int left_x, top_y; | 
| -  gtk_widget_translate_coordinates(toolbar_->widget(), | 
| -      GTK_WIDGET(window_), 0, 0, &left_x, | 
| -      &top_y); | 
| - | 
| -  GtkAllocation window_vbox_allocation; | 
| -  gtk_widget_get_allocation(window_vbox_, &window_vbox_allocation); | 
| -  int center_width = window_vbox_allocation.width; | 
| - | 
| -  gfx::CairoCachedSurface* top_center = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_TOP_CENTER).ToCairo(); | 
| -  gfx::CairoCachedSurface* top_right = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_TOP_RIGHT_CORNER).ToCairo(); | 
| -  gfx::CairoCachedSurface* top_left = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_TOP_LEFT_CORNER).ToCairo(); | 
| - | 
| -  int center_left_x = left_x; | 
| -  if (ShouldDrawContentDropShadow()) { | 
| -    // Don't draw over the corners. | 
| -    center_left_x += top_left->Width() - kContentShadowThickness; | 
| -    center_width -= (top_left->Width() + top_right->Width()); | 
| -    center_width += 2 * kContentShadowThickness; | 
| -  } | 
| - | 
| -  top_center->SetSource(cr, GTK_WIDGET(window_), | 
| -                        center_left_x, top_y - kContentShadowThickness); | 
| -  cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REPEAT); | 
| -  cairo_rectangle(cr, center_left_x, top_y - kContentShadowThickness, | 
| -                  center_width, top_center->Height()); | 
| -  cairo_fill(cr); | 
| - | 
| -  // Only draw the rest of the shadow if the user has the custom frame enabled | 
| -  // and the browser is not maximized. | 
| -  if (!ShouldDrawContentDropShadow()) | 
| -    return; | 
| - | 
| -  // The top left corner has a width of 3 pixels. On Windows, the last column | 
| -  // of pixels overlap the toolbar. We just crop it off on Linux.  The top | 
| -  // corners extend to the base of the toolbar (one pixel above the dividing | 
| -  // line). | 
| -  int right_x = center_left_x + center_width; | 
| -  top_left->SetSource(cr, GTK_WIDGET(window_), | 
| -      left_x - kContentShadowThickness, top_y - kContentShadowThickness); | 
| -  // The toolbar is shorter in location bar only mode so clip the image to the | 
| -  // height of the toolbar + the amount of shadow above the toolbar. | 
| -  cairo_rectangle(cr, | 
| -      left_x - kContentShadowThickness, | 
| -      top_y - kContentShadowThickness, | 
| -      top_left->Width(), | 
| -      top_left->Height()); | 
| -  cairo_fill(cr); | 
| - | 
| -  // Likewise, we crop off the left column of pixels for the top right corner. | 
| -  top_right->SetSource(cr, GTK_WIDGET(window_), | 
| -                       right_x, top_y - kContentShadowThickness); | 
| -  cairo_rectangle(cr, | 
| -      right_x, | 
| -      top_y - kContentShadowThickness, | 
| -      top_right->Width(), | 
| -      top_right->Height()); | 
| -  cairo_fill(cr); | 
| - | 
| -  // Fill in the sides.  As above, we only draw 2 of the 3 columns on Linux. | 
| -  int bottom_y; | 
| -  gtk_widget_translate_coordinates(window_vbox_, | 
| -      GTK_WIDGET(window_), | 
| -      0, window_vbox_allocation.height, | 
| -      NULL, &bottom_y); | 
| -  // |side_y| is where to start drawing the side shadows.  The top corners draw | 
| -  // the sides down to the bottom of the toolbar. | 
| -  int side_y = top_y - kContentShadowThickness + top_right->Height(); | 
| -  // |side_height| is how many pixels to draw for the side borders.  We do one | 
| -  // pixel before the bottom of the web contents because that extra pixel is | 
| -  // drawn by the bottom corners. | 
| -  int side_height = bottom_y - side_y - 1; | 
| -  if (side_height > 0) { | 
| -    gfx::CairoCachedSurface* left = | 
| -        rb.GetNativeImageNamed(IDR_CONTENT_LEFT_SIDE).ToCairo(); | 
| -    left->SetSource(cr, GTK_WIDGET(window_), | 
| -                    left_x - kContentShadowThickness, side_y); | 
| -    cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REPEAT); | 
| -    cairo_rectangle(cr, | 
| -        left_x - kContentShadowThickness, | 
| -        side_y, | 
| -        kContentShadowThickness, | 
| -        side_height); | 
| -    cairo_fill(cr); | 
| - | 
| -    gfx::CairoCachedSurface* right = | 
| -        rb.GetNativeImageNamed(IDR_CONTENT_RIGHT_SIDE).ToCairo(); | 
| -    int right_side_x = | 
| -        right_x + top_right->Width() - kContentShadowThickness - 1; | 
| -    right->SetSource(cr, GTK_WIDGET(window_), right_side_x, side_y); | 
| -    cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REPEAT); | 
| -    cairo_rectangle(cr, | 
| -        right_side_x, | 
| -        side_y, | 
| -        kContentShadowThickness, | 
| -        side_height); | 
| -    cairo_fill(cr); | 
| -  } | 
| - | 
| -  // Draw the bottom corners.  The bottom corners also draw the bottom row of | 
| -  // pixels of the side shadows. | 
| -  gfx::CairoCachedSurface* bottom_left = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_BOTTOM_LEFT_CORNER).ToCairo(); | 
| -  bottom_left->SetSource(cr, GTK_WIDGET(window_), | 
| -                         left_x - kContentShadowThickness, bottom_y - 1); | 
| -  cairo_paint(cr); | 
| - | 
| -  gfx::CairoCachedSurface* bottom_right = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_BOTTOM_RIGHT_CORNER).ToCairo(); | 
| -  bottom_right->SetSource(cr, GTK_WIDGET(window_), right_x - 1, bottom_y - 1); | 
| -  cairo_paint(cr); | 
| - | 
| -  // Finally, draw the bottom row. Since we don't overlap the contents, we clip | 
| -  // the top row of pixels. | 
| -  gfx::CairoCachedSurface* bottom = | 
| -      rb.GetNativeImageNamed(IDR_CONTENT_BOTTOM_CENTER).ToCairo(); | 
| -  bottom->SetSource(cr, GTK_WIDGET(window_), left_x + 1, bottom_y - 1); | 
| -  cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REPEAT); | 
| -  cairo_rectangle(cr, | 
| -      left_x + 1, | 
| -      bottom_y, | 
| -      window_vbox_allocation.width - 2, | 
| -      kContentShadowThickness); | 
| -  cairo_fill(cr); | 
| -} | 
| - | 
| -void BrowserWindowGtk::DrawPopupFrame(cairo_t* cr, | 
| -                                      GtkWidget* widget, | 
| -                                      GdkEventExpose* event) { | 
| -  ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | 
| - | 
| -  // Like DrawCustomFrame(), except that we use the unthemed resources to draw | 
| -  // the background. We do this because we can't rely on sane images in the | 
| -  // theme that we can draw text on. (We tried using the tab background, but | 
| -  // that has inverse saturation from what the user usually expects). | 
| -  int image_name = GetThemeFrameResource(); | 
| -  gfx::CairoCachedSurface* surface = | 
| -      rb.GetNativeImageNamed(image_name).ToCairo(); | 
| -  surface->SetSource(cr, widget, 0, GetVerticalOffset()); | 
| -  cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REFLECT); | 
| -  cairo_rectangle(cr, event->area.x, event->area.y, | 
| -                  event->area.width, event->area.height); | 
| -  cairo_fill(cr); | 
| -} | 
| - | 
| -void BrowserWindowGtk::DrawCustomFrame(cairo_t* cr, | 
| -                                       GtkWidget* widget, | 
| -                                       GdkEventExpose* event) { | 
| -  GtkThemeService* theme_provider = GtkThemeService::GetFrom( | 
| -      browser()->profile()); | 
| - | 
| -  int image_name = GetThemeFrameResource(); | 
| - | 
| -  gfx::CairoCachedSurface* surface = theme_provider->GetImageNamed( | 
| -      image_name).ToCairo(); | 
| -  if (event->area.y < surface->Height()) { | 
| -    surface->SetSource(cr, widget, 0, GetVerticalOffset()); | 
| - | 
| -    // The frame background isn't tiled vertically. | 
| -    cairo_pattern_set_extend(cairo_get_source(cr), CAIRO_EXTEND_REPEAT); | 
| -    cairo_rectangle(cr, event->area.x, event->area.y, | 
| -                    event->area.width, surface->Height() - event->area.y); | 
| -    cairo_fill(cr); | 
| -  } | 
| - | 
| -  if (theme_provider->HasCustomImage(IDR_THEME_FRAME_OVERLAY) && | 
| -      !browser()->profile()->IsOffTheRecord()) { | 
| -    gfx::CairoCachedSurface* theme_overlay = theme_provider->GetImageNamed( | 
| -        DrawFrameAsActive() ? IDR_THEME_FRAME_OVERLAY | 
| -        : IDR_THEME_FRAME_OVERLAY_INACTIVE).ToCairo(); | 
| -    theme_overlay->SetSource(cr, widget, 0, GetVerticalOffset()); | 
| -    cairo_paint(cr); | 
| -  } | 
| -} | 
| - | 
| -int BrowserWindowGtk::GetVerticalOffset() { | 
| -  return (IsMaximized() || (!UseCustomFrame())) ? | 
| -      -kCustomFrameBackgroundVerticalOffset : 0; | 
| -} | 
| - | 
| -int BrowserWindowGtk::GetThemeFrameResource() { | 
| -  bool incognito = browser()->profile()->IsOffTheRecord(); | 
| -  int image_name; | 
| -  if (DrawFrameAsActive()) { | 
| -    image_name = incognito ? IDR_THEME_FRAME_INCOGNITO : IDR_THEME_FRAME; | 
| -  } else { | 
| -    image_name = incognito ? IDR_THEME_FRAME_INCOGNITO_INACTIVE : | 
| -                 IDR_THEME_FRAME_INACTIVE; | 
| -  } | 
| - | 
| -  return image_name; | 
| -} | 
| - | 
| -void BrowserWindowGtk::Show() { | 
| -  // The Browser associated with this browser window must become the active | 
| -  // browser at the time Show() is called. This is the natural behaviour under | 
| -  // Windows, but gtk_widget_show won't show the widget (and therefore won't | 
| -  // call OnFocusIn()) until we return to the runloop. Therefore any calls to | 
| -  // chrome::FindLastActiveWithHostDesktopType will return the previous | 
| -  // browser instead if we don't explicitly set it here. | 
| -  BrowserList::SetLastActive(browser()); | 
| - | 
| -  gtk_window_present(window_); | 
| -  if (show_state_after_show_ == ui::SHOW_STATE_MAXIMIZED) { | 
| -    gtk_window_maximize(window_); | 
| -    show_state_after_show_ = ui::SHOW_STATE_NORMAL; | 
| -  } else if (show_state_after_show_ == ui::SHOW_STATE_MINIMIZED) { | 
| -    gtk_window_iconify(window_); | 
| -    show_state_after_show_ = ui::SHOW_STATE_NORMAL; | 
| -  } | 
| - | 
| -  // If we have sized the window by setting a size request for the render | 
| -  // area, then undo it so that the render view can later adjust its own | 
| -  // size. | 
| -  gtk_widget_set_size_request(devtools_floating_container_, -1, -1); | 
| - | 
| -  window_has_shown_ = true; | 
| -  browser()->OnWindowDidShow(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowInactive() { | 
| -  gtk_window_set_focus_on_map(window_, false); | 
| -  gtk_widget_show(GTK_WIDGET(window_)); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Hide() { | 
| -  // Not implemented. | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetBoundsImpl(const gfx::Rect& bounds, | 
| -                                     bool exterior, | 
| -                                     bool move) { | 
| -  gint x = static_cast<gint>(bounds.x()); | 
| -  gint y = static_cast<gint>(bounds.y()); | 
| -  gint width = static_cast<gint>(bounds.width()); | 
| -  gint height = static_cast<gint>(bounds.height()); | 
| - | 
| -  if (move) | 
| -    gtk_window_move(window_, x, y); | 
| - | 
| -  if (exterior) { | 
| -    gtk_window_util::SetWindowSize(window_, gfx::Size(width, height)); | 
| -  } else { | 
| -    gtk_widget_set_size_request(devtools_floating_container_, | 
| -                                width, height); | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetBounds(const gfx::Rect& bounds) { | 
| -  if (IsFullscreen()) | 
| -    ExitFullscreen(); | 
| -  SetBoundsImpl(bounds, true, true); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Close() { | 
| -  // We're already closing.  Do nothing. | 
| -  if (!window_) | 
| -    return; | 
| - | 
| -  if (!CanClose()) | 
| -    return; | 
| - | 
| -  // We're going to destroy the window, make sure the tab strip isn't running | 
| -  // any animations which may still reference GtkWidgets. | 
| -  tabstrip_->StopAnimation(); | 
| - | 
| -  SaveWindowPosition(); | 
| - | 
| -  if (accel_group_) { | 
| -    // Disconnecting the keys we connected to our accelerator group frees the | 
| -    // closures allocated in ConnectAccelerators. | 
| -    AcceleratorsGtk* accelerators = AcceleratorsGtk::GetInstance(); | 
| -    for (AcceleratorsGtk::const_iterator iter = accelerators->begin(); | 
| -         iter != accelerators->end(); ++iter) { | 
| -      gtk_accel_group_disconnect_key(accel_group_, | 
| -          ui::GetGdkKeyCodeForAccelerator(iter->second), | 
| -          ui::GetGdkModifierForAccelerator(iter->second)); | 
| -    } | 
| -    gtk_window_remove_accel_group(window_, accel_group_); | 
| -    g_object_unref(accel_group_); | 
| -    accel_group_ = NULL; | 
| -  } | 
| - | 
| -  // Cancel any pending callback from the window configure debounce timer. | 
| -  window_configure_debounce_timer_.Stop(); | 
| - | 
| -  // Likewise for the loading animation. | 
| -  loading_animation_timer_.Stop(); | 
| - | 
| -  GtkWidget* window = GTK_WIDGET(window_); | 
| -  // To help catch bugs in any event handlers that might get fired during the | 
| -  // destruction, set window_ to NULL before any handlers will run. | 
| -  window_ = NULL; | 
| -  // Avoid use-after-free in any code that runs after Close() and forgets to | 
| -  // check window_. | 
| -  window_container_ = NULL; | 
| -  window_vbox_ = NULL; | 
| -  render_area_vbox_ = NULL; | 
| -  render_area_floating_container_ = NULL; | 
| -  render_area_event_box_ = NULL; | 
| -  toolbar_border_ = NULL; | 
| -  devtools_floating_container_ = NULL; | 
| - | 
| -  window_has_shown_ = false; | 
| -  titlebar_->set_window(NULL); | 
| - | 
| -  // We don't want GlobalMenuBar handling any notifications or commands after | 
| -  // the window is destroyed. | 
| -  global_menu_bar_->Disable(); | 
| -  gtk_widget_destroy(window); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Activate() { | 
| -  gtk_window_present(window_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Deactivate() { | 
| -  gdk_window_lower(gtk_widget_get_window(GTK_WIDGET(window_))); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsActive() const { | 
| -  if (ui::ActiveWindowWatcherX::WMSupportsActivation()) | 
| -    return is_active_; | 
| - | 
| -  // This still works even though we don't get the activation notification. | 
| -  return window_ && gtk_window_is_active(window_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::FlashFrame(bool flash) { | 
| -  // May not be respected by all window managers. | 
| -  gtk_window_set_urgency_hint(window_, flash); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsAlwaysOnTop() const { | 
| -  return false; | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetAlwaysOnTop(bool always_on_top) { | 
| -  // Not implemented for browser windows. | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -gfx::NativeWindow BrowserWindowGtk::GetNativeWindow() { | 
| -  return window_; | 
| -} | 
| - | 
| -BrowserWindowTesting* BrowserWindowGtk::GetBrowserWindowTesting() { | 
| -  NOTIMPLEMENTED(); | 
| -  return NULL; | 
| -} | 
| - | 
| -StatusBubble* BrowserWindowGtk::GetStatusBubble() { | 
| -  return status_bubble_.get(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateTitleBar() { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::UpdateTitleBar"); | 
| -  base::string16 title = browser_->GetWindowTitleForCurrentTab(); | 
| -  gtk_window_set_title(window_, base::UTF16ToUTF8(title).c_str()); | 
| -  if (ShouldShowWindowIcon()) | 
| -    titlebar_->UpdateTitleAndIcon(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::BookmarkBarStateChanged( | 
| -    BookmarkBar::AnimateChangeType change_type) { | 
| -  MaybeShowBookmarkBar(change_type == BookmarkBar::ANIMATE_STATE_CHANGE); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateDevTools() { | 
| -  UpdateDevToolsForContents( | 
| -      browser_->tab_strip_model()->GetActiveWebContents()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateLoadingAnimations(bool should_animate) { | 
| -  if (should_animate) { | 
| -    if (!loading_animation_timer_.IsRunning()) { | 
| -      // Loads are happening, and the timer isn't running, so start it. | 
| -      loading_animation_timer_.Start(FROM_HERE, | 
| -          base::TimeDelta::FromMilliseconds(kLoadingAnimationFrameTimeMs), this, | 
| -          &BrowserWindowGtk::LoadingAnimationCallback); | 
| -    } | 
| -  } else { | 
| -    if (loading_animation_timer_.IsRunning()) { | 
| -      loading_animation_timer_.Stop(); | 
| -      // Loads are now complete, update the state if a task was scheduled. | 
| -      LoadingAnimationCallback(); | 
| -    } | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::LoadingAnimationCallback() { | 
| -  if (browser_->is_type_tabbed()) { | 
| -    // Loading animations are shown in the tab for tabbed windows.  We check the | 
| -    // browser type instead of calling IsTabStripVisible() because the latter | 
| -    // will return false for fullscreen windows, but we still need to update | 
| -    // their animations (so that when they come out of fullscreen mode they'll | 
| -    // be correct). | 
| -    tabstrip_->UpdateLoadingAnimations(); | 
| -  } else if (ShouldShowWindowIcon()) { | 
| -    // ... or in the window icon area for popups and app windows. | 
| -    WebContents* web_contents = | 
| -        browser_->tab_strip_model()->GetActiveWebContents(); | 
| -    // GetSelectedTabContents can return NULL for example under Purify when | 
| -    // the animations are running slowly and this function is called on | 
| -    // a timer through LoadingAnimationCallback. | 
| -    titlebar_->UpdateThrobber(web_contents); | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetStarredState(bool is_starred) { | 
| -  toolbar_->GetLocationBarView()->SetStarred(is_starred); | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetTranslateIconToggled(bool is_lit) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnActiveTabChanged(WebContents* old_contents, | 
| -                                          WebContents* new_contents, | 
| -                                          int index, | 
| -                                          int reason) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::ActiveTabChanged"); | 
| -  if (old_contents && !old_contents->IsBeingDestroyed()) | 
| -    old_contents->GetView()->StoreFocus(); | 
| - | 
| -  // Update various elements that are interested in knowing the current | 
| -  // WebContents. | 
| -  infobar_container_->ChangeInfoBarManager( | 
| -      InfoBarService::FromWebContents(new_contents)); | 
| -  contents_container_->SetTab(new_contents); | 
| -  UpdateDevToolsForContents(new_contents); | 
| - | 
| -  // TODO(estade): after we manage browser activation, add a check to make sure | 
| -  // we are the active browser before calling RestoreFocus(). | 
| -  if (!browser_->tab_strip_model()->closing_all()) { | 
| -    new_contents->GetView()->RestoreFocus(); | 
| -    FindTabHelper* find_tab_helper = | 
| -        FindTabHelper::FromWebContents(new_contents); | 
| -    if (find_tab_helper->find_ui_active()) | 
| -      browser_->GetFindBarController()->find_bar()->SetFocusAndSelection(); | 
| -  } | 
| - | 
| -  // Update all the UI bits. | 
| -  UpdateTitleBar(); | 
| -  MaybeShowBookmarkBar(false); | 
| -} | 
| -void BrowserWindowGtk::ZoomChangedForActiveTab(bool can_show_bubble) { | 
| -  toolbar_->GetLocationBarView()->ZoomChangedForActiveTab( | 
| -      can_show_bubble && !toolbar_->IsWrenchMenuShowing()); | 
| -} | 
| - | 
| -gfx::Rect BrowserWindowGtk::GetRestoredBounds() const { | 
| -  return restored_bounds_; | 
| -} | 
| - | 
| -ui::WindowShowState BrowserWindowGtk::GetRestoredState() const { | 
| -  if (IsMaximized()) | 
| -    return ui::SHOW_STATE_MAXIMIZED; | 
| -  if (IsMinimized()) | 
| -    return ui::SHOW_STATE_MINIMIZED; | 
| -  return ui::SHOW_STATE_NORMAL; | 
| -} | 
| - | 
| -gfx::Rect BrowserWindowGtk::GetBounds() const { | 
| -  return bounds_; | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsMaximized() const { | 
| -  return (state_ & GDK_WINDOW_STATE_MAXIMIZED); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsMinimized() const { | 
| -  return (state_ & GDK_WINDOW_STATE_ICONIFIED); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Maximize() { | 
| -  gtk_window_maximize(window_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Minimize() { | 
| -  gtk_window_iconify(window_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Restore() { | 
| -  if (IsMaximized()) | 
| -    UnMaximize(); | 
| -  else if (IsMinimized()) | 
| -    gtk_window_deiconify(window_); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::ShouldDrawContentDropShadow() const { | 
| -  return !IsMaximized() && UseCustomFrame(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::EnterFullscreen( | 
| -    const GURL& url, FullscreenExitBubbleType type) { | 
| -  if (IsFullscreen()) | 
| -    return;  // Nothing to do. | 
| -  is_fullscreen_ = true; | 
| - | 
| -  // gtk_window_(un)fullscreen asks the window manager to toggle the EWMH | 
| -  // for fullscreen windows.  Not all window managers support this. | 
| -  gtk_window_fullscreen(window_); | 
| - | 
| -  browser_->WindowFullscreenStateChanged(); | 
| -  UpdateCustomFrame(); | 
| -  toolbar_->Hide(); | 
| -  tabstrip_->Hide(); | 
| -  if (bookmark_bar_.get()) | 
| -    gtk_widget_hide(bookmark_bar_->widget()); | 
| -  if (!chrome::IsRunningInAppMode()) { | 
| -    UpdateFullscreenExitBubbleContent(url, type); | 
| -  } | 
| -  gtk_widget_hide(titlebar_widget()); | 
| -  gtk_widget_hide(toolbar_border_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateFullscreenExitBubbleContent( | 
| -    const GURL& url, FullscreenExitBubbleType bubble_type) { | 
| -  if (!window_) { | 
| -    // Don't create a fullscreen bubble for a closing window. | 
| -    return; | 
| -  } else if (bubble_type == FEB_TYPE_NONE) { | 
| -   fullscreen_exit_bubble_.reset(); | 
| -  } else if (fullscreen_exit_bubble_.get()) { | 
| -   fullscreen_exit_bubble_->UpdateContent(url, bubble_type); | 
| -  } else { | 
| -    fullscreen_exit_bubble_.reset(new FullscreenExitBubbleGtk( | 
| -        GTK_FLOATING_CONTAINER(render_area_floating_container_), | 
| -        browser(), | 
| -        url, | 
| -        bubble_type)); | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::ExitFullscreen() { | 
| -  if (!IsFullscreen()) | 
| -    return;  // Nothing to do. | 
| -  is_fullscreen_ = false; | 
| - | 
| -  // Work around a bug where if we try to unfullscreen, metacity immediately | 
| -  // fullscreens us again.  This is a little flickery and not necessary if | 
| -  // there's a gnome-panel, but it's not easy to detect whether there's a | 
| -  // panel or not. | 
| -  bool unmaximize_before_unfullscreen = IsMaximized() && | 
| -      ui::GuessWindowManager() == ui::WM_METACITY; | 
| -  if (unmaximize_before_unfullscreen) | 
| -    UnMaximize(); | 
| - | 
| -  gtk_window_unfullscreen(window_); | 
| - | 
| -  if (unmaximize_before_unfullscreen) | 
| -    gtk_window_maximize(window_); | 
| - | 
| -  browser_->WindowFullscreenStateChanged(); | 
| -  gtk_widget_show(titlebar_widget()); | 
| -  UpdateFullscreenExitBubbleContent(GURL(), FEB_TYPE_NONE); | 
| -  UpdateCustomFrame(); | 
| -  ShowSupportedWindowFeatures(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::ShouldHideUIForFullscreen() const { | 
| -  return IsFullscreen(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsFullscreen() const { | 
| -  return is_fullscreen_; | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsFullscreenBubbleVisible() const { | 
| -  return fullscreen_exit_bubble_ != NULL; | 
| -} | 
| - | 
| -LocationBar* BrowserWindowGtk::GetLocationBar() const { | 
| -  return toolbar_->GetLocationBar(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetFocusToLocationBar(bool select_all) { | 
| -  if (!IsFullscreen()) | 
| -    GetLocationBar()->FocusLocation(select_all); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateReloadStopState(bool is_loading, bool force) { | 
| -  toolbar_->GetReloadButton()->ChangeMode( | 
| -      is_loading ? ReloadButtonGtk::MODE_STOP : ReloadButtonGtk::MODE_RELOAD, | 
| -      force); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateToolbar(content::WebContents* contents) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::UpdateToolbar"); | 
| -  toolbar_->UpdateWebContents(contents); | 
| -} | 
| - | 
| -void BrowserWindowGtk::FocusToolbar() { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::FocusAppMenu() { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::FocusBookmarksToolbar() { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::FocusInfobars() { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::RotatePaneFocus(bool forwards) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsBookmarkBarVisible() const { | 
| -  return browser_->SupportsWindowFeature(Browser::FEATURE_BOOKMARKBAR) && | 
| -         bookmark_bar_.get() && | 
| -         browser_->profile()->GetPrefs()->GetBoolean(prefs::kShowBookmarkBar); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsBookmarkBarAnimating() const { | 
| -  if (IsBookmarkBarSupported() && bookmark_bar_->IsAnimating()) | 
| -    return true; | 
| -  return false; | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsTabStripEditable() const { | 
| -  return !tabstrip()->IsDragSessionActive() && | 
| -      !tabstrip()->IsActiveDropTarget(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsToolbarVisible() const { | 
| -  return IsToolbarSupported(); | 
| -} | 
| - | 
| -gfx::Rect BrowserWindowGtk::GetRootWindowResizerRect() const { | 
| -  return gfx::Rect(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ConfirmAddSearchProvider(TemplateURL* template_url, | 
| -                                                Profile* profile) { | 
| -  new EditSearchEngineDialog(window_, template_url, NULL, profile); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowUpdateChromeDialog() { | 
| -  UpdateRecommendedDialog::Show(window_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowBookmarkBubble(const GURL& url, | 
| -                                          bool already_bookmarked) { | 
| -  toolbar_->GetLocationBarView()->ShowStarBubble(url, !already_bookmarked); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowBookmarkAppBubble( | 
| -    const WebApplicationInfo& web_app_info, | 
| -    const std::string& extension_id) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowTranslateBubble(content::WebContents* contents, | 
| -                                           translate::TranslateStep step, | 
| -                                           TranslateErrors::Type error_type) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -#if defined(ENABLE_ONE_CLICK_SIGNIN) | 
| -void BrowserWindowGtk::ShowOneClickSigninBubble( | 
| -    OneClickSigninBubbleType type, | 
| -    const base::string16& email, | 
| -    const base::string16& error_message, | 
| -    const StartSyncCallback& start_sync_callback) { | 
| - | 
| -  new OneClickSigninBubbleGtk(this, type, email, | 
| -                              error_message, start_sync_callback); | 
| -} | 
| -#endif | 
| - | 
| -bool BrowserWindowGtk::IsDownloadShelfVisible() const { | 
| -  return download_shelf_.get() && download_shelf_->IsShowing(); | 
| -} | 
| - | 
| -DownloadShelf* BrowserWindowGtk::GetDownloadShelf() { | 
| -  if (!download_shelf_.get()) | 
| -    download_shelf_.reset(new DownloadShelfGtk(browser_.get(), | 
| -                                               render_area_vbox_)); | 
| -  return download_shelf_.get(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UserChangedTheme() { | 
| -  SetBackgroundColor(); | 
| -  InvalidateWindow(); | 
| -  UpdateWindowShape(bounds_.width(), bounds_.height()); | 
| -} | 
| - | 
| -int BrowserWindowGtk::GetExtraRenderViewHeight() const { | 
| -  int sum = infobar_container_->TotalHeightOfAnimatingBars(); | 
| -  if (IsBookmarkBarSupported() && bookmark_bar_->IsAnimating()) | 
| -    sum += bookmark_bar_->GetHeight(); | 
| -  if (download_shelf_.get() && download_shelf_->IsClosing()) | 
| -    sum += download_shelf_->GetHeight(); | 
| -  return sum; | 
| -} | 
| - | 
| -void BrowserWindowGtk::WebContentsFocused(WebContents* contents) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowWebsiteSettings( | 
| -    Profile* profile, | 
| -    content::WebContents* web_contents, | 
| -    const GURL& url, | 
| -    const content::SSLStatus& ssl) { | 
| -  WebsiteSettingsPopupGtk::Show(GetNativeWindow(), profile, web_contents, url, | 
| -                                ssl); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowAppMenu() { | 
| -  toolbar_->ShowAppMenu(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::PreHandleKeyboardEvent( | 
| -    const NativeWebKeyboardEvent& event, bool* is_keyboard_shortcut) { | 
| -  GdkEventKey* os_event = &event.os_event->key; | 
| - | 
| -  if (!os_event || event.type != blink::WebInputEvent::RawKeyDown) | 
| -    return false; | 
| - | 
| -  if (ExtensionKeybindingRegistryGtk::shortcut_handling_suspended()) | 
| -    return false; | 
| - | 
| -  // We first find out the browser command associated to the |event|. | 
| -  // Then if the command is a reserved one, and should be processed immediately | 
| -  // according to the |event|, the command will be executed immediately. | 
| -  // Otherwise we just set |*is_keyboard_shortcut| properly and return false. | 
| - | 
| -  // First check if it's a custom accelerator. | 
| -  int id = GetCustomCommandId(os_event); | 
| - | 
| -  // Then check if it's a predefined accelerator bound to the window. | 
| -  if (id == -1) { | 
| -    // This piece of code is based on the fact that calling | 
| -    // gtk_window_activate_key() method against |window_| may only trigger a | 
| -    // browser command execution, by matching a global accelerator | 
| -    // defined in above |kAcceleratorMap|. | 
| -    // | 
| -    // Here we need to retrieve the command id (if any) associated to the | 
| -    // keyboard event. Instead of looking up the command id in above | 
| -    // |kAcceleratorMap| table by ourselves, we block the command execution of | 
| -    // the |browser_| object then send the keyboard event to the |window_| by | 
| -    // calling gtk_window_activate_key() method, as if we are activating an | 
| -    // accelerator key. Then we can retrieve the command id from the | 
| -    // |browser_| object. | 
| -    // | 
| -    // Pros of this approach: | 
| -    // 1. We don't need to care about keyboard layout problem, as | 
| -    //    gtk_window_activate_key() method handles it for us. | 
| -    // | 
| -    // Cons: | 
| -    // 1. The logic is a little complicated. | 
| -    // 2. We should be careful not to introduce any accelerators that trigger | 
| -    //    customized code instead of browser commands. | 
| -    bool original_block_command_state = | 
| -        browser_->command_controller()->block_command_execution(); | 
| -    browser_->command_controller()->SetBlockCommandExecution(true); | 
| -    gtk_window_activate_key(window_, os_event); | 
| -    // We don't need to care about the WindowOpenDisposition value, | 
| -    // because all commands executed in this path use the default value. | 
| -    id = browser_->command_controller()->GetLastBlockedCommand(NULL); | 
| -    browser_->command_controller()->SetBlockCommandExecution( | 
| -        original_block_command_state); | 
| -  } | 
| - | 
| -  if (id == -1) | 
| -    return false; | 
| - | 
| -  // Executing the command may cause |this| object to be destroyed. | 
| -  if (browser_->command_controller()->IsReservedCommandOrKey(id, event) && | 
| -      !event.match_edit_command) { | 
| -    return chrome::ExecuteCommand(browser_.get(), id); | 
| -  } | 
| - | 
| -  // The |event| is a keyboard shortcut. | 
| -  DCHECK(is_keyboard_shortcut != NULL); | 
| -  *is_keyboard_shortcut = true; | 
| - | 
| -  return false; | 
| -} | 
| - | 
| -void BrowserWindowGtk::HandleKeyboardEvent( | 
| -    const NativeWebKeyboardEvent& event) { | 
| -  GdkEventKey* os_event = &event.os_event->key; | 
| - | 
| -  if (!os_event || event.type != blink::WebInputEvent::RawKeyDown) | 
| -    return; | 
| - | 
| -  // Handles a key event in following sequence: | 
| -  // 1. Our special key accelerators, such as ctrl-tab, etc. | 
| -  // 2. Gtk accelerators. | 
| -  // This sequence matches the default key press handler of GtkWindow. | 
| -  // | 
| -  // It's not necessary to care about the keyboard layout, as | 
| -  // gtk_window_activate_key() takes care of it automatically. | 
| -  int id = GetCustomCommandId(os_event); | 
| -  if (id != -1) | 
| -    chrome::ExecuteCommand(browser_.get(), id); | 
| -  else | 
| -    gtk_window_activate_key(window_, os_event); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Cut() { | 
| -  gtk_window_util::DoCut( | 
| -      window_, browser_->tab_strip_model()->GetActiveWebContents()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Copy() { | 
| -  gtk_window_util::DoCopy( | 
| -      window_, browser_->tab_strip_model()->GetActiveWebContents()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Paste() { | 
| -  gtk_window_util::DoPaste( | 
| -      window_, browser_->tab_strip_model()->GetActiveWebContents()); | 
| -} | 
| - | 
| -WindowOpenDisposition BrowserWindowGtk::GetDispositionForPopupBounds( | 
| -    const gfx::Rect& bounds) { | 
| -  return NEW_POPUP; | 
| -} | 
| - | 
| -FindBar* BrowserWindowGtk::CreateFindBar() { | 
| -  return new FindBarGtk(this); | 
| -} | 
| - | 
| -WebContentsModalDialogHost* BrowserWindowGtk::GetWebContentsModalDialogHost() { | 
| -  return NULL; | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowAvatarBubble(WebContents* web_contents, | 
| -                                        const gfx::Rect& rect) { | 
| -  GtkWidget* widget = web_contents->GetView()->GetContentNativeView(); | 
| -  new AvatarMenuBubbleGtk(browser_.get(), widget, BubbleGtk::ANCHOR_TOP_RIGHT, | 
| -                          &rect); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowAvatarBubbleFromAvatarButton(AvatarBubbleMode mode) { | 
| -  if (titlebar_->avatar_button()) | 
| -    titlebar_->avatar_button()->ShowAvatarBubble(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowPasswordGenerationBubble( | 
| -    const gfx::Rect& rect, | 
| -    const autofill::PasswordForm& form, | 
| -    autofill::PasswordGenerator* password_generator) { | 
| -  WebContents* web_contents = | 
| -      browser_->tab_strip_model()->GetActiveWebContents(); | 
| -  if (!web_contents || !web_contents->GetView()->GetContentNativeView()) { | 
| -    return; | 
| -  } | 
| - | 
| -  new PasswordGenerationBubbleGtk(rect, form, web_contents, password_generator); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ConfirmBrowserCloseWithPendingDownloads( | 
| -    int download_count, | 
| -    Browser::DownloadClosePreventionType dialog_type, | 
| -    bool app_modal, | 
| -    const base::Callback<void(bool)>& callback) { | 
| -  DownloadInProgressDialogGtk::Show( | 
| -      GetNativeWindow(), download_count, dialog_type, app_modal, callback); | 
| -} | 
| - | 
| -int | 
| -BrowserWindowGtk::GetRenderViewHeightInsetWithDetachedBookmarkBar() { | 
| -  if (!bookmark_bar_.get() || | 
| -      browser_->bookmark_bar_state() != BookmarkBar::DETACHED) { | 
| -    return 0; | 
| -  } | 
| -  return bookmark_bar_->max_height(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ExecuteExtensionCommand( | 
| -    const extensions::Extension* extension, | 
| -    const extensions::Command& command) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowPageActionPopup( | 
| -    const extensions::Extension* extension) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowBrowserActionPopup( | 
| -    const extensions::Extension* extension) { | 
| -  NOTIMPLEMENTED(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::Observe(int type, | 
| -                               const content::NotificationSource& source, | 
| -                               const content::NotificationDetails& details) { | 
| -  DCHECK_EQ(chrome::NOTIFICATION_PROFILE_CACHED_INFO_CHANGED, type); | 
| -  // The profile avatar icon may have changed. | 
| -  gtk_util::SetWindowIcon(window_, browser_->profile()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::TabDetachedAt(WebContents* contents, int index) { | 
| -  // We use index here rather than comparing |contents| because by this time | 
| -  // the model has already removed |contents| from its list, so | 
| -  // browser_->tab_strip_model()->GetActiveWebContents() will return NULL or | 
| -  // something else. | 
| -  if (index == browser_->tab_strip_model()->active_index()) { | 
| -    infobar_container_->ChangeInfoBarManager(NULL); | 
| -    UpdateDevToolsForContents(NULL); | 
| -  } | 
| -  contents_container_->DetachTab(contents); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ActiveWindowChanged(GdkWindow* active_window) { | 
| -  // Do nothing if we're in the process of closing the browser window. | 
| -  if (!window_) | 
| -    return; | 
| - | 
| -  bool is_active = gtk_widget_get_window(GTK_WIDGET(window_)) == active_window; | 
| -  bool changed = (is_active != is_active_); | 
| - | 
| -  if (is_active && changed) { | 
| -    // If there's an app modal dialog (e.g., JS alert), try to redirect | 
| -    // the user's attention to the window owning the dialog. | 
| -    if (AppModalDialogQueue::GetInstance()->HasActiveDialog()) { | 
| -      AppModalDialogQueue::GetInstance()->ActivateModalDialog(); | 
| -      return; | 
| -    } | 
| -  } | 
| - | 
| -  is_active_ = is_active; | 
| -  if (changed) { | 
| -    SetBackgroundColor(); | 
| -    InvalidateWindow(); | 
| -    // For some reason, the above two calls cause the window shape to be | 
| -    // lost so reset it. | 
| -    UpdateWindowShape(bounds_.width(), bounds_.height()); | 
| -  } | 
| -} | 
| - | 
| -SkColor BrowserWindowGtk::GetInfoBarSeparatorColor() const { | 
| -  GtkThemeService* theme_service = GtkThemeService::GetFrom( | 
| -      browser()->profile()); | 
| -  return gfx::GdkColorToSkColor(theme_service->GetBorderColor()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::InfoBarContainerStateChanged(bool is_animating) { | 
| -  InvalidateInfoBarBits(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::DrawInfoBarArrows(int* x) const { | 
| -  if (x) { | 
| -    // This is a views specific call that made its way into the interface. We | 
| -    // go through GetXPositionOfLocationIcon() since we need widget relativity. | 
| -    *x = 0; | 
| -    NOTREACHED(); | 
| -  } | 
| -  return true; | 
| -} | 
| - | 
| -extensions::ActiveTabPermissionGranter* | 
| -    BrowserWindowGtk::GetActiveTabPermissionGranter() { | 
| -  WebContents* tab = GetDisplayedTab(); | 
| -  if (!tab) | 
| -    return NULL; | 
| -  return extensions::TabHelper::FromWebContents(tab)-> | 
| -      active_tab_permission_granter(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::DestroyBrowser() { | 
| -  browser_.reset(); | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnConfigure(GtkWidget* widget, | 
| -                                       GdkEventConfigure* event) { | 
| -  gfx::Rect bounds(event->x, event->y, event->width, event->height); | 
| - | 
| -  // When the window moves, we'll get multiple configure-event signals. We can | 
| -  // also get events when the bounds haven't changed, but the window's stacking | 
| -  // has, which we aren't interested in. http://crbug.com/70125 | 
| -  if (bounds == configure_bounds_) | 
| -    return FALSE; | 
| - | 
| -  GetLocationBar()->GetOmniboxView()->CloseOmniboxPopup(); | 
| - | 
| -  WebContents* tab = GetDisplayedTab(); | 
| -  if (tab) | 
| -    tab->GetRenderViewHost()->NotifyMoveOrResizeStarted(); | 
| - | 
| -  if (bounds_.size() != bounds.size()) | 
| -    UpdateWindowShape(bounds.width(), bounds.height()); | 
| - | 
| -  // We update |bounds_| but not |restored_bounds_| here.  The latter needs | 
| -  // to be updated conditionally when the window is non-maximized and non- | 
| -  // fullscreen, but whether those state updates have been processed yet is | 
| -  // window-manager specific.  We update |restored_bounds_| in the debounced | 
| -  // handler below, after the window state has been updated. | 
| -  bounds_ = bounds; | 
| -  configure_bounds_ = bounds; | 
| - | 
| -  // The GdkEventConfigure* we get here doesn't have quite the right | 
| -  // coordinates though (they're relative to the drawable window area, rather | 
| -  // than any window manager decorations, if enabled), so we need to call | 
| -  // gtk_window_get_position() to get the right values. (Otherwise session | 
| -  // restore, if enabled, will restore windows to incorrect positions.) That's | 
| -  // a round trip to the X server though, so we set a debounce timer and only | 
| -  // call it (in OnDebouncedBoundsChanged() below) after we haven't seen a | 
| -  // reconfigure event in a short while. | 
| -  // We don't use Reset() because the timer may not yet be running. | 
| -  // (In that case Stop() is a no-op.) | 
| -  window_configure_debounce_timer_.Stop(); | 
| -  window_configure_debounce_timer_.Start(FROM_HERE, | 
| -      base::TimeDelta::FromMilliseconds(kDebounceTimeoutMilliseconds), this, | 
| -      &BrowserWindowGtk::OnDebouncedBoundsChanged); | 
| - | 
| -  return FALSE; | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnDebouncedBoundsChanged() { | 
| -  gtk_window_util::UpdateWindowPosition(this, &bounds_, &restored_bounds_); | 
| -  SaveWindowPosition(); | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnWindowState(GtkWidget* sender, | 
| -                                         GdkEventWindowState* event) { | 
| -  state_ = event->new_window_state; | 
| - | 
| -  if (event->changed_mask & GDK_WINDOW_STATE_MAXIMIZED) { | 
| -    content::NotificationService::current()->Notify( | 
| -        chrome::NOTIFICATION_BROWSER_WINDOW_MAXIMIZED, | 
| -        content::Source<BrowserWindow>(this), | 
| -        content::NotificationService::NoDetails()); | 
| -  } | 
| - | 
| -  titlebar_->UpdateCustomFrame(UseCustomFrame() && !IsFullscreen()); | 
| -  UpdateWindowShape(bounds_.width(), bounds_.height()); | 
| -  SaveWindowPosition(); | 
| -  return FALSE; | 
| -} | 
| - | 
| -// Callback for the delete event.  This event is fired when the user tries to | 
| -// close the window (e.g., clicking on the X in the window manager title bar). | 
| -gboolean BrowserWindowGtk::OnMainWindowDeleteEvent(GtkWidget* widget, | 
| -                                                   GdkEvent* event) { | 
| -  Close(); | 
| - | 
| -  // Return true to prevent the gtk window from being destroyed.  Close will | 
| -  // destroy it for us. | 
| -  return TRUE; | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnMainWindowDestroy(GtkWidget* widget) { | 
| -  // Make sure we destroy this object while the main window is still valid. | 
| -  extension_keybinding_registry_.reset(); | 
| - | 
| -  // BUG 8712. When we gtk_widget_destroy() in Close(), this will emit the | 
| -  // signal right away, and we will be here (while Close() is still in the | 
| -  // call stack).  In order to not reenter Close(), and to also follow the | 
| -  // expectations of BrowserList, we should run the BrowserWindowGtk destructor | 
| -  // not now, but after the run loop goes back to process messages.  Otherwise | 
| -  // we will remove ourself from BrowserList while it's being iterated. | 
| -  // Additionally, now that we know the window is gone, we need to make sure to | 
| -  // set window_ to NULL, otherwise we will try to close the window again when | 
| -  // we call Close() in the destructor. | 
| -  // | 
| -  // We don't want to use DeleteSoon() here since it won't work on a nested pump | 
| -  // (like in UI tests). | 
| -  base::MessageLoop::current()->PostTask( | 
| -      FROM_HERE, base::Bind(&base::DeletePointer<BrowserWindowGtk>, this)); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UnMaximize() { | 
| -  gtk_window_util::UnMaximize(window_, bounds_, restored_bounds_); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::CanClose() const { | 
| -  // You cannot close a frame for which there is an active originating drag | 
| -  // session. | 
| -  if (tabstrip_->IsDragSessionActive()) | 
| -    return false; | 
| - | 
| -  // Give beforeunload handlers the chance to cancel the close before we hide | 
| -  // the window below. | 
| -  if (!browser_->ShouldCloseWindow()) | 
| -    return false; | 
| - | 
| -  bool fast_tab_closing_enabled = | 
| -      CommandLine::ForCurrentProcess()->HasSwitch(switches::kEnableFastUnload); | 
| - | 
| -  if (!browser_->tab_strip_model()->empty()) { | 
| -    // Tab strip isn't empty.  Hide the window (so it appears to have closed | 
| -    // immediately) and close all the tabs, allowing the renderers to shut | 
| -    // down. When the tab strip is empty we'll be called back again. | 
| -    gtk_widget_hide(GTK_WIDGET(window_)); | 
| -    browser_->OnWindowClosing(); | 
| - | 
| -    if (fast_tab_closing_enabled) | 
| -      browser_->tab_strip_model()->CloseAllTabs(); | 
| -    return false; | 
| -  } else if (fast_tab_closing_enabled && | 
| -      !browser_->HasCompletedUnloadProcessing()) { | 
| -    // The browser needs to finish running unload handlers. | 
| -    // Hide the window (so it appears to have closed immediately), and | 
| -    // the browser will call us back again when it is ready to close. | 
| -    gtk_widget_hide(GTK_WIDGET(window_)); | 
| -    return false; | 
| -  } | 
| - | 
| -  // Empty TabStripModel, it's now safe to allow the Window to be closed. | 
| -  content::NotificationService::current()->Notify( | 
| -      chrome::NOTIFICATION_WINDOW_CLOSED, | 
| -      content::Source<GtkWindow>(window_), | 
| -      content::NotificationService::NoDetails()); | 
| -  return true; | 
| -} | 
| - | 
| -bool BrowserWindowGtk::ShouldShowWindowIcon() const { | 
| -  return browser_->SupportsWindowFeature(Browser::FEATURE_TITLEBAR); | 
| -} | 
| - | 
| -void BrowserWindowGtk::AddFindBar(FindBarGtk* findbar) { | 
| -  gtk_floating_container_add_floating( | 
| -      GTK_FLOATING_CONTAINER(render_area_floating_container_), | 
| -      findbar->widget()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ResetCustomFrameCursor() { | 
| -  if (!frame_cursor_) | 
| -    return; | 
| - | 
| -  frame_cursor_ = NULL; | 
| -  gdk_window_set_cursor(gtk_widget_get_window(GTK_WIDGET(window_)), NULL); | 
| -} | 
| - | 
| -// static | 
| -BrowserWindowGtk* BrowserWindowGtk::GetBrowserWindowForNativeWindow( | 
| -    gfx::NativeWindow window) { | 
| -  if (window) { | 
| -    return static_cast<BrowserWindowGtk*>( | 
| -        g_object_get_qdata(G_OBJECT(window), GetBrowserWindowQuarkKey())); | 
| -  } | 
| - | 
| -  return NULL; | 
| -} | 
| - | 
| -// static | 
| -GtkWindow* BrowserWindowGtk::GetBrowserWindowForXID(XID xid) { | 
| -  GtkWindow* window = ui::GetGtkWindowFromX11Window(xid); | 
| -  // Use GetBrowserWindowForNativeWindow() to verify the GtkWindow we found | 
| -  // is actually a browser window (and not e.g. a dialog). | 
| -  if (!GetBrowserWindowForNativeWindow(window)) | 
| -    return NULL; | 
| -  return window; | 
| -} | 
| - | 
| -GtkWidget* BrowserWindowGtk::titlebar_widget() const { | 
| -  return titlebar_->widget(); | 
| -} | 
| - | 
| -// static | 
| -void BrowserWindowGtk::RegisterProfilePrefs( | 
| -    user_prefs::PrefRegistrySyncable* registry) { | 
| -  bool custom_frame_default = false; | 
| -  // Avoid checking the window manager if we're not connected to an X server (as | 
| -  // is the case in Valgrind tests). | 
| -  if (ui::XDisplayExists()) | 
| -    custom_frame_default = ui::GetCustomFramePrefDefault(); | 
| - | 
| -  registry->RegisterBooleanPref( | 
| -      prefs::kUseCustomChromeFrame, | 
| -      custom_frame_default, | 
| -      user_prefs::PrefRegistrySyncable::UNSYNCABLE_PREF); | 
| -} | 
| - | 
| -WebContents* BrowserWindowGtk::GetDisplayedTab() { | 
| -  return contents_container_->tab(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::QueueToolbarRedraw() { | 
| -  gtk_widget_queue_draw(toolbar_->widget()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetGeometryHints() { | 
| -  // If we call gtk_window_maximize followed by gtk_window_present, compiz gets | 
| -  // confused and maximizes the window, but doesn't set the | 
| -  // GDK_WINDOW_STATE_MAXIMIZED bit.  So instead, we keep track of whether to | 
| -  // maximize and call it after gtk_window_present. | 
| -  gfx::Rect bounds; | 
| -  chrome::GetSavedWindowBoundsAndShowState(browser_.get(), | 
| -                                           &bounds, | 
| -                                           &show_state_after_show_); | 
| -  // We don't blindly call SetBounds here: that sets a forced position | 
| -  // on the window and we intentionally *don't* do that for normal | 
| -  // windows.  Most programs do not restore their window position on | 
| -  // Linux, instead letting the window manager choose a position. | 
| -  // | 
| -  // However, in cases like dropping a tab where the bounds are | 
| -  // specifically set, we do want to position explicitly.  We also | 
| -  // force the position as part of session restore, as applications | 
| -  // that restore other, similar state (for instance GIMP, audacity, | 
| -  // pidgin, dia, and gkrellm) do tend to restore their positions. | 
| -  // | 
| -  // For popup windows, we assume that if x == y == 0, the opening page | 
| -  // did not specify a position.  Let the WM position the popup instead. | 
| -  bool is_popup = browser_->is_type_popup(); | 
| -  bool popup_without_position = is_popup && | 
| -      bounds.x() == 0 && bounds.y() == 0; | 
| -  bool move = browser_->bounds_overridden() && !popup_without_position; | 
| -  SetBoundsImpl(bounds, !is_popup, move); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ConnectHandlersToSignals() { | 
| -  g_signal_connect(window_, "delete-event", | 
| -                   G_CALLBACK(OnMainWindowDeleteEventThunk), this); | 
| -  g_signal_connect(window_, "destroy", | 
| -                   G_CALLBACK(OnMainWindowDestroyThunk), this); | 
| -  g_signal_connect(window_, "configure-event", | 
| -                   G_CALLBACK(OnConfigureThunk), this); | 
| -  g_signal_connect(window_, "window-state-event", | 
| -                   G_CALLBACK(OnWindowStateThunk), this); | 
| -  g_signal_connect(window_, "key-press-event", | 
| -                   G_CALLBACK(OnKeyPressThunk), this); | 
| -  g_signal_connect(window_, "motion-notify-event", | 
| -                   G_CALLBACK(OnMouseMoveEventThunk), this); | 
| -  g_signal_connect(window_, "button-press-event", | 
| -                   G_CALLBACK(OnButtonPressEventThunk), this); | 
| -  g_signal_connect(window_, "focus-in-event", | 
| -                   G_CALLBACK(OnFocusInThunk), this); | 
| -  g_signal_connect(window_, "focus-out-event", | 
| -                   G_CALLBACK(OnFocusOutThunk), this); | 
| -} | 
| - | 
| -void BrowserWindowGtk::InitWidgets() { | 
| -  ConnectHandlersToSignals(); | 
| - | 
| -  bounds_ = configure_bounds_ = restored_bounds_ = | 
| -      GetInitialWindowBounds(window_); | 
| - | 
| -  // This vbox encompasses all of the widgets within the browser.  This is | 
| -  // everything except the custom frame border. | 
| -  window_vbox_ = gtk_vbox_new(FALSE, 0); | 
| -  gtk_widget_show(window_vbox_); | 
| - | 
| -  // We hold an always hidden GtkMenuBar inside our browser window simply to | 
| -  // fool the Unity desktop, which will mirror the contents of the first | 
| -  // GtkMenuBar it sees into the global menu bar. (It doesn't seem to check the | 
| -  // visibility of the GtkMenuBar, so we can just permanently hide it.) | 
| -  global_menu_bar_.reset(new GlobalMenuBar(browser_.get())); | 
| -  gtk_container_add(GTK_CONTAINER(window_vbox_), global_menu_bar_->widget()); | 
| - | 
| -  // The window container draws the custom browser frame. | 
| -  window_container_ = gtk_alignment_new(0.0, 0.0, 1.0, 1.0); | 
| -  gtk_widget_set_name(window_container_, "chrome-custom-frame-border"); | 
| -  gtk_widget_set_app_paintable(window_container_, TRUE); | 
| -  gtk_widget_set_double_buffered(window_container_, FALSE); | 
| -  gtk_widget_set_redraw_on_allocate(window_container_, TRUE); | 
| -  g_signal_connect(window_container_, "expose-event", | 
| -                   G_CALLBACK(OnCustomFrameExposeThunk), this); | 
| -  gtk_container_add(GTK_CONTAINER(window_container_), window_vbox_); | 
| - | 
| -  tabstrip_.reset(new TabStripGtk(browser_->tab_strip_model(), this)); | 
| -  tabstrip_->Init(); | 
| - | 
| -  // Build the titlebar (tabstrip + header space + min/max/close buttons). | 
| -  titlebar_.reset(new BrowserTitlebar(this, window_)); | 
| -  titlebar_->Init(); | 
| - | 
| -  // Insert the tabstrip into the window. | 
| -  gtk_box_pack_start(GTK_BOX(window_vbox_), titlebar_->widget(), FALSE, FALSE, | 
| -                     0); | 
| - | 
| -  toolbar_.reset(new BrowserToolbarGtk(browser_.get(), this)); | 
| -  toolbar_->Init(window_); | 
| -  gtk_box_pack_start(GTK_BOX(window_vbox_), toolbar_->widget(), | 
| -                     FALSE, FALSE, 0); | 
| -  g_signal_connect_after(toolbar_->widget(), "expose-event", | 
| -                         G_CALLBACK(OnExposeDrawInfobarBitsThunk), this); | 
| -  // This vbox surrounds the render area: find bar, info bars and render view. | 
| -  // The reason is that this area as a whole needs to be grouped in its own | 
| -  // GdkWindow hierarchy so that animations originating inside it (infobar, | 
| -  // download shelf, find bar) are all clipped to that area. This is why | 
| -  // |render_area_vbox_| is packed in |render_area_event_box_|. | 
| -  render_area_vbox_ = gtk_vbox_new(FALSE, 0); | 
| -  gtk_widget_set_name(render_area_vbox_, "chrome-render-area-vbox"); | 
| -  render_area_floating_container_ = gtk_floating_container_new(); | 
| -  gtk_container_add(GTK_CONTAINER(render_area_floating_container_), | 
| -                    render_area_vbox_); | 
| - | 
| -  GtkWidget* location_icon = toolbar_->GetLocationBarView()-> | 
| -      location_icon_widget(); | 
| -  g_signal_connect(location_icon, "size-allocate", | 
| -                   G_CALLBACK(OnLocationIconSizeAllocateThunk), this); | 
| -  g_signal_connect_after(location_icon, "expose-event", | 
| -                         G_CALLBACK(OnExposeDrawInfobarBitsThunk), this); | 
| - | 
| -  toolbar_border_ = gtk_event_box_new(); | 
| -  gtk_box_pack_start(GTK_BOX(render_area_vbox_), | 
| -                     toolbar_border_, FALSE, FALSE, 0); | 
| -  gtk_widget_set_size_request(toolbar_border_, -1, 1); | 
| -  gtk_widget_set_no_show_all(toolbar_border_, TRUE); | 
| -  g_signal_connect_after(toolbar_border_, "expose-event", | 
| -                         G_CALLBACK(OnExposeDrawInfobarBitsThunk), this); | 
| - | 
| -  if (IsToolbarSupported()) | 
| -    gtk_widget_show(toolbar_border_); | 
| - | 
| -  infobar_container_.reset( | 
| -      new InfoBarContainerGtk(this, browser_->profile())); | 
| -  gtk_box_pack_start(GTK_BOX(render_area_vbox_), | 
| -                     infobar_container_->widget(), | 
| -                     FALSE, FALSE, 0); | 
| - | 
| -  status_bubble_.reset(new StatusBubbleGtk(browser_->profile())); | 
| - | 
| -  contents_container_.reset(new TabContentsContainerGtk( | 
| -      status_bubble_.get(), | 
| -      implicit_cast<content::WebContentsDelegate*>(browser_.get())-> | 
| -          EmbedsFullscreenWidget())); | 
| -  devtools_container_.reset(new TabContentsContainerGtk(NULL, false)); | 
| -  // DevTools container should only have DevTools-specific view ID. | 
| -  ViewIDUtil::SetDelegateForWidget(devtools_container_->widget(), NULL); | 
| -  ViewIDUtil::SetID(devtools_container_->widget(), VIEW_ID_DEV_TOOLS_DOCKED); | 
| - | 
| -  devtools_floating_container_ = gtk_floating_container_new(); | 
| -  gtk_container_add(GTK_CONTAINER(devtools_floating_container_), | 
| -                    devtools_container_->widget()); | 
| -  gtk_floating_container_add_floating( | 
| -      GTK_FLOATING_CONTAINER(devtools_floating_container_), | 
| -      contents_container_->widget()); | 
| -  g_signal_connect(devtools_floating_container_, "set-floating-position", | 
| -                   G_CALLBACK(OnDevToolsContainerSetFloatingPosition), this); | 
| -  gtk_box_pack_end(GTK_BOX(render_area_vbox_), | 
| -                   devtools_floating_container_, TRUE, TRUE, 0); | 
| - | 
| -  gtk_widget_show_all(render_area_floating_container_); | 
| - | 
| -  render_area_event_box_ = gtk_event_box_new(); | 
| -  // Set a white background so during startup the user sees white in the | 
| -  // content area before we get a WebContents in place. | 
| -  gtk_widget_modify_bg(render_area_event_box_, GTK_STATE_NORMAL, | 
| -                       &ui::kGdkWhite); | 
| -  gtk_container_add(GTK_CONTAINER(render_area_event_box_), | 
| -                    render_area_floating_container_); | 
| -  gtk_widget_show(render_area_event_box_); | 
| -  gtk_box_pack_end(GTK_BOX(window_vbox_), render_area_event_box_, | 
| -                   TRUE, TRUE, 0); | 
| - | 
| -  if (IsBookmarkBarSupported()) { | 
| -    bookmark_bar_.reset(new BookmarkBarGtk(this, | 
| -                                           browser_.get(), | 
| -                                           tabstrip_.get())); | 
| -    PlaceBookmarkBar(false); | 
| -    gtk_widget_show(bookmark_bar_->widget()); | 
| - | 
| -    g_signal_connect_after(bookmark_bar_->widget(), "expose-event", | 
| -                           G_CALLBACK(OnBookmarkBarExposeThunk), this); | 
| -    g_signal_connect(bookmark_bar_->widget(), "size-allocate", | 
| -                     G_CALLBACK(OnBookmarkBarSizeAllocateThunk), this); | 
| -  } | 
| - | 
| -  // We have to realize the window before we try to apply a window shape mask. | 
| -  gtk_widget_realize(GTK_WIDGET(window_)); | 
| -  state_ = gdk_window_get_state(gtk_widget_get_window(GTK_WIDGET(window_))); | 
| -  // Note that calling this the first time is necessary to get the | 
| -  // proper control layout. | 
| -  UpdateCustomFrame(); | 
| - | 
| -  // Add the keybinding registry, now that the window has been realized. | 
| -  extension_keybinding_registry_.reset(new ExtensionKeybindingRegistryGtk( | 
| -      browser_->profile(), | 
| -      window_, | 
| -      extensions::ExtensionKeybindingRegistry::ALL_EXTENSIONS, | 
| -      this)); | 
| - | 
| -  // We have to call this after the first window is created, but after that only | 
| -  // when the theme changes. This sets the icon that will be used for windows | 
| -  // that have not explicitly been assigned an icon. | 
| -  static bool default_icon_set = false; | 
| -  if (!default_icon_set) { | 
| -    gtk_util::SetDefaultWindowIcon(window_); | 
| -    default_icon_set = true; | 
| -  } | 
| -  // Set this window's (potentially profile-avatar-emblemed) icon, overriding | 
| -  // the default. | 
| -  gtk_util::SetWindowIcon(window_, browser_->profile()); | 
| - | 
| -  gtk_container_add(GTK_CONTAINER(window_), window_container_); | 
| -  gtk_widget_show(window_container_); | 
| -  browser_->tab_strip_model()->AddObserver(this); | 
| -} | 
| - | 
| -void BrowserWindowGtk::SetBackgroundColor() { | 
| -  Profile* profile = browser()->profile(); | 
| -  GtkThemeService* theme_provider = GtkThemeService::GetFrom(profile); | 
| -  int frame_color_id; | 
| -  if (UsingCustomPopupFrame()) { | 
| -    frame_color_id = ThemeProperties::COLOR_TOOLBAR; | 
| -  } else if (DrawFrameAsActive()) { | 
| -    frame_color_id = browser()->profile()->IsOffTheRecord() | 
| -       ? ThemeProperties::COLOR_FRAME_INCOGNITO | 
| -       : ThemeProperties::COLOR_FRAME; | 
| -  } else { | 
| -    frame_color_id = browser()->profile()->IsOffTheRecord() | 
| -       ? ThemeProperties::COLOR_FRAME_INCOGNITO_INACTIVE | 
| -       : ThemeProperties::COLOR_FRAME_INACTIVE; | 
| -  } | 
| - | 
| -  SkColor frame_color = theme_provider->GetColor(frame_color_id); | 
| - | 
| -  // Paint the frame color on the left, right and bottom. | 
| -  GdkColor frame_color_gdk = gfx::SkColorToGdkColor(frame_color); | 
| -  gtk_widget_modify_bg(GTK_WIDGET(window_), GTK_STATE_NORMAL, | 
| -                       &frame_color_gdk); | 
| - | 
| -  GdkColor border_color = theme_provider->GetBorderColor(); | 
| -  gtk_widget_modify_bg(toolbar_border_, GTK_STATE_NORMAL, &border_color); | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateWindowShape(int width, int height) { | 
| -  using gtk_window_util::kFrameBorderThickness; | 
| -  GdkRegion* mask = GetWindowShape(width, height); | 
| -  gdk_window_shape_combine_region( | 
| -      gtk_widget_get_window(GTK_WIDGET(window_)), mask, 0, 0); | 
| -  if (mask) | 
| -    gdk_region_destroy(mask); | 
| - | 
| -  if (UseCustomFrame() && !IsFullscreen() && !IsMaximized()) { | 
| -    gtk_alignment_set_padding(GTK_ALIGNMENT(window_container_), 1, | 
| -        kFrameBorderThickness, kFrameBorderThickness, kFrameBorderThickness); | 
| -  } else { | 
| -    gtk_alignment_set_padding(GTK_ALIGNMENT(window_container_), 0, 0, 0, 0); | 
| -  } | 
| -} | 
| - | 
| -GdkRegion* BrowserWindowGtk::GetWindowShape(int width, int height) const { | 
| -  if (UseCustomFrame() && !IsFullscreen() && !IsMaximized()) { | 
| -    // Make the corners rounded.  We set a mask that includes most of the | 
| -    // window except for a few pixels in each corner. | 
| -    GdkRectangle top_top_rect = { 3, 0, width - 6, 1 }; | 
| -    GdkRectangle top_mid_rect = { 1, 1, width - 2, 2 }; | 
| -    GdkRectangle mid_rect = { 0, 3, width, height - 6 }; | 
| -    // The bottom two rects are mirror images of the top two rects. | 
| -    GdkRectangle bot_mid_rect = top_mid_rect; | 
| -    bot_mid_rect.y = height - 3; | 
| -    GdkRectangle bot_bot_rect = top_top_rect; | 
| -    bot_bot_rect.y = height - 1; | 
| -    GdkRegion* mask = gdk_region_rectangle(&top_top_rect); | 
| -    gdk_region_union_with_rect(mask, &top_mid_rect); | 
| -    gdk_region_union_with_rect(mask, &mid_rect); | 
| -    gdk_region_union_with_rect(mask, &bot_mid_rect); | 
| -    gdk_region_union_with_rect(mask, &bot_bot_rect); | 
| -    return mask; | 
| -  } else if (UseCustomFrame()) { | 
| -    // Disable rounded corners.  Simply passing in a NULL region doesn't | 
| -    // seem to work on KWin, so manually set the shape to the whole window. | 
| -    GdkRectangle rect = { 0, 0, width, height }; | 
| -    GdkRegion* mask = gdk_region_rectangle(&rect); | 
| -    return mask; | 
| -  } else { | 
| -    // XFCE disables the system decorations if there's an xshape set. Do not | 
| -    // use the KWin hack when the custom frame is not enabled. | 
| -    return NULL; | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::ConnectAccelerators() { | 
| -  accel_group_ = gtk_accel_group_new(); | 
| -  gtk_window_add_accel_group(window_, accel_group_); | 
| - | 
| -  AcceleratorsGtk* accelerators = AcceleratorsGtk::GetInstance(); | 
| -  for (AcceleratorsGtk::const_iterator iter = accelerators->begin(); | 
| -       iter != accelerators->end(); ++iter) { | 
| -    gtk_accel_group_connect( | 
| -        accel_group_, | 
| -        ui::GetGdkKeyCodeForAccelerator(iter->second), | 
| -        ui::GetGdkModifierForAccelerator(iter->second), | 
| -        GtkAccelFlags(0), | 
| -        g_cclosure_new(G_CALLBACK(OnGtkAccelerator), | 
| -                       GINT_TO_POINTER(iter->first), NULL)); | 
| -  } | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateCustomFrame() { | 
| -  gtk_window_set_decorated(window_, !UseCustomFrame()); | 
| -  titlebar_->UpdateCustomFrame(UseCustomFrame() && !IsFullscreen()); | 
| -  UpdateWindowShape(bounds_.width(), bounds_.height()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::InvalidateWindow() { | 
| -  GtkAllocation allocation; | 
| -  gtk_widget_get_allocation(GTK_WIDGET(window_), &allocation); | 
| -  gdk_window_invalidate_rect(gtk_widget_get_window(GTK_WIDGET(window_)), | 
| -                             &allocation, TRUE); | 
| -} | 
| - | 
| -void BrowserWindowGtk::SaveWindowPosition() { | 
| -  // Browser::SaveWindowPlacement is used for session restore. | 
| -  ui::WindowShowState show_state = ui::SHOW_STATE_NORMAL; | 
| -  if (IsMaximized()) | 
| -    show_state = ui::SHOW_STATE_MAXIMIZED; | 
| -  else if (IsMinimized()) | 
| -    show_state = ui::SHOW_STATE_MINIMIZED; | 
| - | 
| -  if (chrome::ShouldSaveWindowPlacement(browser_.get())) | 
| -    chrome::SaveWindowPlacement(browser_.get(), restored_bounds_, show_state); | 
| - | 
| -  // We also need to save the placement for startup. | 
| -  // This is a web of calls between views and delegates on Windows, but the | 
| -  // crux of the logic follows.  See also cocoa/browser_window_controller.mm. | 
| -  if (!browser_->profile()->GetPrefs()) | 
| -    return; | 
| - | 
| -  std::string window_name = chrome::GetWindowPlacementKey(browser_.get()); | 
| -  DictionaryPrefUpdate update(browser_->profile()->GetPrefs(), | 
| -                              window_name.c_str()); | 
| -  base::DictionaryValue* window_preferences = update.Get(); | 
| -  // Note that we store left/top for consistency with Windows, but that we | 
| -  // *don't* obey them; we only use them for computing width/height.  See | 
| -  // comments in SetGeometryHints(). | 
| -  window_preferences->SetInteger("left", restored_bounds_.x()); | 
| -  window_preferences->SetInteger("top", restored_bounds_.y()); | 
| -  window_preferences->SetInteger("right", restored_bounds_.right()); | 
| -  window_preferences->SetInteger("bottom", restored_bounds_.bottom()); | 
| -  window_preferences->SetBoolean("maximized", IsMaximized()); | 
| - | 
| -  gfx::Rect work_area(gfx::Screen::GetNativeScreen()->GetDisplayMatching( | 
| -      restored_bounds_).work_area()); | 
| -  window_preferences->SetInteger("work_area_left", work_area.x()); | 
| -  window_preferences->SetInteger("work_area_top", work_area.y()); | 
| -  window_preferences->SetInteger("work_area_right", work_area.right()); | 
| -  window_preferences->SetInteger("work_area_bottom", work_area.bottom()); | 
| -} | 
| - | 
| -void BrowserWindowGtk::InvalidateInfoBarBits() { | 
| -  gtk_widget_queue_draw(toolbar_border_); | 
| -  gtk_widget_queue_draw(toolbar_->widget()); | 
| -  if (bookmark_bar_.get() && | 
| -      browser_->bookmark_bar_state() != BookmarkBar::DETACHED) { | 
| -    gtk_widget_queue_draw(bookmark_bar_->widget()); | 
| -  } | 
| -} | 
| - | 
| -int BrowserWindowGtk::GetXPositionOfLocationIcon(GtkWidget* relative_to) { | 
| -  GtkWidget* location_icon = toolbar_->GetLocationBarView()-> | 
| -      location_icon_widget(); | 
| - | 
| -  GtkAllocation location_icon_allocation; | 
| -  gtk_widget_get_allocation(location_icon, &location_icon_allocation); | 
| - | 
| -  int x = 0; | 
| -  gtk_widget_translate_coordinates( | 
| -      location_icon, relative_to, | 
| -      (location_icon_allocation.width + 1) / 2, | 
| -      0, &x, NULL); | 
| - | 
| -  if (!gtk_widget_get_has_window(relative_to)) { | 
| -    GtkAllocation allocation; | 
| -    gtk_widget_get_allocation(relative_to, &allocation); | 
| -    x += allocation.x; | 
| -  } | 
| - | 
| -  return x; | 
| -} | 
| - | 
| -void BrowserWindowGtk::MaybeShowBookmarkBar(bool animate) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::MaybeShowBookmarkBar"); | 
| -  if (!IsBookmarkBarSupported()) | 
| -    return; | 
| - | 
| -  if (GetDisplayedTab()) | 
| -    bookmark_bar_->SetPageNavigator(browser_.get()); | 
| - | 
| -  BookmarkBar::State state = browser_->bookmark_bar_state(); | 
| -  toolbar_->UpdateForBookmarkBarVisibility(state == BookmarkBar::DETACHED); | 
| -  PlaceBookmarkBar(state == BookmarkBar::DETACHED); | 
| -  bookmark_bar_->SetBookmarkBarState( | 
| -      state, | 
| -      animate ? BookmarkBar::ANIMATE_STATE_CHANGE : | 
| -                BookmarkBar::DONT_ANIMATE_STATE_CHANGE); | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnLocationIconSizeAllocate(GtkWidget* sender, | 
| -                                                  GtkAllocation* allocation) { | 
| -  // The position of the arrow may have changed, so we'll have to redraw it. | 
| -  InvalidateInfoBarBits(); | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnExposeDrawInfobarBits(GtkWidget* sender, | 
| -                                                   GdkEventExpose* expose) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::OnExposeDrawInfobarBits"); | 
| -  // Maybe draw infobars | 
| -  infobar_container_->PaintInfobarBitsOn(sender, expose, NULL); | 
| - | 
| -  return FALSE; | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnBookmarkBarExpose(GtkWidget* sender, | 
| -                                               GdkEventExpose* expose) { | 
| -  if (browser_->bookmark_bar_state() == BookmarkBar::DETACHED) | 
| -    return FALSE; | 
| - | 
| -  return OnExposeDrawInfobarBits(sender, expose); | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnBookmarkBarSizeAllocate(GtkWidget* sender, | 
| -                                                 GtkAllocation* allocation) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::OnBookmarkBarSizeAllocate"); | 
| -  // The size of the bookmark bar affects how the infobar arrow is drawn on | 
| -  // the toolbar. | 
| -  if (infobar_container_->ContainsInfobars()) | 
| -    InvalidateInfoBarBits(); | 
| - | 
| -  // Pass the new size to our infobar container. | 
| -  int arrow_size = InfoBar::kDefaultArrowTargetHeight; | 
| -  if (browser_->bookmark_bar_state() != BookmarkBar::DETACHED) | 
| -    arrow_size += allocation->height; | 
| -  infobar_container_->SetMaxTopArrowHeight(arrow_size); | 
| -} | 
| - | 
| -// static | 
| -gboolean BrowserWindowGtk::OnGtkAccelerator(GtkAccelGroup* accel_group, | 
| -                                            GObject* acceleratable, | 
| -                                            guint keyval, | 
| -                                            GdkModifierType modifier, | 
| -                                            void* user_data) { | 
| -  int command_id = GPOINTER_TO_INT(user_data); | 
| -  BrowserWindowGtk* browser_window = | 
| -      GetBrowserWindowForNativeWindow(GTK_WINDOW(acceleratable)); | 
| -  DCHECK(browser_window != NULL); | 
| -  return chrome::ExecuteCommand(browser_window->browser(), command_id); | 
| -} | 
| - | 
| -// Let the focused widget have first crack at the key event so we don't | 
| -// override their accelerators, except if there is a priority keybinding | 
| -// handler registered (it should take precedence). | 
| -gboolean BrowserWindowGtk::OnKeyPress(GtkWidget* widget, GdkEventKey* event) { | 
| -  if (extension_keybinding_registry_->HasPriorityHandler(event)) | 
| -    return FALSE; | 
| - | 
| -  // If a widget besides the native view is focused, we have to try to handle | 
| -  // the custom accelerators before letting it handle them. | 
| -  WebContents* current_web_contents = | 
| -      browser()->tab_strip_model()->GetActiveWebContents(); | 
| -  // The current tab might not have a render view if it crashed. | 
| -  if (!current_web_contents || | 
| -      !current_web_contents->GetView()->GetContentNativeView() || | 
| -      !gtk_widget_is_focus( | 
| -          current_web_contents->GetView()->GetContentNativeView())) { | 
| -    int command_id = GetCustomCommandId(event); | 
| -    if (command_id == -1) | 
| -      command_id = GetPreHandleCommandId(event); | 
| - | 
| -    if (command_id != -1 && chrome::ExecuteCommand(browser_.get(), command_id)) | 
| -      return TRUE; | 
| - | 
| -    // Propagate the key event to child widget first, so we don't override their | 
| -    // accelerators. | 
| -    if (!gtk_window_propagate_key_event(GTK_WINDOW(widget), event)) { | 
| -      if (!gtk_window_activate_key(GTK_WINDOW(widget), event)) { | 
| -        gtk_bindings_activate_event(GTK_OBJECT(widget), event); | 
| -      } | 
| -    } | 
| -  } else { | 
| -    bool rv = gtk_window_propagate_key_event(GTK_WINDOW(widget), event); | 
| -    DCHECK(rv); | 
| -  } | 
| - | 
| -  // Prevents the default handler from handling this event. | 
| -  return TRUE; | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnMouseMoveEvent(GtkWidget* widget, | 
| -                                            GdkEventMotion* event) { | 
| -  // This method is used to update the mouse cursor when over the edge of the | 
| -  // custom frame.  If the custom frame is off or we're over some other widget, | 
| -  // do nothing. | 
| -  if (!UseCustomFrame() || event->window != gtk_widget_get_window(widget)) { | 
| -    // Reset the cursor. | 
| -    if (frame_cursor_) { | 
| -      frame_cursor_ = NULL; | 
| -      gdk_window_set_cursor(gtk_widget_get_window(GTK_WIDGET(window_)), NULL); | 
| -    } | 
| -    return FALSE; | 
| -  } | 
| - | 
| -  // Update the cursor if we're on the custom frame border. | 
| -  GdkWindowEdge edge; | 
| -  bool has_hit_edge = GetWindowEdge(static_cast<int>(event->x), | 
| -                                    static_cast<int>(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) { | 
| -    if (has_hit_edge) { | 
| -      frame_cursor_ = gfx::GetCursor(new_cursor); | 
| -    } else { | 
| -      frame_cursor_ = NULL; | 
| -    } | 
| -    gdk_window_set_cursor(gtk_widget_get_window(GTK_WIDGET(window_)), | 
| -                          frame_cursor_); | 
| -  } | 
| -  return FALSE; | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnButtonPressEvent(GtkWidget* widget, | 
| -                                              GdkEventButton* event) { | 
| -  // Handle back/forward. | 
| -  if (event->type == GDK_BUTTON_PRESS) { | 
| -    if (event->button == 8) { | 
| -      chrome::GoBack(browser_.get(), CURRENT_TAB); | 
| -      return TRUE; | 
| -    } else if (event->button == 9) { | 
| -      chrome::GoForward(browser_.get(), CURRENT_TAB); | 
| -      return TRUE; | 
| -    } | 
| -  } | 
| - | 
| -  // Handle left, middle and right clicks.  In particular, we care about clicks | 
| -  // in the custom frame border and clicks in the titlebar. | 
| - | 
| -  // 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<int>(event->x_root - win_x), | 
| -                   static_cast<int>(event->y_root - win_y)); | 
| -  bool has_hit_edge = GetWindowEdge(point.x(), point.y(), &edge); | 
| - | 
| -  // Ignore clicks that are in/below the browser toolbar. | 
| -  GtkWidget* toolbar = toolbar_->widget(); | 
| -  if (!gtk_widget_get_visible(toolbar)) { | 
| -    // If the toolbar is not showing, use the location of web contents as the | 
| -    // boundary of where to ignore clicks. | 
| -    toolbar = render_area_vbox_; | 
| -  } | 
| -  gint toolbar_y; | 
| -  gtk_widget_get_pointer(toolbar, NULL, &toolbar_y); | 
| -  bool has_hit_titlebar = !IsFullscreen() && (toolbar_y < 0) | 
| -                          && !has_hit_edge; | 
| -  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(gdk_window); | 
| - | 
| -      if (has_hit_titlebar) { | 
| -        return gtk_window_util::HandleTitleBarLeftMousePress( | 
| -            window_, bounds_, event); | 
| -      } else if (has_hit_edge) { | 
| -        gtk_window_begin_resize_drag(window_, edge, event->button, | 
| -                                     static_cast<gint>(event->x_root), | 
| -                                     static_cast<gint>(event->y_root), | 
| -                                     event->time); | 
| -        return TRUE; | 
| -      } | 
| -    } else if (GDK_2BUTTON_PRESS == event->type) { | 
| -      if (has_hit_titlebar) { | 
| -        // Maximize/restore on double click. | 
| -        if (IsMaximized()) { | 
| -          UnMaximize(); | 
| -        } 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; | 
| -  } else if (event->button == 3) { | 
| -    if (has_hit_titlebar) { | 
| -      titlebar_->ShowContextMenu(event); | 
| -      return TRUE; | 
| -    } | 
| -  } | 
| - | 
| -  return FALSE;  // Continue to propagate the event. | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnFocusIn(GtkWidget* widget, | 
| -                                     GdkEventFocus* event) { | 
| -  BrowserList::SetLastActive(browser_.get()); | 
| -  return FALSE; | 
| -} | 
| - | 
| -gboolean BrowserWindowGtk::OnFocusOut(GtkWidget* widget, | 
| -                                      GdkEventFocus* event) { | 
| -  return FALSE; | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowSupportedWindowFeatures() { | 
| -  if (IsTabStripSupported()) | 
| -    tabstrip_->Show(); | 
| - | 
| -  if (IsToolbarSupported()) { | 
| -    toolbar_->Show(); | 
| -    gtk_widget_show(toolbar_border_); | 
| -    gdk_window_lower(gtk_widget_get_window(toolbar_border_)); | 
| -  } | 
| - | 
| -  if (IsBookmarkBarSupported()) | 
| -    MaybeShowBookmarkBar(false); | 
| -} | 
| - | 
| -void BrowserWindowGtk::HideUnsupportedWindowFeatures() { | 
| -  if (!IsTabStripSupported()) | 
| -    tabstrip_->Hide(); | 
| - | 
| -  if (!IsToolbarSupported()) | 
| -    toolbar_->Hide(); | 
| - | 
| -  // If the bookmark bar shelf is unsupported, then we never create it. | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsTabStripSupported() const { | 
| -  return browser_->SupportsWindowFeature(Browser::FEATURE_TABSTRIP); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsToolbarSupported() const { | 
| -  return browser_->SupportsWindowFeature(Browser::FEATURE_TOOLBAR) || | 
| -         browser_->SupportsWindowFeature(Browser::FEATURE_LOCATIONBAR); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::IsBookmarkBarSupported() const { | 
| -  return browser_->SupportsWindowFeature(Browser::FEATURE_BOOKMARKBAR); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::UsingCustomPopupFrame() const { | 
| -  GtkThemeService* theme_provider = GtkThemeService::GetFrom( | 
| -      browser()->profile()); | 
| -  return !theme_provider->UsingNativeTheme() && browser()->is_type_popup(); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::GetWindowEdge(int x, int y, GdkWindowEdge* edge) { | 
| -  if (!UseCustomFrame()) | 
| -    return false; | 
| - | 
| -  if (IsMaximized() || IsFullscreen()) | 
| -    return false; | 
| - | 
| -  return gtk_window_util::GetWindowEdge( | 
| -      bounds_.size(), kTopResizeAdjust, x, y, edge); | 
| -} | 
| - | 
| -bool BrowserWindowGtk::UseCustomFrame() const { | 
| -  // We don't use the custom frame for app mode windows or app window popups. | 
| -  return use_custom_frame_pref_.GetValue() && !browser_->is_app(); | 
| -} | 
| - | 
| -void BrowserWindowGtk::PlaceBookmarkBar(bool is_floating) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::PlaceBookmarkBar"); | 
| - | 
| -  GtkWidget* target_parent = NULL; | 
| -  if (!is_floating) { | 
| -    // Place the bookmark bar at the end of |window_vbox_|; this happens after | 
| -    // we have placed the render area at the end of |window_vbox_| so we will | 
| -    // be above the render area. | 
| -    target_parent = window_vbox_; | 
| -  } else { | 
| -    // Place the bookmark bar at the end of the render area; this happens after | 
| -    // the tab contents container has been placed there so we will be | 
| -    // above the webpage (in terms of y). | 
| -    target_parent = render_area_vbox_; | 
| -  } | 
| - | 
| -  GtkWidget* parent = gtk_widget_get_parent(bookmark_bar_->widget()); | 
| -  if (parent != target_parent) { | 
| -    if (parent) | 
| -      gtk_container_remove(GTK_CONTAINER(parent), bookmark_bar_->widget()); | 
| - | 
| -    gtk_box_pack_end(GTK_BOX(target_parent), bookmark_bar_->widget(), | 
| -                     FALSE, FALSE, 0); | 
| -  } | 
| -} | 
| - | 
| -bool BrowserWindowGtk::DrawFrameAsActive() const { | 
| -  if (ui::ActiveWindowWatcherX::WMSupportsActivation()) | 
| -    return is_active_; | 
| - | 
| -  // Since we don't get notifications when the active state of the frame | 
| -  // changes, we can't consistently repaint the frame at the right time. Instead | 
| -  // we always draw the frame as active. | 
| -  return true; | 
| -} | 
| - | 
| -void BrowserWindowGtk::UpdateDevToolsForContents(WebContents* contents) { | 
| -  TRACE_EVENT0("ui::gtk", "BrowserWindowGtk::UpdateDevToolsForContents"); | 
| -  DevToolsWindow* new_devtools_window = contents ? | 
| -      DevToolsWindow::GetDockedInstanceForInspectedTab(contents) : NULL; | 
| - | 
| -  // Replace tab contents. | 
| -  if (devtools_window_ != new_devtools_window) { | 
| -    if (devtools_window_) | 
| -      devtools_container_->DetachTab(devtools_window_->web_contents()); | 
| -    devtools_container_->SetTab( | 
| -        new_devtools_window ? new_devtools_window->web_contents() : NULL); | 
| -    if (new_devtools_window) { | 
| -      // WebContentsViewGtk::WasShown is not called when a web contents is shown | 
| -      // by anything other than user selecting a Tab. | 
| -      // See TabContentsViewViews::OnWindowPosChanged for reference on how it | 
| -      // should be implemented. | 
| -      new_devtools_window->web_contents()->WasShown(); | 
| -    } | 
| -  } | 
| - | 
| -  // Show / hide container if necessary. | 
| -  bool should_hide = devtools_window_ && !new_devtools_window; | 
| -  bool should_show = new_devtools_window && !devtools_window_; | 
| - | 
| -  if (should_hide) | 
| -    HideDevToolsContainer(); | 
| - | 
| -  devtools_window_ = new_devtools_window; | 
| -  if (devtools_window_) { | 
| -    contents_resizing_strategy_.CopyFrom( | 
| -      devtools_window_->GetContentsResizingStrategy()); | 
| -  } else { | 
| -    contents_resizing_strategy_.CopyFrom(DevToolsContentsResizingStrategy()); | 
| -  } | 
| - | 
| -  if (should_show) | 
| -    ShowDevToolsContainer(); | 
| - | 
| -  gtk_widget_queue_resize(devtools_floating_container_); | 
| -  gtk_widget_queue_draw(devtools_floating_container_); | 
| -} | 
| - | 
| -void BrowserWindowGtk::ShowDevToolsContainer() { | 
| -  // Move devtools below contents. | 
| -  GdkWindow* const devtools_gdk_window = | 
| -      gtk_widget_get_window(devtools_container_->widget()); | 
| -  if (devtools_gdk_window) | 
| -    gdk_window_lower(devtools_gdk_window); | 
| -} | 
| - | 
| -void BrowserWindowGtk::HideDevToolsContainer() { | 
| -  // This method is left intentionally blank. | 
| -} | 
| - | 
| -// static | 
| -void BrowserWindowGtk::OnDevToolsContainerSetFloatingPosition( | 
| -    GtkFloatingContainer* container, GtkAllocation* allocation, | 
| -    BrowserWindowGtk* browser_window) { | 
| -  GtkAllocation contents_allocation; | 
| -  gtk_widget_get_allocation(browser_window->contents_container_->widget(), | 
| -      &contents_allocation); | 
| - | 
| -  gfx::Size container_size(allocation->width, allocation->height); | 
| -  gfx::Rect old_devtools_bounds(0, 0, allocation->width, allocation->height); | 
| -  gfx::Rect old_contents_bounds(contents_allocation.x, contents_allocation.y, | 
| -      contents_allocation.width, contents_allocation.height); | 
| -  gfx::Rect new_devtools_bounds; | 
| -  gfx::Rect new_contents_bounds; | 
| - | 
| -  ApplyDevToolsContentsResizingStrategy( | 
| -      browser_window->contents_resizing_strategy_, container_size, | 
| -      old_devtools_bounds, old_contents_bounds, | 
| -      &new_devtools_bounds, &new_contents_bounds); | 
| - | 
| -  gtk_widget_set_size_request(browser_window->contents_container_->widget(), | 
| -      new_contents_bounds.width(), new_contents_bounds.height()); | 
| - | 
| -  GValue value = { 0, }; | 
| -  g_value_init(&value, G_TYPE_INT); | 
| -  g_value_set_int(&value, new_contents_bounds.x()); | 
| -  gtk_container_child_set_property(GTK_CONTAINER(container), | 
| -      browser_window->contents_container_->widget(), "x", &value); | 
| -  g_value_set_int(&value, new_contents_bounds.y()); | 
| -  gtk_container_child_set_property(GTK_CONTAINER(container), | 
| -      browser_window->contents_container_->widget(), "y", &value); | 
| -  g_value_unset(&value); | 
| -} | 
| - | 
| -void BrowserWindowGtk::OnUseCustomChromeFrameChanged() { | 
| -  UpdateCustomFrame(); | 
| -  ui::SetHideTitlebarWhenMaximizedProperty( | 
| -      ui::GetX11WindowFromGtkWidget(GTK_WIDGET(window_)), | 
| -      UseCustomFrame() ? ui::HIDE_TITLEBAR_WHEN_MAXIMIZED : | 
| -                         ui::SHOW_TITLEBAR_WHEN_MAXIMIZED); | 
| -} | 
| - | 
| -// static | 
| -BrowserWindow* BrowserWindow::CreateBrowserWindow(Browser* browser) { | 
| -  BrowserWindowGtk* browser_window_gtk = new BrowserWindowGtk(browser); | 
| -  browser_window_gtk->Init(); | 
| -  return browser_window_gtk; | 
| -} | 
| - | 
| -// static | 
| -chrome::HostDesktopType BrowserWindow::AdjustHostDesktopType( | 
| -    chrome::HostDesktopType desktop_type) { | 
| -  return desktop_type; | 
| -} | 
|  |