| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235 | 
							- #include <util/dstr.h>
 
- #include "dc-capture.h"
 
- /* clang-format off */
 
- #define TEXT_MONITOR_CAPTURE obs_module_text("MonitorCapture")
 
- #define TEXT_CAPTURE_CURSOR  obs_module_text("CaptureCursor")
 
- #define TEXT_COMPATIBILITY   obs_module_text("Compatibility")
 
- #define TEXT_MONITOR         obs_module_text("Monitor")
 
- #define TEXT_PRIMARY_MONITOR obs_module_text("PrimaryMonitor")
 
- /* clang-format on */
 
- struct monitor_capture {
 
- 	obs_source_t *source;
 
- 	int monitor;
 
- 	bool capture_cursor;
 
- 	bool compatibility;
 
- 	struct dc_capture data;
 
- };
 
- struct monitor_info {
 
- 	int cur_id;
 
- 	int desired_id;
 
- 	int id;
 
- 	RECT rect;
 
- };
 
- /* ------------------------------------------------------------------------- */
 
- static inline void do_log(int level, const char *msg, ...)
 
- {
 
- 	va_list args;
 
- 	struct dstr str = {0};
 
- 	va_start(args, msg);
 
- 	dstr_copy(&str, "[GDI monitor capture]: ");
 
- 	dstr_vcatf(&str, msg, args);
 
- 	blog(level, "%s", str.array);
 
- 	dstr_free(&str);
 
- 	va_end(args);
 
- }
 
- static BOOL CALLBACK enum_monitor(HMONITOR handle, HDC hdc, LPRECT rect, LPARAM param)
 
- {
 
- 	struct monitor_info *monitor = (struct monitor_info *)param;
 
- 	if (monitor->cur_id == 0 || monitor->desired_id == monitor->cur_id) {
 
- 		monitor->rect = *rect;
 
- 		monitor->id = monitor->cur_id;
 
- 	}
 
- 	UNUSED_PARAMETER(hdc);
 
- 	UNUSED_PARAMETER(handle);
 
- 	return (monitor->desired_id > monitor->cur_id++);
 
- }
 
- static void update_monitor(struct monitor_capture *capture, obs_data_t *settings)
 
- {
 
- 	struct monitor_info monitor = {0};
 
- 	uint32_t width, height;
 
- 	monitor.desired_id = (int)obs_data_get_int(settings, "monitor");
 
- 	EnumDisplayMonitors(NULL, NULL, enum_monitor, (LPARAM)&monitor);
 
- 	capture->monitor = monitor.id;
 
- 	width = monitor.rect.right - monitor.rect.left;
 
- 	height = monitor.rect.bottom - monitor.rect.top;
 
- 	dc_capture_init(&capture->data, monitor.rect.left, monitor.rect.top, width, height, capture->capture_cursor,
 
- 			capture->compatibility);
 
- }
 
- static inline void update_settings(struct monitor_capture *capture, obs_data_t *settings)
 
- {
 
- 	capture->monitor = (int)obs_data_get_int(settings, "monitor");
 
- 	capture->capture_cursor = obs_data_get_bool(settings, "capture_cursor");
 
- 	capture->compatibility = obs_data_get_bool(settings, "compatibility");
 
- 	dc_capture_free(&capture->data);
 
- 	update_monitor(capture, settings);
 
- }
 
- /* ------------------------------------------------------------------------- */
 
- static const char *monitor_capture_getname(void *unused)
 
- {
 
- 	UNUSED_PARAMETER(unused);
 
- 	return TEXT_MONITOR_CAPTURE;
 
- }
 
- static void monitor_capture_destroy(void *data)
 
- {
 
- 	struct monitor_capture *capture = data;
 
- 	obs_enter_graphics();
 
- 	dc_capture_free(&capture->data);
 
- 	obs_leave_graphics();
 
- 	bfree(capture);
 
- }
 
- static void monitor_capture_defaults(obs_data_t *settings)
 
- {
 
- 	obs_data_set_default_int(settings, "monitor", 0);
 
- 	obs_data_set_default_bool(settings, "capture_cursor", true);
 
- 	obs_data_set_default_bool(settings, "compatibility", false);
 
- }
 
- static void monitor_capture_update(void *data, obs_data_t *settings)
 
- {
 
- 	struct monitor_capture *mc = data;
 
- 	update_settings(mc, settings);
 
- }
 
- static void *monitor_capture_create(obs_data_t *settings, obs_source_t *source)
 
- {
 
- 	struct monitor_capture *capture;
 
- 	capture = bzalloc(sizeof(struct monitor_capture));
 
- 	capture->source = source;
 
- 	update_settings(capture, settings);
 
- 	return capture;
 
- }
 
- static void monitor_capture_tick(void *data, float seconds)
 
- {
 
- 	struct monitor_capture *capture = data;
 
- 	if (!obs_source_showing(capture->source))
 
- 		return;
 
- 	obs_enter_graphics();
 
- 	dc_capture_capture(&capture->data, NULL);
 
- 	obs_leave_graphics();
 
- 	UNUSED_PARAMETER(seconds);
 
- }
 
- static void monitor_capture_render(void *data, gs_effect_t *effect)
 
- {
 
- 	struct monitor_capture *capture = data;
 
- 	dc_capture_render(&capture->data, obs_source_get_texcoords_centered(capture->source));
 
- 	UNUSED_PARAMETER(effect);
 
- }
 
- static uint32_t monitor_capture_width(void *data)
 
- {
 
- 	struct monitor_capture *capture = data;
 
- 	return capture->data.width;
 
- }
 
- static uint32_t monitor_capture_height(void *data)
 
- {
 
- 	struct monitor_capture *capture = data;
 
- 	return capture->data.height;
 
- }
 
- static BOOL CALLBACK enum_monitor_props(HMONITOR handle, HDC hdc, LPRECT rect, LPARAM param)
 
- {
 
- 	UNUSED_PARAMETER(hdc);
 
- 	UNUSED_PARAMETER(rect);
 
- 	obs_property_t *monitor_list = (obs_property_t *)param;
 
- 	MONITORINFO mi;
 
- 	size_t monitor_id = 0;
 
- 	struct dstr monitor_desc = {0};
 
- 	struct dstr resolution = {0};
 
- 	struct dstr format_string = {0};
 
- 	monitor_id = obs_property_list_item_count(monitor_list);
 
- 	mi.cbSize = sizeof(mi);
 
- 	GetMonitorInfo(handle, &mi);
 
- 	dstr_catf(&resolution, "%dx%d @ %d,%d", mi.rcMonitor.right - mi.rcMonitor.left,
 
- 		  mi.rcMonitor.bottom - mi.rcMonitor.top, mi.rcMonitor.left, mi.rcMonitor.top);
 
- 	dstr_copy(&format_string, "%s %d: %s");
 
- 	if (mi.dwFlags == MONITORINFOF_PRIMARY) {
 
- 		dstr_catf(&format_string, " (%s)", TEXT_PRIMARY_MONITOR);
 
- 	}
 
- 	dstr_catf(&monitor_desc, format_string.array, TEXT_MONITOR, monitor_id + 1, resolution.array);
 
- 	obs_property_list_add_int(monitor_list, monitor_desc.array, (int)monitor_id);
 
- 	dstr_free(&monitor_desc);
 
- 	dstr_free(&resolution);
 
- 	dstr_free(&format_string);
 
- 	return TRUE;
 
- }
 
- static obs_properties_t *monitor_capture_properties(void *unused)
 
- {
 
- 	UNUSED_PARAMETER(unused);
 
- 	obs_properties_t *props = obs_properties_create();
 
- 	obs_property_t *monitors =
 
- 		obs_properties_add_list(props, "monitor", TEXT_MONITOR, OBS_COMBO_TYPE_LIST, OBS_COMBO_FORMAT_INT);
 
- 	obs_properties_add_bool(props, "compatibility", TEXT_COMPATIBILITY);
 
- 	obs_properties_add_bool(props, "capture_cursor", TEXT_CAPTURE_CURSOR);
 
- 	EnumDisplayMonitors(NULL, NULL, enum_monitor_props, (LPARAM)monitors);
 
- 	return props;
 
- }
 
- struct obs_source_info monitor_capture_info = {
 
- 	.id = "monitor_capture",
 
- 	.type = OBS_SOURCE_TYPE_INPUT,
 
- 	.output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_CUSTOM_DRAW | OBS_SOURCE_DO_NOT_DUPLICATE | OBS_SOURCE_SRGB,
 
- 	.get_name = monitor_capture_getname,
 
- 	.create = monitor_capture_create,
 
- 	.destroy = monitor_capture_destroy,
 
- 	.video_render = monitor_capture_render,
 
- 	.video_tick = monitor_capture_tick,
 
- 	.update = monitor_capture_update,
 
- 	.get_width = monitor_capture_width,
 
- 	.get_height = monitor_capture_height,
 
- 	.get_defaults = monitor_capture_defaults,
 
- 	.get_properties = monitor_capture_properties,
 
- 	.icon_type = OBS_ICON_TYPE_DESKTOP_CAPTURE,
 
- };
 
 
  |