diff options
author | saintlou@chromium.org <saintlou@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-09-19 15:03:05 +0000 |
---|---|---|
committer | saintlou@chromium.org <saintlou@chromium.org@0039d316-1c4b-4281-b951-d872f2087c98> | 2011-09-19 15:03:05 +0000 |
commit | 8d6ac4be4114c6732560b78aae4e6f04058d0f73 (patch) | |
tree | 0bd8281fdaaf8956b21ad489484e478787f42023 /views/bubble | |
parent | 0e0d84eb874acd9feaf11e9b853c4af9b1a5389f (diff) | |
download | chromium_src-8d6ac4be4114c6732560b78aae4e6f04058d0f73.zip chromium_src-8d6ac4be4114c6732560b78aae4e6f04058d0f73.tar.gz chromium_src-8d6ac4be4114c6732560b78aae4e6f04058d0f73.tar.bz2 |
re-checkin http://codereview.chromium.org/7720001/ to add bubble views.
The patch caused memory leak in views_unittests. added patch http://codereview.chromium.org/7858015/ to fix unittests memory leak.
BUG=none
TEST=none
Review URL: http://codereview.chromium.org/7831071
Patch from Alice Tull <alicet@chromium.org>.
git-svn-id: svn://svn.chromium.org/chrome/trunk/src@101739 0039d316-1c4b-4281-b951-d872f2087c98
Diffstat (limited to 'views/bubble')
-rw-r--r-- | views/bubble/bubble_delegate.cc | 66 | ||||
-rw-r--r-- | views/bubble/bubble_delegate.h | 63 | ||||
-rw-r--r-- | views/bubble/bubble_delegate_unittest.cc | 48 | ||||
-rw-r--r-- | views/bubble/bubble_frame_view.cc | 73 | ||||
-rw-r--r-- | views/bubble/bubble_frame_view.h | 65 | ||||
-rw-r--r-- | views/bubble/bubble_frame_view_unittest.cc | 76 | ||||
-rw-r--r-- | views/bubble/bubble_view.cc | 128 | ||||
-rw-r--r-- | views/bubble/bubble_view.h | 87 | ||||
-rw-r--r-- | views/bubble/bubble_view_unittest.cc | 90 |
9 files changed, 696 insertions, 0 deletions
diff --git a/views/bubble/bubble_delegate.cc b/views/bubble/bubble_delegate.cc new file mode 100644 index 0000000..14a06e8 --- /dev/null +++ b/views/bubble/bubble_delegate.cc @@ -0,0 +1,66 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "views/bubble/bubble_delegate.h" +#include "views/bubble/bubble_frame_view.h" +#include "views/bubble/bubble_view.h" + +#include "base/logging.h" +#include "views/widget/widget.h" + +namespace views { + +BubbleDelegate* BubbleDelegate::AsBubbleDelegate() { return this; } + +ClientView* BubbleDelegate::CreateClientView(Widget* widget) { + BubbleView* bubble_view = new BubbleView(widget, GetContentsView()); + bubble_view->SetBounds(0, 0, GetBounds().width(), GetBounds().height()); + if (widget->GetFocusManager()) { + widget->GetFocusManager()->RegisterAccelerator( + views::Accelerator(ui::VKEY_ESCAPE, false, false, false), + bubble_view); + } + return bubble_view; +} + +NonClientFrameView* BubbleDelegate::CreateNonClientFrameView() { + return new BubbleFrameView(GetWidget(), + GetBounds(), + GetFrameBackgroundColor(), + GetFrameArrowLocation()); +} + +const BubbleView* BubbleDelegate::GetBubbleView() const { + return GetWidget()->client_view()->AsBubbleView(); +} + +BubbleView* BubbleDelegate::GetBubbleView() { + return GetWidget()->client_view()->AsBubbleView(); +} + +const BubbleFrameView* BubbleDelegate::GetBubbleFrameView() const { + return static_cast<BubbleFrameView*>( + GetWidget()->non_client_view()->frame_view()); +} + +BubbleFrameView* BubbleDelegate::GetBubbleFrameView() { + return static_cast<BubbleFrameView*>( + GetWidget()->non_client_view()->frame_view()); +} + +BubbleDelegateView::BubbleDelegateView(Widget* frame):frame_(frame) { +} + +BubbleDelegateView::~BubbleDelegateView() { +} + +Widget* BubbleDelegateView::GetWidget() { + return frame_; +} + +const Widget* BubbleDelegateView::GetWidget() const { + return frame_; +} + +} // namespace views diff --git a/views/bubble/bubble_delegate.h b/views/bubble/bubble_delegate.h new file mode 100644 index 0000000..32cf0a4 --- /dev/null +++ b/views/bubble/bubble_delegate.h @@ -0,0 +1,63 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef VIEWS_BUBBLE_BUBBLE_DELEGATE_H_ +#define VIEWS_BUBBLE_BUBBLE_DELEGATE_H_ +#pragma once + +#include <string> + +#include "views/bubble/bubble_border.h" +#include "views/view.h" +#include "views/widget/widget_delegate.h" + +namespace views { +class BubbleBorder; +class BubbleFrameView; +class BubbleView; +class View; + +// BubbleDelegate interface to create bubble frame view and bubble client view. +// +/////////////////////////////////////////////////////////////////////////////// +class VIEWS_EXPORT BubbleDelegate : public WidgetDelegate { + public: + virtual BubbleDelegate* AsBubbleDelegate() OVERRIDE; + virtual ClientView* CreateClientView(Widget* widget) OVERRIDE; + virtual NonClientFrameView* CreateNonClientFrameView() OVERRIDE; + + virtual SkColor GetFrameBackgroundColor() = 0; + virtual gfx::Rect GetBounds() = 0; + virtual BubbleBorder::ArrowLocation GetFrameArrowLocation() = 0; + + const BubbleView* GetBubbleView() const; + BubbleView* GetBubbleView(); + + const BubbleFrameView* GetBubbleFrameView() const; + BubbleFrameView* GetBubbleFrameView(); + + protected: + virtual ~BubbleDelegate() {} +}; + +// BubbleDelegateView to create bubble frame view and bubble client view. +// +/////////////////////////////////////////////////////////////////////////////// +class VIEWS_EXPORT BubbleDelegateView : public BubbleDelegate, public View { + public: + explicit BubbleDelegateView(Widget* frame); + virtual ~BubbleDelegateView(); + + // Overridden from WidgetDelegate: + virtual Widget* GetWidget() OVERRIDE; + virtual const Widget* GetWidget() const OVERRIDE; + + private: + Widget* frame_; + DISALLOW_COPY_AND_ASSIGN(BubbleDelegateView); +}; + +} // namespace views + +#endif // VIEWS_BUBBLE_BUBBLE_DELEGATE_H_ diff --git a/views/bubble/bubble_delegate_unittest.cc b/views/bubble/bubble_delegate_unittest.cc new file mode 100644 index 0000000..c8a05e8 --- /dev/null +++ b/views/bubble/bubble_delegate_unittest.cc @@ -0,0 +1,48 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "base/memory/scoped_ptr.h" +#include "base/message_loop.h" +#include "third_party/skia/include/core/SkColor.h" +#include "testing/gtest/include/gtest/gtest.h" +#include "ui/base/animation/slide_animation.h" +#include "views/bubble/bubble_border.h" +#include "views/bubble/bubble_delegate.h" +#include "views/bubble/bubble_view.h" +#include "views/widget/widget.h" + +namespace views { + +namespace { + +class TestBubbleDelegate : public BubbleDelegateView { + public: + explicit TestBubbleDelegate(Widget *frame): BubbleDelegateView(frame) {} + SkColor GetFrameBackgroundColor() { return SK_ColorGREEN; } + gfx::Rect GetBounds() { return gfx::Rect(10, 10, 200, 200); } + BubbleBorder::ArrowLocation GetFrameArrowLocation() { + return BubbleBorder::LEFT_BOTTOM; + } + View* GetContentsView() { return &view_; } + + View view_; +}; + +TEST(BubbleDelegateTest, CreateDelegate) { + MessageLoopForUI message_loop; + scoped_ptr<Widget> bubble_widget(new Widget()); + Widget::InitParams params(Widget::InitParams::TYPE_BUBBLE); + TestBubbleDelegate delegate(bubble_widget.get()); + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; + params.delegate = &delegate; + bubble_widget->Init(params); + EXPECT_EQ(&delegate, bubble_widget->widget_delegate()); + EXPECT_EQ(bubble_widget, delegate.GetWidget()); + bubble_widget->CloseNow(); + bubble_widget.reset(NULL); + MessageLoop::current()->RunAllPending(); +} + +} // namespace +} // namespace views diff --git a/views/bubble/bubble_frame_view.cc b/views/bubble/bubble_frame_view.cc new file mode 100644 index 0000000..b3c2f70 --- /dev/null +++ b/views/bubble/bubble_frame_view.cc @@ -0,0 +1,73 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "views/bubble/bubble_frame_view.h" + +#include "grit/ui_resources.h" +#include "ui/gfx/canvas.h" +#include "ui/gfx/path.h" +#include "views/bubble/bubble_border.h" +#include "views/widget/widget_delegate.h" +#include "views/window/client_view.h" + +namespace views { + +BubbleFrameView::BubbleFrameView(Widget* frame, + const gfx::Rect& bounds, + SkColor color, + BubbleBorder::ArrowLocation location) + : frame_(frame), + frame_bounds_(bounds), + bubble_border_(new BubbleBorder(location)), + bubble_background_(new BubbleBackground(bubble_border_)) { + SetBoundsRect(bounds); + bubble_border_->set_background_color(color); + set_border(bubble_border_); + set_background(bubble_background_); +} + +BubbleFrameView::~BubbleFrameView() {} + +gfx::Rect BubbleFrameView::GetBoundsForClientView() const { + gfx::Insets insets; + bubble_border_->GetInsets(&insets); + return gfx::Rect(insets.left(), insets.top(), + frame_bounds_.width() - insets.left() - insets.right(), + frame_bounds_.height() - insets.top() - insets.bottom()); +} + +gfx::Rect BubbleFrameView::GetWindowBoundsForClientBounds( + const gfx::Rect& client_bounds) const { + return bubble_border_->GetBounds(client_bounds, client_bounds.size()); +} + +void BubbleFrameView::EnableClose(bool enable) { +} + +void BubbleFrameView::ResetWindowControls() { +} + +void BubbleFrameView::UpdateWindowIcon() { +} + +void BubbleFrameView::OnPaint(gfx::Canvas* canvas) { + border()->Paint(*this, canvas); + bubble_background_->Paint(canvas, this); +} + +int BubbleFrameView::NonClientHitTest(const gfx::Point& point) { + return frame_->client_view()->NonClientHitTest(point); +} + +void BubbleFrameView::GetWindowMask(const gfx::Size& size, + gfx::Path* window_mask) { +} + +gfx::Size BubbleFrameView::GetPreferredSize() { + gfx::Size pref = frame_->client_view()->GetPreferredSize(); + gfx::Rect bounds(0, 0, pref.width(), pref.height()); + return frame_->non_client_view()->GetWindowBoundsForClientBounds( + bounds).size(); +} +} // namespace views diff --git a/views/bubble/bubble_frame_view.h b/views/bubble/bubble_frame_view.h new file mode 100644 index 0000000..c6ec085 --- /dev/null +++ b/views/bubble/bubble_frame_view.h @@ -0,0 +1,65 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef VIEWS_BUBBLE_BUBBLE_FRAME_VIEW_H_ +#define VIEWS_BUBBLE_BUBBLE_FRAME_VIEW_H_ +#pragma once + +#include "views/bubble/bubble_border.h" +#include "views/widget/widget.h" +#include "views/window/client_view.h" +#include "views/window/window_resources.h" + +namespace gfx { +class Canvas; +class Font; +class Size; +class Path; +class Point; +} + +namespace views { + +// BubbleFrameView to render BubbleBorder. +// +//////////////////////////////////////////////////////////////////////////////// +class VIEWS_EXPORT BubbleFrameView : public NonClientFrameView { + public: + BubbleFrameView(Widget* frame, + const gfx::Rect& bounds, + SkColor color, + BubbleBorder::ArrowLocation location); + virtual ~BubbleFrameView(); + + // NonClientFrameView overrides: + virtual gfx::Rect GetBoundsForClientView() const OVERRIDE; + virtual gfx::Rect GetWindowBoundsForClientBounds( + const gfx::Rect& client_bounds) const OVERRIDE; + virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE; + virtual void GetWindowMask( + const gfx::Size& size, gfx::Path* window_mask) OVERRIDE; + virtual void EnableClose(bool enable) OVERRIDE; + virtual void ResetWindowControls() OVERRIDE; + virtual void UpdateWindowIcon() OVERRIDE; + + // View overrides: + virtual void OnPaint(gfx::Canvas* canvas) OVERRIDE; + virtual gfx::Size GetPreferredSize() OVERRIDE; + + private: + // Not owned. + Widget* frame_; + // Window bounds in screen coordinates. + gfx::Rect frame_bounds_; + // The bubble border object owned by this view. + BubbleBorder* bubble_border_; + // The bubble background object owned by this view. + BubbleBackground* bubble_background_; + + DISALLOW_COPY_AND_ASSIGN(BubbleFrameView); +}; + +} // namespace views + +#endif // VIEWS_BUBBLE_BUBBLE_FRAME_VIEW_H_ diff --git a/views/bubble/bubble_frame_view_unittest.cc b/views/bubble/bubble_frame_view_unittest.cc new file mode 100644 index 0000000..b7153bd --- /dev/null +++ b/views/bubble/bubble_frame_view_unittest.cc @@ -0,0 +1,76 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "base/memory/scoped_ptr.h" +#include "base/message_loop.h" +#include "third_party/skia/include/core/SkColor.h" +#include "testing/gtest/include/gtest/gtest.h" +#include "views/bubble/bubble_border.h" +#include "views/bubble/bubble_frame_view.h" +#include "views/bubble/bubble_delegate.h" +#include "views/widget/widget.h" +#if !defined(OS_WIN) +#include "views/window/hit_test.h" +#endif +namespace views { + +namespace { + +gfx::Rect kBound = gfx::Rect(10, 10, 200, 200); +SkColor kBackgroundColor = SK_ColorRED; +BubbleBorder::ArrowLocation kArrow = BubbleBorder::LEFT_BOTTOM; + +TEST(BubbleFrameViewBasicTest, GetBoundsForClientView) { + MessageLoopForUI message_loop; + scoped_ptr<Widget> widget(new views::Widget()); + views::Widget::InitParams params(views::Widget::InitParams::TYPE_BUBBLE); + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; + widget->Init(params); + BubbleFrameView frame(widget.get(), kBound, kBackgroundColor, kArrow); + EXPECT_EQ(kBound, frame.bounds()); + EXPECT_EQ(kArrow, + static_cast<BubbleBorder*>(frame.border())->arrow_location()); + EXPECT_EQ(kBackgroundColor, + static_cast<BubbleBorder*>(frame.border())->background_color()); + + gfx::Insets expected_insets; + frame.border()->GetInsets(&expected_insets); + EXPECT_EQ(expected_insets.left(), frame.GetBoundsForClientView().x()); + EXPECT_EQ(expected_insets.top(), frame.GetBoundsForClientView().y()); + widget->CloseNow(); + widget.reset(NULL); + MessageLoop::current()->RunAllPending(); +} + +class TestBubbleDelegate : public BubbleDelegateView { + public: + explicit TestBubbleDelegate(Widget *frame): BubbleDelegateView(frame) {} + SkColor GetFrameBackgroundColor() { return kBackgroundColor; } + gfx::Rect GetBounds() { return gfx::Rect(10, 10, 200, 200); } + BubbleBorder::ArrowLocation GetFrameArrowLocation() { return kArrow; } + View* GetContentsView() { return &view_; } + + View view_; +}; + +TEST(BubbleFrameViewBasicTest, NonClientHitTest) { + MessageLoopForUI message_loop; + scoped_ptr<Widget> widget(new Widget()); + views::Widget::InitParams params(views::Widget::InitParams::TYPE_BUBBLE); + TestBubbleDelegate delegate(widget.get()); + params.delegate = &delegate; + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; + widget->Init(params); + gfx::Point kPtInBound(100, 100); + gfx::Point kPtOutsideBound(1000, 1000); + EXPECT_EQ(HTCLIENT, widget->non_client_view()->NonClientHitTest(kPtInBound)); + EXPECT_EQ(HTNOWHERE, + widget->non_client_view()->NonClientHitTest(kPtOutsideBound)); + widget->CloseNow(); + widget.reset(NULL); + MessageLoop::current()->RunAllPending(); +} + +} // namespace +} // namespace views diff --git a/views/bubble/bubble_view.cc b/views/bubble/bubble_view.cc new file mode 100644 index 0000000..5397f6e --- /dev/null +++ b/views/bubble/bubble_view.cc @@ -0,0 +1,128 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "views/bubble/bubble_view.h" + +#include "ui/base/animation/slide_animation.h" +#include "views/bubble/bubble_border.h" +#include "views/controls/label.h" +#include "views/layout/box_layout.h" +#include "views/layout/layout_constants.h" +#include "views/views_delegate.h" +#include "views/window/client_view.h" +#if defined(OS_WIN) +#include "views/widget/native_widget_win.h" +#else +#include "views/widget/native_widget_gtk.h" +#endif +#include "views/widget/widget.h" + +// How long the fade should last for. +static const int kHideFadeDurationMS = 1000; + +namespace views { + +BubbleView::BubbleView(Widget* owner, View* contents_view) + : ClientView(owner, contents_view), + animation_delegate_(NULL), + registered_accelerator_(false), + should_fade_(false) { + ResetLayoutManager(); + InitAnimation(); +} + +void BubbleView::InitAnimation() { + fade_animation_.reset(new ui::SlideAnimation(this)); + fade_animation_->SetSlideDuration(kHideFadeDurationMS); + fade_animation_->Reset(1.0); +} + +BubbleView* BubbleView::AsBubbleView() { return this; } +const BubbleView* BubbleView::AsBubbleView() const { return this; } + +BubbleView::~BubbleView() {} + +void BubbleView::Show() { + if (!registered_accelerator_) + registered_accelerator_ = true; + GetWidget()->Show(); + GetWidget()->Activate(); + SchedulePaint(); +} + +void BubbleView::StartFade() { + should_fade_ = true; + fade_animation_->Hide(); +} + +void BubbleView::ResetLayoutManager() { + SetLayoutManager(new views::BoxLayout( + views::BoxLayout::kHorizontal, 0, 0, 1)); +} + +void BubbleView::set_animation_delegate(ui::AnimationDelegate* delegate) { + animation_delegate_ = delegate; +} + +void BubbleView::ViewHierarchyChanged(bool is_add, + views::View* parent, + views::View* child) { + if (is_add && parent == this) + child->SetBoundsRect(bounds()); +} + +void BubbleView::Layout() { + gfx::Rect lb = GetContentsBounds(); + contents_view()->SetBoundsRect(lb); + contents_view()->Layout(); +} + +gfx::Size BubbleView::GetPreferredSize() { + return bounds().size(); +} + +bool BubbleView::AcceleratorPressed(const Accelerator& accelerator) { + if (registered_accelerator_) { + GetWidget()->GetFocusManager()->UnregisterAccelerator( + views::Accelerator(ui::VKEY_ESCAPE, false, false, false), this); + registered_accelerator_ = false; + } + // Turn off animation, if any. + if (should_fade_ && fade_animation_->is_animating()) { + fade_animation_->Reset(1.0); + fade_animation_->Show(); + } + GetWidget()->Close(); + return true; +} + +void BubbleView::AnimationEnded(const ui::Animation* animation) { + if (animation_delegate_) + animation_delegate_->AnimationEnded(animation); + + fade_animation_->Reset(0.0); + // Close the widget. + registered_accelerator_ = false; + GetWidget()->Close(); +} + +void BubbleView::AnimationProgressed(const ui::Animation* animation) { + if (fade_animation_->is_animating()) { + if (animation_delegate_) + animation_delegate_->AnimationProgressed(animation); + + SkColor opacity = static_cast<SkColor>( + animation->GetCurrentValue() * 255); +#if defined(OS_WIN) + SetLayeredWindowAttributes(GetWidget()->GetNativeView(), 0, + static_cast<byte>(opacity), LWA_ALPHA); +#else + static_cast<NativeWidgetGtk*>(GetWidget()->native_widget())->SetOpacity( + opacity); +#endif + SchedulePaint(); + } +} + +} // namespace views diff --git a/views/bubble/bubble_view.h b/views/bubble/bubble_view.h new file mode 100644 index 0000000..53fdabf --- /dev/null +++ b/views/bubble/bubble_view.h @@ -0,0 +1,87 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#ifndef VIEWS_BUBBLE_BUBBLE_VIEW_H_ +#define VIEWS_BUBBLE_BUBBLE_VIEW_H_ +#pragma once + +#include <string> + +#include "base/gtest_prod_util.h" +#include "base/task.h" +#include "third_party/skia/include/core/SkColor.h" +#include "ui/base/animation/animation_delegate.h" +#include "views/accelerator.h" +#include "views/focus/focus_manager.h" +#include "views/view.h" +#include "views/window/client_view.h" + +namespace ui { +class SlideAnimation; +} // namespace ui + +namespace views { + +// To show a bubble: +// - Call Show() explicitly. This will not start a fading animation. +// - Call StartFade() this starts a fading out sequence that will be +// cut short on VKEY_ESCAPE. +class VIEWS_EXPORT BubbleView : public ClientView, + public ui::AnimationDelegate { + public: + BubbleView(Widget* widget, View* contents_view); + virtual ~BubbleView(); + + // Starts a fade (out) animation. Unless this method is called, bubble will + // stay until ui::VKEY_ESCAPE is sent. + void StartFade(); + + // Shows the bubble. + void Show(); + + void set_animation_delegate(ui::AnimationDelegate* delegate); + + virtual BubbleView* AsBubbleView() OVERRIDE; + virtual const BubbleView* AsBubbleView() const OVERRIDE; + + protected: + virtual void ViewHierarchyChanged(bool is_add, + views::View* parent, + views::View* child) OVERRIDE; + + virtual bool AcceleratorPressed(const Accelerator& accelerator) OVERRIDE; + virtual void Layout() OVERRIDE; + virtual gfx::Size GetPreferredSize() OVERRIDE; + + private: + FRIEND_TEST(BubbleViewTest, FadeAnimation); + + void InitAnimation(); + + // Sets up the layout manager based on currently initialized views. Should be + // called when a view is initialized or changed. + void ResetLayoutManager(); + + // Close bubble when animation ended. + virtual void AnimationEnded(const ui::Animation* animation); + + // notify on animation progress. + virtual void AnimationProgressed(const ui::Animation* animation); + + // Fade animation for bubble. + scoped_ptr<ui::SlideAnimation> fade_animation_; + + // Not owned. + ui::AnimationDelegate* animation_delegate_; + + bool registered_accelerator_; + + bool should_fade_; + + DISALLOW_COPY_AND_ASSIGN(BubbleView); +}; + +} // namespace views + +#endif // VIEWS_BUBBLE_BUBBLE_VIEW_H_ diff --git a/views/bubble/bubble_view_unittest.cc b/views/bubble/bubble_view_unittest.cc new file mode 100644 index 0000000..d64fbb6 --- /dev/null +++ b/views/bubble/bubble_view_unittest.cc @@ -0,0 +1,90 @@ +// Copyright (c) 2011 The Chromium Authors. All rights reserved. +// Use of this source code is governed by a BSD-style license that can be +// found in the LICENSE file. + +#include "base/message_loop.h" +#include "third_party/skia/include/core/SkColor.h" +#include "testing/gtest/include/gtest/gtest.h" +#include "ui/base/animation/slide_animation.h" +#include "views/bubble/bubble_border.h" +#include "views/bubble/bubble_delegate.h" +#include "views/bubble/bubble_view.h" +#include "views/widget/widget.h" + +namespace views { + +namespace { + +class TestBubbleDelegate : public BubbleDelegateView { + public: + explicit TestBubbleDelegate(Widget *frame): BubbleDelegateView(frame) {} + SkColor GetFrameBackgroundColor() { return SK_ColorGREEN; } + gfx::Rect GetBounds() { return gfx::Rect(10, 10, 200, 200); } + BubbleBorder::ArrowLocation GetFrameArrowLocation() { + return BubbleBorder::LEFT_BOTTOM; + } + View* GetContentsView() { return &view_; } + + View view_; +}; + +class TestAnimationDelegate : public ui::AnimationDelegate { + public: + TestAnimationDelegate():animation_progressed_(0), animation_ended_(0) {} + void AnimationProgressed(const ui::Animation* animation) { + ++animation_progressed_; + } + void AnimationEnded(const ui::Animation* animation) { + ++animation_ended_; + } + int animation_progressed_; + int animation_ended_; +}; + + +TEST(BubbleViewBasicTest, CreateArrowBubble) { + MessageLoopForUI message_loop; + scoped_ptr<Widget> bubble_widget(new Widget()); + Widget::InitParams params(Widget::InitParams::TYPE_BUBBLE); + TestBubbleDelegate delegate(bubble_widget.get()); + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; + params.delegate = &delegate; + bubble_widget->Init(params); + + BubbleBorder* border = + static_cast<BubbleBorder*>(bubble_widget->non_client_view() + ->frame_view()->border()); + EXPECT_EQ(delegate.GetFrameArrowLocation(), border->arrow_location()); + bubble_widget->CloseNow(); + bubble_widget.reset(NULL); + MessageLoop::current()->RunAllPending(); +} + +} // namespace + +TEST(BubbleViewTest, FadeAnimation) { + MessageLoopForUI message_loop; + + scoped_ptr<Widget> bubble_widget(new Widget()); + Widget::InitParams params(Widget::InitParams::TYPE_BUBBLE); + TestBubbleDelegate delegate(bubble_widget.get()); + params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; + params.delegate = &delegate; + bubble_widget->Init(params); + bubble_widget->Show(); + BubbleView* bubble_view = bubble_widget->client_view()->AsBubbleView(); + TestAnimationDelegate test_animation_delegate; + bubble_view->set_animation_delegate(&test_animation_delegate); + bubble_view->StartFade(); + + bubble_view->AnimationProgressed(bubble_view->fade_animation_.get()); + bubble_view->AnimationEnded(bubble_view->fade_animation_.get()); + + EXPECT_LT(0, test_animation_delegate.animation_progressed_); + EXPECT_EQ(1, test_animation_delegate.animation_ended_); + bubble_widget->CloseNow(); + bubble_widget.reset(NULL); + MessageLoop::current()->RunAllPending(); +} + +} // namespace views |