/* * Copyright 2004 The WebRTC Project Authors. All rights reserved. * * Use of this source code is governed by a BSD-style license * that can be found in the LICENSE file in the root of the source * tree. An additional intellectual property rights grant can be found * in the file PATENTS. All contributing project authors may * be found in the AUTHORS file in the root of the source tree. */ #include "rtc_base/win32_window.h" #include "rtc_base/checks.h" #include "rtc_base/logging.h" namespace rtc { /////////////////////////////////////////////////////////////////////////////// // Win32Window /////////////////////////////////////////////////////////////////////////////// static const wchar_t kWindowBaseClassName[] = L"RtcWindowBaseClass"; HINSTANCE Win32Window::instance_ = nullptr; ATOM Win32Window::window_class_ = 0; Win32Window::Win32Window() : wnd_(nullptr) {} Win32Window::~Win32Window() { RTC_DCHECK(nullptr == wnd_); } bool Win32Window::Create(HWND parent, const wchar_t* title, DWORD style, DWORD exstyle, int x, int y, int cx, int cy) { if (wnd_) { // Window already exists. return false; } if (!window_class_) { if (!GetModuleHandleExW(GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS | GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT, reinterpret_cast(&Win32Window::WndProc), &instance_)) { RTC_LOG_GLE(LS_ERROR) << "GetModuleHandleEx failed"; return false; } // Register or reregister the class as necessary. window_class_ == nullptr // is not an infallible indicator that the class is unregistered. WNDCLASSEXW wcex; memset(&wcex, 0, sizeof(wcex)); wcex.cbSize = sizeof(wcex); if (::GetClassInfoExW(instance_, kWindowBaseClassName, &wcex) && !::UnregisterClassW(kWindowBaseClassName, instance_)) { RTC_LOG_GLE(LS_ERROR) << "UnregisterClass failed."; } memset(&wcex, 0, sizeof(wcex)); wcex.cbSize = sizeof(wcex); wcex.hInstance = instance_; wcex.lpfnWndProc = &Win32Window::WndProc; wcex.lpszClassName = kWindowBaseClassName; window_class_ = ::RegisterClassExW(&wcex); if (!window_class_) { RTC_LOG_GLE(LS_ERROR) << "RegisterClassEx failed"; return false; } } wnd_ = ::CreateWindowExW(exstyle, kWindowBaseClassName, title, style, x, y, cx, cy, parent, nullptr, instance_, this); return (nullptr != wnd_); } void Win32Window::Destroy() { const bool success = ::DestroyWindow(wnd_); RTC_DCHECK(success); } void Win32Window::Shutdown() { if (window_class_) { if (!::UnregisterClass(MAKEINTATOM(window_class_), instance_)) { RTC_LOG_GLE(LS_ERROR) << "UnregisterClass failed."; } window_class_ = 0; } } bool Win32Window::OnMessage(UINT uMsg, WPARAM wParam, LPARAM lParam, LRESULT& result) { switch (uMsg) { case WM_CLOSE: if (!OnClose()) { result = 0; return true; } break; } return false; } bool Win32Window::OnClose() { return true; } void Win32Window::OnNcDestroy() { // Do nothing. } } LRESULT Win32Window::WndProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam) { Win32Window* that = reinterpret_cast(::GetWindowLongPtr(hwnd, GWLP_USERDATA)); if (!that && (WM_CREATE == uMsg)) { CREATESTRUCT* cs = reinterpret_cast(lParam); that = static_cast(cs->lpCreateParams); that->wnd_ = hwnd; ::SetWindowLongPtr(hwnd, GWLP_USERDATA, reinterpret_cast(that)); } if (that) { LRESULT result; bool handled = that->OnMessage(uMsg, wParam, lParam, result); if (WM_DESTROY == uMsg) { for (HWND child = ::GetWindow(hwnd, GW_CHILD); child; child = ::GetWindow(child, GW_HWNDNEXT)) { RTC_LOG(LS_INFO) << "Child window: " << static_cast(child); } } if (WM_NCDESTROY == uMsg) { ::SetWindowLongPtr(hwnd, GWLP_USERDATA, NULL); that->wnd_ = nullptr; that->OnNcDestroy(); } if (handled) { return result; } } return ::DefWindowProc(hwnd, uMsg, wParam, lParam); } } // namespace rtc