Продолжительность перехода при наведении на кнопку PyQt5 - PullRequest
1 голос
/ 28 февраля 2020

Я некоторое время работал над небольшим проектом в PyQt5, и я хотел попытаться сделать его немного лучше, используя таблицу стилей. Для тестирования я скопировал пример из W3Schools в новый проект, чтобы посмотреть, сработает ли он. Примером является кнопка, где фон исчезает с белого на зеленый. Все работает, кроме того, что вместо исчезновения, переход происходит мгновенно. Мои вопросы, как я могу получить этот эффект исчезновения? Мне также любопытно, как я могу получить такой же эффект перехода, но с другими аспектами кнопки, такими как размер или форма.

Я провел большой поиск, но все релевантное, что я нашел немного вне моей сферы знаний. Что я знаю, так это то, что мне, вероятно, придется кодировать его вместо использования таблицы стилей, но я не совсем уверен, как это сделать. Я думал, что смогу сделать функцию для постоянного изменения цвета, но я также думаю, что, вероятно, есть лучший способ сделать это.

Это код, который генерируется, когда я конвертирую файл .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_())

Это то, что находится в таблице стилей, но более читабельно.

.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;
}

Вот изображение кнопки.

Перед наведением

и после

1 Ответ

1 голос
/ 28 февраля 2020

Если код OP выполняется из терминала / CMD, появляется следующее предупреждение:

Unknown property display
Unknown property transition-duration
Unknown property cursor
...

И почему вы указываете, что эти свойства неизвестны? Поскольку таблица стилей Qt не CSS, QSS основана на CSS2.1 и имеет ограниченный код свойств (см. здесь для получения дополнительной информации), так что такие свойства, как отображение, длительность перехода и курсор, не определено в этом стандарте, поэтому он не работает.

Если вы хотите сделать анимацию, вам следует использовать некоторую QXAnimation, в этом случае я буду использовать QVariantAnimation, который начинает работать в методах enterEvent() и 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_())
...