mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2025-01-08 08:25:33 +00:00
157 lines
5.2 KiB
C++
157 lines
5.2 KiB
C++
// SPDX-License-Identifier: BSD-2-Clause
|
|
//
|
|
// Copyright (C) 2023, 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.
|
|
|
|
#include <QApplication>
|
|
#include <QQmlApplicationEngine>
|
|
#include <QQuickWindow>
|
|
#include <QQuickItem>
|
|
#include <QRunnable>
|
|
#include <QDirIterator>
|
|
#include <gst/gst.h>
|
|
|
|
class SetPlaying : public QRunnable
|
|
{
|
|
public:
|
|
SetPlaying(GstElement *);
|
|
~SetPlaying();
|
|
|
|
void run ();
|
|
|
|
private:
|
|
GstElement * pipeline_;
|
|
};
|
|
|
|
SetPlaying::SetPlaying (GstElement * pipeline)
|
|
{
|
|
this->pipeline_ = pipeline ? static_cast<GstElement *> (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<QQuickItem *> ("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<QQuickItem *> ("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<QQuickWindow *> (engine.rootObjects().first());
|
|
videoItem = rootObject->findChild<QQuickItem *> ("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;
|
|
}
|