mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-11-26 03:31:05 +00:00
python: Add gst_video_converter.py example
Add an example of using compositor and the samples-selected signal to do frame-by-frame animation by updating pad properties and the pad's GstVideoConverter configuration Part-of: <https://gitlab.freedesktop.org/gstreamer/gstreamer/-/merge_requests/5187>
This commit is contained in:
parent
9b35e0ad26
commit
1de720ad78
1 changed files with 188 additions and 0 deletions
188
subprojects/gst-python/examples/gst_video_converter.py
Executable file
188
subprojects/gst-python/examples/gst_video_converter.py
Executable file
|
@ -0,0 +1,188 @@
|
|||
#!/usr/bin/env python3
|
||||
# Demonstration of using compositor and the samples-selected
|
||||
# signal to do frame-by-frame updates and animation by
|
||||
# udpating compositor pad properties and the GstVideoConverter
|
||||
# config.
|
||||
#
|
||||
# Supply a URI argument to use a video file in the example,
|
||||
# or omit it to just animate a videotestsrc.
|
||||
import gi
|
||||
import math
|
||||
import sys
|
||||
|
||||
gi.require_version('Gst', '1.0')
|
||||
gi.require_version('GstVideo', '1.0')
|
||||
gi.require_version('GLib', '2.0')
|
||||
gi.require_version('GObject', '2.0')
|
||||
from gi.repository import GLib, GObject, Gst, GstVideo
|
||||
|
||||
|
||||
def bus_call(bus, message, loop):
|
||||
t = message.type
|
||||
if t == Gst.MessageType.EOS:
|
||||
sys.stdout.write("End-of-stream\n")
|
||||
loop.quit()
|
||||
elif t == Gst.MessageType.ERROR:
|
||||
err, debug = message.parse_error()
|
||||
sys.stderr.write("Error: %s: %s\n" % (err, debug))
|
||||
loop.quit()
|
||||
return True
|
||||
|
||||
|
||||
def update_compositor(comp, seg, pts, dts, duration, info):
|
||||
sink = comp.get_static_pad("sink_1")
|
||||
# Can peek at the input sample(s) here to get the real video input caps
|
||||
sample = comp.peek_next_sample(sink)
|
||||
caps = sample.get_caps()
|
||||
in_w = caps[0]['width']
|
||||
in_h = caps[0]['height']
|
||||
|
||||
dest_w = 160 + in_w * math.fabs(math.sin(pts / (10 * Gst.SECOND) * math.pi))
|
||||
dest_h = 120 + in_h * math.fabs(math.sin(pts / (11 * Gst.SECOND) * math.pi))
|
||||
x = (1920 - dest_w) * math.fabs(math.sin(pts / (5 * Gst.SECOND) * math.pi))
|
||||
y = (1080 - dest_h) * math.fabs(math.sin(pts / (7 * Gst.SECOND) * math.pi))
|
||||
|
||||
sink.set_property("xpos", x)
|
||||
sink.set_property("ypos", y)
|
||||
sink.set_property("width", dest_w)
|
||||
sink.set_property("height", dest_h)
|
||||
|
||||
# Update video-converter settings
|
||||
cfg = Gst.Structure.new_empty("GstVideoConverter")
|
||||
|
||||
# When scaling down, switch to nearest-neighbour scaling, otherwise use bilinear
|
||||
if in_w < dest_w or in_h < dest_h:
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_RESAMPLER_METHOD, GstVideo.VideoResamplerMethod.NEAREST)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_CHROMA_RESAMPLER_METHOD, GstVideo.VideoResamplerMethod.NEAREST)
|
||||
else:
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_RESAMPLER_METHOD, GstVideo.VideoResamplerMethod.LINEAR)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_CHROMA_RESAMPLER_METHOD, GstVideo.VideoResamplerMethod.LINEAR)
|
||||
|
||||
# Crop some from the top and bottom or sides of the source image
|
||||
crop = 64 * math.cos(pts / (4 * Gst.SECOND) * math.pi)
|
||||
if crop < 0:
|
||||
crop = min(in_w / 2, -crop)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_SRC_X, int(crop))
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_SRC_WIDTH, int(in_w - 2 * crop))
|
||||
else:
|
||||
crop = min(in_h / 2, crop)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_SRC_Y, int(crop))
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_SRC_HEIGHT, int(in_h - 2 * crop))
|
||||
|
||||
# Add add some borders to the result by not filling the destination rect
|
||||
border = 64 * math.sin(pts / (4 * Gst.SECOND) * math.pi)
|
||||
if border < 0:
|
||||
border = min(in_w / 2, -border)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DEST_X, int(border))
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DEST_WIDTH, int(dest_w - 2 * border))
|
||||
else:
|
||||
border = min(in_h / 2, border)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DEST_Y, int(border))
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DEST_HEIGHT, int(dest_h - 2 * border))
|
||||
|
||||
# Set the border colour. Need to set this into a GValue explicitly to ensure it's a uint
|
||||
argb = GObject.Value()
|
||||
argb.init(GObject.TYPE_UINT)
|
||||
argb.set_uint(0x80FF80FF)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_BORDER_ARGB, argb)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_FILL_BORDER, True)
|
||||
|
||||
# When things get wide, do some dithering why not
|
||||
if dest_w > in_w:
|
||||
# Dither quantization must also be a uint
|
||||
dither = GObject.Value()
|
||||
dither.init(GObject.TYPE_UINT)
|
||||
dither.set_uint(64)
|
||||
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DITHER_METHOD, GstVideo.VideoDitherMethod.FLOYD_STEINBERG)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DITHER_QUANTIZATION, dither)
|
||||
else:
|
||||
dither = GObject.Value()
|
||||
dither.init(GObject.TYPE_UINT)
|
||||
dither.set_uint(1)
|
||||
|
||||
# cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DITHER_METHOD, GstVideo.VideoDitherMethod.NONE)
|
||||
cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_DITHER_QUANTIZATION, dither)
|
||||
|
||||
# and fade it in and out every 13 seconds
|
||||
alpha = 0.25 + 0.75 * math.fabs(math.sin(pts / (13 * Gst.SECOND) * math.pi))
|
||||
# Use the video-converter to render alpha
|
||||
# cfg.set_value(GstVideo.VIDEO_CONVERTER_OPT_ALPHA_VALUE, alpha)
|
||||
# or use the pad's alpha setting is better because compositor can use it to skip rendering
|
||||
sink.set_property("alpha", alpha)
|
||||
|
||||
# print(f"Setting config {cfg}")
|
||||
sink.set_property("converter-config", cfg)
|
||||
|
||||
|
||||
if __name__ == "__main__":
|
||||
Gst.init(sys.argv)
|
||||
|
||||
pipeline = Gst.ElementFactory.make('pipeline', None)
|
||||
|
||||
compositor = Gst.ElementFactory.make("compositor", None)
|
||||
compositor.set_property("emit-signals", True)
|
||||
compositor.set_property("background", "black")
|
||||
compositor.connect("samples-selected", update_compositor)
|
||||
|
||||
cf = Gst.ElementFactory.make("capsfilter", None)
|
||||
# Need to make sure the output has RGBA or AYUV, or we can't do alpha blending / fades:
|
||||
caps = Gst.Caps.from_string("video/x-raw,width=1920,height=1080,framerate=25/1,format=RGBA")
|
||||
cf.set_property("caps", caps)
|
||||
|
||||
conv = Gst.ElementFactory.make("videoconvert", None)
|
||||
sink = Gst.ElementFactory.make("autovideosink", None)
|
||||
|
||||
pipeline.add(compositor, cf, conv, sink)
|
||||
Gst.Element.link_many(compositor, cf, conv, sink)
|
||||
|
||||
bgsource = Gst.parse_bin_from_description("videotestsrc pattern=circular ! capsfilter name=cf", False)
|
||||
|
||||
cfsrc = bgsource.get_by_name("cf")
|
||||
caps = Gst.Caps.from_string("video/x-raw,width=320,height=180,framerate=1/1,format=RGB")
|
||||
cfsrc.set_property("caps", caps)
|
||||
src = cfsrc.get_static_pad("src")
|
||||
bgsource.add_pad(Gst.GhostPad.new("src", src))
|
||||
|
||||
pipeline.add(bgsource)
|
||||
bgsource.link(compositor)
|
||||
|
||||
pad = compositor.get_static_pad("sink_0")
|
||||
pad.set_property("width", 1920)
|
||||
pad.set_property("height", 1080)
|
||||
|
||||
if len(sys.argv) > 1:
|
||||
source = Gst.parse_bin_from_description("uridecodebin name=u ! capsfilter name=cf caps=video/x-raw", False)
|
||||
u = source.get_by_name("u")
|
||||
u.set_property("uri", sys.argv[1])
|
||||
|
||||
cfsrc = source.get_by_name("cf")
|
||||
caps = Gst.Caps.from_string("video/x-raw")
|
||||
cfsrc.set_property("caps", caps)
|
||||
|
||||
# Expose the capsfilter source pad as a ghost pad
|
||||
src = cfsrc.get_static_pad("src")
|
||||
source.add_pad(Gst.GhostPad.new("src", src))
|
||||
else:
|
||||
source = Gst.parse_bin_from_description("videotestsrc ! capsfilter name=cf", False)
|
||||
cfsrc = source.get_by_name("cf")
|
||||
caps = Gst.Caps.from_string("video/x-raw,width=320,height=240,framerate=30/1,format=I420")
|
||||
cfsrc.set_property("caps", caps)
|
||||
|
||||
src = cfsrc.get_static_pad("src")
|
||||
source.add_pad(Gst.GhostPad.new("src", src))
|
||||
|
||||
pipeline.add(source)
|
||||
source.link(compositor)
|
||||
|
||||
pipeline.set_state(Gst.State.PLAYING)
|
||||
|
||||
bus = pipeline.get_bus()
|
||||
bus.add_signal_watch()
|
||||
|
||||
loop = GLib.MainLoop()
|
||||
bus.connect("message", bus_call, loop)
|
||||
loop.run()
|
||||
|
||||
pipeline.set_state(Gst.State.NULL)
|
||||
pipeline.get_state(Gst.CLOCK_TIME_NONE)
|
Loading…
Reference in a new issue