在 EnterEvent 中更新 QStyle

update QStyle in EnterEvent

我想在鼠标是 in\out 小部件时更新 LineEdit 的 QStyle (enterEvent\leaveEvent ) 我试图向 drawPrimitive 函数添加一个 bool 变量,但出现此错误 TypeError: drawPrimitive(self, QStyle.PrimitiveElement, QStyleOption, QPainter, widget: QWidget = None): 'a' 不是有效的关键字参数

from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PushButton_2 import Push_Button_
import sys

class LineEditStyle(QProxyStyle):
    def drawPrimitive(self, element, option, painter, widget,a=None):
        if a :
            self.pen = QPen(QColor('green'))
        else :
            self.pen = QPen(QColor('red'))
        self.pen.setWidth(4)
        if element == QStyle.PE_FrameLineEdit:
            painter.setRenderHint(QPainter.Antialiasing)
            painter.setPen(self.pen)
            painter.drawRoundedRect(QRect(0,0,400,40), 10, 10)
        else:
            super().drawPrimitive(element, option, painter, widget)

        
class LineEdit(QLineEdit):
    def __init__(self,*args,**kwargs):
        QLineEdit.__init__(self,*args,**kwargs)
        self.a = 0
    
    def enterEvent(self, a0):
        self.a = 1

    def leaveEvent(self, a0):
        self.a = 0
        
    def paintEvent(self,event):
        option = QStyleOption()
        option.initFrom(self)
        self.style().drawPrimitive(QStyle.PE_FrameLineEdit,option,a=self.a)


if __name__ == '__main__':
    app = QApplication(sys.argv)
    window = QMainWindow()
    window.setGeometry(500,500,400,400)
    window.setStyleSheet('background-color:#373737')
    line = LineEdit(parent=window)
    line.setGeometry(20,200,400,40)
    style = LineEditStyle()
    line.setStyle(style)
    window.show()
    sys.exit(app.exec())

您不能将 QStyleSheet 与 QStyle 一起使用,因为它会造成混淆,您必须将默认参数 Widget 设置为 None

from PyQt5.QtWidgets import *
from PyQt5.QtCore import *
from PyQt5.QtGui import *
from PushButton_2 import Push_Button_
import sys

class LineEditStyle(QProxyStyle):
    def drawPrimitive(self, element, option, painter, widget=None,a=None):
        if a :
            self.pen = QPen(QColor('green'))
        else :
            self.pen = QPen(QColor('red'))
        self.pen.setWidth(4)
        if element == QStyle.PE_FrameLineEdit:
            painter.setRenderHint(QPainter.Antialiasing)
            painter.setPen(self.pen)
            painter.drawRoundedRect(QRect(0,0,400,40), 10, 10)
        else:
            super().drawPrimitive(element, option, painter, widget)
    
    def subElementRect(self, element, option, widget):
        if element == QStyle.SE_LineEditContents :
            return QRect(0,0,50,30)
        else :
            return super().subElementRect(element, option, widget)
    
    def drawItemText(self, painter, rect, flags, pal, enabled, text, textRole):
        rect_ = QRect(20,20,50,50)
        text = text.upper()
        painter.drawText(text,rect_,Qt.AlignCenter)
        
class LineEdit(QLineEdit):
    def __init__(self,*args,**kwargs):
        QLineEdit.__init__(self,*args,**kwargs)
        self.a = 0
    
    def enterEvent(self, a0):
        self.a = 1

    def leaveEvent(self, a0):
        self.a = 0
        
    def paintEvent(self,event):
        option = QStyleOption()
        option.initFrom(self)
        painter = QPainter(self)
        self.style().drawPrimitive(QStyle.PE_FrameLineEdit,option,painter,a=self.a)


if __name__ == '__main__':
    app = QApplication(sys.argv)
    window = QMainWindow()
    window.setGeometry(500,500,400,400)
    #window.setStyleSheet('background-color:#373737')
    line = LineEdit(parent=window)
    line.setGeometry(20,200,400,40)
    style = LineEditStyle()
    line.setStyle(style)
    window.show()
    sys.exit(app.exec())

您可能会发现 QStyleSheets 对这样的事情很有用。

我通过在 styleSheet 属性 中输入以下内容在 Qt Designer 中对此进行了模拟(在代码中您会执行 mywidget.setStyleSheet('<parameters>') ):

QLineEdit {
    border: 3px solid red;
}
QLineEdit:focus {
    border: 3px solid green;
}

编辑:上面的 styleSheet 字符串适用于焦点,但最初的问题是关于 enterEvent/leaveEvent,它在鼠标悬停时触发。正如@musicamente 正确指出的那样,要在鼠标悬停时工作,可以使用 :hover pseudo state 而不是 :focus:

QLineEdit {
    border: 3px solid red;
}
QLineEdit:hover {
    border: 3px solid green;
}

我用 2 个 QLineEdits 制作了一个 QMainWindow:1 个设置了 styleSheet,另一个是默认设置。当焦点移动到常规QLineEdit时,修改后的QLineEdit变为红色。

您可以通过编辑父级的样式表使 QLineEdits 的 ALL 行为相同。在您的情况下,您可以使用 window.setStyleSheet('....。这是 Qt Designer 中的另一个模型: