• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "chrome/browser/ui/gtk/gtk_chrome_shrinkable_hbox.h"
6 
7 #include <vector>
8 
9 #include "testing/gtest/include/gtest/gtest.h"
10 
11 namespace {
12 
13 const int kSpacing = 3;
14 const int kBorderWidth = 5;
15 
16 }  // namespace
17 
18 class GtkChromeShrinkableHBoxTest : public testing::Test {
19  protected:
GtkChromeShrinkableHBoxTest()20   GtkChromeShrinkableHBoxTest()
21       : window_(gtk_window_new(GTK_WINDOW_TOPLEVEL)),
22         box_(gtk_chrome_shrinkable_hbox_new(FALSE, FALSE, kSpacing)) {
23     gtk_widget_set_direction(box_, GTK_TEXT_DIR_LTR);
24     gtk_window_set_default_size(GTK_WINDOW(window_), 200, 200);
25     gtk_container_add(GTK_CONTAINER(window_), box_);
26     gtk_container_set_border_width(GTK_CONTAINER(box_), kBorderWidth);
27   }
28 
~GtkChromeShrinkableHBoxTest()29   ~GtkChromeShrinkableHBoxTest() {
30     gtk_widget_destroy(window_);
31   }
32 
33   // Add some children widgets with arbitrary width and padding.
AddChildren(bool pack_start)34   void AddChildren(bool pack_start) {
35     static struct {
36       int width;
37       int padding;
38     } kChildrenData[] = {
39       { 60, 2 },
40       { 70, 3 },
41       { 80, 5 },
42       { 50, 7 },
43       { 40, 11 },
44       { 60, 0 },
45       { 0, 0 }
46     };
47 
48     for (size_t i = 0; kChildrenData[i].width; ++i) {
49       GtkWidget* child = gtk_fixed_new();
50       gtk_widget_set_size_request(child, kChildrenData[i].width, -1);
51       if (pack_start) {
52         gtk_chrome_shrinkable_hbox_pack_start(
53             GTK_CHROME_SHRINKABLE_HBOX(box_), child, kChildrenData[i].padding);
54       } else {
55         gtk_chrome_shrinkable_hbox_pack_end(
56             GTK_CHROME_SHRINKABLE_HBOX(box_), child, kChildrenData[i].padding);
57       }
58     }
59   }
60 
61   // Check if all children's size allocation are inside the |box_|'s boundary.
Validate(bool pack_start)62   void Validate(bool pack_start) {
63     std::vector<ChildData> children_data;
64     gtk_container_foreach(GTK_CONTAINER(box_), CollectChildData,
65                           &children_data);
66 
67     size_t children_count = children_data.size();
68     size_t visible_children_count = 0;
69     for (size_t i = 0; i < children_count; ++i) {
70       if (children_data[i].visible)
71         ++visible_children_count;
72     }
73 
74     if (visible_children_count == 0)
75       return;
76 
77     int border_width = gtk_container_get_border_width(GTK_CONTAINER(box_));
78     int x = box_->allocation.x + border_width;
79     int width = box_->allocation.width - border_width * 2;
80     int spacing = gtk_box_get_spacing(GTK_BOX(box_));
81     bool homogeneous = gtk_box_get_homogeneous(GTK_BOX(box_));
82 
83     if (homogeneous) {
84       // If the |box_| is in homogeneous mode, then check if the visible
85       // children are not overlapped with each other.
86       int homogeneous_child_width =
87           (width - (visible_children_count - 1) * spacing) /
88           visible_children_count;
89 
90       for (size_t i = 0; i < children_count; ++i) {
91         SCOPED_TRACE(testing::Message() << "Validate homogeneous child " << i
92                      << " visible: " << children_data[i].visible
93                      << " padding: " << children_data[i].padding
94                      << " x: " << children_data[i].x
95                      << " width: " << children_data[i].width);
96 
97         if (children_data[i].visible)
98           ASSERT_LE(children_data[i].width, homogeneous_child_width);
99       }
100     } else {
101       // If the |box_| is not in homogeneous mode, then just check if all
102       // visible children are inside the |box_|'s boundary. And for those
103       // hidden children which are out of the boundary, they should only
104       // be hidden one by one from the end of the |box_|.
105       bool last_visible = pack_start;
106       bool visibility_changed = false;
107       for (size_t i = 0; i < children_count; ++i) {
108         SCOPED_TRACE(testing::Message() << "Validate child " << i
109                      << " visible: " << children_data[i].visible
110                      << " padding: " << children_data[i].padding
111                      << " x: " << children_data[i].x
112                      << " width: " << children_data[i].width);
113 
114         if (last_visible != children_data[i].visible) {
115           ASSERT_FALSE(visibility_changed);
116           visibility_changed = true;
117           last_visible = children_data[i].visible;
118         }
119         if (children_data[i].visible) {
120           ASSERT_GE(children_data[i].x,
121                     x + children_data[i].padding);
122           ASSERT_LE(children_data[i].x + children_data[i].width,
123                     x + width - children_data[i].padding);
124         }
125       }
126     }
127   }
128 
Test(bool pack_start)129   void Test(bool pack_start) {
130     gtk_widget_show_all(window_);
131     GtkAllocation allocation = { 0, 0, 0, 200 };
132     gtk_chrome_shrinkable_hbox_set_hide_child_directly(
133         GTK_CHROME_SHRINKABLE_HBOX(box_), FALSE);
134     for (int width = 500; width > kBorderWidth * 2; --width) {
135       SCOPED_TRACE(testing::Message() << "Shrink hide_child_directly = FALSE,"
136                    << " width = " << width);
137 
138       allocation.width = width;
139       // Reducing the width may cause some children to be hidden, which will
140       // cause queue resize, so it's necessary to do another size allocation to
141       // emulate the queue resize.
142       gtk_widget_size_allocate(box_, &allocation);
143       gtk_widget_size_allocate(box_, &allocation);
144       ASSERT_NO_FATAL_FAILURE(Validate(pack_start)) << "width = " << width;
145     }
146 
147     for (int width = kBorderWidth * 2; width <= 500; ++width) {
148       SCOPED_TRACE(testing::Message() << "Expand hide_child_directly = FALSE,"
149                    << " width = " << width);
150 
151       allocation.width = width;
152       // Expanding the width may cause some invisible children to be shown,
153       // which will cause queue resize, so it's necessary to do another size
154       // allocation to emulate the queue resize.
155       gtk_widget_size_allocate(box_, &allocation);
156       gtk_widget_size_allocate(box_, &allocation);
157       ASSERT_NO_FATAL_FAILURE(Validate(pack_start));
158     }
159 
160     gtk_chrome_shrinkable_hbox_set_hide_child_directly(
161         GTK_CHROME_SHRINKABLE_HBOX(box_), TRUE);
162     for (int width = 500; width > kBorderWidth * 2; --width) {
163       SCOPED_TRACE(testing::Message() << "Shrink hide_child_directly = TRUE,"
164                    << " width = " << width);
165 
166       allocation.width = width;
167       gtk_widget_size_allocate(box_, &allocation);
168       gtk_widget_size_allocate(box_, &allocation);
169       ASSERT_NO_FATAL_FAILURE(Validate(pack_start));
170     }
171 
172     for (int width = kBorderWidth * 2; width <= 500; ++width) {
173       SCOPED_TRACE(testing::Message() << "Expand hide_child_directly = TRUE,"
174                    << " width = " << width);
175 
176       allocation.width = width;
177       gtk_widget_size_allocate(box_, &allocation);
178       gtk_widget_size_allocate(box_, &allocation);
179       ASSERT_NO_FATAL_FAILURE(Validate(pack_start));
180     }
181   }
182 
183  protected:
184   GtkWidget* window_;
185   GtkWidget* box_;
186 
187  private:
188   struct ChildData {
189     bool visible;
190     int padding;
191     int x;
192     int width;
193   };
194 
CollectChildData(GtkWidget * child,gpointer userdata)195   static void CollectChildData(GtkWidget* child, gpointer userdata) {
196     guint padding;
197     gtk_box_query_child_packing(GTK_BOX(gtk_widget_get_parent(child)), child,
198                                 NULL, NULL, &padding, NULL);
199 
200     ChildData data;
201     data.visible = GTK_WIDGET_VISIBLE(child);
202     data.padding = padding;
203     data.x = child->allocation.x;
204     data.width = child->allocation.width;
205 
206     reinterpret_cast<std::vector<ChildData>*>(userdata)->push_back(data);
207   }
208 };
209 
TEST_F(GtkChromeShrinkableHBoxTest,PackStart)210 TEST_F(GtkChromeShrinkableHBoxTest, PackStart) {
211   AddChildren(true);
212 
213   {
214     SCOPED_TRACE("Test LTR");
215     gtk_widget_set_direction(box_, GTK_TEXT_DIR_LTR);
216     EXPECT_NO_FATAL_FAILURE(Test(true));
217   }
218   {
219     SCOPED_TRACE("Test RTL");
220     gtk_widget_set_direction(box_, GTK_TEXT_DIR_RTL);
221     EXPECT_NO_FATAL_FAILURE(Test(true));
222   }
223 }
224 
TEST_F(GtkChromeShrinkableHBoxTest,PackEnd)225 TEST_F(GtkChromeShrinkableHBoxTest, PackEnd) {
226   AddChildren(false);
227 
228   {
229     SCOPED_TRACE("Test LTR");
230     gtk_widget_set_direction(box_, GTK_TEXT_DIR_LTR);
231     EXPECT_NO_FATAL_FAILURE(Test(false));
232   }
233   {
234     SCOPED_TRACE("Test RTL");
235     gtk_widget_set_direction(box_, GTK_TEXT_DIR_RTL);
236     EXPECT_NO_FATAL_FAILURE(Test(false));
237   }
238 }
239 
TEST_F(GtkChromeShrinkableHBoxTest,Homogeneous)240 TEST_F(GtkChromeShrinkableHBoxTest, Homogeneous) {
241   AddChildren(true);
242   gtk_box_set_homogeneous(GTK_BOX(box_), true);
243 
244   {
245     SCOPED_TRACE("Test LTR");
246     gtk_widget_set_direction(box_, GTK_TEXT_DIR_LTR);
247     EXPECT_NO_FATAL_FAILURE(Test(true));
248   }
249   {
250     SCOPED_TRACE("Test RTL");
251     gtk_widget_set_direction(box_, GTK_TEXT_DIR_RTL);
252     EXPECT_NO_FATAL_FAILURE(Test(true));
253   }
254 }
255