| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640 |
- #include <QAction>
- #include <QGuiApplication>
- #include <QMouseEvent>
- #include <QMenu>
- #include <QScreen>
- #include "window-projector.hpp"
- #include "display-helpers.hpp"
- #include "qt-wrappers.hpp"
- #include "platform.hpp"
- static QList<OBSProjector *> multiviewProjectors;
- static bool updatingMultiview = false;
- OBSProjector::OBSProjector(QWidget *widget, obs_source_t *source_, bool window)
- : OBSQTDisplay (widget,
- Qt::Window),
- source (source_),
- removedSignal (obs_source_get_signal_handler(source),
- "remove", OBSSourceRemoved, this)
- {
- if (!window) {
- setWindowFlags(Qt::FramelessWindowHint |
- Qt::X11BypassWindowManagerHint);
- }
- setAttribute(Qt::WA_DeleteOnClose, true);
- //disable application quit when last window closed
- setAttribute(Qt::WA_QuitOnClose, false);
- installEventFilter(CreateShortcutFilter());
- auto addDrawCallback = [this] ()
- {
- bool isMultiview = type == ProjectorType::Multiview;
- obs_display_add_draw_callback(GetDisplay(),
- isMultiview ? OBSRenderMultiview : OBSRender,
- this);
- obs_display_set_background_color(GetDisplay(), 0x000000);
- };
- connect(this, &OBSQTDisplay::DisplayCreated, addDrawCallback);
- bool hideCursor = config_get_bool(GetGlobalConfig(),
- "BasicWindow", "HideProjectorCursor");
- if (hideCursor && !window) {
- QPixmap empty(16, 16);
- empty.fill(Qt::transparent);
- setCursor(QCursor(empty));
- }
- App()->IncrementSleepInhibition();
- resize(480, 270);
- }
- OBSProjector::~OBSProjector()
- {
- bool isMultiview = type == ProjectorType::Multiview;
- obs_display_remove_draw_callback(GetDisplay(),
- isMultiview ? OBSRenderMultiview : OBSRender, this);
- if (source)
- obs_source_dec_showing(source);
- if (isMultiview) {
- for (OBSWeakSource &weakSrc : multiviewScenes) {
- OBSSource src = OBSGetStrongRef(weakSrc);
- if (src)
- obs_source_dec_showing(src);
- }
- obs_enter_graphics();
- gs_vertexbuffer_destroy(outerBox);
- gs_vertexbuffer_destroy(innerBox);
- gs_vertexbuffer_destroy(leftVLine);
- gs_vertexbuffer_destroy(rightVLine);
- gs_vertexbuffer_destroy(leftLine);
- gs_vertexbuffer_destroy(topLine);
- gs_vertexbuffer_destroy(rightLine);
- obs_leave_graphics();
- }
- if (type == ProjectorType::Multiview)
- multiviewProjectors.removeAll(this);
- App()->DecrementSleepInhibition();
- }
- static OBSSource CreateLabel(const char *name, size_t h)
- {
- obs_data_t *settings = obs_data_create();
- obs_data_t *font = obs_data_create();
- std::string text;
- text += " ";
- text += name;
- text += " ";
- #if defined(_WIN32)
- obs_data_set_string(font, "face", "Arial");
- #elif defined(__APPLE__)
- obs_data_set_string(font, "face", "Helvetica");
- #else
- obs_data_set_string(font, "face", "Monospace");
- #endif
- obs_data_set_int(font, "flags", 0);
- obs_data_set_int(font, "size", int(h / 9.81));
- obs_data_set_obj(settings, "font", font);
- obs_data_set_string(settings, "text", text.c_str());
- obs_data_set_bool(settings, "outline", true);
- OBSSource txtSource = obs_source_create_private("text_ft2_source", name,
- settings);
- obs_source_release(txtSource);
- obs_data_release(font);
- obs_data_release(settings);
- return txtSource;
- }
- void OBSProjector::Init(int monitor, bool window, QString title,
- ProjectorType type_)
- {
- QScreen *screen = QGuiApplication::screens()[monitor];
- if (!window)
- setGeometry(screen->geometry());
- bool alwaysOnTop = config_get_bool(GetGlobalConfig(),
- "BasicWindow", "ProjectorAlwaysOnTop");
- if (alwaysOnTop && !window)
- SetAlwaysOnTop(this, true);
- if (window)
- setWindowTitle(title);
- show();
- if (source)
- obs_source_inc_showing(source);
- if (!window) {
- QAction *action = new QAction(this);
- action->setShortcut(Qt::Key_Escape);
- addAction(action);
- connect(action, SIGNAL(triggered()), this,
- SLOT(EscapeTriggered()));
- activateWindow();
- }
- savedMonitor = monitor;
- isWindow = window;
- type = type_;
- if (type == ProjectorType::Multiview) {
- obs_enter_graphics();
- gs_render_start(true);
- gs_vertex2f(0.001f, 0.001f);
- gs_vertex2f(0.001f, 0.997f);
- gs_vertex2f(0.997f, 0.997f);
- gs_vertex2f(0.997f, 0.001f);
- gs_vertex2f(0.001f, 0.001f);
- outerBox = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.04f, 0.04f);
- gs_vertex2f(0.04f, 0.96f);
- gs_vertex2f(0.96f, 0.96f);
- gs_vertex2f(0.96f, 0.04f);
- gs_vertex2f(0.04f, 0.04f);
- innerBox = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.15f, 0.04f);
- gs_vertex2f(0.15f, 0.96f);
- leftVLine = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.85f, 0.04f);
- gs_vertex2f(0.85f, 0.96f);
- rightVLine = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.0f, 0.5f);
- gs_vertex2f(0.075f, 0.5f);
- leftLine = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.5f, 0.0f);
- gs_vertex2f(0.5f, 0.09f);
- topLine = gs_render_save();
- gs_render_start(true);
- gs_vertex2f(0.925f, 0.5f);
- gs_vertex2f(1.0f, 0.5f);
- rightLine = gs_render_save();
- obs_leave_graphics();
- UpdateMultiview();
- multiviewProjectors.push_back(this);
- }
- ready = true;
- }
- static inline void renderVB(gs_effect_t *effect, gs_vertbuffer_t *vb,
- int cx, int cy)
- {
- if (!vb)
- return;
- matrix4 transform;
- matrix4_identity(&transform);
- transform.x.x = cx;
- transform.y.y = cy;
- gs_load_vertexbuffer(vb);
- gs_matrix_push();
- gs_matrix_mul(&transform);
- while (gs_effect_loop(effect, "Solid"))
- gs_draw(GS_LINESTRIP, 0, 0);
- gs_matrix_pop();
- }
- static inline uint32_t labelOffset(obs_source_t *label, uint32_t cx)
- {
- uint32_t w = obs_source_get_width(label);
- w = uint32_t(float(w) * 0.5f);
- return (cx / 2) - w;
- }
- void OBSProjector::OBSRenderMultiview(void *data, uint32_t cx, uint32_t cy)
- {
- OBSProjector *window = (OBSProjector *)data;
- if (updatingMultiview || !window->ready)
- return;
- OBSBasic *main = (OBSBasic *)obs_frontend_get_main_window();
- uint32_t targetCX, targetCY;
- int x, y;
- float fX, fY, halfCX, halfCY, sourceX, sourceY,
- quarterCX, quarterCY, scale, targetCXF, targetCYF,
- hiCX, hiCY, qiX, qiY, qiCX, qiCY,
- hiScaleX, hiScaleY, qiScaleX, qiScaleY;
- uint32_t offset;
- gs_effect_t *solid = obs_get_base_effect(OBS_EFFECT_SOLID);
- gs_eparam_t *color = gs_effect_get_param_by_name(solid, "color");
- struct obs_video_info ovi;
- obs_get_video_info(&ovi);
- targetCX = ovi.base_width;
- targetCY = ovi.base_height;
- GetScaleAndCenterPos(targetCX, targetCY, cx, cy, x, y, scale);
- targetCXF = float(targetCX);
- targetCYF = float(targetCY);
- fX = float(x);
- fY = float(y);
- halfCX = (targetCXF + 1) / 2;
- halfCY = (targetCYF + 1) / 2;
- hiCX = (halfCX - 4.0);
- hiCY = (halfCY - 4.0);
- hiScaleX = hiCX / targetCXF;
- hiScaleY = hiCY / targetCYF;
- quarterCX = (halfCX + 1) / 2;
- quarterCY = (halfCY + 1) / 2;
- qiCX = (quarterCX - 8.0);
- qiCY = (quarterCY - 8.0);
- qiScaleX = qiCX / targetCXF;
- qiScaleY = qiCY / targetCYF;
- OBSSource previewSrc = main->GetCurrentSceneSource();
- OBSSource programSrc = main->GetProgramSource();
- bool studioMode = main->IsPreviewProgramMode();
- auto drawBox = [solid, color] (float cx, float cy,
- uint32_t colorVal)
- {
- gs_effect_set_color(color, colorVal);
- while (gs_effect_loop(solid, "Solid"))
- gs_draw_sprite(nullptr, 0, (uint32_t)cx, (uint32_t)cy);
- };
- auto setRegion = [fX, fY, scale] (float x, float y, float cx, float cy)
- {
- float vX = int(fX + x * scale);
- float vY = int(fY + y * scale);
- float vCX = int(cx * scale);
- float vCY = int(cy * scale);
- float oL = x;
- float oT = y;
- float oR = (x + cx);
- float oB = (y + cy);
- gs_projection_push();
- gs_viewport_push();
- gs_set_viewport(vX, vY, vCX, vCY);
- gs_ortho(oL, oR, oT, oB, -100.0f, 100.0f);
- };
- auto resetRegion = [] ()
- {
- gs_viewport_pop();
- gs_projection_pop();
- };
- /* ----------------------------- */
- /* draw sources */
- gs_projection_push();
- gs_viewport_push();
- gs_set_viewport(x, y, targetCX * scale, targetCY * scale);
- gs_ortho(0.0f, targetCXF, 0.0f, targetCYF, -100.0f, 100.0f);
- for (size_t i = 0; i < 8; i++) {
- OBSSource src = OBSGetStrongRef(window->multiviewScenes[i]);
- obs_source *label = window->multiviewLabels[i + 2];
- if (!src)
- continue;
- if (!label)
- continue;
- if (i < 4) {
- sourceX = (float(i) * quarterCX);
- sourceY = halfCY;
- } else {
- sourceX = (float(i - 4) * quarterCX);
- sourceY = halfCY + quarterCY;
- }
- qiX = sourceX + 4.0f;
- qiY = sourceY + 4.0f;
- /* ----------- */
- if (src == previewSrc || src == programSrc) {
- uint32_t colorVal = src == programSrc
- ? 0xFFFF0000
- : 0xFF00FF00;
- gs_matrix_push();
- gs_matrix_translate3f(sourceX, sourceY, 0.0f);
- drawBox(quarterCX, quarterCY, colorVal);
- gs_matrix_pop();
- gs_matrix_push();
- gs_matrix_translate3f(qiX, qiY, 0.0f);
- drawBox(qiCX, qiCY, 0xFF000000);
- gs_matrix_pop();
- }
- /* ----------- */
- gs_matrix_push();
- gs_matrix_translate3f(qiX, qiY, 0.0f);
- gs_matrix_scale3f(qiScaleX, qiScaleY, 1.0f);
- setRegion(qiX, qiY, qiCX, qiCY);
- obs_source_video_render(src);
- resetRegion();
- gs_effect_set_color(color, 0xFFFFFFFF);
- renderVB(solid, window->outerBox, targetCX, targetCY);
- gs_matrix_pop();
- /* ----------- */
- offset = labelOffset(label, quarterCX);
- cx = obs_source_get_width(label);
- cy = obs_source_get_height(label);
- gs_matrix_push();
- gs_matrix_translate3f(sourceX + offset,
- (quarterCY * 0.8f) + sourceY, 0.0f);
- drawBox(cx, cy + int(quarterCX * 0.015f), 0xD91F1F1F);
- obs_source_video_render(label);
- gs_matrix_pop();
- }
- gs_effect_set_color(color, 0xFFFFFFFF);
- /* ----------------------------- */
- /* draw preview */
- gs_matrix_push();
- gs_matrix_translate3f(2.0f, 2.0f, 0.0f);
- gs_matrix_scale3f(hiScaleX, hiScaleY, 1.0f);
- setRegion(2.0f, 2.0f, hiCX, hiCY);
- if (studioMode) {
- obs_source_video_render(previewSrc);
- } else {
- obs_render_main_texture();
- }
- resetRegion();
- gs_matrix_pop();
- /* ----------- */
- gs_matrix_push();
- gs_matrix_scale3f(0.5f, 0.5f, 1.0f);
- renderVB(solid, window->outerBox, targetCX, targetCY);
- renderVB(solid, window->innerBox, targetCX, targetCY);
- renderVB(solid, window->leftVLine, targetCX, targetCY);
- renderVB(solid, window->rightVLine, targetCX, targetCY);
- renderVB(solid, window->leftLine, targetCX, targetCY);
- renderVB(solid, window->topLine, targetCX, targetCY);
- renderVB(solid, window->rightLine, targetCX, targetCY);
- gs_matrix_pop();
- /* ----------- */
- obs_source_t *previewLabel = window->multiviewLabels[0];
- offset = labelOffset(previewLabel, halfCX);
- cx = obs_source_get_width(previewLabel);
- cy = obs_source_get_height(previewLabel);
- gs_matrix_push();
- gs_matrix_translate3f(offset, (halfCY * 0.8f), 0.0f);
- drawBox(cx, cy + int(halfCX * 0.015f), 0xD91F1F1F);
- obs_source_video_render(previewLabel);
- gs_matrix_pop();
- /* ----------------------------- */
- /* draw program */
- gs_matrix_push();
- gs_matrix_translate3f(halfCX + 2.0, 2.0f, 0.0f);
- gs_matrix_scale3f(hiScaleX, hiScaleY, 1.0f);
- setRegion(halfCX + 2.0f, 2.0f, hiCX, hiCY);
- obs_render_main_texture();
- resetRegion();
- gs_matrix_pop();
- /* ----------- */
- gs_matrix_push();
- gs_matrix_translate3f(halfCX, 0.0f, 0.0f);
- gs_matrix_scale3f(0.5f, 0.5f, 1.0f);
- renderVB(solid, window->outerBox, targetCX, targetCY);
- gs_matrix_pop();
- /* ----------- */
- obs_source_t *programLabel = window->multiviewLabels[1];
- offset = labelOffset(programLabel, halfCX);
- cx = obs_source_get_width(programLabel);
- cy = obs_source_get_height(programLabel);
- gs_matrix_push();
- gs_matrix_translate3f(halfCX + offset, (halfCY * 0.8f), 0.0f);
- drawBox(cx, cy + int(halfCX * 0.015f), 0xD91F1F1F);
- obs_source_video_render(programLabel);
- gs_matrix_pop();
- /* ----------------------------- */
- gs_viewport_pop();
- gs_projection_pop();
- }
- void OBSProjector::OBSRender(void *data, uint32_t cx, uint32_t cy)
- {
- OBSProjector *window = reinterpret_cast<OBSProjector*>(data);
- if (!window->ready)
- return;
- OBSBasic *main = reinterpret_cast<OBSBasic*>(App()->GetMainWindow());
- uint32_t targetCX;
- uint32_t targetCY;
- int x, y;
- int newCX, newCY;
- float scale;
- if (window->source) {
- targetCX = std::max(obs_source_get_width(window->source), 1u);
- targetCY = std::max(obs_source_get_height(window->source), 1u);
- } else {
- struct obs_video_info ovi;
- obs_get_video_info(&ovi);
- targetCX = ovi.base_width;
- targetCY = ovi.base_height;
- }
- GetScaleAndCenterPos(targetCX, targetCY, cx, cy, x, y, scale);
- newCX = int(scale * float(targetCX));
- newCY = int(scale * float(targetCY));
- gs_viewport_push();
- gs_projection_push();
- gs_ortho(0.0f, float(targetCX), 0.0f, float(targetCY), -100.0f, 100.0f);
- gs_set_viewport(x, y, newCX, newCY);
- OBSSource source = window->source;
- if (window->type == ProjectorType::Preview &&
- main->IsPreviewProgramMode()) {
- OBSSource curSource = main->GetCurrentSceneSource();
- if (window->source != curSource) {
- obs_source_dec_showing(window->source);
- obs_source_inc_showing(curSource);
- source = curSource;
- }
- }
- if (source) {
- obs_source_video_render(source);
- } else {
- obs_render_main_texture();
- }
- gs_projection_pop();
- gs_viewport_pop();
- }
- void OBSProjector::OBSSourceRemoved(void *data, calldata_t *params)
- {
- OBSProjector *window = reinterpret_cast<OBSProjector*>(data);
- window->deleteLater();
- UNUSED_PARAMETER(params);
- }
- void OBSProjector::mousePressEvent(QMouseEvent *event)
- {
- OBSQTDisplay::mousePressEvent(event);
- if (event->button() == Qt::RightButton) {
- QMenu popup(this);
- popup.addAction(QTStr("Close"), this, SLOT(EscapeTriggered()));
- popup.exec(QCursor::pos());
- }
- }
- void OBSProjector::EscapeTriggered()
- {
- if (!isWindow) {
- OBSBasic *main = (OBSBasic*)obs_frontend_get_main_window();
- main->RemoveSavedProjectors(savedMonitor);
- }
- deleteLater();
- }
- void OBSProjector::UpdateMultiview()
- {
- for (OBSWeakSource &val : multiviewScenes)
- val = nullptr;
- for (OBSSource &val : multiviewLabels)
- val = nullptr;
- struct obs_video_info ovi;
- obs_get_video_info(&ovi);
- uint32_t h = ovi.base_height;
- struct obs_frontend_source_list scenes = {};
- obs_frontend_get_scenes(&scenes);
- int curIdx = 0;
- multiviewLabels[0] = CreateLabel(Str("StudioMode.Preview"), h / 2);
- multiviewLabels[1] = CreateLabel(Str("StudioMode.Program"), h / 2);
- for (size_t i = 0; i < scenes.sources.num && curIdx < 8; i++) {
- obs_source_t *src = scenes.sources.array[i];
- OBSData data = obs_source_get_private_settings(src);
- obs_data_release(data);
- obs_data_set_default_bool(data, "show_in_multiview", true);
- if (!obs_data_get_bool(data, "show_in_multiview"))
- continue;
- multiviewScenes[curIdx] = OBSGetWeakRef(src);
- obs_source_inc_showing(src);
- std::string name;
- name += std::to_string(curIdx + 1);
- name += " - ";
- name += obs_source_get_name(src);
- if (name.size() > 15)
- name.resize(15);
- multiviewLabels[curIdx + 2] = CreateLabel(name.c_str(), h / 4);
- curIdx++;
- }
- obs_frontend_source_list_free(&scenes);
- }
- void OBSProjector::UpdateMultiviewProjectors()
- {
- obs_enter_graphics();
- updatingMultiview = true;
- obs_leave_graphics();
- for (auto &projector : multiviewProjectors)
- projector->UpdateMultiview();
- obs_enter_graphics();
- updatingMultiview = false;
- obs_leave_graphics();
- }
|