// SPDX-License-Identifier: BSD-2-Clause // // Copyright (C) 2020, 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 #include static GstBusSyncReply on_sync_bus_message (GstBus * bus, GstMessage * msg, gpointer data) { GstElement *pipeline = (GstElement *) data; switch (GST_MESSAGE_TYPE (msg)) { case GST_MESSAGE_HAVE_CONTEXT: { GstContext *context; gst_message_parse_have_context (msg, &context); /* if you need specific behviour or a context from a specific element, * you need to be selective about which context's you set on the * pipeline */ if (gst_context_has_context_type (context, GST_GL_DISPLAY_CONTEXT_TYPE)) { gst_println ("got have-context %p", context); gst_element_set_context (pipeline, context); } if (context) gst_context_unref (context); gst_message_unref (msg); return GST_BUS_DROP; } default: break; } return GST_BUS_PASS; } static void connect_tee (GstElement * tee, GstElement * queue) { gst_println ("attaching tee/queue %p %p", tee, queue); gst_element_link (tee, queue); } static void connect_qmlglsink (GstElement * pipeline, GstElement * tee, QQuickWindow * rootObject) { GstElement *queue = gst_element_factory_make ("queue", NULL); GstElement *qmlglsink = gst_element_factory_make ("qmlglsink", NULL); QQuickItem *videoItem; gst_println ("attaching qmlglsink %s at %p", GST_OBJECT_NAME (qmlglsink), qmlglsink); gst_bin_add (GST_BIN (pipeline), queue); gst_bin_add (GST_BIN (pipeline), qmlglsink); gst_element_link (queue, qmlglsink); gst_element_set_state (queue, GST_STATE_PLAYING); videoItem = rootObject->findChild ("videoItem"); g_assert (videoItem); g_object_set (qmlglsink, "widget", videoItem, NULL); gst_element_set_state (qmlglsink, GST_STATE_PAUSED); connect_tee (tee, queue); gst_element_set_state (qmlglsink, GST_STATE_PLAYING); } int main(int argc, char *argv[]) { int ret; gst_init (&argc, &argv); { QGuiApplication app(argc, argv); /* test a whole bunch of elements respect the change in display * and therefore OpenGL context */ GstElement *pipeline = gst_parse_launch ("gltestsrc ! " "capsfilter caps=video/x-raw(ANY),framerate=10/1 ! glupload ! " "glcolorconvert ! glalpha noise-level=16 method=green angle=40 ! " "glcolorbalance hue=0.25 ! gltransformation rotation-x=30 ! " "glvideomixerelement ! glviewconvert output-mode-override=side-by-side ! " "glstereosplit name=s " "glstereomix name=m ! tee name=t ! queue ! fakesink sync=true " "s.left ! queue ! m.sink_0 " "s.right ! queue ! m.sink_1", NULL); GstBus *bus = gst_element_get_bus (pipeline); gst_bus_set_sync_handler (bus, on_sync_bus_message, pipeline, NULL); gst_object_unref (bus); /* the plugin must be loaded before loading the qml file to register the * GstGLVideoItem qml item */ GstElement *sink = gst_element_factory_make ("qmlglsink", NULL); g_assert (pipeline && sink); gst_object_unref (sink); QQuickWindow *rootObject; /* The Qml scene starts out with the widget not connected to any qmlglsink * element */ QQmlApplicationEngine engine; engine.load(QUrl(QStringLiteral("qrc:/main.qml"))); /* find and set the videoItem on the sink */ rootObject = static_cast (engine.rootObjects().first()); gst_element_set_state (pipeline, GST_STATE_PLAYING); GstElement *t = gst_bin_get_by_name (GST_BIN (pipeline), "t"); gst_object_unref (t); /* ref held by pipeline */ /* add the qmlglsink element */ QTimer::singleShot(5000, [pipeline, t, rootObject]() { connect_qmlglsink (pipeline, t, rootObject); } ); ret = app.exec(); gst_element_set_state (pipeline, GST_STATE_NULL); gst_object_unref (pipeline); } gst_deinit (); return ret; }