| OLD | NEW |
| 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2011 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #ifndef CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ | 5 #ifndef CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ |
| 6 #define CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ | 6 #define CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ |
| 7 #pragma once | 7 #pragma once |
| 8 | 8 |
| 9 #include <gtk/gtk.h> | 9 #include "chrome/browser/ui/gtk/tabs/tab_renderer_gtk.h" |
| 10 #include <map> | 10 // TODO(msw): remove this file once all includes have been updated. |
| 11 | |
| 12 #include "app/gtk_signal.h" | |
| 13 #include "base/basictypes.h" | |
| 14 #include "base/scoped_ptr.h" | |
| 15 #include "base/string16.h" | |
| 16 #include "chrome/browser/gtk/owned_widget_gtk.h" | |
| 17 #include "chrome/common/notification_observer.h" | |
| 18 #include "chrome/common/notification_registrar.h" | |
| 19 #include "gfx/canvas.h" | |
| 20 #include "gfx/font.h" | |
| 21 #include "gfx/rect.h" | |
| 22 #include "third_party/skia/include/core/SkBitmap.h" | |
| 23 #include "ui/base/animation/animation_delegate.h" | |
| 24 | |
| 25 namespace gfx { | |
| 26 class Size; | |
| 27 } // namespace gfx | |
| 28 | |
| 29 class CustomDrawButton; | |
| 30 class GtkThemeProvider; | |
| 31 class TabContents; | |
| 32 class ThemeProvider; | |
| 33 | |
| 34 namespace ui { | |
| 35 class SlideAnimation; | |
| 36 class ThrobAnimation; | |
| 37 } | |
| 38 | |
| 39 class TabRendererGtk : public ui::AnimationDelegate, | |
| 40 public NotificationObserver { | |
| 41 public: | |
| 42 // Possible animation states. | |
| 43 enum AnimationState { | |
| 44 ANIMATION_NONE, | |
| 45 ANIMATION_WAITING, | |
| 46 ANIMATION_LOADING | |
| 47 }; | |
| 48 | |
| 49 class LoadingAnimation : public NotificationObserver { | |
| 50 public: | |
| 51 struct Data { | |
| 52 explicit Data(ThemeProvider* theme_provider); | |
| 53 Data(int loading, int waiting, int waiting_to_loading); | |
| 54 | |
| 55 SkBitmap* waiting_animation_frames; | |
| 56 SkBitmap* loading_animation_frames; | |
| 57 int loading_animation_frame_count; | |
| 58 int waiting_animation_frame_count; | |
| 59 int waiting_to_loading_frame_count_ratio; | |
| 60 }; | |
| 61 | |
| 62 explicit LoadingAnimation(ThemeProvider* theme_provider); | |
| 63 | |
| 64 // Used in unit tests to inject specific data. | |
| 65 explicit LoadingAnimation(const LoadingAnimation::Data& data); | |
| 66 | |
| 67 virtual ~LoadingAnimation(); | |
| 68 | |
| 69 // Advance the loading animation to the next frame, or hide the animation if | |
| 70 // the tab isn't loading. Returns |true| if the icon area needs to be | |
| 71 // repainted. | |
| 72 bool ValidateLoadingAnimation(AnimationState animation_state); | |
| 73 | |
| 74 AnimationState animation_state() const { return animation_state_; } | |
| 75 int animation_frame() const { return animation_frame_; } | |
| 76 | |
| 77 const SkBitmap* waiting_animation_frames() const { | |
| 78 return data_->waiting_animation_frames; | |
| 79 } | |
| 80 const SkBitmap* loading_animation_frames() const { | |
| 81 return data_->loading_animation_frames; | |
| 82 } | |
| 83 | |
| 84 // Provide NotificationObserver implementation. | |
| 85 virtual void Observe(NotificationType type, | |
| 86 const NotificationSource& source, | |
| 87 const NotificationDetails& details); | |
| 88 | |
| 89 private: | |
| 90 scoped_ptr<Data> data_; | |
| 91 | |
| 92 // Used to listen for theme change notifications. | |
| 93 NotificationRegistrar registrar_; | |
| 94 | |
| 95 // Gives us our throbber images. | |
| 96 ThemeProvider* theme_provider_; | |
| 97 | |
| 98 // Current state of the animation. | |
| 99 AnimationState animation_state_; | |
| 100 | |
| 101 // The current index into the Animation image strip. | |
| 102 int animation_frame_; | |
| 103 | |
| 104 DISALLOW_COPY_AND_ASSIGN(LoadingAnimation); | |
| 105 }; | |
| 106 | |
| 107 explicit TabRendererGtk(ThemeProvider* theme_provider); | |
| 108 virtual ~TabRendererGtk(); | |
| 109 | |
| 110 // TabContents. If only the loading state was updated, the loading_only flag | |
| 111 // should be specified. If other things change, set this flag to false to | |
| 112 // update everything. | |
| 113 virtual void UpdateData(TabContents* contents, bool app, bool loading_only); | |
| 114 | |
| 115 // Sets the blocked state of the tab. | |
| 116 void SetBlocked(bool pinned); | |
| 117 bool is_blocked() const; | |
| 118 | |
| 119 // Sets the mini-state of the tab. | |
| 120 void set_mini(bool mini) { data_.mini = mini; } | |
| 121 bool mini() const { return data_.mini; } | |
| 122 | |
| 123 // Sets the app state of the tab. | |
| 124 void set_app(bool app) { data_.app = app; } | |
| 125 bool app() const { return data_.app; } | |
| 126 | |
| 127 // Are we in the process of animating a mini tab state change on this tab? | |
| 128 void set_animating_mini_change(bool value) { | |
| 129 data_.animating_mini_change = value; | |
| 130 } | |
| 131 | |
| 132 // Updates the display to reflect the contents of this TabRenderer's model. | |
| 133 void UpdateFromModel(); | |
| 134 | |
| 135 // Returns true if the Tab is selected, false otherwise. | |
| 136 virtual bool IsSelected() const; | |
| 137 | |
| 138 // Returns true if the Tab is visible, false otherwise. | |
| 139 virtual bool IsVisible() const; | |
| 140 | |
| 141 // Sets the visibility of the Tab. | |
| 142 virtual void SetVisible(bool visible) const; | |
| 143 | |
| 144 // Paints the tab into |canvas|. | |
| 145 virtual void Paint(gfx::Canvas* canvas); | |
| 146 | |
| 147 // Paints the tab into a SkBitmap. | |
| 148 virtual SkBitmap PaintBitmap(); | |
| 149 | |
| 150 // Paints the tab, and keeps the result server-side. The returned surface must | |
| 151 // be freed with cairo_surface_destroy(). | |
| 152 virtual cairo_surface_t* PaintToSurface(); | |
| 153 | |
| 154 // There is no PaintNow available, so the fastest we can do is schedule a | |
| 155 // paint with the windowing system. | |
| 156 virtual void SchedulePaint(); | |
| 157 | |
| 158 // Notifies the Tab that the close button has been clicked. | |
| 159 virtual void CloseButtonClicked(); | |
| 160 | |
| 161 // Sets the bounds of the tab. | |
| 162 virtual void SetBounds(const gfx::Rect& bounds); | |
| 163 | |
| 164 // Provide NotificationObserver implementation. | |
| 165 virtual void Observe(NotificationType type, | |
| 166 const NotificationSource& source, | |
| 167 const NotificationDetails& details); | |
| 168 | |
| 169 // Advance the loading animation to the next frame, or hide the animation if | |
| 170 // the tab isn't loading. Returns |true| if the icon area needs to be | |
| 171 // repainted. | |
| 172 bool ValidateLoadingAnimation(AnimationState animation_state); | |
| 173 | |
| 174 // Repaint only the area of the tab that contains the favicon. | |
| 175 void PaintFavIconArea(GdkEventExpose* event); | |
| 176 | |
| 177 // Returns whether the Tab should display a favicon. | |
| 178 bool ShouldShowIcon() const; | |
| 179 | |
| 180 // Returns the minimum possible size of a single unselected Tab. | |
| 181 static gfx::Size GetMinimumUnselectedSize(); | |
| 182 // Returns the minimum possible size of a selected Tab. Selected tabs must | |
| 183 // always show a close button and have a larger minimum size than unselected | |
| 184 // tabs. | |
| 185 static gfx::Size GetMinimumSelectedSize(); | |
| 186 // Returns the preferred size of a single Tab, assuming space is | |
| 187 // available. | |
| 188 static gfx::Size GetStandardSize(); | |
| 189 | |
| 190 // Returns the width for mini-tabs. Mini-tabs always have this width. | |
| 191 static int GetMiniWidth(); | |
| 192 | |
| 193 // Loads the images to be used for the tab background. | |
| 194 static void LoadTabImages(); | |
| 195 | |
| 196 // Sets the colors used for painting text on the tabs. | |
| 197 static void SetSelectedTitleColor(SkColor color); | |
| 198 static void SetUnselectedTitleColor(SkColor color); | |
| 199 | |
| 200 static gfx::Font* title_font() { return title_font_; } | |
| 201 | |
| 202 // Returns the bounds of the Tab. | |
| 203 int x() const { return bounds_.x(); } | |
| 204 int y() const { return bounds_.y(); } | |
| 205 int width() const { return bounds_.width(); } | |
| 206 int height() const { return bounds_.height(); } | |
| 207 | |
| 208 gfx::Rect bounds() const { return bounds_; } | |
| 209 | |
| 210 gfx::Rect favicon_bounds() const { return favicon_bounds_; } | |
| 211 | |
| 212 // Returns the non-mirrored (LTR) bounds of this tab. | |
| 213 gfx::Rect GetNonMirroredBounds(GtkWidget* parent) const; | |
| 214 | |
| 215 // Returns the requested bounds of the tab. | |
| 216 gfx::Rect GetRequisition() const; | |
| 217 | |
| 218 GtkWidget* widget() const { return tab_.get(); } | |
| 219 | |
| 220 // Start/stop the mini-tab title animation. | |
| 221 void StartMiniTabTitleAnimation(); | |
| 222 void StopMiniTabTitleAnimation(); | |
| 223 | |
| 224 void set_vertical_offset(int offset) { background_offset_y_ = offset; } | |
| 225 | |
| 226 protected: | |
| 227 const gfx::Rect& title_bounds() const { return title_bounds_; } | |
| 228 const gfx::Rect& close_button_bounds() const { return close_button_bounds_; } | |
| 229 | |
| 230 // Returns the title of the Tab. | |
| 231 string16 GetTitle() const; | |
| 232 | |
| 233 // enter-notify-event handler that signals when the mouse enters the tab. | |
| 234 CHROMEGTK_CALLBACK_1(TabRendererGtk, gboolean, OnEnterNotifyEvent, | |
| 235 GdkEventCrossing*); | |
| 236 | |
| 237 // leave-notify-event handler that signals when the mouse enters the tab. | |
| 238 CHROMEGTK_CALLBACK_1(TabRendererGtk, gboolean, OnLeaveNotifyEvent, | |
| 239 GdkEventCrossing*); | |
| 240 | |
| 241 private: | |
| 242 class FavIconCrashAnimation; | |
| 243 | |
| 244 // The data structure used to hold cached bitmaps. We need to manually free | |
| 245 // the bitmap in CachedBitmap when we remove it from |cached_bitmaps_|. We | |
| 246 // handle this when we replace images in the map and in the destructor. | |
| 247 struct CachedBitmap { | |
| 248 int bg_offset_x; | |
| 249 int bg_offset_y; | |
| 250 SkBitmap* bitmap; | |
| 251 }; | |
| 252 typedef std::map<std::pair<const SkBitmap*, const SkBitmap*>, CachedBitmap> | |
| 253 BitmapCache; | |
| 254 | |
| 255 // Model data. We store this here so that we don't need to ask the underlying | |
| 256 // model, which is tricky since instances of this object can outlive the | |
| 257 // corresponding objects in the underlying model. | |
| 258 struct TabData { | |
| 259 TabData() | |
| 260 : is_default_favicon(false), | |
| 261 loading(false), | |
| 262 crashed(false), | |
| 263 off_the_record(false), | |
| 264 show_icon(true), | |
| 265 mini(false), | |
| 266 blocked(false), | |
| 267 animating_mini_change(false), | |
| 268 app(false) { | |
| 269 } | |
| 270 | |
| 271 SkBitmap favicon; | |
| 272 bool is_default_favicon; | |
| 273 string16 title; | |
| 274 bool loading; | |
| 275 bool crashed; | |
| 276 bool off_the_record; | |
| 277 bool show_icon; | |
| 278 bool mini; | |
| 279 bool blocked; | |
| 280 bool animating_mini_change; | |
| 281 bool app; | |
| 282 }; | |
| 283 | |
| 284 // TODO(jhawkins): Move into TabResources class. | |
| 285 struct TabImage { | |
| 286 SkBitmap* image_l; | |
| 287 SkBitmap* image_c; | |
| 288 SkBitmap* image_r; | |
| 289 int l_width; | |
| 290 int r_width; | |
| 291 int y_offset; | |
| 292 }; | |
| 293 | |
| 294 // Overridden from ui::AnimationDelegate: | |
| 295 virtual void AnimationProgressed(const ui::Animation* animation); | |
| 296 virtual void AnimationCanceled(const ui::Animation* animation); | |
| 297 virtual void AnimationEnded(const ui::Animation* animation); | |
| 298 | |
| 299 // Starts/Stops the crash animation. | |
| 300 void StartCrashAnimation(); | |
| 301 void StopCrashAnimation(); | |
| 302 | |
| 303 // Return true if the crash animation is currently running. | |
| 304 bool IsPerformingCrashAnimation() const; | |
| 305 | |
| 306 // Set the temporary offset for the favicon. This is used during animation. | |
| 307 void SetFavIconHidingOffset(int offset); | |
| 308 | |
| 309 void DisplayCrashedFavIcon(); | |
| 310 void ResetCrashedFavIcon(); | |
| 311 | |
| 312 // Generates the bounds for the interior items of the tab. | |
| 313 void Layout(); | |
| 314 | |
| 315 // Returns the local bounds of the tab. This returns the rect | |
| 316 // {0, 0, width(), height()} for now, as we don't yet support borders. | |
| 317 gfx::Rect GetLocalBounds(); | |
| 318 | |
| 319 // Moves the close button widget within the GtkFixed container. | |
| 320 void MoveCloseButtonWidget(); | |
| 321 | |
| 322 // Returns the largest of the favicon, title text, and the close button. | |
| 323 static int GetContentHeight(); | |
| 324 | |
| 325 // A helper method for generating the masked bitmaps used to draw the curved | |
| 326 // edges of tabs. We cache the generated bitmaps because they can take a | |
| 327 // long time to compute. | |
| 328 SkBitmap* GetMaskedBitmap(const SkBitmap* mask, | |
| 329 const SkBitmap* background, | |
| 330 int bg_offset_x, | |
| 331 int bg_offset_y); | |
| 332 BitmapCache cached_bitmaps_; | |
| 333 | |
| 334 // Paints the tab, minus the close button. | |
| 335 void PaintTab(GdkEventExpose* event); | |
| 336 | |
| 337 // Paint various portions of the Tab | |
| 338 void PaintTitle(gfx::Canvas* canvas); | |
| 339 void PaintIcon(gfx::Canvas* canvas); | |
| 340 void PaintTabBackground(gfx::Canvas* canvas); | |
| 341 void PaintInactiveTabBackground(gfx::Canvas* canvas); | |
| 342 void PaintActiveTabBackground(gfx::Canvas* canvas); | |
| 343 void PaintLoadingAnimation(gfx::Canvas* canvas); | |
| 344 | |
| 345 // Returns the number of favicon-size elements that can fit in the tab's | |
| 346 // current size. | |
| 347 int IconCapacity() const; | |
| 348 | |
| 349 | |
| 350 // Returns whether the Tab should display a close button. | |
| 351 bool ShouldShowCloseBox() const; | |
| 352 | |
| 353 CustomDrawButton* MakeCloseButton(); | |
| 354 | |
| 355 // Gets the throb value for the tab. When a tab is not selected the | |
| 356 // active background is drawn at |GetThrobValue()|%. This is used for hover | |
| 357 // and mini-tab title change effects. | |
| 358 double GetThrobValue(); | |
| 359 | |
| 360 // Handles the clicked signal for the close button. | |
| 361 CHROMEGTK_CALLBACK_0(TabRendererGtk, void, OnCloseButtonClicked); | |
| 362 | |
| 363 // Handles middle clicking the close button. | |
| 364 CHROMEGTK_CALLBACK_1(TabRendererGtk, gboolean, OnCloseButtonMouseRelease, | |
| 365 GdkEventButton*); | |
| 366 | |
| 367 // expose-event handler that redraws the tab. | |
| 368 CHROMEGTK_CALLBACK_1(TabRendererGtk, gboolean, OnExposeEvent, | |
| 369 GdkEventExpose*); | |
| 370 | |
| 371 // size-allocate handler used to update the current bounds of the tab. | |
| 372 CHROMEGTK_CALLBACK_1(TabRendererGtk, void, OnSizeAllocate, GtkAllocation*); | |
| 373 | |
| 374 // TODO(jhawkins): Move to TabResources. | |
| 375 static void InitResources(); | |
| 376 static bool initialized_; | |
| 377 | |
| 378 // The bounds of various sections of the display. | |
| 379 gfx::Rect favicon_bounds_; | |
| 380 gfx::Rect title_bounds_; | |
| 381 gfx::Rect close_button_bounds_; | |
| 382 | |
| 383 TabData data_; | |
| 384 | |
| 385 static TabImage tab_active_; | |
| 386 static TabImage tab_inactive_; | |
| 387 static TabImage tab_alpha_; | |
| 388 | |
| 389 static gfx::Font* title_font_; | |
| 390 static int title_font_height_; | |
| 391 | |
| 392 static int close_button_width_; | |
| 393 static int close_button_height_; | |
| 394 | |
| 395 static SkColor selected_title_color_; | |
| 396 static SkColor unselected_title_color_; | |
| 397 | |
| 398 // The GtkDrawingArea we draw the tab on. | |
| 399 OwnedWidgetGtk tab_; | |
| 400 | |
| 401 // Whether we're showing the icon. It is cached so that we can detect when it | |
| 402 // changes and layout appropriately. | |
| 403 bool showing_icon_; | |
| 404 | |
| 405 // Whether we are showing the close button. It is cached so that we can | |
| 406 // detect when it changes and layout appropriately. | |
| 407 bool showing_close_button_; | |
| 408 | |
| 409 // The offset used to animate the favicon location. | |
| 410 int fav_icon_hiding_offset_; | |
| 411 | |
| 412 // The animation object used to swap the favicon with the sad tab icon. | |
| 413 scoped_ptr<FavIconCrashAnimation> crash_animation_; | |
| 414 | |
| 415 // Set when the crashed favicon should be displayed. | |
| 416 bool should_display_crashed_favicon_; | |
| 417 | |
| 418 // The bounds of this Tab. | |
| 419 gfx::Rect bounds_; | |
| 420 | |
| 421 // The requested bounds of this tab. These bounds are relative to the | |
| 422 // tabstrip. | |
| 423 gfx::Rect requisition_; | |
| 424 | |
| 425 // Hover animation. | |
| 426 scoped_ptr<ui::SlideAnimation> hover_animation_; | |
| 427 | |
| 428 // Animation used when the title of an inactive mini-tab changes. | |
| 429 scoped_ptr<ui::ThrobAnimation> mini_title_animation_; | |
| 430 | |
| 431 // Contains the loading animation state. | |
| 432 LoadingAnimation loading_animation_; | |
| 433 | |
| 434 // The offset used to paint the tab theme images. | |
| 435 int background_offset_x_; | |
| 436 | |
| 437 // The vertical offset used to paint the tab theme images. Controlled by the | |
| 438 // tabstrip and plumbed here to offset the theme image by the size of the | |
| 439 // alignment in the BrowserTitlebar. | |
| 440 int background_offset_y_; | |
| 441 | |
| 442 GtkThemeProvider* theme_provider_; | |
| 443 | |
| 444 // The close button. | |
| 445 scoped_ptr<CustomDrawButton> close_button_; | |
| 446 | |
| 447 // The current color of the close button. | |
| 448 SkColor close_button_color_; | |
| 449 | |
| 450 // Used to listen for theme change notifications. | |
| 451 NotificationRegistrar registrar_; | |
| 452 | |
| 453 DISALLOW_COPY_AND_ASSIGN(TabRendererGtk); | |
| 454 }; | |
| 455 | 11 |
| 456 #endif // CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ | 12 #endif // CHROME_BROWSER_GTK_TABS_TAB_RENDERER_GTK_H_ |
| OLD | NEW |