1
0

BasicOutputHandler.cpp 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590
  1. #include "BasicOutputHandler.hpp"
  2. #include "AdvancedOutput.hpp"
  3. #include "SimpleOutput.hpp"
  4. #include <utility/MultitrackVideoError.hpp>
  5. #include <utility/StartMultiTrackVideoStreamingGuard.hpp>
  6. #include <utility/VCamConfig.hpp>
  7. #include <widgets/OBSBasic.hpp>
  8. #include <qt-wrappers.hpp>
  9. #include <QThreadPool>
  10. using namespace std;
  11. extern bool EncoderAvailable(const char *encoder);
  12. volatile bool streaming_active = false;
  13. volatile bool recording_active = false;
  14. volatile bool recording_paused = false;
  15. volatile bool replaybuf_active = false;
  16. volatile bool virtualcam_active = false;
  17. void OBSStreamStarting(void *data, calldata_t *params)
  18. {
  19. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  20. obs_output_t *obj = (obs_output_t *)calldata_ptr(params, "output");
  21. int sec = (int)obs_output_get_active_delay(obj);
  22. if (sec == 0)
  23. return;
  24. output->delayActive = true;
  25. QMetaObject::invokeMethod(output->main, "StreamDelayStarting", Q_ARG(int, sec));
  26. }
  27. void OBSStreamStopping(void *data, calldata_t *params)
  28. {
  29. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  30. obs_output_t *obj = (obs_output_t *)calldata_ptr(params, "output");
  31. int sec = (int)obs_output_get_active_delay(obj);
  32. if (sec == 0)
  33. QMetaObject::invokeMethod(output->main, "StreamStopping");
  34. else
  35. QMetaObject::invokeMethod(output->main, "StreamDelayStopping", Q_ARG(int, sec));
  36. }
  37. void OBSStartStreaming(void *data, calldata_t * /* params */)
  38. {
  39. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  40. output->streamingActive = true;
  41. os_atomic_set_bool(&streaming_active, true);
  42. QMetaObject::invokeMethod(output->main, "StreamingStart");
  43. }
  44. void OBSStopStreaming(void *data, calldata_t *params)
  45. {
  46. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  47. int code = (int)calldata_int(params, "code");
  48. const char *last_error = calldata_string(params, "last_error");
  49. QString arg_last_error = QString::fromUtf8(last_error);
  50. output->streamingActive = false;
  51. output->delayActive = false;
  52. output->multitrackVideoActive = false;
  53. os_atomic_set_bool(&streaming_active, false);
  54. QMetaObject::invokeMethod(output->main, "StreamingStop", Q_ARG(int, code), Q_ARG(QString, arg_last_error));
  55. }
  56. void OBSStartRecording(void *data, calldata_t * /* params */)
  57. {
  58. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  59. output->recordingActive = true;
  60. os_atomic_set_bool(&recording_active, true);
  61. QMetaObject::invokeMethod(output->main, "RecordingStart");
  62. }
  63. void OBSStopRecording(void *data, calldata_t *params)
  64. {
  65. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  66. int code = (int)calldata_int(params, "code");
  67. const char *last_error = calldata_string(params, "last_error");
  68. QString arg_last_error = QString::fromUtf8(last_error);
  69. output->recordingActive = false;
  70. os_atomic_set_bool(&recording_active, false);
  71. os_atomic_set_bool(&recording_paused, false);
  72. QMetaObject::invokeMethod(output->main, "RecordingStop", Q_ARG(int, code), Q_ARG(QString, arg_last_error));
  73. }
  74. void OBSRecordStopping(void *data, calldata_t * /* params */)
  75. {
  76. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  77. QMetaObject::invokeMethod(output->main, "RecordStopping");
  78. }
  79. void OBSRecordFileChanged(void *data, calldata_t *params)
  80. {
  81. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  82. const char *next_file = calldata_string(params, "next_file");
  83. QString arg_last_file = QString::fromUtf8(output->lastRecordingPath.c_str());
  84. QMetaObject::invokeMethod(output->main, "RecordingFileChanged", Q_ARG(QString, arg_last_file));
  85. output->lastRecordingPath = next_file;
  86. }
  87. void OBSStartReplayBuffer(void *data, calldata_t * /* params */)
  88. {
  89. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  90. output->replayBufferActive = true;
  91. os_atomic_set_bool(&replaybuf_active, true);
  92. QMetaObject::invokeMethod(output->main, "ReplayBufferStart");
  93. }
  94. void OBSStopReplayBuffer(void *data, calldata_t *params)
  95. {
  96. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  97. int code = (int)calldata_int(params, "code");
  98. output->replayBufferActive = false;
  99. os_atomic_set_bool(&replaybuf_active, false);
  100. QMetaObject::invokeMethod(output->main, "ReplayBufferStop", Q_ARG(int, code));
  101. }
  102. void OBSReplayBufferStopping(void *data, calldata_t * /* params */)
  103. {
  104. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  105. QMetaObject::invokeMethod(output->main, "ReplayBufferStopping");
  106. }
  107. void OBSReplayBufferSaved(void *data, calldata_t * /* params */)
  108. {
  109. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  110. QMetaObject::invokeMethod(output->main, "ReplayBufferSaved", Qt::QueuedConnection);
  111. }
  112. static void OBSStartVirtualCam(void *data, calldata_t * /* params */)
  113. {
  114. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  115. output->virtualCamActive = true;
  116. os_atomic_set_bool(&virtualcam_active, true);
  117. QMetaObject::invokeMethod(output->main, "OnVirtualCamStart");
  118. }
  119. static void OBSStopVirtualCam(void *data, calldata_t *params)
  120. {
  121. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  122. int code = (int)calldata_int(params, "code");
  123. output->virtualCamActive = false;
  124. os_atomic_set_bool(&virtualcam_active, false);
  125. QMetaObject::invokeMethod(output->main, "OnVirtualCamStop", Q_ARG(int, code));
  126. }
  127. static void OBSDeactivateVirtualCam(void *data, calldata_t * /* params */)
  128. {
  129. BasicOutputHandler *output = static_cast<BasicOutputHandler *>(data);
  130. output->DestroyVirtualCamView();
  131. }
  132. bool return_first_id(void *data, const char *id)
  133. {
  134. const char **output = (const char **)data;
  135. *output = id;
  136. return false;
  137. }
  138. const char *GetStreamOutputType(const obs_service_t *service)
  139. {
  140. const char *protocol = obs_service_get_protocol(service);
  141. const char *output = nullptr;
  142. if (!protocol) {
  143. blog(LOG_WARNING, "The service '%s' has no protocol set", obs_service_get_id(service));
  144. return nullptr;
  145. }
  146. if (!obs_is_output_protocol_registered(protocol)) {
  147. blog(LOG_WARNING, "The protocol '%s' is not registered", protocol);
  148. return nullptr;
  149. }
  150. /* Check if the service has a preferred output type */
  151. output = obs_service_get_preferred_output_type(service);
  152. if (output) {
  153. if ((obs_get_output_flags(output) & OBS_OUTPUT_SERVICE) != 0)
  154. return output;
  155. blog(LOG_WARNING, "The output '%s' is not registered, fallback to another one", output);
  156. }
  157. /* Otherwise, prefer first-party output types */
  158. if (can_use_output(protocol, "rtmp_output", "RTMP", "RTMPS")) {
  159. return "rtmp_output";
  160. } else if (can_use_output(protocol, "ffmpeg_hls_muxer", "HLS")) {
  161. return "ffmpeg_hls_muxer";
  162. } else if (can_use_output(protocol, "ffmpeg_mpegts_muxer", "SRT", "RIST")) {
  163. return "ffmpeg_mpegts_muxer";
  164. }
  165. /* If third-party protocol, use the first enumerated type */
  166. obs_enum_output_types_with_protocol(protocol, &output, return_first_id);
  167. if (output)
  168. return output;
  169. blog(LOG_WARNING, "No output compatible with the service '%s' is registered", obs_service_get_id(service));
  170. return nullptr;
  171. }
  172. BasicOutputHandler::BasicOutputHandler(OBSBasic *main_) : main(main_)
  173. {
  174. if (main->vcamEnabled) {
  175. virtualCam = obs_output_create(VIRTUAL_CAM_ID, "virtualcam_output", nullptr, nullptr);
  176. signal_handler_t *signal = obs_output_get_signal_handler(virtualCam);
  177. startVirtualCam.Connect(signal, "start", OBSStartVirtualCam, this);
  178. stopVirtualCam.Connect(signal, "stop", OBSStopVirtualCam, this);
  179. deactivateVirtualCam.Connect(signal, "deactivate", OBSDeactivateVirtualCam, this);
  180. }
  181. auto service = main_->GetService();
  182. OBSDataAutoRelease settings = obs_service_get_settings(service);
  183. auto multitrack_enabled = config_get_bool(main->Config(), "Stream1", "EnableMultitrackVideo") &&
  184. obs_data_has_user_value(settings, "multitrack_video_configuration_url");
  185. if (multitrack_enabled)
  186. multitrackVideo = make_unique<MultitrackVideoOutput>();
  187. }
  188. extern void log_vcam_changed(const VCamConfig &config, bool starting);
  189. bool BasicOutputHandler::StartVirtualCam()
  190. {
  191. if (!main->vcamEnabled)
  192. return false;
  193. bool typeIsProgram = main->vcamConfig.type == VCamOutputType::ProgramView;
  194. if (!virtualCamView && !typeIsProgram)
  195. virtualCamView = obs_view_create();
  196. UpdateVirtualCamOutputSource();
  197. if (!virtualCamVideo) {
  198. virtualCamVideo = typeIsProgram ? obs_get_video() : obs_view_add(virtualCamView);
  199. if (!virtualCamVideo)
  200. return false;
  201. }
  202. obs_output_set_media(virtualCam, virtualCamVideo, obs_get_audio());
  203. if (!Active())
  204. SetupOutputs();
  205. bool success = obs_output_start(virtualCam);
  206. if (!success) {
  207. QString errorReason;
  208. const char *error = obs_output_get_last_error(virtualCam);
  209. if (error) {
  210. errorReason = QT_UTF8(error);
  211. } else {
  212. errorReason = QTStr("Output.StartFailedGeneric");
  213. }
  214. QMessageBox::critical(main, QTStr("Output.StartVirtualCamFailed"), errorReason);
  215. DestroyVirtualCamView();
  216. }
  217. log_vcam_changed(main->vcamConfig, true);
  218. return success;
  219. }
  220. void BasicOutputHandler::StopVirtualCam()
  221. {
  222. if (main->vcamEnabled) {
  223. obs_output_stop(virtualCam);
  224. }
  225. }
  226. bool BasicOutputHandler::VirtualCamActive() const
  227. {
  228. if (main->vcamEnabled) {
  229. return obs_output_active(virtualCam);
  230. }
  231. return false;
  232. }
  233. void BasicOutputHandler::UpdateVirtualCamOutputSource()
  234. {
  235. if (!main->vcamEnabled || !virtualCamView)
  236. return;
  237. OBSSourceAutoRelease source;
  238. switch (main->vcamConfig.type) {
  239. case VCamOutputType::Invalid:
  240. case VCamOutputType::ProgramView:
  241. DestroyVirtualCameraScene();
  242. return;
  243. case VCamOutputType::PreviewOutput: {
  244. DestroyVirtualCameraScene();
  245. OBSSource s = main->GetCurrentSceneSource();
  246. obs_source_get_ref(s);
  247. source = s.Get();
  248. break;
  249. }
  250. case VCamOutputType::SceneOutput:
  251. DestroyVirtualCameraScene();
  252. source = obs_get_source_by_name(main->vcamConfig.scene.c_str());
  253. break;
  254. case VCamOutputType::SourceOutput:
  255. OBSSourceAutoRelease s = obs_get_source_by_name(main->vcamConfig.source.c_str());
  256. if (!vCamSourceScene)
  257. vCamSourceScene = obs_scene_create_private("vcam_source");
  258. source = obs_source_get_ref(obs_scene_get_source(vCamSourceScene));
  259. if (vCamSourceSceneItem && (obs_sceneitem_get_source(vCamSourceSceneItem) != s)) {
  260. obs_sceneitem_remove(vCamSourceSceneItem);
  261. vCamSourceSceneItem = nullptr;
  262. }
  263. if (!vCamSourceSceneItem) {
  264. vCamSourceSceneItem = obs_scene_add(vCamSourceScene, s);
  265. obs_sceneitem_set_bounds_type(vCamSourceSceneItem, OBS_BOUNDS_SCALE_INNER);
  266. obs_sceneitem_set_bounds_alignment(vCamSourceSceneItem, OBS_ALIGN_CENTER);
  267. const struct vec2 size = {
  268. (float)obs_source_get_width(source),
  269. (float)obs_source_get_height(source),
  270. };
  271. obs_sceneitem_set_bounds(vCamSourceSceneItem, &size);
  272. }
  273. break;
  274. }
  275. OBSSourceAutoRelease current = obs_view_get_source(virtualCamView, 0);
  276. if (source != current)
  277. obs_view_set_source(virtualCamView, 0, source);
  278. }
  279. void BasicOutputHandler::DestroyVirtualCamView()
  280. {
  281. if (main->vcamConfig.type == VCamOutputType::ProgramView) {
  282. virtualCamVideo = nullptr;
  283. return;
  284. }
  285. obs_view_remove(virtualCamView);
  286. obs_view_set_source(virtualCamView, 0, nullptr);
  287. virtualCamVideo = nullptr;
  288. obs_view_destroy(virtualCamView);
  289. virtualCamView = nullptr;
  290. DestroyVirtualCameraScene();
  291. }
  292. void BasicOutputHandler::DestroyVirtualCameraScene()
  293. {
  294. if (!vCamSourceScene)
  295. return;
  296. obs_scene_release(vCamSourceScene);
  297. vCamSourceScene = nullptr;
  298. vCamSourceSceneItem = nullptr;
  299. }
  300. const char *FindAudioEncoderFromCodec(const char *type)
  301. {
  302. const char *alt_enc_id = nullptr;
  303. size_t i = 0;
  304. while (obs_enum_encoder_types(i++, &alt_enc_id)) {
  305. const char *codec = obs_get_encoder_codec(alt_enc_id);
  306. if (strcmp(type, codec) == 0) {
  307. return alt_enc_id;
  308. }
  309. }
  310. return nullptr;
  311. }
  312. void clear_archive_encoder(obs_output_t *output, const char *expected_name)
  313. {
  314. obs_encoder_t *last = obs_output_get_audio_encoder(output, 1);
  315. bool clear = false;
  316. /* ensures that we don't remove twitch's soundtrack encoder */
  317. if (last) {
  318. const char *name = obs_encoder_get_name(last);
  319. clear = name && strcmp(name, expected_name) == 0;
  320. obs_encoder_release(last);
  321. }
  322. if (clear)
  323. obs_output_set_audio_encoder(output, nullptr, 1);
  324. }
  325. void BasicOutputHandler::SetupAutoRemux(const char *&container)
  326. {
  327. bool autoRemux = config_get_bool(main->Config(), "Video", "AutoRemux");
  328. if (autoRemux && strcmp(container, "mp4") == 0)
  329. container = "mkv";
  330. }
  331. std::string BasicOutputHandler::GetRecordingFilename(const char *path, const char *container, bool noSpace,
  332. bool overwrite, const char *format, bool ffmpeg)
  333. {
  334. if (!ffmpeg)
  335. SetupAutoRemux(container);
  336. string dst = GetOutputFilename(path, container, noSpace, overwrite, format);
  337. lastRecordingPath = dst;
  338. return dst;
  339. }
  340. extern std::string DeserializeConfigText(const char *text);
  341. std::shared_future<void> BasicOutputHandler::SetupMultitrackVideo(obs_service_t *service, std::string audio_encoder_id,
  342. size_t main_audio_mixer,
  343. std::optional<size_t> vod_track_mixer,
  344. std::function<void(std::optional<bool>)> continuation)
  345. {
  346. auto start_streaming_guard = std::make_shared<StartMultitrackVideoStreamingGuard>();
  347. if (!multitrackVideo) {
  348. continuation(std::nullopt);
  349. return start_streaming_guard->GetFuture();
  350. }
  351. multitrackVideoActive = false;
  352. streamDelayStarting.Disconnect();
  353. streamStopping.Disconnect();
  354. startStreaming.Disconnect();
  355. stopStreaming.Disconnect();
  356. bool is_custom = strncmp("rtmp_custom", obs_service_get_type(service), 11) == 0;
  357. std::optional<std::string> custom_config = std::nullopt;
  358. if (config_get_bool(main->Config(), "Stream1", "MultitrackVideoConfigOverrideEnabled"))
  359. custom_config = DeserializeConfigText(
  360. config_get_string(main->Config(), "Stream1", "MultitrackVideoConfigOverride"));
  361. std::optional<QString> extraCanvasUUID;
  362. const char *uuid = config_get_string(main->Config(), "Stream1", "MultitrackExtraCanvas");
  363. if (uuid && *uuid) {
  364. extraCanvasUUID = uuid;
  365. }
  366. OBSDataAutoRelease settings = obs_service_get_settings(service);
  367. QString key = obs_data_get_string(settings, "key");
  368. const char *service_name = "<unknown>";
  369. if (is_custom && obs_data_has_user_value(settings, "service_name")) {
  370. service_name = obs_data_get_string(settings, "service_name");
  371. } else if (!is_custom) {
  372. service_name = obs_data_get_string(settings, "service");
  373. }
  374. std::optional<std::string> custom_rtmp_url;
  375. std::optional<bool> use_rtmps;
  376. auto server = obs_data_get_string(settings, "server");
  377. if (strncmp(server, "auto", 4) != 0) {
  378. custom_rtmp_url = server;
  379. } else {
  380. QString server_ = server;
  381. use_rtmps = server_.contains("rtmps", Qt::CaseInsensitive);
  382. }
  383. auto service_custom_server = obs_data_get_bool(settings, "using_custom_server");
  384. if (custom_rtmp_url.has_value()) {
  385. blog(LOG_INFO, "Using %sserver '%s'", service_custom_server ? "custom " : "", custom_rtmp_url->c_str());
  386. }
  387. auto maximum_aggregate_bitrate =
  388. config_get_bool(main->Config(), "Stream1", "MultitrackVideoMaximumAggregateBitrateAuto")
  389. ? std::nullopt
  390. : std::make_optional<uint32_t>(
  391. config_get_int(main->Config(), "Stream1", "MultitrackVideoMaximumAggregateBitrate"));
  392. auto maximum_video_tracks = config_get_bool(main->Config(), "Stream1", "MultitrackVideoMaximumVideoTracksAuto")
  393. ? std::nullopt
  394. : std::make_optional<uint32_t>(config_get_int(
  395. main->Config(), "Stream1", "MultitrackVideoMaximumVideoTracks"));
  396. auto stream_dump_config = GenerateMultitrackVideoStreamDumpConfig();
  397. auto continue_on_main_thread = [&, start_streaming_guard, service = OBSService{service},
  398. continuation =
  399. std::move(continuation)](std::optional<MultitrackVideoError> error) {
  400. if (error) {
  401. OBSDataAutoRelease service_settings = obs_service_get_settings(service);
  402. auto multitrack_video_name = QTStr("Basic.Settings.Stream.MultitrackVideoLabel");
  403. if (obs_data_has_user_value(service_settings, "multitrack_video_name")) {
  404. multitrack_video_name = obs_data_get_string(service_settings, "multitrack_video_name");
  405. }
  406. multitrackVideoActive = false;
  407. if (!error->ShowDialog(main, multitrack_video_name))
  408. return continuation(false);
  409. return continuation(std::nullopt);
  410. }
  411. multitrackVideoActive = true;
  412. auto signal_handler = multitrackVideo->StreamingSignalHandler();
  413. streamDelayStarting.Connect(signal_handler, "starting", OBSStreamStarting, this);
  414. streamStopping.Connect(signal_handler, "stopping", OBSStreamStopping, this);
  415. startStreaming.Connect(signal_handler, "start", OBSStartStreaming, this);
  416. stopStreaming.Connect(signal_handler, "stop", OBSStopStreaming, this);
  417. return continuation(true);
  418. };
  419. QThreadPool::globalInstance()->start([=, multitrackVideo = multitrackVideo.get(),
  420. service_name = std::string{service_name}, service = OBSService{service},
  421. stream_dump_config = OBSData{stream_dump_config},
  422. start_streaming_guard = start_streaming_guard]() mutable {
  423. std::optional<MultitrackVideoError> error;
  424. try {
  425. multitrackVideo->PrepareStreaming(main, service_name.c_str(), service, custom_rtmp_url, key,
  426. audio_encoder_id.c_str(), maximum_aggregate_bitrate,
  427. maximum_video_tracks, custom_config, stream_dump_config,
  428. main_audio_mixer, vod_track_mixer, use_rtmps,
  429. extraCanvasUUID);
  430. } catch (const MultitrackVideoError &error_) {
  431. error.emplace(error_);
  432. }
  433. QMetaObject::invokeMethod(main, [=] { continue_on_main_thread(error); });
  434. });
  435. return start_streaming_guard->GetFuture();
  436. }
  437. OBSDataAutoRelease BasicOutputHandler::GenerateMultitrackVideoStreamDumpConfig()
  438. {
  439. auto stream_dump_enabled = config_get_bool(main->Config(), "Stream1", "MultitrackVideoStreamDumpEnabled");
  440. if (!stream_dump_enabled)
  441. return nullptr;
  442. const char *path = config_get_string(main->Config(), "SimpleOutput", "FilePath");
  443. bool noSpace = config_get_bool(main->Config(), "SimpleOutput", "FileNameWithoutSpace");
  444. const char *filenameFormat = config_get_string(main->Config(), "Output", "FilenameFormatting");
  445. bool overwriteIfExists = config_get_bool(main->Config(), "Output", "OverwriteIfExists");
  446. bool useMP4 = config_get_bool(main->Config(), "Stream1", "MultitrackVideoStreamDumpAsMP4");
  447. string f;
  448. OBSDataAutoRelease settings = obs_data_create();
  449. f = GetFormatString(filenameFormat, nullptr, nullptr);
  450. string strPath = GetRecordingFilename(path, useMP4 ? "mp4" : "flv", noSpace, overwriteIfExists, f.c_str(),
  451. // never remux stream dump
  452. false);
  453. obs_data_set_string(settings, "path", strPath.c_str());
  454. if (useMP4) {
  455. obs_data_set_bool(settings, "use_mp4", true);
  456. obs_data_set_string(settings, "muxer_settings", "write_encoder_info=1");
  457. }
  458. return settings;
  459. }
  460. BasicOutputHandler *CreateSimpleOutputHandler(OBSBasic *main)
  461. {
  462. return new SimpleOutput(main);
  463. }
  464. BasicOutputHandler *CreateAdvancedOutputHandler(OBSBasic *main)
  465. {
  466. return new AdvancedOutput(main);
  467. }