Я использую python3 с библиотекой Qt для сборки моего графического интерфейса (pyqt5).
В основном графическом интерфейсе у меня была метка для размещения изображения внутри него, а listWidget
, который должен содержать элементы в виде значков, иlineEdit
, который берет источник потока, и кнопку для запуска моего приложения.
Моя цель - показать потоковую трансляцию и анализ в реальном времени.
Для этого я использовал QThread
для обновления графического интерфейса пользователя и многопроцессорной обработки.делать анализ.
Вот мой код:
class Stream(QThread):
change_image_signal = pyqtSignal(QImage)
def __init__(self, src, queue):
QThread.__init__(self)
self.stream = cv2.VideoCapture(src)
self.queue = queue
def run(self):
frame_shift = 5
frames_per_sec = 10
sleep_time = 1 / frames_per_sec
frame_count = 0
while self.stream.isOpened():
ret, frame = self.stream.read()
frame_count += 1
if not ret:
break
rgb_image = cv2.cvtColor(frame, cv2.COLOR_BGR2RGB)
if frame_count % (frame_shift + 1) == 0:
self.queue.put(rgb_image)
qt_format_image = QImage(rgb_image.data, rgb_image.shape[1], rgb_image.shape[0], 3 * rgb_image.shape[1],
QImage.Format_RGB888)
image = qt_format_image.scaled(400, 400)
self.change_image_signal.emit(image)
time.sleep(sleep_time)
class ThreadToListWidget(QThread):
append_item_signal = pyqtSignal(QImage)
def __init__(self, analyze_queue):
QThread.__init__(self)
self.analyze_queue = analyze_queue
def run(self):
while True:
frame = self.analyze_queue.get()
if frame is None:
break
qt_format_image = QImage(frame.data, frame.shape[1], frame.shape[0], 3 * frame.shape[1],
QImage.Format_RGB888)
image = qt_format_image.scaled(300, 300)
self.append_item_signal.emit(image)
class AnalyzeProcess(multiprocessing.Process):
def __init__(self, queue, analyze_queue):
multiprocessing.Process.__init__(self)
self.queue = queue
self.analyze_queue = analyze_queue
def run(self):
while True:
frame = self.queue.get()
...
do analyzing to the frame
...
self.analyze_queue.put(frame)
self.analyze_queue.put(None)
class MyGui(QWidget):
def __init__(self):
QWidget.__init__(self)
...
my widgets:
line_edit
image_label
list_widget
button
...
self.button.clicked.connect(self.start)
def start(self):
queue = multiprocessing.Queue()
analyze_queue = multiprocessing.Queue()
self.stream_thread = Stream(self.line_edit.text(), queue)
self.stream_thread.change_image_signal.connect(self.change_image)
self.list_widget_thread = ThreadToListWidget(analyze_queue)
self.list_widget_thread.append_item_signal.connect(self.append_item)
self.analyze_process = AnalyzeProcess(queue, analyze_queue)
self.stream_thread.start()
self.analyze_process.start()
self.list_widget_thread.start()
def change_image(self, image):
self.image_label.setPixmap(QPixmap.fromImage(image))
def append_item(self, image):
pix = QPixmap.fromImage(image)
icon = QIcon(pix)
item = QListWidgetItem()
item.setIcon(icon)
self.list_widget.addItem(item)
self.repaint()
if __name__ == 'main':
app = QApplication(sys.argv)
gui = MyGui()
gui.show()
app.exec_()
Моя проблема в том, что когда я добавил элемент в виджет списка, прямая трансляция, показанная на ярлыке, на секунду застыла.
Может любойодин скажи мне, что не так в моем коде, и если это проблема, потому что нет синхронизации между функциями, которые обновляют графический интерфейс?!