// SPDX-License-Identifier: BSD-2-Clause // // Copyright (C) 2023, Matthew Waters // // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are met: // // a) Redistributions of source code must retain the above copyright notice, // this list of conditions and the following disclaimer. // // b) Redistributions in binary form must reproduce the above copyright // notice, this list of conditions and the following disclaimer in // the documentation and/or other materials provided with the distribution. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, // THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE // ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE // LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR // CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF // SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS // INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF // THE POSSIBILITY OF SUCH DAMAGE. #include #include #include #include #include #include #include class SetPlaying : public QRunnable { public: SetPlaying(GstElement *); ~SetPlaying(); void run (); private: GstElement * pipeline_; }; SetPlaying::SetPlaying (GstElement * pipeline) { this->pipeline_ = pipeline ? static_cast (gst_object_ref (pipeline)) : NULL; } SetPlaying::~SetPlaying () { if (this->pipeline_) gst_object_unref (this->pipeline_); } void SetPlaying::run () { if (this->pipeline_) gst_element_set_state (this->pipeline_, GST_STATE_PLAYING); } static void on_mixer_scene_initialized (GstElement * mixer, gpointer unused) { QQuickItem *rootObject; GST_INFO ("scene initialized"); g_object_get (mixer, "root-item", &rootObject, NULL); QQuickItem *videoItem0 = rootObject->findChild ("inputVideoItem0"); GstPad *sink0 = gst_element_get_static_pad (mixer, "sink_0"); g_object_set (sink0, "widget", videoItem0, NULL); gst_clear_object (&sink0); QQuickItem *videoItem1 = rootObject->findChild ("inputVideoItem1"); GstPad *sink1 = gst_element_get_static_pad (mixer, "sink_1"); g_object_set (sink1, "widget", videoItem1, NULL); gst_clear_object (&sink1); } int main(int argc, char *argv[]) { int ret; gst_init (&argc, &argv); { QGuiApplication app(argc, argv); QQuickWindow::setGraphicsApi(QSGRendererInterface::OpenGL); GstElement *pipeline = gst_pipeline_new (NULL); GstElement *src0 = gst_element_factory_make ("videotestsrc", NULL); GstElement *capsfilter = gst_element_factory_make ("capsfilter", NULL); GstCaps *caps = gst_caps_from_string ("video/x-raw,format=YV12"); g_object_set (capsfilter, "caps", caps, NULL); gst_clear_caps (&caps); GstElement *glupload0 = gst_element_factory_make ("glupload", NULL); GstElement *src1 = gst_element_factory_make ("videotestsrc", NULL); gst_util_set_object_arg ((GObject *) src1, "pattern", "ball"); GstElement *glupload1 = gst_element_factory_make ("glupload", NULL); /* the plugin must be loaded before loading the qml file to register the * GstGLVideoItem qml item */ GstElement *mixer = gst_element_factory_make ("qml6glmixer", NULL); GstElement *sink = gst_element_factory_make ("qml6glsink", NULL); g_assert (src0 && glupload0 && mixer && sink); gst_bin_add_many (GST_BIN (pipeline), src0, capsfilter, glupload0, src1, glupload1, mixer, sink, NULL); gst_element_link_many (src0, capsfilter, glupload0, mixer, sink, NULL); gst_element_link_many (src1, glupload1, mixer, NULL); /* load qmlglsink output */ QQmlApplicationEngine engine; engine.load(QUrl(QStringLiteral("qrc:/main.qml"))); QQuickItem *videoItem; QQuickWindow *rootObject; /* find and set the videoItem on the sink */ rootObject = static_cast (engine.rootObjects().first()); videoItem = rootObject->findChild ("videoItem"); g_assert (videoItem); g_object_set(sink, "widget", videoItem, NULL); QDirIterator it(":", QDirIterator::Subdirectories); while (it.hasNext()) { qDebug() << it.next(); } QFile f(":/mixer.qml"); if(!f.open(QIODevice::ReadOnly)) { qWarning() << "error: " << f.errorString(); return 1; } QByteArray overlay_scene = f.readAll(); qDebug() << overlay_scene; /* load qmlgloverlay contents */ g_signal_connect (mixer, "qml-scene-initialized", G_CALLBACK (on_mixer_scene_initialized), NULL); g_object_set (mixer, "qml-scene", overlay_scene.data(), NULL); rootObject->scheduleRenderJob (new SetPlaying (pipeline), QQuickWindow::BeforeSynchronizingStage); ret = app.exec(); gst_element_set_state (pipeline, GST_STATE_NULL); gst_object_unref (pipeline); } gst_deinit (); return ret; }