/****************************************************************************** Copyright (C) 2013 by Hugh Bailey This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . ******************************************************************************/ #include #include #include "obs-app.hpp" #include "wx-wrappers.hpp" #include "window-basic-settings.hpp" #include "window-basic-main.hpp" #include "window-namedialog.hpp" #ifdef __WXGTK__ #include #endif using namespace std; obs_scene_t OBSBasic::GetCurrentScene() { int sel = scenes->GetSelection(); if (sel == wxNOT_FOUND) return NULL; return (obs_scene_t)scenes->GetClientData(sel); } void OBSBasic::AddScene(obs_source_t source) { const char *name = obs_source_getname(source); obs_scene_t scene = obs_scene_fromsource(source); scenes->Append(WX_UTF8(name), scene); signal_handler_t handler = obs_source_signalhandler(source); signal_handler_connect(handler, "add", OBSBasic::SceneItemAdded, this); signal_handler_connect(handler, "remove", OBSBasic::SceneItemRemoved, this); } void OBSBasic::RemoveScene(obs_source_t source) { const char *name = obs_source_getname(source); int idx = scenes->FindString(name); int sel = scenes->GetSelection(); if (idx != wxNOT_FOUND) { if (sel == idx) sources->Clear(); scenes->Delete(idx); } } void OBSBasic::AddSceneItem(obs_sceneitem_t item) { obs_scene_t scene = obs_sceneitem_getscene(item); obs_source_t source = obs_sceneitem_getsource(item); const char *name = obs_source_getname(source); if (GetCurrentScene() == scene) sources->Insert(WX_UTF8(name), 0, item); sourceSceneRefs[source] = sourceSceneRefs[source] + 1; } void OBSBasic::RemoveSceneItem(obs_sceneitem_t item) { obs_scene_t scene = obs_sceneitem_getscene(item); if (GetCurrentScene() == scene) { for (unsigned int idx = 0; idx < sources->GetCount(); idx++) { obs_sceneitem_t curItem; curItem = (obs_sceneitem_t)sources->GetClientData(idx); if (item == curItem) { sources->Delete(idx); break; } } } obs_source_t source = obs_sceneitem_getsource(item); obs_source_addref(source); obs_source_release(source); int scenes = sourceSceneRefs[source] - 1; if (scenes == 0) { obs_source_remove(source); sourceSceneRefs.erase(source); } } void OBSBasic::UpdateSources(obs_scene_t scene) { sources->Clear(); obs_scene_enum_items(scene, [] (obs_scene_t scene, obs_sceneitem_t item, void *p) { OBSBasic *window = static_cast(p); window->AddSceneItem(item); return true; }, this); } void OBSBasic::UpdateSceneSelection(obs_source_t source) { if (source) { obs_source_type type; obs_source_gettype(source, &type, NULL); if (type == SOURCE_SCENE) { obs_scene_t scene = obs_scene_fromsource(source); const char *name = obs_source_getname(source); int idx = scenes->FindString(WX_UTF8(name)); int sel = scenes->GetSelection(); if (idx != sel) { scenes->SetSelection(idx); UpdateSources(scene); } } } else { scenes->SetSelection(wxNOT_FOUND); } } /* OBS Callbacks */ void OBSBasic::SceneItemAdded(void *data, calldata_t params) { OBSBasic *window = static_cast(data); obs_scene_t scene = (obs_scene_t)calldata_ptr(params, "scene"); obs_sceneitem_t item = (obs_sceneitem_t)calldata_ptr(params, "item"); window->AddSceneItem(item); } void OBSBasic::SceneItemRemoved(void *data, calldata_t params) { OBSBasic *window = static_cast(data); obs_scene_t scene = (obs_scene_t)calldata_ptr(params, "scene"); obs_sceneitem_t item = (obs_sceneitem_t)calldata_ptr(params, "item"); window->RemoveSceneItem(item); } void OBSBasic::SourceAdded(void *data, calldata_t params) { obs_source_t source = (obs_source_t)calldata_ptr(params, "source"); obs_source_type type; obs_source_gettype(source, &type, NULL); if (type == SOURCE_SCENE) static_cast(data)->AddScene(source); } void OBSBasic::SourceDestroyed(void *data, calldata_t params) { obs_source_t source = (obs_source_t)calldata_ptr(params, "source"); obs_source_type type; obs_source_gettype(source, &type, NULL); if (type == SOURCE_SCENE) static_cast(data)->RemoveScene(source); } void OBSBasic::ChannelChanged(void *data, calldata_t params) { obs_source_t source = (obs_source_t)calldata_ptr(params, "source"); uint32_t channel = calldata_uint32(params, "channel"); if (channel == 0) static_cast(data)->UpdateSceneSelection(source); } /* Main class functions */ bool OBSBasic::Init() { if (!obs_startup()) return false; if (!InitGraphics()) return false; if (!InitAudio()) return false; signal_handler_connect(obs_signalhandler(), "source-add", OBSBasic::SourceAdded, this); signal_handler_connect(obs_signalhandler(), "source-destroy", OBSBasic::SourceDestroyed, this); signal_handler_connect(obs_signalhandler(), "channel-change", OBSBasic::ChannelChanged, this); /* TODO: this is a test */ obs_load_module("test-input"); return true; } OBSBasic::~OBSBasic() { obs_shutdown(); } bool OBSBasic::InitGraphics() { struct obs_video_info ovi; wxGetApp().GetConfigFPS(ovi.fps_num, ovi.fps_den); ovi.graphics_module = wxGetApp().GetRenderModule(); ovi.base_width = (uint32_t)config_get_uint(GetGlobalConfig(), "Video", "BaseCX"); ovi.base_height = (uint32_t)config_get_uint(GetGlobalConfig(), "Video", "BaseCY"); ovi.output_width = (uint32_t)config_get_uint(GetGlobalConfig(), "Video", "OutputCX"); ovi.output_height = (uint32_t)config_get_uint(GetGlobalConfig(), "Video", "OutputCY"); ovi.output_format = VIDEO_FORMAT_RGBA; ovi.adapter = 0; #ifdef __WXGTK__ /* Ugly hack for GTK, I'm hoping this can be avoided eventually... */ gtk_widget_realize(previewPanel->GetHandle()); #endif ovi.window = WxToGSWindow(previewPanel); //required to make opengl display stuff on osx(?) ResizePreview(ovi.base_width, ovi.base_height); SendSizeEvent(); wxSize size = previewPanel->GetClientSize(); ovi.window_width = size.x; ovi.window_height = size.y; return obs_reset_video(&ovi); } bool OBSBasic::InitAudio() { /* TODO: load audio settings from config */ struct audio_info ai; ai.name = "test"; ai.samples_per_sec = 44100; ai.format = AUDIO_FORMAT_16BIT; ai.speakers = SPEAKERS_STEREO; ai.buffer_ms = 700; return obs_reset_audio(&ai); } void OBSBasic::OnClose(wxCloseEvent &event) { wxGetApp().ExitMainLoop(); event.Skip(); } void OBSBasic::OnMinimize(wxIconizeEvent &event) { event.Skip(); } void OBSBasic::ResizePreview(uint32_t cx, uint32_t cy) { /* resize preview panel to fix to the top section of the window */ wxSize targetSize = GetPreviewContainer()->GetSize(); double targetAspect = double(targetSize.x) / double(targetSize.y); double baseAspect = double(cx) / double(cy); wxSize newSize; if (targetAspect > baseAspect) newSize = wxSize(targetSize.y * baseAspect, targetSize.y); else newSize = wxSize(targetSize.x, targetSize.x / baseAspect); GetPreviewPanel()->SetMinSize(newSize); } void OBSBasic::OnSize(wxSizeEvent &event) { struct obs_video_info ovi; event.Skip(); if (!obs_get_video_info(&ovi)) return; ResizePreview(ovi.base_width, ovi.base_height); } void OBSBasic::OnResizePreview(wxSizeEvent &event) { event.Skip(); wxSize newSize = previewPanel->GetClientSize(); graphics_t graphics = obs_graphics(); if (graphics) { gs_entercontext(graphics); gs_resize(newSize.x, newSize.y); gs_leavecontext(); } } void OBSBasic::fileNewClicked(wxCommandEvent &event) { } void OBSBasic::fileOpenClicked(wxCommandEvent &event) { } void OBSBasic::fileSaveClicked(wxCommandEvent &event) { } void OBSBasic::fileExitClicked(wxCommandEvent &event) { wxGetApp().ExitMainLoop(); } void OBSBasic::scenesClicked(wxCommandEvent &event) { int sel = scenes->GetSelection(); obs_source_t source = NULL; if (sel != wxNOT_FOUND) { obs_scene_t scene = (obs_scene_t)scenes->GetClientData(sel); source = obs_scene_getsource(scene); UpdateSources(scene); } /* TODO: allow transitions */ obs_set_output_source(0, source); } void OBSBasic::scenesRDown(wxMouseEvent &event) { } void OBSBasic::sceneAddClicked(wxCommandEvent &event) { string name; int ret = NameDialog::AskForName(this, Str("MainWindow.AddSceneDlg.Title"), Str("MainWindow.AddSceneDlg.Text"), name); if (ret == wxID_OK) { obs_source_t source = obs_get_source_by_name(name.c_str()); if (source) { wxMessageBox(WXStr("MainWindow.NameExists.Text"), WXStr("MainWindow.NameExists.Title"), wxOK|wxCENTRE, this); obs_source_release(source); sceneAddClicked(event); return; } obs_scene_t scene = obs_scene_create(name.c_str()); source = obs_scene_getsource(scene); obs_add_source(source); obs_scene_release(scene); obs_set_output_source(0, source); } } void OBSBasic::sceneRemoveClicked(wxCommandEvent &event) { int sel = scenes->GetSelection(); if (sel == wxNOT_FOUND) return; obs_scene_t scene = (obs_scene_t)scenes->GetClientData(sel); obs_source_t source = obs_scene_getsource(scene); obs_source_remove(source); } void OBSBasic::scenePropertiesClicked(wxCommandEvent &event) { } void OBSBasic::sceneUpClicked(wxCommandEvent &event) { } void OBSBasic::sceneDownClicked(wxCommandEvent &event) { } void OBSBasic::sourcesClicked(wxCommandEvent &event) { } void OBSBasic::sourcesToggled(wxCommandEvent &event) { } void OBSBasic::sourcesRDown(wxMouseEvent &event) { } void OBSBasic::AddSource(obs_scene_t scene, const char *id) { string name; bool success = false; while (!success) { int ret = NameDialog::AskForName(this, Str("MainWindow.AddSourceDlg.Title"), Str("MainWindow.AddSourceDlg.Text"), name); if (ret == wxID_CANCEL) break; obs_source_t source = obs_get_source_by_name( name.c_str()); if (!source) { success = true; } else { wxMessageBox(WXStr("MainWindow.NameExists.Text"), WXStr("MainWindow.NameExists.Title"), wxOK|wxCENTRE, this); obs_source_release(source); } } if (success) { obs_source_t source = obs_source_create(SOURCE_INPUT, id, name.c_str(), NULL); sourceSceneRefs[source] = 0; obs_add_source(source); obs_sceneitem_t item = obs_scene_add(scene, source); obs_source_release(source); } } void OBSBasic::AddSourcePopupMenu() { int sceneSel = scenes->GetSelection(); size_t idx = 0; const char *type; vector types; if (sceneSel == wxNOT_FOUND) return; obs_scene_t scene = (obs_scene_t)scenes->GetClientData(sceneSel); obs_scene_addref(scene); unique_ptr popup(new wxMenu()); while (obs_enum_input_types(idx, &type)) { const char *name = obs_source_getdisplayname(SOURCE_INPUT, type, wxGetApp().GetLocale()); types.push_back(type); popup->Append((int)idx+1, wxString(name, wxConvUTF8)); idx++; } if (idx) { int id = WXDoPopupMenu(this, popup.get()); if (id != -1) AddSource(scene, types[id-1]); } obs_scene_release(scene); } void OBSBasic::sourceAddClicked(wxCommandEvent &event) { AddSourcePopupMenu(); } void OBSBasic::sourceRemoveClicked(wxCommandEvent &event) { int sel = sources->GetSelection(); if (sel == wxNOT_FOUND) return; obs_sceneitem_t item = (obs_sceneitem_t)sources->GetClientData(sel); obs_source_t source = obs_sceneitem_getsource(item); obs_sceneitem_destroy(item); } void OBSBasic::sourcePropertiesClicked(wxCommandEvent &event) { } void OBSBasic::sourceUpClicked(wxCommandEvent &event) { } void OBSBasic::sourceDownClicked(wxCommandEvent &event) { } void OBSBasic::settingsClicked(wxCommandEvent &event) { OBSBasicSettings test(this); test.ShowModal(); } void OBSBasic::exitClicked(wxCommandEvent &event) { wxGetApp().ExitMainLoop(); }