Search code examples
pythonpyqtpyqt5vispy

Integrating vispy with pyqt5 and uic


Some questions on vispy mention using canvas.native when adding a widget. How can a widget made as a placeholder in qt designer be used for vispy?

The idea is going from this

canvas = vispy.app.Canvas()
w = QMainWindow()
widget = QWidget()
w.setCentralWidget(widget)
widget.setLayout(QVBoxLayout())
widget.layout().addWidget(canvas.native)
widget.layout().addWidget(QPushButton())
w.show()
vispy.app.run()

to a version where there is a ui or generated python file that has a frame with the name "frameFor3d".

class myWindow(QtWidgets.QMainWindow):
    def __init__(self):

        super(myWindow, self).__init__()
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

        #declare this here?
        canvas = vispy.app.Canvas()
        self.ui.frameFor3d.layout().addWidget(canvas.native)

if __name__ == '__main__': 
    app = QtWidgets.QApplication([])
    application = myWindow()
    vispy.app.run() #does it go here?
    application.show()
    sys.exit(app.exec())

This errors with None because layout() is none.

The uic sample

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

# Form implementation generated from reading ui file 'myWindow.ui'
#
# Created by: PyQt5 UI code generator 5.9.2
#
# WARNING! All changes made in this file will be lost!

from PyQt5 import QtCore, QtGui, QtWidgets

class Ui_MainWindow(object):
    def setupUi(self, MainWindow):
        MainWindow.setObjectName("MainWindow")
        MainWindow.resize(440, 299)
        self.centralwidget = QtWidgets.QWidget(MainWindow)
        self.centralwidget.setObjectName("centralwidget")
        self.gridLayout = QtWidgets.QGridLayout(self.centralwidget)
        self.gridLayout.setObjectName("gridLayout")
        self.frameFor3d = QtWidgets.QFrame(self.centralwidget)
        self.frameFor3d.setFrameShape(QtWidgets.QFrame.StyledPanel)
        self.frameFor3d.setFrameShadow(QtWidgets.QFrame.Raised)
        self.frameFor3d.setObjectName("frameFor3d")
        self.gridLayout.addWidget(self.frameFor3d, 0, 0, 1, 1)
        self.horizontalSlider = QtWidgets.QSlider(self.centralwidget)
        self.horizontalSlider.setOrientation(QtCore.Qt.Horizontal)
        self.horizontalSlider.setObjectName("horizontalSlider")
        self.gridLayout.addWidget(self.horizontalSlider, 1, 0, 1, 1)
        MainWindow.setCentralWidget(self.centralwidget)
        self.menubar = QtWidgets.QMenuBar(MainWindow)
        self.menubar.setGeometry(QtCore.QRect(0, 0, 440, 18))
        self.menubar.setObjectName("menubar")
        MainWindow.setMenuBar(self.menubar)
        self.statusbar = QtWidgets.QStatusBar(MainWindow)
        self.statusbar.setObjectName("statusbar")
        MainWindow.setStatusBar(self.statusbar)

        self.retranslateUi(MainWindow)
        QtCore.QMetaObject.connectSlotsByName(MainWindow)

    def retranslateUi(self, MainWindow):
        _translate = QtCore.QCoreApplication.translate
        MainWindow.setWindowTitle(_translate("MainWindow", "MainWindow"))

Solution

  • QFrame does not have a layout, so you must set a layout, and the run() method is a blocker that internally calls app.exec(), so it is not necessary to call it again.

    class myWindow(QtWidgets.QMainWindow):
        def __init__(self):
            super(myWindow, self).__init__()
            self.ui = Ui_MainWindow()
            self.ui.setupUi(self)
    
            canvas = vispy.app.Canvas()
            lay = QtWidgets.QVBoxLayout(self.ui.frameFor3d) # create layout
            lay.addWidget(canvas.native)
    
    if __name__ == '__main__': 
        app = QtWidgets.QApplication([])
        application = myWindow()
        application.show()
        vispy.app.run()
        # sys.exit(app.exec())