• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libjingle
3  * Copyright 2012, Google Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  *
8  *  1. Redistributions of source code must retain the above copyright notice,
9  *     this list of conditions and the following disclaimer.
10  *  2. Redistributions in binary form must reproduce the above copyright notice,
11  *     this list of conditions and the following disclaimer in the documentation
12  *     and/or other materials provided with the distribution.
13  *  3. The name of the author may not be used to endorse or promote products
14  *     derived from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 #include "talk/examples/peerconnection/client/main_wnd.h"
29 
30 #include <math.h>
31 
32 #include "talk/base/common.h"
33 #include "talk/base/logging.h"
34 #include "talk/examples/peerconnection/client/defaults.h"
35 
36 ATOM MainWnd::wnd_class_ = 0;
37 const wchar_t MainWnd::kClassName[] = L"WebRTC_MainWnd";
38 
39 namespace {
40 
41 const char kConnecting[] = "Connecting... ";
42 const char kNoVideoStreams[] = "(no video streams either way)";
43 const char kNoIncomingStream[] = "(no incoming video)";
44 
CalculateWindowSizeForText(HWND wnd,const wchar_t * text,size_t * width,size_t * height)45 void CalculateWindowSizeForText(HWND wnd, const wchar_t* text,
46                                 size_t* width, size_t* height) {
47   HDC dc = ::GetDC(wnd);
48   RECT text_rc = {0};
49   ::DrawText(dc, text, -1, &text_rc, DT_CALCRECT | DT_SINGLELINE);
50   ::ReleaseDC(wnd, dc);
51   RECT client, window;
52   ::GetClientRect(wnd, &client);
53   ::GetWindowRect(wnd, &window);
54 
55   *width = text_rc.right - text_rc.left;
56   *width += (window.right - window.left) -
57             (client.right - client.left);
58   *height = text_rc.bottom - text_rc.top;
59   *height += (window.bottom - window.top) -
60              (client.bottom - client.top);
61 }
62 
GetDefaultFont()63 HFONT GetDefaultFont() {
64   static HFONT font = reinterpret_cast<HFONT>(GetStockObject(DEFAULT_GUI_FONT));
65   return font;
66 }
67 
GetWindowText(HWND wnd)68 std::string GetWindowText(HWND wnd) {
69   char text[MAX_PATH] = {0};
70   ::GetWindowTextA(wnd, &text[0], ARRAYSIZE(text));
71   return text;
72 }
73 
AddListBoxItem(HWND listbox,const std::string & str,LPARAM item_data)74 void AddListBoxItem(HWND listbox, const std::string& str, LPARAM item_data) {
75   LRESULT index = ::SendMessageA(listbox, LB_ADDSTRING, 0,
76       reinterpret_cast<LPARAM>(str.c_str()));
77   ::SendMessageA(listbox, LB_SETITEMDATA, index, item_data);
78 }
79 
80 }  // namespace
81 
MainWnd()82 MainWnd::MainWnd()
83   : ui_(CONNECT_TO_SERVER), wnd_(NULL), edit1_(NULL), edit2_(NULL),
84     label1_(NULL), label2_(NULL), button_(NULL), listbox_(NULL),
85     destroyed_(false), callback_(NULL), nested_msg_(NULL) {
86 }
87 
~MainWnd()88 MainWnd::~MainWnd() {
89   ASSERT(!IsWindow());
90 }
91 
Create()92 bool MainWnd::Create() {
93   ASSERT(wnd_ == NULL);
94   if (!RegisterWindowClass())
95     return false;
96 
97   ui_thread_id_ = ::GetCurrentThreadId();
98   wnd_ = ::CreateWindowExW(WS_EX_OVERLAPPEDWINDOW, kClassName, L"WebRTC",
99       WS_OVERLAPPEDWINDOW | WS_VISIBLE | WS_CLIPCHILDREN,
100       CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
101       NULL, NULL, GetModuleHandle(NULL), this);
102 
103   ::SendMessage(wnd_, WM_SETFONT, reinterpret_cast<WPARAM>(GetDefaultFont()),
104                 TRUE);
105 
106   CreateChildWindows();
107   SwitchToConnectUI();
108 
109   return wnd_ != NULL;
110 }
111 
Destroy()112 bool MainWnd::Destroy() {
113   BOOL ret = FALSE;
114   if (IsWindow()) {
115     ret = ::DestroyWindow(wnd_);
116   }
117 
118   return ret != FALSE;
119 }
120 
RegisterObserver(MainWndCallback * callback)121 void MainWnd::RegisterObserver(MainWndCallback* callback) {
122   callback_ = callback;
123 }
124 
IsWindow()125 bool MainWnd::IsWindow() {
126   return wnd_ && ::IsWindow(wnd_) != FALSE;
127 }
128 
PreTranslateMessage(MSG * msg)129 bool MainWnd::PreTranslateMessage(MSG* msg) {
130   bool ret = false;
131   if (msg->message == WM_CHAR) {
132     if (msg->wParam == VK_TAB) {
133       HandleTabbing();
134       ret = true;
135     } else if (msg->wParam == VK_RETURN) {
136       OnDefaultAction();
137       ret = true;
138     } else if (msg->wParam == VK_ESCAPE) {
139       if (callback_) {
140         if (ui_ == STREAMING) {
141           callback_->DisconnectFromCurrentPeer();
142         } else {
143           callback_->DisconnectFromServer();
144         }
145       }
146     }
147   } else if (msg->hwnd == NULL && msg->message == UI_THREAD_CALLBACK) {
148     callback_->UIThreadCallback(static_cast<int>(msg->wParam),
149                                 reinterpret_cast<void*>(msg->lParam));
150     ret = true;
151   }
152   return ret;
153 }
154 
SwitchToConnectUI()155 void MainWnd::SwitchToConnectUI() {
156   ASSERT(IsWindow());
157   LayoutPeerListUI(false);
158   ui_ = CONNECT_TO_SERVER;
159   LayoutConnectUI(true);
160   ::SetFocus(edit1_);
161 }
162 
SwitchToPeerList(const Peers & peers)163 void MainWnd::SwitchToPeerList(const Peers& peers) {
164   LayoutConnectUI(false);
165 
166   ::SendMessage(listbox_, LB_RESETCONTENT, 0, 0);
167 
168   AddListBoxItem(listbox_, "List of currently connected peers:", -1);
169   Peers::const_iterator i = peers.begin();
170   for (; i != peers.end(); ++i)
171     AddListBoxItem(listbox_, i->second.c_str(), i->first);
172 
173   ui_ = LIST_PEERS;
174   LayoutPeerListUI(true);
175   ::SetFocus(listbox_);
176 }
177 
SwitchToStreamingUI()178 void MainWnd::SwitchToStreamingUI() {
179   LayoutConnectUI(false);
180   LayoutPeerListUI(false);
181   ui_ = STREAMING;
182 }
183 
MessageBox(const char * caption,const char * text,bool is_error)184 void MainWnd::MessageBox(const char* caption, const char* text, bool is_error) {
185   DWORD flags = MB_OK;
186   if (is_error)
187     flags |= MB_ICONERROR;
188 
189   ::MessageBoxA(handle(), text, caption, flags);
190 }
191 
192 
StartLocalRenderer(webrtc::VideoTrackInterface * local_video)193 void MainWnd::StartLocalRenderer(webrtc::VideoTrackInterface* local_video) {
194   local_renderer_.reset(new VideoRenderer(handle(), 1, 1, local_video));
195 }
196 
StopLocalRenderer()197 void MainWnd::StopLocalRenderer() {
198   local_renderer_.reset();
199 }
200 
StartRemoteRenderer(webrtc::VideoTrackInterface * remote_video)201 void MainWnd::StartRemoteRenderer(webrtc::VideoTrackInterface* remote_video) {
202   remote_renderer_.reset(new VideoRenderer(handle(), 1, 1, remote_video));
203 }
204 
StopRemoteRenderer()205 void MainWnd::StopRemoteRenderer() {
206   remote_renderer_.reset();
207 }
208 
QueueUIThreadCallback(int msg_id,void * data)209 void MainWnd::QueueUIThreadCallback(int msg_id, void* data) {
210   ::PostThreadMessage(ui_thread_id_, UI_THREAD_CALLBACK,
211       static_cast<WPARAM>(msg_id), reinterpret_cast<LPARAM>(data));
212 }
213 
OnPaint()214 void MainWnd::OnPaint() {
215   PAINTSTRUCT ps;
216   ::BeginPaint(handle(), &ps);
217 
218   RECT rc;
219   ::GetClientRect(handle(), &rc);
220 
221   VideoRenderer* local_renderer = local_renderer_.get();
222   VideoRenderer* remote_renderer = remote_renderer_.get();
223   if (ui_ == STREAMING && remote_renderer && local_renderer) {
224     AutoLock<VideoRenderer> local_lock(local_renderer);
225     AutoLock<VideoRenderer> remote_lock(remote_renderer);
226 
227     const BITMAPINFO& bmi = remote_renderer->bmi();
228     int height = abs(bmi.bmiHeader.biHeight);
229     int width = bmi.bmiHeader.biWidth;
230 
231     const uint8* image = remote_renderer->image();
232     if (image != NULL) {
233       HDC dc_mem = ::CreateCompatibleDC(ps.hdc);
234       ::SetStretchBltMode(dc_mem, HALFTONE);
235 
236       // Set the map mode so that the ratio will be maintained for us.
237       HDC all_dc[] = { ps.hdc, dc_mem };
238       for (int i = 0; i < ARRAY_SIZE(all_dc); ++i) {
239         SetMapMode(all_dc[i], MM_ISOTROPIC);
240         SetWindowExtEx(all_dc[i], width, height, NULL);
241         SetViewportExtEx(all_dc[i], rc.right, rc.bottom, NULL);
242       }
243 
244       HBITMAP bmp_mem = ::CreateCompatibleBitmap(ps.hdc, rc.right, rc.bottom);
245       HGDIOBJ bmp_old = ::SelectObject(dc_mem, bmp_mem);
246 
247       POINT logical_area = { rc.right, rc.bottom };
248       DPtoLP(ps.hdc, &logical_area, 1);
249 
250       HBRUSH brush = ::CreateSolidBrush(RGB(0, 0, 0));
251       RECT logical_rect = {0, 0, logical_area.x, logical_area.y };
252       ::FillRect(dc_mem, &logical_rect, brush);
253       ::DeleteObject(brush);
254 
255       int x = (logical_area.x / 2) - (width / 2);
256       int y = (logical_area.y / 2) - (height / 2);
257 
258       StretchDIBits(dc_mem, x, y, width, height,
259                     0, 0, width, height, image, &bmi, DIB_RGB_COLORS, SRCCOPY);
260 
261       if ((rc.right - rc.left) > 200 && (rc.bottom - rc.top) > 200) {
262         const BITMAPINFO& bmi = local_renderer->bmi();
263         image = local_renderer->image();
264         int thumb_width = bmi.bmiHeader.biWidth / 4;
265         int thumb_height = abs(bmi.bmiHeader.biHeight) / 4;
266         StretchDIBits(dc_mem,
267             logical_area.x - thumb_width - 10,
268             logical_area.y - thumb_height - 10,
269             thumb_width, thumb_height,
270             0, 0, bmi.bmiHeader.biWidth, -bmi.bmiHeader.biHeight,
271             image, &bmi, DIB_RGB_COLORS, SRCCOPY);
272       }
273 
274       BitBlt(ps.hdc, 0, 0, logical_area.x, logical_area.y,
275              dc_mem, 0, 0, SRCCOPY);
276 
277       // Cleanup.
278       ::SelectObject(dc_mem, bmp_old);
279       ::DeleteObject(bmp_mem);
280       ::DeleteDC(dc_mem);
281     } else {
282       // We're still waiting for the video stream to be initialized.
283       HBRUSH brush = ::CreateSolidBrush(RGB(0, 0, 0));
284       ::FillRect(ps.hdc, &rc, brush);
285       ::DeleteObject(brush);
286 
287       HGDIOBJ old_font = ::SelectObject(ps.hdc, GetDefaultFont());
288       ::SetTextColor(ps.hdc, RGB(0xff, 0xff, 0xff));
289       ::SetBkMode(ps.hdc, TRANSPARENT);
290 
291       std::string text(kConnecting);
292       if (!local_renderer->image()) {
293         text += kNoVideoStreams;
294       } else {
295         text += kNoIncomingStream;
296       }
297       ::DrawTextA(ps.hdc, text.c_str(), -1, &rc,
298           DT_SINGLELINE | DT_CENTER | DT_VCENTER);
299       ::SelectObject(ps.hdc, old_font);
300     }
301   } else {
302     HBRUSH brush = ::CreateSolidBrush(::GetSysColor(COLOR_WINDOW));
303     ::FillRect(ps.hdc, &rc, brush);
304     ::DeleteObject(brush);
305   }
306 
307   ::EndPaint(handle(), &ps);
308 }
309 
OnDestroyed()310 void MainWnd::OnDestroyed() {
311   PostQuitMessage(0);
312 }
313 
OnDefaultAction()314 void MainWnd::OnDefaultAction() {
315   if (!callback_)
316     return;
317   if (ui_ == CONNECT_TO_SERVER) {
318     std::string server(GetWindowText(edit1_));
319     std::string port_str(GetWindowText(edit2_));
320     int port = port_str.length() ? atoi(port_str.c_str()) : 0;
321     callback_->StartLogin(server, port);
322   } else if (ui_ == LIST_PEERS) {
323     LRESULT sel = ::SendMessage(listbox_, LB_GETCURSEL, 0, 0);
324     if (sel != LB_ERR) {
325       LRESULT peer_id = ::SendMessage(listbox_, LB_GETITEMDATA, sel, 0);
326       if (peer_id != -1 && callback_) {
327         callback_->ConnectToPeer(peer_id);
328       }
329     }
330   } else {
331     MessageBoxA(wnd_, "OK!", "Yeah", MB_OK);
332   }
333 }
334 
OnMessage(UINT msg,WPARAM wp,LPARAM lp,LRESULT * result)335 bool MainWnd::OnMessage(UINT msg, WPARAM wp, LPARAM lp, LRESULT* result) {
336   switch (msg) {
337     case WM_ERASEBKGND:
338       *result = TRUE;
339       return true;
340 
341     case WM_PAINT:
342       OnPaint();
343       return true;
344 
345     case WM_SETFOCUS:
346       if (ui_ == CONNECT_TO_SERVER) {
347         SetFocus(edit1_);
348       } else if (ui_ == LIST_PEERS) {
349         SetFocus(listbox_);
350       }
351       return true;
352 
353     case WM_SIZE:
354       if (ui_ == CONNECT_TO_SERVER) {
355         LayoutConnectUI(true);
356       } else if (ui_ == LIST_PEERS) {
357         LayoutPeerListUI(true);
358       }
359       break;
360 
361     case WM_CTLCOLORSTATIC:
362       *result = reinterpret_cast<LRESULT>(GetSysColorBrush(COLOR_WINDOW));
363       return true;
364 
365     case WM_COMMAND:
366       if (button_ == reinterpret_cast<HWND>(lp)) {
367         if (BN_CLICKED == HIWORD(wp))
368           OnDefaultAction();
369       } else if (listbox_ == reinterpret_cast<HWND>(lp)) {
370         if (LBN_DBLCLK == HIWORD(wp)) {
371           OnDefaultAction();
372         }
373       }
374       return true;
375 
376     case WM_CLOSE:
377       if (callback_)
378         callback_->Close();
379       break;
380   }
381   return false;
382 }
383 
384 // static
WndProc(HWND hwnd,UINT msg,WPARAM wp,LPARAM lp)385 LRESULT CALLBACK MainWnd::WndProc(HWND hwnd, UINT msg, WPARAM wp, LPARAM lp) {
386   MainWnd* me = reinterpret_cast<MainWnd*>(
387       ::GetWindowLongPtr(hwnd, GWLP_USERDATA));
388   if (!me && WM_CREATE == msg) {
389     CREATESTRUCT* cs = reinterpret_cast<CREATESTRUCT*>(lp);
390     me = reinterpret_cast<MainWnd*>(cs->lpCreateParams);
391     me->wnd_ = hwnd;
392     ::SetWindowLongPtr(hwnd, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(me));
393   }
394 
395   LRESULT result = 0;
396   if (me) {
397     void* prev_nested_msg = me->nested_msg_;
398     me->nested_msg_ = &msg;
399 
400     bool handled = me->OnMessage(msg, wp, lp, &result);
401     if (WM_NCDESTROY == msg) {
402       me->destroyed_ = true;
403     } else if (!handled) {
404       result = ::DefWindowProc(hwnd, msg, wp, lp);
405     }
406 
407     if (me->destroyed_ && prev_nested_msg == NULL) {
408       me->OnDestroyed();
409       me->wnd_ = NULL;
410       me->destroyed_ = false;
411     }
412 
413     me->nested_msg_ = prev_nested_msg;
414   } else {
415     result = ::DefWindowProc(hwnd, msg, wp, lp);
416   }
417 
418   return result;
419 }
420 
421 // static
RegisterWindowClass()422 bool MainWnd::RegisterWindowClass() {
423   if (wnd_class_)
424     return true;
425 
426   WNDCLASSEX wcex = { sizeof(WNDCLASSEX) };
427   wcex.style = CS_DBLCLKS;
428   wcex.hInstance = GetModuleHandle(NULL);
429   wcex.hbrBackground = reinterpret_cast<HBRUSH>(COLOR_WINDOW + 1);
430   wcex.hCursor = ::LoadCursor(NULL, IDC_ARROW);
431   wcex.lpfnWndProc = &WndProc;
432   wcex.lpszClassName = kClassName;
433   wnd_class_ = ::RegisterClassEx(&wcex);
434   ASSERT(wnd_class_ != 0);
435   return wnd_class_ != 0;
436 }
437 
CreateChildWindow(HWND * wnd,MainWnd::ChildWindowID id,const wchar_t * class_name,DWORD control_style,DWORD ex_style)438 void MainWnd::CreateChildWindow(HWND* wnd, MainWnd::ChildWindowID id,
439                                 const wchar_t* class_name, DWORD control_style,
440                                 DWORD ex_style) {
441   if (::IsWindow(*wnd))
442     return;
443 
444   // Child windows are invisible at first, and shown after being resized.
445   DWORD style = WS_CHILD | control_style;
446   *wnd = ::CreateWindowEx(ex_style, class_name, L"", style,
447                           100, 100, 100, 100, wnd_,
448                           reinterpret_cast<HMENU>(id),
449                           GetModuleHandle(NULL), NULL);
450   ASSERT(::IsWindow(*wnd) != FALSE);
451   ::SendMessage(*wnd, WM_SETFONT, reinterpret_cast<WPARAM>(GetDefaultFont()),
452                 TRUE);
453 }
454 
CreateChildWindows()455 void MainWnd::CreateChildWindows() {
456   // Create the child windows in tab order.
457   CreateChildWindow(&label1_, LABEL1_ID, L"Static", ES_CENTER | ES_READONLY, 0);
458   CreateChildWindow(&edit1_, EDIT_ID, L"Edit",
459                     ES_LEFT | ES_NOHIDESEL | WS_TABSTOP, WS_EX_CLIENTEDGE);
460   CreateChildWindow(&label2_, LABEL2_ID, L"Static", ES_CENTER | ES_READONLY, 0);
461   CreateChildWindow(&edit2_, EDIT_ID, L"Edit",
462                     ES_LEFT | ES_NOHIDESEL | WS_TABSTOP, WS_EX_CLIENTEDGE);
463   CreateChildWindow(&button_, BUTTON_ID, L"Button", BS_CENTER | WS_TABSTOP, 0);
464 
465   CreateChildWindow(&listbox_, LISTBOX_ID, L"ListBox",
466                     LBS_HASSTRINGS | LBS_NOTIFY, WS_EX_CLIENTEDGE);
467 
468   ::SetWindowTextA(edit1_, GetDefaultServerName().c_str());
469   ::SetWindowTextA(edit2_, "8888");
470 }
471 
LayoutConnectUI(bool show)472 void MainWnd::LayoutConnectUI(bool show) {
473   struct Windows {
474     HWND wnd;
475     const wchar_t* text;
476     size_t width;
477     size_t height;
478   } windows[] = {
479     { label1_, L"Server" },
480     { edit1_, L"XXXyyyYYYgggXXXyyyYYYggg" },
481     { label2_, L":" },
482     { edit2_, L"XyXyX" },
483     { button_, L"Connect" },
484   };
485 
486   if (show) {
487     const size_t kSeparator = 5;
488     size_t total_width = (ARRAYSIZE(windows) - 1) * kSeparator;
489 
490     for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
491       CalculateWindowSizeForText(windows[i].wnd, windows[i].text,
492                                  &windows[i].width, &windows[i].height);
493       total_width += windows[i].width;
494     }
495 
496     RECT rc;
497     ::GetClientRect(wnd_, &rc);
498     size_t x = (rc.right / 2) - (total_width / 2);
499     size_t y = rc.bottom / 2;
500     for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
501       size_t top = y - (windows[i].height / 2);
502       ::MoveWindow(windows[i].wnd, static_cast<int>(x), static_cast<int>(top),
503                    static_cast<int>(windows[i].width),
504                    static_cast<int>(windows[i].height),
505                    TRUE);
506       x += kSeparator + windows[i].width;
507       if (windows[i].text[0] != 'X')
508         ::SetWindowText(windows[i].wnd, windows[i].text);
509       ::ShowWindow(windows[i].wnd, SW_SHOWNA);
510     }
511   } else {
512     for (size_t i = 0; i < ARRAYSIZE(windows); ++i) {
513       ::ShowWindow(windows[i].wnd, SW_HIDE);
514     }
515   }
516 }
517 
LayoutPeerListUI(bool show)518 void MainWnd::LayoutPeerListUI(bool show) {
519   if (show) {
520     RECT rc;
521     ::GetClientRect(wnd_, &rc);
522     ::MoveWindow(listbox_, 0, 0, rc.right, rc.bottom, TRUE);
523     ::ShowWindow(listbox_, SW_SHOWNA);
524   } else {
525     ::ShowWindow(listbox_, SW_HIDE);
526     InvalidateRect(wnd_, NULL, TRUE);
527   }
528 }
529 
HandleTabbing()530 void MainWnd::HandleTabbing() {
531   bool shift = ((::GetAsyncKeyState(VK_SHIFT) & 0x8000) != 0);
532   UINT next_cmd = shift ? GW_HWNDPREV : GW_HWNDNEXT;
533   UINT loop_around_cmd = shift ? GW_HWNDLAST : GW_HWNDFIRST;
534   HWND focus = GetFocus(), next;
535   do {
536     next = ::GetWindow(focus, next_cmd);
537     if (IsWindowVisible(next) &&
538         (GetWindowLong(next, GWL_STYLE) & WS_TABSTOP)) {
539       break;
540     }
541 
542     if (!next) {
543       next = ::GetWindow(focus, loop_around_cmd);
544       if (IsWindowVisible(next) &&
545           (GetWindowLong(next, GWL_STYLE) & WS_TABSTOP)) {
546         break;
547       }
548     }
549     focus = next;
550   } while (true);
551   ::SetFocus(next);
552 }
553 
554 //
555 // MainWnd::VideoRenderer
556 //
557 
VideoRenderer(HWND wnd,int width,int height,webrtc::VideoTrackInterface * track_to_render)558 MainWnd::VideoRenderer::VideoRenderer(
559     HWND wnd, int width, int height,
560     webrtc::VideoTrackInterface* track_to_render)
561     : wnd_(wnd), rendered_track_(track_to_render) {
562   ::InitializeCriticalSection(&buffer_lock_);
563   ZeroMemory(&bmi_, sizeof(bmi_));
564   bmi_.bmiHeader.biSize = sizeof(BITMAPINFOHEADER);
565   bmi_.bmiHeader.biPlanes = 1;
566   bmi_.bmiHeader.biBitCount = 32;
567   bmi_.bmiHeader.biCompression = BI_RGB;
568   bmi_.bmiHeader.biWidth = width;
569   bmi_.bmiHeader.biHeight = -height;
570   bmi_.bmiHeader.biSizeImage = width * height *
571                               (bmi_.bmiHeader.biBitCount >> 3);
572   rendered_track_->AddRenderer(this);
573 }
574 
~VideoRenderer()575 MainWnd::VideoRenderer::~VideoRenderer() {
576   rendered_track_->RemoveRenderer(this);
577   ::DeleteCriticalSection(&buffer_lock_);
578 }
579 
SetSize(int width,int height)580 void MainWnd::VideoRenderer::SetSize(int width, int height) {
581   AutoLock<VideoRenderer> lock(this);
582 
583   bmi_.bmiHeader.biWidth = width;
584   bmi_.bmiHeader.biHeight = -height;
585   bmi_.bmiHeader.biSizeImage = width * height *
586                                (bmi_.bmiHeader.biBitCount >> 3);
587   image_.reset(new uint8[bmi_.bmiHeader.biSizeImage]);
588 }
589 
RenderFrame(const cricket::VideoFrame * frame)590 void MainWnd::VideoRenderer::RenderFrame(const cricket::VideoFrame* frame) {
591   if (!frame)
592     return;
593 
594   {
595     AutoLock<VideoRenderer> lock(this);
596 
597     ASSERT(image_.get() != NULL);
598     frame->ConvertToRgbBuffer(cricket::FOURCC_ARGB,
599                               image_.get(),
600                               bmi_.bmiHeader.biSizeImage,
601                               bmi_.bmiHeader.biWidth *
602                               bmi_.bmiHeader.biBitCount / 8);
603   }
604   InvalidateRect(wnd_, NULL, TRUE);
605 }
606