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