blob: 275237f4a65c154ed67eb37a8dc5cfcf4fc4de94 [file] [log] [blame]
/*
* 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<LPCWSTR>(&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<Win32Window*>(::GetWindowLongPtr(hwnd, GWLP_USERDATA));
if (!that && (WM_CREATE == uMsg)) {
CREATESTRUCT* cs = reinterpret_cast<CREATESTRUCT*>(lParam);
that = static_cast<Win32Window*>(cs->lpCreateParams);
that->wnd_ = hwnd;
::SetWindowLongPtr(hwnd, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(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<void*>(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