| // Copyright 2013 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 <stddef.h> |
| |
| #include "base/macros.h" |
| #include "base/strings/utf_string_conversions.h" |
| #include "ui/base/hit_test.h" |
| #include "ui/events/event_processor.h" |
| #include "ui/views/bubble/bubble_border.h" |
| #include "ui/views/bubble/bubble_frame_view.h" |
| #include "ui/views/controls/button/checkbox.h" |
| #include "ui/views/controls/button/label_button.h" |
| #include "ui/views/controls/textfield/textfield.h" |
| #include "ui/views/test/views_test_base.h" |
| #include "ui/views/widget/widget.h" |
| #include "ui/views/window/dialog_client_view.h" |
| #include "ui/views/window/dialog_delegate.h" |
| |
| namespace views { |
| |
| namespace { |
| |
| class TestDialog : public DialogDelegateView, public ButtonListener { |
| public: |
| TestDialog() |
| : input_(new views::Textfield()), |
| canceled_(false), |
| accepted_(false), |
| closed_(false), |
| closeable_(false), |
| last_pressed_button_(nullptr), |
| should_handle_escape_(false) { |
| AddChildView(input_); |
| } |
| ~TestDialog() override {} |
| |
| void Init() { |
| // Add the accelerator before being added to the widget hierarchy (before |
| // DCV has registered its accelerator) to make sure accelerator handling is |
| // not dependent on the order of AddAccelerator calls. |
| EXPECT_FALSE(GetWidget()); |
| AddAccelerator(ui::Accelerator(ui::VKEY_ESCAPE, ui::EF_NONE)); |
| } |
| |
| // WidgetDelegate overrides: |
| bool ShouldShowWindowTitle() const override { |
| return !title_.empty(); |
| } |
| |
| // DialogDelegateView overrides: |
| bool Cancel() override { |
| canceled_ = true; |
| return closeable_; |
| } |
| bool Accept() override { |
| accepted_ = true; |
| return closeable_; |
| } |
| bool Close() override { |
| closed_ = true; |
| return closeable_; |
| } |
| |
| // DialogDelegateView overrides: |
| gfx::Size GetPreferredSize() const override { return gfx::Size(200, 200); } |
| bool AcceleratorPressed(const ui::Accelerator& accelerator) override { |
| return should_handle_escape_; |
| } |
| base::string16 GetWindowTitle() const override { return title_; } |
| View* GetInitiallyFocusedView() override { return input_; } |
| bool ShouldUseCustomFrame() const override { return true; } |
| |
| // ButtonListener override: |
| void ButtonPressed(Button* sender, const ui::Event& event) override { |
| last_pressed_button_ = sender; |
| } |
| |
| Button* last_pressed_button() const { return last_pressed_button_; } |
| |
| void CheckAndResetStates(bool canceled, |
| bool accepted, |
| bool closed, |
| Button* last_pressed) { |
| EXPECT_EQ(canceled, canceled_); |
| canceled_ = false; |
| EXPECT_EQ(accepted, accepted_); |
| accepted_ = false; |
| EXPECT_EQ(last_pressed, last_pressed_button_); |
| last_pressed_button_ = nullptr; |
| EXPECT_EQ(closed, closed_); |
| closed_ = false; |
| } |
| |
| void TearDown() { |
| closeable_ = true; |
| GetWidget()->Close(); |
| } |
| |
| void set_title(const base::string16& title) { title_ = title; } |
| void set_should_handle_escape(bool should_handle_escape) { |
| should_handle_escape_ = should_handle_escape; |
| } |
| |
| views::Textfield* input() { return input_; } |
| |
| private: |
| views::Textfield* input_; |
| bool canceled_; |
| bool accepted_; |
| bool closed_; |
| // Prevent the dialog from closing, for repeated ok and cancel button clicks. |
| bool closeable_; |
| Button* last_pressed_button_; |
| base::string16 title_; |
| bool should_handle_escape_; |
| |
| DISALLOW_COPY_AND_ASSIGN(TestDialog); |
| }; |
| |
| class DialogTest : public ViewsTestBase { |
| public: |
| DialogTest() : dialog_(nullptr) {} |
| ~DialogTest() override {} |
| |
| void SetUp() override { |
| ViewsTestBase::SetUp(); |
| dialog_ = new TestDialog(); |
| dialog_->Init(); |
| DialogDelegate::CreateDialogWidget(dialog_, GetContext(), nullptr)->Show(); |
| } |
| |
| void TearDown() override { |
| dialog_->TearDown(); |
| ViewsTestBase::TearDown(); |
| } |
| |
| void SimulateKeyEvent(const ui::KeyEvent& event) { |
| ui::KeyEvent event_copy = event; |
| if (dialog()->GetFocusManager()->OnKeyEvent(event_copy)) |
| dialog()->GetWidget()->OnKeyEvent(&event_copy); |
| } |
| |
| TestDialog* dialog() const { return dialog_; } |
| |
| private: |
| TestDialog* dialog_; |
| |
| DISALLOW_COPY_AND_ASSIGN(DialogTest); |
| }; |
| |
| } // namespace |
| |
| TEST_F(DialogTest, AcceptAndCancel) { |
| DialogClientView* client_view = dialog()->GetDialogClientView(); |
| LabelButton* ok_button = client_view->ok_button(); |
| LabelButton* cancel_button = client_view->cancel_button(); |
| |
| // Check that return/escape accelerators accept/close dialogs. |
| EXPECT_EQ(dialog()->input(), dialog()->GetFocusManager()->GetFocusedView()); |
| const ui::KeyEvent return_event( |
| ui::ET_KEY_PRESSED, ui::VKEY_RETURN, ui::EF_NONE); |
| SimulateKeyEvent(return_event); |
| dialog()->CheckAndResetStates(false, true, false, nullptr); |
| const ui::KeyEvent escape_event( |
| ui::ET_KEY_PRESSED, ui::VKEY_ESCAPE, ui::EF_NONE); |
| SimulateKeyEvent(escape_event); |
| dialog()->CheckAndResetStates(false, false, true, nullptr); |
| |
| // Check ok and cancel button behavior on a directed return key events. |
| ok_button->OnKeyPressed(return_event); |
| dialog()->CheckAndResetStates(false, true, false, nullptr); |
| cancel_button->OnKeyPressed(return_event); |
| dialog()->CheckAndResetStates(true, false, false, nullptr); |
| |
| // Check that return accelerators cancel dialogs if cancel is focused. |
| cancel_button->RequestFocus(); |
| EXPECT_EQ(cancel_button, dialog()->GetFocusManager()->GetFocusedView()); |
| SimulateKeyEvent(return_event); |
| dialog()->CheckAndResetStates(true, false, false, nullptr); |
| |
| // Check that escape can be overridden. |
| dialog()->set_should_handle_escape(true); |
| SimulateKeyEvent(escape_event); |
| dialog()->CheckAndResetStates(false, false, false, nullptr); |
| } |
| |
| TEST_F(DialogTest, RemoveDefaultButton) { |
| // Removing buttons from the dialog here should not cause a crash on close. |
| delete dialog()->GetDialogClientView()->ok_button(); |
| delete dialog()->GetDialogClientView()->cancel_button(); |
| } |
| |
| TEST_F(DialogTest, HitTest_HiddenTitle) { |
| // Ensure that BubbleFrameView hit-tests as expected when the title is hidden. |
| const NonClientView* view = dialog()->GetWidget()->non_client_view(); |
| BubbleFrameView* frame = static_cast<BubbleFrameView*>(view->frame_view()); |
| const int border = frame->bubble_border()->GetBorderThickness(); |
| |
| struct { |
| const int point; |
| const int hit; |
| } cases[] = { |
| { border, HTSYSMENU }, |
| { border + 10, HTSYSMENU }, |
| { border + 20, HTCLIENT }, |
| { border + 50, HTCLIENT }, |
| { border + 60, HTCLIENT }, |
| { 1000, HTNOWHERE }, |
| }; |
| |
| for (size_t i = 0; i < arraysize(cases); ++i) { |
| gfx::Point point(cases[i].point, cases[i].point); |
| EXPECT_EQ(cases[i].hit, frame->NonClientHitTest(point)) |
| << " case " << i << " with border: " << border << ", at point " |
| << cases[i].point; |
| } |
| } |
| |
| TEST_F(DialogTest, HitTest_WithTitle) { |
| // Ensure that BubbleFrameView hit-tests as expected when the title is shown. |
| const NonClientView* view = dialog()->GetWidget()->non_client_view(); |
| dialog()->set_title(base::ASCIIToUTF16("Title")); |
| dialog()->GetWidget()->UpdateWindowTitle(); |
| BubbleFrameView* frame = static_cast<BubbleFrameView*>(view->frame_view()); |
| const int border = frame->bubble_border()->GetBorderThickness(); |
| |
| struct { |
| const int point; |
| const int hit; |
| } cases[] = { |
| { border, HTSYSMENU }, |
| { border + 10, HTSYSMENU }, |
| { border + 20, HTCAPTION }, |
| { border + 50, HTCLIENT }, |
| { border + 60, HTCLIENT }, |
| { 1000, HTNOWHERE }, |
| }; |
| |
| for (size_t i = 0; i < arraysize(cases); ++i) { |
| gfx::Point point(cases[i].point, cases[i].point); |
| EXPECT_EQ(cases[i].hit, frame->NonClientHitTest(point)) |
| << " with border: " << border << ", at point " << cases[i].point; |
| } |
| } |
| |
| TEST_F(DialogTest, BoundsAccommodateTitle) { |
| TestDialog* dialog2(new TestDialog()); |
| dialog2->set_title(base::ASCIIToUTF16("Title")); |
| DialogDelegate::CreateDialogWidget(dialog2, GetContext(), nullptr); |
| |
| // Titled dialogs have taller initial frame bounds than untitled dialogs. |
| View* frame1 = dialog()->GetWidget()->non_client_view()->frame_view(); |
| View* frame2 = dialog2->GetWidget()->non_client_view()->frame_view(); |
| EXPECT_LT(frame1->GetPreferredSize().height(), |
| frame2->GetPreferredSize().height()); |
| |
| // Giving the default test dialog a title will yield the same bounds. |
| dialog()->set_title(base::ASCIIToUTF16("Title")); |
| dialog()->GetWidget()->UpdateWindowTitle(); |
| EXPECT_EQ(frame1->GetPreferredSize().height(), |
| frame2->GetPreferredSize().height()); |
| |
| dialog2->TearDown(); |
| } |
| |
| // Tests default focus is assigned correctly when showing a new dialog. |
| TEST_F(DialogTest, InitialFocus) { |
| EXPECT_TRUE(dialog()->input()->HasFocus()); |
| EXPECT_EQ(dialog()->input(), dialog()->GetFocusManager()->GetFocusedView()); |
| } |
| |
| } // namespace views |