mac-window-capture.m 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. #include <obs-module.h>
  2. #include <util/darray.h>
  3. #include <util/threading.h>
  4. #include <util/platform.h>
  5. #import <CoreGraphics/CGWindow.h>
  6. #import <Cocoa/Cocoa.h>
  7. #include "window-utils.h"
  8. struct window_capture {
  9. obs_source_t *source;
  10. struct cocoa_window window;
  11. //CGRect bounds;
  12. //CGWindowListOption window_option;
  13. CGWindowImageOption image_option;
  14. CGColorSpaceRef color_space;
  15. DARRAY(uint8_t) buffer;
  16. pthread_t capture_thread;
  17. os_event_t *capture_event;
  18. os_event_t *stop_event;
  19. };
  20. static CGImageRef get_image(struct window_capture *wc)
  21. {
  22. NSArray *arr = (NSArray *)CGWindowListCreate(
  23. kCGWindowListOptionIncludingWindow, wc->window.window_id);
  24. [arr autorelease];
  25. if (!arr.count && !find_window(&wc->window, NULL, false))
  26. return NULL;
  27. return CGWindowListCreateImage(CGRectNull,
  28. kCGWindowListOptionIncludingWindow,
  29. wc->window.window_id, wc->image_option);
  30. }
  31. static inline void capture_frame(struct window_capture *wc)
  32. {
  33. uint64_t ts = os_gettime_ns();
  34. CGImageRef img = get_image(wc);
  35. if (!img)
  36. return;
  37. size_t width = CGImageGetWidth(img);
  38. size_t height = CGImageGetHeight(img);
  39. CGRect rect = {{0, 0}, {width, height}};
  40. da_reserve(wc->buffer, width * height * 4);
  41. uint8_t *data = wc->buffer.array;
  42. CGContextRef cg_context = CGBitmapContextCreate(
  43. data, width, height, 8, width * 4, wc->color_space,
  44. kCGBitmapByteOrder32Host | kCGImageAlphaPremultipliedFirst);
  45. CGContextSetBlendMode(cg_context, kCGBlendModeCopy);
  46. CGContextDrawImage(cg_context, rect, img);
  47. CGContextRelease(cg_context);
  48. CGImageRelease(img);
  49. struct obs_source_frame frame = {
  50. .format = VIDEO_FORMAT_BGRA,
  51. .width = width,
  52. .height = height,
  53. .data[0] = data,
  54. .linesize[0] = width * 4,
  55. .timestamp = ts,
  56. };
  57. obs_source_output_video(wc->source, &frame);
  58. }
  59. static void *capture_thread(void *data)
  60. {
  61. struct window_capture *wc = data;
  62. for (;;) {
  63. os_event_wait(wc->capture_event);
  64. if (os_event_try(wc->stop_event) != EAGAIN)
  65. break;
  66. @autoreleasepool {
  67. capture_frame(wc);
  68. }
  69. }
  70. return NULL;
  71. }
  72. static inline void *window_capture_create_internal(obs_data_t *settings,
  73. obs_source_t *source)
  74. {
  75. struct window_capture *wc = bzalloc(sizeof(struct window_capture));
  76. wc->source = source;
  77. wc->color_space = CGColorSpaceCreateDeviceRGB();
  78. da_init(wc->buffer);
  79. init_window(&wc->window, settings);
  80. wc->image_option = obs_data_get_bool(settings, "show_shadow")
  81. ? kCGWindowImageDefault
  82. : kCGWindowImageBoundsIgnoreFraming;
  83. os_event_init(&wc->capture_event, OS_EVENT_TYPE_AUTO);
  84. os_event_init(&wc->stop_event, OS_EVENT_TYPE_MANUAL);
  85. pthread_create(&wc->capture_thread, NULL, capture_thread, wc);
  86. return wc;
  87. }
  88. static void *window_capture_create(obs_data_t *settings, obs_source_t *source)
  89. {
  90. @autoreleasepool {
  91. return window_capture_create_internal(settings, source);
  92. }
  93. }
  94. static void window_capture_destroy(void *data)
  95. {
  96. struct window_capture *cap = data;
  97. os_event_signal(cap->stop_event);
  98. os_event_signal(cap->capture_event);
  99. pthread_join(cap->capture_thread, NULL);
  100. CGColorSpaceRelease(cap->color_space);
  101. da_free(cap->buffer);
  102. os_event_destroy(cap->capture_event);
  103. os_event_destroy(cap->stop_event);
  104. destroy_window(&cap->window);
  105. bfree(cap);
  106. }
  107. static void window_capture_defaults(obs_data_t *settings)
  108. {
  109. obs_data_set_default_bool(settings, "show_shadow", false);
  110. window_defaults(settings);
  111. }
  112. static obs_properties_t *window_capture_properties(void *unused)
  113. {
  114. UNUSED_PARAMETER(unused);
  115. obs_properties_t *props = obs_properties_create();
  116. add_window_properties(props);
  117. obs_properties_add_bool(props, "show_shadow",
  118. obs_module_text("WindowCapture.ShowShadow"));
  119. return props;
  120. }
  121. static inline void window_capture_update_internal(struct window_capture *wc,
  122. obs_data_t *settings)
  123. {
  124. wc->image_option = obs_data_get_bool(settings, "show_shadow")
  125. ? kCGWindowImageDefault
  126. : kCGWindowImageBoundsIgnoreFraming;
  127. update_window(&wc->window, settings);
  128. if (wc->window.window_name.length) {
  129. blog(LOG_INFO,
  130. "[window-capture: '%s'] update settings:\n"
  131. "\twindow: %s\n"
  132. "\towner: %s",
  133. obs_source_get_name(wc->source),
  134. [wc->window.window_name UTF8String],
  135. [wc->window.owner_name UTF8String]);
  136. }
  137. }
  138. static void window_capture_update(void *data, obs_data_t *settings)
  139. {
  140. @autoreleasepool {
  141. return window_capture_update_internal(data, settings);
  142. }
  143. }
  144. static const char *window_capture_getname(void *unused)
  145. {
  146. UNUSED_PARAMETER(unused);
  147. return obs_module_text("WindowCapture");
  148. }
  149. static inline void window_capture_tick_internal(struct window_capture *wc,
  150. float seconds)
  151. {
  152. UNUSED_PARAMETER(seconds);
  153. os_event_signal(wc->capture_event);
  154. }
  155. static void window_capture_tick(void *data, float seconds)
  156. {
  157. struct window_capture *wc = data;
  158. if (!obs_source_showing(wc->source))
  159. return;
  160. @autoreleasepool {
  161. return window_capture_tick_internal(data, seconds);
  162. }
  163. }
  164. struct obs_source_info window_capture_info = {
  165. .id = "window_capture",
  166. .type = OBS_SOURCE_TYPE_INPUT,
  167. .get_name = window_capture_getname,
  168. .create = window_capture_create,
  169. .destroy = window_capture_destroy,
  170. .output_flags = OBS_SOURCE_ASYNC_VIDEO,
  171. .video_tick = window_capture_tick,
  172. .get_defaults = window_capture_defaults,
  173. .get_properties = window_capture_properties,
  174. .update = window_capture_update,
  175. .icon_type = OBS_ICON_TYPE_WINDOW_CAPTURE,
  176. };