Compare commits
4 Commits
e803956d75
...
proto
| Author | SHA1 | Date | |
|---|---|---|---|
|
0ec8b77554
|
|||
|
b6196c1d46
|
|||
|
fff635f5c2
|
|||
|
be3b56856d
|
53
pve.py
53
pve.py
@ -1,53 +0,0 @@
|
|||||||
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
|
||||||
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
|
||||||
#
|
|
||||||
# This file is part of People's Video Editor.
|
|
||||||
#
|
|
||||||
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
|
||||||
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
|
||||||
# or (at your option) any later version.
|
|
||||||
#
|
|
||||||
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
|
||||||
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
|
||||||
# See the GNU General Public License for more details.
|
|
||||||
#
|
|
||||||
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
|
||||||
# If not, see <https://www.gnu.org/licenses/>.
|
|
||||||
|
|
||||||
import multiprocessing
|
|
||||||
import cProfile
|
|
||||||
|
|
||||||
from pyav.demuxer import Demuxer
|
|
||||||
from pyav.decoder import Decoder
|
|
||||||
|
|
||||||
demuxer = Demuxer('test.mp4')
|
|
||||||
|
|
||||||
print(f"nb_streams = {demuxer.nb_streams}")
|
|
||||||
print(f"video codec = {demuxer.video_stream.codec.description()}")
|
|
||||||
print(f"audio codec = {demuxer.audio_stream.codec.description()}")
|
|
||||||
|
|
||||||
cpu_count = multiprocessing.cpu_count()
|
|
||||||
|
|
||||||
print(f"using {cpu_count} threads for video decoding")
|
|
||||||
|
|
||||||
video_decoder = Decoder(demuxer.video_stream, cpu_count)
|
|
||||||
audio_decoder = Decoder(demuxer.audio_stream, 1)
|
|
||||||
|
|
||||||
num_video_frames = 0
|
|
||||||
num_audio_frames = 0
|
|
||||||
|
|
||||||
with cProfile.Profile() as pr:
|
|
||||||
while True:
|
|
||||||
packet = demuxer.read_packet()
|
|
||||||
eof = (packet is None)
|
|
||||||
if eof or demuxer.video_stream.contains(packet):
|
|
||||||
video_frames = video_decoder.decode(packet)
|
|
||||||
num_video_frames += len(video_frames)
|
|
||||||
if eof or demuxer.audio_stream.contains(packet):
|
|
||||||
audio_frames = audio_decoder.decode(packet)
|
|
||||||
num_audio_frames += len(audio_frames)
|
|
||||||
if eof:
|
|
||||||
break
|
|
||||||
|
|
||||||
print(f"num video frames: {num_video_frames}, audio frames: {num_audio_frames}")
|
|
||||||
pr.print_stats()
|
|
||||||
15
pve/__init__.py
Normal file
15
pve/__init__.py
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
100
pve/node.py
Normal file
100
pve/node.py
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from weakref import ref
|
||||||
|
|
||||||
|
class Node:
|
||||||
|
__slots__ = 'parent', 'children', 'state', '__weakref__'
|
||||||
|
|
||||||
|
STATE_NEW = 0
|
||||||
|
STATE_MODIFIED = 1
|
||||||
|
STATE_BUILT = 2
|
||||||
|
|
||||||
|
def __init__(self, parent = None, children = None):
|
||||||
|
self.parent = None
|
||||||
|
self.children = None
|
||||||
|
self.state = Node.STATE_NEW
|
||||||
|
if children is not None:
|
||||||
|
self.add_children(children)
|
||||||
|
if parent is not None:
|
||||||
|
parent.add_child(self)
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
if self.parent is not None:
|
||||||
|
self.parent().remove_child(self)
|
||||||
|
|
||||||
|
def add_child(self, child: Node):
|
||||||
|
assert child.parent is None
|
||||||
|
child.parent = ref(self)
|
||||||
|
if self.children is None:
|
||||||
|
self.children = [child]
|
||||||
|
else:
|
||||||
|
self.children.append(child)
|
||||||
|
child.state = Node.STATE_NEW
|
||||||
|
self.set_modified()
|
||||||
|
|
||||||
|
def add_children(self, children):
|
||||||
|
for child in children:
|
||||||
|
self.add_child(child)
|
||||||
|
|
||||||
|
def remove_child(self, child: Node):
|
||||||
|
assert child.parent is self
|
||||||
|
child.parent = None
|
||||||
|
self.children.remove(child)
|
||||||
|
self.set_modified()
|
||||||
|
|
||||||
|
def remove_children(self, children):
|
||||||
|
for child in children:
|
||||||
|
self.remove_child(child)
|
||||||
|
|
||||||
|
def is_new(self):
|
||||||
|
return self.state == Node.STATE_NEW
|
||||||
|
|
||||||
|
def needs_rebuilding(self):
|
||||||
|
return self.state != Node.STATE_BUILT
|
||||||
|
|
||||||
|
def set_modified(self):
|
||||||
|
if self.needs_rebuilding():
|
||||||
|
return
|
||||||
|
self.state = Node.STATE_MODIFIED
|
||||||
|
if self.parent is not None:
|
||||||
|
self.set_parent_modified(self.parent())
|
||||||
|
|
||||||
|
def set_parent_modified(self, parent):
|
||||||
|
parent.set_modified()
|
||||||
|
|
||||||
|
def rebuild(self):
|
||||||
|
if not self.needs_rebuilding():
|
||||||
|
return True
|
||||||
|
if self.children is not None:
|
||||||
|
if not self.rebuild_children(self.children):
|
||||||
|
return False
|
||||||
|
if not self.build():
|
||||||
|
return False
|
||||||
|
self.state = Node.STATE_BUILT
|
||||||
|
return True
|
||||||
|
|
||||||
|
def rebuild_children(self, children):
|
||||||
|
for child in children:
|
||||||
|
if not self.rebuild_child(child):
|
||||||
|
return False
|
||||||
|
return True
|
||||||
|
|
||||||
|
def rebuild_child(self, child):
|
||||||
|
return child.rebuild()
|
||||||
|
|
||||||
|
def build(self):
|
||||||
|
return True
|
||||||
131
pve/pixmap.py
Normal file
131
pve/pixmap.py
Normal file
@ -0,0 +1,131 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from fractions import Fraction
|
||||||
|
|
||||||
|
class Geometry:
|
||||||
|
__slots__ = '_width', '_height', '_ratio', '_size'
|
||||||
|
|
||||||
|
def __init__(self, width = 0, height = 0, ratio = None):
|
||||||
|
if width > 0 and height > 0:
|
||||||
|
assert ratio is None
|
||||||
|
self._width = width
|
||||||
|
self._height = height
|
||||||
|
self._ratio = Fraction(width, height).limit_denominator()
|
||||||
|
self._size = width * height
|
||||||
|
elif width > 0 and ratio:
|
||||||
|
self._width = width
|
||||||
|
self._height = round((width * ratio.numerator) / ratio.denominator)
|
||||||
|
self._ratio = ratio
|
||||||
|
self._size = width * self._height
|
||||||
|
elif height > 0 and ratio:
|
||||||
|
self._width = round((height * ratio.denominator) / ratio.numerator)
|
||||||
|
self._height = height
|
||||||
|
self._ratio = ratio
|
||||||
|
self._size = self._width * height
|
||||||
|
else:
|
||||||
|
raise ValueError
|
||||||
|
|
||||||
|
def __ge__(self, other):
|
||||||
|
return (self._width >= other._width and self._height >= other._height)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def width(self):
|
||||||
|
return self._width
|
||||||
|
|
||||||
|
@property
|
||||||
|
def height(self):
|
||||||
|
return self._height
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ratio(self):
|
||||||
|
return self._ratio
|
||||||
|
|
||||||
|
class Format:
|
||||||
|
__slots__ = '_channels', '_depth', '_type', '_size'
|
||||||
|
|
||||||
|
def __init__(self, channels, depth):
|
||||||
|
self._channels = channels
|
||||||
|
self._depth = depth
|
||||||
|
if depth == 8:
|
||||||
|
self._type = 'B'
|
||||||
|
self._size = channels
|
||||||
|
elif depth == 16:
|
||||||
|
self._type = 'H'
|
||||||
|
self._size = channels * 2
|
||||||
|
else:
|
||||||
|
raise ValueError
|
||||||
|
|
||||||
|
def __eq__(self, other):
|
||||||
|
return (self._channels == other._channels and self._depth == other._depth)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channels(self):
|
||||||
|
return self._channels
|
||||||
|
|
||||||
|
@property
|
||||||
|
def depth(self):
|
||||||
|
return self._depth
|
||||||
|
|
||||||
|
class Pixmap:
|
||||||
|
__slots__ = '_geometry', '_format', '_line_size', '_bytes', '_view'
|
||||||
|
|
||||||
|
def __init__(self, geometry, format):
|
||||||
|
self._geometry = geometry
|
||||||
|
self._format = format
|
||||||
|
self._line_size = geometry._width * format._size
|
||||||
|
self._bytes = bytearray(geometry._size * format._size)
|
||||||
|
self._view = memoryview(self._bytes)
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
del self._view
|
||||||
|
del self._bytes
|
||||||
|
|
||||||
|
@property
|
||||||
|
def geometry(self):
|
||||||
|
return self._geometry
|
||||||
|
|
||||||
|
@property
|
||||||
|
def width(self):
|
||||||
|
return self._geometry._width
|
||||||
|
|
||||||
|
@property
|
||||||
|
def height(self):
|
||||||
|
return self._geometry._height
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ratio(self):
|
||||||
|
return self._geometry._ratio
|
||||||
|
|
||||||
|
@property
|
||||||
|
def format(self):
|
||||||
|
return self._format
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channels(self):
|
||||||
|
return self._format._channels
|
||||||
|
|
||||||
|
@property
|
||||||
|
def depth(self):
|
||||||
|
return self._format._depth
|
||||||
|
|
||||||
|
@property
|
||||||
|
def line_size(self):
|
||||||
|
return self._line_size
|
||||||
|
|
||||||
|
@property
|
||||||
|
def view(self):
|
||||||
|
return self._view
|
||||||
107
pve/polygon.py
Normal file
107
pve/polygon.py
Normal file
@ -0,0 +1,107 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from array import array
|
||||||
|
|
||||||
|
from .node import Node
|
||||||
|
from .vector import Vector2
|
||||||
|
|
||||||
|
class Point:
|
||||||
|
__slots__ = 'position', 'mapping'
|
||||||
|
|
||||||
|
def __init__(self, xy = Vector2(), uv = Vector2()):
|
||||||
|
assert isinstance(xy, Vector2) and isinstance(uv, Vector2)
|
||||||
|
self.position = xy
|
||||||
|
self.mapping = uv
|
||||||
|
|
||||||
|
class Polygon(Node):
|
||||||
|
__slots__ = '_positions', '_mappings', '_len'
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def rectangle(cls, geometry):
|
||||||
|
tl = Vector2(0.0, 0.0)
|
||||||
|
tr = Vector2(geometry._width, 0.0)
|
||||||
|
br = Vector2(geometry._width, geometry._height)
|
||||||
|
bl = Vector2(0.0, geometry._height)
|
||||||
|
return cls(points = (
|
||||||
|
Point(tl, tl),
|
||||||
|
Point(tr, tr),
|
||||||
|
Point(br, br),
|
||||||
|
Point(bl, bl)))
|
||||||
|
|
||||||
|
def __init__(self, parent = None, children = None, size = 0, points = None):
|
||||||
|
Node.__init__(self, parent, children)
|
||||||
|
self.clear_points()
|
||||||
|
self.add_points(size, points)
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return self._len
|
||||||
|
|
||||||
|
def __getitem__(self, index):
|
||||||
|
begin = index * 2
|
||||||
|
end = begin + 2
|
||||||
|
return Point(
|
||||||
|
Vector2(*self._positions[begin:end]),
|
||||||
|
Vector2(*self._mappings[begin:end]))
|
||||||
|
|
||||||
|
def __setitem__(self, index, point):
|
||||||
|
begin = index * 2
|
||||||
|
end = begin + 2
|
||||||
|
self._positions[begin:end] = array(Vector2.TYPE, point.position)
|
||||||
|
self._mappings[begin:end] = array(Vector2.TYPE, point.mapping)
|
||||||
|
|
||||||
|
def add_child(self, child):
|
||||||
|
assert isinstance(child, Polygon)
|
||||||
|
Node.add_child(self, child)
|
||||||
|
|
||||||
|
def add_points(self, count = 0, points = None):
|
||||||
|
assert (count > 0) ^ len(points)
|
||||||
|
if count:
|
||||||
|
initializer = bytes(count * Vector2.SIZE)
|
||||||
|
self._positions.frombytes(initializer)
|
||||||
|
self._mappings.frombytes(initializer)
|
||||||
|
self._len += count
|
||||||
|
else:
|
||||||
|
for point in points:
|
||||||
|
self._positions.extend(point.position)
|
||||||
|
self._mappings.extend(point.mapping)
|
||||||
|
self._len += len(points)
|
||||||
|
|
||||||
|
def insert_points(self, index, count = 0, points = None):
|
||||||
|
assert (count == 0) ^ (points is None)
|
||||||
|
index *= 2
|
||||||
|
if count:
|
||||||
|
initializer = array(Vector2.TYPE, bytes(count * Vector2.SIZE))
|
||||||
|
self._positions[index:index] = initializer
|
||||||
|
self._mappings[index:index] = initializer
|
||||||
|
self._len += count
|
||||||
|
else:
|
||||||
|
polygon = Polygon(points = points)
|
||||||
|
self._positions[index:index] = polygon._positions
|
||||||
|
self._mappings[index:index] = polygon._mappings
|
||||||
|
self._len += polygon._len
|
||||||
|
|
||||||
|
def remove_points(self, index, count = 1):
|
||||||
|
begin = index * 2
|
||||||
|
end = begin + count * 2
|
||||||
|
del self._positions[begin:end]
|
||||||
|
del self._mappings[begin:end]
|
||||||
|
self._len -= count
|
||||||
|
|
||||||
|
def clear_points(self):
|
||||||
|
self._positions = array(Vector2.TYPE)
|
||||||
|
self._mappings = array(Vector2.TYPE)
|
||||||
|
self._len = 0
|
||||||
26
pve/shape.py
Normal file
26
pve/shape.py
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from .node import Node
|
||||||
|
|
||||||
|
class Shape(Node):
|
||||||
|
|
||||||
|
def __init__(self, parent = None, children = None):
|
||||||
|
Node.__init__(self, parent, children)
|
||||||
|
|
||||||
|
def rebuild(self):
|
||||||
|
#TODO
|
||||||
|
raise NotImplementedError
|
||||||
29
pve/vector.py
Normal file
29
pve/vector.py
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
class Vector2:
|
||||||
|
__slots__ = 'x', 'y'
|
||||||
|
|
||||||
|
TYPE = 'f'
|
||||||
|
SIZE = 8
|
||||||
|
|
||||||
|
def __init__(self, x = 0.0, y = 0.0):
|
||||||
|
assert isinstance(x, float) and isinstance(y, float)
|
||||||
|
self.x = x
|
||||||
|
self.y = y
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
return iter((self.x, self.y))
|
||||||
58
pve_editor.py
Normal file
58
pve_editor.py
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from pyui.application import Application
|
||||||
|
|
||||||
|
application = Application()
|
||||||
|
application.MainLoop()
|
||||||
|
|
||||||
|
# import multiprocessing
|
||||||
|
# import cProfile
|
||||||
|
#
|
||||||
|
# from pyav.demuxer import Demuxer
|
||||||
|
# from pyav.decoder import Decoder
|
||||||
|
#
|
||||||
|
# demuxer = Demuxer('test.mp4')
|
||||||
|
#
|
||||||
|
# print(f"nb_streams = {demuxer.nb_streams}")
|
||||||
|
# print(f"video codec = {demuxer.video_stream.codec.description()}")
|
||||||
|
# print(f"audio codec = {demuxer.audio_stream.codec.description()}")
|
||||||
|
#
|
||||||
|
# cpu_count = multiprocessing.cpu_count()
|
||||||
|
#
|
||||||
|
# print(f"using {cpu_count} threads for video decoding")
|
||||||
|
#
|
||||||
|
# video_decoder = Decoder(demuxer.video_stream, cpu_count)
|
||||||
|
# audio_decoder = Decoder(demuxer.audio_stream, 1)
|
||||||
|
#
|
||||||
|
# num_video_frames = 0
|
||||||
|
# num_audio_frames = 0
|
||||||
|
#
|
||||||
|
# with cProfile.Profile() as pr:
|
||||||
|
# while True:
|
||||||
|
# packet = demuxer.read_packet()
|
||||||
|
# eof = (packet is None)
|
||||||
|
# if eof or demuxer.video_stream.contains(packet):
|
||||||
|
# video_frames = video_decoder.decode(packet)
|
||||||
|
# num_video_frames += len(video_frames)
|
||||||
|
# if eof or demuxer.audio_stream.contains(packet):
|
||||||
|
# audio_frames = audio_decoder.decode(packet)
|
||||||
|
# num_audio_frames += len(audio_frames)
|
||||||
|
# if eof:
|
||||||
|
# break
|
||||||
|
#
|
||||||
|
# print(f"num video frames: {num_video_frames}, audio frames: {num_audio_frames}")
|
||||||
|
# pr.print_stats()
|
||||||
@ -19,4 +19,4 @@ shared_library_path="./pyav/libav"
|
|||||||
absolute_shared_library_path=$(realpath ${shared_library_path})
|
absolute_shared_library_path=$(realpath ${shared_library_path})
|
||||||
|
|
||||||
LD_LIBRARY_PATH=${absolute_shared_library_path} \
|
LD_LIBRARY_PATH=${absolute_shared_library_path} \
|
||||||
python3 pve.py
|
python3 pve_editor.py
|
||||||
15
pygl/__init__.py
Normal file
15
pygl/__init__.py
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
41
pygl/libgl/__init__.py
Normal file
41
pygl/libgl/__init__.py
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
import ctypes
|
||||||
|
|
||||||
|
Texture_p = ctypes.c_void_p
|
||||||
|
VertexBuffer_p = ctypes.c_void_p
|
||||||
|
|
||||||
|
def create_texture(width, height, channels, depth):
|
||||||
|
return None
|
||||||
|
|
||||||
|
def delete_texture(texture):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def upload_texture(texture, view, line_size):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def dowload_texture(texture, view, line_size):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def create_vertex_buffer(size, channels, elems):
|
||||||
|
return None
|
||||||
|
|
||||||
|
def delete_vertex_buffer(buffer):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def upload_vertex_buffer(buffer, views, size):
|
||||||
|
pass
|
||||||
66
pygl/texture.py
Normal file
66
pygl/texture.py
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import libgl
|
||||||
|
|
||||||
|
class Texture:
|
||||||
|
__slots__ = '_geometry', '_format', '_texture'
|
||||||
|
|
||||||
|
def __init__(self, geometry, format):
|
||||||
|
self._geometry = geometry
|
||||||
|
self._format = format
|
||||||
|
self._texture = libgl.create_texture(geometry.width, geometry.height, format.channels, format.depth)
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
libgl.delete_texture(self._texture)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def geometry(self):
|
||||||
|
return self._geometry
|
||||||
|
|
||||||
|
@property
|
||||||
|
def width(self):
|
||||||
|
return self._geometry._width
|
||||||
|
|
||||||
|
@property
|
||||||
|
def height(self):
|
||||||
|
return self._geometry._height
|
||||||
|
|
||||||
|
@property
|
||||||
|
def ratio(self):
|
||||||
|
return self._geometry._ratio
|
||||||
|
|
||||||
|
@property
|
||||||
|
def format(self):
|
||||||
|
return self._format
|
||||||
|
|
||||||
|
@property
|
||||||
|
def channels(self):
|
||||||
|
return self._format._channels
|
||||||
|
|
||||||
|
@property
|
||||||
|
def depth(self):
|
||||||
|
return self._format._depth
|
||||||
|
|
||||||
|
def upload(self, pixmap):
|
||||||
|
assert (self._format == pixmap.format)
|
||||||
|
assert (self._geometry >= pixmap.geometry)
|
||||||
|
return libgl.upload_texture(self._texture, pixmap.view, pixmap.line_size)
|
||||||
|
|
||||||
|
def download(self, pixmap):
|
||||||
|
assert (self._format == pixmap.format)
|
||||||
|
assert (self._geometry >= pixmap.geometry)
|
||||||
|
return libgl.upload_texture(self._texture, pixmap.view, pixmap.line_size)
|
||||||
35
pygl/vertexbuffer.py
Normal file
35
pygl/vertexbuffer.py
Normal file
@ -0,0 +1,35 @@
|
|||||||
|
# People's Video Editor: high quality, GPU accelerated mp4 editor
|
||||||
|
# Copyright (C) 2025 Roz K <roz@rozk.net>
|
||||||
|
#
|
||||||
|
# This file is part of People's Video Editor.
|
||||||
|
#
|
||||||
|
# People's Video Editor is free software: you can redistribute it and/or modify it under the terms of the
|
||||||
|
# GNU General Public License as published by the Free Software Foundation, either version 3 of the License,
|
||||||
|
# or (at your option) any later version.
|
||||||
|
#
|
||||||
|
# People's Video Editor is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
|
||||||
|
# without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||||||
|
# See the GNU General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License along with People's Video Editor.
|
||||||
|
# If not, see <https://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
|
from . import libgl
|
||||||
|
|
||||||
|
class VertexBuffer:
|
||||||
|
__slots__ = '_buffer', '_size'
|
||||||
|
|
||||||
|
def __init__(self, size, channels, elems):
|
||||||
|
self._buffer = libgl.create_vertex_buffer(size, channels, elems)
|
||||||
|
self._size = size
|
||||||
|
|
||||||
|
def __del__(self):
|
||||||
|
libgl.delete_vertex_buffer(self._buffer)
|
||||||
|
|
||||||
|
@property
|
||||||
|
def size(self):
|
||||||
|
return self._size
|
||||||
|
|
||||||
|
def upload(self, mesh):
|
||||||
|
assert (self._size >= mesh.size)
|
||||||
|
return libgl.upload_vertex_buffer(self._buffer, mesh.views, mesh.size)
|
||||||
@ -17,15 +17,14 @@
|
|||||||
from . import preview_generated
|
from . import preview_generated
|
||||||
|
|
||||||
class PreviewPanel(preview_generated.PreviewPanel):
|
class PreviewPanel(preview_generated.PreviewPanel):
|
||||||
__slots__ = 'canvas', 'context'
|
__slots__ = 'video_canvas'
|
||||||
|
|
||||||
def __init__(self, parent, id, pos, size, style):
|
def __init__(self, parent, id, pos, size, style):
|
||||||
preview_generated.PreviewPanel.__init__(self, parent, id = id, pos = pos, size = size, style = style)
|
preview_generated.PreviewPanel.__init__(self, parent, id = id, pos = pos, size = size, style = style)
|
||||||
self.canvas = self._video_canvas
|
self.video_canvas = self._video_canvas
|
||||||
self.context = self._video_canvas.context
|
|
||||||
|
|
||||||
def initdialog_event_handler(self, event):
|
def initdialog_event_handler(self, event):
|
||||||
self.canvas.SetCurrent(self.context)
|
self.video_canvas.SetCurrent(self.video_canvas.context)
|
||||||
|
|
||||||
def size_event_handler(self, event):
|
def size_event_handler(self, event):
|
||||||
pass
|
pass
|
||||||
|
|||||||
@ -25,3 +25,5 @@ class VideoCanvas(wx.glcanvas.GLCanvas):
|
|||||||
wx.glcanvas.GLAttributes().PlatformDefaults().RGBA().FrameBuffersRGB().DoubleBuffer().EndList())
|
wx.glcanvas.GLAttributes().PlatformDefaults().RGBA().FrameBuffersRGB().DoubleBuffer().EndList())
|
||||||
self.context = wx.glcanvas.GLContext(self, ctxAttrs =
|
self.context = wx.glcanvas.GLContext(self, ctxAttrs =
|
||||||
wx.glcanvas.GLContextAttrs().CoreProfile().OGLVersion(4,6).EndList())
|
wx.glcanvas.GLContextAttrs().CoreProfile().OGLVersion(4,6).EndList())
|
||||||
|
if not self.context.IsOK()):
|
||||||
|
raise RuntimeError("Failed to create GL Context")
|
||||||
|
|||||||
Reference in New Issue
Block a user