mac-display-capture.m 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679
  1. #include <stdlib.h>
  2. #include <obs-module.h>
  3. #include <util/threading.h>
  4. #include <pthread.h>
  5. #import <AvailabilityMacros.h>
  6. #import <CoreGraphics/CGDisplayStream.h>
  7. #import <Cocoa/Cocoa.h>
  8. #include "window-utils.h"
  9. enum crop_mode {
  10. CROP_NONE,
  11. CROP_MANUAL,
  12. CROP_TO_WINDOW,
  13. CROP_TO_WINDOW_AND_MANUAL,
  14. CROP_INVALID
  15. };
  16. static inline bool requires_window(enum crop_mode mode)
  17. {
  18. return mode == CROP_TO_WINDOW || mode == CROP_TO_WINDOW_AND_MANUAL;
  19. }
  20. struct display_capture {
  21. obs_source_t *source;
  22. gs_samplerstate_t *sampler;
  23. gs_effect_t *effect;
  24. gs_texture_t *tex;
  25. gs_vertbuffer_t *vertbuf;
  26. NSScreen *screen;
  27. unsigned display;
  28. NSRect frame;
  29. bool hide_cursor;
  30. enum crop_mode crop;
  31. CGRect crop_rect;
  32. struct cocoa_window window;
  33. CGRect window_rect;
  34. bool on_screen;
  35. bool hide_when_minimized;
  36. os_event_t *disp_finished;
  37. CGDisplayStreamRef disp;
  38. IOSurfaceRef current, prev;
  39. pthread_mutex_t mutex;
  40. };
  41. static inline bool crop_mode_valid(enum crop_mode mode)
  42. {
  43. return CROP_NONE <= mode && mode < CROP_INVALID;
  44. }
  45. static void destroy_display_stream(struct display_capture *dc)
  46. {
  47. if (dc->disp) {
  48. CGDisplayStreamStop(dc->disp);
  49. os_event_wait(dc->disp_finished);
  50. }
  51. if (dc->tex) {
  52. gs_texture_destroy(dc->tex);
  53. dc->tex = NULL;
  54. }
  55. if (dc->current) {
  56. IOSurfaceDecrementUseCount(dc->current);
  57. CFRelease(dc->current);
  58. dc->current = NULL;
  59. }
  60. if (dc->prev) {
  61. IOSurfaceDecrementUseCount(dc->prev);
  62. CFRelease(dc->prev);
  63. dc->prev = NULL;
  64. }
  65. if (dc->disp) {
  66. CFRelease(dc->disp);
  67. dc->disp = NULL;
  68. }
  69. if (dc->screen) {
  70. [dc->screen release];
  71. dc->screen = nil;
  72. }
  73. os_event_destroy(dc->disp_finished);
  74. }
  75. static void display_capture_destroy(void *data)
  76. {
  77. struct display_capture *dc = data;
  78. if (!dc)
  79. return;
  80. obs_enter_graphics();
  81. destroy_display_stream(dc);
  82. if (dc->sampler)
  83. gs_samplerstate_destroy(dc->sampler);
  84. if (dc->vertbuf)
  85. gs_vertexbuffer_destroy(dc->vertbuf);
  86. obs_leave_graphics();
  87. destroy_window(&dc->window);
  88. pthread_mutex_destroy(&dc->mutex);
  89. bfree(dc);
  90. }
  91. static inline void update_window_params(struct display_capture *dc)
  92. {
  93. if (!requires_window(dc->crop))
  94. return;
  95. NSArray *arr = (NSArray *) CGWindowListCopyWindowInfo(kCGWindowListOptionIncludingWindow, dc->window.window_id);
  96. if (arr.count) {
  97. NSDictionary *dict = arr[0];
  98. NSDictionary *ref = dict[(NSString *) kCGWindowBounds];
  99. CGRectMakeWithDictionaryRepresentation((CFDictionaryRef) ref, &dc->window_rect);
  100. dc->on_screen = dict[(NSString *) kCGWindowIsOnscreen] != nil;
  101. dc->window_rect = [dc->screen convertRectToBacking:dc->window_rect];
  102. } else {
  103. if (find_window(&dc->window, NULL, false))
  104. update_window_params(dc);
  105. else
  106. dc->on_screen = false;
  107. }
  108. [arr release];
  109. }
  110. static inline void display_stream_update(struct display_capture *dc, CGDisplayStreamFrameStatus status,
  111. uint64_t display_time, IOSurfaceRef frame_surface,
  112. CGDisplayStreamUpdateRef update_ref)
  113. {
  114. UNUSED_PARAMETER(display_time);
  115. if (status == kCGDisplayStreamFrameStatusStopped) {
  116. os_event_signal(dc->disp_finished);
  117. return;
  118. }
  119. IOSurfaceRef prev_current = NULL;
  120. if (frame_surface && !pthread_mutex_lock(&dc->mutex)) {
  121. prev_current = dc->current;
  122. dc->current = frame_surface;
  123. CFRetain(dc->current);
  124. IOSurfaceIncrementUseCount(dc->current);
  125. update_window_params(dc);
  126. pthread_mutex_unlock(&dc->mutex);
  127. }
  128. if (prev_current) {
  129. IOSurfaceDecrementUseCount(prev_current);
  130. CFRelease(prev_current);
  131. }
  132. size_t dropped_frames = CGDisplayStreamUpdateGetDropCount(update_ref);
  133. if (dropped_frames > 0)
  134. blog(LOG_INFO, "%s: Dropped %zu frames", obs_source_get_name(dc->source), dropped_frames);
  135. }
  136. static bool init_display_stream(struct display_capture *dc)
  137. {
  138. [[NSScreen screens] enumerateObjectsUsingBlock:^(NSScreen *_Nonnull screen, NSUInteger index __unused,
  139. BOOL *_Nonnull stop __unused) {
  140. NSNumber *screenNumber = screen.deviceDescription[@"NSScreenNumber"];
  141. CGDirectDisplayID display_id = (CGDirectDisplayID) screenNumber.intValue;
  142. if (display_id == dc->display) {
  143. dc->screen = [screen retain];
  144. *stop = YES;
  145. }
  146. }];
  147. if (!dc->screen) {
  148. return false;
  149. }
  150. dc->frame = [dc->screen convertRectToBacking:dc->screen.frame];
  151. NSNumber *screen_num = dc->screen.deviceDescription[@"NSScreenNumber"];
  152. CGDirectDisplayID disp_id = screen_num.unsignedIntValue;
  153. NSDictionary *rect_dict = CFBridgingRelease(CGRectCreateDictionaryRepresentation(
  154. CGRectMake(0, 0, dc->screen.frame.size.width, dc->screen.frame.size.height)));
  155. CFBooleanRef show_cursor_cf = dc->hide_cursor ? kCFBooleanFalse : kCFBooleanTrue;
  156. NSDictionary *dict = @{
  157. (__bridge NSString *) kCGDisplayStreamSourceRect: rect_dict,
  158. (__bridge NSString *) kCGDisplayStreamQueueDepth: @5,
  159. (__bridge NSString *) kCGDisplayStreamShowCursor: (id) show_cursor_cf,
  160. };
  161. os_event_init(&dc->disp_finished, OS_EVENT_TYPE_MANUAL);
  162. FourCharCode bgra_code = 0;
  163. bgra_code = ('B' << 24) | ('G' << 16) | ('R' << 8) | 'A';
  164. const CGSize *size = &dc->frame.size;
  165. dc->disp = CGDisplayStreamCreateWithDispatchQueue(
  166. disp_id, (size_t) size->width, (size_t) size->height, bgra_code, (__bridge CFDictionaryRef) dict,
  167. dispatch_queue_create(NULL, NULL),
  168. ^(CGDisplayStreamFrameStatus status, uint64_t displayTime, IOSurfaceRef frameSurface,
  169. CGDisplayStreamUpdateRef updateRef) {
  170. display_stream_update(dc, status, displayTime, frameSurface, updateRef);
  171. });
  172. return !CGDisplayStreamStart(dc->disp);
  173. }
  174. bool init_vertbuf(struct display_capture *dc)
  175. {
  176. struct gs_vb_data *vb_data = gs_vbdata_create();
  177. vb_data->num = 4;
  178. vb_data->points = bzalloc(sizeof(struct vec3) * 4);
  179. if (!vb_data->points)
  180. return false;
  181. vb_data->num_tex = 1;
  182. vb_data->tvarray = bzalloc(sizeof(struct gs_tvertarray));
  183. if (!vb_data->tvarray)
  184. return false;
  185. vb_data->tvarray[0].width = 2;
  186. vb_data->tvarray[0].array = bzalloc(sizeof(struct vec2) * 4);
  187. if (!vb_data->tvarray[0].array)
  188. return false;
  189. dc->vertbuf = gs_vertexbuffer_create(vb_data, GS_DYNAMIC);
  190. return dc->vertbuf != NULL;
  191. }
  192. void load_crop(struct display_capture *dc, obs_data_t *settings);
  193. static void *display_capture_create(obs_data_t *settings, obs_source_t *source)
  194. {
  195. struct display_capture *dc = bzalloc(sizeof(struct display_capture));
  196. dc->source = source;
  197. dc->hide_cursor = !obs_data_get_bool(settings, "show_cursor");
  198. dc->effect = obs_get_base_effect(OBS_EFFECT_DEFAULT_RECT);
  199. if (!dc->effect)
  200. goto fail;
  201. obs_enter_graphics();
  202. struct gs_sampler_info info = {
  203. .filter = GS_FILTER_LINEAR,
  204. .address_u = GS_ADDRESS_CLAMP,
  205. .address_v = GS_ADDRESS_CLAMP,
  206. .address_w = GS_ADDRESS_CLAMP,
  207. .max_anisotropy = 1,
  208. };
  209. dc->sampler = gs_samplerstate_create(&info);
  210. if (!dc->sampler)
  211. goto fail;
  212. if (!init_vertbuf(dc))
  213. goto fail;
  214. obs_leave_graphics();
  215. init_window(&dc->window, settings);
  216. load_crop(dc, settings);
  217. dc->display = get_display_migrate_settings(settings);
  218. pthread_mutex_init(&dc->mutex, NULL);
  219. if (!init_display_stream(dc))
  220. goto fail;
  221. return dc;
  222. fail:
  223. obs_leave_graphics();
  224. display_capture_destroy(dc);
  225. return NULL;
  226. }
  227. static void build_sprite(struct gs_vb_data *data, float fcx, float fcy, float start_u, float end_u, float start_v,
  228. float end_v)
  229. {
  230. struct vec2 *tvarray = data->tvarray[0].array;
  231. vec3_set(data->points + 1, fcx, 0.0f, 0.0f);
  232. vec3_set(data->points + 2, 0.0f, fcy, 0.0f);
  233. vec3_set(data->points + 3, fcx, fcy, 0.0f);
  234. vec2_set(tvarray, start_u, start_v);
  235. vec2_set(tvarray + 1, end_u, start_v);
  236. vec2_set(tvarray + 2, start_u, end_v);
  237. vec2_set(tvarray + 3, end_u, end_v);
  238. }
  239. static inline void build_sprite_rect(struct gs_vb_data *data, float origin_x, float origin_y, float end_x, float end_y)
  240. {
  241. build_sprite(data, fabsf(end_x - origin_x), fabsf(end_y - origin_y), origin_x, end_x, origin_y, end_y);
  242. }
  243. static void display_capture_video_tick(void *data, float seconds)
  244. {
  245. UNUSED_PARAMETER(seconds);
  246. struct display_capture *dc = data;
  247. if (!dc->current)
  248. return;
  249. if (!obs_source_showing(dc->source))
  250. return;
  251. IOSurfaceRef prev_prev = dc->prev;
  252. if (pthread_mutex_lock(&dc->mutex))
  253. return;
  254. dc->prev = dc->current;
  255. dc->current = NULL;
  256. pthread_mutex_unlock(&dc->mutex);
  257. if (prev_prev == dc->prev)
  258. return;
  259. if (requires_window(dc->crop) && !dc->on_screen)
  260. goto cleanup;
  261. CGPoint origin = {0.f};
  262. CGPoint end = {0.f};
  263. double x, y;
  264. switch (dc->crop) {
  265. case CROP_INVALID:
  266. break;
  267. case CROP_MANUAL:
  268. origin.x += dc->crop_rect.origin.x;
  269. origin.y += dc->crop_rect.origin.y;
  270. end.y -= dc->crop_rect.size.height;
  271. end.x -= dc->crop_rect.size.width;
  272. case CROP_NONE:
  273. end.y += dc->frame.size.height;
  274. end.x += dc->frame.size.width;
  275. break;
  276. case CROP_TO_WINDOW_AND_MANUAL:
  277. origin.x += dc->crop_rect.origin.x;
  278. origin.y += dc->crop_rect.origin.y;
  279. end.y -= dc->crop_rect.size.height;
  280. end.x -= dc->crop_rect.size.width;
  281. case CROP_TO_WINDOW:
  282. origin.x += x = dc->window_rect.origin.x - dc->frame.origin.x;
  283. origin.y += y = dc->window_rect.origin.y - dc->frame.origin.y;
  284. end.y += dc->window_rect.size.height + y;
  285. end.x += dc->window_rect.size.width + x;
  286. break;
  287. }
  288. obs_enter_graphics();
  289. build_sprite_rect(gs_vertexbuffer_get_data(dc->vertbuf), (float) origin.x, (float) origin.y, (float) end.x,
  290. (float) end.y);
  291. if (dc->tex)
  292. gs_texture_rebind_iosurface(dc->tex, dc->prev);
  293. else
  294. dc->tex = gs_texture_create_from_iosurface(dc->prev);
  295. obs_leave_graphics();
  296. cleanup:
  297. if (prev_prev) {
  298. IOSurfaceDecrementUseCount(prev_prev);
  299. CFRelease(prev_prev);
  300. }
  301. }
  302. static void display_capture_video_render(void *data, gs_effect_t *effect)
  303. {
  304. UNUSED_PARAMETER(effect);
  305. struct display_capture *dc = data;
  306. if (!dc->tex || (requires_window(dc->crop) && !dc->on_screen))
  307. return;
  308. const bool linear_srgb = gs_get_linear_srgb();
  309. const bool previous = gs_framebuffer_srgb_enabled();
  310. gs_enable_framebuffer_srgb(linear_srgb);
  311. gs_vertexbuffer_flush(dc->vertbuf);
  312. gs_load_vertexbuffer(dc->vertbuf);
  313. gs_load_indexbuffer(NULL);
  314. gs_load_samplerstate(dc->sampler, 0);
  315. gs_technique_t *tech = gs_effect_get_technique(dc->effect, "Draw");
  316. gs_eparam_t *param = gs_effect_get_param_by_name(dc->effect, "image");
  317. if (linear_srgb)
  318. gs_effect_set_texture_srgb(param, dc->tex);
  319. else
  320. gs_effect_set_texture(param, dc->tex);
  321. gs_technique_begin(tech);
  322. gs_technique_begin_pass(tech, 0);
  323. gs_draw(GS_TRISTRIP, 0, 4);
  324. gs_technique_end_pass(tech);
  325. gs_technique_end(tech);
  326. gs_enable_framebuffer_srgb(previous);
  327. }
  328. static const char *display_capture_getname(void *unused)
  329. {
  330. UNUSED_PARAMETER(unused);
  331. return obs_module_text("DisplayCapture");
  332. }
  333. static uint32_t display_capture_getwidth(void *data)
  334. {
  335. struct display_capture *dc = data;
  336. double crop = dc->crop_rect.origin.x + dc->crop_rect.size.width;
  337. switch (dc->crop) {
  338. case CROP_NONE:
  339. return (uint32_t) dc->frame.size.width;
  340. case CROP_MANUAL:
  341. return (uint32_t) fabs(dc->frame.size.width - crop);
  342. case CROP_TO_WINDOW:
  343. return (uint32_t) dc->window_rect.size.width;
  344. case CROP_TO_WINDOW_AND_MANUAL:
  345. return (uint32_t) fabs(dc->window_rect.size.width - crop);
  346. case CROP_INVALID:
  347. break;
  348. }
  349. return 0;
  350. }
  351. static uint32_t display_capture_getheight(void *data)
  352. {
  353. struct display_capture *dc = data;
  354. double crop = dc->crop_rect.origin.y + dc->crop_rect.size.height;
  355. switch (dc->crop) {
  356. case CROP_NONE:
  357. return (uint32_t) dc->frame.size.height;
  358. case CROP_MANUAL:
  359. return (uint32_t) fabs(dc->frame.size.height - crop);
  360. case CROP_TO_WINDOW:
  361. return (uint32_t) dc->window_rect.size.height;
  362. case CROP_TO_WINDOW_AND_MANUAL:
  363. return (uint32_t) fabs(dc->window_rect.size.height - crop);
  364. case CROP_INVALID:
  365. break;
  366. }
  367. return 0;
  368. }
  369. static void display_capture_defaults(obs_data_t *settings)
  370. {
  371. NSNumber *screen = [[NSScreen mainScreen] deviceDescription][@"NSScreenNumber"];
  372. CFUUIDRef display_uuid = CGDisplayCreateUUIDFromDisplayID((CGDirectDisplayID) screen.intValue);
  373. CFStringRef uuid_string = CFUUIDCreateString(kCFAllocatorDefault, display_uuid);
  374. obs_data_set_default_string(settings, "display_uuid", CFStringGetCStringPtr(uuid_string, kCFStringEncodingUTF8));
  375. CFRelease(uuid_string);
  376. CFRelease(display_uuid);
  377. obs_data_set_default_bool(settings, "show_cursor", true);
  378. obs_data_set_default_int(settings, "crop_mode", CROP_NONE);
  379. window_defaults(settings);
  380. }
  381. void load_crop_mode(enum crop_mode *mode, obs_data_t *settings)
  382. {
  383. *mode = (int) obs_data_get_int(settings, "crop_mode");
  384. if (!crop_mode_valid(*mode))
  385. *mode = CROP_NONE;
  386. }
  387. void load_crop(struct display_capture *dc, obs_data_t *settings)
  388. {
  389. load_crop_mode(&dc->crop, settings);
  390. #define CROP_VAR_NAME(var, mode) (mode "." #var)
  391. #define LOAD_CROP_VAR(var, mode) dc->crop_rect.var = obs_data_get_double(settings, CROP_VAR_NAME(var, mode));
  392. switch (dc->crop) {
  393. case CROP_MANUAL:
  394. LOAD_CROP_VAR(origin.x, "manual");
  395. LOAD_CROP_VAR(origin.y, "manual");
  396. LOAD_CROP_VAR(size.width, "manual");
  397. LOAD_CROP_VAR(size.height, "manual");
  398. break;
  399. case CROP_TO_WINDOW_AND_MANUAL:
  400. LOAD_CROP_VAR(origin.x, "window");
  401. LOAD_CROP_VAR(origin.y, "window");
  402. LOAD_CROP_VAR(size.width, "window");
  403. LOAD_CROP_VAR(size.height, "window");
  404. break;
  405. case CROP_NONE:
  406. case CROP_TO_WINDOW:
  407. case CROP_INVALID:
  408. break;
  409. }
  410. #undef LOAD_CROP_VAR
  411. }
  412. static void display_capture_update(void *data, obs_data_t *settings)
  413. {
  414. struct display_capture *dc = data;
  415. load_crop(dc, settings);
  416. if (requires_window(dc->crop))
  417. update_window(&dc->window, settings);
  418. CGDirectDisplayID display = get_display_migrate_settings(settings);
  419. bool show_cursor = obs_data_get_bool(settings, "show_cursor");
  420. if (dc->display == display && dc->hide_cursor != show_cursor)
  421. return;
  422. obs_enter_graphics();
  423. destroy_display_stream(dc);
  424. dc->display = display;
  425. dc->hide_cursor = !show_cursor;
  426. init_display_stream(dc);
  427. obs_leave_graphics();
  428. }
  429. static bool switch_crop_mode(obs_properties_t *props, obs_property_t *p, obs_data_t *settings)
  430. {
  431. UNUSED_PARAMETER(p);
  432. enum crop_mode crop;
  433. load_crop_mode(&crop, settings);
  434. const char *name;
  435. bool visible;
  436. #define LOAD_CROP_VAR(var, mode) \
  437. name = CROP_VAR_NAME(var, mode); \
  438. obs_property_set_visible(obs_properties_get(props, name), visible);
  439. visible = crop == CROP_MANUAL;
  440. LOAD_CROP_VAR(origin.x, "manual");
  441. LOAD_CROP_VAR(origin.y, "manual");
  442. LOAD_CROP_VAR(size.width, "manual");
  443. LOAD_CROP_VAR(size.height, "manual");
  444. visible = crop == CROP_TO_WINDOW_AND_MANUAL;
  445. LOAD_CROP_VAR(origin.x, "window");
  446. LOAD_CROP_VAR(origin.y, "window");
  447. LOAD_CROP_VAR(size.width, "window");
  448. LOAD_CROP_VAR(size.height, "window");
  449. #undef LOAD_CROP_VAR
  450. show_window_properties(props, visible || crop == CROP_TO_WINDOW);
  451. return true;
  452. }
  453. static const char *crop_names[] = {"CropMode.None", "CropMode.Manual", "CropMode.ToWindow",
  454. "CropMode.ToWindowAndManual"};
  455. #ifndef COUNTOF
  456. #define COUNTOF(x) (sizeof(x) / sizeof(x[0]))
  457. #endif
  458. static obs_properties_t *display_capture_properties(void *unused)
  459. {
  460. UNUSED_PARAMETER(unused);
  461. obs_properties_t *props = obs_properties_create();
  462. obs_property_t *list = obs_properties_add_list(props, "display_uuid", obs_module_text("DisplayCapture.Display"),
  463. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_STRING);
  464. [[NSScreen screens] enumerateObjectsUsingBlock:^(NSScreen *_Nonnull screen, NSUInteger index __unused,
  465. BOOL *_Nonnull stop __unused) {
  466. char dimension_buffer[4][12];
  467. char name_buffer[256];
  468. snprintf(dimension_buffer[0], sizeof(dimension_buffer[0]), "%u", (uint32_t)[screen frame].size.width);
  469. snprintf(dimension_buffer[1], sizeof(dimension_buffer[0]), "%u", (uint32_t)[screen frame].size.height);
  470. snprintf(dimension_buffer[2], sizeof(dimension_buffer[0]), "%d", (int32_t)[screen frame].origin.x);
  471. snprintf(dimension_buffer[3], sizeof(dimension_buffer[0]), "%d", (int32_t)[screen frame].origin.y);
  472. snprintf(name_buffer, sizeof(name_buffer), "%.200s: %.12sx%.12s @ %.12s,%.12s",
  473. [[screen localizedName] UTF8String], dimension_buffer[0], dimension_buffer[1], dimension_buffer[2],
  474. dimension_buffer[3]);
  475. NSNumber *screenNumber = screen.deviceDescription[@"NSScreenNumber"];
  476. CGDirectDisplayID display_id = (CGDirectDisplayID) screenNumber.intValue;
  477. CFUUIDRef display_uuid = CGDisplayCreateUUIDFromDisplayID(display_id);
  478. CFStringRef uuid_string = CFUUIDCreateString(kCFAllocatorDefault, display_uuid);
  479. obs_property_list_add_string(list, name_buffer, CFStringGetCStringPtr(uuid_string, kCFStringEncodingUTF8));
  480. CFRelease(uuid_string);
  481. CFRelease(display_uuid);
  482. }];
  483. obs_properties_add_bool(props, "show_cursor", obs_module_text("DisplayCapture.ShowCursor"));
  484. obs_property_t *crop = obs_properties_add_list(props, "crop_mode", obs_module_text("CropMode"), OBS_COMBO_TYPE_LIST,
  485. OBS_COMBO_FORMAT_INT);
  486. obs_property_set_modified_callback(crop, switch_crop_mode);
  487. for (unsigned i = 0; i < COUNTOF(crop_names); i++) {
  488. const char *name = obs_module_text(crop_names[i]);
  489. obs_property_list_add_int(crop, name, i);
  490. }
  491. add_window_properties(props);
  492. show_window_properties(props, false);
  493. obs_property_t *p;
  494. const char *name;
  495. float min;
  496. #define LOAD_CROP_VAR(var, mode) \
  497. name = CROP_VAR_NAME(var, mode); \
  498. p = obs_properties_add_float(props, name, obs_module_text("Crop." #var), min, 4096.f, .5f); \
  499. obs_property_set_visible(p, false);
  500. min = 0.f;
  501. LOAD_CROP_VAR(origin.x, "manual");
  502. LOAD_CROP_VAR(origin.y, "manual");
  503. LOAD_CROP_VAR(size.width, "manual");
  504. LOAD_CROP_VAR(size.height, "manual");
  505. min = -4096.f;
  506. LOAD_CROP_VAR(origin.x, "window");
  507. LOAD_CROP_VAR(origin.y, "window");
  508. LOAD_CROP_VAR(size.width, "window");
  509. LOAD_CROP_VAR(size.height, "window");
  510. #undef LOAD_CROP_VAR
  511. return props;
  512. }
  513. struct obs_source_info display_capture_info = {
  514. .id = "display_capture",
  515. .type = OBS_SOURCE_TYPE_INPUT,
  516. .get_name = display_capture_getname,
  517. .create = display_capture_create,
  518. .destroy = display_capture_destroy,
  519. .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_CUSTOM_DRAW | OBS_SOURCE_DO_NOT_DUPLICATE | OBS_SOURCE_SRGB,
  520. .video_tick = display_capture_video_tick,
  521. .video_render = display_capture_video_render,
  522. .get_width = display_capture_getwidth,
  523. .get_height = display_capture_getheight,
  524. .get_defaults = display_capture_defaults,
  525. .get_properties = display_capture_properties,
  526. .update = display_capture_update,
  527. .icon_type = OBS_ICON_TYPE_DESKTOP_CAPTURE,
  528. };