2014-03-05 10:43:14 -07:00
|
|
|
#include "dc-capture.h"
|
|
|
|
|
|
|
|
#define WIN32_MEAN_AND_LEAN
|
|
|
|
#include <windows.h>
|
|
|
|
|
|
|
|
static inline void init_textures(struct dc_capture *capture)
|
|
|
|
{
|
2021-07-06 22:10:57 -07:00
|
|
|
if (capture->compatibility) {
|
2017-09-28 06:25:51 -07:00
|
|
|
capture->texture = gs_texture_create(capture->width,
|
|
|
|
capture->height, GS_BGRA,
|
|
|
|
1, NULL, GS_DYNAMIC);
|
2021-07-06 22:10:57 -07:00
|
|
|
} else {
|
2017-09-28 06:25:51 -07:00
|
|
|
capture->texture =
|
|
|
|
gs_texture_create_gdi(capture->width, capture->height);
|
|
|
|
|
2021-07-06 22:10:57 -07:00
|
|
|
if (capture->texture) {
|
|
|
|
capture->extra_texture = gs_texture_create(
|
|
|
|
capture->width, capture->height, GS_BGRA, 1,
|
|
|
|
NULL, 0);
|
|
|
|
if (!capture->extra_texture) {
|
|
|
|
blog(LOG_WARNING, "[dc_capture_init] Failed to "
|
|
|
|
"create textures");
|
|
|
|
gs_texture_destroy(capture->texture);
|
|
|
|
capture->texture = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-28 06:25:51 -07:00
|
|
|
if (!capture->texture) {
|
|
|
|
blog(LOG_WARNING, "[dc_capture_init] Failed to "
|
|
|
|
"create textures");
|
|
|
|
return;
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
capture->valid = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dc_capture_init(struct dc_capture *capture, int x, int y, uint32_t width,
|
|
|
|
uint32_t height, bool cursor, bool compatibility)
|
|
|
|
{
|
2014-03-16 18:26:46 -07:00
|
|
|
memset(capture, 0, sizeof(struct dc_capture));
|
|
|
|
|
2014-03-05 10:43:14 -07:00
|
|
|
capture->x = x;
|
|
|
|
capture->y = y;
|
|
|
|
capture->width = width;
|
|
|
|
capture->height = height;
|
|
|
|
capture->capture_cursor = cursor;
|
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_enter_graphics();
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
if (!gs_gdi_texture_available())
|
|
|
|
compatibility = true;
|
|
|
|
|
|
|
|
capture->compatibility = compatibility;
|
|
|
|
|
|
|
|
init_textures(capture);
|
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_leave_graphics();
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
if (!capture->valid)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (compatibility) {
|
|
|
|
BITMAPINFO bi = {0};
|
|
|
|
BITMAPINFOHEADER *bih = &bi.bmiHeader;
|
|
|
|
bih->biSize = sizeof(BITMAPINFOHEADER);
|
|
|
|
bih->biBitCount = 32;
|
|
|
|
bih->biWidth = width;
|
|
|
|
bih->biHeight = height;
|
|
|
|
bih->biPlanes = 1;
|
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
const HDC hdc = CreateCompatibleDC(NULL);
|
|
|
|
if (hdc) {
|
|
|
|
const HBITMAP bmp = CreateDIBSection(
|
|
|
|
capture->hdc, &bi, DIB_RGB_COLORS,
|
|
|
|
(void **)&capture->bits, NULL, 0);
|
|
|
|
if (bmp) {
|
|
|
|
capture->hdc = hdc;
|
|
|
|
capture->bmp = bmp;
|
|
|
|
capture->old_bmp = SelectObject(capture->hdc,
|
|
|
|
capture->bmp);
|
|
|
|
} else {
|
|
|
|
DeleteDC(capture->hdc);
|
|
|
|
}
|
|
|
|
}
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dc_capture_free(struct dc_capture *capture)
|
|
|
|
{
|
|
|
|
if (capture->hdc) {
|
|
|
|
SelectObject(capture->hdc, capture->old_bmp);
|
|
|
|
DeleteDC(capture->hdc);
|
|
|
|
DeleteObject(capture->bmp);
|
|
|
|
}
|
|
|
|
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_enter_graphics();
|
2021-07-06 22:10:57 -07:00
|
|
|
gs_texture_destroy(capture->extra_texture);
|
2017-09-28 06:25:51 -07:00
|
|
|
gs_texture_destroy(capture->texture);
|
2014-08-04 05:48:58 -07:00
|
|
|
obs_leave_graphics();
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
memset(capture, 0, sizeof(struct dc_capture));
|
|
|
|
}
|
|
|
|
|
2014-06-14 23:49:35 -07:00
|
|
|
static void draw_cursor(struct dc_capture *capture, HDC hdc, HWND window)
|
2014-03-05 10:43:14 -07:00
|
|
|
{
|
2014-06-14 23:49:35 -07:00
|
|
|
HICON icon;
|
|
|
|
ICONINFO ii;
|
2014-03-05 10:43:14 -07:00
|
|
|
CURSORINFO *ci = &capture->ci;
|
2014-06-14 23:49:35 -07:00
|
|
|
POINT win_pos = {capture->x, capture->y};
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
if (!(capture->ci.flags & CURSOR_SHOWING))
|
|
|
|
return;
|
|
|
|
|
|
|
|
icon = CopyIcon(capture->ci.hCursor);
|
|
|
|
if (!icon)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (GetIconInfo(icon, &ii)) {
|
|
|
|
POINT pos;
|
2014-06-14 23:49:35 -07:00
|
|
|
|
|
|
|
if (window)
|
|
|
|
ClientToScreen(window, &win_pos);
|
|
|
|
|
|
|
|
pos.x = ci->ptScreenPos.x - (int)ii.xHotspot - win_pos.x;
|
|
|
|
pos.y = ci->ptScreenPos.y - (int)ii.yHotspot - win_pos.y;
|
2014-03-05 10:43:14 -07:00
|
|
|
|
2018-05-06 17:46:42 +02:00
|
|
|
DrawIconEx(hdc, pos.x, pos.y, icon, 0, 0, 0, NULL, DI_NORMAL);
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
DeleteObject(ii.hbmColor);
|
|
|
|
DeleteObject(ii.hbmMask);
|
|
|
|
}
|
|
|
|
|
|
|
|
DestroyIcon(icon);
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline HDC dc_capture_get_dc(struct dc_capture *capture)
|
|
|
|
{
|
|
|
|
if (!capture->valid)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
if (capture->compatibility)
|
|
|
|
return capture->hdc;
|
|
|
|
else
|
2017-09-28 06:25:51 -07:00
|
|
|
return gs_texture_get_dc(capture->texture);
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void dc_capture_release_dc(struct dc_capture *capture)
|
|
|
|
{
|
|
|
|
if (capture->compatibility) {
|
2017-09-28 06:25:51 -07:00
|
|
|
gs_texture_set_image(capture->texture, capture->bits,
|
2014-03-05 10:43:14 -07:00
|
|
|
capture->width * 4, false);
|
|
|
|
} else {
|
2017-09-28 06:25:51 -07:00
|
|
|
gs_texture_release_dc(capture->texture);
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void dc_capture_capture(struct dc_capture *capture, HWND window)
|
|
|
|
{
|
|
|
|
HDC hdc_target;
|
|
|
|
HDC hdc;
|
|
|
|
|
|
|
|
if (capture->capture_cursor) {
|
|
|
|
memset(&capture->ci, 0, sizeof(CURSORINFO));
|
|
|
|
capture->ci.cbSize = sizeof(CURSORINFO);
|
|
|
|
capture->cursor_captured = GetCursorInfo(&capture->ci);
|
|
|
|
}
|
|
|
|
|
|
|
|
hdc = dc_capture_get_dc(capture);
|
|
|
|
if (!hdc) {
|
|
|
|
blog(LOG_WARNING, "[capture_screen] Failed to get "
|
|
|
|
"texture DC");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
hdc_target = GetDC(window);
|
|
|
|
|
|
|
|
BitBlt(hdc, 0, 0, capture->width, capture->height, hdc_target,
|
|
|
|
capture->x, capture->y, SRCCOPY);
|
|
|
|
|
|
|
|
ReleaseDC(NULL, hdc_target);
|
|
|
|
|
2017-05-10 23:47:44 +02:00
|
|
|
if (capture->cursor_captured && !capture->cursor_hidden)
|
2014-06-14 23:49:35 -07:00
|
|
|
draw_cursor(capture, hdc, window);
|
2014-03-05 10:43:14 -07:00
|
|
|
|
|
|
|
dc_capture_release_dc(capture);
|
|
|
|
|
2017-09-28 06:25:51 -07:00
|
|
|
capture->texture_written = true;
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
void dc_capture_render(struct dc_capture *capture, bool texcoords_centered)
|
2014-03-05 10:43:14 -07:00
|
|
|
{
|
2022-03-23 22:25:05 -07:00
|
|
|
if (capture->valid && capture->texture_written) {
|
|
|
|
gs_texture_t *texture = capture->texture;
|
|
|
|
const bool compatibility = capture->compatibility;
|
|
|
|
bool linear_sample = compatibility;
|
|
|
|
if (!linear_sample && !texcoords_centered) {
|
|
|
|
gs_texture_t *const extra_texture =
|
|
|
|
capture->extra_texture;
|
|
|
|
gs_copy_texture(extra_texture, texture);
|
|
|
|
texture = extra_texture;
|
|
|
|
linear_sample = true;
|
|
|
|
}
|
2021-01-20 08:20:51 -08:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
const char *tech_name = "Draw";
|
|
|
|
float multiplier = 1.f;
|
|
|
|
switch (gs_get_color_space()) {
|
|
|
|
case GS_CS_SRGB_16F:
|
|
|
|
case GS_CS_709_EXTENDED:
|
|
|
|
if (!linear_sample)
|
|
|
|
tech_name = "DrawSrgbDecompress";
|
|
|
|
break;
|
|
|
|
case GS_CS_709_SCRGB:
|
|
|
|
if (linear_sample)
|
|
|
|
tech_name = "DrawMultiply";
|
|
|
|
else
|
|
|
|
tech_name = "DrawSrgbDecompressMultiply";
|
|
|
|
multiplier = obs_get_video_sdr_white_level() / 80.f;
|
|
|
|
}
|
2021-01-20 08:20:51 -08:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
gs_effect_t *effect = obs_get_base_effect(OBS_EFFECT_OPAQUE);
|
|
|
|
gs_technique_t *tech =
|
|
|
|
gs_effect_get_technique(effect, tech_name);
|
|
|
|
gs_eparam_t *image =
|
|
|
|
gs_effect_get_param_by_name(effect, "image");
|
2014-03-05 10:43:14 -07:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
const bool previous = gs_framebuffer_srgb_enabled();
|
|
|
|
gs_enable_framebuffer_srgb(linear_sample);
|
|
|
|
gs_enable_blending(false);
|
2014-03-05 10:43:14 -07:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
if (linear_sample)
|
|
|
|
gs_effect_set_texture_srgb(image, texture);
|
|
|
|
else
|
|
|
|
gs_effect_set_texture(image, texture);
|
2021-01-20 08:20:51 -08:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
gs_eparam_t *multiplier_param =
|
|
|
|
gs_effect_get_param_by_name(effect, "multiplier");
|
|
|
|
gs_effect_set_float(multiplier_param, multiplier);
|
2014-03-05 10:43:14 -07:00
|
|
|
|
2022-03-23 22:25:05 -07:00
|
|
|
const uint32_t flip = compatibility ? GS_FLIP_V : 0;
|
|
|
|
const size_t passes = gs_technique_begin(tech);
|
|
|
|
for (size_t i = 0; i < passes; i++) {
|
|
|
|
if (gs_technique_begin_pass(tech, i)) {
|
|
|
|
gs_draw_sprite(texture, flip, 0, 0);
|
|
|
|
|
|
|
|
gs_technique_end_pass(tech);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
gs_technique_end(tech);
|
|
|
|
|
|
|
|
gs_enable_blending(true);
|
|
|
|
gs_enable_framebuffer_srgb(previous);
|
|
|
|
}
|
2014-03-05 10:43:14 -07:00
|
|
|
}
|