nysh/src/widgets/notifications/NotificationToast.qml

163 lines
4.7 KiB
QML

pragma ComponentBehavior: Bound
import QtQuick
import QtQuick.Layouts
import QtQuick.Controls
import Quickshell
import Quickshell.Widgets
import "../../base"
import "../../provider"
import Quickshell.Services.Notifications
MouseArea {
id: toast
required property Notification notif
property int actionHeight: 30
property int expansionSpeed: 200
property bool showTimeBar: false
height: box.height
hoverEnabled: true
signal close
BRectangle {
id: box
width: parent.width
height: header.height + toast.actionHeight + bodyBox.height + (5 * 3)
clip: true
Item {
id: inner
anchors.margins: 5
anchors.fill: parent
RowLayout {
id: header
width: parent.width
height: 25
IconImage {
source: toast.notif.appIcon ? Quickshell.iconPath(toast.notif.appIcon) : ""
height: parent.height
width: height
visible: toast.notif.appIcon
}
Text {
text: `${toast.notif.appIcon ? "" : `${toast.notif.appName}:`} ${toast.notif.summary}`
Layout.fillWidth: true
elide: Text.ElideRight
font.pointSize: 12.5
}
Button {
onClicked: toast.close()
height: 16
width: 16
}
}
Rectangle {
id: bodyBox
width: parent.width
anchors.top: header.bottom
height: 60
clip: true
property int maxHeight: 0
color: "transparent"
Text {
id: text
anchors.topMargin: 5
text: toast.notif.body
width: parent.width
height: parent.height
wrapMode: Text.Wrap
elide: Text.ElideRight
font.pointSize: 12.5
Component.onCompleted: () => {
if (text.implicitHeight < bodyBox.height) {
bodyBox.height = text.implicitHeight;
}
bodyBox.maxHeight = Qt.binding(() => text.implicitHeight);
}
}
states: State {
name: "expand"
when: toast.containsMouse
PropertyChanges {
target: bodyBox
height: bodyBox.maxHeight
}
}
transitions: Transition {
NumberAnimation {
properties: "height"
duration: toast.expansionSpeed
}
}
}
RowLayout {
id: actions
width: parent.width
anchors.top: bodyBox.bottom
anchors.topMargin: 5
anchors.bottomMargin: 5
Repeater {
id: rep
model: toast.notif.actions
delegate: NotificationToastAction {
required property var modelData
notifAction: modelData
hasIcons: toast.notif.hasActionIcons
height: toast.actionHeight
}
}
visible: toast?.notif.actions ? true : false
}
}
NumberAnimation on width {
duration: toast.expansionSpeed
}
Rectangle {
id: timeBar
visible: toast.showTimeBar
anchors.margins: 2
anchors.bottom: box.bottom
anchors.right: box.right
width: box.width - box.border.width - anchors.margins
height: 2
bottomLeftRadius: box.radius
bottomRightRadius: box.radius
color: {
switch (toast.urgency) {
case NotificationUrgency.Critical:
return "red";
break;
case NotificationUrgency.Normal:
return "green";
break;
default:
return "white";
}
}
NumberAnimation on width {
to: 0
duration: Config.notifications.toastDuration
paused: toast.containsMouse && timeBar.visible
running: timeBar.visible
}
}
}
}