原文: http://zetcode.com/gui/pyqt5/menustoolbars/

在 PyQt5 教程的这一部分中,我们创建一个状态栏,菜单栏和工具栏。 菜单是位于菜单栏中的一组命令。 工具栏上的按钮带有应用中的一些常用命令。 状态栏通常在应用窗口的底部显示状态信息。

QMainWindow

QMainWindow类提供一个主应用窗口。 这样可以创建带有状态栏,工具栏和菜单栏的经典应用框架。

状态栏

状态栏是用于显示状态信息的小部件。

statusbar.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a statusbar.
  6. Author: Jan Bodnar
  7. Website: zetcode.com
  8. Last edited: August 2017
  9. """
  10. import sys
  11. from PyQt5.QtWidgets import QMainWindow, QApplication
  12. class Example(QMainWindow):
  13. def __init__(self):
  14. super().__init__()
  15. self.initUI()
  16. def initUI(self):
  17. self.statusBar().showMessage('Ready')
  18. self.setGeometry(300, 300, 250, 150)
  19. self.setWindowTitle('Statusbar')
  20. self.show()
  21. if __name__ == '__main__':
  22. app = QApplication(sys.argv)
  23. ex = Example()
  24. sys.exit(app.exec_())

状态栏是在QMainWindow小部件的帮助下创建的。

  1. self.statusBar().showMessage('Ready')

要获取状态栏,我们调用QtGui.QMainWindow类的statusBar()方法。 该方法的第一次调用将创建一个状态栏。 后续调用返回状态栏对象。 showMessage()在状态栏上显示一条消息。

简单菜单

菜单栏是 GUI 应用的常见部分。 它是位于各个菜单中的一组命令。 (Mac OS 对菜单栏的处理不同。要获得相似的结果,我们可以添加以下行:menubar.setNativeMenuBar(False)。)

simplemenu.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a menubar. The
  6. menubar has one menu with an exit action.
  7. Author: Jan Bodnar
  8. Website: zetcode.com
  9. Last edited: January 2017
  10. """
  11. import sys
  12. from PyQt5.QtWidgets import QMainWindow, QAction, qApp, QApplication
  13. from PyQt5.QtGui import QIcon
  14. class Example(QMainWindow):
  15. def __init__(self):
  16. super().__init__()
  17. self.initUI()
  18. def initUI(self):
  19. exitAct = QAction(QIcon('exit.png'), '&Exit', self)
  20. exitAct.setShortcut('Ctrl+Q')
  21. exitAct.setStatusTip('Exit application')
  22. exitAct.triggered.connect(qApp.quit)
  23. self.statusBar()
  24. menubar = self.menuBar()
  25. fileMenu = menubar.addMenu('&File')
  26. fileMenu.addAction(exitAct)
  27. self.setGeometry(300, 300, 300, 200)
  28. self.setWindowTitle('Simple menu')
  29. self.show()
  30. if __name__ == '__main__':
  31. app = QApplication(sys.argv)
  32. ex = Example()
  33. sys.exit(app.exec_())

在上面的示例中,我们创建一个带有一个菜单的菜单栏。 该菜单将包含一个动作,如果选中该动作,它将终止应用。 也会创建一个状态栏。 可通过 Ctrl + Q 快捷方式访问该操作。

  1. exitAct = QAction(QIcon('exit.png'), '&Exit', self)
  2. exitAct.setShortcut('Ctrl+Q')
  3. exitAct.setStatusTip('Exit application')

QAction是使用菜单栏,工具栏或自定义键盘快捷键执行的操作的抽象。 在以上三行中,我们创建一个带有特定图标和“退出”标签的动作。 此外,为此操作定义了快捷方式。 第三行创建一个状态提示,当我们将鼠标指针悬停在菜单项上时,状态提示将显示在状态栏中。

  1. exitAct.triggered.connect(qApp.quit)

当我们选择此特定动作时,将触发信号。 信号连接到QApplication小部件的quit()方法。 这将终止应用。

  1. menubar = self.menuBar()
  2. fileMenu = menubar.addMenu('&File')
  3. fileMenu.addAction(exitAction)

menuBar()方法创建一个菜单栏。 我们使用addMenu()创建文件菜单,并使用addAction()添加操作。

子菜单

子菜单是位于另一个菜单内的菜单。

submenu.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a submenu.
  6. Author: Jan Bodnar
  7. Website: zetcode.com
  8. Last edited: August 2017
  9. """
  10. import sys
  11. from PyQt5.QtWidgets import QMainWindow, QAction, QMenu, QApplication
  12. class Example(QMainWindow):
  13. def __init__(self):
  14. super().__init__()
  15. self.initUI()
  16. def initUI(self):
  17. menubar = self.menuBar()
  18. fileMenu = menubar.addMenu('File')
  19. impMenu = QMenu('Import', self)
  20. impAct = QAction('Import mail', self)
  21. impMenu.addAction(impAct)
  22. newAct = QAction('New', self)
  23. fileMenu.addAction(newAct)
  24. fileMenu.addMenu(impMenu)
  25. self.setGeometry(300, 300, 300, 200)
  26. self.setWindowTitle('Submenu')
  27. self.show()
  28. if __name__ == '__main__':
  29. app = QApplication(sys.argv)
  30. ex = Example()
  31. sys.exit(app.exec_())

在示例中,我们有两个菜单项; 一个位于“文件”菜单中,另一个位于“文件的导入”子菜单中。

  1. impMenu = QMenu('Import', self)

使用QMenu创建新菜单。

  1. impAct = QAction('Import mail', self)
  2. impMenu.addAction(impAct)

使用addAction()将操作添加到子菜单。

PyQt5 中的菜单和工具栏 - 图1

图:子菜单

复选菜单

在以下示例中,我们创建一个可以选中和取消选中的菜单。

checkmenu.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a checkable menu.
  6. Author: Jan Bodnar
  7. Website: zetcode.com
  8. Last edited: August 2017
  9. """
  10. import sys
  11. from PyQt5.QtWidgets import QMainWindow, QAction, QApplication
  12. class Example(QMainWindow):
  13. def __init__(self):
  14. super().__init__()
  15. self.initUI()
  16. def initUI(self):
  17. self.statusbar = self.statusBar()
  18. self.statusbar.showMessage('Ready')
  19. menubar = self.menuBar()
  20. viewMenu = menubar.addMenu('View')
  21. viewStatAct = QAction('View statusbar', self, checkable=True)
  22. viewStatAct.setStatusTip('View statusbar')
  23. viewStatAct.setChecked(True)
  24. viewStatAct.triggered.connect(self.toggleMenu)
  25. viewMenu.addAction(viewStatAct)
  26. self.setGeometry(300, 300, 300, 200)
  27. self.setWindowTitle('Check menu')
  28. self.show()
  29. def toggleMenu(self, state):
  30. if state:
  31. self.statusbar.show()
  32. else:
  33. self.statusbar.hide()
  34. if __name__ == '__main__':
  35. app = QApplication(sys.argv)
  36. ex = Example()
  37. sys.exit(app.exec_())

该代码示例使用一个动作创建一个视图菜单。 该操作显示或隐藏状态栏。 当状态栏可见时,将选中菜单项。

  1. viewStatAct = QAction('View statusbar', self, checkable=True)

使用checkable选项,我们创建一个复选菜单。

  1. viewStatAct.setChecked(True)

由于状态栏从一开始就可见,因此我们使用setChecked()方法检查操作。

  1. def toggleMenu(self, state):
  2. if state:
  3. self.statusbar.show()
  4. else:
  5. self.statusbar.hide()

根据操作的状态,我们显示或隐藏状态栏。

PyQt5 中的菜单和工具栏 - 图2

图:复现菜单

上下文菜单

上下文菜单,也称为弹出菜单,是在某些上下文下显示的命令列表。 例如,在 Opera Web 浏览器中,当我们右键单击某个网页时,将获得一个上下文菜单。 在这里,我们可以重新加载页面,返回页面或查看页面源。 如果右键单击工具栏,则将获得另一个用于管理工具栏的上下文菜单。

contextmenu.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a context menu.
  6. Author: Jan Bodnar
  7. Website: zetcode.com
  8. Last edited: August 2017
  9. """
  10. import sys
  11. from PyQt5.QtWidgets import QMainWindow, qApp, QMenu, QApplication
  12. class Example(QMainWindow):
  13. def __init__(self):
  14. super().__init__()
  15. self.initUI()
  16. def initUI(self):
  17. self.setGeometry(300, 300, 300, 200)
  18. self.setWindowTitle('Context menu')
  19. self.show()
  20. def contextMenuEvent(self, event):
  21. cmenu = QMenu(self)
  22. newAct = cmenu.addAction("New")
  23. opnAct = cmenu.addAction("Open")
  24. quitAct = cmenu.addAction("Quit")
  25. action = cmenu.exec_(self.mapToGlobal(event.pos()))
  26. if action == quitAct:
  27. qApp.quit()
  28. if __name__ == '__main__':
  29. app = QApplication(sys.argv)
  30. ex = Example()
  31. sys.exit(app.exec_())

要使用上下文菜单,我们必须重新实现contextMenuEvent()方法。

  1. action = cmenu.exec_(self.mapToGlobal(event.pos()))

使用exec_()方法显示上下文菜单。 从事件对象获取鼠标指针的坐标。 mapToGlobal()方法将微件坐标转换为全局屏幕坐标。

  1. if action == quitAct:
  2. qApp.quit()

如果从上下文菜单返回的操作等于退出操作,则我们终止该应用。

工具栏

菜单将我们可以在应用中使用的所有命令分组。 使用工具栏可以快速访问最常用的命令。

toolbar.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a toolbar.
  6. The toolbar has one action, which
  7. terminates the application, if triggered.
  8. Author: Jan Bodnar
  9. Website: zetcode.com
  10. Last edited: August 2017
  11. """
  12. import sys
  13. from PyQt5.QtWidgets import QMainWindow, QAction, qApp, QApplication
  14. from PyQt5.QtGui import QIcon
  15. class Example(QMainWindow):
  16. def __init__(self):
  17. super().__init__()
  18. self.initUI()
  19. def initUI(self):
  20. exitAct = QAction(QIcon('exit24.png'), 'Exit', self)
  21. exitAct.setShortcut('Ctrl+Q')
  22. exitAct.triggered.connect(qApp.quit)
  23. self.toolbar = self.addToolBar('Exit')
  24. self.toolbar.addAction(exitAct)
  25. self.setGeometry(300, 300, 300, 200)
  26. self.setWindowTitle('Toolbar')
  27. self.show()
  28. if __name__ == '__main__':
  29. app = QApplication(sys.argv)
  30. ex = Example()
  31. sys.exit(app.exec_())

在上面的示例中,我们创建了一个简单的工具栏。 工具栏上有一个工具动作,一个退出动作,在触发时会终止应用。

  1. exitAct = QAction(QIcon('exit24.png'), 'Exit', self)
  2. exitAct.setShortcut('Ctrl+Q')
  3. exitAct.triggered.connect(qApp.quit)

与上面的菜单栏示例类似,我们创建一个动作对象。 该对象具有标签,图标和快捷方式。 QtGui.QMainWindowquit()方法连接到触发信号。

  1. self.toolbar = self.addToolBar('Exit')
  2. self.toolbar.addAction(exitAction)

工具栏是使用addToolBar()方法创建的。 我们使用addAction()将动作对象添加到工具栏。

PyQt5 中的菜单和工具栏 - 图3

图:工具栏

把它放在一起

在本节的最后一个示例中,我们将创建一个菜单栏,工具栏和一个状态栏。 我们还将创建一个中央小部件。

mainwindow.py

  1. #!/usr/bin/python3
  2. # -*- coding: utf-8 -*-
  3. """
  4. ZetCode PyQt5 tutorial
  5. This program creates a skeleton of
  6. a classic GUI application with a menubar,
  7. toolbar, statusbar, and a central widget.
  8. Author: Jan Bodnar
  9. Website: zetcode.com
  10. Last edited: August 2017
  11. """
  12. import sys
  13. from PyQt5.QtWidgets import QMainWindow, QTextEdit, QAction, QApplication
  14. from PyQt5.QtGui import QIcon
  15. class Example(QMainWindow):
  16. def __init__(self):
  17. super().__init__()
  18. self.initUI()
  19. def initUI(self):
  20. textEdit = QTextEdit()
  21. self.setCentralWidget(textEdit)
  22. exitAct = QAction(QIcon('exit24.png'), 'Exit', self)
  23. exitAct.setShortcut('Ctrl+Q')
  24. exitAct.setStatusTip('Exit application')
  25. exitAct.triggered.connect(self.close)
  26. self.statusBar()
  27. menubar = self.menuBar()
  28. fileMenu = menubar.addMenu('&File')
  29. fileMenu.addAction(exitAct)
  30. toolbar = self.addToolBar('Exit')
  31. toolbar.addAction(exitAct)
  32. self.setGeometry(300, 300, 350, 250)
  33. self.setWindowTitle('Main window')
  34. self.show()
  35. if __name__ == '__main__':
  36. app = QApplication(sys.argv)
  37. ex = Example()
  38. sys.exit(app.exec_())

此代码示例使用菜单栏,工具栏和状态栏创建经典 GUI 应用的框架。

  1. textEdit = QTextEdit()
  2. self.setCentralWidget(textEdit)

在这里,我们创建一个文本编辑小部件。 我们将其设置为QMainWindow的中央小部件。 中央窗口小部件将占据剩余的所有空间。

PyQt5 中的菜单和工具栏 - 图4

图:主窗口

在 PyQt5 教程的这一部分中,我们使用了菜单,工具栏,状态栏和主应用窗口。