Upstream version 8.37.180.0
[platform/framework/web/crosswalk.git] / src / ui / views / controls / single_split_view_unittest.cc
1 // Copyright (c) 2011 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "ui/views/controls/single_split_view.h"
6
7 #include "base/logging.h"
8 #include "testing/gtest/include/gtest/gtest.h"
9 #include "ui/views/controls/single_split_view_listener.h"
10
11 namespace {
12
13 static void VerifySplitViewLayout(const views::SingleSplitView& split) {
14   ASSERT_EQ(2, split.child_count());
15
16   const views::View* leading = split.child_at(0);
17   const views::View* trailing = split.child_at(1);
18
19   if (split.bounds().IsEmpty()) {
20     EXPECT_TRUE(leading->bounds().IsEmpty());
21     EXPECT_TRUE(trailing->bounds().IsEmpty());
22     return;
23   }
24
25   EXPECT_FALSE(leading->bounds().IsEmpty());
26   EXPECT_FALSE(trailing->bounds().IsEmpty());
27   EXPECT_FALSE(leading->bounds().Intersects(trailing->bounds()));
28
29   if (split.orientation() == views::SingleSplitView::HORIZONTAL_SPLIT) {
30     EXPECT_EQ(leading->bounds().height(), split.bounds().height());
31     EXPECT_EQ(trailing->bounds().height(), split.bounds().height());
32     EXPECT_LT(leading->bounds().width() + trailing->bounds().width(),
33               split.bounds().width());
34   } else if (split.orientation() == views::SingleSplitView::VERTICAL_SPLIT) {
35     EXPECT_EQ(leading->bounds().width(), split.bounds().width());
36     EXPECT_EQ(trailing->bounds().width(), split.bounds().width());
37     EXPECT_LT(leading->bounds().height() + trailing->bounds().height(),
38               split.bounds().height());
39   } else {
40     NOTREACHED();
41   }
42 }
43
44 class SingleSplitViewListenerImpl : public views::SingleSplitViewListener {
45  public:
46   SingleSplitViewListenerImpl() : count_(0) {}
47
48   virtual bool SplitHandleMoved(views::SingleSplitView* sender) OVERRIDE {
49     ++count_;
50     return false;
51   }
52
53   int count() const { return count_; }
54
55  private:
56   int count_;
57
58   DISALLOW_COPY_AND_ASSIGN(SingleSplitViewListenerImpl);
59 };
60
61 class MinimumSizedView: public views::View {
62  public:
63   MinimumSizedView(gfx::Size min_size) : min_size_(min_size) {}
64
65  private:
66   gfx::Size min_size_;
67   virtual gfx::Size GetMinimumSize() const OVERRIDE;
68 };
69
70 gfx::Size MinimumSizedView::GetMinimumSize() const {
71   return min_size_;
72 }
73
74 }  // namespace
75
76 namespace views {
77
78 TEST(SingleSplitViewTest, Resize) {
79   // Test cases to iterate through for horizontal and vertical split views.
80   struct TestCase {
81     // Split view resize policy for this test case.
82     bool resize_leading_on_bounds_change;
83     // Split view size to set.
84     int primary_axis_size;
85     int secondary_axis_size;
86     // Expected divider offset.
87     int divider_offset;
88   } test_cases[] = {
89     // The initial split size is 100x100, divider at 33.
90     { true, 100, 100, 33 },
91     // Grow the split view, leading view should grow.
92     { true, 1000, 100, 933 },
93     // Shrink the split view, leading view should shrink.
94     { true, 200, 100, 133 },
95     // Minimize the split view, divider should not move.
96     { true, 0, 0, 133 },
97     // Restore the split view, divider should not move.
98     { false, 500, 100, 133 },
99     // Resize the split view by secondary axis, divider should not move.
100     { false,  500, 600, 133 }
101   };
102
103   SingleSplitView::Orientation orientations[] = {
104     SingleSplitView::HORIZONTAL_SPLIT,
105     SingleSplitView::VERTICAL_SPLIT
106   };
107
108   for (size_t orientation = 0; orientation < arraysize(orientations);
109        ++orientation) {
110     // Create a split view.
111     SingleSplitView split(
112         new View(), new View(), orientations[orientation], NULL);
113
114     // Set initial size and divider offset.
115     EXPECT_EQ(test_cases[0].primary_axis_size,
116               test_cases[0].secondary_axis_size);
117     split.SetBounds(0, 0, test_cases[0].primary_axis_size,
118                     test_cases[0].secondary_axis_size);
119     split.set_divider_offset(test_cases[0].divider_offset);
120     split.Layout();
121
122     // Run all test cases.
123     for (size_t i = 0; i < ARRAYSIZE_UNSAFE(test_cases); ++i) {
124       split.set_resize_leading_on_bounds_change(
125           test_cases[i].resize_leading_on_bounds_change);
126       if (split.orientation() == SingleSplitView::HORIZONTAL_SPLIT) {
127         split.SetBounds(0, 0, test_cases[i].primary_axis_size,
128                         test_cases[i].secondary_axis_size);
129       } else {
130         split.SetBounds(0, 0, test_cases[i].secondary_axis_size,
131                         test_cases[i].primary_axis_size);
132       }
133
134       EXPECT_EQ(test_cases[i].divider_offset, split.divider_offset());
135       VerifySplitViewLayout(split);
136     }
137
138     // Special cases, one of the child views is hidden.
139     split.child_at(0)->SetVisible(false);
140     split.Layout();
141
142     EXPECT_EQ(split.size(), split.child_at(1)->size());
143
144     split.child_at(0)->SetVisible(true);
145     split.child_at(1)->SetVisible(false);
146     split.Layout();
147
148     EXPECT_EQ(split.size(), split.child_at(0)->size());
149   }
150 }
151
152 TEST(SingleSplitViewTest, MouseDrag) {
153   const int kMinimumChildSize = 25;
154   MinimumSizedView *child0 =
155       new MinimumSizedView(gfx::Size(5, kMinimumChildSize));
156   MinimumSizedView *child1 =
157       new MinimumSizedView(gfx::Size(5, kMinimumChildSize));
158   SingleSplitViewListenerImpl listener;
159   SingleSplitView split(
160       child0, child1, SingleSplitView::VERTICAL_SPLIT, &listener);
161
162   const int kTotalSplitSize = 100;
163   split.SetBounds(0, 0, 10, kTotalSplitSize);
164   const int kInitialDividerOffset = 33;
165   const int kMouseOffset = 2;  // Mouse offset in the divider.
166   const int kMouseMoveDelta = 7;
167   split.set_divider_offset(kInitialDividerOffset);
168   split.Layout();
169
170   gfx::Point press_point(7, kInitialDividerOffset + kMouseOffset);
171   ui::MouseEvent mouse_pressed(
172       ui::ET_MOUSE_PRESSED, press_point, press_point, 0, 0);
173   ASSERT_TRUE(split.OnMousePressed(mouse_pressed));
174   EXPECT_EQ(kInitialDividerOffset, split.divider_offset());
175   EXPECT_EQ(0, listener.count());
176
177   // Drag divider to the bottom.
178   gfx::Point drag_1_point(
179       5, kInitialDividerOffset + kMouseOffset + kMouseMoveDelta);
180   ui::MouseEvent mouse_dragged_1(
181       ui::ET_MOUSE_DRAGGED, drag_1_point, drag_1_point, 0, 0);
182   ASSERT_TRUE(split.OnMouseDragged(mouse_dragged_1));
183   EXPECT_EQ(kInitialDividerOffset + kMouseMoveDelta, split.divider_offset());
184   EXPECT_EQ(1, listener.count());
185
186   // Drag divider to the top, beyond first child minimum size.
187   gfx::Point drag_2_point(
188       7, kMinimumChildSize - 5);
189   ui::MouseEvent mouse_dragged_2(
190       ui::ET_MOUSE_DRAGGED, drag_2_point, drag_2_point, 0,0 );
191   ASSERT_TRUE(split.OnMouseDragged(mouse_dragged_2));
192   EXPECT_EQ(kMinimumChildSize, split.divider_offset());
193   EXPECT_EQ(2, listener.count());
194
195   // Drag divider to the bottom, beyond second child minimum size.
196   gfx::Point drag_3_point(
197       7, kTotalSplitSize - kMinimumChildSize + 5);
198   ui::MouseEvent mouse_dragged_3(
199       ui::ET_MOUSE_DRAGGED, drag_3_point, drag_3_point, 0, 0);
200   ASSERT_TRUE(split.OnMouseDragged(mouse_dragged_3));
201   EXPECT_EQ(kTotalSplitSize - kMinimumChildSize - split.GetDividerSize(),
202             split.divider_offset());
203   EXPECT_EQ(3, listener.count());
204
205   // Drag divider between childs' minimum sizes.
206   gfx::Point drag_4_point(
207       6, kInitialDividerOffset + kMouseOffset + kMouseMoveDelta * 2);
208   ui::MouseEvent mouse_dragged_4(
209       ui::ET_MOUSE_DRAGGED, drag_4_point, drag_4_point, 0, 0);
210   ASSERT_TRUE(split.OnMouseDragged(mouse_dragged_4));
211   EXPECT_EQ(kInitialDividerOffset + kMouseMoveDelta * 2,
212             split.divider_offset());
213   EXPECT_EQ(4, listener.count());
214
215   gfx::Point release_point(
216       7, kInitialDividerOffset + kMouseOffset + kMouseMoveDelta * 2);
217   ui::MouseEvent mouse_released(
218       ui::ET_MOUSE_RELEASED, release_point, release_point, 0, 0);
219   split.OnMouseReleased(mouse_released);
220   EXPECT_EQ(kInitialDividerOffset + kMouseMoveDelta * 2,
221             split.divider_offset());
222
223   // Expect intial offset after a system/user gesture cancels the drag.
224   // This shouldn't occur after mouse release, but it's sufficient for testing.
225   split.OnMouseCaptureLost();
226   EXPECT_EQ(kInitialDividerOffset, split.divider_offset());
227   EXPECT_EQ(5, listener.count());
228 }
229
230 }  // namespace views