Как указывает @musicamante в комментариях, решение состоит в том, чтобы получить доступ к QMediaStreamsControl, но PyQt5 не предоставляет его.
Вместо этого PySide2 предоставляет его, и решение заключается в приведении с использованием shiboken2:
import os
from PySide2 import QtCore, QtWidgets, QtMultimedia, QtMultimediaWidgets
import shiboken2
class MainWindow(QtWidgets.QMainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
video_widget = QtMultimediaWidgets.QVideoWidget()
self.player = QtMultimedia.QMediaPlayer(
self, QtMultimedia.QMediaPlayer.VideoSurface
)
file = os.path.join(os.path.dirname(os.path.realpath(__file__)), "test5.mkv")
self.player.setMedia(
QtMultimedia.QMediaContent(QtCore.QUrl.fromLocalFile(file))
)
self.player.setVideoOutput(video_widget)
self.player.play()
self.setCentralWidget(video_widget)
control = self.player.service().requestControl(
"org.qt-project.qt.mediastreamscontrol/5.0"
)
qptr = shiboken2.getCppPointer(control)[0]
self.qcontrol = shiboken2.wrapInstance(qptr, QtMultimedia.QMediaStreamsControl)
self.resize(640, 480)
def contextMenuEvent(self, event):
menu = QtWidgets.QMenu()
group = QtWidgets.QActionGroup(menu)
group.setExclusive(True)
index = 0
for i in range(self.qcontrol.streamCount()):
t = self.qcontrol.streamType(i)
if t == QtMultimedia.QMediaStreamsControl.AudioStream:
action = menu.addAction("Audio-{}".format(index))
action.setCheckable(True)
if self.qcontrol.isActive(i):
action.setChecked(True)
action.setData(i)
menu.addAction(action)
index += 1
action = menu.exec_(self.mapToGlobal(event.pos()))
if action is not None:
i = action.data()
self.qcontrol.setActive(i, True)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())
В случае pyqt5 вы должны использовать sip со следующим кодом:
import os
from PyQt5 import QtCore, QtWidgets, QtMultimedia, QtMultimediaWidgets
import sip
class MainWindow(QtWidgets.QMainWindow):
def __init__(self, parent=None):
super(MainWindow, self).__init__(parent)
video_widget = QtMultimediaWidgets.QVideoWidget()
self.player = QtMultimedia.QMediaPlayer(
self, QtMultimedia.QMediaPlayer.VideoSurface
)
file = os.path.join(os.path.dirname(os.path.realpath(__file__)), "test5.mkv")
self.player.setMedia(
QtMultimedia.QMediaContent(QtCore.QUrl.fromLocalFile(file))
)
self.player.setVideoOutput(video_widget)
self.player.play()
self.setCentralWidget(video_widget)
control = self.player.service().requestControl(
"org.qt-project.qt.mediastreamscontrol/5.0"
)
self.qcontrol = sip.cast(control, QtMultimedia.QMediaStreamsControl)
self.resize(640, 480)
def contextMenuEvent(self, event):
menu = QtWidgets.QMenu()
group = QtWidgets.QActionGroup(menu)
group.setExclusive(True)
index = 0
for i in range(self.qcontrol.streamCount()):
t = self.qcontrol.streamType(i)
if t == QtMultimedia.QMediaStreamsControl.AudioStream:
action = menu.addAction("Audio-{}".format(index))
action.setCheckable(True)
if self.qcontrol.isActive(i):
action.setChecked(True)
action.setData(i)
menu.addAction(action)
index += 1
action = menu.exec_(self.mapToGlobal(event.pos()))
if action is not None:
i = action.data()
self.qcontrol.setActive(i, True)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
w.show()
sys.exit(app.exec_())
Но, как указано в QMediaStreamsControl, он недоступен в PyQt5, поэтому решение состоит в том, чтобы выставить его и для этого Вы должны:
- Загрузить исходный код PyQt5: https://pypi.python.org/packages/source/P/PyQt5/PyQt5-5.14.2.tar.gz
Создать файл qmediastreamscontrol.sip в папке "sip / QtMultimedia" папки Исходный код PyQt5.
qmediastreamscontrol.sip
class QMediaStreamsControl : QMediaControl
{
%TypeHeaderCode
#include <qmediastreamscontrol.h>
%End
public:
enum StreamType {
UnknownStream,
VideoStream,
AudioStream,
SubPictureStream,
DataStream
};
virtual ~QMediaStreamsControl();
virtual int streamCount() = 0;
virtual QMediaStreamsControl::StreamType streamType(int streamNumber) = 0;
virtual QVariant metaData(int streamNumber, const QString &key) = 0;
virtual bool isActive(int streamNumber) = 0;
virtual void setActive(int streamNumber, bool state) = 0;
signals:
void streamsChanged();
void activeStreamsChanged();
protected:
%If (Qt_5_6_1 -)
explicit QMediaStreamsControl(QObject *parent /TransferThis/ = 0);
%End
%If (- Qt_5_6_1)
QMediaStreamsControl(QObject *parent /TransferThis/ = 0);
%End
};
Добавить %Include qmediastreamscontrol.sip
в конец файла sip / QtMultimedia / QtMultimediamod.sip
Скомпилируйте и установите PyQt5, используя измененный исходный код.
В заключение:
Если вы используете pyside2, решение простое.
Если вы используете pyq t5 вам придется изменить его исходный код, скомпилировать и установить его. Надеемся, что с отчетом @musicamante класс QMediaStreamsControl будет представлен в будущих выпусках pyqt5