1 // Copyright (c) 2012 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.
5 #include "ui/aura/root_window_host_win.h"
11 #include "base/message_loop/message_loop.h"
12 #include "ui/aura/client/cursor_client.h"
13 #include "ui/aura/root_window.h"
14 #include "ui/base/cursor/cursor_loader_win.h"
15 #include "ui/events/event.h"
16 #include "ui/base/view_prop.h"
17 #include "ui/gfx/display.h"
18 #include "ui/gfx/insets.h"
19 #include "ui/gfx/screen.h"
27 bool use_popup_as_root_window_for_test
= false;
32 WindowTreeHost
* WindowTreeHost::Create(const gfx::Rect
& bounds
) {
33 return new WindowTreeHostWin(bounds
);
37 gfx::Size
WindowTreeHost::GetNativeScreenSize() {
38 return gfx::Size(GetSystemMetrics(SM_CXSCREEN
),
39 GetSystemMetrics(SM_CYSCREEN
));
42 WindowTreeHostWin::WindowTreeHostWin(const gfx::Rect
& bounds
)
45 saved_window_style_(0),
46 saved_window_ex_style_(0) {
47 if (use_popup_as_root_window_for_test
)
48 set_window_style(WS_POPUP
);
50 SetWindowText(hwnd(), L
"aura::RootWindow!");
51 CreateCompositor(GetAcceleratedWidget());
54 WindowTreeHostWin::~WindowTreeHostWin() {
56 DestroyWindow(hwnd());
59 RootWindow
* WindowTreeHostWin::GetRootWindow() {
60 return delegate_
->AsRootWindow();
63 gfx::AcceleratedWidget
WindowTreeHostWin::GetAcceleratedWidget() {
67 void WindowTreeHostWin::Show() {
68 ShowWindow(hwnd(), SW_SHOWNORMAL
);
71 void WindowTreeHostWin::Hide() {
75 void WindowTreeHostWin::ToggleFullScreen() {
76 gfx::Rect target_rect
;
79 saved_window_style_
= GetWindowLong(hwnd(), GWL_STYLE
);
80 saved_window_ex_style_
= GetWindowLong(hwnd(), GWL_EXSTYLE
);
81 GetWindowRect(hwnd(), &saved_window_rect_
);
82 SetWindowLong(hwnd(), GWL_STYLE
,
83 saved_window_style_
& ~(WS_CAPTION
| WS_THICKFRAME
));
84 SetWindowLong(hwnd(), GWL_EXSTYLE
,
85 saved_window_ex_style_
& ~(WS_EX_DLGMODALFRAME
|
86 WS_EX_WINDOWEDGE
| WS_EX_CLIENTEDGE
| WS_EX_STATICEDGE
));
89 mi
.cbSize
= sizeof(mi
);
90 GetMonitorInfo(MonitorFromWindow(hwnd(), MONITOR_DEFAULTTONEAREST
), &mi
);
91 target_rect
= gfx::Rect(mi
.rcMonitor
);
94 SetWindowLong(hwnd(), GWL_STYLE
, saved_window_style_
);
95 SetWindowLong(hwnd(), GWL_EXSTYLE
, saved_window_ex_style_
);
96 target_rect
= gfx::Rect(saved_window_rect_
);
103 target_rect
.height(),
104 SWP_NOZORDER
| SWP_NOACTIVATE
| SWP_FRAMECHANGED
);
107 gfx::Rect
WindowTreeHostWin::GetBounds() const {
109 GetClientRect(hwnd(), &r
);
113 void WindowTreeHostWin::SetBounds(const gfx::Rect
& bounds
) {
115 saved_window_rect_
.right
= saved_window_rect_
.left
+ bounds
.width();
116 saved_window_rect_
.bottom
= saved_window_rect_
.top
+ bounds
.height();
120 window_rect
.left
= bounds
.x();
121 window_rect
.top
= bounds
.y();
122 window_rect
.right
= bounds
.right() ;
123 window_rect
.bottom
= bounds
.bottom();
124 AdjustWindowRectEx(&window_rect
,
125 GetWindowLong(hwnd(), GWL_STYLE
),
127 GetWindowLong(hwnd(), GWL_EXSTYLE
));
133 window_rect
.right
- window_rect
.left
,
134 window_rect
.bottom
- window_rect
.top
,
135 SWP_NOMOVE
| SWP_NOOWNERZORDER
| SWP_NOREDRAW
| SWP_NOREPOSITION
);
137 // Explicity call NotifyHostResized when the scale has changed because
138 // the window size may not have changed.
139 float current_scale
= compositor()->device_scale_factor();
140 float new_scale
= gfx::Screen::GetScreenFor(
141 delegate_
->AsRootWindow()->window())->GetDisplayNearestWindow(
142 delegate_
->AsRootWindow()->window()).device_scale_factor();
143 if (current_scale
!= new_scale
)
144 NotifyHostResized(bounds
.size());
147 gfx::Insets
WindowTreeHostWin::GetInsets() const {
148 return gfx::Insets();
151 void WindowTreeHostWin::SetInsets(const gfx::Insets
& insets
) {
154 gfx::Point
WindowTreeHostWin::GetLocationOnNativeScreen() const {
156 GetClientRect(hwnd(), &r
);
157 return gfx::Point(r
.left
, r
.top
);
161 void WindowTreeHostWin::SetCursor(gfx::NativeCursor native_cursor
) {
162 // Custom web cursors are handled directly.
163 if (native_cursor
== ui::kCursorCustom
)
166 ui::CursorLoaderWin cursor_loader
;
167 cursor_loader
.SetPlatformCursor(&native_cursor
);
168 ::SetCursor(native_cursor
.platform());
171 void WindowTreeHostWin::SetCapture() {
174 ::SetCapture(hwnd());
178 void WindowTreeHostWin::ReleaseCapture() {
180 has_capture_
= false;
185 bool WindowTreeHostWin::QueryMouseLocation(gfx::Point
* location_return
) {
186 client::CursorClient
* cursor_client
=
187 client::GetCursorClient(GetRootWindow()->window());
188 if (cursor_client
&& !cursor_client
->IsMouseEventsEnabled()) {
189 *location_return
= gfx::Point(0, 0);
195 ScreenToClient(hwnd(), &pt
);
196 const gfx::Size size
= GetBounds().size();
198 gfx::Point(max(0, min(size
.width(), static_cast<int>(pt
.x
))),
199 max(0, min(size
.height(), static_cast<int>(pt
.y
))));
200 return (pt
.x
>= 0 && static_cast<int>(pt
.x
) < size
.width() &&
201 pt
.y
>= 0 && static_cast<int>(pt
.y
) < size
.height());
204 bool WindowTreeHostWin::ConfineCursorToRootWindow() {
206 GetWindowRect(hwnd(), &window_rect
);
207 return ClipCursor(&window_rect
) != 0;
210 void WindowTreeHostWin::UnConfineCursor() {
214 void WindowTreeHostWin::OnCursorVisibilityChanged(bool show
) {
218 void WindowTreeHostWin::MoveCursorTo(const gfx::Point
& location
) {
219 // Deliberately not implemented.
222 void WindowTreeHostWin::PostNativeEvent(const base::NativeEvent
& native_event
) {
224 hwnd(), native_event
.message
, native_event
.wParam
, native_event
.lParam
);
227 void WindowTreeHostWin::OnDeviceScaleFactorChanged(
228 float device_scale_factor
) {
232 void WindowTreeHostWin::PrepareForShutdown() {
236 void WindowTreeHostWin::OnClose() {
237 // TODO: this obviously shouldn't be here.
238 base::MessageLoopForUI::current()->Quit();
241 LRESULT
WindowTreeHostWin::OnKeyEvent(UINT message
,
244 MSG msg
= { hwnd(), message
, w_param
, l_param
};
245 ui::KeyEvent
keyev(msg
, message
== WM_CHAR
);
246 SetMsgHandled(delegate_
->OnHostKeyEvent(&keyev
));
250 LRESULT
WindowTreeHostWin::OnMouseRange(UINT message
,
253 MSG msg
= { hwnd(), message
, w_param
, l_param
, 0,
254 { GET_X_LPARAM(l_param
), GET_Y_LPARAM(l_param
) } };
255 ui::MouseEvent
event(msg
);
256 bool handled
= false;
257 if (!(event
.flags() & ui::EF_IS_NON_CLIENT
))
258 handled
= delegate_
->OnHostMouseEvent(&event
);
259 SetMsgHandled(handled
);
263 LRESULT
WindowTreeHostWin::OnCaptureChanged(UINT message
,
267 has_capture_
= false;
268 delegate_
->OnHostLostWindowCapture();
273 LRESULT
WindowTreeHostWin::OnNCActivate(UINT message
,
277 delegate_
->OnHostActivated();
278 return DefWindowProc(hwnd(), message
, w_param
, l_param
);
281 void WindowTreeHostWin::OnMove(const CPoint
& point
) {
283 delegate_
->OnHostMoved(gfx::Point(point
.x
, point
.y
));
286 void WindowTreeHostWin::OnPaint(HDC dc
) {
287 gfx::Rect damage_rect
;
288 RECT update_rect
= {0};
289 if (GetUpdateRect(hwnd(), &update_rect
, FALSE
))
290 damage_rect
= gfx::Rect(update_rect
);
291 compositor()->ScheduleRedrawRect(damage_rect
);
292 ValidateRect(hwnd(), NULL
);
295 void WindowTreeHostWin::OnSize(UINT param
, const CSize
& size
) {
296 // Minimizing resizes the window to 0x0 which causes our layout to go all
297 // screwy, so we just ignore it.
298 if (delegate_
&& param
!= SIZE_MINIMIZED
)
299 NotifyHostResized(gfx::Size(size
.cx
, size
.cy
));
305 void SetUsePopupAsRootWindowForTest(bool use
) {
306 use_popup_as_root_window_for_test
= use
;