621 lines
18 KiB
C++
621 lines
18 KiB
C++
|
/*
|
||
|
* Copyright 2012 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 "webrtc/examples/peerconnection/client/main_wnd.h"
|
||
|
|
||
|
#include <math.h>
|
||
|
|
||
|
#include "webrtc/examples/peerconnection/client/defaults.h"
|
||
|
#include "webrtc/base/arraysize.h"
|
||
|
#include "webrtc/base/common.h"
|
||
|
#include "webrtc/base/logging.h"
|
||
|
|
||
|
ATOM MainWnd::wnd_class_ = 0;
|
||
|
const wchar_t MainWnd::kClassName[] = L"WebRTC_MainWnd";
|
||
|
|
||
|
using rtc::sprintfn;
|
||
|
|
||
|
namespace {
|
||
|
|
||
|
const char kConnecting[] = "Connecting... ";
|
||
|
const char kNoVideoStreams[] = "(no video streams either way)";
|
||
|
const char kNoIncomingStream[] = "(no incoming video)";
|
||
|
|
||
|
void CalculateWindowSizeForText(HWND wnd, const wchar_t* text,
|
||
|
size_t* width, size_t* height) {
|
||
|
HDC dc = ::GetDC(wnd);
|
||
|
RECT text_rc = {0};
|
||
|
::DrawText(dc, text, -1, &text_rc, DT_CALCRECT | DT_SINGLELINE);
|
||
|
::ReleaseDC(wnd, dc);
|
||
|
RECT client, window;
|
||
|
::GetClientRect(wnd, &client);
|
||
|
::GetWindowRect(wnd, &window);
|
||
|
|
||
|
*width = text_rc.right - text_rc.left;
|
||
|
*width += (window.right - window.left) -
|
||
|
(client.right - client.left);
|
||
|
*height = text_rc.bottom - text_rc.top;
|
||
|
*height += (window.bottom - window.top) -
|
||
|
(client.bottom - client.top);
|
||
|
}
|
||
|
|
||
|
HFONT GetDefaultFont() {
|
||
|
static HFONT font = reinterpret_cast<HFONT>(GetStockObject(DEFAULT_GUI_FONT));
|
||
|
return font;
|
||
|
}
|
||
|
|
||
|
std::string GetWindowText(HWND wnd) {
|
||
|
char text[MAX_PATH] = {0};
|
||
|
::GetWindowTextA(wnd, &text[0], ARRAYSIZE(text));
|
||
|
return text;
|
||
|
}
|
||
|
|
||
|
void AddListBoxItem(HWND listbox, const std::string& str, LPARAM item_data) {
|
||
|
LRESULT index = ::SendMessageA(listbox, LB_ADDSTRING, 0,
|
||
|
reinterpret_cast<LPARAM>(str.c_str()));
|
||
|
::SendMessageA(listbox, LB_SETITEMDATA, index, item_data);
|
||
|
}
|
||
|
|
||
|
} // namespace
|
||
|
|
||
|
MainWnd::MainWnd(const char* server, int port, bool auto_connect,
|
||
|
bool auto_call)
|
||
|
: ui_(CONNECT_TO_SERVER), wnd_(NULL), edit1_(NULL), edit2_(NULL),
|
||
|
label1_(NULL), label2_(NULL), button_(NULL), listbox_(NULL),
|
||
|
destroyed_(false), callback_(NULL), nested_msg_(NULL),
|
||
|
server_(server), auto_connect_(auto_connect), auto_call_(auto_call) {
|
||
|
char buffer[10] = {0};
|
||
|
sprintfn(buffer, sizeof(buffer), "%i", port);
|
||
|
port_ = buffer;
|
||
|
}
|
||
|
|
||
|
MainWnd::~MainWnd() {
|
||
|
ASSERT(!IsWindow());
|
||
|
}
|
||
|
|
||
|
bool MainWnd::Create() {
|
||
|
ASSERT(wnd_ == NULL);
|
||
|
if (!RegisterWindowClass())
|
||
|
return false;
|
||
|
|
||
|
ui_thread_id_ = ::GetCurrentThreadId();
|
||
|
wnd_ = ::CreateWindowExW(WS_EX_OVERLAPPEDWINDOW, kClassName, L"WebRTC",
|
||
|
WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN,
|
||
|
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
|
||
|
NULL, NULL, GetModuleHandle(NULL), this);
|
||
|
|
||
|
::SendMessage(wnd_, WM_SETFONT, reinterpret_cast<WPARAM>(GetDefaultFont()),
|
||
|
TRUE);
|
||
|
|
||
|
CreateChildWindows();
|
||
|
SwitchToConnectUI();
|
||
|
|
||
|
return wnd_ != NULL;
|
||
|
}
|
||
|
|
||
|
bool MainWnd::Destroy() {
|
||
|
BOOL ret = FALSE;
|
||
|
if (IsWindow()) {
|
||
|
ret = ::DestroyWindow(wnd_);
|
||
|
}
|
||
|
|
||
|
return ret != FALSE;
|
||
|
}
|
||
|
|
||
|
void MainWnd::RegisterObserver(MainWndCallback* callback) {
|
||
|
callback_ = callback;
|
||
|
}
|
||
|
|
||
|
bool MainWnd::IsWindow() {
|
||
|
return wnd_ && ::IsWindow(wnd_) != FALSE;
|
||
|
}
|
||
|
|
||
|
bool MainWnd::PreTranslateMessage(MSG* msg) {
|
||
|
bool ret = false;
|
||
|
if (msg->message == WM_CHAR) {
|
||
|
if (msg->wParam == VK_TAB) {
|
||
|
HandleTabbing();
|
||
|
ret = true;
|
||
|
} else if (msg->wParam == VK_RETURN) {
|
||
|
OnDefaultAction();
|
||
|
ret = true;
|
||
|
} else if (msg->wParam == VK_ESCAPE) {
|
||
|
if (callback_) {
|
||
|
if (ui_ == STREAMING) {
|
||
|
callback_->DisconnectFromCurrentPeer();
|
||
|
} else {
|
||
|
callback_->DisconnectFromServer();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} else if (msg->hwnd == NULL && msg->message == UI_THREAD_CALLBACK) {
|
||
|
callback_->UIThreadCallback(static_cast<int>(msg->wParam),
|
||
|
reinterpret_cast<void*>(msg->lParam));
|
||
|
ret = true;
|
||
|
}
|
||
|
return ret;
|
||
|
}
|
||
|
|
||
|
void MainWnd::SwitchToConnectUI() {
|
||
|
ASSERT(IsWindow());
|
||
|
LayoutPeerListUI(false);
|
||
|
ui_ = CONNECT_TO_SERVER;
|
||
|
LayoutConnectUI(true);
|
||
|
::SetFocus(edit1_);
|
||
|
|
||
|
if (auto_connect_)
|
||
|
::PostMessage(button_, BM_CLICK, 0, 0);
|
||
|
}
|
||
|
|
||
|
void MainWnd::SwitchToPeerList(const Peers& peers) {
|
||
|
LayoutConnectUI(false);
|
||
|
|
||
|
::SendMessage(listbox_, LB_RESETCONTENT, 0, 0);
|
||
|
|
||
|
AddListBoxItem(listbox_, "List of currently connected peers:", -1);
|
||
|
Peers::const_iterator i = peers.begin();
|
||
|
for (; i != peers.end(); ++i)
|
||
|
AddListBoxItem(listbox_, i->second.c_str(), i->first);
|
||
|
|
||
|
ui_ = LIST_PEERS;
|
||
|
LayoutPeerListUI(true);
|
||
|
::SetFocus(listbox_);
|
||
|
|
||
|
if (auto_call_ && peers.begin() != peers.end()) {
|
||
|
// Get the number of items in the list
|
||
|
LRESULT count = ::SendMessage(listbox_, LB_GETCOUNT, 0, 0);
|
||
|
if (count != LB_ERR) {
|
||
|
// Select the last item in the list
|
||
|
LRESULT selection = ::SendMessage(listbox_, LB_SETCURSEL , count - 1, 0);
|
||
|
if (selection != LB_ERR)
|
||
|
::PostMessage(wnd_, WM_COMMAND, MAKEWPARAM(GetDlgCtrlID(listbox_),
|
||
|
LBN_DBLCLK),
|
||
|
reinterpret_cast<LPARAM>(listbox_));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MainWnd::SwitchToStreamingUI() {
|
||
|
LayoutConnectUI(false);
|
||
|
LayoutPeerListUI(false);
|
||
|
ui_ = STREAMING;
|
||
|
}
|
||
|
|
||
|
void MainWnd::MessageBox(const char* caption, const char* text, bool is_error) {
|
||
|
DWORD flags = MB_OK;
|
||
|
if (is_error)
|
||
|
flags |= MB_ICONERROR;
|
||
|
|
||
|
::MessageBoxA(handle(), text, caption, flags);
|
||
|
}
|
||
|
|
||
|
|
||
|
void MainWnd::StartLocalRenderer(webrtc::VideoTrackInterface* local_video) {
|
||
|
local_renderer_.reset(new VideoRenderer(handle(), 1, 1, local_video));
|
||
|
}
|
||
|
|
||
|
void MainWnd::StopLocalRenderer() {
|
||
|
local_renderer_.reset();
|
||
|
}
|
||
|
|
||
|
void MainWnd::StartRemoteRenderer(webrtc::VideoTrackInterface* remote_video) {
|
||
|
remote_renderer_.reset(new VideoRenderer(handle(), 1, 1, remote_video));
|
||
|
}
|
||
|
|
||
|
void MainWnd::StopRemoteRenderer() {
|
||
|
remote_renderer_.reset();
|
||
|
}
|
||
|
|
||
|
void MainWnd::QueueUIThreadCallback(int msg_id, void* data) {
|
||
|
::PostThreadMessage(ui_thread_id_, UI_THREAD_CALLBACK,
|
||
|
static_cast<WPARAM>(msg_id), reinterpret_cast<LPARAM>(data));
|
||
|
}
|
||
|
|
||
|
void MainWnd::OnPaint() {
|
||
|
PAINTSTRUCT ps;
|
||
|
::BeginPaint(handle(), &ps);
|
||
|
|
||
|
RECT rc;
|
||
|
::GetClientRect(handle(), &rc);
|
||
|
|
||
|
VideoRenderer* local_renderer = local_renderer_.get();
|
||
|
VideoRenderer* remote_renderer = remote_renderer_.get();
|
||
|
if (ui_ == STREAMING && remote_renderer && local_renderer) {
|
||
|
AutoLock<VideoRenderer> local_lock(local_renderer);
|
||
|
AutoLock<VideoRenderer> remote_lock(remote_renderer);
|
||
|
|
||
|
const BITMAPINFO& bmi = remote_renderer->bmi();
|
||
|
int height = abs(bmi.bmiHeader.biHeight);
|
||
|
int width = bmi.bmiHeader.biWidth;
|
||
|
|
||
|
const uint8_t* image = remote_renderer->image();
|
||
|
if (image != NULL) {
|
||
|
HDC dc_mem = ::CreateCompatibleDC(ps.hdc);
|
||
|
::SetStretchBltMode(dc_mem, HALFTONE);
|
||
|
|
||
|
// Set the map mode so that the ratio will be maintained for us.
|
||
|
HDC all_dc[] = { ps.hdc, dc_mem };
|
||
|
for (int i = 0; i < arraysize(all_dc); ++i) {
|
||
|
SetMapMode(all_dc[i], MM_ISOTROPIC);
|
||
|
SetWindowExtEx(all_dc[i], width, height, NULL);
|
||
|
SetViewportExtEx(all_dc[i], rc.right, rc.bottom, NULL);
|
||
|
}
|
||
|
|
||
|
HBITMAP bmp_mem = ::CreateCompatibleBitmap(ps.hdc, rc.right, rc.bottom);
|
||
|
HGDIOBJ bmp_old = ::SelectObject(dc_mem, bmp_mem);
|
||
|
|
||
|
POINT logical_area = { rc.right, rc.bottom };
|
||
|
DPtoLP(ps.hdc, &logical_area, 1);
|
||
|
|
||
|
HBRUSH brush = ::CreateSolidBrush(RGB(0, 0, 0));
|
||
|
RECT logical_rect = {0, 0, logical_area.x, logical_area.y };
|
||
|
::FillRect(dc_mem, &logical_rect, brush);
|
||
|
::DeleteObject(brush);
|
||
|
|
||
|
int x = (logical_area.x / 2) - (width / 2);
|
||
|
int y = (logical_area.y / 2) - (height / 2);
|
||
|
|
||
|
StretchDIBits(dc_mem, x, y, width, height,
|
||
|
0, 0, width, height, image, &bmi, DIB_RGB_COLORS, SRCCOPY);
|
||
|
|
||
|
if ((rc.right - rc.left) > 200 && (rc.bottom - rc.top) > 200) {
|
||
|
const BITMAPINFO& bmi = local_renderer->bmi();
|
||
|
image = local_renderer->image();
|
||
|
int thumb_width = bmi.bmiHeader.biWidth / 4;
|
||
|
int thumb_height = abs(bmi.bmiHeader.biHeight) / 4;
|
||
|
StretchDIBits(dc_mem,
|
||
|
logical_area.x - thumb_width - 10,
|
||
|
logical_area.y - thumb_height - 10,
|
||
|
thumb_width, thumb_height,
|
||
|
0, 0, bmi.bmiHeader.biWidth, -bmi.bmiHeader.biHeight,
|
||
|
image, &bmi, DIB_RGB_COLORS, SRCCOPY);
|
||
|
}
|
||
|
|
||
|
BitBlt(ps.hdc, 0, 0, logical_area.x, logical_area.y,
|
||
|
dc_mem, 0, 0, SRCCOPY);
|
||
|
|
||
|
// Cleanup.
|
||
|
::SelectObject(dc_mem, bmp_old);
|
||
|
::DeleteObject(bmp_mem);
|
||
|
::DeleteDC(dc_mem);
|
||
|
} else {
|
||
|
// We're still waiting for the video stream to be initialized.
|
||
|
HBRUSH brush = ::CreateSolidBrush(RGB(0, 0, 0));
|
||
|
::FillRect(ps.hdc, &rc, brush);
|
||
|
::DeleteObject(brush);
|
||
|
|
||
|
HGDIOBJ old_font = ::SelectObject(ps.hdc, GetDefaultFont());
|
||
|
::SetTextColor(ps.hdc, RGB(0xff, 0xff, 0xff));
|
||
|
::SetBkMode(ps.hdc, TRANSPARENT);
|
||
|
|
||
|
std::string text(kConnecting);
|
||
|
if (!local_renderer->image()) {
|
||
|
text += kNoVideoStreams;
|
||
|
} else {
|
||
|
text += kNoIncomingStream;
|
||
|
}
|
||
|
::DrawTextA(ps.hdc, text.c_str(), -1, &rc,
|
||
|
DT_SINGLELINE | DT_CENTER | DT_VCENTER);
|
||
|
::SelectObject(ps.hdc, old_font);
|
||
|
}
|
||
|
} else {
|
||
|
HBRUSH brush = ::CreateSolidBrush(::GetSysColor(COLOR_WINDOW));
|
||
|
::FillRect(ps.hdc, &rc, brush);
|
||
|
::DeleteObject(brush);
|
||
|
}
|
||
|
|
||
|
::EndPaint(handle(), &ps);
|
||
|
}
|
||
|
|
||
|
void MainWnd::OnDestroyed() {
|
||
|
PostQuitMessage(0);
|
||
|
}
|
||
|
|
||
|
void MainWnd::OnDefaultAction() {
|
||
|
if (!callback_)
|
||
|
return;
|
||
|
if (ui_ == CONNECT_TO_SERVER) {
|
||
|
std::string server(GetWindowText(edit1_));
|
||
|
std::string port_str(GetWindowText(edit2_));
|
||
|
int port = port_str.length() ? atoi(port_str.c_str()) : 0;
|
||
|
callback_->StartLogin(server, port);
|
||
|
} else if (ui_ == LIST_PEERS) {
|
||
|
LRESULT sel = ::SendMessage(listbox_, LB_GETCURSEL, 0, 0);
|
||
|
if (sel != LB_ERR) {
|
||
|
LRESULT peer_id = ::SendMessage(listbox_, LB_GETITEMDATA, sel, 0);
|
||
|
if (peer_id != -1 && callback_) {
|
||
|
callback_->ConnectToPeer(peer_id);
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
MessageBoxA(wnd_, "OK!", "Yeah", MB_OK);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool MainWnd::OnMessage(UINT msg, WPARAM wp, LPARAM lp, LRESULT* result) {
|
||
|
switch (msg) {
|
||
|
case WM_ERASEBKGND:
|
||
|
*result = TRUE;
|
||
|
return true;
|
||
|
|
||
|
case WM_PAINT:
|
||
|
OnPaint();
|
||
|
return true;
|
||
|
|
||
|
case WM_SETFOCUS:
|
||
|
if (ui_ == CONNECT_TO_SERVER) {
|
||
|
SetFocus(edit1_);
|
||
|
} else if (ui_ == LIST_PEERS) {
|
||
|
SetFocus(listbox_);
|
||
|
}
|
||
|
return true;
|
||
|
|
||
|
case WM_SIZE:
|
||
|
if (ui_ == CONNECT_TO_SERVER) {
|
||
|
LayoutConnectUI(true);
|
||
|
} else if (ui_ == LIST_PEERS) {
|
||
|
LayoutPeerListUI(true);
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case WM_CTLCOLORSTATIC:
|
||
|
*result = reinterpret_cast<LRESULT>(GetSysColorBrush(COLOR_WINDOW));
|
||
|
return true;
|
||
|
|
||
|
case WM_COMMAND:
|
||
|
if (button_ == reinterpret_cast<HWND>(lp)) {
|
||
|
if (BN_CLICKED == HIWORD(wp))
|
||
|
OnDefaultAction();
|
||
|
} else if (listbox_ == reinterpret_cast<HWND>(lp)) {
|
||
|
if (LBN_DBLCLK == HIWORD(wp)) {
|
||
|
OnDefaultAction();
|
||
|
}
|
||
|
}
|
||
|
return true;
|
||
|
|
||
|
case WM_CLOSE:
|
||
|
if (callback_)
|
||
|
callback_->Close();
|
||
|
break;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
LRESULT CALLBACK MainWnd::WndProc(HWND hwnd, UINT msg, WPARAM wp, LPARAM lp) {
|
||
|
MainWnd* me = reinterpret_cast<MainWnd*>(
|
||
|
::GetWindowLongPtr(hwnd, GWLP_USERDATA));
|
||
|
if (!me && WM_CREATE == msg) {
|
||
|
CREATESTRUCT* cs = reinterpret_cast<CREATESTRUCT*>(lp);
|
||
|
me = reinterpret_cast<MainWnd*>(cs->lpCreateParams);
|
||
|
me->wnd_ = hwnd;
|
||
|
::SetWindowLongPtr(hwnd, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(me));
|
||
|
}
|
||
|
|
||
|
LRESULT result = 0;
|
||
|
if (me) {
|
||
|
void* prev_nested_msg = me->nested_msg_;
|
||
|
me->nested_msg_ = &msg;
|
||
|
|
||
|
bool handled = me->OnMessage(msg, wp, lp, &result);
|
||
|
if (WM_NCDESTROY == msg) {
|
||
|
me->destroyed_ = true;
|
||
|
} else if (!handled) {
|
||
|
result = ::DefWindowProc(hwnd, msg, wp, lp);
|
||
|
}
|
||
|
|
||
|
if (me->destroyed_ && prev_nested_msg == NULL) {
|
||
|
me->OnDestroyed();
|
||
|
me->wnd_ = NULL;
|
||
|
me->destroyed_ = false;
|
||
|
}
|
||
|
|
||
|
me->nested_msg_ = prev_nested_msg;
|
||
|
} else {
|
||
|
result = ::DefWindowProc(hwnd, msg, wp, lp);
|
||
|
}
|
||
|
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
// static
|
||
|
bool MainWnd::RegisterWindowClass() {
|
||
|
if (wnd_class_)
|
||
|
return true;
|
||
|
|
||
|
WNDCLASSEX wcex = { sizeof(WNDCLASSEX) };
|
||
|
wcex.style = CS_DBLCLKS;
|
||
|
wcex.hInstance = GetModuleHandle(NULL);
|
||
|
wcex.hbrBackground = reinterpret_cast<HBRUSH>(COLOR_WINDOW + 1);
|
||
|
wcex.hCursor = ::LoadCursor(NULL, IDC_ARROW);
|
||
|
wcex.lpfnWndProc = &WndProc;
|
||
|
wcex.lpszClassName = kClassName;
|
||
|
wnd_class_ = ::RegisterClassEx(&wcex);
|
||
|
ASSERT(wnd_class_ != 0);
|
||
|
return wnd_class_ != 0;
|
||
|
}
|
||
|
|
||
|
void MainWnd::CreateChildWindow(HWND* wnd, MainWnd::ChildWindowID id,
|
||
|
const wchar_t* class_name, DWORD control_style,
|
||
|
DWORD ex_style) {
|
||
|
if (::IsWindow(*wnd))
|
||
|
return;
|
||
|
|
||
|
// Child windows are invisible at first, and shown after being resized.
|
||
|
DWORD style = WS_CHILD | control_style;
|
||
|
*wnd = ::CreateWindowEx(ex_style, class_name, L"", style,
|
||
|
100, 100, 100, 100, wnd_,
|
||
|
reinterpret_cast<HMENU>(id),
|
||
|
GetModuleHandle(NULL), NULL);
|
||
|
ASSERT(::IsWindow(*wnd) != FALSE);
|
||
|
::SendMessage(*wnd, WM_SETFONT, reinterpret_cast<WPARAM>(GetDefaultFont()),
|
||
|
TRUE);
|
||
|
}
|
||
|
|
||
|
void MainWnd::CreateChildWindows() {
|
||
|
// Create the child windows in tab order.
|
||
|
CreateChildWindow(&label1_, LABEL1_ID, L"Static", ES_CENTER | ES_READONLY, 0);
|
||
|
CreateChildWindow(&edit1_, EDIT_ID, L"Edit",
|
||
|
ES_LEFT | ES_NOHIDESEL | WS_TABSTOP, WS_EX_CLIENTEDGE);
|
||
|
CreateChildWindow(&label2_, LABEL2_ID, L"Static", ES_CENTER | ES_READONLY, 0);
|
||
|
CreateChildWindow(&edit2_, EDIT_ID, L"Edit",
|
||
|
ES_LEFT | ES_NOHIDESEL | WS_TABSTOP, WS_EX_CLIENTEDGE);
|
||
|
CreateChildWindow(&button_, BUTTON_ID, L"Button", BS_CENTER | WS_TABSTOP, 0);
|
||
|
|
||
|
CreateChildWindow(&listbox_, LISTBOX_ID, L"ListBox",
|
||
|
LBS_HASSTRINGS | LBS_NOTIFY, WS_EX_CLIENTEDGE);
|
||
|
|
||
|
::SetWindowTextA(edit1_, server_.c_str());
|
||
|
::SetWindowTextA(edit2_, port_.c_str());
|
||
|
}
|
||
|
|
||
|
void MainWnd::LayoutConnectUI(bool show) {
|
||
|
struct Windows {
|
||
|
HWND wnd;
|
||
|
const wchar_t* text;
|
||
|
size_t width;
|
||
|
size_t height;
|
||
|
} windows[] = {
|
||
|
{ label1_, L"Server" },
|
||
|
{ edit1_, L"XXXyyyYYYgggXXXyyyYYYggg" },
|
||
|
{ label2_, L":" },
|
||
|
{ edit2_, L"XyXyX" },
|
||
|
{ button_, L"Connect" },
|
||
|
};
|
||
|
|
||
|
if (show) {
|
||
|
const size_t kSeparator = 5;
|
||
|
size_t total_width = (ARRAYSIZE(windows) - 1) * kSeparator;
|
||
|
|
||
|
for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
|
||
|
CalculateWindowSizeForText(windows[i].wnd, windows[i].text,
|
||
|
&windows[i].width, &windows[i].height);
|
||
|
total_width += windows[i].width;
|
||
|
}
|
||
|
|
||
|
RECT rc;
|
||
|
::GetClientRect(wnd_, &rc);
|
||
|
size_t x = (rc.right / 2) - (total_width / 2);
|
||
|
size_t y = rc.bottom / 2;
|
||
|
for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
|
||
|
size_t top = y - (windows[i].height / 2);
|
||
|
::MoveWindow(windows[i].wnd, static_cast<int>(x), static_cast<int>(top),
|
||
|
static_cast<int>(windows[i].width),
|
||
|
static_cast<int>(windows[i].height),
|
||
|
TRUE);
|
||
|
x += kSeparator + windows[i].width;
|
||
|
if (windows[i].text[0] != 'X')
|
||
|
::SetWindowText(windows[i].wnd, windows[i].text);
|
||
|
::ShowWindow(windows[i].wnd, SW_SHOWNA);
|
||
|
}
|
||
|
} else {
|
||
|
for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
|
||
|
::ShowWindow(windows[i].wnd, SW_HIDE);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MainWnd::LayoutPeerListUI(bool show) {
|
||
|
if (show) {
|
||
|
RECT rc;
|
||
|
::GetClientRect(wnd_, &rc);
|
||
|
::MoveWindow(listbox_, 0, 0, rc.right, rc.bottom, TRUE);
|
||
|
::ShowWindow(listbox_, SW_SHOWNA);
|
||
|
} else {
|
||
|
::ShowWindow(listbox_, SW_HIDE);
|
||
|
InvalidateRect(wnd_, NULL, TRUE);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void MainWnd::HandleTabbing() {
|
||
|
bool shift = ((::GetAsyncKeyState(VK_SHIFT) & 0x8000) != 0);
|
||
|
UINT next_cmd = shift ? GW_HWNDPREV : GW_HWNDNEXT;
|
||
|
UINT loop_around_cmd = shift ? GW_HWNDLAST : GW_HWNDFIRST;
|
||
|
HWND focus = GetFocus(), next;
|
||
|
do {
|
||
|
next = ::GetWindow(focus, next_cmd);
|
||
|
if (IsWindowVisible(next) &&
|
||
|
(GetWindowLong(next, GWL_STYLE) & WS_TABSTOP)) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if (!next) {
|
||
|
next = ::GetWindow(focus, loop_around_cmd);
|
||
|
if (IsWindowVisible(next) &&
|
||
|
(GetWindowLong(next, GWL_STYLE) & WS_TABSTOP)) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
focus = next;
|
||
|
} while (true);
|
||
|
::SetFocus(next);
|
||
|
}
|
||
|
|
||
|
//
|
||
|
// MainWnd::VideoRenderer
|
||
|
//
|
||
|
|
||
|
MainWnd::VideoRenderer::VideoRenderer(
|
||
|
HWND wnd, int width, int height,
|
||
|
webrtc::VideoTrackInterface* track_to_render)
|
||
|
: wnd_(wnd), rendered_track_(track_to_render) {
|
||
|
::InitializeCriticalSection(&buffer_lock_);
|
||
|
ZeroMemory(&bmi_, sizeof(bmi_));
|
||
|
bmi_.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
|
||
|
bmi_.bmiHeader.biPlanes = 1;
|
||
|
bmi_.bmiHeader.biBitCount = 32;
|
||
|
bmi_.bmiHeader.biCompression = BI_RGB;
|
||
|
bmi_.bmiHeader.biWidth = width;
|
||
|
bmi_.bmiHeader.biHeight = -height;
|
||
|
bmi_.bmiHeader.biSizeImage = width * height *
|
||
|
(bmi_.bmiHeader.biBitCount >> 3);
|
||
|
rendered_track_->AddOrUpdateSink(this, rtc::VideoSinkWants());
|
||
|
}
|
||
|
|
||
|
MainWnd::VideoRenderer::~VideoRenderer() {
|
||
|
rendered_track_->RemoveSink(this);
|
||
|
::DeleteCriticalSection(&buffer_lock_);
|
||
|
}
|
||
|
|
||
|
void MainWnd::VideoRenderer::SetSize(int width, int height) {
|
||
|
AutoLock<VideoRenderer> lock(this);
|
||
|
|
||
|
if (width == bmi_.bmiHeader.biWidth && height == bmi_.bmiHeader.biHeight) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
bmi_.bmiHeader.biWidth = width;
|
||
|
bmi_.bmiHeader.biHeight = -height;
|
||
|
bmi_.bmiHeader.biSizeImage = width * height *
|
||
|
(bmi_.bmiHeader.biBitCount >> 3);
|
||
|
image_.reset(new uint8_t[bmi_.bmiHeader.biSizeImage]);
|
||
|
}
|
||
|
|
||
|
void MainWnd::VideoRenderer::OnFrame(
|
||
|
const cricket::VideoFrame& video_frame) {
|
||
|
|
||
|
{
|
||
|
AutoLock<VideoRenderer> lock(this);
|
||
|
|
||
|
const cricket::VideoFrame* frame =
|
||
|
video_frame.GetCopyWithRotationApplied();
|
||
|
|
||
|
SetSize(frame->width(), frame->height());
|
||
|
|
||
|
ASSERT(image_.get() != NULL);
|
||
|
frame->ConvertToRgbBuffer(cricket::FOURCC_ARGB,
|
||
|
image_.get(),
|
||
|
bmi_.bmiHeader.biSizeImage,
|
||
|
bmi_.bmiHeader.biWidth *
|
||
|
bmi_.bmiHeader.biBitCount / 8);
|
||
|
}
|
||
|
InvalidateRect(wnd_, NULL, TRUE);
|
||
|
}
|