2014-08-20 00:39:41 +02:00
|
|
|
#include <obs-module.h>
|
|
|
|
#include <util/darray.h>
|
|
|
|
#include <util/threading.h>
|
|
|
|
#include <util/platform.h>
|
|
|
|
|
|
|
|
#import <CoreGraphics/CGWindow.h>
|
|
|
|
#import <Cocoa/Cocoa.h>
|
|
|
|
|
|
|
|
#include "window-utils.h"
|
|
|
|
|
|
|
|
struct window_capture {
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_source_t *source;
|
2014-08-20 00:39:41 +02:00
|
|
|
|
|
|
|
struct cocoa_window window;
|
|
|
|
|
|
|
|
//CGRect bounds;
|
|
|
|
//CGWindowListOption window_option;
|
|
|
|
CGWindowImageOption image_option;
|
|
|
|
|
|
|
|
CGColorSpaceRef color_space;
|
|
|
|
|
|
|
|
DARRAY(uint8_t) buffer;
|
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
pthread_t capture_thread;
|
2014-09-25 17:44:05 -07:00
|
|
|
os_event_t *capture_event;
|
|
|
|
os_event_t *stop_event;
|
2014-08-20 00:39:41 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static CGImageRef get_image(struct window_capture *wc)
|
|
|
|
{
|
2019-07-09 13:29:39 -05:00
|
|
|
NSArray *arr = (NSArray *)CGWindowListCreate(
|
|
|
|
kCGWindowListOptionIncludingWindow, wc->window.window_id);
|
2014-08-20 00:39:41 +02:00
|
|
|
[arr autorelease];
|
|
|
|
|
2020-01-30 06:46:33 -08:00
|
|
|
if (!arr.count && !find_window(&wc->window, NULL, false))
|
2014-08-20 00:39:41 +02:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
return CGWindowListCreateImage(CGRectNull,
|
2019-07-09 13:29:39 -05:00
|
|
|
kCGWindowListOptionIncludingWindow,
|
|
|
|
wc->window.window_id, wc->image_option);
|
2014-08-20 00:39:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static inline void capture_frame(struct window_capture *wc)
|
|
|
|
{
|
2019-07-09 13:29:39 -05:00
|
|
|
uint64_t ts = os_gettime_ns();
|
2014-08-20 00:39:41 +02:00
|
|
|
CGImageRef img = get_image(wc);
|
|
|
|
if (!img)
|
|
|
|
return;
|
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
size_t width = CGImageGetWidth(img);
|
2014-08-20 00:39:41 +02:00
|
|
|
size_t height = CGImageGetHeight(img);
|
|
|
|
|
|
|
|
CGRect rect = {{0, 0}, {width, height}};
|
2020-05-09 09:12:48 -07:00
|
|
|
da_resize(wc->buffer, width * height * 4);
|
2014-08-20 00:39:41 +02:00
|
|
|
uint8_t *data = wc->buffer.array;
|
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
CGContextRef cg_context = CGBitmapContextCreate(
|
|
|
|
data, width, height, 8, width * 4, wc->color_space,
|
|
|
|
kCGBitmapByteOrder32Host | kCGImageAlphaPremultipliedFirst);
|
2014-08-20 00:39:41 +02:00
|
|
|
CGContextSetBlendMode(cg_context, kCGBlendModeCopy);
|
|
|
|
CGContextDrawImage(cg_context, rect, img);
|
|
|
|
CGContextRelease(cg_context);
|
|
|
|
CGImageRelease(img);
|
|
|
|
|
|
|
|
struct obs_source_frame frame = {
|
2019-07-09 13:29:39 -05:00
|
|
|
.format = VIDEO_FORMAT_BGRA,
|
|
|
|
.width = width,
|
|
|
|
.height = height,
|
|
|
|
.data[0] = data,
|
2014-08-20 00:39:41 +02:00
|
|
|
.linesize[0] = width * 4,
|
2019-07-09 13:29:39 -05:00
|
|
|
.timestamp = ts,
|
2014-08-20 00:39:41 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
obs_source_output_video(wc->source, &frame);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void *capture_thread(void *data)
|
|
|
|
{
|
|
|
|
struct window_capture *wc = data;
|
|
|
|
|
|
|
|
for (;;) {
|
|
|
|
os_event_wait(wc->capture_event);
|
|
|
|
if (os_event_try(wc->stop_event) != EAGAIN)
|
|
|
|
break;
|
|
|
|
|
|
|
|
@autoreleasepool {
|
|
|
|
capture_frame(wc);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static inline void *window_capture_create_internal(obs_data_t *settings,
|
2019-07-09 13:29:39 -05:00
|
|
|
obs_source_t *source)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
|
|
|
struct window_capture *wc = bzalloc(sizeof(struct window_capture));
|
|
|
|
|
|
|
|
wc->source = source;
|
|
|
|
|
|
|
|
wc->color_space = CGColorSpaceCreateDeviceRGB();
|
|
|
|
|
|
|
|
da_init(wc->buffer);
|
|
|
|
|
|
|
|
init_window(&wc->window, settings);
|
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
wc->image_option = obs_data_get_bool(settings, "show_shadow")
|
|
|
|
? kCGWindowImageDefault
|
|
|
|
: kCGWindowImageBoundsIgnoreFraming;
|
2014-08-20 00:39:41 +02:00
|
|
|
|
|
|
|
os_event_init(&wc->capture_event, OS_EVENT_TYPE_AUTO);
|
|
|
|
os_event_init(&wc->stop_event, OS_EVENT_TYPE_MANUAL);
|
|
|
|
|
|
|
|
pthread_create(&wc->capture_thread, NULL, capture_thread, wc);
|
|
|
|
|
|
|
|
return wc;
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void *window_capture_create(obs_data_t *settings, obs_source_t *source)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
|
|
|
@autoreleasepool {
|
|
|
|
return window_capture_create_internal(settings, source);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void window_capture_destroy(void *data)
|
|
|
|
{
|
|
|
|
struct window_capture *cap = data;
|
|
|
|
|
|
|
|
os_event_signal(cap->stop_event);
|
|
|
|
os_event_signal(cap->capture_event);
|
2019-07-09 13:29:39 -05:00
|
|
|
|
2014-08-20 00:39:41 +02:00
|
|
|
pthread_join(cap->capture_thread, NULL);
|
|
|
|
|
|
|
|
CGColorSpaceRelease(cap->color_space);
|
|
|
|
|
|
|
|
da_free(cap->buffer);
|
|
|
|
|
|
|
|
os_event_destroy(cap->capture_event);
|
|
|
|
os_event_destroy(cap->stop_event);
|
|
|
|
|
|
|
|
destroy_window(&cap->window);
|
|
|
|
|
|
|
|
bfree(cap);
|
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void window_capture_defaults(obs_data_t *settings)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
|
|
|
obs_data_set_default_bool(settings, "show_shadow", false);
|
|
|
|
window_defaults(settings);
|
|
|
|
}
|
|
|
|
|
2014-09-29 17:36:13 +02:00
|
|
|
static obs_properties_t *window_capture_properties(void *unused)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
2014-09-29 17:36:13 +02:00
|
|
|
UNUSED_PARAMETER(unused);
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
obs_properties_t *props = obs_properties_create();
|
2014-08-20 00:39:41 +02:00
|
|
|
|
|
|
|
add_window_properties(props);
|
|
|
|
|
|
|
|
obs_properties_add_bool(props, "show_shadow",
|
2019-07-09 13:29:39 -05:00
|
|
|
obs_module_text("WindowCapture.ShowShadow"));
|
2014-08-20 00:39:41 +02:00
|
|
|
|
|
|
|
return props;
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void window_capture_update_internal(struct window_capture *wc,
|
2019-07-09 13:29:39 -05:00
|
|
|
obs_data_t *settings)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
2019-07-09 13:29:39 -05:00
|
|
|
wc->image_option = obs_data_get_bool(settings, "show_shadow")
|
|
|
|
? kCGWindowImageDefault
|
|
|
|
: kCGWindowImageBoundsIgnoreFraming;
|
2014-08-20 00:39:41 +02:00
|
|
|
|
|
|
|
update_window(&wc->window, settings);
|
2017-05-29 22:13:33 -04:00
|
|
|
|
|
|
|
if (wc->window.window_name.length) {
|
2019-07-09 13:29:39 -05:00
|
|
|
blog(LOG_INFO,
|
|
|
|
"[window-capture: '%s'] update settings:\n"
|
|
|
|
"\twindow: %s\n"
|
|
|
|
"\towner: %s",
|
|
|
|
obs_source_get_name(wc->source),
|
|
|
|
[wc->window.window_name UTF8String],
|
|
|
|
[wc->window.owner_name UTF8String]);
|
2017-05-29 22:13:33 -04:00
|
|
|
}
|
2014-08-20 00:39:41 +02:00
|
|
|
}
|
|
|
|
|
2014-09-25 17:44:05 -07:00
|
|
|
static void window_capture_update(void *data, obs_data_t *settings)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
|
|
|
@autoreleasepool {
|
|
|
|
return window_capture_update_internal(data, settings);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-16 01:30:51 -07:00
|
|
|
static const char *window_capture_getname(void *unused)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
2015-09-16 01:30:51 -07:00
|
|
|
UNUSED_PARAMETER(unused);
|
2014-08-20 00:39:41 +02:00
|
|
|
return obs_module_text("WindowCapture");
|
|
|
|
}
|
|
|
|
|
|
|
|
static inline void window_capture_tick_internal(struct window_capture *wc,
|
2019-07-09 13:29:39 -05:00
|
|
|
float seconds)
|
2014-08-20 00:39:41 +02:00
|
|
|
{
|
|
|
|
UNUSED_PARAMETER(seconds);
|
|
|
|
os_event_signal(wc->capture_event);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void window_capture_tick(void *data, float seconds)
|
|
|
|
{
|
2015-01-03 22:39:21 -08:00
|
|
|
struct window_capture *wc = data;
|
|
|
|
|
|
|
|
if (!obs_source_showing(wc->source))
|
|
|
|
return;
|
|
|
|
|
2014-08-20 00:39:41 +02:00
|
|
|
@autoreleasepool {
|
|
|
|
return window_capture_tick_internal(data, seconds);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct obs_source_info window_capture_info = {
|
2019-07-09 13:29:39 -05:00
|
|
|
.id = "window_capture",
|
|
|
|
.type = OBS_SOURCE_TYPE_INPUT,
|
|
|
|
.get_name = window_capture_getname,
|
2014-08-20 00:39:41 +02:00
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
.create = window_capture_create,
|
|
|
|
.destroy = window_capture_destroy,
|
2014-08-20 00:39:41 +02:00
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
.output_flags = OBS_SOURCE_ASYNC_VIDEO,
|
|
|
|
.video_tick = window_capture_tick,
|
2014-08-20 00:39:41 +02:00
|
|
|
|
2019-07-09 13:29:39 -05:00
|
|
|
.get_defaults = window_capture_defaults,
|
2014-08-20 00:39:41 +02:00
|
|
|
.get_properties = window_capture_properties,
|
2019-07-09 13:29:39 -05:00
|
|
|
.update = window_capture_update,
|
2019-07-27 23:59:16 -05:00
|
|
|
.icon_type = OBS_ICON_TYPE_WINDOW_CAPTURE,
|
2014-08-20 00:39:41 +02:00
|
|
|
};
|