PyQt5:QMediaPlayer 无法重播来自 QBuffer 的音频

PyQt5 : QMediaPlayer can't replay audio from QBuffer

我在 RAM 中有一个 QBuffer 和一个临时 wav 文件,我想让用户从任何地方听他想听多少次就听多少次。但是,它只允许播放一次,不允许重播。如果我播放文件 (QUrl.fromLocalFile) 中的音频,它可以重播它。有什么不同?如何解决?

1) 要从 RAM 播放 wav 文件,我使用以下代码:

    data = b""
    with open(fname, "rb") as file:
        data = file.read()
    buf = QBuffer()
    buf.setData(data) #For debugging. Real buffer is filled differently.
    buf.open(QIODevice.ReadOnly);

    self.mediaPlayer=QMediaPlayer(self)
    self.mediaPlayer.setMedia(QMediaContent(),buf)

然后,如果我调用self.mediaplayer.play(),它会播放文件到最后。但是,所有对 self.mediaplayer.play() 的后续调用均无效。这不是我想要的。

2) 如果我从文件初始化媒体播放器,使用:

self.mediaPlayer.setMedia(QMediaContent(QUrl.fromLocalFile("/home/me/myTestApp/test.wav")))

它工作正常 - 如果我在上一个播放结束后调用 play(),QMediaPlayer 只会重复播放。

A​​ QBuffer 是一个 io 设备 - 阅读后,您需要重置其位置才能再次阅读。所以在你的代码中你需要做这样的事情:

    ...
    self._buffer = buf
    self.mediaPlayer=QMediaPlayer(self)
    self.mediaPlayer.setMedia(QMediaContent(), self._buffer)

def play(self):
    self._buffer.seek(0)
    self.mediaPlayer.play()

编辑:

经过一些实际测试,我发现只需要保留对缓冲区的引用就可以重放音频。下面的脚本是一个完整的示例,对我来说效果很好(在 Linux,使用 GStreamer 后端):

import sys
from PyQt5 import QtCore, QtWidgets, QtMultimedia

class Window(QtWidgets.QWidget):
    def __init__(self):
        super(Window, self).__init__()
        self.buttonOpen = QtWidgets.QPushButton('Open', self)
        self.buttonOpen.clicked.connect(self.handleOpen)
        self.buttonPlay = QtWidgets.QPushButton('Play', self)
        self.buttonPlay.clicked.connect(self.handlePlay)
        layout = QtWidgets.QHBoxLayout(self)
        layout.addWidget(self.buttonOpen)
        layout.addWidget(self.buttonPlay)
        self.mediaPlayer = QtMultimedia.QMediaPlayer(self)
        self._buffer = QtCore.QBuffer()

    def handlePlay(self):
        if self.buttonPlay.text() == 'Play':
            self.buttonPlay.setText('Stop')
            # self._buffer.seek(0)
            self.mediaPlayer.play()
        else:
            self.buttonPlay.setText('Play')
            self.mediaPlayer.stop()

    def handleOpen(self):
        path, ok = QtWidgets.QFileDialog.getOpenFileName(
            self, filter='WAV Files (*.wav)')
        if ok:
            self._buffer.close()
            with open(path, 'rb') as stream:
                self._buffer.setData(stream.read())
            if self._buffer.open(QtCore.QIODevice.ReadOnly):
                self.mediaPlayer.setMedia(
                    QtMultimedia.QMediaContent(), self._buffer)

if __name__ == '__main__':

    app = QtWidgets.QApplication(sys.argv)
    window = Window()
    window.setGeometry(600, 100, 200, 50)
    window.show()
    sys.exit(app.exec_())