python - 为什么带有自定义绘画的小部件不可见?
问题描述
所以我正在尝试使用自定义小部件构建一个简单的 PyQt 应用程序。然而,画家不画任何东西。如果我注释掉第 44-45 行 ( label = QLabel('Map');box.addWidget(label)
),我可以看到一个大的彩色矩形。但是,当我尝试在矩形上方添加标签时,矩形不再显示。
我想我可能用错了画家,但我不确定。
我是 PyQt 的新手,对我的编码风格或逻辑的任何评论也将不胜感激。
from PyQt5.QtCore import Qt
from PyQt5.QtGui import QPainter, QColor
from PyQt5.QtWidgets import (QMainWindow,
QWidget,
QFrame,
QDesktopWidget,
QGridLayout,
QLabel,
QTextEdit,
QSplitter,
QVBoxLayout,
QApplication)
class Simulator(QMainWindow):
def __init__(self):
super().__init__()
self.stdout = QTextEdit()
self.stderr = QTextEdit()
self.exec = QTextEdit()
self.frame = QFrame()
self.setCentralWidget(self.frame)
self.screen = QDesktopWidget().screenGeometry()
self.setGeometry(self.screen)
self.grid = QGridLayout()
self.frame.setLayout(self.grid)
self.map = SimulatedFieldMap()
# -- setting splitters
splitter_r = QSplitter(Qt.Vertical)
splitter_l = QSplitter(Qt.Vertical)
splitter_h = QSplitter(Qt.Horizontal)
splitter_h.addWidget(splitter_l)
splitter_h.addWidget(splitter_r)
# --------------
# -- top left --
frame = QFrame()
box = QVBoxLayout()
frame.setLayout(box)
splitter_l.addWidget(frame)
label = QLabel('Map')
box.addWidget(label)
box.addWidget(self.map)
# ------
# -- bottom left --
box = QVBoxLayout()
frame = QFrame()
frame.setLayout(box)
box.addWidget(QLabel('Exec'))
box.addWidget(self.exec)
splitter_l.addWidget(frame)
# -------
# -- top right --
box = QVBoxLayout()
frame = QFrame()
frame.setLayout(box)
splitter_r.addWidget(frame)
box.addWidget(QLabel('STDOUT'))
box.addWidget(self.stdout)
# -------
# -- bottom right --
box = QVBoxLayout()
frame = QFrame()
frame.setLayout(box)
splitter_r.addWidget(frame)
box.addWidget(QLabel('STDERR'))
box.addWidget(self.stderr)
# -------
self.grid.addWidget(splitter_h, 0, 0)
splitter_h.setSizes((self.screen.width() * 0.7, self.screen.width() * 0.3))
splitter_l.setSizes((self.screen.height() * 0.7, self.screen.height() * 0.3))
splitter_r.setSizes((self.screen.height() * 0.7, self.screen.height() * 0.3))
class SimulatedFieldMap(QWidget):
def __init__(self):
super().__init__()
def paintEvent(self, event):
qp = QPainter()
qp.begin(self)
self.paintMap(qp)
qp.end()
def paintMap(self, qp):
qp.setBrush(QColor(200, 162, 200)) # lilac
qp.setPen(QColor(200, 162, 200))
geo = self.geometry()
qp.drawRect(geo)
if __name__ == '__main__':
app = QApplication([])
app.setStyle('Fusion')
sim = Simulator()
sim.show()
status = app.exec_()
exit(status)
我在 macOS 10.13.5 上使用 Python3.7。
解决方案
绘制小部件时,使用内部坐标,但geometry()
它是相对于父级的坐标,因此您不应该使用它,而应该使用rect()
.
from PyQt5 import QtCore, QtGui, QtWidgets
class Simulator(QtWidgets.QMainWindow):
def __init__(self):
super().__init__()
self.stdout = QtWidgets.QTextEdit()
self.stderr = QtWidgets.QTextEdit()
self.exec = QtWidgets.QTextEdit()
self.frame = QtWidgets.QFrame()
self.setCentralWidget(self.frame)
self.grid = QtWidgets.QGridLayout(self.frame)
self.map = SimulatedFieldMap()
# -- setting splitters
splitter_r = QtWidgets.QSplitter(QtCore.Qt.Vertical)
splitter_l = QtWidgets.QSplitter(QtCore.Qt.Vertical)
splitter_h = QtWidgets.QSplitter(QtCore.Qt.Horizontal)
splitter_h.addWidget(splitter_l)
splitter_h.addWidget(splitter_r)
# --------------
# -- top left --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
splitter_l.addWidget(frame)
label = QtWidgets.QLabel('Map')
box.addWidget(label)
box.addWidget(self.map)
# ------
# -- bottom left --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
box.addWidget(QtWidgets.QLabel('Exec'))
box.addWidget(self.exec)
splitter_l.addWidget(frame)
# -------
# -- top right --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
splitter_r.addWidget(frame)
box.addWidget(QtWidgets.QLabel('STDOUT'))
box.addWidget(self.stdout)
# -------
# -- bottom right --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
splitter_r.addWidget(frame)
box.addWidget(QtWidgets.QLabel('STDERR'))
box.addWidget(self.stderr)
# -------
screen = QtWidgets.QDesktopWidget().screenGeometry()
self.grid.addWidget(splitter_h, 0, 0)
splitter_h.setSizes((screen.width() * 0.7, screen.width() * 0.3))
splitter_l.setSizes((screen.height() * 0.7, screen.height() * 0.3))
splitter_r.setSizes((screen.height() * 0.7, screen.height() * 0.3))
class SimulatedFieldMap(QtWidgets.QWidget):
def __init__(self):
super().__init__()
def paintEvent(self, event):
qp = QtGui.QPainter(self)
self.paintMap(qp)
def paintMap(self, qp):
qp.setBrush(QtGui.QColor(200, 162, 200)) # lilac
qp.setPen(QtGui.QColor(200, 162, 200))
qp.drawRect(self.rect())
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
app.setStyle('Fusion')
sim = Simulator()
sim.showMaximized()
sys.exit(app.exec_())
更新:如果您希望将其放置在某个位置,则不应将其放置在布局中,但标签必须是地图的子级,并用于move()
建立相对于地图左上角的位置。
# -- top left --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
box.addWidget(self.map)
splitter_l.addWidget(frame)
label = QtWidgets.QLabel('Map', self.map)
label.move(0, 100)
# ------
更新:问题是由 QLabel 的垂直 sizePolicyQSizePolicy::Preferred
导致它扩展,一个简单的解决方案是将其更改为QSizePolicy::Maximum
,因此根据字体计算正确的高度。
# -- top left --
frame = QtWidgets.QFrame()
box = QtWidgets.QVBoxLayout(frame)
splitter_l.addWidget(frame)
label = QtWidgets.QLabel('Map')
sp = label.sizePolicy()
sp.setVerticalPolicy(QtWidgets.QSizePolicy.Maximum)
label.setSizePolicy(sp)
box.addWidget(label)
box.addWidget(self.map)
# ------
推荐阅读
- tensorflow - Tensorflow-gpu,没有属性“python_io”
- python - 通用 n 门蒙蒂霍尔问题模拟器
- python - 使用 python 将文件移动到特定目录获取错误文件已存在
- java - 复制到 vaadin 14 中的剪贴板组件
- javascript - 从选定的单选按钮显示输入文本框
- c - 编译“Hello World”程序时出错
- javascript - XMLHttp 请求上的混合内容
- java - JAVA使用匹配器/模式查找字符串中括号包围的部分
- html - 在同一行中的 div 中使用工具提示的 input 和 span 元素
- c++ - Visual Studio 2019 v16.3.9 是否完全支持 C++20?