如何向 QTabWidget 添加新选项卡

Posted

技术标签:

【中文标题】如何向 QTabWidget 添加新选项卡【英文标题】:How to add a new tab to QTabWidget 【发布时间】:2019-11-28 00:20:12 【问题描述】:

我有一个带有 1(或 2)个小部件(我的主页小部件)的 QtTabWidget,如下所示:

330     for i in range(1):
331         win = MainWindow()
332         tabs.addTab(win, QIcon('running.png'), "Test-%d" % i)
333     tabs.show()
334     print("tab count = %d" % tabs.count())

AFAIK,'tabs' 现在是'win' 的父级。 我还有一个带有“添加选项卡”和“删除选项卡”的工具栏。删除/退出处理程序正在工作,就像

254     # ----------------------- quitHandler() ------------------------
255     def quitHandler(self):
256         if ( self.parentWidget().count() == 1 ):
257             self.statusBar.setText('I am the last one, you can not kill me ....')
258             return
259         tab = self.parentWidget().currentWidget()
260         self.close()
261         self.parentWidget().removeWidget(tab)
262 

但我的“添加标签”处理程序写成

263     # ----------------------- newTabHandler() ------------------------
264     def newTabHandler(self):
265         count = self.parentWidget().count()
266         if ( count > 10 ):
267             self.statusBar.setText('I only support 10 tabs ....')
268             return
269         win = MainWindow()
270         self.parentWidget().addWidget(win)
271         self.parentWidget().show()
272         print(self.parentWidget().count())

没有呈现新标签。但我看到标签数量正在增加..这是日志

medi@medi:~/proto/python/d1> ./utg
tab count = 1
2
3
4
5

感谢您的帮助。

根据要求,这里是显示问题陈述的最小代码:

  1 #!/usr/bin/python3
  2 
  3 import sys
  4 import os
  5 
  6 from PyQt5 import (QtCore, QtWidgets, QtGui)
  7 from PyQt5.QtGui import (QIcon)
  8 from PyQt5.QtWidgets import (QMainWindow, QPushButton, QLabel, QLineEdit, QTextEdit)
  9 from PyQt5.QtWidgets import (QWidget, QHBoxLayout, QVBoxLayout, QGroupBox, QGridLayou    t)
 10 from PyQt5.QtWidgets import (QFormLayout, QSizePolicy, QAction, QToolBar)
 11 from PyQt5.QtCore import (QSize, QProcess)
 12 
 13 class MainWindow(QMainWindow):
 14     def __init__(self):
 15         super().__init__()
 16         self.setWindowTitle('tabs mgmnt test')
 17         self.setGeometry(50, 50, 600, 600)
 18 
 19         self.toolbar = QToolBar('My Main Tool Bar')
 20         self.addToolBar(self.toolbar)
 21         newTabAct = QAction('New Tab', self)
 22         self.toolbar.addAction(newTabAct)
 23         newTabAct.triggered.connect(self.newTabHandler)
 24         
 25    # ----------------------- newTabHandler() ------------------------
 26     def newTabHandler(self):
 27         print("before new tab, tab-count = %d" % self.parentWidget().count() )
 28         win = MainWindow()
 29         self.parentWidget().addWidget(win)
 30         self.parentWidget().show()
 31         print("after new tab, tab-count = %d" % self.parentWidget().count() )
 32 
 33 # ================================= main() ==========================
 34 if (__name__ == "__main__"):
 35     app = QtWidgets.QApplication(sys.argv)
 36     tabs = QtWidgets.QTabWidget()
 37     win = MainWindow()
 38     tabs.addTab(win, "Tab-1" )
 39     tabs.show()
 40     sys.exit ( app.exec_() )
 41 

正如所见,在 main() 和第 36 行中,我创建了一个 QTabWidget 和一个窗口并将窗口挂到选项卡上。所以我相信,“选项卡”是父级,“赢”是一个子级。 然后第 26 行的“newTabHandler()”应该向选项卡添加另一个选项卡。但是这个方法是 MainWindow 的一个实例方法,看不到 main() 本地的“选项卡”。所以我通过parentWidget()来遍历父子关系。另请注意,第 27 行和第 31 行的 print() 语句报告了正确的选项卡计数。但我没有看到呈现的新标签。也许我错过了针对某个对象的 show() 。我在第 30 行执行此操作。但没有渲染新选项卡,我只看到一个选项卡,即在 main() 中创建的一个选项卡。

【问题讨论】:

什么是'self.parentWidget()'?看来这是QStackedWidget,你可以数一下附加到'QStackedWidget'的小部件的数量。你能上传可执行代码吗? 什么是addWidget?如果父级应该是QTabWidget,则没有这样的方法。 【参考方案1】:

要理解问题,您必须在 newTabHandler 中打印 parentWidget():

def newTabHandler(self):
    print(self.parentWidget())

您会注意到,每次尝试添加标签时都会得到:

<PyQt5.QtWidgets.QStackedWidget object at 0x7f3e4e2680d0>
<PyQt5.QtWidgets.QStackedWidget object at 0x7f3e4e2680d0>
<PyQt5.QtWidgets.QStackedWidget object at 0x7f3e4e2680d0>
...

所以 parentWidget() 不是 QTabWidget 而是 QStackedWidget。

说明:

QTabWidget 是 QTabBar + QStackedWidget,每次使用 addTab() 添加小部件时,小部件都会添加到 QStackedWidget 并在 QTabBar 中创建一个新选项卡,因此小部件的父级是 QStackedWidget。

解决方案:

考虑到上述情况,解决方案是使用 QStackedWidget 的 parentWidget(),即 QTabWidget,或者在您提供的代码中,因为 QTabWidget 是窗口,然后使用 window() 方法:

def newTabHandler(self):
    tab_widget = self.parentWidget().parentWidget()
    #              QStackedLayout    QStackedWidget
    # or
    # tab_widget = self.window()
    print(tab_widget)
    count = tab_widget.count()
    win = MainWindow()
    tab_widget.addTab(win, "Tab-".format(count + 1))

【讨论】:

感谢工作。我之前发现 QTabWidget 确实使用 QStackWidget 来保存它的孩子。但我不认为我(作为 QTabWidget 对象的用户)应该关心(或知道)它如何处理其内部数据结构。我以为我应该知道的只是层级关系,父级如何处理其业务是他的域。无论如何...我选择了 tabs = self.window() 因为我不喜欢显式级别(xpath)引用(parent.parent.etc)。再次感谢。

以上是关于如何向 QTabWidget 添加新选项卡的主要内容,如果未能解决你的问题,请参考以下文章

如何访问 QTabWidget 不同选项卡中的小部件?

将选项卡添加到 QTabWidget 时显示问题

避免 QTabWidget 中的重复选项卡

选项卡未添加到 QTabWidget

PyQt4 菜单操作以将新选项卡添加到 QTabWidget

QT。向 QLayout 添加新布局