- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在尝试创建带有文本轮廓的标签。我只想要一个带有黑色轮廓的简单白色文本。我首先尝试在 css 中这样做 label.setStyleSheet("color:white; outline:2px black;")
但大纲没有做任何事情。
我进行了大量搜索并找到了使用 qpainter 路径的方法。但问题是文本总是被截断。
根据函数,文本应该从左下角开始,但它开始太低太左。我知道我可以通过试错找到一个点,这样它就不会被切断——你可以 -20 到高度,这对这个来说已经足够了。但它只会修复这个特定的文本!对于具有不同大小、文本或字体的任何标签,它都不会相同。
我将把最小的代码示例放在这里
from PyQt5 import QtCore, QtGui, QtWidgets
from PyQt5.QtWidgets import QWidget, QLabel, QMainWindow
class MainLabel(QLabel):
def __init__(self, text):
super(MainLabel, self).__init__(text)
def paintEvent(self, event):
qp = QtGui.QPainter()
qp.begin(self)
qp.setRenderHint(QtGui.QPainter.Antialiasing)
font=QtGui.QFont()
font.setPointSize(70)
painterPath = QtGui.QPainterPath()
#how to get the right positioning for addText
painterPath.addText(0, self.height(), font,self.text())#HERE
qp.strokePath(painterPath, QtGui.QPen(QtGui.QColor(0,0,0), 6))
qp.fillPath(painterPath, QtGui.QColor(255,255,255))
qp.end()
class MainWindow(QMainWindow):
def __init__(self):
super(MainWindow, self).__init__()
self.centralWidget=QWidget(self)
self.setCentralWidget(self.centralWidget)
self.lay = QtWidgets.QVBoxLayout()
self.centralWidget.setLayout(self.lay)
self.label = MainLabel("text gets cut off")
self.label.setStyleSheet("font-size:70pt;color:white; outline:2px black;")
self.lay.addWidget(self.label)
self.show()
if __name__ == '__main__':
import sys
app = QtWidgets.QApplication(sys.argv)
w = MainWindow()
sys.exit(app.exec_())
这是一个很常见的事情,我几乎到处都能看到,但是在 PYQT 中没有一个简单的函数可以解决这个问题而不引起更多问题吗?所以普通的Qlabel会自动处理定位,但如果你想要文本轮廓你就得放弃!
所以我想问的是,如果这是获得文本轮廓的唯一方法,如何像普通 Qlabel 一样找到正确的定位,否则如果有其他更好的方法请告诉我。
最佳答案
仅仅因为没有方便的功能或样式表属性并不意味着没有一致的解决方案!
设置文本的基线位置需要考虑许多属性:QLabel 的几何形状、文本的 boundingRect、对齐方式、缩进、字体规范。轮廓文本总体上将比相同磅值的常规文本大,因此重新实现了 sizeHint
和 minimumSizeHint
来解决这个问题。文档解释了如何 indent计算并与对齐一起使用。文本和字符的几何形状、上升、下降和方位角是从 QFontMetrics 获得的。有了这些信息,就可以确定 QPainterPath.addText
的位置来模拟 QLabel。
import sys, math
from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
class OutlinedLabel(QLabel):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.w = 1 / 25
self.mode = True
self.setBrush(Qt.white)
self.setPen(Qt.black)
def scaledOutlineMode(self):
return self.mode
def setScaledOutlineMode(self, state):
self.mode = state
def outlineThickness(self):
return self.w * self.font().pointSize() if self.mode else self.w
def setOutlineThickness(self, value):
self.w = value
def setBrush(self, brush):
if not isinstance(brush, QBrush):
brush = QBrush(brush)
self.brush = brush
def setPen(self, pen):
if not isinstance(pen, QPen):
pen = QPen(pen)
pen.setJoinStyle(Qt.RoundJoin)
self.pen = pen
def sizeHint(self):
w = math.ceil(self.outlineThickness() * 2)
return super().sizeHint() + QSize(w, w)
def minimumSizeHint(self):
w = math.ceil(self.outlineThickness() * 2)
return super().minimumSizeHint() + QSize(w, w)
def paintEvent(self, event):
w = self.outlineThickness()
rect = self.rect()
metrics = QFontMetrics(self.font())
tr = metrics.boundingRect(self.text()).adjusted(0, 0, w, w)
if self.indent() == -1:
if self.frameWidth():
indent = (metrics.boundingRect('x').width() + w * 2) / 2
else:
indent = w
else:
indent = self.indent()
if self.alignment() & Qt.AlignLeft:
x = rect.left() + indent - min(metrics.leftBearing(self.text()[0]), 0)
elif self.alignment() & Qt.AlignRight:
x = rect.x() + rect.width() - indent - tr.width()
else:
x = (rect.width() - tr.width()) / 2
if self.alignment() & Qt.AlignTop:
y = rect.top() + indent + metrics.ascent()
elif self.alignment() & Qt.AlignBottom:
y = rect.y() + rect.height() - indent - metrics.descent()
else:
y = (rect.height() + metrics.ascent() - metrics.descent()) / 2
path = QPainterPath()
path.addText(x, y, self.font(), self.text())
qp = QPainter(self)
qp.setRenderHint(QPainter.Antialiasing)
self.pen.setWidthF(w * 2)
qp.strokePath(path, self.pen)
if 1 < self.brush.style() < 15:
qp.fillPath(path, self.palette().window())
qp.fillPath(path, self.brush)
您可以使用setBrush
和setPen
设置OutlinedLabel
填充和轮廓颜色。默认为带有黑色轮廓的白色文本。轮廓粗细基于字体的点大小,默认比例为 1/25(即 25pt 字体将具有 1px 粗轮廓)。使用 setOutlineThickness
来改变它。如果您想要一个不基于点大小(例如 3px)的固定轮廓,请调用 setScaledOutlineMode(False)
和 setOutlineThickness(3)
。
此类仅支持左/右/上/下/居中对齐的单行纯文本字符串。如果您想要其他 QLabel 功能,如超链接、自动换行、省略文本等,也需要实现这些功能。但在这些情况下,您很可能不会使用文本大纲。
这是一个例子,表明它适用于各种标签:
class Template(QWidget):
def __init__(self):
super().__init__()
vbox = QVBoxLayout(self)
label = OutlinedLabel('Lorem ipsum dolor sit amet consectetur adipiscing elit,')
label.setStyleSheet('font-family: Monaco; font-size: 20pt')
vbox.addWidget(label)
label = OutlinedLabel('sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.')
label.setStyleSheet('font-family: Helvetica; font-size: 30pt; font-weight: bold')
vbox.addWidget(label)
label = OutlinedLabel('Ut enim ad minim veniam,', alignment=Qt.AlignCenter)
label.setStyleSheet('font-family: Comic Sans MS; font-size: 40pt')
vbox.addWidget(label)
label = OutlinedLabel('quis nostrud exercitation ullamco laboris nisi ut')
label.setStyleSheet('font-family: Arial; font-size: 50pt; font-style: italic')
vbox.addWidget(label)
label = OutlinedLabel('aliquip ex ea commodo consequat.')
label.setStyleSheet('font-family: American Typewriter; font-size: 60pt')
label.setPen(Qt.red)
vbox.addWidget(label)
label = OutlinedLabel('Duis aute irure dolor', alignment=Qt.AlignRight)
label.setStyleSheet('font-family: Luminari; font-size: 70pt')
label.setPen(Qt.red); label.setBrush(Qt.black)
vbox.addWidget(label)
label = OutlinedLabel('in reprehenderit')
label.setStyleSheet('font-family: Zapfino; font-size: 80pt')
label.setBrush(Qt.red)
vbox.addWidget(label)
if __name__ == '__main__':
app = QApplication(sys.argv)
window = Template()
window.show()
sys.exit(app.exec_())
现在 Qt 实际上发挥了重要作用,因为您可以从中获得更多,而不仅仅是使用所有 QBrush/QPen 选项的纯色文本和轮廓:
class Template(QWidget):
def __init__(self):
super().__init__()
vbox = QVBoxLayout(self)
text = 'Felicitations'
label = OutlinedLabel(text)
linearGrad = QLinearGradient(0, 1, 0, 0)
linearGrad.setCoordinateMode(QGradient.ObjectBoundingMode)
linearGrad.setColorAt(0, QColor('#0fd850'))
linearGrad.setColorAt(1, QColor('#f9f047'))
label.setBrush(linearGrad)
label.setPen(Qt.darkGreen)
vbox.addWidget(label)
label = OutlinedLabel(text)
radialGrad = QRadialGradient(0.3, 0.7, 0.05)
radialGrad.setCoordinateMode(QGradient.ObjectBoundingMode)
radialGrad.setSpread(QGradient.ReflectSpread)
radialGrad.setColorAt(0, QColor('#0250c5'))
radialGrad.setColorAt(1, QColor('#2575fc'))
label.setBrush(radialGrad)
label.setPen(QColor('Navy'))
vbox.addWidget(label)
label = OutlinedLabel(text)
linearGrad.setStart(0, 0); linearGrad.setFinalStop(1, 0)
linearGrad.setColorAt(0, Qt.cyan); linearGrad.setColorAt(1, Qt.magenta)
label.setPen(QPen(linearGrad, 1)) # pen width is ignored
vbox.addWidget(label)
label = OutlinedLabel(text)
linearGrad.setFinalStop(1, 1)
for x in [(0, '#231557'), (0.29, '#44107A'), (0.67, '#FF1361'), (1, '#FFF800')]:
linearGrad.setColorAt(x[0], QColor(x[1]))
label.setBrush(linearGrad)
label.setPen(QPen(QBrush(QColor('RoyalBlue'), Qt.Dense4Pattern), 1))
label.setOutlineThickness(1 / 15)
vbox.addWidget(label)
label = OutlinedLabel(text)
label.setBrush(QBrush(Qt.darkBlue, Qt.BDiagPattern))
label.setPen(Qt.darkGray)
vbox.addWidget(label)
label = OutlinedLabel(text, styleSheet='background-color: black')
label.setBrush(QPixmap('paint.jpg'))
label.setPen(QColor('Lavender'))
vbox.addWidget(label)
self.setStyleSheet('''
OutlinedLabel {
font-family: Ubuntu;
font-size: 60pt;
font-weight: bold;
}''')
if __name__ == '__main__':
app = QApplication(sys.argv)
window = Template()
window.show()
sys.exit(app.exec_())
请注意,我选择使用 setBrush
/setPen
方法将 OutlinedLabel 视为 QGraphicsItem。如果你想为文本颜色使用样式表,请使用 qp.fillPath(path, self.palette().text())
替代调用 QPainter.strokePath
然后调用 QPainter.fillPath
的另一个选项是使用 QPainterPathStroker 生成文本路径的可填充轮廓,但我注意到它更慢.我只会用它来调整非常小的文本的清晰度,方法是为笔触设置比笔更大的宽度。要尝试将 paintEvent
中的最后 5 行替换为:
qp.setBrush(self.brush)
self.pen.setWidthF(w)
qp.setPen(self.pen)
stroker = QPainterPathStroker()
stroker.setWidth(w)
qp.drawPath(stroker.createStroke(path).united(path))
关于python - QLabel 文本轮廓的正确定位,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/64290561/
这个问题已经有答案了: How to do case insensitive string comparison? (23 个回答) 已关闭 3 年前。 用户在我的输入栏中写入“足球”,然后执行第 6
啊,不习惯 javascript 中的字符串。 character_id= + id + correct= + correctOrIncorrect 这就是我需要制作成字符串的内容。如果您无法猜测字符
$(function() { var base_price = 0; CalculatePrice(); $(".math1").on('change', function(e) { Calc
我找不到任何文章回答问题:将Spinnaker部署到Spinnaker将管理的同一Kubernetes集群是否安全/正确?我主要是指生产,HA部署。 最佳答案 我认为Spinnaker和Kuberne
我正在使用MSVC在Windows上从源代码(官方源代码发布,而不是从仓库中)构建Qt5(Qt 5.15.0)。 我正在设置环境。变量,依赖项等,然后运行具有1600万个选项的configure,最后
我需要打印一个包含重复单词的数组。我的数组已经可以工作,但我不知道如何正确计算单词数。我已经知道,当我的索引计数器 (i) 为 49 时,并且当 (i) 想要计数到 50 时,我会收到错误,但我不知道
我正在遵循一个指南,该指南允许 Google map 屏幕根据屏幕尺寸禁用滚动。我唯一挣扎的部分是编写一个代码,当我手动调整屏幕大小时动态更改 True/False 值。 这是我按照说明操作的网站,但
我有一个类“FileButton”。它的目的是将文件链接到 JButton,FileButton 继承自 JButton。子类继承自此以使用链接到按钮的文件做有用的事情。 JingleCardButt
我的 friend 数组只返回一个数字而不是所有数字。 ($myfriends = 3) 应该是…… ($myfriends = 3 5 7 8 9 12). 如果我让它进入 while 循环……整个
这个问题在这里已经有了答案: Is there a workaround to make CSS classes with names that start with numbers valid?
我正在制作一个 JavaScript 函数,当调整窗口大小时,它会自动将 div 的大小调整为与窗口相同的宽度/高度。 该功能非常基本,但我注意到在调整窗口大小时出现明显的“绘制”滞后。在 JS fi
此问题的基本视觉效果可在 http://sevenx.de/demo/bootstrap-carousel/inc.carousel/tabbed-slider.html 获得。 - 如果你想看一看。
我明白,如果我想从函数返回一个字符串文字或一个数组,我应该将其声明为静态的,这样当被调用的函数被返回时,内容就不会“消亡”。 但我的问题是,当我在函数内部使用 malloc 分配内存时会怎样? 在下面
在 mySQL 数据库中存储 true/false/1/0 值最合适(读取数据消耗最少)的数据字段是什么? 我以前使用过一个字符长的 tinyint,但我不确定它是否是最佳解决方案? 谢谢! 最佳答案
我想一次读取并处理CSV文件第一行中的条目(例如打印)。我假设使用Unix风格的\n换行符,没有条目长度超过255个字符,并且(现在)在EOF之前有一个换行符。这意味着它是fgets()后跟strto
所以,我们都知道 -1 > 2u == true 的 C/C++ 有符号/无符号比较规则,并且我有一种情况,我想有效地实现“正确”比较。 我的问题是,考虑到人们熟悉的尽可能多的架构,哪种方法更有效。显
**摘要:**文章的标题看似自相矛盾。 本文分享自华为云社区《Java异常处理:如何写出“正确”但被编译器认为有语法错误的程序》,作者: Jerry Wang 。 文章的标题看似自相矛盾,然而我在“正
我有一个数据框,看起来像: dataDemo % mutate_each(funs(ifelse(. == '.', REF, as.character(.))), -POS) # POS REF
有人可以帮助我使用 VBScript 重新格式化/正确格式化带分隔符的文本文件吗? 我有一个文本文件 ^分界如下: AGREE^NAME^ADD1^ADD2^ADD3^ADD4^PCODE^BAL^A
就目前而言,这个问题不适合我们的问答形式。我们希望答案得到事实、引用或专业知识的支持,但这个问题可能会引起辩论、争论、投票或扩展讨论。如果您觉得这个问题可以改进并可能重新打开,visit the he
我是一名优秀的程序员,十分优秀!