PyQt5 QTabWidget 显示当前 Widget() "NoneType"



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

我想使重新加载按钮在页面加载时被隐藏;"停止装载";按钮可见,当加载完成时,重新加载按钮将再次可见;"停止装载";按钮将被隐藏。

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

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吗?有办法解决它吗?询问我是否需要更多详细信息

谢谢!

错误的原因是:

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

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

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

  1. 删除

    self.tabs.currentWidget().loadProgress.connect(self.loadProgressHandler)
    self.tabs.currentWidget().loadFinished.connect(self.loadFinishedHandler)
    
  2. 添加连接:

    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)
    
  3. 验证:

    @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%加载时,也会调用相关联的插槽,因此它可以被隐藏,因为它可以与完成一起调用。

  • 最好验证可以为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()

最新更新