Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(139)

Side by Side Diff: chrome/browser/gtk/tabs/tab_strip_gtk.cc

Issue 6251001: Move chrome/browser/gtk/ to chrome/browser/ui/gtk/... (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src/
Patch Set: '' Created 9 years, 11 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « chrome/browser/gtk/tabs/tab_strip_gtk.h ('k') | chrome/browser/gtk/tabstrip_origin_provider.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
(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/tabs/tab_strip_gtk.h"
6
7 #include <algorithm>
8
9 #include "app/gtk_dnd_util.h"
10 #include "app/resource_bundle.h"
11 #include "base/i18n/rtl.h"
12 #include "base/string_util.h"
13 #include "base/utf_string_conversions.h"
14 #include "chrome/browser/autocomplete/autocomplete.h"
15 #include "chrome/browser/autocomplete/autocomplete_classifier.h"
16 #include "chrome/browser/autocomplete/autocomplete_match.h"
17 #include "chrome/browser/gtk/browser_window_gtk.h"
18 #include "chrome/browser/gtk/custom_button.h"
19 #include "chrome/browser/gtk/gtk_theme_provider.h"
20 #include "chrome/browser/gtk/gtk_util.h"
21 #include "chrome/browser/gtk/tabs/dragged_tab_controller_gtk.h"
22 #include "chrome/browser/profiles/profile.h"
23 #include "chrome/browser/tab_contents/tab_contents.h"
24 #include "chrome/browser/tabs/tab_strip_model_delegate.h"
25 #include "chrome/browser/themes/browser_theme_provider.h"
26 #include "chrome/browser/ui/browser.h"
27 #include "chrome/browser/ui/browser_navigator.h"
28 #include "chrome/browser/ui/tab_contents/tab_contents_wrapper.h"
29 #include "chrome/common/notification_service.h"
30 #include "chrome/common/notification_type.h"
31 #include "gfx/gtk_util.h"
32 #include "gfx/point.h"
33 #include "grit/app_resources.h"
34 #include "grit/theme_resources.h"
35 #include "ui/base/animation/animation_delegate.h"
36 #include "ui/base/animation/slide_animation.h"
37
38 namespace {
39
40 const int kDefaultAnimationDurationMs = 100;
41 const int kResizeLayoutAnimationDurationMs = 166;
42 const int kReorderAnimationDurationMs = 166;
43 const int kAnimateToBoundsDurationMs = 150;
44 const int kMiniTabAnimationDurationMs = 150;
45
46 const int kNewTabButtonHOffset = -5;
47 const int kNewTabButtonVOffset = 5;
48
49 // The delay between when the mouse leaves the tabstrip and the resize animation
50 // is started.
51 const int kResizeTabsTimeMs = 300;
52
53 // The range outside of the tabstrip where the pointer must enter/leave to
54 // start/stop the resize animation.
55 const int kTabStripAnimationVSlop = 40;
56
57 const int kHorizontalMoveThreshold = 16; // pixels
58
59 // The horizontal offset from one tab to the next, which results in overlapping
60 // tabs.
61 const int kTabHOffset = -16;
62
63 // A linux specific menu item for toggling window decorations.
64 const int kShowWindowDecorationsCommand = 200;
65
66 // Size of the drop indicator.
67 static int drop_indicator_width;
68 static int drop_indicator_height;
69
70 inline int Round(double x) {
71 return static_cast<int>(x + 0.5);
72 }
73
74 // widget->allocation is not guaranteed to be set. After window creation,
75 // we pick up the normal bounds by connecting to the configure-event signal.
76 gfx::Rect GetInitialWidgetBounds(GtkWidget* widget) {
77 GtkRequisition request;
78 gtk_widget_size_request(widget, &request);
79 return gfx::Rect(0, 0, request.width, request.height);
80 }
81
82 // Sort rectangles based on their x position. We don't care about y position
83 // so we don't bother breaking ties.
84 int CompareGdkRectangles(const void* p1, const void* p2) {
85 int p1_x = static_cast<const GdkRectangle*>(p1)->x;
86 int p2_x = static_cast<const GdkRectangle*>(p2)->x;
87 if (p1_x < p2_x)
88 return -1;
89 else if (p1_x == p2_x)
90 return 0;
91 return 1;
92 }
93
94 bool GdkRectMatchesTabFavIconBounds(const GdkRectangle& gdk_rect, TabGtk* tab) {
95 gfx::Rect favicon_bounds = tab->favicon_bounds();
96 return gdk_rect.x == favicon_bounds.x() + tab->x() &&
97 gdk_rect.y == favicon_bounds.y() + tab->y() &&
98 gdk_rect.width == favicon_bounds.width() &&
99 gdk_rect.height == favicon_bounds.height();
100 }
101
102 } // namespace
103
104 ////////////////////////////////////////////////////////////////////////////////
105 //
106 // TabAnimation
107 //
108 // A base class for all TabStrip animations.
109 //
110 class TabStripGtk::TabAnimation : public ui::AnimationDelegate {
111 public:
112 friend class TabStripGtk;
113
114 // Possible types of animation.
115 enum Type {
116 INSERT,
117 REMOVE,
118 MOVE,
119 RESIZE,
120 MINI,
121 MINI_MOVE
122 };
123
124 TabAnimation(TabStripGtk* tabstrip, Type type)
125 : tabstrip_(tabstrip),
126 animation_(this),
127 start_selected_width_(0),
128 start_unselected_width_(0),
129 end_selected_width_(0),
130 end_unselected_width_(0),
131 layout_on_completion_(false),
132 type_(type) {
133 }
134 virtual ~TabAnimation() {}
135
136 Type type() const { return type_; }
137
138 void Start() {
139 animation_.SetSlideDuration(GetDuration());
140 animation_.SetTweenType(ui::Tween::EASE_OUT);
141 if (!animation_.IsShowing()) {
142 animation_.Reset();
143 animation_.Show();
144 }
145 }
146
147 void Stop() {
148 animation_.Stop();
149 }
150
151 void set_layout_on_completion(bool layout_on_completion) {
152 layout_on_completion_ = layout_on_completion;
153 }
154
155 // Retrieves the width for the Tab at the specified index if an animation is
156 // active.
157 static double GetCurrentTabWidth(TabStripGtk* tabstrip,
158 TabStripGtk::TabAnimation* animation,
159 int index) {
160 TabGtk* tab = tabstrip->GetTabAt(index);
161 double tab_width;
162 if (tab->mini()) {
163 tab_width = TabGtk::GetMiniWidth();
164 } else {
165 double unselected, selected;
166 tabstrip->GetCurrentTabWidths(&unselected, &selected);
167 tab_width = tab->IsSelected() ? selected : unselected;
168 }
169
170 if (animation) {
171 double specified_tab_width = animation->GetWidthForTab(index);
172 if (specified_tab_width != -1)
173 tab_width = specified_tab_width;
174 }
175
176 return tab_width;
177 }
178
179 // Overridden from ui::AnimationDelegate:
180 virtual void AnimationProgressed(const ui::Animation* animation) {
181 tabstrip_->AnimationLayout(end_unselected_width_);
182 }
183
184 virtual void AnimationEnded(const ui::Animation* animation) {
185 tabstrip_->FinishAnimation(this, layout_on_completion_);
186 // This object is destroyed now, so we can't do anything else after this.
187 }
188
189 virtual void AnimationCanceled(const ui::Animation* animation) {
190 AnimationEnded(animation);
191 }
192
193 // Returns the gap before the tab at the specified index. Subclass if during
194 // an animation you need to insert a gap before a tab.
195 virtual double GetGapWidth(int index) {
196 return 0;
197 }
198
199 protected:
200 // Returns the duration of the animation.
201 virtual int GetDuration() const {
202 return kDefaultAnimationDurationMs;
203 }
204
205 // Subclasses override to return the width of the Tab at the specified index
206 // at the current animation frame. -1 indicates the default width should be
207 // used for the Tab.
208 virtual double GetWidthForTab(int index) const {
209 return -1; // Use default.
210 }
211
212 // Figure out the desired start and end widths for the specified pre- and
213 // post- animation tab counts.
214 void GenerateStartAndEndWidths(int start_tab_count, int end_tab_count,
215 int start_mini_count,
216 int end_mini_count) {
217 tabstrip_->GetDesiredTabWidths(start_tab_count, start_mini_count,
218 &start_unselected_width_,
219 &start_selected_width_);
220 double standard_tab_width =
221 static_cast<double>(TabRendererGtk::GetStandardSize().width());
222
223 if ((end_tab_count - start_tab_count) > 0 &&
224 start_unselected_width_ < standard_tab_width) {
225 double minimum_tab_width = static_cast<double>(
226 TabRendererGtk::GetMinimumUnselectedSize().width());
227 start_unselected_width_ -= minimum_tab_width / start_tab_count;
228 }
229
230 tabstrip_->GenerateIdealBounds();
231 tabstrip_->GetDesiredTabWidths(end_tab_count, end_mini_count,
232 &end_unselected_width_,
233 &end_selected_width_);
234 }
235
236 TabStripGtk* tabstrip_;
237 ui::SlideAnimation animation_;
238
239 double start_selected_width_;
240 double start_unselected_width_;
241 double end_selected_width_;
242 double end_unselected_width_;
243
244 private:
245 // True if a complete re-layout is required upon completion of the animation.
246 // Subclasses set this if they don't perform a complete layout
247 // themselves and canceling the animation may leave the strip in an
248 // inconsistent state.
249 bool layout_on_completion_;
250
251 const Type type_;
252
253 DISALLOW_COPY_AND_ASSIGN(TabAnimation);
254 };
255
256 ////////////////////////////////////////////////////////////////////////////////
257
258 // Handles insertion of a Tab at |index|.
259 class InsertTabAnimation : public TabStripGtk::TabAnimation {
260 public:
261 explicit InsertTabAnimation(TabStripGtk* tabstrip, int index)
262 : TabAnimation(tabstrip, INSERT),
263 index_(index) {
264 int tab_count = tabstrip->GetTabCount();
265 int end_mini_count = tabstrip->GetMiniTabCount();
266 int start_mini_count = end_mini_count;
267 if (index < end_mini_count)
268 start_mini_count--;
269 GenerateStartAndEndWidths(tab_count - 1, tab_count, start_mini_count,
270 end_mini_count);
271 }
272 virtual ~InsertTabAnimation() {}
273
274 protected:
275 // Overridden from TabStripGtk::TabAnimation:
276 virtual double GetWidthForTab(int index) const {
277 if (index == index_) {
278 bool is_selected = tabstrip_->model()->selected_index() == index;
279 double start_width, target_width;
280 if (index < tabstrip_->GetMiniTabCount()) {
281 start_width = TabGtk::GetMinimumSelectedSize().width();
282 target_width = TabGtk::GetMiniWidth();
283 } else {
284 target_width =
285 is_selected ? end_unselected_width_ : end_selected_width_;
286 start_width =
287 is_selected ? TabGtk::GetMinimumSelectedSize().width() :
288 TabGtk::GetMinimumUnselectedSize().width();
289 }
290
291 double delta = target_width - start_width;
292 if (delta > 0)
293 return start_width + (delta * animation_.GetCurrentValue());
294
295 return start_width;
296 }
297
298 if (tabstrip_->GetTabAt(index)->mini())
299 return TabGtk::GetMiniWidth();
300
301 if (tabstrip_->GetTabAt(index)->IsSelected()) {
302 double delta = end_selected_width_ - start_selected_width_;
303 return start_selected_width_ + (delta * animation_.GetCurrentValue());
304 }
305
306 double delta = end_unselected_width_ - start_unselected_width_;
307 return start_unselected_width_ + (delta * animation_.GetCurrentValue());
308 }
309
310 private:
311 int index_;
312
313 DISALLOW_COPY_AND_ASSIGN(InsertTabAnimation);
314 };
315
316 ////////////////////////////////////////////////////////////////////////////////
317
318 // Handles removal of a Tab from |index|
319 class RemoveTabAnimation : public TabStripGtk::TabAnimation {
320 public:
321 RemoveTabAnimation(TabStripGtk* tabstrip, int index, TabContents* contents)
322 : TabAnimation(tabstrip, REMOVE),
323 index_(index) {
324 int tab_count = tabstrip->GetTabCount();
325 int start_mini_count = tabstrip->GetMiniTabCount();
326 int end_mini_count = start_mini_count;
327 if (index < start_mini_count)
328 end_mini_count--;
329 GenerateStartAndEndWidths(tab_count, tab_count - 1, start_mini_count,
330 end_mini_count);
331 // If the last non-mini-tab is being removed we force a layout on
332 // completion. This is necessary as the value returned by GetTabHOffset
333 // changes once the tab is actually removed (which happens at the end of
334 // the animation), and unless we layout GetTabHOffset won't be called after
335 // the removal.
336 // We do the same when the last mini-tab is being removed for the same
337 // reason.
338 set_layout_on_completion(start_mini_count > 0 &&
339 (end_mini_count == 0 ||
340 (start_mini_count == end_mini_count &&
341 tab_count == start_mini_count + 1)));
342 }
343
344 virtual ~RemoveTabAnimation() {}
345
346 // Returns the index of the tab being removed.
347 int index() const { return index_; }
348
349 protected:
350 // Overridden from TabStripGtk::TabAnimation:
351 virtual double GetWidthForTab(int index) const {
352 TabGtk* tab = tabstrip_->GetTabAt(index);
353
354 if (index == index_) {
355 // The tab(s) being removed are gradually shrunken depending on the state
356 // of the animation.
357 if (tab->mini()) {
358 return animation_.CurrentValueBetween(TabGtk::GetMiniWidth(),
359 -kTabHOffset);
360 }
361
362 // Removed animated Tabs are never selected.
363 double start_width = start_unselected_width_;
364 // Make sure target_width is at least abs(kTabHOffset), otherwise if
365 // less than kTabHOffset during layout tabs get negatively offset.
366 double target_width =
367 std::max(abs(kTabHOffset),
368 TabGtk::GetMinimumUnselectedSize().width() + kTabHOffset);
369 return animation_.CurrentValueBetween(start_width, target_width);
370 }
371
372 if (tab->mini())
373 return TabGtk::GetMiniWidth();
374
375 if (tabstrip_->available_width_for_tabs_ != -1 &&
376 index_ != tabstrip_->GetTabCount() - 1) {
377 return TabStripGtk::TabAnimation::GetWidthForTab(index);
378 }
379
380 // All other tabs are sized according to the start/end widths specified at
381 // the start of the animation.
382 if (tab->IsSelected()) {
383 double delta = end_selected_width_ - start_selected_width_;
384 return start_selected_width_ + (delta * animation_.GetCurrentValue());
385 }
386
387 double delta = end_unselected_width_ - start_unselected_width_;
388 return start_unselected_width_ + (delta * animation_.GetCurrentValue());
389 }
390
391 virtual void AnimationEnded(const ui::Animation* animation) {
392 tabstrip_->RemoveTabAt(index_);
393 TabStripGtk::TabAnimation::AnimationEnded(animation);
394 }
395
396 private:
397 int index_;
398
399 DISALLOW_COPY_AND_ASSIGN(RemoveTabAnimation);
400 };
401
402 ////////////////////////////////////////////////////////////////////////////////
403
404 // Handles the movement of a Tab from one position to another.
405 class MoveTabAnimation : public TabStripGtk::TabAnimation {
406 public:
407 MoveTabAnimation(TabStripGtk* tabstrip, int tab_a_index, int tab_b_index)
408 : TabAnimation(tabstrip, MOVE),
409 start_tab_a_bounds_(tabstrip_->GetIdealBounds(tab_b_index)),
410 start_tab_b_bounds_(tabstrip_->GetIdealBounds(tab_a_index)) {
411 tab_a_ = tabstrip_->GetTabAt(tab_a_index);
412 tab_b_ = tabstrip_->GetTabAt(tab_b_index);
413
414 // Since we don't do a full TabStrip re-layout, we need to force a full
415 // layout upon completion since we're not guaranteed to be in a good state
416 // if for example the animation is canceled.
417 set_layout_on_completion(true);
418 }
419 virtual ~MoveTabAnimation() {}
420
421 // Overridden from ui::AnimationDelegate:
422 virtual void AnimationProgressed(const ui::Animation* animation) {
423 // Position Tab A
424 double distance = start_tab_b_bounds_.x() - start_tab_a_bounds_.x();
425 double delta = distance * animation_.GetCurrentValue();
426 double new_x = start_tab_a_bounds_.x() + delta;
427 gfx::Rect bounds(Round(new_x), start_tab_a_bounds_.y(), tab_a_->width(),
428 tab_a_->height());
429 tabstrip_->SetTabBounds(tab_a_, bounds);
430
431 // Position Tab B
432 distance = start_tab_a_bounds_.x() - start_tab_b_bounds_.x();
433 delta = distance * animation_.GetCurrentValue();
434 new_x = start_tab_b_bounds_.x() + delta;
435 bounds = gfx::Rect(Round(new_x), start_tab_b_bounds_.y(), tab_b_->width(),
436 tab_b_->height());
437 tabstrip_->SetTabBounds(tab_b_, bounds);
438 }
439
440 protected:
441 // Overridden from TabStripGtk::TabAnimation:
442 virtual int GetDuration() const { return kReorderAnimationDurationMs; }
443
444 private:
445 // The two tabs being exchanged.
446 TabGtk* tab_a_;
447 TabGtk* tab_b_;
448
449 // ...and their bounds.
450 gfx::Rect start_tab_a_bounds_;
451 gfx::Rect start_tab_b_bounds_;
452
453 DISALLOW_COPY_AND_ASSIGN(MoveTabAnimation);
454 };
455
456 ////////////////////////////////////////////////////////////////////////////////
457
458 // Handles the animated resize layout of the entire TabStrip from one width
459 // to another.
460 class ResizeLayoutAnimation : public TabStripGtk::TabAnimation {
461 public:
462 explicit ResizeLayoutAnimation(TabStripGtk* tabstrip)
463 : TabAnimation(tabstrip, RESIZE) {
464 int tab_count = tabstrip->GetTabCount();
465 int mini_tab_count = tabstrip->GetMiniTabCount();
466 GenerateStartAndEndWidths(tab_count, tab_count, mini_tab_count,
467 mini_tab_count);
468 InitStartState();
469 }
470 virtual ~ResizeLayoutAnimation() {}
471
472 // Overridden from ui::AnimationDelegate:
473 virtual void AnimationEnded(const ui::Animation* animation) {
474 tabstrip_->needs_resize_layout_ = false;
475 TabStripGtk::TabAnimation::AnimationEnded(animation);
476 }
477
478 protected:
479 // Overridden from TabStripGtk::TabAnimation:
480 virtual int GetDuration() const {
481 return kResizeLayoutAnimationDurationMs;
482 }
483
484 virtual double GetWidthForTab(int index) const {
485 TabGtk* tab = tabstrip_->GetTabAt(index);
486
487 if (tab->mini())
488 return TabGtk::GetMiniWidth();
489
490 if (tab->IsSelected()) {
491 return animation_.CurrentValueBetween(start_selected_width_,
492 end_selected_width_);
493 }
494
495 return animation_.CurrentValueBetween(start_unselected_width_,
496 end_unselected_width_);
497 }
498
499 private:
500 // We need to start from the current widths of the Tabs as they were last
501 // laid out, _not_ the last known good state, which is what'll be done if we
502 // don't measure the Tab sizes here and just go with the default TabAnimation
503 // behavior...
504 void InitStartState() {
505 for (int i = 0; i < tabstrip_->GetTabCount(); ++i) {
506 TabGtk* current_tab = tabstrip_->GetTabAt(i);
507 if (!current_tab->mini()) {
508 if (current_tab->IsSelected()) {
509 start_selected_width_ = current_tab->width();
510 } else {
511 start_unselected_width_ = current_tab->width();
512 }
513 }
514 }
515 }
516
517 DISALLOW_COPY_AND_ASSIGN(ResizeLayoutAnimation);
518 };
519
520 // Handles a tabs mini-state changing while the tab does not change position
521 // in the model.
522 class MiniTabAnimation : public TabStripGtk::TabAnimation {
523 public:
524 explicit MiniTabAnimation(TabStripGtk* tabstrip, int index)
525 : TabAnimation(tabstrip, MINI),
526 index_(index) {
527 int tab_count = tabstrip->GetTabCount();
528 int start_mini_count = tabstrip->GetMiniTabCount();
529 int end_mini_count = start_mini_count;
530 if (tabstrip->GetTabAt(index)->mini())
531 start_mini_count--;
532 else
533 start_mini_count++;
534 tabstrip_->GetTabAt(index)->set_animating_mini_change(true);
535 GenerateStartAndEndWidths(tab_count, tab_count, start_mini_count,
536 end_mini_count);
537 }
538
539 protected:
540 // Overridden from TabStripGtk::TabAnimation:
541 virtual int GetDuration() const {
542 return kMiniTabAnimationDurationMs;
543 }
544
545 virtual double GetWidthForTab(int index) const {
546 TabGtk* tab = tabstrip_->GetTabAt(index);
547
548 if (index == index_) {
549 if (tab->mini()) {
550 return animation_.CurrentValueBetween(
551 start_selected_width_,
552 static_cast<double>(TabGtk::GetMiniWidth()));
553 } else {
554 return animation_.CurrentValueBetween(
555 static_cast<double>(TabGtk::GetMiniWidth()),
556 end_selected_width_);
557 }
558 } else if (tab->mini()) {
559 return TabGtk::GetMiniWidth();
560 }
561
562 if (tab->IsSelected()) {
563 return animation_.CurrentValueBetween(start_selected_width_,
564 end_selected_width_);
565 }
566
567 return animation_.CurrentValueBetween(start_unselected_width_,
568 end_unselected_width_);
569 }
570
571 private:
572 // Index of the tab whose mini-state changed.
573 int index_;
574
575 DISALLOW_COPY_AND_ASSIGN(MiniTabAnimation);
576 };
577
578 ////////////////////////////////////////////////////////////////////////////////
579
580 // Handles the animation when a tabs mini-state changes and the tab moves as a
581 // result.
582 class MiniMoveAnimation : public TabStripGtk::TabAnimation {
583 public:
584 explicit MiniMoveAnimation(TabStripGtk* tabstrip,
585 int from_index,
586 int to_index,
587 const gfx::Rect& start_bounds)
588 : TabAnimation(tabstrip, MINI_MOVE),
589 tab_(tabstrip->GetTabAt(to_index)),
590 start_bounds_(start_bounds),
591 from_index_(from_index),
592 to_index_(to_index) {
593 int tab_count = tabstrip->GetTabCount();
594 int start_mini_count = tabstrip->GetMiniTabCount();
595 int end_mini_count = start_mini_count;
596 if (tabstrip->GetTabAt(to_index)->mini())
597 start_mini_count--;
598 else
599 start_mini_count++;
600 GenerateStartAndEndWidths(tab_count, tab_count, start_mini_count,
601 end_mini_count);
602 target_bounds_ = tabstrip->GetIdealBounds(to_index);
603 tab_->set_animating_mini_change(true);
604 }
605
606 // Overridden from ui::AnimationDelegate:
607 virtual void AnimationProgressed(const ui::Animation* animation) {
608 // Do the normal layout.
609 TabAnimation::AnimationProgressed(animation);
610
611 // Then special case the position of the tab being moved.
612 int x = animation_.CurrentValueBetween(start_bounds_.x(),
613 target_bounds_.x());
614 int width = animation_.CurrentValueBetween(start_bounds_.width(),
615 target_bounds_.width());
616 gfx::Rect tab_bounds(x, start_bounds_.y(), width,
617 start_bounds_.height());
618 tabstrip_->SetTabBounds(tab_, tab_bounds);
619 }
620
621 virtual void AnimationEnded(const ui::Animation* animation) {
622 tabstrip_->needs_resize_layout_ = false;
623 TabStripGtk::TabAnimation::AnimationEnded(animation);
624 }
625
626 virtual double GetGapWidth(int index) {
627 if (to_index_ < from_index_) {
628 // The tab was made mini.
629 if (index == to_index_) {
630 double current_size =
631 animation_.CurrentValueBetween(0, target_bounds_.width());
632 if (current_size < -kTabHOffset)
633 return -(current_size + kTabHOffset);
634 } else if (index == from_index_ + 1) {
635 return animation_.CurrentValueBetween(start_bounds_.width(), 0);
636 }
637 } else {
638 // The tab was was made a normal tab.
639 if (index == from_index_) {
640 return animation_.CurrentValueBetween(
641 TabGtk::GetMiniWidth() + kTabHOffset, 0);
642 }
643 }
644 return 0;
645 }
646
647 protected:
648 // Overridden from TabStripGtk::TabAnimation:
649 virtual int GetDuration() const { return kReorderAnimationDurationMs; }
650
651 virtual double GetWidthForTab(int index) const {
652 TabGtk* tab = tabstrip_->GetTabAt(index);
653
654 if (index == to_index_)
655 return animation_.CurrentValueBetween(0, target_bounds_.width());
656
657 if (tab->mini())
658 return TabGtk::GetMiniWidth();
659
660 if (tab->IsSelected()) {
661 return animation_.CurrentValueBetween(start_selected_width_,
662 end_selected_width_);
663 }
664
665 return animation_.CurrentValueBetween(start_unselected_width_,
666 end_unselected_width_);
667 }
668
669 private:
670 // The tab being moved.
671 TabGtk* tab_;
672
673 // Initial bounds of tab_.
674 gfx::Rect start_bounds_;
675
676 // Target bounds.
677 gfx::Rect target_bounds_;
678
679 // Start and end indices of the tab.
680 int from_index_;
681 int to_index_;
682
683 DISALLOW_COPY_AND_ASSIGN(MiniMoveAnimation);
684 };
685
686 ////////////////////////////////////////////////////////////////////////////////
687 // TabStripGtk, public:
688
689 // static
690 const int TabStripGtk::mini_to_non_mini_gap_ = 3;
691
692 TabStripGtk::TabStripGtk(TabStripModel* model, BrowserWindowGtk* window)
693 : current_unselected_width_(TabGtk::GetStandardSize().width()),
694 current_selected_width_(TabGtk::GetStandardSize().width()),
695 available_width_for_tabs_(-1),
696 needs_resize_layout_(false),
697 tab_vertical_offset_(0),
698 model_(model),
699 window_(window),
700 theme_provider_(GtkThemeProvider::GetFrom(model->profile())),
701 resize_layout_factory_(this),
702 added_as_message_loop_observer_(false) {
703 theme_provider_->InitThemesFor(this);
704 registrar_.Add(this, NotificationType::BROWSER_THEME_CHANGED,
705 NotificationService::AllSources());
706 }
707
708 TabStripGtk::~TabStripGtk() {
709 model_->RemoveObserver(this);
710 tabstrip_.Destroy();
711
712 // Free any remaining tabs. This is needed to free the very last tab,
713 // because it is not animated on close. This also happens when all of the
714 // tabs are closed at once.
715 std::vector<TabData>::iterator iterator = tab_data_.begin();
716 for (; iterator < tab_data_.end(); iterator++) {
717 delete iterator->tab;
718 }
719
720 tab_data_.clear();
721
722 // Make sure we unhook ourselves as a message loop observer so that we don't
723 // crash in the case where the user closes the last tab in a window.
724 RemoveMessageLoopObserver();
725 }
726
727 void TabStripGtk::Init() {
728 model_->AddObserver(this);
729
730 tabstrip_.Own(gtk_fixed_new());
731 ViewIDUtil::SetID(tabstrip_.get(), VIEW_ID_TAB_STRIP);
732 // We want the tab strip to be horizontally shrinkable, so that the Chrome
733 // window can be resized freely.
734 gtk_widget_set_size_request(tabstrip_.get(), 0,
735 TabGtk::GetMinimumUnselectedSize().height());
736 gtk_widget_set_app_paintable(tabstrip_.get(), TRUE);
737 gtk_drag_dest_set(tabstrip_.get(), GTK_DEST_DEFAULT_ALL,
738 NULL, 0,
739 static_cast<GdkDragAction>(
740 GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK));
741 static const int targets[] = { gtk_dnd_util::TEXT_URI_LIST,
742 gtk_dnd_util::NETSCAPE_URL,
743 gtk_dnd_util::TEXT_PLAIN,
744 -1 };
745 gtk_dnd_util::SetDestTargetList(tabstrip_.get(), targets);
746
747 g_signal_connect(tabstrip_.get(), "expose-event",
748 G_CALLBACK(OnExposeThunk), this);
749 g_signal_connect(tabstrip_.get(), "size-allocate",
750 G_CALLBACK(OnSizeAllocateThunk), this);
751 g_signal_connect(tabstrip_.get(), "drag-motion",
752 G_CALLBACK(OnDragMotionThunk), this);
753 g_signal_connect(tabstrip_.get(), "drag-drop",
754 G_CALLBACK(OnDragDropThunk), this);
755 g_signal_connect(tabstrip_.get(), "drag-leave",
756 G_CALLBACK(OnDragLeaveThunk), this);
757 g_signal_connect(tabstrip_.get(), "drag-data-received",
758 G_CALLBACK(OnDragDataReceivedThunk), this);
759
760 newtab_button_.reset(MakeNewTabButton());
761
762 gtk_widget_show_all(tabstrip_.get());
763
764 bounds_ = GetInitialWidgetBounds(tabstrip_.get());
765
766 if (drop_indicator_width == 0) {
767 // Direction doesn't matter, both images are the same size.
768 GdkPixbuf* drop_image = GetDropArrowImage(true);
769 drop_indicator_width = gdk_pixbuf_get_width(drop_image);
770 drop_indicator_height = gdk_pixbuf_get_height(drop_image);
771 }
772
773 ViewIDUtil::SetDelegateForWidget(widget(), this);
774 }
775
776 void TabStripGtk::Show() {
777 gtk_widget_show(tabstrip_.get());
778 }
779
780 void TabStripGtk::Hide() {
781 gtk_widget_hide(tabstrip_.get());
782 }
783
784 bool TabStripGtk::IsActiveDropTarget() const {
785 for (int i = 0; i < GetTabCount(); ++i) {
786 TabGtk* tab = GetTabAt(i);
787 if (tab->dragging())
788 return true;
789 }
790 return false;
791 }
792
793 void TabStripGtk::Layout() {
794 // Called from:
795 // - window resize
796 // - animation completion
797 StopAnimation();
798
799 GenerateIdealBounds();
800 int tab_count = GetTabCount();
801 int tab_right = 0;
802 for (int i = 0; i < tab_count; ++i) {
803 const gfx::Rect& bounds = tab_data_.at(i).ideal_bounds;
804 TabGtk* tab = GetTabAt(i);
805 tab->set_animating_mini_change(false);
806 tab->set_vertical_offset(tab_vertical_offset_);
807 SetTabBounds(tab, bounds);
808 tab_right = bounds.right();
809 tab_right += GetTabHOffset(i + 1);
810 }
811
812 LayoutNewTabButton(static_cast<double>(tab_right), current_unselected_width_);
813 }
814
815 void TabStripGtk::SchedulePaint() {
816 gtk_widget_queue_draw(tabstrip_.get());
817 }
818
819 void TabStripGtk::SetBounds(const gfx::Rect& bounds) {
820 bounds_ = bounds;
821 }
822
823 void TabStripGtk::UpdateLoadingAnimations() {
824 for (int i = 0, index = 0; i < GetTabCount(); ++i, ++index) {
825 TabGtk* current_tab = GetTabAt(i);
826 if (current_tab->closing()) {
827 --index;
828 } else {
829 TabRendererGtk::AnimationState state;
830 TabContentsWrapper* contents = model_->GetTabContentsAt(index);
831 if (!contents || !contents->tab_contents()->is_loading()) {
832 state = TabGtk::ANIMATION_NONE;
833 } else if (contents->tab_contents()->waiting_for_response()) {
834 state = TabGtk::ANIMATION_WAITING;
835 } else {
836 state = TabGtk::ANIMATION_LOADING;
837 }
838 if (current_tab->ValidateLoadingAnimation(state)) {
839 // Queue the tab's icon area to be repainted.
840 gfx::Rect favicon_bounds = current_tab->favicon_bounds();
841 gtk_widget_queue_draw_area(tabstrip_.get(),
842 favicon_bounds.x() + current_tab->x(),
843 favicon_bounds.y() + current_tab->y(),
844 favicon_bounds.width(),
845 favicon_bounds.height());
846 }
847 }
848 }
849 }
850
851 bool TabStripGtk::IsCompatibleWith(TabStripGtk* other) {
852 return model_->profile() == other->model()->profile();
853 }
854
855 bool TabStripGtk::IsAnimating() const {
856 return active_animation_.get() != NULL;
857 }
858
859 void TabStripGtk::DestroyDragController() {
860 drag_controller_.reset();
861 }
862
863 void TabStripGtk::DestroyDraggedSourceTab(TabGtk* tab) {
864 // We could be running an animation that references this Tab.
865 StopAnimation();
866
867 // Make sure we leave the tab_data_ vector in a consistent state, otherwise
868 // we'll be pointing to tabs that have been deleted and removed from the
869 // child view list.
870 std::vector<TabData>::iterator it = tab_data_.begin();
871 for (; it != tab_data_.end(); ++it) {
872 if (it->tab == tab) {
873 if (!model_->closing_all())
874 NOTREACHED() << "Leaving in an inconsistent state!";
875 tab_data_.erase(it);
876 break;
877 }
878 }
879
880 gtk_container_remove(GTK_CONTAINER(tabstrip_.get()), tab->widget());
881 // If we delete the dragged source tab here, the DestroyDragWidget posted
882 // task will be run after the tab is deleted, leading to a crash.
883 MessageLoop::current()->DeleteSoon(FROM_HERE, tab);
884
885 // Force a layout here, because if we've just quickly drag detached a Tab,
886 // the stopping of the active animation above may have left the TabStrip in a
887 // bad (visual) state.
888 Layout();
889 }
890
891 gfx::Rect TabStripGtk::GetIdealBounds(int index) {
892 DCHECK(index >= 0 && index < GetTabCount());
893 return tab_data_.at(index).ideal_bounds;
894 }
895
896 void TabStripGtk::SetVerticalOffset(int offset) {
897 tab_vertical_offset_ = offset;
898 Layout();
899 }
900
901 gfx::Point TabStripGtk::GetTabStripOriginForWidget(GtkWidget* target) {
902 int x, y;
903 if (!gtk_widget_translate_coordinates(widget(), target,
904 -widget()->allocation.x, 0, &x, &y)) {
905 // If the tab strip isn't showing, give the coordinates relative to the
906 // toplevel instead.
907 if (!gtk_widget_translate_coordinates(
908 gtk_widget_get_toplevel(widget()), target, 0, 0, &x, &y)) {
909 NOTREACHED();
910 }
911 }
912 if (GTK_WIDGET_NO_WINDOW(target)) {
913 x += target->allocation.x;
914 y += target->allocation.y;
915 }
916 return gfx::Point(x, y);
917 }
918
919 ////////////////////////////////////////////////////////////////////////////////
920 // ViewIDUtil::Delegate implementation
921
922 GtkWidget* TabStripGtk::GetWidgetForViewID(ViewID view_id) {
923 if (GetTabCount() > 0) {
924 if (view_id == VIEW_ID_TAB_LAST) {
925 return GetTabAt(GetTabCount() - 1)->widget();
926 } else if ((view_id >= VIEW_ID_TAB_0) && (view_id < VIEW_ID_TAB_LAST)) {
927 int index = view_id - VIEW_ID_TAB_0;
928 if (index >= 0 && index < GetTabCount()) {
929 return GetTabAt(index)->widget();
930 } else {
931 return NULL;
932 }
933 }
934 }
935
936 return NULL;
937 }
938
939 ////////////////////////////////////////////////////////////////////////////////
940 // TabStripGtk, TabStripModelObserver implementation:
941
942 void TabStripGtk::TabInsertedAt(TabContentsWrapper* contents,
943 int index,
944 bool foreground) {
945 DCHECK(contents);
946 DCHECK(index == TabStripModel::kNoTab || model_->ContainsIndex(index));
947
948 StopAnimation();
949
950 bool contains_tab = false;
951 TabGtk* tab = NULL;
952 // First see if this Tab is one that was dragged out of this TabStrip and is
953 // now being dragged back in. In this case, the DraggedTabController actually
954 // has the Tab already constructed and we can just insert it into our list
955 // again.
956 if (IsDragSessionActive()) {
957 tab = drag_controller_->GetDragSourceTabForContents(
958 contents->tab_contents());
959 if (tab) {
960 // If the Tab was detached, it would have been animated closed but not
961 // removed, so we need to reset this property.
962 tab->set_closing(false);
963 tab->ValidateLoadingAnimation(TabRendererGtk::ANIMATION_NONE);
964 tab->SetVisible(true);
965 }
966
967 // See if we're already in the list. We don't want to add ourselves twice.
968 std::vector<TabData>::const_iterator iter = tab_data_.begin();
969 for (; iter != tab_data_.end() && !contains_tab; ++iter) {
970 if (iter->tab == tab)
971 contains_tab = true;
972 }
973 }
974
975 if (!tab)
976 tab = new TabGtk(this);
977
978 // Only insert if we're not already in the list.
979 if (!contains_tab) {
980 TabData d = { tab, gfx::Rect() };
981 tab_data_.insert(tab_data_.begin() + index, d);
982 tab->UpdateData(contents->tab_contents(), model_->IsAppTab(index), false);
983 }
984 tab->set_mini(model_->IsMiniTab(index));
985 tab->set_app(model_->IsAppTab(index));
986 tab->SetBlocked(model_->IsTabBlocked(index));
987
988 if (gtk_widget_get_parent(tab->widget()) != tabstrip_.get())
989 gtk_fixed_put(GTK_FIXED(tabstrip_.get()), tab->widget(), 0, 0);
990
991 // Don't animate the first tab; it looks weird.
992 if (GetTabCount() > 1) {
993 StartInsertTabAnimation(index);
994 // We added the tab at 0x0, we need to force an animation step otherwise
995 // if GTK paints before the animation event the tab is painted at 0x0
996 // which is most likely not where it should be positioned.
997 active_animation_->AnimationProgressed(NULL);
998 } else {
999 Layout();
1000 }
1001 }
1002
1003 void TabStripGtk::TabDetachedAt(TabContentsWrapper* contents, int index) {
1004 GenerateIdealBounds();
1005 StartRemoveTabAnimation(index, contents->tab_contents());
1006 // Have to do this _after_ calling StartRemoveTabAnimation, so that any
1007 // previous remove is completed fully and index is valid in sync with the
1008 // model index.
1009 GetTabAt(index)->set_closing(true);
1010 }
1011
1012 void TabStripGtk::TabSelectedAt(TabContentsWrapper* old_contents,
1013 TabContentsWrapper* new_contents,
1014 int index,
1015 bool user_gesture) {
1016 DCHECK(index >= 0 && index < static_cast<int>(GetTabCount()));
1017
1018 // We have "tiny tabs" if the tabs are so tiny that the unselected ones are
1019 // a different size to the selected ones.
1020 bool tiny_tabs = current_unselected_width_ != current_selected_width_;
1021 if (!IsAnimating() && (!needs_resize_layout_ || tiny_tabs))
1022 Layout();
1023
1024 GetTabAt(index)->SchedulePaint();
1025
1026 int old_index = model_->GetIndexOfTabContents(old_contents);
1027 if (old_index >= 0) {
1028 GetTabAt(old_index)->SchedulePaint();
1029 GetTabAt(old_index)->StopMiniTabTitleAnimation();
1030 }
1031 }
1032
1033 void TabStripGtk::TabMoved(TabContentsWrapper* contents,
1034 int from_index,
1035 int to_index) {
1036 gfx::Rect start_bounds = GetIdealBounds(from_index);
1037 TabGtk* tab = GetTabAt(from_index);
1038 tab_data_.erase(tab_data_.begin() + from_index);
1039 TabData data = {tab, gfx::Rect()};
1040 tab->set_mini(model_->IsMiniTab(to_index));
1041 tab->SetBlocked(model_->IsTabBlocked(to_index));
1042 tab_data_.insert(tab_data_.begin() + to_index, data);
1043 GenerateIdealBounds();
1044 StartMoveTabAnimation(from_index, to_index);
1045 }
1046
1047 void TabStripGtk::TabChangedAt(TabContentsWrapper* contents, int index,
1048 TabChangeType change_type) {
1049 // Index is in terms of the model. Need to make sure we adjust that index in
1050 // case we have an animation going.
1051 TabGtk* tab = GetTabAtAdjustForAnimation(index);
1052 if (change_type == TITLE_NOT_LOADING) {
1053 if (tab->mini() && !tab->IsSelected())
1054 tab->StartMiniTabTitleAnimation();
1055 // We'll receive another notification of the change asynchronously.
1056 return;
1057 }
1058 tab->UpdateData(contents->tab_contents(),
1059 model_->IsAppTab(index),
1060 change_type == LOADING_ONLY);
1061 tab->UpdateFromModel();
1062 }
1063
1064 void TabStripGtk::TabReplacedAt(TabStripModel* tab_strip_model,
1065 TabContentsWrapper* old_contents,
1066 TabContentsWrapper* new_contents,
1067 int index) {
1068 TabChangedAt(new_contents, index, ALL);
1069 }
1070
1071 void TabStripGtk::TabMiniStateChanged(TabContentsWrapper* contents, int index) {
1072 // Don't do anything if we've already picked up the change from TabMoved.
1073 if (GetTabAt(index)->mini() == model_->IsMiniTab(index))
1074 return;
1075
1076 GetTabAt(index)->set_mini(model_->IsMiniTab(index));
1077 // Don't animate if the window isn't visible yet. The window won't be visible
1078 // when dragging a mini-tab to a new window.
1079 if (window_ && window_->window() &&
1080 GTK_WIDGET_VISIBLE(GTK_WIDGET(window_->window()))) {
1081 StartMiniTabAnimation(index);
1082 } else {
1083 Layout();
1084 }
1085 }
1086
1087 void TabStripGtk::TabBlockedStateChanged(TabContentsWrapper* contents,
1088 int index) {
1089 GetTabAt(index)->SetBlocked(model_->IsTabBlocked(index));
1090 }
1091
1092 ////////////////////////////////////////////////////////////////////////////////
1093 // TabStripGtk, TabGtk::TabDelegate implementation:
1094
1095 bool TabStripGtk::IsTabSelected(const TabGtk* tab) const {
1096 if (tab->closing())
1097 return false;
1098
1099 return GetIndexOfTab(tab) == model_->selected_index();
1100 }
1101
1102 bool TabStripGtk::IsTabDetached(const TabGtk* tab) const {
1103 if (drag_controller_.get())
1104 return drag_controller_->IsTabDetached(tab);
1105 return false;
1106 }
1107
1108 void TabStripGtk::GetCurrentTabWidths(double* unselected_width,
1109 double* selected_width) const {
1110 *unselected_width = current_unselected_width_;
1111 *selected_width = current_selected_width_;
1112 }
1113
1114 bool TabStripGtk::IsTabPinned(const TabGtk* tab) const {
1115 if (tab->closing())
1116 return false;
1117
1118 return model_->IsTabPinned(GetIndexOfTab(tab));
1119 }
1120
1121 void TabStripGtk::SelectTab(TabGtk* tab) {
1122 int index = GetIndexOfTab(tab);
1123 if (model_->ContainsIndex(index))
1124 model_->SelectTabContentsAt(index, true);
1125 }
1126
1127 void TabStripGtk::CloseTab(TabGtk* tab) {
1128 int tab_index = GetIndexOfTab(tab);
1129 if (model_->ContainsIndex(tab_index)) {
1130 TabGtk* last_tab = GetTabAt(GetTabCount() - 1);
1131 // Limit the width available to the TabStrip for laying out Tabs, so that
1132 // Tabs are not resized until a later time (when the mouse pointer leaves
1133 // the TabStrip).
1134 available_width_for_tabs_ = GetAvailableWidthForTabs(last_tab);
1135 needs_resize_layout_ = true;
1136 // We hook into the message loop in order to receive mouse move events when
1137 // the mouse is outside of the tabstrip. We unhook once the resize layout
1138 // animation is started.
1139 AddMessageLoopObserver();
1140 model_->CloseTabContentsAt(tab_index,
1141 TabStripModel::CLOSE_USER_GESTURE |
1142 TabStripModel::CLOSE_CREATE_HISTORICAL_TAB);
1143 }
1144 }
1145
1146 bool TabStripGtk::IsCommandEnabledForTab(
1147 TabStripModel::ContextMenuCommand command_id, const TabGtk* tab) const {
1148 int index = GetIndexOfTab(tab);
1149 if (model_->ContainsIndex(index))
1150 return model_->IsContextMenuCommandEnabled(index, command_id);
1151 return false;
1152 }
1153
1154 void TabStripGtk::ExecuteCommandForTab(
1155 TabStripModel::ContextMenuCommand command_id, TabGtk* tab) {
1156 int index = GetIndexOfTab(tab);
1157 if (model_->ContainsIndex(index))
1158 model_->ExecuteContextMenuCommand(index, command_id);
1159 }
1160
1161 void TabStripGtk::StartHighlightTabsForCommand(
1162 TabStripModel::ContextMenuCommand command_id, TabGtk* tab) {
1163 if (command_id == TabStripModel::CommandCloseOtherTabs ||
1164 command_id == TabStripModel::CommandCloseTabsToRight) {
1165 NOTIMPLEMENTED();
1166 }
1167 }
1168
1169 void TabStripGtk::StopHighlightTabsForCommand(
1170 TabStripModel::ContextMenuCommand command_id, TabGtk* tab) {
1171 if (command_id == TabStripModel::CommandCloseTabsToRight ||
1172 command_id == TabStripModel::CommandCloseOtherTabs) {
1173 // Just tell all Tabs to stop pulsing - it's safe.
1174 StopAllHighlighting();
1175 }
1176 }
1177
1178 void TabStripGtk::StopAllHighlighting() {
1179 // TODO(jhawkins): Hook up animations.
1180 NOTIMPLEMENTED();
1181 }
1182
1183 void TabStripGtk::MaybeStartDrag(TabGtk* tab, const gfx::Point& point) {
1184 // Don't accidentally start any drag operations during animations if the
1185 // mouse is down.
1186 if (IsAnimating() || tab->closing() || !HasAvailableDragActions())
1187 return;
1188
1189 drag_controller_.reset(new DraggedTabControllerGtk(tab, this));
1190 drag_controller_->CaptureDragInfo(point);
1191 }
1192
1193 void TabStripGtk::ContinueDrag(GdkDragContext* context) {
1194 // We can get called even if |MaybeStartDrag| wasn't called in the event of
1195 // a TabStrip animation when the mouse button is down. In this case we should
1196 // _not_ continue the drag because it can lead to weird bugs.
1197 if (drag_controller_.get())
1198 drag_controller_->Drag();
1199 }
1200
1201 bool TabStripGtk::EndDrag(bool canceled) {
1202 return drag_controller_.get() ? drag_controller_->EndDrag(canceled) : false;
1203 }
1204
1205 bool TabStripGtk::HasAvailableDragActions() const {
1206 return model_->delegate()->GetDragActions() != 0;
1207 }
1208
1209 ThemeProvider* TabStripGtk::GetThemeProvider() {
1210 return theme_provider_;
1211 }
1212
1213 ///////////////////////////////////////////////////////////////////////////////
1214 // TabStripGtk, MessageLoop::Observer implementation:
1215
1216 void TabStripGtk::WillProcessEvent(GdkEvent* event) {
1217 // Nothing to do.
1218 }
1219
1220 void TabStripGtk::DidProcessEvent(GdkEvent* event) {
1221 switch (event->type) {
1222 case GDK_MOTION_NOTIFY:
1223 case GDK_LEAVE_NOTIFY:
1224 HandleGlobalMouseMoveEvent();
1225 break;
1226 default:
1227 break;
1228 }
1229 }
1230
1231 void TabStripGtk::Observe(NotificationType type,
1232 const NotificationSource& source,
1233 const NotificationDetails& details) {
1234 if (type == NotificationType::BROWSER_THEME_CHANGED) {
1235 TabRendererGtk::SetSelectedTitleColor(theme_provider_->GetColor(
1236 BrowserThemeProvider::COLOR_TAB_TEXT));
1237 TabRendererGtk::SetUnselectedTitleColor(theme_provider_->GetColor(
1238 BrowserThemeProvider::COLOR_BACKGROUND_TAB_TEXT));
1239 }
1240 }
1241
1242 ////////////////////////////////////////////////////////////////////////////////
1243 // TabStripGtk, private:
1244
1245 int TabStripGtk::GetTabCount() const {
1246 return static_cast<int>(tab_data_.size());
1247 }
1248
1249 int TabStripGtk::GetMiniTabCount() const {
1250 int mini_count = 0;
1251 for (size_t i = 0; i < tab_data_.size(); ++i) {
1252 if (tab_data_[i].tab->mini())
1253 mini_count++;
1254 else
1255 return mini_count;
1256 }
1257 return mini_count;
1258 }
1259
1260 int TabStripGtk::GetAvailableWidthForTabs(TabGtk* last_tab) const {
1261 if (!base::i18n::IsRTL())
1262 return last_tab->x() - bounds_.x() + last_tab->width();
1263 else
1264 return bounds_.width() - last_tab->x();
1265 }
1266
1267 int TabStripGtk::GetIndexOfTab(const TabGtk* tab) const {
1268 for (int i = 0, index = 0; i < GetTabCount(); ++i, ++index) {
1269 TabGtk* current_tab = GetTabAt(i);
1270 if (current_tab->closing()) {
1271 --index;
1272 } else if (current_tab == tab) {
1273 return index;
1274 }
1275 }
1276 return -1;
1277 }
1278
1279 TabGtk* TabStripGtk::GetTabAt(int index) const {
1280 DCHECK_GE(index, 0);
1281 DCHECK_LT(index, GetTabCount());
1282 return tab_data_.at(index).tab;
1283 }
1284
1285 TabGtk* TabStripGtk::GetTabAtAdjustForAnimation(int index) const {
1286 if (active_animation_.get() &&
1287 active_animation_->type() == TabAnimation::REMOVE &&
1288 index >=
1289 static_cast<RemoveTabAnimation*>(active_animation_.get())->index()) {
1290 index++;
1291 }
1292 return GetTabAt(index);
1293 }
1294
1295 void TabStripGtk::RemoveTabAt(int index) {
1296 TabGtk* removed = tab_data_.at(index).tab;
1297
1298 // Remove the Tab from the TabStrip's list.
1299 tab_data_.erase(tab_data_.begin() + index);
1300
1301 if (!IsDragSessionActive() || !drag_controller_->IsDragSourceTab(removed)) {
1302 gtk_container_remove(GTK_CONTAINER(tabstrip_.get()), removed->widget());
1303 delete removed;
1304 }
1305 }
1306
1307 void TabStripGtk::HandleGlobalMouseMoveEvent() {
1308 if (!IsCursorInTabStripZone()) {
1309 // Mouse moved outside the tab slop zone, start a timer to do a resize
1310 // layout after a short while...
1311 if (resize_layout_factory_.empty()) {
1312 MessageLoop::current()->PostDelayedTask(FROM_HERE,
1313 resize_layout_factory_.NewRunnableMethod(
1314 &TabStripGtk::ResizeLayoutTabs),
1315 kResizeTabsTimeMs);
1316 }
1317 } else {
1318 // Mouse moved quickly out of the tab strip and then into it again, so
1319 // cancel the timer so that the strip doesn't move when the mouse moves
1320 // back over it.
1321 resize_layout_factory_.RevokeAll();
1322 }
1323 }
1324
1325 void TabStripGtk::GenerateIdealBounds() {
1326 int tab_count = GetTabCount();
1327 double unselected, selected;
1328 GetDesiredTabWidths(tab_count, GetMiniTabCount(), &unselected, &selected);
1329
1330 current_unselected_width_ = unselected;
1331 current_selected_width_ = selected;
1332
1333 // NOTE: This currently assumes a tab's height doesn't differ based on
1334 // selected state or the number of tabs in the strip!
1335 int tab_height = TabGtk::GetStandardSize().height();
1336 double tab_x = tab_start_x();
1337 for (int i = 0; i < tab_count; ++i) {
1338 TabGtk* tab = GetTabAt(i);
1339 double tab_width = unselected;
1340 if (tab->mini())
1341 tab_width = TabGtk::GetMiniWidth();
1342 else if (tab->IsSelected())
1343 tab_width = selected;
1344 double end_of_tab = tab_x + tab_width;
1345 int rounded_tab_x = Round(tab_x);
1346 gfx::Rect state(rounded_tab_x, 0, Round(end_of_tab) - rounded_tab_x,
1347 tab_height);
1348 tab_data_.at(i).ideal_bounds = state;
1349 tab_x = end_of_tab + GetTabHOffset(i + 1);
1350 }
1351 }
1352
1353 void TabStripGtk::LayoutNewTabButton(double last_tab_right,
1354 double unselected_width) {
1355 gfx::Rect bounds(0, kNewTabButtonVOffset,
1356 newtab_button_->width(), newtab_button_->height());
1357 int delta = abs(Round(unselected_width) - TabGtk::GetStandardSize().width());
1358 if (delta > 1 && !needs_resize_layout_) {
1359 // We're shrinking tabs, so we need to anchor the New Tab button to the
1360 // right edge of the TabStrip's bounds, rather than the right edge of the
1361 // right-most Tab, otherwise it'll bounce when animating.
1362 bounds.set_x(bounds_.width() - newtab_button_->width());
1363 } else {
1364 bounds.set_x(Round(last_tab_right - kTabHOffset) + kNewTabButtonHOffset);
1365 }
1366 bounds.set_x(gtk_util::MirroredLeftPointForRect(tabstrip_.get(), bounds));
1367
1368 gtk_fixed_move(GTK_FIXED(tabstrip_.get()), newtab_button_->widget(),
1369 bounds.x(), bounds.y());
1370 }
1371
1372 void TabStripGtk::GetDesiredTabWidths(int tab_count,
1373 int mini_tab_count,
1374 double* unselected_width,
1375 double* selected_width) const {
1376 DCHECK(tab_count >= 0 && mini_tab_count >= 0 && mini_tab_count <= tab_count);
1377 const double min_unselected_width =
1378 TabGtk::GetMinimumUnselectedSize().width();
1379 const double min_selected_width =
1380 TabGtk::GetMinimumSelectedSize().width();
1381
1382 *unselected_width = min_unselected_width;
1383 *selected_width = min_selected_width;
1384
1385 if (tab_count == 0) {
1386 // Return immediately to avoid divide-by-zero below.
1387 return;
1388 }
1389
1390 // Determine how much space we can actually allocate to tabs.
1391 int available_width = tabstrip_->allocation.width;
1392 if (available_width_for_tabs_ < 0) {
1393 available_width = bounds_.width();
1394 available_width -=
1395 (kNewTabButtonHOffset + newtab_button_->width());
1396 } else {
1397 // Interesting corner case: if |available_width_for_tabs_| > the result
1398 // of the calculation in the conditional arm above, the strip is in
1399 // overflow. We can either use the specified width or the true available
1400 // width here; the first preserves the consistent "leave the last tab under
1401 // the user's mouse so they can close many tabs" behavior at the cost of
1402 // prolonging the glitchy appearance of the overflow state, while the second
1403 // gets us out of overflow as soon as possible but forces the user to move
1404 // their mouse for a few tabs' worth of closing. We choose visual
1405 // imperfection over behavioral imperfection and select the first option.
1406 available_width = available_width_for_tabs_;
1407 }
1408
1409 if (mini_tab_count > 0) {
1410 available_width -= mini_tab_count * (TabGtk::GetMiniWidth() + kTabHOffset);
1411 tab_count -= mini_tab_count;
1412 if (tab_count == 0) {
1413 *selected_width = *unselected_width = TabGtk::GetStandardSize().width();
1414 return;
1415 }
1416 // Account for gap between the last mini-tab and first normal tab.
1417 available_width -= mini_to_non_mini_gap_;
1418 }
1419
1420 // Calculate the desired tab widths by dividing the available space into equal
1421 // portions. Don't let tabs get larger than the "standard width" or smaller
1422 // than the minimum width for each type, respectively.
1423 const int total_offset = kTabHOffset * (tab_count - 1);
1424 const double desired_tab_width = std::min(
1425 (static_cast<double>(available_width - total_offset) /
1426 static_cast<double>(tab_count)),
1427 static_cast<double>(TabGtk::GetStandardSize().width()));
1428
1429 *unselected_width = std::max(desired_tab_width, min_unselected_width);
1430 *selected_width = std::max(desired_tab_width, min_selected_width);
1431
1432 // When there are multiple tabs, we'll have one selected and some unselected
1433 // tabs. If the desired width was between the minimum sizes of these types,
1434 // try to shrink the tabs with the smaller minimum. For example, if we have
1435 // a strip of width 10 with 4 tabs, the desired width per tab will be 2.5. If
1436 // selected tabs have a minimum width of 4 and unselected tabs have a minimum
1437 // width of 1, the above code would set *unselected_width = 2.5,
1438 // *selected_width = 4, which results in a total width of 11.5. Instead, we
1439 // want to set *unselected_width = 2, *selected_width = 4, for a total width
1440 // of 10.
1441 if (tab_count > 1) {
1442 if ((min_unselected_width < min_selected_width) &&
1443 (desired_tab_width < min_selected_width)) {
1444 double calc_width =
1445 static_cast<double>(
1446 available_width - total_offset - min_selected_width) /
1447 static_cast<double>(tab_count - 1);
1448 *unselected_width = std::max(calc_width, min_unselected_width);
1449 } else if ((min_unselected_width > min_selected_width) &&
1450 (desired_tab_width < min_unselected_width)) {
1451 *selected_width = std::max(available_width - total_offset -
1452 (min_unselected_width * (tab_count - 1)), min_selected_width);
1453 }
1454 }
1455 }
1456
1457 int TabStripGtk::GetTabHOffset(int tab_index) {
1458 if (tab_index < GetTabCount() && GetTabAt(tab_index - 1)->mini() &&
1459 !GetTabAt(tab_index)->mini()) {
1460 return mini_to_non_mini_gap_ + kTabHOffset;
1461 }
1462 return kTabHOffset;
1463 }
1464
1465 int TabStripGtk::tab_start_x() const {
1466 return 0;
1467 }
1468
1469 bool TabStripGtk::ResizeLayoutTabs() {
1470 resize_layout_factory_.RevokeAll();
1471
1472 // It is critically important that this is unhooked here, otherwise we will
1473 // keep spying on messages forever.
1474 RemoveMessageLoopObserver();
1475
1476 available_width_for_tabs_ = -1;
1477 int mini_tab_count = GetMiniTabCount();
1478 if (mini_tab_count == GetTabCount()) {
1479 // Only mini tabs, we know the tab widths won't have changed (all mini-tabs
1480 // have the same width), so there is nothing to do.
1481 return false;
1482 }
1483 TabGtk* first_tab = GetTabAt(mini_tab_count);
1484 double unselected, selected;
1485 GetDesiredTabWidths(GetTabCount(), mini_tab_count, &unselected, &selected);
1486 int w = Round(first_tab->IsSelected() ? selected : unselected);
1487
1488 // We only want to run the animation if we're not already at the desired
1489 // size.
1490 if (abs(first_tab->width() - w) > 1) {
1491 StartResizeLayoutAnimation();
1492 return true;
1493 }
1494
1495 return false;
1496 }
1497
1498 bool TabStripGtk::IsCursorInTabStripZone() const {
1499 gfx::Point tabstrip_topleft;
1500 gtk_util::ConvertWidgetPointToScreen(tabstrip_.get(), &tabstrip_topleft);
1501
1502 gfx::Rect bds = bounds();
1503 bds.set_origin(tabstrip_topleft);
1504 bds.set_height(bds.height() + kTabStripAnimationVSlop);
1505
1506 GdkScreen* screen = gdk_screen_get_default();
1507 GdkDisplay* display = gdk_screen_get_display(screen);
1508 gint x, y;
1509 gdk_display_get_pointer(display, NULL, &x, &y, NULL);
1510 gfx::Point cursor_point(x, y);
1511
1512 return bds.Contains(cursor_point);
1513 }
1514
1515 void TabStripGtk::AddMessageLoopObserver() {
1516 if (!added_as_message_loop_observer_) {
1517 MessageLoopForUI::current()->AddObserver(this);
1518 added_as_message_loop_observer_ = true;
1519 }
1520 }
1521
1522 void TabStripGtk::RemoveMessageLoopObserver() {
1523 if (added_as_message_loop_observer_) {
1524 MessageLoopForUI::current()->RemoveObserver(this);
1525 added_as_message_loop_observer_ = false;
1526 }
1527 }
1528
1529 gfx::Rect TabStripGtk::GetDropBounds(int drop_index,
1530 bool drop_before,
1531 bool* is_beneath) {
1532 DCHECK_NE(drop_index, -1);
1533 int center_x;
1534 if (drop_index < GetTabCount()) {
1535 TabGtk* tab = GetTabAt(drop_index);
1536 gfx::Rect bounds = tab->GetNonMirroredBounds(tabstrip_.get());
1537 // TODO(sky): update these for pinned tabs.
1538 if (drop_before)
1539 center_x = bounds.x() - (kTabHOffset / 2);
1540 else
1541 center_x = bounds.x() + (bounds.width() / 2);
1542 } else {
1543 TabGtk* last_tab = GetTabAt(drop_index - 1);
1544 gfx::Rect bounds = last_tab->GetNonMirroredBounds(tabstrip_.get());
1545 center_x = bounds.x() + bounds.width() + (kTabHOffset / 2);
1546 }
1547
1548 center_x = gtk_util::MirroredXCoordinate(tabstrip_.get(), center_x);
1549
1550 // Determine the screen bounds.
1551 gfx::Point drop_loc(center_x - drop_indicator_width / 2,
1552 -drop_indicator_height);
1553 gtk_util::ConvertWidgetPointToScreen(tabstrip_.get(), &drop_loc);
1554 gfx::Rect drop_bounds(drop_loc.x(), drop_loc.y(), drop_indicator_width,
1555 drop_indicator_height);
1556
1557 // TODO(jhawkins): We always display the arrow underneath the tab because we
1558 // don't have custom frame support yet.
1559 *is_beneath = true;
1560 if (*is_beneath)
1561 drop_bounds.Offset(0, drop_bounds.height() + bounds().height());
1562
1563 return drop_bounds;
1564 }
1565
1566 void TabStripGtk::UpdateDropIndex(GdkDragContext* context, gint x, gint y) {
1567 // If the UI layout is right-to-left, we need to mirror the mouse
1568 // coordinates since we calculate the drop index based on the
1569 // original (and therefore non-mirrored) positions of the tabs.
1570 x = gtk_util::MirroredXCoordinate(tabstrip_.get(), x);
1571 // We don't allow replacing the urls of mini-tabs.
1572 for (int i = GetMiniTabCount(); i < GetTabCount(); ++i) {
1573 TabGtk* tab = GetTabAt(i);
1574 gfx::Rect bounds = tab->GetNonMirroredBounds(tabstrip_.get());
1575 const int tab_max_x = bounds.x() + bounds.width();
1576 const int hot_width = bounds.width() / 3;
1577 if (x < tab_max_x) {
1578 if (x < bounds.x() + hot_width)
1579 SetDropIndex(i, true);
1580 else if (x >= tab_max_x - hot_width)
1581 SetDropIndex(i + 1, true);
1582 else
1583 SetDropIndex(i, false);
1584 return;
1585 }
1586 }
1587
1588 // The drop isn't over a tab, add it to the end.
1589 SetDropIndex(GetTabCount(), true);
1590 }
1591
1592 void TabStripGtk::SetDropIndex(int index, bool drop_before) {
1593 bool is_beneath;
1594 gfx::Rect drop_bounds = GetDropBounds(index, drop_before, &is_beneath);
1595
1596 if (!drop_info_.get()) {
1597 drop_info_.reset(new DropInfo(index, drop_before, !is_beneath));
1598 } else {
1599 if (!GTK_IS_WIDGET(drop_info_->container)) {
1600 drop_info_->CreateContainer();
1601 } else if (drop_info_->drop_index == index &&
1602 drop_info_->drop_before == drop_before) {
1603 return;
1604 }
1605
1606 drop_info_->drop_index = index;
1607 drop_info_->drop_before = drop_before;
1608 if (is_beneath == drop_info_->point_down) {
1609 drop_info_->point_down = !is_beneath;
1610 drop_info_->drop_arrow= GetDropArrowImage(drop_info_->point_down);
1611 }
1612 }
1613
1614 gtk_window_move(GTK_WINDOW(drop_info_->container),
1615 drop_bounds.x(), drop_bounds.y());
1616 gtk_window_resize(GTK_WINDOW(drop_info_->container),
1617 drop_bounds.width(), drop_bounds.height());
1618 }
1619
1620 bool TabStripGtk::CompleteDrop(guchar* data, bool is_plain_text) {
1621 if (!drop_info_.get())
1622 return false;
1623
1624 const int drop_index = drop_info_->drop_index;
1625 const bool drop_before = drop_info_->drop_before;
1626
1627 // Destroy the drop indicator.
1628 drop_info_.reset();
1629
1630 GURL url;
1631 if (is_plain_text) {
1632 AutocompleteMatch match;
1633 model_->profile()->GetAutocompleteClassifier()->Classify(
1634 UTF8ToWide(reinterpret_cast<char*>(data)), std::wstring(), false,
1635 &match, NULL);
1636 url = match.destination_url;
1637 } else {
1638 std::string url_string(reinterpret_cast<char*>(data));
1639 url = GURL(url_string.substr(0, url_string.find_first_of('\n')));
1640 }
1641 if (!url.is_valid())
1642 return false;
1643
1644 browser::NavigateParams params(window()->browser(), url,
1645 PageTransition::LINK);
1646 params.tabstrip_index = drop_index;
1647
1648 if (drop_before) {
1649 params.disposition = NEW_FOREGROUND_TAB;
1650 } else {
1651 params.disposition = CURRENT_TAB;
1652 params.source_contents = model_->GetTabContentsAt(drop_index);
1653 }
1654
1655 browser::Navigate(&params);
1656
1657 return true;
1658 }
1659
1660 // static
1661 GdkPixbuf* TabStripGtk::GetDropArrowImage(bool is_down) {
1662 return ResourceBundle::GetSharedInstance().GetPixbufNamed(
1663 is_down ? IDR_TAB_DROP_DOWN : IDR_TAB_DROP_UP);
1664 }
1665
1666 // TabStripGtk::DropInfo -------------------------------------------------------
1667
1668 TabStripGtk::DropInfo::DropInfo(int drop_index, bool drop_before,
1669 bool point_down)
1670 : drop_index(drop_index),
1671 drop_before(drop_before),
1672 point_down(point_down) {
1673 CreateContainer();
1674 drop_arrow = GetDropArrowImage(point_down);
1675 }
1676
1677 TabStripGtk::DropInfo::~DropInfo() {
1678 DestroyContainer();
1679 }
1680
1681 gboolean TabStripGtk::DropInfo::OnExposeEvent(GtkWidget* widget,
1682 GdkEventExpose* event) {
1683 if (gtk_util::IsScreenComposited()) {
1684 SetContainerTransparency();
1685 } else {
1686 SetContainerShapeMask();
1687 }
1688
1689 gdk_pixbuf_render_to_drawable(drop_arrow,
1690 container->window,
1691 0, 0, 0,
1692 0, 0,
1693 drop_indicator_width,
1694 drop_indicator_height,
1695 GDK_RGB_DITHER_NONE, 0, 0);
1696
1697 return FALSE;
1698 }
1699
1700 // Sets the color map of the container window to allow the window to be
1701 // transparent.
1702 void TabStripGtk::DropInfo::SetContainerColorMap() {
1703 GdkScreen* screen = gtk_widget_get_screen(container);
1704 GdkColormap* colormap = gdk_screen_get_rgba_colormap(screen);
1705
1706 // If rgba is not available, use rgb instead.
1707 if (!colormap)
1708 colormap = gdk_screen_get_rgb_colormap(screen);
1709
1710 gtk_widget_set_colormap(container, colormap);
1711 }
1712
1713 // Sets full transparency for the container window. This is used if
1714 // compositing is available for the screen.
1715 void TabStripGtk::DropInfo::SetContainerTransparency() {
1716 cairo_t* cairo_context = gdk_cairo_create(container->window);
1717 if (!cairo_context)
1718 return;
1719
1720 // Make the background of the dragged tab window fully transparent. All of
1721 // the content of the window (child widgets) will be completely opaque.
1722
1723 cairo_scale(cairo_context, static_cast<double>(drop_indicator_width),
1724 static_cast<double>(drop_indicator_height));
1725 cairo_set_source_rgba(cairo_context, 1.0f, 1.0f, 1.0f, 0.0f);
1726 cairo_set_operator(cairo_context, CAIRO_OPERATOR_SOURCE);
1727 cairo_paint(cairo_context);
1728 cairo_destroy(cairo_context);
1729 }
1730
1731 // Sets the shape mask for the container window to emulate a transparent
1732 // container window. This is used if compositing is not available for the
1733 // screen.
1734 void TabStripGtk::DropInfo::SetContainerShapeMask() {
1735 // Create a 1bpp bitmap the size of |container|.
1736 GdkPixmap* pixmap = gdk_pixmap_new(NULL,
1737 drop_indicator_width,
1738 drop_indicator_height, 1);
1739 cairo_t* cairo_context = gdk_cairo_create(GDK_DRAWABLE(pixmap));
1740
1741 // Set the transparency.
1742 cairo_set_source_rgba(cairo_context, 1, 1, 1, 0);
1743
1744 // Blit the rendered bitmap into a pixmap. Any pixel set in the pixmap will
1745 // be opaque in the container window.
1746 cairo_set_operator(cairo_context, CAIRO_OPERATOR_SOURCE);
1747 gdk_cairo_set_source_pixbuf(cairo_context, drop_arrow, 0, 0);
1748 cairo_paint(cairo_context);
1749 cairo_destroy(cairo_context);
1750
1751 // Set the shape mask.
1752 gdk_window_shape_combine_mask(container->window, pixmap, 0, 0);
1753 g_object_unref(pixmap);
1754 }
1755
1756 void TabStripGtk::DropInfo::CreateContainer() {
1757 container = gtk_window_new(GTK_WINDOW_POPUP);
1758 SetContainerColorMap();
1759 gtk_widget_set_app_paintable(container, TRUE);
1760 g_signal_connect(container, "expose-event",
1761 G_CALLBACK(OnExposeEventThunk), this);
1762 gtk_widget_add_events(container, GDK_STRUCTURE_MASK);
1763 gtk_window_move(GTK_WINDOW(container), 0, 0);
1764 gtk_window_resize(GTK_WINDOW(container),
1765 drop_indicator_width, drop_indicator_height);
1766 gtk_widget_show_all(container);
1767 }
1768
1769 void TabStripGtk::DropInfo::DestroyContainer() {
1770 if (GTK_IS_WIDGET(container))
1771 gtk_widget_destroy(container);
1772 }
1773
1774 void TabStripGtk::StopAnimation() {
1775 if (active_animation_.get())
1776 active_animation_->Stop();
1777 }
1778
1779 // Called from:
1780 // - animation tick
1781 void TabStripGtk::AnimationLayout(double unselected_width) {
1782 int tab_height = TabGtk::GetStandardSize().height();
1783 double tab_x = tab_start_x();
1784 for (int i = 0; i < GetTabCount(); ++i) {
1785 TabAnimation* animation = active_animation_.get();
1786 if (animation)
1787 tab_x += animation->GetGapWidth(i);
1788 double tab_width = TabAnimation::GetCurrentTabWidth(this, animation, i);
1789 double end_of_tab = tab_x + tab_width;
1790 int rounded_tab_x = Round(tab_x);
1791 TabGtk* tab = GetTabAt(i);
1792 gfx::Rect bounds(rounded_tab_x, 0, Round(end_of_tab) - rounded_tab_x,
1793 tab_height);
1794 SetTabBounds(tab, bounds);
1795 tab_x = end_of_tab + GetTabHOffset(i + 1);
1796 }
1797 LayoutNewTabButton(tab_x, unselected_width);
1798 }
1799
1800 void TabStripGtk::StartInsertTabAnimation(int index) {
1801 // The TabStrip can now use its entire width to lay out Tabs.
1802 available_width_for_tabs_ = -1;
1803 StopAnimation();
1804 active_animation_.reset(new InsertTabAnimation(this, index));
1805 active_animation_->Start();
1806 }
1807
1808 void TabStripGtk::StartRemoveTabAnimation(int index, TabContents* contents) {
1809 if (active_animation_.get()) {
1810 // Some animations (e.g. MoveTabAnimation) cause there to be a Layout when
1811 // they're completed (which includes canceled). Since |tab_data_| is now
1812 // inconsistent with TabStripModel, doing this Layout will crash now, so
1813 // we ask the MoveTabAnimation to skip its Layout (the state will be
1814 // corrected by the RemoveTabAnimation we're about to initiate).
1815 active_animation_->set_layout_on_completion(false);
1816 active_animation_->Stop();
1817 }
1818
1819 active_animation_.reset(new RemoveTabAnimation(this, index, contents));
1820 active_animation_->Start();
1821 }
1822
1823 void TabStripGtk::StartMoveTabAnimation(int from_index, int to_index) {
1824 StopAnimation();
1825 active_animation_.reset(new MoveTabAnimation(this, from_index, to_index));
1826 active_animation_->Start();
1827 }
1828
1829 void TabStripGtk::StartResizeLayoutAnimation() {
1830 StopAnimation();
1831 active_animation_.reset(new ResizeLayoutAnimation(this));
1832 active_animation_->Start();
1833 }
1834
1835 void TabStripGtk::StartMiniTabAnimation(int index) {
1836 StopAnimation();
1837 active_animation_.reset(new MiniTabAnimation(this, index));
1838 active_animation_->Start();
1839 }
1840
1841 void TabStripGtk::StartMiniMoveTabAnimation(int from_index,
1842 int to_index,
1843 const gfx::Rect& start_bounds) {
1844 StopAnimation();
1845 active_animation_.reset(
1846 new MiniMoveAnimation(this, from_index, to_index, start_bounds));
1847 active_animation_->Start();
1848 }
1849
1850 void TabStripGtk::FinishAnimation(TabStripGtk::TabAnimation* animation,
1851 bool layout) {
1852 active_animation_.reset(NULL);
1853
1854 // Reset the animation state of each tab.
1855 for (int i = 0, count = GetTabCount(); i < count; ++i)
1856 GetTabAt(i)->set_animating_mini_change(false);
1857
1858 if (layout)
1859 Layout();
1860 }
1861
1862 gboolean TabStripGtk::OnExpose(GtkWidget* widget, GdkEventExpose* event) {
1863 if (gdk_region_empty(event->region))
1864 return TRUE;
1865
1866 // If we're only repainting favicons, optimize the paint path and only draw
1867 // the favicons.
1868 GdkRectangle* rects;
1869 gint num_rects;
1870 gdk_region_get_rectangles(event->region, &rects, &num_rects);
1871 qsort(rects, num_rects, sizeof(GdkRectangle), CompareGdkRectangles);
1872 std::vector<int> tabs_to_repaint;
1873 if (!IsDragSessionActive() &&
1874 CanPaintOnlyFavIcons(rects, num_rects, &tabs_to_repaint)) {
1875 PaintOnlyFavIcons(event, tabs_to_repaint);
1876 g_free(rects);
1877 return TRUE;
1878 }
1879 g_free(rects);
1880
1881 // TODO(jhawkins): Ideally we'd like to only draw what's needed in the damage
1882 // rect, but the tab widgets overlap each other, and painting on one widget
1883 // will cause an expose-event to be sent to the widgets underneath. The
1884 // underlying widget does not need to be redrawn as we control the order of
1885 // expose-events. Currently we hack it to redraw the entire tabstrip. We
1886 // could change the damage rect to just contain the tabs + the new tab button.
1887 event->area.x = 0;
1888 event->area.y = 0;
1889 event->area.width = bounds_.width();
1890 event->area.height = bounds_.height();
1891 gdk_region_union_with_rect(event->region, &event->area);
1892
1893 // Paint the New Tab button.
1894 gtk_container_propagate_expose(GTK_CONTAINER(tabstrip_.get()),
1895 newtab_button_->widget(), event);
1896
1897 // Paint the tabs in reverse order, so they stack to the left.
1898 TabGtk* selected_tab = NULL;
1899 int tab_count = GetTabCount();
1900 for (int i = tab_count - 1; i >= 0; --i) {
1901 TabGtk* tab = GetTabAt(i);
1902 // We must ask the _Tab's_ model, not ourselves, because in some situations
1903 // the model will be different to this object, e.g. when a Tab is being
1904 // removed after its TabContents has been destroyed.
1905 if (!tab->IsSelected()) {
1906 gtk_container_propagate_expose(GTK_CONTAINER(tabstrip_.get()),
1907 tab->widget(), event);
1908 } else {
1909 selected_tab = tab;
1910 }
1911 }
1912
1913 // Paint the selected tab last, so it overlaps all the others.
1914 if (selected_tab) {
1915 gtk_container_propagate_expose(GTK_CONTAINER(tabstrip_.get()),
1916 selected_tab->widget(), event);
1917 }
1918
1919 return TRUE;
1920 }
1921
1922 void TabStripGtk::OnSizeAllocate(GtkWidget* widget, GtkAllocation* allocation) {
1923 gfx::Rect bounds = gfx::Rect(allocation->x, allocation->y,
1924 allocation->width, allocation->height);
1925
1926 // Nothing to do if the bounds are the same. If we don't catch this, we'll
1927 // get an infinite loop of size-allocate signals.
1928 if (bounds_ == bounds)
1929 return;
1930
1931 SetBounds(bounds);
1932
1933 // No tabs, nothing to layout. This happens when a browser window is created
1934 // and shown before tabs are added (as in a popup window).
1935 if (GetTabCount() == 0)
1936 return;
1937
1938 // When there is only one tab, Layout() so we don't animate it. With more
1939 // tabs, do ResizeLayoutTabs(). In RTL(), we will also need to manually
1940 // Layout() when ResizeLayoutTabs() is a no-op.
1941 if ((GetTabCount() == 1) || (!ResizeLayoutTabs() && base::i18n::IsRTL()))
1942 Layout();
1943 }
1944
1945 gboolean TabStripGtk::OnDragMotion(GtkWidget* widget, GdkDragContext* context,
1946 gint x, gint y, guint time) {
1947 UpdateDropIndex(context, x, y);
1948 return TRUE;
1949 }
1950
1951 gboolean TabStripGtk::OnDragDrop(GtkWidget* widget, GdkDragContext* context,
1952 gint x, gint y, guint time) {
1953 if (!drop_info_.get())
1954 return FALSE;
1955
1956 GdkAtom target = gtk_drag_dest_find_target(widget, context, NULL);
1957 if (target != GDK_NONE)
1958 gtk_drag_finish(context, FALSE, FALSE, time);
1959 else
1960 gtk_drag_get_data(widget, context, target, time);
1961
1962 return TRUE;
1963 }
1964
1965 gboolean TabStripGtk::OnDragLeave(GtkWidget* widget, GdkDragContext* context,
1966 guint time) {
1967 // Destroy the drop indicator.
1968 drop_info_->DestroyContainer();
1969 return FALSE;
1970 }
1971
1972 gboolean TabStripGtk::OnDragDataReceived(GtkWidget* widget,
1973 GdkDragContext* context,
1974 gint x, gint y,
1975 GtkSelectionData* data,
1976 guint info, guint time) {
1977 bool success = false;
1978
1979 if (info == gtk_dnd_util::TEXT_URI_LIST ||
1980 info == gtk_dnd_util::NETSCAPE_URL ||
1981 info == gtk_dnd_util::TEXT_PLAIN) {
1982 success = CompleteDrop(data->data, info == gtk_dnd_util::TEXT_PLAIN);
1983 }
1984
1985 gtk_drag_finish(context, success, success, time);
1986 return TRUE;
1987 }
1988
1989 void TabStripGtk::OnNewTabClicked(GtkWidget* widget) {
1990 GdkEvent* event = gtk_get_current_event();
1991 DCHECK_EQ(event->type, GDK_BUTTON_RELEASE);
1992 int mouse_button = event->button.button;
1993 gdk_event_free(event);
1994
1995 switch (mouse_button) {
1996 case 1:
1997 model_->delegate()->AddBlankTab(true);
1998 break;
1999 case 2: {
2000 // On middle-click, try to parse the PRIMARY selection as a URL and load
2001 // it instead of creating a blank page.
2002 GURL url;
2003 if (!gtk_util::URLFromPrimarySelection(model_->profile(), &url))
2004 return;
2005
2006 Browser* browser = window_->browser();
2007 DCHECK(browser);
2008 browser->AddSelectedTabWithURL(url, PageTransition::TYPED);
2009 break;
2010 }
2011 default:
2012 NOTREACHED() << "Got click on new tab button with unhandled mouse "
2013 << "button " << mouse_button;
2014 }
2015 }
2016
2017 void TabStripGtk::SetTabBounds(TabGtk* tab, const gfx::Rect& bounds) {
2018 gfx::Rect bds = bounds;
2019 bds.set_x(gtk_util::MirroredLeftPointForRect(tabstrip_.get(), bounds));
2020 tab->SetBounds(bds);
2021 gtk_fixed_move(GTK_FIXED(tabstrip_.get()), tab->widget(),
2022 bds.x(), bds.y());
2023 }
2024
2025 bool TabStripGtk::CanPaintOnlyFavIcons(const GdkRectangle* rects,
2026 int num_rects, std::vector<int>* tabs_to_paint) {
2027 // |rects| are sorted so we just need to scan from left to right and compare
2028 // it to the tab favicon positions from left to right.
2029 int t = 0;
2030 for (int r = 0; r < num_rects; ++r) {
2031 while (t < GetTabCount()) {
2032 TabGtk* tab = GetTabAt(t);
2033 if (GdkRectMatchesTabFavIconBounds(rects[r], tab) &&
2034 tab->ShouldShowIcon()) {
2035 tabs_to_paint->push_back(t);
2036 ++t;
2037 break;
2038 }
2039 ++t;
2040 }
2041 }
2042 return static_cast<int>(tabs_to_paint->size()) == num_rects;
2043 }
2044
2045 void TabStripGtk::PaintOnlyFavIcons(GdkEventExpose* event,
2046 const std::vector<int>& tabs_to_paint) {
2047 for (size_t i = 0; i < tabs_to_paint.size(); ++i)
2048 GetTabAt(tabs_to_paint[i])->PaintFavIconArea(event);
2049 }
2050
2051 CustomDrawButton* TabStripGtk::MakeNewTabButton() {
2052 CustomDrawButton* button = new CustomDrawButton(IDR_NEWTAB_BUTTON,
2053 IDR_NEWTAB_BUTTON_P, IDR_NEWTAB_BUTTON_H, 0);
2054
2055 // Let the middle mouse button initiate clicks as well.
2056 gtk_util::SetButtonTriggersNavigation(button->widget());
2057 g_signal_connect(button->widget(), "clicked",
2058 G_CALLBACK(OnNewTabClickedThunk), this);
2059 GTK_WIDGET_UNSET_FLAGS(button->widget(), GTK_CAN_FOCUS);
2060 gtk_fixed_put(GTK_FIXED(tabstrip_.get()), button->widget(), 0, 0);
2061
2062 return button;
2063 }
OLDNEW
« no previous file with comments | « chrome/browser/gtk/tabs/tab_strip_gtk.h ('k') | chrome/browser/gtk/tabstrip_origin_provider.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698