mirror of
https://gitlab.freedesktop.org/gstreamer/gstreamer.git
synced 2024-12-28 03:00:35 +00:00
1de720ad78
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>
188 lines
7.3 KiB
Python
Executable file
188 lines
7.3 KiB
Python
Executable file
#!/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)
|