按钮悬停过渡持续时间
Button hover transition duration
我已经在 PyQt5 中开发了一个小项目,我想尝试使用 sheet 风格让它看起来更好一些。为了测试,我将 an example from W3Schools 复制到一个新项目中,只是为了看看它是否可行。该示例是一个背景从白色渐变为绿色的按钮。一切正常,除了不褪色,过渡立即发生。我的问题是如何获得淡入淡出效果?我也很好奇如何获得相同类型的过渡效果,但按钮的其他方面(例如大小或形状)。
我进行了大量搜索,但我发现的所有相关内容都超出了我的知识范围。我所知道的是,我可能必须对其进行编码,而不是使用 sheet 样式,但我不确定该怎么做。我在想我可以做一个函数来不断改变颜色,但我也认为可能有更好的方法来做到这一点。
这是我转换.ui文件时生成的代码。
from PyQt5 import QtCore, QtGui, QtWidgets
class Ui_Dialog(object):
def setupUi(self, Dialog):
Dialog.setObjectName("Dialog")
Dialog.resize(400, 300)
self.pushButton = QtWidgets.QPushButton(Dialog)
self.pushButton.setGeometry(QtCore.QRect(10, 10, 381, 281))
self.pushButton.setMouseTracking(False)
self.pushButton.setAutoFillBackground(False)
self.pushButton.setStyleSheet(".QPushButton{\n"
"background-color: #4CAF50;\n"
"border: none;\n"
"color: white;\n"
"padding: 16px 32px;\n"
"text-align: center;\n"
"text-decoration: none;\n"
"display: inline-block;\n"
"font-size: 16px;\n"
"margin: 4px 2px;\n"
"transition-duration: 0.4s;\n"
"cursor: pointer;\n"
"}\n"
"\n"
"#pushButton{\n"
"background-color: white;\n"
"color: black;\n"
"border: 2px solid #4CAF50;\n"
"}\n"
"\n"
"#pushButton:hover{\n"
"background-color: #4CAF50;\n"
"color: white;\n"
"}")
self.pushButton.setObjectName("pushButton")
self.retranslateUi(Dialog)
QtCore.QMetaObject.connectSlotsByName(Dialog)
def retranslateUi(self, Dialog):
_translate = QtCore.QCoreApplication.translate
Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
self.pushButton.setText(_translate("Dialog", "Click Here"))
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
Dialog = QtWidgets.QDialog()
ui = Ui_Dialog()
ui.setupUi(Dialog)
Dialog.show()
sys.exit(app.exec_())
这是 sheet 的风格,但更具可读性。
.QPushButton{
background-color: #4CAF50;
border: none;
color: white;
padding: 16px 32px;
text-align: center;
text-decoration: none;
display: inline-block;
font-size: 16px;
margin: 4px 2px;
transition-duration: 0.4s;
cursor: pointer;
}
#pushButton{
background-color: white;
color: black;
border: 2px solid #4CAF50;
}
#pushButton:hover{
background-color: #4CAF50;
color: white;
}
这是按钮的图片。
悬停前
及之后
如果从 terminal/CMD 执行 OP 代码,会得到以下警告:
Unknown property display
Unknown property transition-duration
Unknown property cursor
...
你为什么指出这些属性是未知的?由于 Qt Style Sheet 不是 CSS,QSS 是基于 CSS2.1 并且具有有限的属性代码(更多信息请参见 here)以便显示等属性, transition-duration 和 cursor 在该标准中没有定义,所以它不起作用。
如果你想制作动画你应该使用一些 QXAnimation,在这种情况下我将使用 QVariantAnimation
that starts to run in the enterEvent()
and leaveEvent()
方法:
from PyQt5 import QtCore, QtGui, QtWidgets
class PushButton(QtWidgets.QPushButton):
def __init__(self, parent=None):
super().__init__(parent)
self._animation = QtCore.QVariantAnimation(
startValue=QtGui.QColor("#4CAF50"),
endValue=QtGui.QColor("white"),
valueChanged=self._on_value_changed,
duration=400,
)
self._update_stylesheet(QtGui.QColor("white"), QtGui.QColor("black"))
self.setCursor(QtGui.QCursor(QtCore.Qt.PointingHandCursor))
def _on_value_changed(self, color):
foreground = (
QtGui.QColor("black")
if self._animation.direction() == QtCore.QAbstractAnimation.Forward
else QtGui.QColor("white")
)
self._update_stylesheet(color, foreground)
def _update_stylesheet(self, background, foreground):
self.setStyleSheet(
"""
QPushButton{
background-color: %s;
border: none;
color: %s;
padding: 16px 32px;
text-align: center;
text-decoration: none;
font-size: 16px;
margin: 4px 2px;
border: 2px solid #4CAF50;
}
"""
% (background.name(), foreground.name())
)
def enterEvent(self, event):
self._animation.setDirection(QtCore.QAbstractAnimation.Backward)
self._animation.start()
super().enterEvent(event)
def leaveEvent(self, event):
self._animation.setDirection(QtCore.QAbstractAnimation.Forward)
self._animation.start()
super().leaveEvent(event)
class Dialog(QtWidgets.QDialog):
def __init__(self, parent=None):
super().__init__(parent)
self.setWindowTitle(self.tr("Dialog"))
self.pushButton = PushButton()
self.pushButton.setText(self.tr("Click Here"))
self.pushButton.setSizePolicy(
QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Minimum
)
lay = QtWidgets.QVBoxLayout(self)
lay.addWidget(self.pushButton)
self.resize(400, 300)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = Dialog()
w.show()
sys.exit(app.exec_())
我已经在 PyQt5 中开发了一个小项目,我想尝试使用 sheet 风格让它看起来更好一些。为了测试,我将 an example from W3Schools 复制到一个新项目中,只是为了看看它是否可行。该示例是一个背景从白色渐变为绿色的按钮。一切正常,除了不褪色,过渡立即发生。我的问题是如何获得淡入淡出效果?我也很好奇如何获得相同类型的过渡效果,但按钮的其他方面(例如大小或形状)。
我进行了大量搜索,但我发现的所有相关内容都超出了我的知识范围。我所知道的是,我可能必须对其进行编码,而不是使用 sheet 样式,但我不确定该怎么做。我在想我可以做一个函数来不断改变颜色,但我也认为可能有更好的方法来做到这一点。
这是我转换.ui文件时生成的代码。
from PyQt5 import QtCore, QtGui, QtWidgets
class Ui_Dialog(object):
def setupUi(self, Dialog):
Dialog.setObjectName("Dialog")
Dialog.resize(400, 300)
self.pushButton = QtWidgets.QPushButton(Dialog)
self.pushButton.setGeometry(QtCore.QRect(10, 10, 381, 281))
self.pushButton.setMouseTracking(False)
self.pushButton.setAutoFillBackground(False)
self.pushButton.setStyleSheet(".QPushButton{\n"
"background-color: #4CAF50;\n"
"border: none;\n"
"color: white;\n"
"padding: 16px 32px;\n"
"text-align: center;\n"
"text-decoration: none;\n"
"display: inline-block;\n"
"font-size: 16px;\n"
"margin: 4px 2px;\n"
"transition-duration: 0.4s;\n"
"cursor: pointer;\n"
"}\n"
"\n"
"#pushButton{\n"
"background-color: white;\n"
"color: black;\n"
"border: 2px solid #4CAF50;\n"
"}\n"
"\n"
"#pushButton:hover{\n"
"background-color: #4CAF50;\n"
"color: white;\n"
"}")
self.pushButton.setObjectName("pushButton")
self.retranslateUi(Dialog)
QtCore.QMetaObject.connectSlotsByName(Dialog)
def retranslateUi(self, Dialog):
_translate = QtCore.QCoreApplication.translate
Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
self.pushButton.setText(_translate("Dialog", "Click Here"))
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
Dialog = QtWidgets.QDialog()
ui = Ui_Dialog()
ui.setupUi(Dialog)
Dialog.show()
sys.exit(app.exec_())
这是 sheet 的风格,但更具可读性。
.QPushButton{
background-color: #4CAF50;
border: none;
color: white;
padding: 16px 32px;
text-align: center;
text-decoration: none;
display: inline-block;
font-size: 16px;
margin: 4px 2px;
transition-duration: 0.4s;
cursor: pointer;
}
#pushButton{
background-color: white;
color: black;
border: 2px solid #4CAF50;
}
#pushButton:hover{
background-color: #4CAF50;
color: white;
}
这是按钮的图片。
悬停前
及之后
如果从 terminal/CMD 执行 OP 代码,会得到以下警告:
Unknown property display
Unknown property transition-duration
Unknown property cursor
...
你为什么指出这些属性是未知的?由于 Qt Style Sheet 不是 CSS,QSS 是基于 CSS2.1 并且具有有限的属性代码(更多信息请参见 here)以便显示等属性, transition-duration 和 cursor 在该标准中没有定义,所以它不起作用。
如果你想制作动画你应该使用一些 QXAnimation,在这种情况下我将使用 QVariantAnimation
that starts to run in the enterEvent()
and leaveEvent()
方法:
from PyQt5 import QtCore, QtGui, QtWidgets
class PushButton(QtWidgets.QPushButton):
def __init__(self, parent=None):
super().__init__(parent)
self._animation = QtCore.QVariantAnimation(
startValue=QtGui.QColor("#4CAF50"),
endValue=QtGui.QColor("white"),
valueChanged=self._on_value_changed,
duration=400,
)
self._update_stylesheet(QtGui.QColor("white"), QtGui.QColor("black"))
self.setCursor(QtGui.QCursor(QtCore.Qt.PointingHandCursor))
def _on_value_changed(self, color):
foreground = (
QtGui.QColor("black")
if self._animation.direction() == QtCore.QAbstractAnimation.Forward
else QtGui.QColor("white")
)
self._update_stylesheet(color, foreground)
def _update_stylesheet(self, background, foreground):
self.setStyleSheet(
"""
QPushButton{
background-color: %s;
border: none;
color: %s;
padding: 16px 32px;
text-align: center;
text-decoration: none;
font-size: 16px;
margin: 4px 2px;
border: 2px solid #4CAF50;
}
"""
% (background.name(), foreground.name())
)
def enterEvent(self, event):
self._animation.setDirection(QtCore.QAbstractAnimation.Backward)
self._animation.start()
super().enterEvent(event)
def leaveEvent(self, event):
self._animation.setDirection(QtCore.QAbstractAnimation.Forward)
self._animation.start()
super().leaveEvent(event)
class Dialog(QtWidgets.QDialog):
def __init__(self, parent=None):
super().__init__(parent)
self.setWindowTitle(self.tr("Dialog"))
self.pushButton = PushButton()
self.pushButton.setText(self.tr("Click Here"))
self.pushButton.setSizePolicy(
QtWidgets.QSizePolicy.Minimum, QtWidgets.QSizePolicy.Minimum
)
lay = QtWidgets.QVBoxLayout(self)
lay.addWidget(self.pushButton)
self.resize(400, 300)
if __name__ == "__main__":
import sys
app = QtWidgets.QApplication(sys.argv)
w = Dialog()
w.show()
sys.exit(app.exec_())