Index: talk/examples/peerconnection/client/main_wnd.cc |
diff --git a/talk/examples/peerconnection/client/main_wnd.cc b/talk/examples/peerconnection/client/main_wnd.cc |
deleted file mode 100644 |
index 122af679c64fef3b354488553f633b594b72711e..0000000000000000000000000000000000000000 |
--- a/talk/examples/peerconnection/client/main_wnd.cc |
+++ /dev/null |
@@ -1,639 +0,0 @@ |
-/* |
- * libjingle |
- * Copyright 2012 Google Inc. |
- * |
- * Redistribution and use in source and binary forms, with or without |
- * modification, are permitted provided that the following conditions are met: |
- * |
- * 1. Redistributions of source code must retain the above copyright notice, |
- * this list of conditions and the following disclaimer. |
- * 2. Redistributions in binary form must reproduce the above copyright notice, |
- * this list of conditions and the following disclaimer in the documentation |
- * and/or other materials provided with the distribution. |
- * 3. The name of the author may not be used to endorse or promote products |
- * derived from this software without specific prior written permission. |
- * |
- * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED |
- * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF |
- * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO |
- * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, |
- * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, |
- * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; |
- * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, |
- * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR |
- * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF |
- * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. |
- */ |
- |
-#include "talk/examples/peerconnection/client/main_wnd.h" |
- |
-#include <math.h> |
- |
-#include "talk/examples/peerconnection/client/defaults.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* 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 < ARRAY_SIZE(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_->AddRenderer(this); |
-} |
- |
-MainWnd::VideoRenderer::~VideoRenderer() { |
- rendered_track_->RemoveRenderer(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[bmi_.bmiHeader.biSizeImage]); |
-} |
- |
-void MainWnd::VideoRenderer::RenderFrame( |
- const cricket::VideoFrame* video_frame) { |
- if (!video_frame) |
- return; |
- |
- { |
- AutoLock<VideoRenderer> lock(this); |
- |
- const cricket::VideoFrame* frame = |
- video_frame->GetCopyWithRotationApplied(); |
- |
- SetSize(static_cast<int>(frame->GetWidth()), |
- static_cast<int>(frame->GetHeight())); |
- |
- 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); |
-} |