
BUG=webrtc:496110 R=sergeyu@chromium.org Review URL: https://codereview.webrtc.org/1199073002 . Cr-Commit-Position: refs/heads/master@{#9595}
219 lines
7.6 KiB
C++
219 lines
7.6 KiB
C++
/*
|
|
* Copyright (c) 2014 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/modules/desktop_capture/cropping_window_capturer.h"
|
|
|
|
#include "webrtc/base/win32.h"
|
|
#include "webrtc/modules/desktop_capture/win/scoped_gdi_object.h"
|
|
#include "webrtc/modules/desktop_capture/win/screen_capture_utils.h"
|
|
#include "webrtc/modules/desktop_capture/win/window_capture_utils.h"
|
|
#include "webrtc/system_wrappers/interface/logging.h"
|
|
|
|
namespace webrtc {
|
|
|
|
namespace {
|
|
|
|
// Used to pass input/output data during the EnumWindow call for verifying if
|
|
// the selected window is on top.
|
|
struct TopWindowVerifierContext {
|
|
TopWindowVerifierContext(HWND selected_window, HWND excluded_window)
|
|
: selected_window(selected_window),
|
|
excluded_window(excluded_window),
|
|
is_top_window(false),
|
|
selected_window_process_id(0) {}
|
|
|
|
HWND selected_window;
|
|
HWND excluded_window;
|
|
bool is_top_window;
|
|
DWORD selected_window_process_id;
|
|
DesktopRect selected_window_rect;
|
|
};
|
|
|
|
// The function is called during EnumWindow for every window enumerated and is
|
|
// responsible for verifying if the selected window is on top.
|
|
BOOL CALLBACK TopWindowVerifier(HWND hwnd, LPARAM param) {
|
|
TopWindowVerifierContext* context =
|
|
reinterpret_cast<TopWindowVerifierContext*>(param);
|
|
|
|
if (hwnd == context->selected_window) {
|
|
context->is_top_window = true;
|
|
return FALSE;
|
|
}
|
|
|
|
// Ignore the excluded window.
|
|
if (hwnd == context->excluded_window) {
|
|
return TRUE;
|
|
}
|
|
|
|
// Ignore hidden or minimized window.
|
|
if (IsIconic(hwnd) || !IsWindowVisible(hwnd)) {
|
|
return TRUE;
|
|
}
|
|
|
|
// Ignore descendant/owned windows since we want to capture them.
|
|
// This check does not work for tooltips and context menus. Drop down menus
|
|
// and popup windows are fine.
|
|
if (GetAncestor(hwnd, GA_ROOTOWNER) == context->selected_window) {
|
|
return TRUE;
|
|
}
|
|
|
|
// If |hwnd| has no title and belongs to the same process, assume it's a
|
|
// tooltip or context menu from the selected window and ignore it.
|
|
const size_t kTitleLength = 32;
|
|
WCHAR window_title[kTitleLength];
|
|
GetWindowText(hwnd, window_title, kTitleLength);
|
|
if (wcsnlen_s(window_title, kTitleLength) == 0) {
|
|
DWORD enumerated_process;
|
|
GetWindowThreadProcessId(hwnd, &enumerated_process);
|
|
if (!context->selected_window_process_id) {
|
|
GetWindowThreadProcessId(context->selected_window,
|
|
&context->selected_window_process_id);
|
|
}
|
|
if (context->selected_window_process_id == enumerated_process) {
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
// Check if the enumerated window intersects with the selected window.
|
|
RECT enumerated_rect;
|
|
if (!GetWindowRect(hwnd, &enumerated_rect)) {
|
|
// Bail out if failed to get the window area.
|
|
context->is_top_window = false;
|
|
return FALSE;
|
|
}
|
|
|
|
DesktopRect intersect_rect = context->selected_window_rect;
|
|
DesktopRect enumerated_desktop_rect =
|
|
DesktopRect::MakeLTRB(enumerated_rect.left,
|
|
enumerated_rect.top,
|
|
enumerated_rect.right,
|
|
enumerated_rect.bottom);
|
|
intersect_rect.IntersectWith(enumerated_desktop_rect);
|
|
|
|
// If intersection is not empty, the selected window is not on top.
|
|
if (!intersect_rect.is_empty()) {
|
|
context->is_top_window = false;
|
|
return FALSE;
|
|
}
|
|
// Otherwise, keep enumerating.
|
|
return TRUE;
|
|
}
|
|
|
|
class CroppingWindowCapturerWin : public CroppingWindowCapturer {
|
|
public:
|
|
CroppingWindowCapturerWin(
|
|
const DesktopCaptureOptions& options)
|
|
: CroppingWindowCapturer(options) {}
|
|
|
|
private:
|
|
bool ShouldUseScreenCapturer() override;
|
|
DesktopRect GetWindowRectInVirtualScreen() override;
|
|
|
|
// The region from GetWindowRgn in the desktop coordinate if the region is
|
|
// rectangular, or the rect from GetWindowRect if the region is not set.
|
|
DesktopRect window_region_rect_;
|
|
|
|
AeroChecker aero_checker_;
|
|
};
|
|
|
|
bool CroppingWindowCapturerWin::ShouldUseScreenCapturer() {
|
|
if (!rtc::IsWindows8OrLater() && aero_checker_.IsAeroEnabled())
|
|
return false;
|
|
|
|
// Check if the window is a translucent layered window.
|
|
HWND selected = reinterpret_cast<HWND>(selected_window());
|
|
LONG window_ex_style = GetWindowLong(selected, GWL_EXSTYLE);
|
|
if (window_ex_style & WS_EX_LAYERED) {
|
|
COLORREF color_ref_key = 0;
|
|
BYTE alpha = 0;
|
|
DWORD flags = 0;
|
|
|
|
// GetLayeredWindowAttributes fails if the window was setup with
|
|
// UpdateLayeredWindow. We have no way to know the opacity of the window in
|
|
// that case. This happens for Stiky Note (crbug/412726).
|
|
if (!GetLayeredWindowAttributes(selected, &color_ref_key, &alpha, &flags))
|
|
return false;
|
|
|
|
// UpdateLayeredWindow is the only way to set per-pixel alpha and will cause
|
|
// the previous GetLayeredWindowAttributes to fail. So we only need to check
|
|
// the window wide color key or alpha.
|
|
if ((flags & LWA_COLORKEY) || ((flags & LWA_ALPHA) && (alpha < 255)))
|
|
return false;
|
|
}
|
|
|
|
TopWindowVerifierContext context(
|
|
selected, reinterpret_cast<HWND>(excluded_window()));
|
|
|
|
RECT selected_window_rect;
|
|
if (!GetWindowRect(selected, &selected_window_rect)) {
|
|
return false;
|
|
}
|
|
context.selected_window_rect = DesktopRect::MakeLTRB(
|
|
selected_window_rect.left,
|
|
selected_window_rect.top,
|
|
selected_window_rect.right,
|
|
selected_window_rect.bottom);
|
|
|
|
// Get the window region and check if it is rectangular.
|
|
win::ScopedGDIObject<HRGN, win::DeleteObjectTraits<HRGN> >
|
|
scoped_hrgn(CreateRectRgn(0, 0, 0, 0));
|
|
int region_type = GetWindowRgn(selected, scoped_hrgn.Get());
|
|
|
|
// Do not use the screen capturer if the region is empty or not rectangular.
|
|
if (region_type == COMPLEXREGION || region_type == NULLREGION)
|
|
return false;
|
|
|
|
if (region_type == SIMPLEREGION) {
|
|
RECT region_rect;
|
|
GetRgnBox(scoped_hrgn.Get(), ®ion_rect);
|
|
DesktopRect rgn_rect =
|
|
DesktopRect::MakeLTRB(region_rect.left,
|
|
region_rect.top,
|
|
region_rect.right,
|
|
region_rect.bottom);
|
|
rgn_rect.Translate(context.selected_window_rect.left(),
|
|
context.selected_window_rect.top());
|
|
context.selected_window_rect.IntersectWith(rgn_rect);
|
|
}
|
|
window_region_rect_ = context.selected_window_rect;
|
|
|
|
// Check if the window is occluded by any other window, excluding the child
|
|
// windows, context menus, and |excluded_window_|.
|
|
EnumWindows(&TopWindowVerifier, reinterpret_cast<LPARAM>(&context));
|
|
return context.is_top_window;
|
|
}
|
|
|
|
DesktopRect CroppingWindowCapturerWin::GetWindowRectInVirtualScreen() {
|
|
DesktopRect original_rect;
|
|
DesktopRect window_rect;
|
|
HWND hwnd = reinterpret_cast<HWND>(selected_window());
|
|
if (!GetCroppedWindowRect(hwnd, &window_rect, &original_rect)) {
|
|
LOG(LS_WARNING) << "Failed to get window info: " << GetLastError();
|
|
return window_rect;
|
|
}
|
|
window_rect.IntersectWith(window_region_rect_);
|
|
|
|
// Convert |window_rect| to be relative to the top-left of the virtual screen.
|
|
DesktopRect screen_rect(GetScreenRect(kFullDesktopScreenId, L""));
|
|
window_rect.IntersectWith(screen_rect);
|
|
window_rect.Translate(-screen_rect.left(), -screen_rect.top());
|
|
return window_rect;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
// static
|
|
WindowCapturer*
|
|
CroppingWindowCapturer::Create(const DesktopCaptureOptions& options) {
|
|
return new CroppingWindowCapturerWin(options);
|
|
}
|
|
|
|
} // namespace webrtc
|