| OLD | NEW |
| (Empty) |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "chrome/browser/gtk/reload_button_gtk.h" | |
| 6 | |
| 7 #include <algorithm> | |
| 8 | |
| 9 #include "app/l10n_util.h" | |
| 10 #include "base/logging.h" | |
| 11 #include "chrome/app/chrome_command_ids.h" | |
| 12 #include "chrome/browser/gtk/gtk_chrome_button.h" | |
| 13 #include "chrome/browser/gtk/gtk_theme_provider.h" | |
| 14 #include "chrome/browser/gtk/gtk_util.h" | |
| 15 #include "chrome/browser/gtk/location_bar_view_gtk.h" | |
| 16 #include "chrome/browser/ui/browser.h" | |
| 17 #include "chrome/common/notification_source.h" | |
| 18 #include "grit/generated_resources.h" | |
| 19 #include "grit/theme_resources.h" | |
| 20 | |
| 21 // The width of this button in GTK+ theme mode. The Stop and Refresh stock icons | |
| 22 // can be different sizes; this variable is used to make sure that the button | |
| 23 // doesn't change sizes when switching between the two. | |
| 24 static int GtkButtonWidth = 0; | |
| 25 | |
| 26 //////////////////////////////////////////////////////////////////////////////// | |
| 27 // ReloadButton, public: | |
| 28 | |
| 29 ReloadButtonGtk::ReloadButtonGtk(LocationBarViewGtk* location_bar, | |
| 30 Browser* browser) | |
| 31 : location_bar_(location_bar), | |
| 32 browser_(browser), | |
| 33 intended_mode_(MODE_RELOAD), | |
| 34 visible_mode_(MODE_RELOAD), | |
| 35 theme_provider_(browser ? | |
| 36 GtkThemeProvider::GetFrom(browser->profile()) : NULL), | |
| 37 reload_(theme_provider_, IDR_RELOAD, IDR_RELOAD_P, IDR_RELOAD_H, 0), | |
| 38 stop_(theme_provider_, IDR_STOP, IDR_STOP_P, IDR_STOP_H, IDR_STOP_D), | |
| 39 widget_(gtk_chrome_button_new()), | |
| 40 stop_to_reload_timer_delay_(base::TimeDelta::FromMilliseconds(1350)), | |
| 41 testing_mouse_hovered_(false), | |
| 42 testing_reload_count_(0) { | |
| 43 gtk_widget_set_size_request(widget(), reload_.Width(), reload_.Height()); | |
| 44 | |
| 45 gtk_widget_set_app_paintable(widget(), TRUE); | |
| 46 | |
| 47 g_signal_connect(widget(), "clicked", G_CALLBACK(OnClickedThunk), this); | |
| 48 g_signal_connect(widget(), "expose-event", G_CALLBACK(OnExposeThunk), this); | |
| 49 g_signal_connect(widget(), "leave-notify-event", | |
| 50 G_CALLBACK(OnLeaveNotifyThunk), this); | |
| 51 GTK_WIDGET_UNSET_FLAGS(widget(), GTK_CAN_FOCUS); | |
| 52 | |
| 53 gtk_widget_set_has_tooltip(widget(), TRUE); | |
| 54 g_signal_connect(widget(), "query-tooltip", G_CALLBACK(OnQueryTooltipThunk), | |
| 55 this); | |
| 56 | |
| 57 hover_controller_.Init(widget()); | |
| 58 gtk_util::SetButtonTriggersNavigation(widget()); | |
| 59 | |
| 60 if (theme_provider_) { | |
| 61 theme_provider_->InitThemesFor(this); | |
| 62 registrar_.Add(this, | |
| 63 NotificationType::BROWSER_THEME_CHANGED, | |
| 64 Source<GtkThemeProvider>(theme_provider_)); | |
| 65 } | |
| 66 | |
| 67 // Set the default double-click timer delay to the system double-click time. | |
| 68 int timer_delay_ms; | |
| 69 GtkSettings* settings = gtk_settings_get_default(); | |
| 70 g_object_get(G_OBJECT(settings), "gtk-double-click-time", &timer_delay_ms, | |
| 71 NULL); | |
| 72 double_click_timer_delay_ = base::TimeDelta::FromMilliseconds(timer_delay_ms); | |
| 73 } | |
| 74 | |
| 75 ReloadButtonGtk::~ReloadButtonGtk() { | |
| 76 widget_.Destroy(); | |
| 77 } | |
| 78 | |
| 79 void ReloadButtonGtk::ChangeMode(Mode mode, bool force) { | |
| 80 intended_mode_ = mode; | |
| 81 | |
| 82 // If the change is forced, or the user isn't hovering the icon, or it's safe | |
| 83 // to change it to the other image type, make the change immediately; | |
| 84 // otherwise we'll let it happen later. | |
| 85 if (force || ((GTK_WIDGET_STATE(widget()) == GTK_STATE_NORMAL) && | |
| 86 !testing_mouse_hovered_) || ((mode == MODE_STOP) ? | |
| 87 !double_click_timer_.IsRunning() : (visible_mode_ != MODE_STOP))) { | |
| 88 double_click_timer_.Stop(); | |
| 89 stop_to_reload_timer_.Stop(); | |
| 90 visible_mode_ = mode; | |
| 91 | |
| 92 stop_.set_paint_override(-1); | |
| 93 gtk_chrome_button_unset_paint_state(GTK_CHROME_BUTTON(widget_.get())); | |
| 94 | |
| 95 UpdateThemeButtons(); | |
| 96 gtk_widget_queue_draw(widget()); | |
| 97 } else if (visible_mode_ != MODE_RELOAD) { | |
| 98 // If you read the views implementation of reload_button.cc, you'll see | |
| 99 // that instead of screwing with paint states, the views implementation | |
| 100 // just changes whether the view is enabled. We can't do that here because | |
| 101 // changing the widget state to GTK_STATE_INSENSITIVE will cause a cascade | |
| 102 // of messages on all its children and will also trigger a synthesized | |
| 103 // leave notification and prevent the real leave notification from turning | |
| 104 // the button back to normal. So instead, override the stop_ paint state | |
| 105 // for chrome-theme mode, and use this as a flag to discard click events. | |
| 106 stop_.set_paint_override(GTK_STATE_INSENSITIVE); | |
| 107 | |
| 108 // Also set the gtk_chrome_button paint state to insensitive to hide | |
| 109 // the border drawn around the stop icon. | |
| 110 gtk_chrome_button_set_paint_state(GTK_CHROME_BUTTON(widget_.get()), | |
| 111 GTK_STATE_INSENSITIVE); | |
| 112 | |
| 113 // If we're in GTK theme mode, we need to also render the correct icon for | |
| 114 // the stop/insensitive since we won't be using |stop_| to render the icon. | |
| 115 UpdateThemeButtons(); | |
| 116 | |
| 117 // Go ahead and change to reload after a bit, which allows repeated reloads | |
| 118 // without moving the mouse. | |
| 119 if (!stop_to_reload_timer_.IsRunning()) { | |
| 120 stop_to_reload_timer_.Start(stop_to_reload_timer_delay_, this, | |
| 121 &ReloadButtonGtk::OnStopToReloadTimer); | |
| 122 } | |
| 123 } | |
| 124 } | |
| 125 | |
| 126 //////////////////////////////////////////////////////////////////////////////// | |
| 127 // ReloadButtonGtk, NotificationObserver implementation: | |
| 128 | |
| 129 void ReloadButtonGtk::Observe(NotificationType type, | |
| 130 const NotificationSource& source, | |
| 131 const NotificationDetails& /* details */) { | |
| 132 DCHECK(NotificationType::BROWSER_THEME_CHANGED == type); | |
| 133 | |
| 134 GtkThemeProvider* provider = static_cast<GtkThemeProvider*>( | |
| 135 Source<GtkThemeProvider>(source).ptr()); | |
| 136 DCHECK_EQ(provider, theme_provider_); | |
| 137 GtkButtonWidth = 0; | |
| 138 UpdateThemeButtons(); | |
| 139 } | |
| 140 | |
| 141 //////////////////////////////////////////////////////////////////////////////// | |
| 142 // ReloadButtonGtk, private: | |
| 143 | |
| 144 void ReloadButtonGtk::OnClicked(GtkWidget* /* sender */) { | |
| 145 if (visible_mode_ == MODE_STOP) { | |
| 146 // Do nothing if Stop was disabled due to an attempt to change back to | |
| 147 // RELOAD mode while hovered. | |
| 148 if (stop_.paint_override() == GTK_STATE_INSENSITIVE) | |
| 149 return; | |
| 150 | |
| 151 if (browser_) | |
| 152 browser_->Stop(); | |
| 153 | |
| 154 // The user has clicked, so we can feel free to update the button, | |
| 155 // even if the mouse is still hovering. | |
| 156 ChangeMode(MODE_RELOAD, true); | |
| 157 } else if (!double_click_timer_.IsRunning()) { | |
| 158 // Shift-clicking or Ctrl-clicking the reload button means we should ignore | |
| 159 // any cached content. | |
| 160 int command; | |
| 161 GdkModifierType modifier_state; | |
| 162 gtk_get_current_event_state(&modifier_state); | |
| 163 guint modifier_state_uint = modifier_state; | |
| 164 if (modifier_state_uint & (GDK_SHIFT_MASK | GDK_CONTROL_MASK)) { | |
| 165 command = IDC_RELOAD_IGNORING_CACHE; | |
| 166 // Mask off Shift and Control so they don't affect the disposition below. | |
| 167 modifier_state_uint &= ~(GDK_SHIFT_MASK | GDK_CONTROL_MASK); | |
| 168 } else { | |
| 169 command = IDC_RELOAD; | |
| 170 } | |
| 171 | |
| 172 WindowOpenDisposition disposition = | |
| 173 event_utils::DispositionFromEventFlags(modifier_state_uint); | |
| 174 if ((disposition == CURRENT_TAB) && location_bar_) { | |
| 175 // Forcibly reset the location bar, since otherwise it won't discard any | |
| 176 // ongoing user edits, since it doesn't realize this is a user-initiated | |
| 177 // action. | |
| 178 location_bar_->Revert(); | |
| 179 } | |
| 180 | |
| 181 // Start a timer - while this timer is running, the reload button cannot be | |
| 182 // changed to a stop button. We do not set |intended_mode_| to MODE_STOP | |
| 183 // here as the browser will do that when it actually starts loading (which | |
| 184 // may happen synchronously, thus the need to do this before telling the | |
| 185 // browser to execute the reload command). | |
| 186 double_click_timer_.Start(double_click_timer_delay_, this, | |
| 187 &ReloadButtonGtk::OnDoubleClickTimer); | |
| 188 | |
| 189 if (browser_) | |
| 190 browser_->ExecuteCommandWithDisposition(command, disposition); | |
| 191 ++testing_reload_count_; | |
| 192 } | |
| 193 } | |
| 194 | |
| 195 gboolean ReloadButtonGtk::OnExpose(GtkWidget* widget, | |
| 196 GdkEventExpose* e) { | |
| 197 if (theme_provider_ && theme_provider_->UseGtkTheme()) | |
| 198 return FALSE; | |
| 199 return ((visible_mode_ == MODE_RELOAD) ? reload_ : stop_).OnExpose( | |
| 200 widget, e, hover_controller_.GetCurrentValue()); | |
| 201 } | |
| 202 | |
| 203 gboolean ReloadButtonGtk::OnLeaveNotify(GtkWidget* /* widget */, | |
| 204 GdkEventCrossing* /* event */) { | |
| 205 ChangeMode(intended_mode_, true); | |
| 206 return FALSE; | |
| 207 } | |
| 208 | |
| 209 gboolean ReloadButtonGtk::OnQueryTooltip(GtkWidget* /* sender */, | |
| 210 gint /* x */, | |
| 211 gint /* y */, | |
| 212 gboolean /* keyboard_mode */, | |
| 213 GtkTooltip* tooltip) { | |
| 214 // |location_bar_| can be NULL in tests. | |
| 215 if (!location_bar_) | |
| 216 return FALSE; | |
| 217 | |
| 218 gtk_tooltip_set_text(tooltip, l10n_util::GetStringUTF8( | |
| 219 (visible_mode_ == MODE_RELOAD) ? | |
| 220 IDS_TOOLTIP_RELOAD : IDS_TOOLTIP_STOP).c_str()); | |
| 221 return TRUE; | |
| 222 } | |
| 223 | |
| 224 void ReloadButtonGtk::UpdateThemeButtons() { | |
| 225 bool use_gtk = theme_provider_ && theme_provider_->UseGtkTheme(); | |
| 226 | |
| 227 if (use_gtk) { | |
| 228 gtk_widget_ensure_style(widget()); | |
| 229 GtkIconSet* icon_set = gtk_style_lookup_icon_set( | |
| 230 widget()->style, | |
| 231 (visible_mode_ == MODE_RELOAD) ? GTK_STOCK_REFRESH : GTK_STOCK_STOP); | |
| 232 if (icon_set) { | |
| 233 GtkStateType state = static_cast<GtkStateType>( | |
| 234 GTK_WIDGET_STATE(widget())); | |
| 235 if (visible_mode_ == MODE_STOP && stop_.paint_override() != -1) | |
| 236 state = static_cast<GtkStateType>(stop_.paint_override()); | |
| 237 | |
| 238 GdkPixbuf* pixbuf = gtk_icon_set_render_icon( | |
| 239 icon_set, | |
| 240 widget()->style, | |
| 241 gtk_widget_get_direction(widget()), | |
| 242 state, | |
| 243 GTK_ICON_SIZE_SMALL_TOOLBAR, | |
| 244 widget(), | |
| 245 NULL); | |
| 246 | |
| 247 gtk_button_set_image(GTK_BUTTON(widget()), | |
| 248 gtk_image_new_from_pixbuf(pixbuf)); | |
| 249 g_object_unref(pixbuf); | |
| 250 } | |
| 251 | |
| 252 gtk_widget_set_size_request(widget(), -1, -1); | |
| 253 GtkRequisition req; | |
| 254 gtk_widget_size_request(widget(), &req); | |
| 255 GtkButtonWidth = std::max(GtkButtonWidth, req.width); | |
| 256 gtk_widget_set_size_request(widget(), GtkButtonWidth, -1); | |
| 257 | |
| 258 gtk_widget_set_app_paintable(widget(), FALSE); | |
| 259 gtk_widget_set_double_buffered(widget(), TRUE); | |
| 260 } else { | |
| 261 gtk_button_set_image(GTK_BUTTON(widget()), NULL); | |
| 262 | |
| 263 gtk_widget_set_size_request(widget(), reload_.Width(), reload_.Height()); | |
| 264 | |
| 265 gtk_widget_set_app_paintable(widget(), TRUE); | |
| 266 // We effectively double-buffer by virtue of having only one image... | |
| 267 gtk_widget_set_double_buffered(widget(), FALSE); | |
| 268 } | |
| 269 | |
| 270 gtk_chrome_button_set_use_gtk_rendering(GTK_CHROME_BUTTON(widget()), use_gtk); | |
| 271 } | |
| 272 | |
| 273 void ReloadButtonGtk::OnDoubleClickTimer() { | |
| 274 ChangeMode(intended_mode_, false); | |
| 275 } | |
| 276 | |
| 277 void ReloadButtonGtk::OnStopToReloadTimer() { | |
| 278 ChangeMode(intended_mode_, true); | |
| 279 } | |
| OLD | NEW |