| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591 | 
							- #include "BasicOutputHandler.hpp"
 
- #include "AdvancedOutput.hpp"
 
- #include "SimpleOutput.hpp"
 
- #include <utility/MultitrackVideoError.hpp>
 
- #include <utility/StartMultiTrackVideoStreamingGuard.hpp>
 
- #include <utility/VCamConfig.hpp>
 
- #include <widgets/OBSBasic.hpp>
 
- #include <qt-wrappers.hpp>
 
- #include <QThreadPool>
 
- using namespace std;
 
- extern bool EncoderAvailable(const char *encoder);
 
- volatile bool streaming_active = false;
 
- volatile bool recording_active = false;
 
- volatile bool recording_paused = false;
 
- volatile bool replaybuf_active = false;
 
- volatile bool virtualcam_active = false;
 
- void OBSStreamStarting(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	obs_output_t *obj = (obs_output_t *)calldata_ptr(params, "output");
 
- 	int sec = (int)obs_output_get_active_delay(obj);
 
- 	if (sec == 0)
 
- 		return;
 
- 	output->delayActive = true;
 
- 	QMetaObject::invokeMethod(output->main, "StreamDelayStarting", Q_ARG(int, sec));
 
- }
 
- void OBSStreamStopping(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	obs_output_t *obj = (obs_output_t *)calldata_ptr(params, "output");
 
- 	int sec = (int)obs_output_get_active_delay(obj);
 
- 	if (sec == 0)
 
- 		QMetaObject::invokeMethod(output->main, "StreamStopping");
 
- 	else
 
- 		QMetaObject::invokeMethod(output->main, "StreamDelayStopping", Q_ARG(int, sec));
 
- }
 
- void OBSStartStreaming(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	output->streamingActive = true;
 
- 	os_atomic_set_bool(&streaming_active, true);
 
- 	QMetaObject::invokeMethod(output->main, "StreamingStart");
 
- }
 
- void OBSStopStreaming(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	int code = (int)calldata_int(params, "code");
 
- 	const char *last_error = calldata_string(params, "last_error");
 
- 	QString arg_last_error = QString::fromUtf8(last_error);
 
- 	output->streamingActive = false;
 
- 	output->delayActive = false;
 
- 	output->multitrackVideoActive = false;
 
- 	os_atomic_set_bool(&streaming_active, false);
 
- 	QMetaObject::invokeMethod(output->main, "StreamingStop", Q_ARG(int, code), Q_ARG(QString, arg_last_error));
 
- }
 
- void OBSStartRecording(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	output->recordingActive = true;
 
- 	os_atomic_set_bool(&recording_active, true);
 
- 	QMetaObject::invokeMethod(output->main, "RecordingStart");
 
- }
 
- void OBSStopRecording(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	int code = (int)calldata_int(params, "code");
 
- 	const char *last_error = calldata_string(params, "last_error");
 
- 	QString arg_last_error = QString::fromUtf8(last_error);
 
- 	output->recordingActive = false;
 
- 	os_atomic_set_bool(&recording_active, false);
 
- 	os_atomic_set_bool(&recording_paused, false);
 
- 	QMetaObject::invokeMethod(output->main, "RecordingStop", Q_ARG(int, code), Q_ARG(QString, arg_last_error));
 
- }
 
- void OBSRecordStopping(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	QMetaObject::invokeMethod(output->main, "RecordStopping");
 
- }
 
- void OBSRecordFileChanged(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	const char *next_file = calldata_string(params, "next_file");
 
- 	QString arg_last_file = QString::fromUtf8(output->lastRecordingPath.c_str());
 
- 	QMetaObject::invokeMethod(output->main, "RecordingFileChanged", Q_ARG(QString, arg_last_file));
 
- 	output->lastRecordingPath = next_file;
 
- }
 
- void OBSStartReplayBuffer(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	output->replayBufferActive = true;
 
- 	os_atomic_set_bool(&replaybuf_active, true);
 
- 	QMetaObject::invokeMethod(output->main, "ReplayBufferStart");
 
- }
 
- void OBSStopReplayBuffer(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	int code = (int)calldata_int(params, "code");
 
- 	output->replayBufferActive = false;
 
- 	os_atomic_set_bool(&replaybuf_active, false);
 
- 	QMetaObject::invokeMethod(output->main, "ReplayBufferStop", Q_ARG(int, code));
 
- }
 
- void OBSReplayBufferStopping(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	QMetaObject::invokeMethod(output->main, "ReplayBufferStopping");
 
- }
 
- void OBSReplayBufferSaved(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	QMetaObject::invokeMethod(output->main, "ReplayBufferSaved", Qt::QueuedConnection);
 
- }
 
- static void OBSStartVirtualCam(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	output->virtualCamActive = true;
 
- 	os_atomic_set_bool(&virtualcam_active, true);
 
- 	QMetaObject::invokeMethod(output->main, "OnVirtualCamStart");
 
- }
 
- static void OBSStopVirtualCam(void *data, calldata_t *params)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	int code = (int)calldata_int(params, "code");
 
- 	output->virtualCamActive = false;
 
- 	os_atomic_set_bool(&virtualcam_active, false);
 
- 	QMetaObject::invokeMethod(output->main, "OnVirtualCamStop", Q_ARG(int, code));
 
- }
 
- static void OBSDeactivateVirtualCam(void *data, calldata_t * /* params */)
 
- {
 
- 	BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
 
- 	output->DestroyVirtualCamView();
 
- }
 
- bool return_first_id(void *data, const char *id)
 
- {
 
- 	const char **output = (const char **)data;
 
- 	*output = id;
 
- 	return false;
 
- }
 
- const char *GetStreamOutputType(const obs_service_t *service)
 
- {
 
- 	const char *protocol = obs_service_get_protocol(service);
 
- 	const char *output = nullptr;
 
- 	if (!protocol) {
 
- 		blog(LOG_WARNING, "The service '%s' has no protocol set", obs_service_get_id(service));
 
- 		return nullptr;
 
- 	}
 
- 	if (!obs_is_output_protocol_registered(protocol)) {
 
- 		blog(LOG_WARNING, "The protocol '%s' is not registered", protocol);
 
- 		return nullptr;
 
- 	}
 
- 	/* Check if the service has a preferred output type */
 
- 	output = obs_service_get_preferred_output_type(service);
 
- 	if (output) {
 
- 		if ((obs_get_output_flags(output) & OBS_OUTPUT_SERVICE) != 0)
 
- 			return output;
 
- 		blog(LOG_WARNING, "The output '%s' is not registered, fallback to another one", output);
 
- 	}
 
- 	/* Otherwise, prefer first-party output types */
 
- 	if (can_use_output(protocol, "rtmp_output", "RTMP", "RTMPS")) {
 
- 		return "rtmp_output";
 
- 	} else if (can_use_output(protocol, "ffmpeg_hls_muxer", "HLS")) {
 
- 		return "ffmpeg_hls_muxer";
 
- 	} else if (can_use_output(protocol, "ffmpeg_mpegts_muxer", "SRT", "RIST")) {
 
- 		return "ffmpeg_mpegts_muxer";
 
- 	}
 
- 	/* If third-party protocol, use the first enumerated type */
 
- 	obs_enum_output_types_with_protocol(protocol, &output, return_first_id);
 
- 	if (output)
 
- 		return output;
 
- 	blog(LOG_WARNING, "No output compatible with the service '%s' is registered", obs_service_get_id(service));
 
- 	return nullptr;
 
- }
 
- BasicOutputHandler::BasicOutputHandler(OBSBasic *main_) : main(main_)
 
- {
 
- 	if (main->vcamEnabled) {
 
- 		virtualCam = obs_output_create(VIRTUAL_CAM_ID, "virtualcam_output", nullptr, nullptr);
 
- 		signal_handler_t *signal = obs_output_get_signal_handler(virtualCam);
 
- 		startVirtualCam.Connect(signal, "start", OBSStartVirtualCam, this);
 
- 		stopVirtualCam.Connect(signal, "stop", OBSStopVirtualCam, this);
 
- 		deactivateVirtualCam.Connect(signal, "deactivate", OBSDeactivateVirtualCam, this);
 
- 	}
 
- 	auto multitrack_enabled = config_get_bool(main->Config(), "Stream1", "EnableMultitrackVideo");
 
- 	if (!config_has_user_value(main->Config(), "Stream1", "EnableMultitrackVideo")) {
 
- 		auto service = main_->GetService();
 
- 		OBSDataAutoRelease settings = obs_service_get_settings(service);
 
- 		multitrack_enabled = obs_data_has_user_value(settings, "multitrack_video_configuration_url");
 
- 	}
 
- 	if (multitrack_enabled)
 
- 		multitrackVideo = make_unique<MultitrackVideoOutput>();
 
- }
 
- extern void log_vcam_changed(const VCamConfig &config, bool starting);
 
- bool BasicOutputHandler::StartVirtualCam()
 
- {
 
- 	if (!main->vcamEnabled)
 
- 		return false;
 
- 	bool typeIsProgram = main->vcamConfig.type == VCamOutputType::ProgramView;
 
- 	if (!virtualCamView && !typeIsProgram)
 
- 		virtualCamView = obs_view_create();
 
- 	UpdateVirtualCamOutputSource();
 
- 	if (!virtualCamVideo) {
 
- 		virtualCamVideo = typeIsProgram ? obs_get_video() : obs_view_add(virtualCamView);
 
- 		if (!virtualCamVideo)
 
- 			return false;
 
- 	}
 
- 	obs_output_set_media(virtualCam, virtualCamVideo, obs_get_audio());
 
- 	if (!Active())
 
- 		SetupOutputs();
 
- 	bool success = obs_output_start(virtualCam);
 
- 	if (!success) {
 
- 		QString errorReason;
 
- 		const char *error = obs_output_get_last_error(virtualCam);
 
- 		if (error) {
 
- 			errorReason = QT_UTF8(error);
 
- 		} else {
 
- 			errorReason = QTStr("Output.StartFailedGeneric");
 
- 		}
 
- 		QMessageBox::critical(main, QTStr("Output.StartVirtualCamFailed"), errorReason);
 
- 		DestroyVirtualCamView();
 
- 	}
 
- 	log_vcam_changed(main->vcamConfig, true);
 
- 	return success;
 
- }
 
- void BasicOutputHandler::StopVirtualCam()
 
- {
 
- 	if (main->vcamEnabled) {
 
- 		obs_output_stop(virtualCam);
 
- 	}
 
- }
 
- bool BasicOutputHandler::VirtualCamActive() const
 
- {
 
- 	if (main->vcamEnabled) {
 
- 		return obs_output_active(virtualCam);
 
- 	}
 
- 	return false;
 
- }
 
- void BasicOutputHandler::UpdateVirtualCamOutputSource()
 
- {
 
- 	if (!main->vcamEnabled || !virtualCamView)
 
- 		return;
 
- 	OBSSourceAutoRelease source;
 
- 	switch (main->vcamConfig.type) {
 
- 	case VCamOutputType::Invalid:
 
- 	case VCamOutputType::ProgramView:
 
- 		DestroyVirtualCameraScene();
 
- 		return;
 
- 	case VCamOutputType::PreviewOutput: {
 
- 		DestroyVirtualCameraScene();
 
- 		OBSSource s = main->GetCurrentSceneSource();
 
- 		obs_source_get_ref(s);
 
- 		source = s.Get();
 
- 		break;
 
- 	}
 
- 	case VCamOutputType::SceneOutput:
 
- 		DestroyVirtualCameraScene();
 
- 		source = obs_get_source_by_name(main->vcamConfig.scene.c_str());
 
- 		break;
 
- 	case VCamOutputType::SourceOutput:
 
- 		OBSSourceAutoRelease s = obs_get_source_by_name(main->vcamConfig.source.c_str());
 
- 		if (!vCamSourceScene)
 
- 			vCamSourceScene = obs_scene_create_private("vcam_source");
 
- 		source = obs_source_get_ref(obs_scene_get_source(vCamSourceScene));
 
- 		if (vCamSourceSceneItem && (obs_sceneitem_get_source(vCamSourceSceneItem) != s)) {
 
- 			obs_sceneitem_remove(vCamSourceSceneItem);
 
- 			vCamSourceSceneItem = nullptr;
 
- 		}
 
- 		if (!vCamSourceSceneItem) {
 
- 			vCamSourceSceneItem = obs_scene_add(vCamSourceScene, s);
 
- 			obs_sceneitem_set_bounds_type(vCamSourceSceneItem, OBS_BOUNDS_SCALE_INNER);
 
- 			obs_sceneitem_set_bounds_alignment(vCamSourceSceneItem, OBS_ALIGN_CENTER);
 
- 			const struct vec2 size = {
 
- 				(float)obs_source_get_width(source),
 
- 				(float)obs_source_get_height(source),
 
- 			};
 
- 			obs_sceneitem_set_bounds(vCamSourceSceneItem, &size);
 
- 		}
 
- 		break;
 
- 	}
 
- 	OBSSourceAutoRelease current = obs_view_get_source(virtualCamView, 0);
 
- 	if (source != current)
 
- 		obs_view_set_source(virtualCamView, 0, source);
 
- }
 
- void BasicOutputHandler::DestroyVirtualCamView()
 
- {
 
- 	if (main->vcamConfig.type == VCamOutputType::ProgramView) {
 
- 		virtualCamVideo = nullptr;
 
- 		return;
 
- 	}
 
- 	obs_view_remove(virtualCamView);
 
- 	obs_view_set_source(virtualCamView, 0, nullptr);
 
- 	virtualCamVideo = nullptr;
 
- 	obs_view_destroy(virtualCamView);
 
- 	virtualCamView = nullptr;
 
- 	DestroyVirtualCameraScene();
 
- }
 
- void BasicOutputHandler::DestroyVirtualCameraScene()
 
- {
 
- 	if (!vCamSourceScene)
 
- 		return;
 
- 	obs_scene_release(vCamSourceScene);
 
- 	vCamSourceScene = nullptr;
 
- 	vCamSourceSceneItem = nullptr;
 
- }
 
- const char *FindAudioEncoderFromCodec(const char *type)
 
- {
 
- 	const char *alt_enc_id = nullptr;
 
- 	size_t i = 0;
 
- 	while (obs_enum_encoder_types(i++, &alt_enc_id)) {
 
- 		const char *codec = obs_get_encoder_codec(alt_enc_id);
 
- 		if (strcmp(type, codec) == 0) {
 
- 			return alt_enc_id;
 
- 		}
 
- 	}
 
- 	return nullptr;
 
- }
 
- void clear_archive_encoder(obs_output_t *output, const char *expected_name)
 
- {
 
- 	obs_encoder_t *last = obs_output_get_audio_encoder(output, 1);
 
- 	bool clear = false;
 
- 	/* ensures that we don't remove twitch's soundtrack encoder */
 
- 	if (last) {
 
- 		const char *name = obs_encoder_get_name(last);
 
- 		clear = name && strcmp(name, expected_name) == 0;
 
- 		obs_encoder_release(last);
 
- 	}
 
- 	if (clear)
 
- 		obs_output_set_audio_encoder(output, nullptr, 1);
 
- }
 
- void BasicOutputHandler::SetupAutoRemux(const char *&container)
 
- {
 
- 	bool autoRemux = config_get_bool(main->Config(), "Video", "AutoRemux");
 
- 	if (autoRemux && strcmp(container, "mp4") == 0)
 
- 		container = "mkv";
 
- }
 
- std::string BasicOutputHandler::GetRecordingFilename(const char *path, const char *container, bool noSpace,
 
- 						     bool overwrite, const char *format, bool ffmpeg)
 
- {
 
- 	if (!ffmpeg)
 
- 		SetupAutoRemux(container);
 
- 	string dst = GetOutputFilename(path, container, noSpace, overwrite, format);
 
- 	lastRecordingPath = dst;
 
- 	return dst;
 
- }
 
- extern std::string DeserializeConfigText(const char *text);
 
- std::shared_future<void> BasicOutputHandler::SetupMultitrackVideo(obs_service_t *service, std::string audio_encoder_id,
 
- 								  size_t main_audio_mixer,
 
- 								  std::optional<size_t> vod_track_mixer,
 
- 								  std::function<void(std::optional<bool>)> continuation)
 
- {
 
- 	auto start_streaming_guard = std::make_shared<StartMultitrackVideoStreamingGuard>();
 
- 	if (!multitrackVideo) {
 
- 		continuation(std::nullopt);
 
- 		return start_streaming_guard->GetFuture();
 
- 	}
 
- 	multitrackVideoActive = false;
 
- 	streamDelayStarting.Disconnect();
 
- 	streamStopping.Disconnect();
 
- 	startStreaming.Disconnect();
 
- 	stopStreaming.Disconnect();
 
- 	bool is_custom = strncmp("rtmp_custom", obs_service_get_type(service), 11) == 0;
 
- 	std::optional<std::string> custom_config = std::nullopt;
 
- 	if (config_get_bool(main->Config(), "Stream1", "MultitrackVideoConfigOverrideEnabled"))
 
- 		custom_config = DeserializeConfigText(
 
- 			config_get_string(main->Config(), "Stream1", "MultitrackVideoConfigOverride"));
 
- 	std::optional<QString> extraCanvasUUID;
 
- 	const char *uuid = config_get_string(main->Config(), "Stream1", "MultitrackExtraCanvas");
 
- 	if (uuid && *uuid) {
 
- 		extraCanvasUUID = uuid;
 
- 	}
 
- 	OBSDataAutoRelease settings = obs_service_get_settings(service);
 
- 	QString key = obs_data_get_string(settings, "key");
 
- 	const char *service_name = "<unknown>";
 
- 	if (is_custom && obs_data_has_user_value(settings, "service_name")) {
 
- 		service_name = obs_data_get_string(settings, "service_name");
 
- 	} else if (!is_custom) {
 
- 		service_name = obs_data_get_string(settings, "service");
 
- 	}
 
- 	std::optional<std::string> custom_rtmp_url;
 
- 	std::optional<bool> use_rtmps;
 
- 	auto server = obs_data_get_string(settings, "server");
 
- 	if (strncmp(server, "auto", 4) != 0) {
 
- 		custom_rtmp_url = server;
 
- 	} else {
 
- 		QString server_ = server;
 
- 		use_rtmps = server_.contains("rtmps", Qt::CaseInsensitive);
 
- 	}
 
- 	auto service_custom_server = obs_data_get_bool(settings, "using_custom_server");
 
- 	if (custom_rtmp_url.has_value()) {
 
- 		blog(LOG_INFO, "Using %sserver '%s'", service_custom_server ? "custom " : "", custom_rtmp_url->c_str());
 
- 	}
 
- 	auto maximum_aggregate_bitrate =
 
- 		config_get_bool(main->Config(), "Stream1", "MultitrackVideoMaximumAggregateBitrateAuto")
 
- 			? std::nullopt
 
- 			: std::make_optional<uint32_t>(
 
- 				  config_get_int(main->Config(), "Stream1", "MultitrackVideoMaximumAggregateBitrate"));
 
- 	auto maximum_video_tracks = config_get_bool(main->Config(), "Stream1", "MultitrackVideoMaximumVideoTracksAuto")
 
- 					    ? std::nullopt
 
- 					    : std::make_optional<uint32_t>(config_get_int(
 
- 						      main->Config(), "Stream1", "MultitrackVideoMaximumVideoTracks"));
 
- 	auto stream_dump_config = GenerateMultitrackVideoStreamDumpConfig();
 
- 	auto continue_on_main_thread = [&, start_streaming_guard, service = OBSService{service},
 
- 					continuation =
 
- 						std::move(continuation)](std::optional<MultitrackVideoError> error) {
 
- 		if (error) {
 
- 			OBSDataAutoRelease service_settings = obs_service_get_settings(service);
 
- 			auto multitrack_video_name = QTStr("Basic.Settings.Stream.MultitrackVideoLabel");
 
- 			if (obs_data_has_user_value(service_settings, "multitrack_video_name")) {
 
- 				multitrack_video_name = obs_data_get_string(service_settings, "multitrack_video_name");
 
- 			}
 
- 			multitrackVideoActive = false;
 
- 			if (!error->ShowDialog(main, multitrack_video_name))
 
- 				return continuation(false);
 
- 			return continuation(std::nullopt);
 
- 		}
 
- 		multitrackVideoActive = true;
 
- 		auto signal_handler = multitrackVideo->StreamingSignalHandler();
 
- 		streamDelayStarting.Connect(signal_handler, "starting", OBSStreamStarting, this);
 
- 		streamStopping.Connect(signal_handler, "stopping", OBSStreamStopping, this);
 
- 		startStreaming.Connect(signal_handler, "start", OBSStartStreaming, this);
 
- 		stopStreaming.Connect(signal_handler, "stop", OBSStopStreaming, this);
 
- 		return continuation(true);
 
- 	};
 
- 	QThreadPool::globalInstance()->start([=, multitrackVideo = multitrackVideo.get(),
 
- 					      service_name = std::string{service_name}, service = OBSService{service},
 
- 					      stream_dump_config = OBSData{stream_dump_config},
 
- 					      start_streaming_guard = start_streaming_guard]() mutable {
 
- 		std::optional<MultitrackVideoError> error;
 
- 		try {
 
- 			multitrackVideo->PrepareStreaming(main, service_name.c_str(), service, custom_rtmp_url, key,
 
- 							  audio_encoder_id.c_str(), maximum_aggregate_bitrate,
 
- 							  maximum_video_tracks, custom_config, stream_dump_config,
 
- 							  main_audio_mixer, vod_track_mixer, use_rtmps,
 
- 							  extraCanvasUUID);
 
- 		} catch (const MultitrackVideoError &error_) {
 
- 			error.emplace(error_);
 
- 		}
 
- 		QMetaObject::invokeMethod(main, [=] { continue_on_main_thread(error); });
 
- 	});
 
- 	return start_streaming_guard->GetFuture();
 
- }
 
- OBSDataAutoRelease BasicOutputHandler::GenerateMultitrackVideoStreamDumpConfig()
 
- {
 
- 	auto stream_dump_enabled = config_get_bool(main->Config(), "Stream1", "MultitrackVideoStreamDumpEnabled");
 
- 	if (!stream_dump_enabled)
 
- 		return nullptr;
 
- 	const char *path = config_get_string(main->Config(), "SimpleOutput", "FilePath");
 
- 	bool noSpace = config_get_bool(main->Config(), "SimpleOutput", "FileNameWithoutSpace");
 
- 	const char *filenameFormat = config_get_string(main->Config(), "Output", "FilenameFormatting");
 
- 	bool overwriteIfExists = config_get_bool(main->Config(), "Output", "OverwriteIfExists");
 
- 	bool useMP4 = config_get_bool(main->Config(), "Stream1", "MultitrackVideoStreamDumpAsMP4");
 
- 	string f;
 
- 	OBSDataAutoRelease settings = obs_data_create();
 
- 	f = GetFormatString(filenameFormat, nullptr, nullptr);
 
- 	string strPath = GetRecordingFilename(path, useMP4 ? "mp4" : "flv", noSpace, overwriteIfExists, f.c_str(),
 
- 					      // never remux stream dump
 
- 					      false);
 
- 	obs_data_set_string(settings, "path", strPath.c_str());
 
- 	if (useMP4) {
 
- 		obs_data_set_bool(settings, "use_mp4", true);
 
- 		obs_data_set_string(settings, "muxer_settings", "write_encoder_info=1");
 
- 	}
 
- 	return settings;
 
- }
 
- BasicOutputHandler *CreateSimpleOutputHandler(OBSBasic *main)
 
- {
 
- 	return new SimpleOutput(main);
 
- }
 
- BasicOutputHandler *CreateAdvancedOutputHandler(OBSBasic *main)
 
- {
 
- 	return new AdvancedOutput(main);
 
- }
 
 
  |