import QtQuick import QtQuick.Layouts import QtQuick.Controls import Quickshell import Quickshell.Wayland import Quickshell.Services.Notifications import qs.Commons import qs.Services import qs.Widgets // Notification History panel NPanel { id: root preferredWidth: 380 * Style.uiScaleRatio preferredHeight: 480 * Style.uiScaleRatio onOpened: function () { NotificationService.updateLastSeenTs() } panelContent: Rectangle { id: notificationRect color: Color.transparent ColumnLayout { anchors.fill: parent anchors.margins: Style.marginL spacing: Style.marginM // Header section NBox { Layout.fillWidth: true implicitHeight: headerRow.implicitHeight + (Style.marginM * 2) RowLayout { id: headerRow anchors.fill: parent anchors.margins: Style.marginM spacing: Style.marginM NIcon { icon: "bell" pointSize: Style.fontSizeXXL color: Color.mPrimary } NText { text: I18n.tr("notifications.panel.title") pointSize: Style.fontSizeL font.weight: Style.fontWeightBold color: Color.mOnSurface Layout.fillWidth: true } NIconButton { icon: Settings.data.notifications.doNotDisturb ? "bell-off" : "bell" tooltipText: Settings.data.notifications.doNotDisturb ? I18n.tr("tooltips.do-not-disturb-enabled") : I18n.tr("tooltips.do-not-disturb-disabled") baseSize: Style.baseWidgetSize * 0.8 onClicked: Settings.data.notifications.doNotDisturb = !Settings.data.notifications.doNotDisturb } NIconButton { icon: "trash" tooltipText: I18n.tr("tooltips.clear-history") baseSize: Style.baseWidgetSize * 0.8 onClicked: { NotificationService.clearHistory() // Close panel as there is nothing more to see. root.close() } } NIconButton { icon: "close" tooltipText: I18n.tr("tooltips.close") baseSize: Style.baseWidgetSize * 0.8 onClicked: root.close() } } } // Empty state when no notifications ColumnLayout { Layout.fillWidth: true Layout.fillHeight: true Layout.alignment: Qt.AlignHCenter visible: NotificationService.historyList.count === 0 spacing: Style.marginL Item { Layout.fillHeight: true } NIcon { icon: "bell-off" pointSize: 64 color: Color.mOnSurfaceVariant Layout.alignment: Qt.AlignHCenter } NText { text: I18n.tr("notifications.panel.no-notifications") pointSize: Style.fontSizeL color: Color.mOnSurfaceVariant Layout.alignment: Qt.AlignHCenter } NText { text: I18n.tr("notifications.panel.description") pointSize: Style.fontSizeS color: Color.mOnSurfaceVariant Layout.alignment: Qt.AlignHCenter Layout.fillWidth: true wrapMode: Text.Wrap horizontalAlignment: Text.AlignHCenter } Item { Layout.fillHeight: true } } // Notification list NListView { id: notificationList Layout.fillWidth: true Layout.fillHeight: true horizontalPolicy: ScrollBar.AlwaysOff verticalPolicy: ScrollBar.AsNeeded model: NotificationService.historyList spacing: Style.marginM clip: true boundsBehavior: Flickable.StopAtBounds visible: NotificationService.historyList.count > 0 // Track which notification is expanded property string expandedId: "" delegate: Rectangle { property string notificationId: model.id property bool isExpanded: notificationList.expandedId === notificationId width: notificationList.width height: notificationLayout.implicitHeight + (Style.marginM * 2) radius: Style.radiusM color: Color.mSurfaceVariant border.color: Qt.alpha(Color.mOutline, Style.opacityMedium) border.width: Style.borderS Behavior on height { enabled: !Settings.data.general.animationDisabled NumberAnimation { duration: Style.animationNormal easing.type: Easing.InOutQuad } } // Smooth color transition on hover Behavior on color { enabled: !Settings.data.general.animationDisabled ColorAnimation { duration: Style.animationFast } } // Click to expand/collapse MouseArea { anchors.fill: parent // Don't capture clicks on the delete button anchors.rightMargin: 48 enabled: (summaryText.truncated || bodyText.truncated) onClicked: { if (notificationList.expandedId === notificationId) { notificationList.expandedId = "" } else { notificationList.expandedId = notificationId } } cursorShape: enabled ? Qt.PointingHandCursor : Qt.ArrowCursor } RowLayout { id: notificationLayout anchors.fill: parent anchors.margins: Style.marginM spacing: Style.marginM ColumnLayout { NImageCircled { Layout.preferredWidth: Math.round(40 * Style.uiScaleRatio) Layout.preferredHeight: Math.round(40 * Style.uiScaleRatio) Layout.alignment: Qt.AlignTop Layout.topMargin: 20 imagePath: model.cachedImage || model.originalImage || "" borderColor: Color.transparent borderWidth: 0 fallbackIcon: "bell" fallbackIconSize: 24 } Item { Layout.fillHeight: true } } // Notification content column ColumnLayout { Layout.fillWidth: true Layout.alignment: Qt.AlignTop spacing: Style.marginXS Layout.rightMargin: -(Style.marginM + Style.baseWidgetSize * 0.6) // Header row with app name and timestamp RowLayout { Layout.fillWidth: true spacing: Style.marginS // Urgency indicator Rectangle { Layout.preferredWidth: 6 Layout.preferredHeight: 6 Layout.alignment: Qt.AlignVCenter radius: 3 visible: model.urgency !== 1 color: { if (model.urgency === 2) return Color.mError else if (model.urgency === 0) return Color.mOnSurfaceVariant else return Color.transparent } } NText { text: model.appName || "Unknown App" pointSize: Style.fontSizeXS color: Color.mSecondary } NText { text: Time.formatRelativeTime(model.timestamp) pointSize: Style.fontSizeXS color: Color.mSecondary } Item { Layout.fillWidth: true } } // Summary NText { id: summaryText text: model.summary || I18n.tr("general.no-summary") pointSize: Style.fontSizeM font.weight: Font.Medium color: Color.mOnSurface textFormat: Text.PlainText wrapMode: Text.Wrap Layout.fillWidth: true maximumLineCount: isExpanded ? 999 : 2 elide: Text.ElideRight } // Body NText { id: bodyText text: model.body || "" pointSize: Style.fontSizeS color: Color.mOnSurfaceVariant textFormat: Text.PlainText wrapMode: Text.Wrap Layout.fillWidth: true maximumLineCount: isExpanded ? 999 : 3 elide: Text.ElideRight visible: text.length > 0 } // Spacer for expand indicator Item { Layout.fillWidth: true Layout.preferredHeight: (!isExpanded && (summaryText.truncated || bodyText.truncated)) ? (Style.marginS) : 0 } // Expand indicator RowLayout { Layout.fillWidth: true visible: !isExpanded && (summaryText.truncated || bodyText.truncated) spacing: Style.marginXS Item { Layout.fillWidth: true } NText { text: I18n.tr("notifications.panel.click-to-expand") || "Click to expand" pointSize: Style.fontSizeXS color: Color.mPrimary font.weight: Font.Medium } NIcon { icon: "chevron-down" pointSize: Style.fontSizeS color: Color.mPrimary } } } // Delete button NIconButton { icon: "trash" tooltipText: I18n.tr("tooltips.delete-notification") baseSize: Style.baseWidgetSize * 0.7 Layout.alignment: Qt.AlignTop onClicked: { // Remove from history using the service API NotificationService.removeFromHistory(notificationId) } } } } } } } }