124 lines
3.9 KiB
Python
124 lines
3.9 KiB
Python
import cython
|
|
from cython.cimports import libav as lib
|
|
from cython.cimports.av.packet import Packet
|
|
from cython.cimports.av.utils import avrational_to_fraction, to_avrational
|
|
from cython.cimports.av.video.frame import VideoFrame
|
|
|
|
|
|
@cython.cclass
|
|
class VideoStream(Stream):
|
|
def __repr__(self):
|
|
return (
|
|
f"<av.VideoStream #{self.index} {self.name}, "
|
|
f"{self.format.name if self.format else None} {self.codec_context.width}x"
|
|
f"{self.codec_context.height} at 0x{id(self):x}>"
|
|
)
|
|
|
|
def __getattr__(self, name):
|
|
if name in ("framerate", "rate"):
|
|
raise AttributeError(
|
|
f"'{type(self).__name__}' object has no attribute '{name}'"
|
|
)
|
|
|
|
return getattr(self.codec_context, name)
|
|
|
|
@cython.ccall
|
|
def encode(self, frame: VideoFrame | None = None):
|
|
"""
|
|
Encode an :class:`.VideoFrame` and return a list of :class:`.Packet`.
|
|
|
|
:rtype: list[Packet]
|
|
|
|
.. seealso:: This is mostly a passthrough to :meth:`.CodecContext.encode`.
|
|
"""
|
|
|
|
packets = self.codec_context.encode(frame)
|
|
packet: Packet
|
|
for packet in packets:
|
|
packet._stream = self
|
|
packet.ptr.stream_index = self.ptr.index
|
|
return packets
|
|
|
|
@cython.ccall
|
|
def decode(self, packet: Packet | None = None):
|
|
"""
|
|
Decode a :class:`.Packet` and return a list of :class:`.VideoFrame`.
|
|
|
|
:rtype: list[VideoFrame]
|
|
|
|
.. seealso:: This is a passthrough to :meth:`.CodecContext.decode`.
|
|
"""
|
|
return self.codec_context.decode(packet)
|
|
|
|
@property
|
|
def average_rate(self):
|
|
"""
|
|
The average frame rate of this video stream.
|
|
|
|
This is calculated when the file is opened by looking at the first
|
|
few frames and averaging their rate.
|
|
|
|
:type: fractions.Fraction | None
|
|
"""
|
|
return avrational_to_fraction(cython.address(self.ptr.avg_frame_rate))
|
|
|
|
@property
|
|
def base_rate(self):
|
|
"""
|
|
The base frame rate of this stream.
|
|
|
|
This is calculated as the lowest framerate at which the timestamps of
|
|
frames can be represented accurately. See :ffmpeg:`AVStream.r_frame_rate`
|
|
for more.
|
|
|
|
:type: fractions.Fraction | None
|
|
"""
|
|
return avrational_to_fraction(cython.address(self.ptr.r_frame_rate))
|
|
|
|
@property
|
|
def guessed_rate(self):
|
|
"""The guessed frame rate of this stream.
|
|
|
|
This is a wrapper around :ffmpeg:`av_guess_frame_rate`, and uses multiple
|
|
heuristics to decide what is "the" frame rate.
|
|
|
|
:type: fractions.Fraction | None
|
|
"""
|
|
val: lib.AVRational = lib.av_guess_frame_rate(
|
|
cython.NULL, self.ptr, cython.NULL
|
|
)
|
|
return avrational_to_fraction(cython.address(val))
|
|
|
|
@property
|
|
def sample_aspect_ratio(self):
|
|
"""The guessed sample aspect ratio (SAR) of this stream.
|
|
|
|
This is a wrapper around :ffmpeg:`av_guess_sample_aspect_ratio`, and uses multiple
|
|
heuristics to decide what is "the" sample aspect ratio.
|
|
|
|
:type: fractions.Fraction | None
|
|
"""
|
|
sar: lib.AVRational = lib.av_guess_sample_aspect_ratio(
|
|
self.container.ptr, self.ptr, cython.NULL
|
|
)
|
|
return avrational_to_fraction(cython.address(sar))
|
|
|
|
@property
|
|
def display_aspect_ratio(self):
|
|
"""The guessed display aspect ratio (DAR) of this stream.
|
|
|
|
This is calculated from :meth:`.VideoStream.guessed_sample_aspect_ratio`.
|
|
|
|
:type: fractions.Fraction | None
|
|
"""
|
|
dar = cython.declare(lib.AVRational)
|
|
lib.av_reduce(
|
|
cython.address(dar.num),
|
|
cython.address(dar.den),
|
|
self.format.width * self.sample_aspect_ratio.num,
|
|
self.format.height * self.sample_aspect_ratio.den,
|
|
1024 * 1024,
|
|
)
|
|
|
|
return avrational_to_fraction(cython.address(dar))
|