// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "ui/aura/root_window_host_win.h" #include #include #include "base/message_loop.h" #include "ui/aura/client/capture_client.h" #include "ui/aura/env.h" #include "ui/aura/event.h" #include "ui/aura/root_window.h" #include "ui/base/view_prop.h" using std::max; using std::min; namespace aura { namespace { const char* kRootWindowHostWinKey = "__AURA_ROOT_WINDOW_HOST_WIN__"; const wchar_t* GetCursorId(gfx::NativeCursor native_cursor) { switch (native_cursor.native_type()) { case ui::kCursorNull: return IDC_ARROW; case ui::kCursorPointer: return IDC_ARROW; case ui::kCursorCross: return IDC_CROSS; case ui::kCursorHand: return IDC_HAND; case ui::kCursorIBeam: return IDC_IBEAM; case ui::kCursorWait: return IDC_WAIT; case ui::kCursorHelp: return IDC_HELP; case ui::kCursorEastResize: return IDC_SIZEWE; case ui::kCursorNorthResize: return IDC_SIZENS; case ui::kCursorNorthEastResize: return IDC_SIZENESW; case ui::kCursorNorthWestResize: return IDC_SIZENWSE; case ui::kCursorSouthResize: return IDC_SIZENS; case ui::kCursorSouthEastResize: return IDC_SIZENWSE; case ui::kCursorSouthWestResize: return IDC_SIZENESW; case ui::kCursorWestResize: return IDC_SIZEWE; case ui::kCursorNorthSouthResize: return IDC_SIZENS; case ui::kCursorEastWestResize: return IDC_SIZEWE; case ui::kCursorNorthEastSouthWestResize: return IDC_SIZENESW; case ui::kCursorNorthWestSouthEastResize: return IDC_SIZENWSE; case ui::kCursorMove: return IDC_SIZEALL; case ui::kCursorProgress: return IDC_APPSTARTING; case ui::kCursorNoDrop: return IDC_NO; case ui::kCursorNotAllowed: return IDC_NO; case ui::kCursorColumnResize: case ui::kCursorRowResize: case ui::kCursorMiddlePanning: case ui::kCursorEastPanning: case ui::kCursorNorthPanning: case ui::kCursorNorthEastPanning: case ui::kCursorNorthWestPanning: case ui::kCursorSouthPanning: case ui::kCursorSouthEastPanning: case ui::kCursorSouthWestPanning: case ui::kCursorWestPanning: case ui::kCursorVerticalText: case ui::kCursorCell: case ui::kCursorContextMenu: case ui::kCursorAlias: case ui::kCursorCopy: case ui::kCursorNone: case ui::kCursorZoomIn: case ui::kCursorZoomOut: case ui::kCursorGrab: case ui::kCursorGrabbing: case ui::kCursorCustom: // TODO(jamescook): Should we use WebKit glue resources for these? // Or migrate those resources to someplace ui/aura can share? NOTIMPLEMENTED(); return IDC_ARROW; default: NOTREACHED(); return IDC_ARROW; } } } // namespace // static RootWindowHost* RootWindowHost::Create(const gfx::Rect& bounds) { return new RootWindowHostWin(bounds); } // static RootWindowHost* RootWindowHost::GetForAcceleratedWidget( gfx::AcceleratedWidget accelerated_widget) { return reinterpret_cast( ui::ViewProp::GetValue(accelerated_widget, kRootWindowHostWinKey)); } // static gfx::Size RootWindowHost::GetNativeScreenSize() { return gfx::Size(GetSystemMetrics(SM_CXSCREEN), GetSystemMetrics(SM_CYSCREEN)); } RootWindowHostWin::RootWindowHostWin(const gfx::Rect& bounds) : root_window_(NULL), fullscreen_(false), has_capture_(false), saved_window_style_(0), saved_window_ex_style_(0) { Init(NULL, bounds); SetWindowText(hwnd(), L"aura::RootWindow!"); prop_.reset(new ui::ViewProp(hwnd(), kRootWindowHostWinKey, this)); } RootWindowHostWin::~RootWindowHostWin() { DestroyWindow(hwnd()); } void RootWindowHostWin::SetRootWindow(RootWindow* root_window) { root_window_ = root_window; } RootWindow* RootWindowHostWin::GetRootWindow() { return root_window_; } gfx::AcceleratedWidget RootWindowHostWin::GetAcceleratedWidget() { return hwnd(); } void RootWindowHostWin::Show() { ShowWindow(hwnd(), SW_SHOWNORMAL); } void RootWindowHostWin::ToggleFullScreen() { gfx::Rect target_rect; if (!fullscreen_) { fullscreen_ = true; saved_window_style_ = GetWindowLong(hwnd(), GWL_STYLE); saved_window_ex_style_ = GetWindowLong(hwnd(), GWL_EXSTYLE); GetWindowRect(hwnd(), &saved_window_rect_); SetWindowLong(hwnd(), GWL_STYLE, saved_window_style_ & ~(WS_CAPTION | WS_THICKFRAME)); SetWindowLong(hwnd(), GWL_EXSTYLE, saved_window_ex_style_ & ~(WS_EX_DLGMODALFRAME | WS_EX_WINDOWEDGE | WS_EX_CLIENTEDGE | WS_EX_STATICEDGE)); MONITORINFO mi; mi.cbSize = sizeof(mi); GetMonitorInfo(MonitorFromWindow(hwnd(), MONITOR_DEFAULTTONEAREST), &mi); target_rect = mi.rcMonitor; } else { fullscreen_ = false; SetWindowLong(hwnd(), GWL_STYLE, saved_window_style_); SetWindowLong(hwnd(), GWL_EXSTYLE, saved_window_ex_style_); target_rect = saved_window_rect_; } SetWindowPos(hwnd(), NULL, target_rect.x(), target_rect.y(), target_rect.width(), target_rect.height(), SWP_NOZORDER | SWP_NOACTIVATE | SWP_FRAMECHANGED); } gfx::Rect RootWindowHostWin::GetBounds() const { RECT r; GetClientRect(hwnd(), &r); return gfx::Rect(r); } void RootWindowHostWin::SetBounds(const gfx::Rect& bounds) { if (fullscreen_) { saved_window_rect_.right = saved_window_rect_.left + bounds.width(); saved_window_rect_.bottom = saved_window_rect_.top + bounds.height(); return; } RECT window_rect; window_rect.left = bounds.x(); window_rect.top = bounds.y(); window_rect.right = bounds.right() ; window_rect.bottom = bounds.bottom(); AdjustWindowRectEx(&window_rect, GetWindowLong(hwnd(), GWL_STYLE), FALSE, GetWindowLong(hwnd(), GWL_EXSTYLE)); SetWindowPos( hwnd(), NULL, 0, 0, window_rect.right - window_rect.left, window_rect.bottom - window_rect.top, SWP_NOMOVE | SWP_NOOWNERZORDER | SWP_NOREDRAW | SWP_NOREPOSITION); } gfx::Point RootWindowHostWin::GetLocationOnNativeScreen() const { RECT r; GetClientRect(hwnd(), &r); return gfx::Point(r.left, r.top); } void RootWindowHostWin::SetCursor(gfx::NativeCursor native_cursor) { // Custom web cursors are handled directly. if (native_cursor == ui::kCursorCustom) return; const wchar_t* cursor_id = GetCursorId(native_cursor); // TODO(jamescook): Support for non-system cursors will require finding // the appropriate module to pass to LoadCursor(). ::SetCursor(LoadCursor(NULL, cursor_id)); } void RootWindowHostWin::SetCapture() { if (!has_capture_) { has_capture_ = true; ::SetCapture(hwnd()); } } void RootWindowHostWin::ReleaseCapture() { if (has_capture_) { has_capture_ = false; ::ReleaseCapture(); } } void RootWindowHostWin::ShowCursor(bool show) { // NOTIMPLEMENTED(); } bool RootWindowHostWin::QueryMouseLocation(gfx::Point* location_return) { POINT pt; GetCursorPos(&pt); ScreenToClient(hwnd(), &pt); const gfx::Size size = GetBounds().size(); *location_return = gfx::Point(max(0, min(size.width(), static_cast(pt.x))), max(0, min(size.height(), static_cast(pt.y)))); return (pt.x >= 0 && static_cast(pt.x) < size.width() && pt.y >= 0 && static_cast(pt.y) < size.height()); } bool RootWindowHostWin::ConfineCursorToRootWindow() { RECT window_rect; GetWindowRect(hwnd(), &window_rect); return ClipCursor(&window_rect) != 0; } bool RootWindowHostWin::GrabSnapshot( const gfx::Rect& snapshot_bounds, std::vector* png_representation) { NOTIMPLEMENTED(); return false; } void RootWindowHostWin::UnConfineCursor() { ClipCursor(NULL); } void RootWindowHostWin::MoveCursorTo(const gfx::Point& location) { POINT pt; ClientToScreen(hwnd(), &pt); SetCursorPos(pt.x, pt.y); } void RootWindowHostWin::SetFocusWhenShown(bool focus_when_shown) { NOTIMPLEMENTED(); } void RootWindowHostWin::PostNativeEvent(const base::NativeEvent& native_event) { ::PostMessage( hwnd(), native_event.message, native_event.wParam, native_event.lParam); } void RootWindowHostWin::OnDeviceScaleFactorChanged( float device_scale_factor) { NOTIMPLEMENTED(); } void RootWindowHostWin::OnClose() { // TODO: this obviously shouldn't be here. MessageLoopForUI::current()->Quit(); } LRESULT RootWindowHostWin::OnKeyEvent(UINT message, WPARAM w_param, LPARAM l_param) { MSG msg = { hwnd(), message, w_param, l_param }; KeyEvent keyev(msg, message == WM_CHAR); SetMsgHandled(root_window_->DispatchKeyEvent(&keyev)); return 0; } LRESULT RootWindowHostWin::OnMouseRange(UINT message, WPARAM w_param, LPARAM l_param) { MSG msg = { hwnd(), message, w_param, l_param, 0, { GET_X_LPARAM(l_param), GET_Y_LPARAM(l_param) } }; MouseEvent event(msg); bool handled = false; if (!(event.flags() & ui::EF_IS_NON_CLIENT)) handled = root_window_->DispatchMouseEvent(&event); SetMsgHandled(handled); return 0; } LRESULT RootWindowHostWin::OnCaptureChanged(UINT message, WPARAM w_param, LPARAM l_param) { if (has_capture_) { has_capture_ = false; Window* capture_window = client::GetCaptureWindow(root_window_); if (capture_window && capture_window->GetRootWindow() == root_window_) capture_window->ReleaseCapture(); } return 0; } void RootWindowHostWin::OnPaint(HDC dc) { root_window_->Draw(); ValidateRect(hwnd(), NULL); } void RootWindowHostWin::OnSize(UINT param, const CSize& size) { // Minimizing resizes the window to 0x0 which causes our layout to go all // screwy, so we just ignore it. if (param != SIZE_MINIMIZED) root_window_->OnHostResized(gfx::Size(size.cx, size.cy)); } } // namespace aura