1 Star 0 Fork 0

emacs-eaf / eaf-video-player

加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
克隆/下载
buffer.py 10.03 KB
一键复制 编辑 原始数据 按行查看 历史
manateelazycat 提交于 2023-06-18 18:24 . Use ruff refactor code.
#!/usr/bin/env python3
# -*- coding: utf-8 -*-
# Copyright (C) 2018 Andy Stewart
#
# Author: Andy Stewart <lazycat.manatee@gmail.com>
# Maintainer: Andy Stewart <lazycat.manatee@gmail.com>
#
# This program 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
# any later version.
#
# This program 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 this program. If not, see <http://www.gnu.org/licenses/>.
from core.buffer import Buffer
from core.utils import interactive, message_to_emacs
from PyQt6 import QtCore, QtWidgets
from PyQt6.QtCore import QEvent, QRectF, QSizeF, Qt, QUrl
from PyQt6.QtGui import QBrush, QColor, QPainter
from PyQt6.QtMultimedia import QAudioOutput, QMediaPlayer
from PyQt6.QtMultimediaWidgets import QGraphicsVideoItem
from PyQt6.QtWidgets import QGraphicsScene, QGraphicsView, QHBoxLayout, QVBoxLayout, QWidget
class AppBuffer(Buffer):
def __init__(self, buffer_id, url, arguments):
Buffer.__init__(self, buffer_id, url, arguments, True)
self.background_color = QColor(0, 0, 0)
self.add_widget(VideoPlayer(self.theme_background_color, self.theme_foreground_color))
self.buffer_widget.play(url)
self.build_all_methods(self.buffer_widget)
def all_views_hide(self):
# Pause video before all views hdie, otherwise will got error "Internal data stream error".
if self.buffer_widget.media_player.playbackState() == QMediaPlayer.PlaybackState.PlayingState:
self.buffer_widget.media_player.pause()
self.buffer_widget.video_need_replay = True
def some_view_show(self):
if self.buffer_widget.video_need_replay is True:
self.buffer_widget.media_player.play()
def save_session_data(self):
return str(self.buffer_widget.media_player.position())
def restore_session_data(self, session_data):
position = int(session_data)
self.buffer_widget.media_player.setPosition(position)
def toggle_play(self):
if self.buffer_widget.media_player.playbackState() == QMediaPlayer.PlaybackState.PlayingState:
self.buffer_widget.media_player.pause()
self.buffer_widget.video_need_replay = False
else:
self.buffer_widget.media_player.play()
self.buffer_widget.video_need_replay = True
def destroy_buffer(self):
self.buffer_widget.media_player.pause()
super().destroy_buffer()
class VideoPlayer(QWidget):
def __init__(self, theme_background_color, theme_foreground_color):
super(VideoPlayer, self).__init__()
self.scene = QGraphicsScene(self)
self.scene.setBackgroundBrush(QBrush(QColor(0, 0, 0, 255)))
self.graphics_view = QGraphicsView(self.scene)
self.graphics_view.setHorizontalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
self.graphics_view.setVerticalScrollBarPolicy(Qt.ScrollBarPolicy.ScrollBarAlwaysOff)
self.graphics_view.setFrameStyle(0)
self.graphics_view.setStyleSheet("QGraphicsView {background: transparent; border: 3px; outline: none;}")
self.is_button_press = False
self.video_item = QGraphicsVideoItem()
self.panel_height = 60
self.progress_bar_height = 60
self.panel_padding_x = 0
self.panel_padding_y = (self.panel_height - self.progress_bar_height) / 2
self.control_panel_widget = QWidget()
self.control_panel_widget.setStyleSheet("background-color: transparent;")
self.progress_bar_layout = QHBoxLayout(self.control_panel_widget)
self.progress_bar_layout.setContentsMargins(
int(self.panel_padding_x),
int(self.panel_padding_y),
int(self.panel_padding_x),
int(self.panel_padding_x))
self.control_panel = ControlPanel()
self.progress_bar = ProgressBar(theme_background_color, theme_foreground_color)
self.progress_bar.progress_changed.connect(self.update_video_progress)
self.progress_bar_layout.addWidget(self.progress_bar)
self.layout = QVBoxLayout(self)
self.layout.setContentsMargins(0, 0, 0, 0)
self.layout.addWidget(self.graphics_view)
self.scene.addItem(self.video_item)
self.scene.addItem(self.control_panel)
self.control_panel_proxy_widget = self.scene.addWidget(self.control_panel_widget)
self.media_player = QMediaPlayer()
self.audio_output = QAudioOutput()
self.media_player.positionChanged.connect(self.progress_change)
self.media_player.setVideoOutput(self.video_item)
self.media_player.setAudioOutput(self.audio_output)
self.video_need_replay = False
self.video_seek_durcation = 10000 # in milliseconds
QtCore.QTimer().singleShot(2000, self.hide_control_panel)
self.graphics_view.viewport().installEventFilter(self)
def update_video_progress(self, percent):
self.media_player.setPosition(int(self.media_player.duration() * percent))
def progress_change(self, position):
self.progress_bar.update_progress(self.media_player.duration(), position)
def resizeEvent(self, event):
self.video_item.setSize(QSizeF(event.size().width(), event.size().height()))
self.control_panel.update_size(event.size().width(), self.panel_height)
self.control_panel.setPos(0, event.size().height() - self.panel_height)
self.control_panel_widget.resize(event.size().width(), self.panel_height)
self.control_panel_proxy_widget.setPos(0, event.size().height() - self.panel_height)
self.progress_bar.resize(event.size().width() - self.panel_padding_x * 2, self.progress_bar_height)
QWidget.resizeEvent(self, event)
def play(self, url):
self.media_player.setSource(QUrl.fromLocalFile(url))
self.media_player.play()
def eventFilter(self, obj, event):
if event.type() in [QEvent.Type.MouseButtonPress]:
self.is_button_press = True
elif event.type() in [QEvent.Type.MouseButtonRelease]:
self.is_button_press = False
if event.type() == QEvent.Type.MouseMove:
if event.position().y() > self.height() - self.progress_bar_height:
self.show_control_panel()
else:
self.hide_control_panel()
return False
def hide_control_panel(self):
self.control_panel.hide()
self.control_panel_proxy_widget.hide()
def show_control_panel(self):
self.control_panel.show()
self.control_panel_proxy_widget.show()
@interactive
def play_forward(self):
video_position = self.media_player.position()
self.media_player.setPosition(video_position + self.video_seek_durcation)
message_to_emacs("Forward to: {}%".format(self.media_player.position() / self.media_player.duration() * 100))
@interactive
def play_backward(self):
video_position = self.media_player.position()
self.media_player.setPosition(max(video_position - self.video_seek_durcation, 0))
message_to_emacs("Forward to: {}%".format(self.media_player.position() / self.media_player.duration() * 100))
@interactive
def increase_volume(self):
self.audio_output.setVolume(self.audio_output.volume() + 0.1)
message_to_emacs("Increase volume to: {}%".format(self.audio_output.volume() * 100))
@interactive
def decrease_volume(self):
self.audio_output.setVolume(self.audio_output.volume() - 0.1)
message_to_emacs("Decrease volume to: {}%".format(self.audio_output.volume() * 100))
@interactive
def restart(self):
self.media_player.setPosition(0)
class ControlPanel(QtWidgets.QGraphicsItem):
def __init__(self, parent=None):
super(ControlPanel, self).__init__(parent)
self.height = 0
self.width = 0
self.background_color = QColor(0, 0, 0, 255)
self.setOpacity(0.9)
def update_size(self, width, height):
self.width = width
self.height = height
self.update()
def paint(self, painter, option, widget):
painter.setPen(self.background_color)
painter.setBrush(self.background_color)
painter.drawRect(0, 0, self.width, self.height)
def boundingRect(self):
return QRectF(0, 0, self.width, self.height)
class ProgressBar(QWidget):
progress_changed = QtCore.pyqtSignal(float)
def __init__(self, theme_background_color, theme_foreground_color):
super(QWidget, self).__init__()
self.foreground_color = QColor(theme_foreground_color)
self.background_color = QColor(theme_background_color)
self.position = 0
self.duration = 0
self.is_press = False
self.render_height = 10
def update_progress(self, duration, position):
self.position = position
self.duration = duration
self.update()
def mousePressEvent(self, event):
self.is_press = True
self.progress_changed.emit(event.position().x() * 1.0 / self.width())
def mouseReleaseEvent(self, event):
self.is_press = False
def mouseMoveEvent(self, event):
if self.is_press:
self.progress_changed.emit(event.position().x() * 1.0 / self.width())
def paintEvent(self, event):
painter = QPainter(self)
render_y = (self.height() - self.render_height) / 2
painter.setPen(self.background_color)
painter.setBrush(self.background_color)
painter.drawRect(0, int(render_y), int(self.width()), int(self.render_height))
if self.duration > 0:
painter.setPen(self.foreground_color)
painter.setBrush(self.foreground_color)
painter.drawRect(0, int(render_y), int(self.width() * self.position / self.duration), int(self.render_height))
Python
1
https://gitee.com/emacs-eaf/eaf-video-player.git
git@gitee.com:emacs-eaf/eaf-video-player.git
emacs-eaf
eaf-video-player
eaf-video-player
master

搜索帮助

53164aa7 5694891 3bd8fe86 5694891