Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
Q
qgroundcontrol
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Valentin Platzgummer
qgroundcontrol
Commits
d3315c2c
Commit
d3315c2c
authored
Apr 12, 2019
by
Willian Galvani
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Implement overlay on recorded videos by using subtitles
parent
e684dbe5
Changes
8
Hide whitespace changes
Inline
Side-by-side
Showing
8 changed files
with
258 additions
and
0 deletions
+258
-0
qgroundcontrol.pro
qgroundcontrol.pro
+2
-0
VideoManager.cc
src/FlightDisplay/VideoManager.cc
+1
-0
VideoManager.h
src/FlightDisplay/VideoManager.h
+2
-0
CMakeLists.txt
src/VideoStreaming/CMakeLists.txt
+1
-0
SubtitleWriter.cc
src/VideoStreaming/SubtitleWriter.cc
+193
-0
SubtitleWriter.h
src/VideoStreaming/SubtitleWriter.h
+57
-0
VideoReceiver.cc
src/VideoStreaming/VideoReceiver.cc
+1
-0
VideoReceiver.h
src/VideoStreaming/VideoReceiver.h
+1
-0
No files found.
qgroundcontrol.pro
View file @
d3315c2c
...
...
@@ -1189,12 +1189,14 @@ HEADERS += \
src
/
VideoStreaming
/
VideoStreaming
.
h
\
src
/
VideoStreaming
/
VideoSurface
.
h
\
src
/
VideoStreaming
/
VideoSurface_p
.
h
\
src
/
VideoStreaming
/
SubtitleWriter
.
h
\
SOURCES
+=
\
src
/
VideoStreaming
/
VideoItem
.
cc
\
src
/
VideoStreaming
/
VideoReceiver
.
cc
\
src
/
VideoStreaming
/
VideoStreaming
.
cc
\
src
/
VideoStreaming
/
VideoSurface
.
cc
\
src
/
VideoStreaming
/
SubtitleWriter
.
cc
\
contains
(
CONFIG
,
DISABLE_VIDEOSTREAMING
)
{
message
(
"Skipping support for video streaming (manual override from command line)"
)
...
...
src/FlightDisplay/VideoManager.cc
View file @
d3315c2c
...
...
@@ -81,6 +81,7 @@ VideoManager::setToolbox(QGCToolbox *toolbox)
_updateSettings
();
if
(
isGStreamer
())
{
startVideo
();
_subtitleWriter
.
setVideoReceiver
(
_videoReceiver
);
}
else
{
stopVideo
();
}
...
...
src/FlightDisplay/VideoManager.h
View file @
d3315c2c
...
...
@@ -20,6 +20,7 @@
#include "QGCLoggingCategory.h"
#include "VideoReceiver.h"
#include "QGCToolbox.h"
#include "SubtitleWriter.h"
Q_DECLARE_LOGGING_CATEGORY
(
VideoManagerLog
)
...
...
@@ -105,6 +106,7 @@ private:
void
_updateSettings
();
private:
SubtitleWriter
_subtitleWriter
;
bool
_isTaisync
=
false
;
VideoReceiver
*
_videoReceiver
=
nullptr
;
VideoReceiver
*
_thermalVideoReceiver
=
nullptr
;
...
...
src/VideoStreaming/CMakeLists.txt
View file @
d3315c2c
...
...
@@ -27,6 +27,7 @@ add_library(VideoStreaming
VideoReceiver.cc
VideoStreaming.cc
VideoSurface.cc
SubtitleWriter.cc
${
EXTRA_SRC
}
)
...
...
src/VideoStreaming/SubtitleWriter.cc
0 → 100644
View file @
d3315c2c
/****************************************************************************
*
* (c) 2009-2019 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
*
* QGroundControl is licensed according to the terms in the file
* COPYING.md in the root of the source code directory.
*
****************************************************************************/
/**
* @file
* @brief QGC Video Subtitle Writer
* @author Willian Galvani <williangalvani@gmail.com>
*/
#include "SubtitleWriter.h"
#include "SettingsManager.h"
#include "VideoReceiver.h"
#include "VideoManager.h"
#include "QGCApplication.h"
#include "QGCCorePlugin.h"
#include <QDateTime>
#include <QString>
#include <QDate>
QGC_LOGGING_CATEGORY
(
SubtitleWriterLog
,
"SubtitleWriterLog"
)
const
int
SubtitleWriter
::
_sampleRate
=
1
;
// Sample rate in Hz for getting telemetry data, most players do weird stuff when > 1Hz
SubtitleWriter
::
SubtitleWriter
(
QObject
*
parent
)
:
QObject
(
parent
)
{
}
void
SubtitleWriter
::
setVideoReceiver
(
VideoReceiver
*
videoReceiver
)
{
if
(
!
videoReceiver
)
{
qCWarning
(
SubtitleWriterLog
)
<<
"Invalid VideoReceiver pointer! Aborting subtitle capture!"
;
return
;
}
_videoReceiver
=
videoReceiver
;
// Only start writing subtitles once the recording pipeline actually starts
connect
(
_videoReceiver
,
&
VideoReceiver
::
gotFirstRecordingKeyFrame
,
this
,
&
SubtitleWriter
::
_startCapturingTelemetry
);
// Captures recordingChanged() signals to stop writing subtitles
connect
(
_videoReceiver
,
&
VideoReceiver
::
recordingChanged
,
this
,
&
SubtitleWriter
::
_onVideoRecordingChanged
);
// Timer for telemetry capture and writing to file
connect
(
&
_timer
,
&
QTimer
::
timeout
,
this
,
&
SubtitleWriter
::
_captureTelemetry
);
}
void
SubtitleWriter
::
_onVideoRecordingChanged
()
{
// Stop capturing data if recording stopped
if
(
!
_videoReceiver
->
recording
())
{
qCDebug
(
SubtitleWriterLog
)
<<
"Stopping writing"
;
_timer
.
stop
();
_file
.
close
();
}
}
void
SubtitleWriter
::
_startCapturingTelemetry
()
{
if
(
!
_videoReceiver
)
{
qCWarning
(
SubtitleWriterLog
)
<<
"Invalid VideoReceiver pointer! Aborting subtitle capture!"
;
_timer
.
stop
();
return
;
}
// Get the facts displayed in the values widget and capture them, removing the "Vehicle." prefix.
QSettings
settings
;
settings
.
beginGroup
(
"ValuesWidget"
);
_values
=
settings
.
value
(
"large"
).
toStringList
().
replaceInStrings
(
QStringLiteral
(
"Vehicle."
),
QString
());
_values
+=
settings
.
value
(
"small"
).
toStringList
().
replaceInStrings
(
QStringLiteral
(
"Vehicle."
),
QString
());
_startTime
=
QDateTime
::
currentDateTime
();
QFileInfo
videoFile
(
_videoReceiver
->
videoFile
());
QString
subtitleFilePath
=
QStringLiteral
(
"%1/%2.ass"
).
arg
(
videoFile
.
path
(),
videoFile
.
completeBaseName
());
qCDebug
(
SubtitleWriterLog
)
<<
"Writing overlay to file:"
<<
subtitleFilePath
;
_file
.
setFileName
(
subtitleFilePath
);
if
(
!
_file
.
open
(
QIODevice
::
ReadWrite
))
{
qCWarning
(
SubtitleWriterLog
)
<<
"Unable to write subtitle data to file"
;
return
;
}
QTextStream
stream
(
&
_file
);
// This is file header
stream
<<
QStringLiteral
(
"[Script Info]
\n
"
"Title: QGroundControl Subtitle Telemetry file
\n
"
"ScriptType: v4.00+
\n
"
"WrapStyle: 0
\n
"
"ScaledBorderAndShadow: yes
\n
"
"YCbCr Matrix: TV.601
\n
"
"PlayResX: 1920
\n
"
"PlayResY: 1080
\n
"
"
\n
"
"[V4+ Styles]
\n
"
"Format: Name, Fontname, Fontsize, PrimaryColour, SecondaryColour, OutlineColour, BackColour, Bold, Italic, Underline, StrikeOut, ScaleX, ScaleY, Spacing, Angle, BorderStyle, Outline, Shadow, Alignment, MarginL, MarginR, MarginV, Encoding
\n
"
"Style: Default,Monospace,30,&H00FFFFFF,&H000000FF,&H00000000,&H00000000,0,0,0,0,100,100,0,0,1,2,2,1,10,10,10,1
\n
"
"
\n
"
"[Events]
\n
"
"Format: Layer, Start, End, Style, Name, MarginL, MarginR, MarginV, Effect, Text
\n
"
);
// TODO: Find a good way to input title
//stream << QStringLiteral("Dialogue: 0,0:00:00.00,999:00:00.00,Default,,0,0,0,,{\\pos(5,35)}%1\n");
_timer
.
start
(
1000
/
_sampleRate
);
}
void
SubtitleWriter
::
_captureTelemetry
()
{
if
(
!
_videoReceiver
)
{
qCWarning
(
SubtitleWriterLog
)
<<
"Invalid VideoReceiver pointer! Aborting subtitle capture!"
;
_timer
.
stop
();
return
;
}
static
const
float
nRows
=
3
;
// number of rows used for displaying data
static
const
int
offsetFactor
=
700
;
// Used to simulate a larger resolution and reduce the borders in the layout
auto
*
vehicle
=
qgcApp
()
->
toolbox
()
->
multiVehicleManager
()
->
activeVehicle
();
if
(
!
vehicle
)
{
qCWarning
(
SubtitleWriterLog
)
<<
"Attempting to capture fact data with no active vehicle!"
;
return
;
}
// Each list corresponds to a column in the subtitles
QStringList
namesStrings
;
QStringList
valuesStrings
;
// Make a list of "factname:" strings and other with the values, so one can be aligned left and the other right
for
(
const
auto
&
i
:
_values
)
{
valuesStrings
<<
QStringLiteral
(
"%2 %3"
).
arg
(
vehicle
->
getFact
(
i
)
->
cookedValueString
())
.
arg
(
vehicle
->
getFact
(
i
)
->
cookedUnits
());
namesStrings
<<
QStringLiteral
(
"%1:"
).
arg
(
vehicle
->
getFact
(
i
)
->
shortDescription
());
}
// The time to start displaying this subtitle text
QTime
start
=
QTime
(
0
,
0
).
addMSecs
(
_startTime
.
time
().
msecsTo
(
QDateTime
::
currentDateTime
().
time
()));
// The time to stop displaying this subtitle text
QTime
end
=
start
.
addMSecs
(
1000
/
_sampleRate
);
// This splits the screen in N parts and uses the N-1 internal parts to align the subtitles to.
// Should we try to get the resolution from the pipeline? This seems to work fine with other resolutions too.
static
const
int
rowWidth
=
(
1920
+
offsetFactor
)
/
(
nRows
+
1
);
int
nValuesByRow
=
ceil
(
_values
.
length
()
/
nRows
);
QList
<
QStringList
>
dataColumns
;
QStringList
stringColumns
;
// These templates are used for the data columns, one right-aligned for names and one for
// the facts values. The arguments expected are: start time, end time, xposition, and string content.
QString
namesLine
=
QStringLiteral
(
"Dialogue: 0,%2,%3,Default,,0,0,0,,{
\\
an3
\\
pos(%1,1075)}%4
\n
"
);
QString
valuesLine
=
QStringLiteral
(
"Dialogue: 0,%2,%3,Default,,0,0,0,,{
\\
pos(%1,1075)}%4
\n
"
);
// Split values into N columns and create a subtitle entry for each column
for
(
int
i
=
0
;
i
<
nRows
;
i
++
)
{
QStringList
currentColumnNameStrings
=
namesStrings
.
mid
((
i
)
*
nValuesByRow
,
nValuesByRow
);
QStringList
currentColumnValueStrings
=
valuesStrings
.
mid
((
i
)
*
nValuesByRow
,
nValuesByRow
);
// Fill templates for names of column i
QString
names
=
namesLine
.
arg
(
-
offsetFactor
/
2
+
rowWidth
*
(
i
+
1
)
-
10
)
.
arg
(
start
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
end
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
currentColumnNameStrings
.
join
(
"
\\
N"
));
stringColumns
<<
names
;
// Fill templates for values of column i
QString
values
=
valuesLine
.
arg
(
-
offsetFactor
/
2
+
rowWidth
*
(
i
+
1
))
.
arg
(
start
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
end
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
currentColumnValueStrings
.
join
(
"
\\
N"
));
stringColumns
<<
values
;
}
// Write the date to the corner
stringColumns
<<
QStringLiteral
(
"Dialogue: 0,%1,%2,Default,,0,0,0,,{
\\
pos(10,35)}%3
\n
"
)
.
arg
(
start
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
end
.
toString
(
"H:mm:ss.zzz"
).
chopped
(
2
))
.
arg
(
QDateTime
::
currentDateTime
().
toString
(
Qt
::
SystemLocaleShortDate
));
// Write new data
QTextStream
stream
(
&
_file
);
for
(
const
auto
&
i
:
stringColumns
)
{
stream
<<
i
;
}
}
src/VideoStreaming/SubtitleWriter.h
0 → 100644
View file @
d3315c2c
/****************************************************************************
*
* (c) 2009-2019 QGROUNDCONTROL PROJECT <http://www.qgroundcontrol.org>
*
* QGroundControl is licensed according to the terms in the file
* COPYING.md in the root of the source code directory.
*
****************************************************************************/
/**
* @file
* @brief QGC Video Subtitle Writer
* @author Willian Galvani <williangalvani@gmail.com>
*/
#pragma once
#include "QGCLoggingCategory.h"
#include "VideoReceiver.h"
#include <QObject>
#include <QTimer>
#include <QDateTime>
#include <QFile>
Q_DECLARE_LOGGING_CATEGORY
(
SubtitleWriterLog
)
class
SubtitleWriter
:
public
QObject
{
Q_OBJECT
public:
explicit
SubtitleWriter
(
QObject
*
parent
=
nullptr
);
~
SubtitleWriter
()
=
default
;
void
setVideoReceiver
(
VideoReceiver
*
videoReceiver
);
private
slots
:
// Fires with every "videoRecordingChanged() signal, stops capturing telemetry if video stopped."
void
_onVideoRecordingChanged
();
// Captures a snapshot of telemetry data from vehicle into the subtitles file.
void
_captureTelemetry
();
// starts capturing vehicle telemetry.
void
_startCapturingTelemetry
();
private:
QTimer
_timer
;
QStringList
_values
;
QDateTime
_startTime
;
QFile
_file
;
VideoReceiver
*
_videoReceiver
;
static
const
int
_sampleRate
;
};
src/VideoStreaming/VideoReceiver.cc
View file @
d3315c2c
...
...
@@ -828,6 +828,7 @@ VideoReceiver::_keyframeWatch(GstPad* pad, GstPadProbeInfo* info, gpointer user_
gst_element_sync_state_with_parent
(
pThis
->
_sink
->
filesink
);
qCDebug
(
VideoReceiverLog
)
<<
"Got keyframe, stop dropping buffers"
;
pThis
->
gotFirstRecordingKeyFrame
();
}
}
...
...
src/VideoStreaming/VideoReceiver.h
View file @
d3315c2c
...
...
@@ -79,6 +79,7 @@ signals:
void
msgErrorReceived
();
void
msgEOSReceived
();
void
msgStateChangedReceived
();
void
gotFirstRecordingKeyFrame
();
#endif
public
slots
:
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment