// Config created by Keyitdev https://github.com/Keyitdev/sddm-astronaut-theme // Copyright (C) 2022-2025 Keyitdev // Based on https://github.com/MarianArlt/sddm-sugar-dark // Distributed under the GPLv3+ License https://www.gnu.org/licenses/gpl-3.0.html import QtQuick 2.15 import QtQuick.Layouts 1.15 import QtQuick.Controls 2.15 Column { id: inputContainer Layout.fillWidth: true property ComboBox exposeSession: sessionSelect.exposeSession property bool failed Item { id: errorMessageField // change also in selectSession height: root.font.pointSize * 2 width: parent.width / 2 anchors.horizontalCenter: parent.horizontalCenter Label { id: errorMessage width: parent.width horizontalAlignment: Text.AlignHCenter text: failed ? config.TranslateLoginFailedWarning || textConstants.loginFailed + "!" : keyboard.capsLock ? config.TranslateCapslockWarning || textConstants.capslockWarning : null font.pointSize: root.font.pointSize * 0.8 font.italic: true color: config.WarningColor opacity: 0 states: [ State { name: "fail" when: failed PropertyChanges { target: errorMessage opacity: 1 } }, State { name: "capslock" when: keyboard.capsLock PropertyChanges { target: errorMessage opacity: 1 } } ] transitions: [ Transition { PropertyAnimation { properties: "opacity" duration: 100 } } ] } } Item { id: usernameField height: root.font.pointSize * 4.5 width: parent.width / 2 anchors.horizontalCenter: parent.horizontalCenter ComboBox { id: selectUser width: parent.height height: parent.height anchors.left: parent.left z: 2 model: userModel currentIndex: model.lastIndex textRole: "name" hoverEnabled: true onActivated: { username.text = currentText } property var popkey: config.RightToLeftLayout == "true" ? Qt.Key_Right : Qt.Key_Left Keys.onPressed: function(event) { if (event.key == Qt.Key_Down && !popup.opened) username.forceActiveFocus(); if ((event.key == Qt.Key_Up || event.key == popkey) && !popup.opened) popup.open(); } KeyNavigation.down: username KeyNavigation.right: username delegate: ItemDelegate { // minus padding width: popupHandler.width - 20 anchors.horizontalCenter: popupHandler.horizontalCenter contentItem: Text { verticalAlignment: Text.AlignVCenter horizontalAlignment: Text.AlignHCenter text: model.name font.pointSize: root.font.pointSize * 0.8 font.capitalization: Font.AllLowercase font.family: root.font.family color: config.DropdownTextColor } background: Rectangle { color: selectUser.highlightedIndex === index ? config.DropdownSelectedBackgroundColor : "transparent" } } indicator: Button { id: usernameIcon width: selectUser.height * 1 height: parent.height anchors.left: parent.left anchors.verticalCenter: parent.verticalCenter anchors.leftMargin: selectUser.height * 0 icon.height: parent.height * 0.25 icon.width: parent.height * 0.25 enabled: false icon.color: config.UserIconColor icon.source: Qt.resolvedUrl("../Assets/User.svg") background: Rectangle { color: "transparent" border.color: "transparent" } } background: Rectangle { color: "transparent" border.color: "transparent" } popup: Popup { id: popupHandler implicitHeight: contentItem.implicitHeight width: usernameField.width y: parent.height - username.height / 3 x: config.RightToLeftLayout == "true" ? -loginButton.width + selectUser.width : 0 rightMargin: config.RightToLeftLayout == "true" ? root.padding + usernameField.width / 2 : undefined padding: 10 contentItem: ListView { implicitHeight: contentHeight + 20 clip: true model: selectUser.popup.visible ? selectUser.delegateModel : null currentIndex: selectUser.highlightedIndex ScrollIndicator.vertical: ScrollIndicator { } } background: Rectangle { radius: config.RoundCorners / 2 color: config.DropdownBackgroundColor layer.enabled: true } enter: Transition { NumberAnimation { property: "opacity"; from: 0; to: 1 } } } states: [ State { name: "pressed" when: selectUser.down PropertyChanges { target: usernameIcon icon.color: Qt.lighter(config.HoverUserIconColor, 1.1) } }, State { name: "hovered" when: selectUser.hovered PropertyChanges { target: usernameIcon icon.color: Qt.lighter(config.HoverUserIconColor, 1.2) } }, State { name: "focused" when: selectUser.activeFocus PropertyChanges { target: usernameIcon icon.color: config.HoverUserIconColor } } ] transitions: [ Transition { PropertyAnimation { properties: "color, border.color, icon.color" duration: 150 } } ] } TextField { id: username anchors.centerIn: parent height: root.font.pointSize * 3 width: parent.width horizontalAlignment: TextInput.AlignHCenter z: 1 text: config.ForceLastUser == "true" ? selectUser.currentText : null color: config.LoginFieldTextColor font.bold: true font.capitalization: config.AllowUppercaseLettersInUsernames == "false" ? Font.AllLowercase : Font.MixedCase placeholderText: config.TranslatePlaceholderUsername || textConstants.userName placeholderTextColor: config.PlaceholderTextColor selectByMouse: true renderType: Text.QtRendering onFocusChanged:{ if(focus) selectAll() } background: Rectangle { color: config.LoginFieldBackgroundColor opacity: 0.6 border.color: "transparent" border.width: parent.activeFocus ? 2 : 1 radius: config.RoundCorners || 0 } onAccepted: config.AllowUppercaseLettersInUsernames == "false" ? sddm.login(username.text.toLowerCase(), password.text, sessionSelect.selectedSession) : sddm.login(username.text, password.text, sessionSelect.selectedSession) KeyNavigation.down: passwordIcon states: [ State { name: "focused" when: username.activeFocus PropertyChanges { target: username.background border.color: config.HighlightBorderColor } PropertyChanges { target: username color: Qt.lighter(config.LoginFieldTextColor, 1.15) } } ] } } Item { id: passwordField height: root.font.pointSize * 4.5 width: parent.width / 2 anchors.horizontalCenter: parent.horizontalCenter Button { id: passwordIcon height: parent.height width: selectUser.height * 1 anchors.left: parent.left anchors.leftMargin: selectUser.height * 0 anchors.verticalCenter: parent.verticalCenter z: 2 icon.height: parent.height * 0.25 icon.width: parent.height * 0.25 icon.color: config.PasswordIconColor icon.source: Qt.resolvedUrl("../Assets/Password2.svg") background: Rectangle { color: "transparent" border.color: "transparent" } states: [ State { name: "visiblePasswordFocused" when: passwordIcon.checked && passwordIcon.activeFocus PropertyChanges { target: passwordIcon icon.source: Qt.resolvedUrl("../Assets/Password.svg") icon.color: config.HoverPasswordIconColor } }, State { name: "visiblePasswordHovered" when: passwordIcon.checked && passwordIcon.hovered PropertyChanges { target: passwordIcon icon.source: Qt.resolvedUrl("../Assets/Password.svg") icon.color: config.HoverPasswordIconColor } }, State { name: "visiblePassword" when: passwordIcon.checked PropertyChanges { target: passwordIcon icon.source: Qt.resolvedUrl("../Assets/Password.svg") } }, State { name: "hiddenPasswordFocused" when: passwordIcon.enabled && passwordIcon.activeFocus PropertyChanges { target: passwordIcon icon.source: Qt.resolvedUrl("../Assets/Password2.svg") icon.color: config.HoverPasswordIconColor } }, State { name: "hiddenPasswordHovered" when: passwordIcon.hovered PropertyChanges { target: passwordIcon icon.source: Qt.resolvedUrl("../Assets/Password2.svg") icon.color: config.HoverPasswordIconColor } } ] onClicked: toggle() Keys.onReturnPressed: toggle() Keys.onEnterPressed: toggle() KeyNavigation.down: password } TextField { id: password height: root.font.pointSize * 3 width: parent.width anchors.centerIn: parent horizontalAlignment: TextInput.AlignHCenter font.bold: true color: config.PasswordFieldTextColor focus: config.PasswordFocus == "true" ? true : false echoMode: passwordIcon.checked ? TextInput.Normal : TextInput.Password placeholderText: config.TranslatePlaceholderPassword || textConstants.password placeholderTextColor: config.PlaceholderTextColor passwordCharacter: "•" passwordMaskDelay: config.HideCompletePassword == "true" ? undefined : 1000 renderType: Text.QtRendering selectByMouse: true background: Rectangle { color: config.PasswordFieldBackgroundColor opacity: 0.6 border.color: "transparent" border.width: parent.activeFocus ? 2 : 1 radius: config.RoundCorners || 0 } onAccepted: config.AllowUppercaseLettersInUsernames == "false" ? sddm.login(username.text.toLowerCase(), password.text, sessionSelect.selectedSession) : sddm.login(username.text, password.text, sessionSelect.selectedSession) KeyNavigation.down: loginButton } states: [ State { name: "focused" when: password.activeFocus PropertyChanges { target: password.background border.color: config.HighlightBorderColor } PropertyChanges { target: password color: Qt.lighter(config.LoginFieldTextColor, 1.15) } } ] transitions: [ Transition { PropertyAnimation { properties: "color, border.color" duration: 150 } } ] } Item { id: login // important // try 4 or 9 ... height: root.font.pointSize * 9 width: parent.width / 2 anchors.horizontalCenter: parent.horizontalCenter visible: config.HideLoginButton == "true" ? false : true Button { id: loginButton height: root.font.pointSize * 3 implicitWidth: parent.width anchors.horizontalCenter: parent.horizontalCenter anchors.verticalCenter: parent.verticalCenter text: config.TranslateLogin || textConstants.login enabled: config.AllowEmptyPassword == "true" || username.text != "" && password.text != "" ? true : false hoverEnabled: true contentItem: Text { horizontalAlignment: Text.AlignHCenter verticalAlignment: Text.AlignVCenter font.bold: true font.pointSize: root.font.pointSize font.family: root.font.family color: config.LoginButtonTextColor text: parent.text opacity: 0.0 } background: Rectangle { id: buttonBackground color: config.LoginButtonBackgroundColor opacity: 0.0 radius: config.RoundCorners || 0 } states: [ State { name: "pressed" when: loginButton.down PropertyChanges { target: buttonBackground color: Qt.darker(config.LoginButtonBackgroundColor, 1.1) opacity: 0.0 } PropertyChanges { target: loginButton.contentItem } }, State { name: "hovered" when: loginButton.hovered PropertyChanges { target: buttonBackground color: Qt.lighter(config.LoginButtonBackgroundColor, 1.15) opacity: 0.0 } PropertyChanges { target: loginButton.contentItem opacity: 0.0 } }, State { name: "focused" when: loginButton.activeFocus PropertyChanges { target: buttonBackground color: Qt.lighter(config.LoginButtonBackgroundColor, 1.2) opacity: 0.0 } PropertyChanges { target: loginButton.contentItem opacity: 0.0 } }, State { name: "enabled" when: loginButton.enabled PropertyChanges { target: buttonBackground; color: config.LoginButtonBackgroundColor; opacity: 0.0 } PropertyChanges { target: loginButton.contentItem; opacity: 0.0 } } ] transitions: [ Transition { PropertyAnimation { properties: "opacity, color"; duration: 300 } } ] onClicked: config.AllowUppercaseLettersInUsernames == "false" ? sddm.login(username.text.toLowerCase(), password.text, sessionSelect.selectedSession) : sddm.login(username.text, password.text, sessionSelect.selectedSession) Keys.onReturnPressed: clicked() Keys.onEnterPressed: clicked() } } Connections { target: sddm function onLoginSucceeded() {} function onLoginFailed() { failed = true resetError.running ? resetError.stop() && resetError.start() : resetError.start() } } Timer { id: resetError interval: 2000 onTriggered: failed = false running: false } }