monitor-capture.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. #include <util/dstr.h>
  2. #include "dc-capture.h"
  3. #define TEXT_MONITOR_CAPTURE obs_module_text("MonitorCapture")
  4. #define TEXT_CAPTURE_CURSOR obs_module_text("CaptureCursor")
  5. #define TEXT_COMPATIBILITY obs_module_text("Compatibility")
  6. #define TEXT_MONITOR obs_module_text("Monitor")
  7. #define TEXT_PRIMARY_MONITOR obs_module_text("PrimaryMonitor")
  8. struct monitor_capture {
  9. obs_source_t *source;
  10. int monitor;
  11. bool capture_cursor;
  12. bool compatibility;
  13. struct dc_capture data;
  14. gs_effect_t *opaque_effect;
  15. };
  16. struct monitor_info {
  17. int cur_id;
  18. int desired_id;
  19. int id;
  20. RECT rect;
  21. };
  22. /* ------------------------------------------------------------------------- */
  23. static inline void do_log(int level, const char *msg, ...)
  24. {
  25. va_list args;
  26. struct dstr str = {0};
  27. va_start(args, msg);
  28. dstr_copy(&str, "[GDI monitor capture]: ");
  29. dstr_vcatf(&str, msg, args);
  30. blog(level, "%s", str.array);
  31. dstr_free(&str);
  32. va_end(args);
  33. }
  34. static BOOL CALLBACK enum_monitor(HMONITOR handle, HDC hdc, LPRECT rect,
  35. LPARAM param)
  36. {
  37. struct monitor_info *monitor = (struct monitor_info *)param;
  38. if (monitor->cur_id == 0 || monitor->desired_id == monitor->cur_id) {
  39. monitor->rect = *rect;
  40. monitor->id = monitor->cur_id;
  41. }
  42. UNUSED_PARAMETER(hdc);
  43. UNUSED_PARAMETER(handle);
  44. return (monitor->desired_id > monitor->cur_id++);
  45. }
  46. static void update_monitor(struct monitor_capture *capture,
  47. obs_data_t *settings)
  48. {
  49. struct monitor_info monitor = {0};
  50. uint32_t width, height;
  51. monitor.desired_id = (int)obs_data_get_int(settings, "monitor");
  52. EnumDisplayMonitors(NULL, NULL, enum_monitor, (LPARAM)&monitor);
  53. capture->monitor = monitor.id;
  54. width = monitor.rect.right - monitor.rect.left;
  55. height = monitor.rect.bottom - monitor.rect.top;
  56. dc_capture_init(&capture->data, monitor.rect.left, monitor.rect.top,
  57. width, height, capture->capture_cursor,
  58. capture->compatibility);
  59. }
  60. static inline void update_settings(struct monitor_capture *capture,
  61. obs_data_t *settings)
  62. {
  63. capture->monitor = (int)obs_data_get_int(settings, "monitor");
  64. capture->capture_cursor = obs_data_get_bool(settings, "capture_cursor");
  65. capture->compatibility = obs_data_get_bool(settings, "compatibility");
  66. dc_capture_free(&capture->data);
  67. update_monitor(capture, settings);
  68. }
  69. /* ------------------------------------------------------------------------- */
  70. static const char *monitor_capture_getname(void)
  71. {
  72. return TEXT_MONITOR_CAPTURE;
  73. }
  74. static void monitor_capture_destroy(void *data)
  75. {
  76. struct monitor_capture *capture = data;
  77. obs_enter_graphics();
  78. dc_capture_free(&capture->data);
  79. gs_effect_destroy(capture->opaque_effect);
  80. obs_leave_graphics();
  81. bfree(capture);
  82. }
  83. static void monitor_capture_defaults(obs_data_t *settings)
  84. {
  85. obs_data_set_default_int(settings, "monitor", 0);
  86. obs_data_set_default_bool(settings, "capture_cursor", true);
  87. obs_data_set_default_bool(settings, "compatibility", false);
  88. }
  89. static void monitor_capture_update(void *data, obs_data_t *settings)
  90. {
  91. struct monitor_capture *mc = data;
  92. update_settings(mc, settings);
  93. }
  94. static void *monitor_capture_create(obs_data_t *settings, obs_source_t *source)
  95. {
  96. struct monitor_capture *capture;
  97. gs_effect_t *opaque_effect = create_opaque_effect();
  98. if (!opaque_effect)
  99. return NULL;
  100. capture = bzalloc(sizeof(struct monitor_capture));
  101. capture->opaque_effect = opaque_effect;
  102. update_settings(capture, settings);
  103. UNUSED_PARAMETER(source);
  104. return capture;
  105. }
  106. static void monitor_capture_tick(void *data, float seconds)
  107. {
  108. struct monitor_capture *capture = data;
  109. obs_enter_graphics();
  110. dc_capture_capture(&capture->data, NULL);
  111. obs_leave_graphics();
  112. UNUSED_PARAMETER(seconds);
  113. }
  114. static void monitor_capture_render(void *data, gs_effect_t *effect)
  115. {
  116. struct monitor_capture *capture = data;
  117. dc_capture_render(&capture->data, capture->opaque_effect);
  118. UNUSED_PARAMETER(effect);
  119. }
  120. static uint32_t monitor_capture_width(void *data)
  121. {
  122. struct monitor_capture *capture = data;
  123. return capture->data.width;
  124. }
  125. static uint32_t monitor_capture_height(void *data)
  126. {
  127. struct monitor_capture *capture = data;
  128. return capture->data.height;
  129. }
  130. static BOOL CALLBACK enum_monitor_props(HMONITOR handle, HDC hdc, LPRECT rect,
  131. LPARAM param)
  132. {
  133. UNUSED_PARAMETER(hdc);
  134. UNUSED_PARAMETER(rect);
  135. obs_property_t *monitor_list = (obs_property_t*)param;
  136. MONITORINFO mi;
  137. size_t monitor_id = 0;
  138. struct dstr monitor_desc = { 0 };
  139. struct dstr resolution = { 0 };
  140. struct dstr format_string = { 0 };
  141. monitor_id = obs_property_list_item_count(monitor_list);
  142. mi.cbSize = sizeof(mi);
  143. GetMonitorInfo(handle, &mi);
  144. dstr_catf(&resolution,
  145. "%dx%d @ %d,%d",
  146. mi.rcMonitor.right - mi.rcMonitor.left,
  147. mi.rcMonitor.bottom - mi.rcMonitor.top,
  148. mi.rcMonitor.left,
  149. mi.rcMonitor.top);
  150. dstr_copy(&format_string, "%s %d: %s");
  151. if (mi.dwFlags == MONITORINFOF_PRIMARY) {
  152. dstr_catf(&format_string, " (%s)", TEXT_PRIMARY_MONITOR);
  153. }
  154. dstr_catf(&monitor_desc,
  155. format_string.array,
  156. TEXT_MONITOR,
  157. monitor_id,
  158. resolution.array);
  159. obs_property_list_add_int(monitor_list,
  160. monitor_desc.array, (int)monitor_id);
  161. dstr_free(&monitor_desc);
  162. dstr_free(&resolution);
  163. dstr_free(&format_string);
  164. return TRUE;
  165. }
  166. static obs_properties_t *monitor_capture_properties(void *unused)
  167. {
  168. UNUSED_PARAMETER(unused);
  169. obs_properties_t *props = obs_properties_create();
  170. obs_property_t *monitors = obs_properties_add_list(props,
  171. "monitor", TEXT_MONITOR,
  172. OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
  173. obs_property_t *compatmode = obs_properties_add_bool(props,
  174. "compatibility", TEXT_COMPATIBILITY);
  175. obs_property_t *capture_cursor = obs_properties_add_bool(props,
  176. "capture_cursor", TEXT_CAPTURE_CURSOR);
  177. EnumDisplayMonitors(NULL, NULL, enum_monitor_props, (LPARAM)monitors);
  178. return props;
  179. }
  180. struct obs_source_info monitor_capture_info = {
  181. .id = "monitor_capture",
  182. .type = OBS_SOURCE_TYPE_INPUT,
  183. .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_CUSTOM_DRAW,
  184. .get_name = monitor_capture_getname,
  185. .create = monitor_capture_create,
  186. .destroy = monitor_capture_destroy,
  187. .video_render = monitor_capture_render,
  188. .video_tick = monitor_capture_tick,
  189. .update = monitor_capture_update,
  190. .get_width = monitor_capture_width,
  191. .get_height = monitor_capture_height,
  192. .get_defaults = monitor_capture_defaults,
  193. .get_properties = monitor_capture_properties
  194. };