|
@@ -0,0 +1,977 @@
|
|
|
+#include <AvailabilityMacros.h>
|
|
|
+#include <Cocoa/Cocoa.h>
|
|
|
+
|
|
|
+bool is_screen_capture_available(void)
|
|
|
+{
|
|
|
+ return (NSClassFromString(@"SCStream") != NULL);
|
|
|
+}
|
|
|
+
|
|
|
+#if __MAC_OS_X_VERSION_MAX_ALLOWED >= 120300 // __MAC_12_3
|
|
|
+#pragma clang diagnostic push
|
|
|
+#pragma clang diagnostic ignored "-Wunguarded-availability-new"
|
|
|
+
|
|
|
+#include <stdlib.h>
|
|
|
+#include <obs-module.h>
|
|
|
+#include <util/threading.h>
|
|
|
+#include <pthread.h>
|
|
|
+
|
|
|
+#include <IOSurface/IOSurface.h>
|
|
|
+#include <ScreenCaptureKit/ScreenCaptureKit.h>
|
|
|
+#include <CoreMedia/CMSampleBuffer.h>
|
|
|
+#include <CoreVideo/CVPixelBuffer.h>
|
|
|
+
|
|
|
+#include "window-utils.h"
|
|
|
+
|
|
|
+#define MACCAP_LOG(level, msg, ...) \
|
|
|
+ blog(level, "[ mac-screencapture ]: " msg, ##__VA_ARGS__)
|
|
|
+#define MACCAP_ERR(msg, ...) MACCAP_LOG(LOG_ERROR, msg, ##__VA_ARGS__)
|
|
|
+
|
|
|
+typedef enum {
|
|
|
+ ScreenCaptureDisplayStream = 0,
|
|
|
+ ScreenCaptureWindowStream = 1,
|
|
|
+ ScreenCaptureApplicationStream = 2,
|
|
|
+} ScreenCaptureStreamType;
|
|
|
+
|
|
|
+@interface ScreenCaptureDelegate : NSObject <SCStreamOutput>
|
|
|
+
|
|
|
+@property struct screen_capture *sc;
|
|
|
+
|
|
|
+@end
|
|
|
+
|
|
|
+struct screen_capture {
|
|
|
+ obs_source_t *source;
|
|
|
+
|
|
|
+ gs_samplerstate_t *sampler;
|
|
|
+ gs_effect_t *effect;
|
|
|
+ gs_texture_t *tex;
|
|
|
+ gs_vertbuffer_t *vertbuf;
|
|
|
+
|
|
|
+ NSRect frame;
|
|
|
+ bool hide_cursor;
|
|
|
+ bool show_hidden_windows;
|
|
|
+ bool show_empty_names;
|
|
|
+
|
|
|
+ SCStream *disp;
|
|
|
+ SCStreamConfiguration *stream_properties;
|
|
|
+ SCShareableContent *shareable_content;
|
|
|
+ ScreenCaptureDelegate *capture_delegate;
|
|
|
+
|
|
|
+ os_event_t *disp_finished;
|
|
|
+ os_event_t *stream_start_completed;
|
|
|
+ os_sem_t *shareable_content_available;
|
|
|
+ IOSurfaceRef current, prev;
|
|
|
+
|
|
|
+ pthread_mutex_t mutex;
|
|
|
+
|
|
|
+ unsigned capture_type;
|
|
|
+ CGDirectDisplayID display;
|
|
|
+ struct cocoa_window window;
|
|
|
+ NSString *application_id;
|
|
|
+};
|
|
|
+
|
|
|
+static void destroy_screen_stream(struct screen_capture *sc)
|
|
|
+{
|
|
|
+ if (sc->disp) {
|
|
|
+ [sc->disp stopCaptureWithCompletionHandler:^(
|
|
|
+ NSError *_Nullable error) {
|
|
|
+ if (error && error.code != 3808) {
|
|
|
+ MACCAP_ERR(
|
|
|
+ "destroy_screen_stream: Failed to stop stream with error %s\n",
|
|
|
+ [[error localizedFailureReason]
|
|
|
+ cStringUsingEncoding:
|
|
|
+ NSUTF8StringEncoding]);
|
|
|
+ }
|
|
|
+ os_event_signal(sc->disp_finished);
|
|
|
+ }];
|
|
|
+ os_event_wait(sc->disp_finished);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->stream_properties) {
|
|
|
+ [sc->stream_properties release];
|
|
|
+ sc->stream_properties = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->tex) {
|
|
|
+ gs_texture_destroy(sc->tex);
|
|
|
+ sc->tex = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->current) {
|
|
|
+ IOSurfaceDecrementUseCount(sc->current);
|
|
|
+ CFRelease(sc->current);
|
|
|
+ sc->current = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->prev) {
|
|
|
+ IOSurfaceDecrementUseCount(sc->prev);
|
|
|
+ CFRelease(sc->prev);
|
|
|
+ sc->prev = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->disp) {
|
|
|
+ [sc->disp release];
|
|
|
+ sc->disp = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ os_event_destroy(sc->disp_finished);
|
|
|
+ os_event_destroy(sc->stream_start_completed);
|
|
|
+}
|
|
|
+
|
|
|
+static void screen_capture_destroy(void *data)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ if (!sc)
|
|
|
+ return;
|
|
|
+
|
|
|
+ obs_enter_graphics();
|
|
|
+
|
|
|
+ destroy_screen_stream(sc);
|
|
|
+
|
|
|
+ if (sc->sampler)
|
|
|
+ gs_samplerstate_destroy(sc->sampler);
|
|
|
+ if (sc->vertbuf)
|
|
|
+ gs_vertexbuffer_destroy(sc->vertbuf);
|
|
|
+
|
|
|
+ obs_leave_graphics();
|
|
|
+
|
|
|
+ if (sc->shareable_content) {
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+ [sc->shareable_content release];
|
|
|
+ os_sem_destroy(sc->shareable_content_available);
|
|
|
+ sc->shareable_content_available = NULL;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (sc->capture_delegate) {
|
|
|
+ [sc->capture_delegate release];
|
|
|
+ }
|
|
|
+
|
|
|
+ destroy_window(&sc->window);
|
|
|
+
|
|
|
+ pthread_mutex_destroy(&sc->mutex);
|
|
|
+ bfree(sc);
|
|
|
+}
|
|
|
+
|
|
|
+static inline void screen_stream_update(struct screen_capture *sc,
|
|
|
+ CMSampleBufferRef sample_buffer)
|
|
|
+{
|
|
|
+ bool frame_detail_errored = false;
|
|
|
+ float scale_factor = 1.0f;
|
|
|
+ CGRect window_rect = {};
|
|
|
+
|
|
|
+ CFArrayRef attachments_array =
|
|
|
+ CMSampleBufferGetSampleAttachmentsArray(sample_buffer, false);
|
|
|
+ if (sc->capture_type == ScreenCaptureWindowStream &&
|
|
|
+ attachments_array != NULL &&
|
|
|
+ CFArrayGetCount(attachments_array) > 0) {
|
|
|
+ CFDictionaryRef attachments_dict =
|
|
|
+ CFArrayGetValueAtIndex(attachments_array, 0);
|
|
|
+ if (attachments_dict != NULL) {
|
|
|
+
|
|
|
+ CFTypeRef frame_scale_factor = CFDictionaryGetValue(
|
|
|
+ attachments_dict, SCStreamFrameInfoScaleFactor);
|
|
|
+ if (frame_scale_factor != NULL) {
|
|
|
+ Boolean result = CFNumberGetValue(
|
|
|
+ (CFNumberRef)frame_scale_factor,
|
|
|
+ kCFNumberFloatType, &scale_factor);
|
|
|
+ if (result == false) {
|
|
|
+ scale_factor = 1.0f;
|
|
|
+ frame_detail_errored = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ CFTypeRef content_rect_dict = CFDictionaryGetValue(
|
|
|
+ attachments_dict, SCStreamFrameInfoContentRect);
|
|
|
+ CFTypeRef content_scale_factor = CFDictionaryGetValue(
|
|
|
+ attachments_dict,
|
|
|
+ SCStreamFrameInfoContentScale);
|
|
|
+ if ((content_rect_dict != NULL) &&
|
|
|
+ (content_scale_factor != NULL)) {
|
|
|
+ CGRect content_rect = {};
|
|
|
+ float points_to_pixels = 0.0f;
|
|
|
+
|
|
|
+ Boolean result =
|
|
|
+ CGRectMakeWithDictionaryRepresentation(
|
|
|
+ (__bridge CFDictionaryRef)
|
|
|
+ content_rect_dict,
|
|
|
+ &content_rect);
|
|
|
+ if (result == false) {
|
|
|
+ content_rect = CGRectZero;
|
|
|
+ frame_detail_errored = true;
|
|
|
+ }
|
|
|
+ result = CFNumberGetValue(
|
|
|
+ (CFNumberRef)content_scale_factor,
|
|
|
+ kCFNumberFloatType, &points_to_pixels);
|
|
|
+ if (result == false) {
|
|
|
+ points_to_pixels = 1.0f;
|
|
|
+ frame_detail_errored = true;
|
|
|
+ }
|
|
|
+
|
|
|
+ window_rect.origin = content_rect.origin;
|
|
|
+ window_rect.size.width =
|
|
|
+ content_rect.size.width /
|
|
|
+ points_to_pixels * scale_factor;
|
|
|
+ window_rect.size.height =
|
|
|
+ content_rect.size.height /
|
|
|
+ points_to_pixels * scale_factor;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ CVImageBufferRef image_buffer =
|
|
|
+ CMSampleBufferGetImageBuffer(sample_buffer);
|
|
|
+
|
|
|
+ CVPixelBufferLockBaseAddress(image_buffer, 0);
|
|
|
+ IOSurfaceRef frame_surface = CVPixelBufferGetIOSurface(image_buffer);
|
|
|
+ CVPixelBufferUnlockBaseAddress(image_buffer, 0);
|
|
|
+
|
|
|
+ IOSurfaceRef prev_current = NULL;
|
|
|
+
|
|
|
+ if (frame_surface && !pthread_mutex_lock(&sc->mutex)) {
|
|
|
+
|
|
|
+ bool needs_to_update_properties = false;
|
|
|
+
|
|
|
+ if (!frame_detail_errored) {
|
|
|
+ if (sc->capture_type == ScreenCaptureWindowStream) {
|
|
|
+ if ((sc->frame.size.width !=
|
|
|
+ window_rect.size.width) ||
|
|
|
+ (sc->frame.size.height !=
|
|
|
+ window_rect.size.height)) {
|
|
|
+ sc->frame.size.width =
|
|
|
+ window_rect.size.width;
|
|
|
+ sc->frame.size.height =
|
|
|
+ window_rect.size.height;
|
|
|
+ needs_to_update_properties = true;
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ size_t width =
|
|
|
+ CVPixelBufferGetWidth(image_buffer);
|
|
|
+ size_t height =
|
|
|
+ CVPixelBufferGetHeight(image_buffer);
|
|
|
+
|
|
|
+ if ((sc->frame.size.width != width) ||
|
|
|
+ (sc->frame.size.height != height)) {
|
|
|
+ sc->frame.size.width = width;
|
|
|
+ sc->frame.size.height = height;
|
|
|
+ needs_to_update_properties = true;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if (needs_to_update_properties) {
|
|
|
+ [sc->stream_properties setWidth:sc->frame.size.width];
|
|
|
+ [sc->stream_properties setHeight:sc->frame.size.height];
|
|
|
+
|
|
|
+ [sc->disp
|
|
|
+ updateConfiguration:sc->stream_properties
|
|
|
+ completionHandler:^(
|
|
|
+ NSError *_Nullable error) {
|
|
|
+ if (error) {
|
|
|
+ MACCAP_ERR(
|
|
|
+ "screen_stream_video_update: Failed to update stream properties with error %s\n",
|
|
|
+ [[error localizedFailureReason]
|
|
|
+ cStringUsingEncoding:
|
|
|
+ NSUTF8StringEncoding]);
|
|
|
+ }
|
|
|
+ }];
|
|
|
+ }
|
|
|
+
|
|
|
+ prev_current = sc->current;
|
|
|
+ sc->current = frame_surface;
|
|
|
+ CFRetain(sc->current);
|
|
|
+ IOSurfaceIncrementUseCount(sc->current);
|
|
|
+
|
|
|
+ pthread_mutex_unlock(&sc->mutex);
|
|
|
+ }
|
|
|
+
|
|
|
+ if (prev_current) {
|
|
|
+ IOSurfaceDecrementUseCount(prev_current);
|
|
|
+ CFRelease(prev_current);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static bool init_screen_stream(struct screen_capture *sc)
|
|
|
+{
|
|
|
+ SCContentFilter *content_filter;
|
|
|
+
|
|
|
+ sc->frame = CGRectZero;
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+
|
|
|
+ __block SCDisplay *target_display = nil;
|
|
|
+ {
|
|
|
+ [sc->shareable_content.displays
|
|
|
+ indexOfObjectPassingTest:^BOOL(
|
|
|
+ SCDisplay *_Nonnull display, NSUInteger idx,
|
|
|
+ BOOL *_Nonnull stop) {
|
|
|
+ if (display.displayID == sc->display) {
|
|
|
+ target_display = sc->shareable_content
|
|
|
+ .displays[idx];
|
|
|
+ *stop = TRUE;
|
|
|
+ }
|
|
|
+ return *stop;
|
|
|
+ }];
|
|
|
+ }
|
|
|
+
|
|
|
+ __block SCWindow *target_window = nil;
|
|
|
+ if (sc->window.window_id != 0) {
|
|
|
+ [sc->shareable_content.windows indexOfObjectPassingTest:^BOOL(
|
|
|
+ SCWindow *_Nonnull window,
|
|
|
+ NSUInteger idx,
|
|
|
+ BOOL *_Nonnull stop) {
|
|
|
+ if (window.windowID == sc->window.window_id) {
|
|
|
+ target_window =
|
|
|
+ sc->shareable_content.windows[idx];
|
|
|
+ *stop = TRUE;
|
|
|
+ }
|
|
|
+ return *stop;
|
|
|
+ }];
|
|
|
+ }
|
|
|
+
|
|
|
+ __block SCRunningApplication *target_application = nil;
|
|
|
+ {
|
|
|
+ [sc->shareable_content.applications
|
|
|
+ indexOfObjectPassingTest:^BOOL(
|
|
|
+ SCRunningApplication *_Nonnull application,
|
|
|
+ NSUInteger idx, BOOL *_Nonnull stop) {
|
|
|
+ if ([application.bundleIdentifier
|
|
|
+ isEqualToString:sc->
|
|
|
+ application_id]) {
|
|
|
+ target_application =
|
|
|
+ sc->shareable_content
|
|
|
+ .applications[idx];
|
|
|
+ *stop = TRUE;
|
|
|
+ }
|
|
|
+ return *stop;
|
|
|
+ }];
|
|
|
+ }
|
|
|
+ NSArray *target_application_array =
|
|
|
+ [[NSArray alloc] initWithObjects:target_application, nil];
|
|
|
+
|
|
|
+ switch (sc->capture_type) {
|
|
|
+ case ScreenCaptureDisplayStream: {
|
|
|
+ content_filter = [[SCContentFilter alloc]
|
|
|
+ initWithDisplay:target_display
|
|
|
+ excludingWindows:[[NSArray alloc] init]];
|
|
|
+ } break;
|
|
|
+ case ScreenCaptureWindowStream: {
|
|
|
+ content_filter = [[SCContentFilter alloc]
|
|
|
+ initWithDesktopIndependentWindow:target_window];
|
|
|
+ } break;
|
|
|
+ case ScreenCaptureApplicationStream: {
|
|
|
+ content_filter = [[SCContentFilter alloc]
|
|
|
+ initWithDisplay:target_display
|
|
|
+ includingApplications:target_application_array
|
|
|
+ exceptingWindows:[[NSArray alloc] init]];
|
|
|
+ } break;
|
|
|
+ }
|
|
|
+ os_sem_post(sc->shareable_content_available);
|
|
|
+
|
|
|
+ sc->stream_properties = [[SCStreamConfiguration alloc] init];
|
|
|
+ [sc->stream_properties setQueueDepth:8];
|
|
|
+ [sc->stream_properties setShowsCursor:!sc->hide_cursor];
|
|
|
+ [sc->stream_properties setPixelFormat:'BGRA'];
|
|
|
+
|
|
|
+ switch (sc->capture_type) {
|
|
|
+ case ScreenCaptureDisplayStream:
|
|
|
+ case ScreenCaptureApplicationStream:
|
|
|
+ if (target_display) {
|
|
|
+ CGDisplayModeRef display_mode =
|
|
|
+ CGDisplayCopyDisplayMode(
|
|
|
+ target_display.displayID);
|
|
|
+ [sc->stream_properties
|
|
|
+ setWidth:CGDisplayModeGetPixelWidth(
|
|
|
+ display_mode)];
|
|
|
+ [sc->stream_properties
|
|
|
+ setHeight:CGDisplayModeGetPixelHeight(
|
|
|
+ display_mode)];
|
|
|
+ CGDisplayModeRelease(display_mode);
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ case ScreenCaptureWindowStream:
|
|
|
+ if (target_window) {
|
|
|
+ [sc->stream_properties
|
|
|
+ setWidth:target_window.frame.size.width];
|
|
|
+ [sc->stream_properties
|
|
|
+ setHeight:target_window.frame.size.height];
|
|
|
+ }
|
|
|
+ break;
|
|
|
+ }
|
|
|
+
|
|
|
+ sc->disp = [[SCStream alloc] initWithFilter:content_filter
|
|
|
+ configuration:sc->stream_properties
|
|
|
+ delegate:nil];
|
|
|
+
|
|
|
+ NSError *error = nil;
|
|
|
+ BOOL did_add_output = [sc->disp addStreamOutput:sc->capture_delegate
|
|
|
+ type:SCStreamOutputTypeScreen
|
|
|
+ sampleHandlerQueue:nil
|
|
|
+ error:&error];
|
|
|
+ if (!did_add_output) {
|
|
|
+ MACCAP_ERR(
|
|
|
+ "init_screen_stream: Failed to add stream output with error %s\n",
|
|
|
+ [[error localizedFailureReason]
|
|
|
+ cStringUsingEncoding:NSUTF8StringEncoding]);
|
|
|
+ [error release];
|
|
|
+ return !did_add_output;
|
|
|
+ }
|
|
|
+
|
|
|
+ os_event_init(&sc->disp_finished, OS_EVENT_TYPE_MANUAL);
|
|
|
+ os_event_init(&sc->stream_start_completed, OS_EVENT_TYPE_MANUAL);
|
|
|
+
|
|
|
+ __block BOOL did_stream_start = false;
|
|
|
+ [sc->disp startCaptureWithCompletionHandler:^(
|
|
|
+ NSError *_Nullable error) {
|
|
|
+ did_stream_start = (BOOL)(error == nil);
|
|
|
+ if (!did_stream_start) {
|
|
|
+ MACCAP_ERR(
|
|
|
+ "init_screen_stream: Failed to start capture with error %s\n",
|
|
|
+ [[error localizedFailureReason]
|
|
|
+ cStringUsingEncoding:
|
|
|
+ NSUTF8StringEncoding]);
|
|
|
+ // Clean up disp so it isn't stopped
|
|
|
+ [sc->disp release];
|
|
|
+ sc->disp = NULL;
|
|
|
+ }
|
|
|
+ os_event_signal(sc->stream_start_completed);
|
|
|
+ }];
|
|
|
+ os_event_wait(sc->stream_start_completed);
|
|
|
+
|
|
|
+ return did_stream_start;
|
|
|
+}
|
|
|
+
|
|
|
+bool init_vertbuf_screen_capture(struct screen_capture *sc)
|
|
|
+{
|
|
|
+ struct gs_vb_data *vb_data = gs_vbdata_create();
|
|
|
+ vb_data->num = 4;
|
|
|
+ vb_data->points = bzalloc(sizeof(struct vec3) * 4);
|
|
|
+ if (!vb_data->points)
|
|
|
+ return false;
|
|
|
+
|
|
|
+ vb_data->num_tex = 1;
|
|
|
+ vb_data->tvarray = bzalloc(sizeof(struct gs_tvertarray));
|
|
|
+ if (!vb_data->tvarray)
|
|
|
+ return false;
|
|
|
+
|
|
|
+ vb_data->tvarray[0].width = 2;
|
|
|
+ vb_data->tvarray[0].array = bzalloc(sizeof(struct vec2) * 4);
|
|
|
+ if (!vb_data->tvarray[0].array)
|
|
|
+ return false;
|
|
|
+
|
|
|
+ sc->vertbuf = gs_vertexbuffer_create(vb_data, GS_DYNAMIC);
|
|
|
+ return sc->vertbuf != NULL;
|
|
|
+}
|
|
|
+
|
|
|
+static void *screen_capture_build_content_list(struct screen_capture *sc)
|
|
|
+{
|
|
|
+ typedef void (^shareable_content_callback)(SCShareableContent *,
|
|
|
+ NSError *);
|
|
|
+ shareable_content_callback new_content_received = ^void(
|
|
|
+ SCShareableContent *shareable_content, NSError *error) {
|
|
|
+ if (error == nil && sc->shareable_content_available != NULL) {
|
|
|
+ sc->shareable_content = [shareable_content retain];
|
|
|
+ } else {
|
|
|
+ MACCAP_ERR(
|
|
|
+ "screen_capture_properties: Failed to get shareable content with error %s\n",
|
|
|
+ [[error localizedFailureReason]
|
|
|
+ cStringUsingEncoding:
|
|
|
+ NSUTF8StringEncoding]);
|
|
|
+ }
|
|
|
+ os_sem_post(sc->shareable_content_available);
|
|
|
+ };
|
|
|
+
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+ [sc->shareable_content release];
|
|
|
+ [SCShareableContent
|
|
|
+ getShareableContentExcludingDesktopWindows:true
|
|
|
+ onScreenWindowsOnly:!sc->show_hidden_windows
|
|
|
+ completionHandler:new_content_received];
|
|
|
+}
|
|
|
+
|
|
|
+static void *screen_capture_create(obs_data_t *settings, obs_source_t *source)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = bzalloc(sizeof(struct screen_capture));
|
|
|
+
|
|
|
+ sc->source = source;
|
|
|
+ sc->hide_cursor = !obs_data_get_bool(settings, "show_cursor");
|
|
|
+ sc->show_empty_names = obs_data_get_bool(settings, "show_empty_names");
|
|
|
+ sc->show_hidden_windows =
|
|
|
+ obs_data_get_bool(settings, "show_hidden_windows");
|
|
|
+
|
|
|
+ init_window(&sc->window, settings);
|
|
|
+ update_window(&sc->window, settings);
|
|
|
+
|
|
|
+ os_sem_init(&sc->shareable_content_available, 1);
|
|
|
+ screen_capture_build_content_list(sc);
|
|
|
+
|
|
|
+ sc->capture_delegate = [[ScreenCaptureDelegate alloc] init];
|
|
|
+ sc->capture_delegate.sc = sc;
|
|
|
+
|
|
|
+ sc->effect = obs_get_base_effect(OBS_EFFECT_DEFAULT_RECT);
|
|
|
+ if (!sc->effect)
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ obs_enter_graphics();
|
|
|
+
|
|
|
+ struct gs_sampler_info info = {
|
|
|
+ .filter = GS_FILTER_LINEAR,
|
|
|
+ .address_u = GS_ADDRESS_CLAMP,
|
|
|
+ .address_v = GS_ADDRESS_CLAMP,
|
|
|
+ .address_w = GS_ADDRESS_CLAMP,
|
|
|
+ .max_anisotropy = 1,
|
|
|
+ };
|
|
|
+ sc->sampler = gs_samplerstate_create(&info);
|
|
|
+ if (!sc->sampler)
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ if (!init_vertbuf_screen_capture(sc))
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ obs_leave_graphics();
|
|
|
+
|
|
|
+ sc->capture_type = obs_data_get_int(settings, "type");
|
|
|
+ sc->display = obs_data_get_int(settings, "display");
|
|
|
+ sc->application_id = [[NSString alloc]
|
|
|
+ initWithUTF8String:obs_data_get_string(settings,
|
|
|
+ "application")];
|
|
|
+ pthread_mutex_init(&sc->mutex, NULL);
|
|
|
+
|
|
|
+ if (!init_screen_stream(sc))
|
|
|
+ goto fail;
|
|
|
+
|
|
|
+ return sc;
|
|
|
+
|
|
|
+fail:
|
|
|
+ obs_leave_graphics();
|
|
|
+ screen_capture_destroy(sc);
|
|
|
+ return NULL;
|
|
|
+}
|
|
|
+
|
|
|
+static void build_sprite(struct gs_vb_data *data, float fcx, float fcy,
|
|
|
+ float start_u, float end_u, float start_v, float end_v)
|
|
|
+{
|
|
|
+ struct vec2 *tvarray = data->tvarray[0].array;
|
|
|
+
|
|
|
+ vec3_set(data->points + 1, fcx, 0.0f, 0.0f);
|
|
|
+ vec3_set(data->points + 2, 0.0f, fcy, 0.0f);
|
|
|
+ vec3_set(data->points + 3, fcx, fcy, 0.0f);
|
|
|
+ vec2_set(tvarray, start_u, start_v);
|
|
|
+ vec2_set(tvarray + 1, end_u, start_v);
|
|
|
+ vec2_set(tvarray + 2, start_u, end_v);
|
|
|
+ vec2_set(tvarray + 3, end_u, end_v);
|
|
|
+}
|
|
|
+
|
|
|
+static inline void build_sprite_rect(struct gs_vb_data *data, float origin_x,
|
|
|
+ float origin_y, float end_x, float end_y)
|
|
|
+{
|
|
|
+ build_sprite(data, fabs(end_x - origin_x), fabs(end_y - origin_y),
|
|
|
+ origin_x, end_x, origin_y, end_y);
|
|
|
+}
|
|
|
+
|
|
|
+static void screen_capture_video_tick(void *data, float seconds)
|
|
|
+{
|
|
|
+ UNUSED_PARAMETER(seconds);
|
|
|
+
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ if (!sc->current)
|
|
|
+ return;
|
|
|
+ if (!obs_source_showing(sc->source))
|
|
|
+ return;
|
|
|
+
|
|
|
+ IOSurfaceRef prev_prev = sc->prev;
|
|
|
+ if (pthread_mutex_lock(&sc->mutex))
|
|
|
+ return;
|
|
|
+ sc->prev = sc->current;
|
|
|
+ sc->current = NULL;
|
|
|
+ pthread_mutex_unlock(&sc->mutex);
|
|
|
+
|
|
|
+ if (prev_prev == sc->prev)
|
|
|
+ return;
|
|
|
+
|
|
|
+ CGPoint origin = {0.f, 0.f};
|
|
|
+ CGPoint end = {sc->frame.size.width, sc->frame.size.height};
|
|
|
+
|
|
|
+ obs_enter_graphics();
|
|
|
+ build_sprite_rect(gs_vertexbuffer_get_data(sc->vertbuf), origin.x,
|
|
|
+ origin.y, end.x, end.y);
|
|
|
+
|
|
|
+ if (sc->tex)
|
|
|
+ gs_texture_rebind_iosurface(sc->tex, sc->prev);
|
|
|
+ else
|
|
|
+ sc->tex = gs_texture_create_from_iosurface(sc->prev);
|
|
|
+ obs_leave_graphics();
|
|
|
+
|
|
|
+ if (prev_prev) {
|
|
|
+ IOSurfaceDecrementUseCount(prev_prev);
|
|
|
+ CFRelease(prev_prev);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void screen_capture_video_render(void *data, gs_effect_t *effect)
|
|
|
+{
|
|
|
+ UNUSED_PARAMETER(effect);
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ if (!sc->tex)
|
|
|
+ return;
|
|
|
+
|
|
|
+ const bool linear_srgb = gs_get_linear_srgb();
|
|
|
+
|
|
|
+ const bool previous = gs_framebuffer_srgb_enabled();
|
|
|
+ gs_enable_framebuffer_srgb(linear_srgb);
|
|
|
+
|
|
|
+ gs_vertexbuffer_flush(sc->vertbuf);
|
|
|
+ gs_load_vertexbuffer(sc->vertbuf);
|
|
|
+ gs_load_indexbuffer(NULL);
|
|
|
+ gs_load_samplerstate(sc->sampler, 0);
|
|
|
+ gs_technique_t *tech = gs_effect_get_technique(sc->effect, "Draw");
|
|
|
+ gs_eparam_t *param = gs_effect_get_param_by_name(sc->effect, "image");
|
|
|
+ if (linear_srgb)
|
|
|
+ gs_effect_set_texture_srgb(param, sc->tex);
|
|
|
+ else
|
|
|
+ gs_effect_set_texture(param, sc->tex);
|
|
|
+ gs_technique_begin(tech);
|
|
|
+ gs_technique_begin_pass(tech, 0);
|
|
|
+
|
|
|
+ gs_draw(GS_TRISTRIP, 0, 4);
|
|
|
+
|
|
|
+ gs_technique_end_pass(tech);
|
|
|
+ gs_technique_end(tech);
|
|
|
+
|
|
|
+ gs_enable_framebuffer_srgb(previous);
|
|
|
+}
|
|
|
+
|
|
|
+static const char *screen_capture_getname(void *unused)
|
|
|
+{
|
|
|
+ UNUSED_PARAMETER(unused);
|
|
|
+ return "macOS ScreenCapture";
|
|
|
+}
|
|
|
+
|
|
|
+static uint32_t screen_capture_getwidth(void *data)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ return sc->frame.size.width;
|
|
|
+}
|
|
|
+
|
|
|
+static uint32_t screen_capture_getheight(void *data)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ return sc->frame.size.height;
|
|
|
+}
|
|
|
+
|
|
|
+static void screen_capture_defaults(obs_data_t *settings)
|
|
|
+{
|
|
|
+ CGDirectDisplayID initial_display = 0;
|
|
|
+ {
|
|
|
+ NSScreen *mainScreen = [NSScreen mainScreen];
|
|
|
+ if (mainScreen) {
|
|
|
+ NSNumber *screen_num =
|
|
|
+ mainScreen.deviceDescription[@"NSScreenNumber"];
|
|
|
+ if (screen_num) {
|
|
|
+ initial_display =
|
|
|
+ (CGDirectDisplayID)
|
|
|
+ screen_num.pointerValue;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ obs_data_set_default_int(settings, "type", 0);
|
|
|
+ obs_data_set_default_int(settings, "display", initial_display);
|
|
|
+ obs_data_set_default_int(settings, "window", kCGNullWindowID);
|
|
|
+ obs_data_set_default_obj(settings, "application", NULL);
|
|
|
+ obs_data_set_default_bool(settings, "show_cursor", true);
|
|
|
+ obs_data_set_default_bool(settings, "show_empty_names", false);
|
|
|
+ obs_data_set_default_bool(settings, "show_hidden_windows", false);
|
|
|
+
|
|
|
+ window_defaults(settings);
|
|
|
+}
|
|
|
+
|
|
|
+static void screen_capture_update(void *data, obs_data_t *settings)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ CGWindowID old_window_id = sc->window.window_id;
|
|
|
+ update_window(&sc->window, settings);
|
|
|
+
|
|
|
+ ScreenCaptureStreamType capture_type =
|
|
|
+ (ScreenCaptureStreamType)obs_data_get_int(settings, "type");
|
|
|
+ CGDirectDisplayID display =
|
|
|
+ (CGDirectDisplayID)obs_data_get_int(settings, "display");
|
|
|
+ NSString *application_id = [[NSString alloc]
|
|
|
+ initWithUTF8String:obs_data_get_string(settings,
|
|
|
+ "application")];
|
|
|
+ bool show_cursor = obs_data_get_bool(settings, "show_cursor");
|
|
|
+ bool show_empty_names = obs_data_get_bool(settings, "show_empty_names");
|
|
|
+ bool show_hidden_windows =
|
|
|
+ obs_data_get_bool(settings, "show_hidden_windows");
|
|
|
+
|
|
|
+ if (capture_type == sc->capture_type) {
|
|
|
+ switch (sc->capture_type) {
|
|
|
+ case ScreenCaptureDisplayStream: {
|
|
|
+ if (sc->display == display &&
|
|
|
+ sc->hide_cursor != show_cursor)
|
|
|
+ return;
|
|
|
+ } break;
|
|
|
+ case ScreenCaptureWindowStream: {
|
|
|
+ if (old_window_id == sc->window.window_id &&
|
|
|
+ sc->hide_cursor != show_cursor)
|
|
|
+ return;
|
|
|
+ } break;
|
|
|
+ case ScreenCaptureApplicationStream: {
|
|
|
+ if (sc->display == display &&
|
|
|
+ [application_id
|
|
|
+ isEqualToString:sc->application_id] &&
|
|
|
+ sc->hide_cursor != show_cursor)
|
|
|
+ return;
|
|
|
+ } break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ obs_enter_graphics();
|
|
|
+
|
|
|
+ destroy_screen_stream(sc);
|
|
|
+ sc->capture_type = capture_type;
|
|
|
+ sc->display = display;
|
|
|
+ sc->application_id = application_id;
|
|
|
+ sc->hide_cursor = !show_cursor;
|
|
|
+ sc->show_empty_names = show_empty_names;
|
|
|
+ sc->show_hidden_windows = show_hidden_windows;
|
|
|
+ init_screen_stream(sc);
|
|
|
+
|
|
|
+ obs_leave_graphics();
|
|
|
+}
|
|
|
+
|
|
|
+static bool build_display_list(struct screen_capture *sc,
|
|
|
+ obs_properties_t *props)
|
|
|
+{
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+
|
|
|
+ obs_property_t *display_list = obs_properties_get(props, "display");
|
|
|
+ obs_property_list_clear(display_list);
|
|
|
+ [sc->shareable_content.displays
|
|
|
+ enumerateObjectsUsingBlock:^(SCDisplay *_Nonnull display,
|
|
|
+ NSUInteger idx,
|
|
|
+ BOOL *_Nonnull stop) {
|
|
|
+ UNUSED_PARAMETER(idx);
|
|
|
+ UNUSED_PARAMETER(stop);
|
|
|
+
|
|
|
+ NSUInteger screen_index = [NSScreen.screens
|
|
|
+ indexOfObjectPassingTest:^BOOL(
|
|
|
+ NSScreen *_Nonnull screen,
|
|
|
+ NSUInteger index, BOOL *_Nonnull stop) {
|
|
|
+ UNUSED_PARAMETER(index);
|
|
|
+ NSNumber *screen_num =
|
|
|
+ screen.deviceDescription
|
|
|
+ [@"NSScreenNumber"];
|
|
|
+ CGDirectDisplayID screen_display_id =
|
|
|
+ (CGDirectDisplayID)
|
|
|
+ screen_num.pointerValue;
|
|
|
+ stop = (BOOL)(screen_display_id ==
|
|
|
+ display.displayID);
|
|
|
+ return stop;
|
|
|
+ }];
|
|
|
+ NSScreen *screen =
|
|
|
+ [NSScreen.screens objectAtIndex:screen_index];
|
|
|
+
|
|
|
+ char dimension_buffer[4][12] = {};
|
|
|
+ char name_buffer[256] = {};
|
|
|
+ sprintf(dimension_buffer[0], "%u",
|
|
|
+ (uint32_t)screen.frame.size.width);
|
|
|
+ sprintf(dimension_buffer[1], "%u",
|
|
|
+ (uint32_t)screen.frame.size.height);
|
|
|
+ sprintf(dimension_buffer[2], "%d",
|
|
|
+ (int32_t)screen.frame.origin.x);
|
|
|
+ sprintf(dimension_buffer[3], "%d",
|
|
|
+ (int32_t)screen.frame.origin.y);
|
|
|
+
|
|
|
+ sprintf(name_buffer,
|
|
|
+ "%.200s: %.12sx%.12s @ %.12s,%.12s",
|
|
|
+ screen.localizedName.UTF8String,
|
|
|
+ dimension_buffer[0], dimension_buffer[1],
|
|
|
+ dimension_buffer[2], dimension_buffer[3]);
|
|
|
+
|
|
|
+ obs_property_list_add_int(display_list, name_buffer,
|
|
|
+ display.displayID);
|
|
|
+ }];
|
|
|
+
|
|
|
+ os_sem_post(sc->shareable_content_available);
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+static bool build_window_list(struct screen_capture *sc,
|
|
|
+ obs_properties_t *props)
|
|
|
+{
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+
|
|
|
+ obs_property_t *window_list = obs_properties_get(props, "window");
|
|
|
+ obs_property_list_clear(window_list);
|
|
|
+ [sc->shareable_content.windows enumerateObjectsUsingBlock:^(
|
|
|
+ SCWindow *_Nonnull window,
|
|
|
+ NSUInteger idx,
|
|
|
+ BOOL *_Nonnull stop) {
|
|
|
+ UNUSED_PARAMETER(idx);
|
|
|
+ UNUSED_PARAMETER(stop);
|
|
|
+ NSString *app_name = window.owningApplication.applicationName;
|
|
|
+ NSString *title = window.title;
|
|
|
+
|
|
|
+ if (!sc->show_empty_names) {
|
|
|
+ if (app_name == NULL || title == NULL) {
|
|
|
+ return;
|
|
|
+ } else if ([app_name isEqualToString:@""] ||
|
|
|
+ [title isEqualToString:@""]) {
|
|
|
+ return;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ const char *list_text =
|
|
|
+ [[NSString stringWithFormat:@"[%@] %@", app_name, title]
|
|
|
+ UTF8String];
|
|
|
+ obs_property_list_add_int(window_list, list_text,
|
|
|
+ window.windowID);
|
|
|
+ }];
|
|
|
+
|
|
|
+ os_sem_post(sc->shareable_content_available);
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+static bool build_application_list(struct screen_capture *sc,
|
|
|
+ obs_properties_t *props)
|
|
|
+{
|
|
|
+ os_sem_wait(sc->shareable_content_available);
|
|
|
+
|
|
|
+ obs_property_t *application_list =
|
|
|
+ obs_properties_get(props, "application");
|
|
|
+ obs_property_list_clear(application_list);
|
|
|
+ [sc->shareable_content.applications
|
|
|
+ enumerateObjectsUsingBlock:^(
|
|
|
+ SCRunningApplication *_Nonnull application,
|
|
|
+ NSUInteger idx, BOOL *_Nonnull stop) {
|
|
|
+ UNUSED_PARAMETER(idx);
|
|
|
+ UNUSED_PARAMETER(stop);
|
|
|
+ const char *name =
|
|
|
+ [application.applicationName UTF8String];
|
|
|
+ const char *bundle_id =
|
|
|
+ [application.bundleIdentifier UTF8String];
|
|
|
+ obs_property_list_add_string(application_list, name,
|
|
|
+ bundle_id);
|
|
|
+ }];
|
|
|
+
|
|
|
+ os_sem_post(sc->shareable_content_available);
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+static bool content_changed(struct screen_capture *sc, obs_properties_t *props)
|
|
|
+{
|
|
|
+ screen_capture_build_content_list(sc);
|
|
|
+
|
|
|
+ build_display_list(sc, props);
|
|
|
+ build_window_list(sc, props);
|
|
|
+ build_application_list(sc, props);
|
|
|
+
|
|
|
+ return true;
|
|
|
+}
|
|
|
+
|
|
|
+static bool content_settings_changed(void *priv, obs_properties_t *props,
|
|
|
+ obs_property_t *property,
|
|
|
+ obs_data_t *settings)
|
|
|
+{
|
|
|
+ UNUSED_PARAMETER(property);
|
|
|
+
|
|
|
+ struct screen_capture *sc = (struct screen_capture *)priv;
|
|
|
+
|
|
|
+ sc->show_empty_names = obs_data_get_bool(settings, "show_empty_names");
|
|
|
+ sc->show_hidden_windows =
|
|
|
+ obs_data_get_bool(settings, "show_hidden_windows");
|
|
|
+
|
|
|
+ return content_changed(sc, props);
|
|
|
+}
|
|
|
+
|
|
|
+static obs_properties_t *screen_capture_properties(void *data)
|
|
|
+{
|
|
|
+ struct screen_capture *sc = data;
|
|
|
+
|
|
|
+ screen_capture_build_content_list(sc);
|
|
|
+
|
|
|
+ obs_properties_t *props = obs_properties_create();
|
|
|
+
|
|
|
+ obs_property_t *capture_type = obs_properties_add_list(
|
|
|
+ props, "type", obs_module_text("Method"), OBS_COMBO_TYPE_LIST,
|
|
|
+ OBS_COMBO_FORMAT_INT);
|
|
|
+ obs_property_list_add_int(capture_type,
|
|
|
+ obs_module_text("DisplayCapture"), 0);
|
|
|
+ obs_property_list_add_int(capture_type,
|
|
|
+ obs_module_text("WindowCapture"), 1);
|
|
|
+ obs_property_list_add_int(capture_type, "Application Capture", 2);
|
|
|
+
|
|
|
+ obs_property_t *display_list = obs_properties_add_list(
|
|
|
+ props, "display", obs_module_text("DisplayCapture.Display"),
|
|
|
+ OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
|
|
|
+
|
|
|
+ obs_property_t *window_list = obs_properties_add_list(
|
|
|
+ props, "window", obs_module_text("WindowUtils.Window"),
|
|
|
+ OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
|
|
|
+
|
|
|
+ obs_property_t *empty = obs_properties_add_bool(
|
|
|
+ props, "show_empty_names",
|
|
|
+ obs_module_text("WindowUtils.ShowEmptyNames"));
|
|
|
+ obs_property_set_modified_callback2(empty, content_settings_changed,
|
|
|
+ sc);
|
|
|
+
|
|
|
+ obs_property_t *hidden = obs_properties_add_bool(
|
|
|
+ props, "show_hidden_windows",
|
|
|
+ obs_module_text("WindowUtils.ShowHidden"));
|
|
|
+ obs_property_set_modified_callback2(hidden, content_settings_changed,
|
|
|
+ sc);
|
|
|
+
|
|
|
+ obs_property_t *application_list = obs_properties_add_list(
|
|
|
+ props, "application", obs_module_text("Application"),
|
|
|
+ OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
|
|
|
+
|
|
|
+ content_changed(sc, props);
|
|
|
+
|
|
|
+ obs_properties_add_bool(props, "show_cursor",
|
|
|
+ obs_module_text("DisplayCapture.ShowCursor"));
|
|
|
+
|
|
|
+ return props;
|
|
|
+}
|
|
|
+
|
|
|
+struct obs_source_info screen_capture_info = {
|
|
|
+ .id = "screen_capture",
|
|
|
+ .type = OBS_SOURCE_TYPE_INPUT,
|
|
|
+ .get_name = screen_capture_getname,
|
|
|
+
|
|
|
+ .create = screen_capture_create,
|
|
|
+ .destroy = screen_capture_destroy,
|
|
|
+
|
|
|
+ .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_CUSTOM_DRAW |
|
|
|
+ OBS_SOURCE_DO_NOT_DUPLICATE | OBS_SOURCE_SRGB,
|
|
|
+ .video_tick = screen_capture_video_tick,
|
|
|
+ .video_render = screen_capture_video_render,
|
|
|
+
|
|
|
+ .get_width = screen_capture_getwidth,
|
|
|
+ .get_height = screen_capture_getheight,
|
|
|
+
|
|
|
+ .get_defaults = screen_capture_defaults,
|
|
|
+ .get_properties = screen_capture_properties,
|
|
|
+ .update = screen_capture_update,
|
|
|
+ .icon_type = OBS_ICON_TYPE_GAME_CAPTURE,
|
|
|
+};
|
|
|
+
|
|
|
+@implementation ScreenCaptureDelegate
|
|
|
+
|
|
|
+- (void)stream:(SCStream *)stream
|
|
|
+ didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer
|
|
|
+ ofType:(SCStreamOutputType)type
|
|
|
+{
|
|
|
+ if (self.sc != NULL) {
|
|
|
+ screen_stream_update(self.sc, sampleBuffer);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+@end
|
|
|
+
|
|
|
+// "-Wunguarded-availability-new"
|
|
|
+#pragma clang diagnostic pop
|
|
|
+#endif
|