PyQt5 QTabWidget 显示 currentWidget() "NoneType"

Posted

技术标签:

【中文标题】PyQt5 QTabWidget 显示 currentWidget() "NoneType"【英文标题】:PyQt5 QTabWidget showing currentWidget() "NoneType" 【发布时间】:2021-06-09 19:23:26 【问题描述】:

我正在用python构建一个简单的浏览器,PyQt5 QWebEnjineView。

我想让重新加载按钮在页面加载时隐藏并且“停止加载”按钮可见, 加载完成后,重新加载按钮将再次可见,“停止加载”按钮将被隐藏。

我的 QWebEnjineView 在一个名为 add_new_tab 的方法中,我在 init 方法中将 QTabWidget 定义为 self.tabs。

import os
import sys
from PyQt5 import QtGui
from PyQt5 import QtCore
from PyQt5.QtGui import *
from PyQt5.QtCore import *
from PyQt5.QtWidgets import *
from PyQt5.QtWebEngineWidgets import QWebEngineView

class mainWindow(QMainWindow):
    def __init__(self, *args, **kwargs):
        super(mainWindow, self).__init__()
        
        # create tabs
        self.tabs = QTabWidget()
        
        self.tabs.tabBarDoubleClicked.connect(self.tab_open_doubleclick)
        self.tabs.setTabsClosable(True)
        self.tabs.tabCloseRequested.connect(self.close_current_tab)

        self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler)
        self.tabs.currentWidget().loadFinished.connect(self.loadFinishedHandler)

        self.setCentralWidget(self.tabs)

        # self.setCentralWidget(self.browser)
        self.showMaximized()


        # nav bar
        self.navbar = QToolBar()
        self.navbar.setMovable(False)
        self.addToolBar(self.navbar)

        # Refresh button
        self.reload_butn = QPushButton(self, text="Reload")
        self.reload_butn.clicked.connect(self.reload_tab)

        # Set reload button visible
        self.reload_butn.setHidden(False)

        # Stop button
        self.stop_btn = QPushButton(self, text="Stop")
        self.stop_btn.clicked.connect(self.stop_loading_tab)
        self.stop_btn.setHidden(True)

        # Set stop_butn hidden initially
        self.stop_btn.setHidden(True)
        
        # Add Refresh and Stop button
        self.navbar.addWidget(self.stop_btn)
        self.navbar.addWidget(self.reload_butn)
        
       # Add Address bar
        self.url_bar = QLineEdit()
        self.url_bar.returnPressed.connect(self.navigate_to_url)
        self.navbar.addWidget(self.url_bar)

        # on startup
        self.add_new_tab(QUrl("https://www.google.com/"), "Homepage")
        self.show()
    
    @QtCore.pyqtSlot(int)
    def loadProgressHandler(self, prog):
        self.stop_btn.setHidden(False)  # When any page is loading, then stop_butn will visible
        self.reload_butn.setHidden(True)    # When any page is loading, then reload_butn will hidde  

    @QtCore.pyqtSlot()
    def loadFinishedHandler(self):
        self.reload_butn.setHidden(False)    # When loading is finished, then reload_butn will be visible again for the user
        self.stop_btn.setHidden(True)   # When load finished, stop button will be hidden

    # reload tab
    def reload_tab(self):
        self.tabs.currentWidget().reload()
    
    def stop_loading_tab(self):
        self.tabs.currentWidget().stop()    

    def close_current_tab(self, i):
        if self.tabs.count() < 2 :
            return
        self.tabs.removeTab(i)
    
    # stop load current tab
    def stop_loading_tab(self):
        self.tabs.currentWidget().stop()    

    # doubleclick on empty space for new tab
    def tab_open_doubleclick(self, i):
        if i == -1: # No tab under the click
            self.add_new_tab(QUrl("http://www.google.com/"), label="New tab")
    
  

    # function to add new tab
    def add_new_tab(self, qurl=None, label="Blank"):
        if qurl is None:
            qurl = QUrl('https://www.google.com/')
        
        browser = QWebEngineView()
        browser.setUrl(qurl)
        i = self.tabs.addTab(browser, label)

        self.tabs.setCurrentIndex(i)


    def navigate_to_url(self): 
        self.tabs.currentWidget().setUrl(QUrl(self.url_bar.text()))



app = QApplication(sys.argv)
app.setApplicationName("browser")
window = mainWindow()
app.exec_()

我有一些用于重新加载、返回、主页等的按钮。我在其中调用了self.tabs.currentWidget().reload(),例如在重新加载方法中, 但是当我添加self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler) 对于操作,然后它给我一个错误

    self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler)
AttributeError: 'NoneType' object has no attribute 'loadProgress'

谁能告诉我为什么显示 self.tabs.currentWidget() 是 NoneType? 有没有办法解决它?如果您需要更多详细信息,请询问我

谢谢!

【问题讨论】:

在哪里您尝试连接?请提供minimal, reproducible example。 【参考方案1】:

错误原因是:

self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler)
self.tabs.currentWidget().loadFinished.connect(self.loadFinishedHandler)

OP 假设所有页面都将发生连接,这是不正确的,因为它只会发生在当前小部件上,在这种情况下是 None 导致错误。

在这种情况下,解决方案是连接每个创建的 QWebEngineView,如果 sender() 与 currentWidget() 匹配,则检查插槽。

    删除

    self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler)
    self.tabs.currentWidget().loadFinished.connect(self.loadFinishedHandler)
    

    添加连接:

    def add_new_tab(self, qurl=None, label="Blank"):
        if qurl is None:
            qurl = QUrl('https://www.google.com/')
        browser = QWebEngineView()
    
        browser.loadProgress.connect(self.loadProgressHandler)
        browser.loadFinished.connect(self.loadFinishedHandler)
    
        i = self.tabs.addTab(browser, label)
        self.tabs.setCurrentIndex(i)
    
        browser.load(qurl)
    

    验证:

    @QtCore.pyqtSlot(int)
    def loadProgressHandler(self, prog):
        if self.tabs.currentWidget() is not self.sender():
            return
        self.stop_btn.show()
        self.reload_butn.hide()
    
    @QtCore.pyqtSlot()
    def loadFinishedHandler(self):
        if self.tabs.currentWidget() is not self.sender():
            return
        self.reload_butn.show()
        self.stop_btn.hide()
    

更新:

有以下错误:

添加到 QToolBar 的小部件的可见性是使用关联的 QActions 管理的。

不应管理与加载进度和完成相关的 2 个插槽,而应只使用其中一个,因为例如,关联的插槽也会在以 100% 加载时被调用,因此它可以被隐藏,因为它可以与finished一起调用。

最好验证可以为None的变量,避免异常。

综合以上,解决办法是:

import sys

from PyQt5.QtCore import pyqtSlot, QUrl
from PyQt5.QtWidgets import (
    QApplication,
    QLineEdit,
    QMainWindow,
    QPushButton,
    QTabWidget,
    QToolBar,
)
from PyQt5.QtWebEngineWidgets import QWebEngineView


class mainWindow(QMainWindow):
    def __init__(self, parent=None):
        super(mainWindow, self).__init__(parent)

        self.tabs = QTabWidget(tabsClosable=True)

        self.tabs.tabBarDoubleClicked.connect(self.tab_open_doubleclick)
        self.tabs.tabCloseRequested.connect(self.close_current_tab)

        self.navbar = QToolBar(movable=True)
        self.addToolBar(self.navbar)

        self.reload_butn = QPushButton(self, text="Reload")
        self.reload_butn.clicked.connect(self.reload_tab)

        self.stop_btn = QPushButton(self, text="Stop")
        self.stop_btn.clicked.connect(self.stop_loading_tab)

        self.url_bar = QLineEdit()
        self.url_bar.returnPressed.connect(self.navigate_to_url)

        self.stop_action = self.navbar.addWidget(self.stop_btn)
        self.reload_action = self.navbar.addWidget(self.reload_butn)
        self.navbar.addWidget(self.url_bar)

        self.stop_action.setVisible(False)

        self.add_new_tab(QUrl("https://www.google.com/"), "Homepage")
        self.setCentralWidget(self.tabs)
        self.showMaximized()

    @pyqtSlot(int)
    def loadProgressHandler(self, prog):
        if self.tabs.currentWidget() is not self.sender():
            return
        loading = prog < 100
        self.stop_action.setVisible(loading)
        self.reload_action.setVisible(not loading)

    def reload_tab(self):
        self.tabs.currentWidget().reload()

    def stop_loading_tab(self):
        self.tabs.currentWidget().stop()

    def close_current_tab(self, i):
        if self.tabs.count() < 2:
            return
        self.tabs.removeTab(i)

    def stop_loading_tab(self):
        if self.tabs.currentWidget() is None:
            return
        self.tabs.currentWidget().stop()

    def tab_open_doubleclick(self, i):
        if i == -1:
            self.add_new_tab(QUrl("http://www.google.com/"), label="New tab")

    def add_new_tab(self, qurl=None, label="Blank"):
        if qurl is None:
            qurl = QUrl("https://www.google.com/")
        browser = QWebEngineView()
        browser.loadProgress.connect(self.loadProgressHandler)
        i = self.tabs.addTab(browser, label)
        self.tabs.setCurrentIndex(i)
        browser.load(qurl)

    def navigate_to_url(self):
        if self.tabs.currentWidget() is None:
            return
        self.tabs.currentWidget().load(QUrl.fromString(self.url_bar.text()))


def main():

    app = QApplication(sys.argv)
    app.setApplicationName("browser")
    window = mainWindow()
    app.exec_()


if __name__ == "__main__":
    main()

【讨论】:

谢谢,但是我的 QWebEnjineView 不在构造函数中,它在 add_new_tab 方法中,有没有办法从 add_new_tab 中做到这一点?如果没有,那么我必须进行大量重构。 @SaminSakur 您是否进行了我指出的详细更改?请应用我指出的更改并告诉我它不起作用 感谢您的回答,但我希望在页面加载时隐藏重新加载按钮并且“停止加载”按钮可见,加载完成后,重新加载按钮将再次可见并且“停止加载”按钮将被隐藏,但这不起作用,你知道为什么吗? 非常感谢您的帮助,现在可以正常使用了。

以上是关于PyQt5 QTabWidget 显示 currentWidget() "NoneType"的主要内容,如果未能解决你的问题,请参考以下文章

如何仅隐藏 QTabWidget 栏 PyQt5 [关闭]

PyQt5 - QTabWidget 内容是不是可以扩展到主窗口边缘,即使没有内容?

PyQt5 动态将 QFormLayouts 添加到 QTabWidget 的选项卡中

PyQt5 QTabWidget:如何在类和同一窗口中包含的选项卡之间切换?

PyQt5:QTabWidget中的可关闭选项卡?

PyQt5组件之QStackedWidget