pythonpyside2qcamera

No viewfinder available while trying to display webcam with PySide2


I am working on Ubuntu 20.04 with Python3 and PySide2, Qt is version 5.

I wrote this simple Python script in order to test the ability to save webcam video for a larger project I will later work on.

Basically, I would like that this demo script at first and simply displays the stream coming from the camera, in a widget I named "frame", but it seems I am missing something, because a message gets printed in the terminal:

Starting camera without viewfinder available

I made the GUI within QtCreator, then converted it with pyside-uic.

I applied a black background-color to the widget called "frame". When I click the "Connect" button I would expect to see the video starting and displaying, but the "frame" keeps black and I get the message about the viewfinder in the terminal. Isn't it the viewfinder the place (aka widget ) where the video is displayed?

I feel something is wrong, but I think it could be solved easily, because when I click the button to connect, the webcam's LED (correctly) lights up.

enter image description here

Here it is the python code:

from PySide2.QtMultimedia import QCamera, QCameraInfo, QCamera
from PySide2.QtMultimedia import  *
from PySide2.QtMultimedia import  QCameraViewfinderSettings
from PySide2.QtCore import *
from PySide2.QtGui import *
from PySide2.QtWidgets import *
import MainWindow
import sys
from MainWindow import Ui_MainWindow

class MainWindow(QMainWindow, Ui_MainWindow):
    def __init__(self, *args, **kwargs):
        super(MainWindow, self).__init__(*args, **kwargs)
        self.setupUi(self)
        self.available_cameras = QCameraInfo.availableCameras()
        print(self.available_cameras[0].description())
        self.camera_combobox.addItem( self.available_cameras[0].description() )
        self.camera_combobox.addItem( self.available_cameras[1].description() )
        self.connect_btn.clicked.connect(self.connect_btn_clicked)

    def connect_btn_clicked(self):
        self.camera = QCamera(self.available_cameras[0])
        self.frame.setSizePolicy(QSizePolicy.Maximum,QSizePolicy.Maximum)
        self.camera.setViewfinder(self.frame)
        self.camera.start()
        
app = QApplication(sys.argv)
window = MainWindow()
window.show()
app.exec_()

Here it is the MainWindow.py code

# -*- coding: utf-8 -*-

################################################################################
## Form generated from reading UI file 'mainwindow.ui'
##
## Created by: Qt User Interface Compiler version 5.15.1
##
## WARNING! All changes made in this file will be lost when recompiling UI file!
################################################################################

from PySide2.QtCore import *
from PySide2.QtGui import *
from PySide2.QtWidgets import *


class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        if not MainWindow.objectName():
            MainWindow.setObjectName(u"MainWindow")
        MainWindow.resize(800, 600)
        self.centralwidget = QWidget(MainWindow)
        self.centralwidget.setObjectName(u"centralwidget")
        self.horizontalLayoutWidget = QWidget(self.centralwidget)
        self.horizontalLayoutWidget.setObjectName(u"horizontalLayoutWidget")
        self.horizontalLayoutWidget.setGeometry(QRect(50, 30, 298, 80))
        self.horizontalLayout = QHBoxLayout(self.horizontalLayoutWidget)
        self.horizontalLayout.setObjectName(u"horizontalLayout")
        self.horizontalLayout.setContentsMargins(0, 0, 0, 0)
        self.label = QLabel(self.horizontalLayoutWidget)
        self.label.setObjectName(u"label")

        self.horizontalLayout.addWidget(self.label)

        self.camera_combobox = QComboBox(self.horizontalLayoutWidget)
        self.camera_combobox.setObjectName(u"camera_combobox")

        self.horizontalLayout.addWidget(self.camera_combobox)

        self.connect_btn = QPushButton(self.horizontalLayoutWidget)
        self.connect_btn.setObjectName(u"connect_btn")

        self.horizontalLayout.addWidget(self.connect_btn)

        self.frame = QFrame(self.centralwidget)
        self.frame.setObjectName(u"frame")
        self.frame.setGeometry(QRect(60, 210, 301, 171))
        sizePolicy = QSizePolicy(QSizePolicy.Expanding, QSizePolicy.Expanding)
        sizePolicy.setHorizontalStretch(0)
        sizePolicy.setVerticalStretch(0)
        sizePolicy.setHeightForWidth(self.frame.sizePolicy().hasHeightForWidth())
        self.frame.setSizePolicy(sizePolicy)
        self.frame.setStyleSheet(u"background-color: rgb(0, 0, 0);")
        self.horizontalLayout_2 = QHBoxLayout(self.frame)
        self.horizontalLayout_2.setObjectName(u"horizontalLayout_2")
        self.horizontalLayoutWidget_2 = QWidget(self.centralwidget)
        self.horizontalLayoutWidget_2.setObjectName(u"horizontalLayoutWidget_2")
        self.horizontalLayoutWidget_2.setGeometry(QRect(60, 110, 291, 51))
        self.horizontalLayout_3 = QHBoxLayout(self.horizontalLayoutWidget_2)
        self.horizontalLayout_3.setObjectName(u"horizontalLayout_3")
        self.horizontalLayout_3.setContentsMargins(0, 0, 0, 0)
        self.capture_btn = QPushButton(self.horizontalLayoutWidget_2)
        self.capture_btn.setObjectName(u"capture_btn")

        self.horizontalLayout_3.addWidget(self.capture_btn)

        self.record_btn = QPushButton(self.horizontalLayoutWidget_2)
        self.record_btn.setObjectName(u"record_btn")

        self.horizontalLayout_3.addWidget(self.record_btn)

        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QMenuBar(MainWindow)
        self.menubar.setObjectName(u"menubar")
        self.menubar.setGeometry(QRect(0, 0, 800, 26))
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QStatusBar(MainWindow)
        self.statusbar.setObjectName(u"statusbar")
        MainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MainWindow)

        QMetaObject.connectSlotsByName(MainWindow)
    # setupUi

    def retranslateUi(self, MainWindow):
        MainWindow.setWindowTitle(QCoreApplication.translate("MainWindow", u"MainWindow", None))
        self.label.setText(QCoreApplication.translate("MainWindow", u"Device:", None))
        self.connect_btn.setText(QCoreApplication.translate("MainWindow", u"Connect", None))
        self.capture_btn.setText(QCoreApplication.translate("MainWindow", u"Capture", None))
        self.record_btn.setText(QCoreApplication.translate("MainWindow", u"Record", None))
    # retranslateUi

The above code is the output of

pyside2-uic mainwindow.ui -o MainWindow.py

Here is the content of mainwindow.ui

<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
 <class>MainWindow</class>
 <widget class="QMainWindow" name="MainWindow">
  <property name="geometry">
   <rect>
    <x>0</x>
    <y>0</y>
    <width>800</width>
    <height>600</height>
   </rect>
  </property>
  <property name="windowTitle">
   <string>MainWindow</string>
  </property>
  <widget class="QWidget" name="centralwidget">
   <widget class="QWidget" name="horizontalLayoutWidget">
    <property name="geometry">
     <rect>
      <x>50</x>
      <y>30</y>
      <width>298</width>
      <height>80</height>
     </rect>
    </property>
    <layout class="QHBoxLayout" name="horizontalLayout">
     <item>
      <widget class="QLabel" name="label">
       <property name="text">
        <string>Device:</string>
       </property>
      </widget>
     </item>
     <item>
      <widget class="QComboBox" name="camera_combobox"/>
     </item>
     <item>
      <widget class="QPushButton" name="connect_btn">
       <property name="text">
        <string>Connect</string>
       </property>
      </widget>
     </item>
    </layout>
   </widget>
   <widget class="QFrame" name="frame">
    <property name="geometry">
     <rect>
      <x>60</x>
      <y>210</y>
      <width>301</width>
      <height>171</height>
     </rect>
    </property>
    <property name="sizePolicy">
     <sizepolicy hsizetype="Expanding" vsizetype="Expanding">
      <horstretch>0</horstretch>
      <verstretch>0</verstretch>
     </sizepolicy>
    </property>
    <property name="styleSheet">
     <string notr="true">background-color: rgb(0, 0, 0);</string>
    </property>
    <layout class="QHBoxLayout" name="horizontalLayout_2"/>
   </widget>
   <widget class="QWidget" name="horizontalLayoutWidget_2">
    <property name="geometry">
     <rect>
      <x>60</x>
      <y>110</y>
      <width>291</width>
      <height>51</height>
     </rect>
    </property>
    <layout class="QHBoxLayout" name="horizontalLayout_3">
     <item>
      <widget class="QPushButton" name="capture_btn">
       <property name="text">
        <string>Capture</string>
       </property>
      </widget>
     </item>
     <item>
      <widget class="QPushButton" name="record_btn">
       <property name="text">
        <string>Record</string>
       </property>
      </widget>
     </item>
    </layout>
   </widget>
  </widget>
  <widget class="QMenuBar" name="menubar">
   <property name="geometry">
    <rect>
     <x>0</x>
     <y>0</y>
     <width>800</width>
     <height>26</height>
    </rect>
   </property>
  </widget>
  <widget class="QStatusBar" name="statusbar"/>
 </widget>
 <resources/>
 <connections/>
</ui>

Solution

  • QCamera needs a QCameraViewfinder which is a widget that allows to display the information captured by QCamera, so you cannot pass a QFrame to it.

    import sys
    
    from PySide2.QtWidgets import QApplication, QMainWindow
    from PySide2.QtMultimedia import QCamera, QCameraInfo
    from PySide2.QtMultimediaWidgets import QCameraViewfinder
    
    from MainWindow import Ui_MainWindow
    
    
    class MainWindow(QMainWindow, Ui_MainWindow):
        def __init__(self, *args, **kwargs):
            super(MainWindow, self).__init__(*args, **kwargs)
            self.setupUi(self)
            for info in QCameraInfo.availableCameras():
                self.camera_combobox.addItem(info.description(), info)
            self.connect_btn.clicked.connect(self.connect_btn_clicked)
    
            self.view_finder = QCameraViewfinder()
            self.frame.layout().addWidget(self.view_finder)
    
        def connect_btn_clicked(self):
            info = self.camera_combobox.currentData()
            self.camera = QCamera(info)
            self.camera.setViewfinder(self.view_finder)
            self.camera.start()
    
    
    app = QApplication(sys.argv)
    window = MainWindow()
    window.show()
    app.exec_()