OLD | NEW |
(Empty) | |
| 1 // Copyright 2014 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 <vector> |
| 6 |
| 7 #include <X11/extensions/shape.h> |
| 8 #include <X11/Xlib.h> |
| 9 |
| 10 // Get rid of X11 macros which conflict with gtest. |
| 11 #undef Bool |
| 12 #undef None |
| 13 |
| 14 #include "base/memory/scoped_ptr.h" |
| 15 #include "base/run_loop.h" |
| 16 #include "ui/aura/window.h" |
| 17 #include "ui/aura/window_tree_host.h" |
| 18 #include "ui/base/hit_test.h" |
| 19 #include "ui/base/x/x11_util.h" |
| 20 #include "ui/events/platform/platform_event_dispatcher.h" |
| 21 #include "ui/events/platform/platform_event_source.h" |
| 22 #include "ui/events/platform/scoped_event_dispatcher.h" |
| 23 #include "ui/events/platform/x11/x11_event_source.h" |
| 24 #include "ui/gfx/path.h" |
| 25 #include "ui/gfx/point.h" |
| 26 #include "ui/gfx/rect.h" |
| 27 #include "ui/gfx/x/x11_atom_cache.h" |
| 28 #include "ui/views/test/views_test_base.h" |
| 29 #include "ui/views/widget/desktop_aura/desktop_native_widget_aura.h" |
| 30 #include "ui/views/widget/widget_delegate.h" |
| 31 #include "ui/views/window/non_client_view.h" |
| 32 |
| 33 namespace views { |
| 34 |
| 35 namespace { |
| 36 |
| 37 // Blocks till |window| becomes maximized. |
| 38 class MaximizeWaiter : public ui::PlatformEventDispatcher { |
| 39 public: |
| 40 explicit MaximizeWaiter(XID window) |
| 41 : xwindow_(window), |
| 42 wait_(true) { |
| 43 const char* kAtomsToCache[] = { |
| 44 "_NET_WM_STATE", |
| 45 "_NET_WM_STATE_MAXIMIZED_VERT", |
| 46 NULL |
| 47 }; |
| 48 atom_cache_.reset(new ui::X11AtomCache(gfx::GetXDisplay(), kAtomsToCache)); |
| 49 |
| 50 // Override the dispatcher so that we get events before |
| 51 // DesktopWindowTreeHostX11 does. We must do this because |
| 52 // DesktopWindowTreeHostX11 stops propagation. |
| 53 dispatcher_ = ui::PlatformEventSource::GetInstance()-> |
| 54 OverrideDispatcher(this).Pass(); |
| 55 } |
| 56 |
| 57 virtual ~MaximizeWaiter() { |
| 58 } |
| 59 |
| 60 void Wait() { |
| 61 if (wait_) { |
| 62 base::RunLoop run_loop; |
| 63 quit_closure_ = run_loop.QuitClosure(); |
| 64 run_loop.Run(); |
| 65 } |
| 66 dispatcher_.reset(); |
| 67 } |
| 68 |
| 69 virtual bool CanDispatchEvent(const ui::PlatformEvent& event) OVERRIDE { |
| 70 NOTREACHED(); |
| 71 return true; |
| 72 } |
| 73 |
| 74 virtual uint32_t DispatchEvent(const ui::PlatformEvent& event) OVERRIDE { |
| 75 if (event->type != PropertyNotify || |
| 76 event->xproperty.window != xwindow_ || |
| 77 event->xproperty.atom != atom_cache_->GetAtom("_NET_WM_STATE")) { |
| 78 return ui::POST_DISPATCH_PERFORM_DEFAULT; |
| 79 } |
| 80 |
| 81 std::vector<Atom> wm_states; |
| 82 if (!ui::GetAtomArrayProperty(xwindow_, "_NET_WM_STATE", &wm_states)) |
| 83 return ui::POST_DISPATCH_PERFORM_DEFAULT; |
| 84 |
| 85 std::vector<Atom>::iterator it = std::find( |
| 86 wm_states.begin(), |
| 87 wm_states.end(), |
| 88 atom_cache_->GetAtom("_NET_WM_STATE_MAXIMIZED_VERT")); |
| 89 if (it == wm_states.end()) |
| 90 return ui::POST_DISPATCH_PERFORM_DEFAULT; |
| 91 |
| 92 wait_ = false; |
| 93 if (!quit_closure_.is_null()) |
| 94 quit_closure_.Run(); |
| 95 return ui::POST_DISPATCH_PERFORM_DEFAULT; |
| 96 } |
| 97 |
| 98 // The window we are waiting to get maximized. |
| 99 XID xwindow_; |
| 100 |
| 101 // Whether Wait() should block. |
| 102 bool wait_; |
| 103 |
| 104 // Ends the run loop. |
| 105 base::Closure quit_closure_; |
| 106 |
| 107 scoped_ptr<ui::ScopedEventDispatcher> dispatcher_; |
| 108 |
| 109 scoped_ptr<ui::X11AtomCache> atom_cache_; |
| 110 |
| 111 DISALLOW_COPY_AND_ASSIGN(MaximizeWaiter); |
| 112 }; |
| 113 |
| 114 // A NonClientFrameView with a window mask with the bottom right corner cut out. |
| 115 class ShapedNonClientFrameView : public NonClientFrameView { |
| 116 public: |
| 117 explicit ShapedNonClientFrameView() { |
| 118 } |
| 119 |
| 120 virtual ~ShapedNonClientFrameView() { |
| 121 } |
| 122 |
| 123 // NonClientFrameView: |
| 124 virtual gfx::Rect GetBoundsForClientView() const OVERRIDE { |
| 125 return bounds(); |
| 126 } |
| 127 virtual gfx::Rect GetWindowBoundsForClientBounds( |
| 128 const gfx::Rect& client_bounds) const OVERRIDE { |
| 129 return client_bounds; |
| 130 } |
| 131 virtual int NonClientHitTest(const gfx::Point& point) OVERRIDE { |
| 132 return HTNOWHERE; |
| 133 } |
| 134 virtual void GetWindowMask(const gfx::Size& size, |
| 135 gfx::Path* window_mask) OVERRIDE { |
| 136 int right = size.width(); |
| 137 int bottom = size.height(); |
| 138 |
| 139 window_mask->moveTo(0, 0); |
| 140 window_mask->lineTo(0, bottom); |
| 141 window_mask->lineTo(right, bottom); |
| 142 window_mask->lineTo(right, 10); |
| 143 window_mask->lineTo(right - 10, 10); |
| 144 window_mask->lineTo(right - 10, 0); |
| 145 window_mask->close(); |
| 146 } |
| 147 virtual void ResetWindowControls() OVERRIDE { |
| 148 } |
| 149 virtual void UpdateWindowIcon() OVERRIDE { |
| 150 } |
| 151 virtual void UpdateWindowTitle() OVERRIDE { |
| 152 } |
| 153 |
| 154 private: |
| 155 DISALLOW_COPY_AND_ASSIGN(ShapedNonClientFrameView); |
| 156 }; |
| 157 |
| 158 class ShapedWidgetDelegate : public WidgetDelegateView { |
| 159 public: |
| 160 ShapedWidgetDelegate() { |
| 161 } |
| 162 |
| 163 virtual ~ShapedWidgetDelegate() { |
| 164 } |
| 165 |
| 166 // WidgetDelegateView: |
| 167 virtual NonClientFrameView* CreateNonClientFrameView( |
| 168 Widget* widget) OVERRIDE { |
| 169 return new ShapedNonClientFrameView; |
| 170 } |
| 171 |
| 172 private: |
| 173 DISALLOW_COPY_AND_ASSIGN(ShapedWidgetDelegate); |
| 174 }; |
| 175 |
| 176 // Creates a widget of size 100x100. |
| 177 scoped_ptr<Widget> CreateWidget(WidgetDelegate* delegate) { |
| 178 scoped_ptr<Widget> widget(new Widget); |
| 179 Widget::InitParams params(Widget::InitParams::TYPE_WINDOW); |
| 180 params.delegate = delegate; |
| 181 params.ownership = Widget::InitParams::WIDGET_OWNS_NATIVE_WIDGET; |
| 182 params.remove_standard_frame = true; |
| 183 params.native_widget = new DesktopNativeWidgetAura(widget.get()); |
| 184 params.bounds = gfx::Rect(100, 100, 100, 100); |
| 185 widget->Init(params); |
| 186 return widget.Pass(); |
| 187 } |
| 188 |
| 189 // Returns the list of rectangles which describe |xid|'s bounding region via the |
| 190 // X shape extension. |
| 191 std::vector<gfx::Rect> GetShapeRects(XID xid) { |
| 192 int dummy; |
| 193 int shape_rects_size; |
| 194 XRectangle* shape_rects = XShapeGetRectangles(gfx::GetXDisplay(), |
| 195 xid, |
| 196 ShapeBounding, |
| 197 &shape_rects_size, |
| 198 &dummy); |
| 199 |
| 200 std::vector<gfx::Rect> shape_vector; |
| 201 for (int i = 0; i < shape_rects_size; ++i) { |
| 202 shape_vector.push_back(gfx::Rect( |
| 203 shape_rects[i].x, |
| 204 shape_rects[i].y, |
| 205 shape_rects[i].width, |
| 206 shape_rects[i].height)); |
| 207 } |
| 208 XFree(shape_rects); |
| 209 return shape_vector; |
| 210 } |
| 211 |
| 212 // Returns true if one of |rects| contains point (x,y). |
| 213 bool ShapeRectContainsPoint(const std::vector<gfx::Rect>& shape_rects, |
| 214 int x, |
| 215 int y) { |
| 216 gfx::Point point(x, y); |
| 217 for (size_t i = 0; i < shape_rects.size(); ++i) { |
| 218 if (shape_rects[i].Contains(point)) |
| 219 return true; |
| 220 } |
| 221 return false; |
| 222 } |
| 223 |
| 224 } // namespace |
| 225 |
| 226 class DesktopWindowTreeHostX11Test : public ViewsTestBase { |
| 227 public: |
| 228 DesktopWindowTreeHostX11Test() { |
| 229 } |
| 230 virtual ~DesktopWindowTreeHostX11Test() { |
| 231 } |
| 232 |
| 233 virtual void SetUp() OVERRIDE { |
| 234 ViewsTestBase::SetUp(); |
| 235 |
| 236 // Make X11 synchronous for our display connection. This does not force the |
| 237 // window manager to behave synchronously. |
| 238 XSynchronize(gfx::GetXDisplay(), True); |
| 239 } |
| 240 |
| 241 virtual void TearDown() OVERRIDE { |
| 242 XSynchronize(gfx::GetXDisplay(), False); |
| 243 ViewsTestBase::TearDown(); |
| 244 } |
| 245 |
| 246 private: |
| 247 DISALLOW_COPY_AND_ASSIGN(DesktopWindowTreeHostX11Test); |
| 248 }; |
| 249 |
| 250 // Tests that the shape is properly set on the x window. |
| 251 TEST_F(DesktopWindowTreeHostX11Test, Shape) { |
| 252 if (!ui::IsShapeExtensionAvailable()) |
| 253 return; |
| 254 |
| 255 // 1) Test setting the window shape via the NonClientFrameView. This technique |
| 256 // is used to get rounded corners on Chrome windows when not using the native |
| 257 // window frame. |
| 258 scoped_ptr<Widget> widget1 = CreateWidget(new ShapedWidgetDelegate()); |
| 259 widget1->Show(); |
| 260 ui::X11EventSource::GetInstance()->DispatchXEvents(); |
| 261 |
| 262 XID xid1 = widget1->GetNativeWindow()->GetHost()->GetAcceleratedWidget(); |
| 263 std::vector<gfx::Rect> shape_rects = GetShapeRects(xid1); |
| 264 ASSERT_FALSE(shape_rects.empty()); |
| 265 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 85, 5)); |
| 266 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 95, 5)); |
| 267 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 95, 15)); |
| 268 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 105, 15)); |
| 269 |
| 270 // Changing widget's size should update the shape. |
| 271 widget1->SetBounds(gfx::Rect(100, 100, 200, 200)); |
| 272 ui::X11EventSource::GetInstance()->DispatchXEvents(); |
| 273 |
| 274 shape_rects = GetShapeRects(xid1); |
| 275 ASSERT_FALSE(shape_rects.empty()); |
| 276 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 85, 5)); |
| 277 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 95, 5)); |
| 278 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 185, 5)); |
| 279 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 195, 5)); |
| 280 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 195, 15)); |
| 281 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 205, 15)); |
| 282 |
| 283 // The shape should be changed to a rectangle which fills the entire screen |
| 284 // when |widget1| is maximized. |
| 285 { |
| 286 MaximizeWaiter waiter(xid1); |
| 287 widget1->Maximize(); |
| 288 waiter.Wait(); |
| 289 } |
| 290 |
| 291 // xvfb does not support Xrandr so we cannot check the maximized window's |
| 292 // bounds. |
| 293 gfx::Rect maximized_bounds; |
| 294 ui::GetWindowRect(xid1, &maximized_bounds); |
| 295 |
| 296 shape_rects = GetShapeRects(xid1); |
| 297 ASSERT_FALSE(shape_rects.empty()); |
| 298 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, |
| 299 maximized_bounds.width() - 1, |
| 300 5)); |
| 301 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, |
| 302 maximized_bounds.width() - 1, |
| 303 15)); |
| 304 |
| 305 // 2) Test setting the window shape via Widget::SetShape(). |
| 306 gfx::Path shape2; |
| 307 shape2.moveTo(10, 0); |
| 308 shape2.lineTo(10, 10); |
| 309 shape2.lineTo(0, 10); |
| 310 shape2.lineTo(0, 100); |
| 311 shape2.lineTo(100, 100); |
| 312 shape2.lineTo(100, 0); |
| 313 shape2.close(); |
| 314 |
| 315 scoped_ptr<Widget> widget2(CreateWidget(NULL)); |
| 316 widget2->Show(); |
| 317 widget2->SetShape(shape2.CreateNativeRegion()); |
| 318 ui::X11EventSource::GetInstance()->DispatchXEvents(); |
| 319 |
| 320 XID xid2 = widget2->GetNativeWindow()->GetHost()->GetAcceleratedWidget(); |
| 321 shape_rects = GetShapeRects(xid2); |
| 322 ASSERT_FALSE(shape_rects.empty()); |
| 323 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 5, 5)); |
| 324 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 15, 5)); |
| 325 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 95, 15)); |
| 326 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 105, 15)); |
| 327 |
| 328 // Changing the widget's size should not affect the shape. |
| 329 widget2->SetBounds(gfx::Rect(100, 100, 200, 200)); |
| 330 shape_rects = GetShapeRects(xid2); |
| 331 ASSERT_FALSE(shape_rects.empty()); |
| 332 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 5, 5)); |
| 333 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 15, 5)); |
| 334 EXPECT_TRUE(ShapeRectContainsPoint(shape_rects, 95, 15)); |
| 335 EXPECT_FALSE(ShapeRectContainsPoint(shape_rects, 105, 15)); |
| 336 } |
| 337 |
| 338 } // namespace views |
OLD | NEW |