2024-07-26 01:09:44 +00:00
|
|
|
// SPDX-License-Identifier: BSD-2-Clause
|
|
|
|
//
|
|
|
|
// Copyright (C) 2020, Matthew Waters <matthew@centricular.com>
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
2020-02-04 08:43:52 +00:00
|
|
|
#include <QApplication>
|
|
|
|
#include <QQmlApplicationEngine>
|
|
|
|
#include <QQuickWindow>
|
|
|
|
#include <QQuickItem>
|
|
|
|
#include <QRunnable>
|
|
|
|
#include <QTimer>
|
|
|
|
#include <gst/gst.h>
|
|
|
|
#include <gst/gl/gl.h>
|
|
|
|
|
|
|
|
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<QQuickItem *> ("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<QQuickWindow *> (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;
|
|
|
|
}
|