• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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 #ifndef CHROME_BROWSER_INFOBARS_INFOBAR_H_
6 #define CHROME_BROWSER_INFOBARS_INFOBAR_H_
7 
8 #include <utility>
9 
10 #include "base/memory/scoped_ptr.h"
11 #include "chrome/browser/infobars/infobar_delegate.h"
12 #include "third_party/skia/include/core/SkColor.h"
13 #include "ui/gfx/animation/animation_delegate.h"
14 #include "ui/gfx/animation/slide_animation.h"
15 #include "ui/gfx/size.h"
16 
17 class InfoBarContainer;
18 class InfoBarService;
19 
20 // InfoBar is a cross-platform base class for an infobar "view" (in the MVC
21 // sense), which owns a corresponding InfoBarDelegate "model".  Typically,
22 // a caller will call XYZInfoBarDelegate::Create() and pass in the
23 // InfoBarService for the relevant tab.  This will create an XYZInfoBarDelegate,
24 // create a platform-specific subclass of InfoBar to own it, and then call
25 // InfoBarService::AddInfoBar() to give it ownership of the infobar.
26 // During its life, the InfoBar may be shown and hidden as the owning tab is
27 // switched between the foreground and background.  Eventually, InfoBarService
28 // will instruct the InfoBar to close itself.  At this point, the InfoBar will
29 // optionally animate closed; once it's no longer visible, it deletes itself,
30 // destroying the InfoBarDelegate in the process.
31 //
32 // Thus, InfoBarDelegate and InfoBar implementations can assume they share
33 // lifetimes, and not NULL-check each other; but if one needs to reach back into
34 // the owning InfoBarService, it must check whether that's still possible.
35 class InfoBar : public gfx::AnimationDelegate {
36  public:
37   // These are the types passed as Details for infobar-related notifications.
38   typedef InfoBar AddedDetails;
39   typedef std::pair<InfoBar*, bool> RemovedDetails;
40   typedef std::pair<InfoBar*, InfoBar*> ReplacedDetails;
41 
42   // Platforms must define these.
43   static const int kDefaultBarTargetHeight;
44   static const int kSeparatorLineHeight;
45   static const int kDefaultArrowTargetHeight;
46   static const int kMaximumArrowTargetHeight;
47   // The half-width (see comments on |arrow_half_width_| below) scales to its
48   // default and maximum values proportionally to how the height scales to its.
49   static const int kDefaultArrowTargetHalfWidth;
50   static const int kMaximumArrowTargetHalfWidth;
51 
52   explicit InfoBar(scoped_ptr<InfoBarDelegate> delegate);
53   virtual ~InfoBar();
54 
55   static SkColor GetTopColor(InfoBarDelegate::Type infobar_type);
56   static SkColor GetBottomColor(InfoBarDelegate::Type infobar_type);
57 
owner()58   InfoBarService* owner() { return owner_; }
delegate()59   InfoBarDelegate* delegate() { return delegate_.get(); }
delegate()60   const InfoBarDelegate* delegate() const { return delegate_.get(); }
set_container(InfoBarContainer * container)61   void set_container(InfoBarContainer* container) { container_ = container; }
62 
63   // Sets |owner_|.  This also calls StoreActiveEntryUniqueID() on |delegate_|.
64   // This must only be called once as there's no way to extract an infobar from
65   // its owner without deleting it, for reparenting in another tab.
66   void SetOwner(InfoBarService* owner);
67 
68   // Makes the infobar visible.  If |animate| is true, the infobar is then
69   // animated to full size.
70   void Show(bool animate);
71 
72   // Makes the infobar hidden.  If |animate| is false, the infobar is
73   // immediately removed from the container, and, if now unowned, deleted.  If
74   // |animate| is true, the infobar is animated to zero size, ultimately
75   // triggering a call to AnimationEnded().
76   void Hide(bool animate);
77 
78   // Changes the target height of the arrow portion of the infobar.  This has no
79   // effect once the infobar is animating closed.
80   void SetArrowTargetHeight(int height);
81 
82   // Notifies the infobar that it is no longer owned and should delete itself
83   // once it is invisible.
84   void CloseSoon();
85 
86   // Forwards a close request to our owner.  This is a no-op if we're already
87   // unowned.
88   void RemoveSelf();
89 
90   // Changes the target height of the main ("bar") portion of the infobar.
91   void SetBarTargetHeight(int height);
92 
animation()93   const gfx::SlideAnimation& animation() const { return animation_; }
arrow_height()94   int arrow_height() const { return arrow_height_; }
arrow_target_height()95   int arrow_target_height() const { return arrow_target_height_; }
arrow_half_width()96   int arrow_half_width() const { return arrow_half_width_; }
total_height()97   int total_height() const { return arrow_height_ + bar_height_; }
98 
99  protected:
100   // gfx::AnimationDelegate:
101   virtual void AnimationProgressed(const gfx::Animation* animation) OVERRIDE;
102 
container()103   const InfoBarContainer* container() const { return container_; }
container()104   InfoBarContainer* container() { return container_; }
animation()105   gfx::SlideAnimation* animation() { return &animation_; }
bar_height()106   int bar_height() const { return bar_height_; }
bar_target_height()107   int bar_target_height() const { return bar_target_height_; }
108 
109   // Platforms may optionally override these if they need to do work during
110   // processing of the given calls.
PlatformSpecificSetOwner()111   virtual void PlatformSpecificSetOwner() {}
PlatformSpecificShow(bool animate)112   virtual void PlatformSpecificShow(bool animate) {}
PlatformSpecificHide(bool animate)113   virtual void PlatformSpecificHide(bool animate) {}
PlatformSpecificOnCloseSoon()114   virtual void PlatformSpecificOnCloseSoon() {}
PlatformSpecificOnHeightsRecalculated()115   virtual void PlatformSpecificOnHeightsRecalculated() {}
116 
117  private:
118   // gfx::AnimationDelegate:
119   virtual void AnimationEnded(const gfx::Animation* animation) OVERRIDE;
120 
121   // Finds the new desired arrow and bar heights, and if they differ from the
122   // current ones, calls PlatformSpecificOnHeightRecalculated().  Informs our
123   // container our state has changed if either the heights have changed or
124   // |force_notify| is set.
125   void RecalculateHeights(bool force_notify);
126 
127   // Checks whether the infobar is unowned and done with all animations.  If so,
128   // notifies the container that it should remove this infobar, and deletes
129   // itself.
130   void MaybeDelete();
131 
132   InfoBarService* owner_;
133   scoped_ptr<InfoBarDelegate> delegate_;
134   InfoBarContainer* container_;
135   gfx::SlideAnimation animation_;
136 
137   // The current and target heights of the arrow and bar portions, and half the
138   // current arrow width.  (It's easier to work in half-widths as we draw the
139   // arrow as two halves on either side of a center point.)
140   int arrow_height_;         // Includes both fill and top stroke.
141   int arrow_target_height_;
142   int arrow_half_width_;     // Includes only fill.
143   int bar_height_;           // Includes both fill and bottom separator.
144   int bar_target_height_;
145 
146   DISALLOW_COPY_AND_ASSIGN(InfoBar);
147 };
148 
149 #endif  // CHROME_BROWSER_INFOBARS_INFOBAR_H_
150