| 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 |