Browse Source

Add Window Capture to mac-capture plugin

Palana 11 years ago
parent
commit
f8ed37dec4

+ 2 - 0
plugins/mac-capture/CMakeLists.txt

@@ -22,9 +22,11 @@ set(mac-capture_SOURCES
 	audio-device-enum.c
 	audio-device-enum.c
 	mac-audio.c
 	mac-audio.c
 	mac-display-capture.m
 	mac-display-capture.m
+	mac-window-capture.m
 	window-utils.m)
 	window-utils.m)
 	
 	
 set_source_files_properties(mac-display-capture.m
 set_source_files_properties(mac-display-capture.m
+			    mac-window-capture.m
 			    window-utils.m
 			    window-utils.m
 	PROPERTIES LANGUAGE C)
 	PROPERTIES LANGUAGE C)
 
 

+ 2 - 0
plugins/mac-capture/data/locale/en-US.ini

@@ -5,6 +5,8 @@ CoreAudio.Device.Default="Default"
 DisplayCapture="Display Capture"
 DisplayCapture="Display Capture"
 DisplayCapture.Display="Display"
 DisplayCapture.Display="Display"
 DisplayCapture.ShowCursor="Show Cursor"
 DisplayCapture.ShowCursor="Show Cursor"
+WindowCapture="Window Capture"
+WindowCapture.ShowShadow="Show Window shadow"
 WindowUtils.Window="Window"
 WindowUtils.Window="Window"
 WindowUtils.ShowEmptyNames="Show Windows with empty names"
 WindowUtils.ShowEmptyNames="Show Windows with empty names"
 CropMode="Crop"
 CropMode="Crop"

+ 220 - 0
plugins/mac-capture/mac-window-capture.m

@@ -0,0 +1,220 @@
+#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 {
+	obs_source_t source;
+
+	struct cocoa_window window;
+
+	//CGRect              bounds;
+	//CGWindowListOption  window_option;
+	CGWindowImageOption image_option;
+
+	CGColorSpaceRef color_space;
+
+	DARRAY(uint8_t) buffer;
+
+	pthread_t  capture_thread;
+	os_event_t capture_event;
+	os_event_t stop_event;
+};
+
+static CGImageRef get_image(struct window_capture *wc)
+{
+	NSArray *arr = (NSArray*)CGWindowListCreate(
+			kCGWindowListOptionIncludingWindow,
+			wc->window.window_id);
+	[arr autorelease];
+
+	if (arr.count)
+		return CGWindowListCreateImage(CGRectNull,
+				kCGWindowListOptionIncludingWindow,
+				wc->window.window_id, wc->image_option);
+
+	if (!find_window(&wc->window, NULL, false))
+		return NULL;
+
+	return CGWindowListCreateImage(CGRectNull,
+			kCGWindowListOptionIncludingWindow,
+			wc->window.window_id, wc->image_option);
+}
+
+static inline void capture_frame(struct window_capture *wc)
+{
+	uint64_t   ts  = os_gettime_ns();
+	CGImageRef img = get_image(wc);
+	if (!img)
+		return;
+
+	size_t width  = CGImageGetWidth(img);
+	size_t height = CGImageGetHeight(img);
+
+	CGRect rect = {{0, 0}, {width, height}};
+	da_reserve(wc->buffer, width * height * 4);
+	uint8_t *data = wc->buffer.array;
+
+	CGContextRef cg_context = CGBitmapContextCreate(data, width, height,
+			8, width * 4, wc->color_space,
+			kCGBitmapByteOrder32Host |
+			kCGImageAlphaPremultipliedFirst);
+	CGContextSetBlendMode(cg_context, kCGBlendModeCopy);
+	CGContextDrawImage(cg_context, rect, img);
+	CGContextRelease(cg_context);
+	CGImageRelease(img);
+
+	struct obs_source_frame frame = {
+		.format      = VIDEO_FORMAT_BGRA,
+		.width       = width,
+		.height      = height,
+		.data[0]     = data,
+		.linesize[0] = width * 4,
+		.timestamp   = ts,
+	};
+
+	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;
+}
+
+static inline void *window_capture_create_internal(obs_data_t settings,
+		obs_source_t source)
+{
+	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);
+
+	wc->image_option = obs_data_get_bool(settings, "show_shadow") ?
+		kCGWindowImageDefault : kCGWindowImageBoundsIgnoreFraming;
+
+	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;
+}
+
+static void *window_capture_create(obs_data_t settings, obs_source_t source)
+{
+	@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);
+	
+	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);
+}
+
+static void window_capture_defaults(obs_data_t settings)
+{
+	obs_data_set_default_bool(settings, "show_shadow", false);
+	window_defaults(settings);
+}
+
+static obs_properties_t window_capture_properties(void)
+{
+	obs_properties_t props = obs_properties_create();
+
+	add_window_properties(props);
+
+	obs_properties_add_bool(props, "show_shadow",
+			obs_module_text("WindowCapture.ShowShadow"));
+
+	return props;
+}
+
+static inline void window_capture_update_internal(struct window_capture *wc,
+		obs_data_t settings)
+{
+	wc->image_option = obs_data_get_bool(settings, "show_shadow") ?
+		kCGWindowImageDefault : kCGWindowImageBoundsIgnoreFraming;
+
+	update_window(&wc->window, settings);
+}
+
+static void window_capture_update(void *data, obs_data_t settings)
+{
+	@autoreleasepool {
+		return window_capture_update_internal(data, settings);
+	}
+}
+
+static const char *window_capture_getname(void)
+{
+	return obs_module_text("WindowCapture");
+}
+
+static inline void window_capture_tick_internal(struct window_capture *wc,
+		float seconds)
+{
+	UNUSED_PARAMETER(seconds);
+	os_event_signal(wc->capture_event);
+}
+
+static void window_capture_tick(void *data, float seconds)
+{
+	@autoreleasepool {
+		return window_capture_tick_internal(data, seconds);
+	}
+}
+
+struct obs_source_info window_capture_info = {
+	.id             = "window_capture",
+	.type           = OBS_SOURCE_TYPE_INPUT,
+	.get_name       = window_capture_getname,
+
+	.create         = window_capture_create,
+	.destroy        = window_capture_destroy,
+
+	.output_flags   = OBS_SOURCE_ASYNC_VIDEO,
+	.video_tick     = window_capture_tick,
+
+	.get_defaults   = window_capture_defaults,
+	.get_properties = window_capture_properties,
+	.update         = window_capture_update,
+};

+ 2 - 0
plugins/mac-capture/plugin-main.c

@@ -6,11 +6,13 @@ OBS_MODULE_USE_DEFAULT_LOCALE("mac-capture", "en-US")
 extern struct obs_source_info coreaudio_input_capture_info;
 extern struct obs_source_info coreaudio_input_capture_info;
 extern struct obs_source_info coreaudio_output_capture_info;
 extern struct obs_source_info coreaudio_output_capture_info;
 extern struct obs_source_info display_capture_info;
 extern struct obs_source_info display_capture_info;
+extern struct obs_source_info window_capture_info;
 
 
 bool obs_module_load(void)
 bool obs_module_load(void)
 {
 {
 	obs_register_source(&coreaudio_input_capture_info);
 	obs_register_source(&coreaudio_input_capture_info);
 	obs_register_source(&coreaudio_output_capture_info);
 	obs_register_source(&coreaudio_output_capture_info);
 	obs_register_source(&display_capture_info);
 	obs_register_source(&display_capture_info);
+	obs_register_source(&window_capture_info);
 	return true;
 	return true;
 }
 }