PyQt5 QTabWidget显示currentWidget()“非类型”

2024-06-16 14:25:49 发布

您现在位置:Python中文网/ 问答频道 /正文

我正在用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()是非类型的吗? 有办法解决吗?问我你是否需要更多的细节

谢谢大家!


Tags: fromimportselfnewdefconnecttabreload
1条回答
网友
1楼 · 发布于 2024-06-16 14:25:49

错误的原因是:

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

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

在这种情况下,解决方案是连接创建的每个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()

相关问题 更多 >