This change removes ScreenCapturer and WindowCapturer from WebRTC. BUG=webrtc:6513 Review-Url: https://codereview.webrtc.org/2490063002 Cr-Commit-Position: refs/heads/master@{#15033}
272 lines
8.8 KiB
C++
272 lines
8.8 KiB
C++
/*
|
|
* Copyright (c) 2013 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 <assert.h>
|
|
|
|
#include <memory>
|
|
|
|
#include "webrtc/base/checks.h"
|
|
#include "webrtc/base/constructormagic.h"
|
|
#include "webrtc/base/win32.h"
|
|
#include "webrtc/modules/desktop_capture/desktop_capturer.h"
|
|
#include "webrtc/modules/desktop_capture/desktop_frame_win.h"
|
|
#include "webrtc/modules/desktop_capture/win/window_capture_utils.h"
|
|
#include "webrtc/system_wrappers/include/logging.h"
|
|
|
|
namespace webrtc {
|
|
|
|
namespace {
|
|
|
|
BOOL CALLBACK WindowsEnumerationHandler(HWND hwnd, LPARAM param) {
|
|
DesktopCapturer::SourceList* list =
|
|
reinterpret_cast<DesktopCapturer::SourceList*>(param);
|
|
|
|
// Skip windows that are invisible, minimized, have no title, or are owned,
|
|
// unless they have the app window style set.
|
|
int len = GetWindowTextLength(hwnd);
|
|
HWND owner = GetWindow(hwnd, GW_OWNER);
|
|
LONG exstyle = GetWindowLong(hwnd, GWL_EXSTYLE);
|
|
if (len == 0 || IsIconic(hwnd) || !IsWindowVisible(hwnd) ||
|
|
(owner && !(exstyle & WS_EX_APPWINDOW))) {
|
|
return TRUE;
|
|
}
|
|
|
|
// Skip the Program Manager window and the Start button.
|
|
const size_t kClassLength = 256;
|
|
WCHAR class_name[kClassLength];
|
|
const int class_name_length = GetClassName(hwnd, class_name, kClassLength);
|
|
RTC_DCHECK(class_name_length)
|
|
<< "Error retrieving the application's class name";
|
|
|
|
// Skip Program Manager window and the Start button. This is the same logic
|
|
// that's used in Win32WindowPicker in libjingle. Consider filtering other
|
|
// windows as well (e.g. toolbars).
|
|
if (wcscmp(class_name, L"Progman") == 0 || wcscmp(class_name, L"Button") == 0)
|
|
return TRUE;
|
|
|
|
// Windows 8 introduced a "Modern App" identified by their class name being
|
|
// either ApplicationFrameWindow or windows.UI.Core.coreWindow. The
|
|
// associated windows cannot be captured, so we skip them.
|
|
// http://crbug.com/526883.
|
|
if (rtc::IsWindows8OrLater() &&
|
|
(wcscmp(class_name, L"ApplicationFrameWindow") == 0 ||
|
|
wcscmp(class_name, L"Windows.UI.Core.CoreWindow") == 0)) {
|
|
return TRUE;
|
|
}
|
|
|
|
DesktopCapturer::Source window;
|
|
window.id = reinterpret_cast<WindowId>(hwnd);
|
|
|
|
const size_t kTitleLength = 500;
|
|
WCHAR window_title[kTitleLength];
|
|
// Truncate the title if it's longer than kTitleLength.
|
|
GetWindowText(hwnd, window_title, kTitleLength);
|
|
window.title = rtc::ToUtf8(window_title);
|
|
|
|
// Skip windows when we failed to convert the title or it is empty.
|
|
if (window.title.empty())
|
|
return TRUE;
|
|
|
|
list->push_back(window);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
class WindowCapturerWin : public DesktopCapturer {
|
|
public:
|
|
WindowCapturerWin();
|
|
~WindowCapturerWin() override;
|
|
|
|
// DesktopCapturer interface.
|
|
void Start(Callback* callback) override;
|
|
void CaptureFrame() override;
|
|
bool GetSourceList(SourceList* sources) override;
|
|
bool SelectSource(SourceId id) override;
|
|
bool FocusOnSelectedSource() override;
|
|
|
|
private:
|
|
Callback* callback_ = nullptr;
|
|
|
|
// HWND and HDC for the currently selected window or nullptr if window is not
|
|
// selected.
|
|
HWND window_ = nullptr;
|
|
|
|
DesktopSize previous_size_;
|
|
|
|
AeroChecker aero_checker_;
|
|
|
|
// This map is used to avoid flickering for the case when SelectWindow() calls
|
|
// are interleaved with Capture() calls.
|
|
std::map<HWND, DesktopSize> window_size_map_;
|
|
|
|
RTC_DISALLOW_COPY_AND_ASSIGN(WindowCapturerWin);
|
|
};
|
|
|
|
WindowCapturerWin::WindowCapturerWin() {}
|
|
WindowCapturerWin::~WindowCapturerWin() {}
|
|
|
|
bool WindowCapturerWin::GetSourceList(SourceList* sources) {
|
|
SourceList result;
|
|
LPARAM param = reinterpret_cast<LPARAM>(&result);
|
|
if (!EnumWindows(&WindowsEnumerationHandler, param))
|
|
return false;
|
|
sources->swap(result);
|
|
|
|
std::map<HWND, DesktopSize> new_map;
|
|
for (const auto& item : *sources) {
|
|
HWND hwnd = reinterpret_cast<HWND>(item.id);
|
|
new_map[hwnd] = window_size_map_[hwnd];
|
|
}
|
|
window_size_map_.swap(new_map);
|
|
|
|
return true;
|
|
}
|
|
|
|
bool WindowCapturerWin::SelectSource(SourceId id) {
|
|
HWND window = reinterpret_cast<HWND>(id);
|
|
if (!IsWindow(window) || !IsWindowVisible(window) || IsIconic(window))
|
|
return false;
|
|
window_ = window;
|
|
// When a window is not in the map, window_size_map_[window] will create an
|
|
// item with DesktopSize (0, 0).
|
|
previous_size_ = window_size_map_[window];
|
|
return true;
|
|
}
|
|
|
|
bool WindowCapturerWin::FocusOnSelectedSource() {
|
|
if (!window_)
|
|
return false;
|
|
|
|
if (!IsWindow(window_) || !IsWindowVisible(window_) || IsIconic(window_))
|
|
return false;
|
|
|
|
return BringWindowToTop(window_) != FALSE &&
|
|
SetForegroundWindow(window_) != FALSE;
|
|
}
|
|
|
|
void WindowCapturerWin::Start(Callback* callback) {
|
|
assert(!callback_);
|
|
assert(callback);
|
|
|
|
callback_ = callback;
|
|
}
|
|
|
|
void WindowCapturerWin::CaptureFrame() {
|
|
if (!window_) {
|
|
LOG(LS_ERROR) << "Window hasn't been selected: " << GetLastError();
|
|
callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
|
|
return;
|
|
}
|
|
|
|
// Stop capturing if the window has been closed.
|
|
if (!IsWindow(window_)) {
|
|
callback_->OnCaptureResult(Result::ERROR_PERMANENT, nullptr);
|
|
return;
|
|
}
|
|
|
|
// Return a 1x1 black frame if the window is minimized or invisible, to match
|
|
// behavior on mace. Window can be temporarily invisible during the
|
|
// transition of full screen mode on/off.
|
|
if (IsIconic(window_) || !IsWindowVisible(window_)) {
|
|
std::unique_ptr<DesktopFrame> frame(
|
|
new BasicDesktopFrame(DesktopSize(1, 1)));
|
|
memset(frame->data(), 0, frame->stride() * frame->size().height());
|
|
|
|
previous_size_ = frame->size();
|
|
window_size_map_[window_] = previous_size_;
|
|
callback_->OnCaptureResult(Result::SUCCESS, std::move(frame));
|
|
return;
|
|
}
|
|
|
|
DesktopRect original_rect;
|
|
DesktopRect cropped_rect;
|
|
if (!GetCroppedWindowRect(window_, &cropped_rect, &original_rect)) {
|
|
LOG(LS_WARNING) << "Failed to get window info: " << GetLastError();
|
|
callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
|
|
return;
|
|
}
|
|
|
|
HDC window_dc = GetWindowDC(window_);
|
|
if (!window_dc) {
|
|
LOG(LS_WARNING) << "Failed to get window DC: " << GetLastError();
|
|
callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
|
|
return;
|
|
}
|
|
|
|
std::unique_ptr<DesktopFrameWin> frame(
|
|
DesktopFrameWin::Create(cropped_rect.size(), nullptr, window_dc));
|
|
if (!frame.get()) {
|
|
ReleaseDC(window_, window_dc);
|
|
callback_->OnCaptureResult(Result::ERROR_TEMPORARY, nullptr);
|
|
return;
|
|
}
|
|
|
|
HDC mem_dc = CreateCompatibleDC(window_dc);
|
|
HGDIOBJ previous_object = SelectObject(mem_dc, frame->bitmap());
|
|
BOOL result = FALSE;
|
|
|
|
// When desktop composition (Aero) is enabled each window is rendered to a
|
|
// private buffer allowing BitBlt() to get the window content even if the
|
|
// window is occluded. PrintWindow() is slower but lets rendering the window
|
|
// contents to an off-screen device context when Aero is not available.
|
|
// PrintWindow() is not supported by some applications.
|
|
//
|
|
// If Aero is enabled, we prefer BitBlt() because it's faster and avoids
|
|
// window flickering. Otherwise, we prefer PrintWindow() because BitBlt() may
|
|
// render occluding windows on top of the desired window.
|
|
//
|
|
// When composition is enabled the DC returned by GetWindowDC() doesn't always
|
|
// have window frame rendered correctly. Windows renders it only once and then
|
|
// caches the result between captures. We hack it around by calling
|
|
// PrintWindow() whenever window size changes, including the first time of
|
|
// capturing - it somehow affects what we get from BitBlt() on the subsequent
|
|
// captures.
|
|
|
|
if (!aero_checker_.IsAeroEnabled() || !previous_size_.equals(frame->size())) {
|
|
result = PrintWindow(window_, mem_dc, 0);
|
|
}
|
|
|
|
// Aero is enabled or PrintWindow() failed, use BitBlt.
|
|
if (!result) {
|
|
result = BitBlt(mem_dc, 0, 0, frame->size().width(), frame->size().height(),
|
|
window_dc,
|
|
cropped_rect.left() - original_rect.left(),
|
|
cropped_rect.top() - original_rect.top(),
|
|
SRCCOPY);
|
|
}
|
|
|
|
SelectObject(mem_dc, previous_object);
|
|
DeleteDC(mem_dc);
|
|
ReleaseDC(window_, window_dc);
|
|
|
|
previous_size_ = frame->size();
|
|
window_size_map_[window_] = previous_size_;
|
|
|
|
frame->mutable_updated_region()->SetRect(
|
|
DesktopRect::MakeSize(frame->size()));
|
|
|
|
if (!result) {
|
|
LOG(LS_ERROR) << "Both PrintWindow() and BitBlt() failed.";
|
|
frame.reset();
|
|
}
|
|
|
|
callback_->OnCaptureResult(Result::SUCCESS, std::move(frame));
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// static
|
|
std::unique_ptr<DesktopCapturer> DesktopCapturer::CreateRawWindowCapturer(
|
|
const DesktopCaptureOptions& options) {
|
|
return std::unique_ptr<DesktopCapturer>(new WindowCapturerWin());
|
|
}
|
|
|
|
} // namespace webrtc
|