mirror of
https://invent.kde.org/marcoa/shift-shell.git
synced 2026-04-26 22:33:08 +00:00
only allow a phone call notification to execute actions without unlocking for the rest open the pin dialog and execute the action only after successful authentication
226 lines
8.5 KiB
QML
226 lines
8.5 KiB
QML
/*
|
|
Copyright (C) 2020 Devin Lin <espidev@gmail.com>
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2 of the License, or
|
|
(at your option) any later version.
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import QtQuick 2.12
|
|
import QtQuick.Controls 1.1
|
|
import QtQuick.Layouts 1.1
|
|
import QtGraphicalEffects 1.12
|
|
|
|
import org.kde.plasma.components 2.0 as PlasmaComponents
|
|
import org.kde.plasma.components 3.0 as PlasmaComponents3
|
|
import org.kde.plasma.core 2.0 as PlasmaCore
|
|
import org.kde.plasma.extras 2.0 as PlasmaExtras
|
|
import org.kde.notificationmanager 1.1 as Notifications
|
|
|
|
import org.kde.kquickcontrolsaddons 2.0 as KQCAddons
|
|
|
|
import "../components"
|
|
|
|
// meant to be temporary, until the notifications components in plasma-workspace are available to used
|
|
// https://invent.kde.org/plasma/plasma-workspace/-/blob/master/applets/notifications/package/contents/ui/NotificationItem.qml
|
|
Item {
|
|
id: notificationItem
|
|
property var notification
|
|
|
|
anchors.left: parent.left
|
|
anchors.right: parent.right
|
|
height: notifLayout.height + units.gridUnit
|
|
|
|
opacity: 1 - Math.min(1, 1.5 * Math.abs(rect.x) / width) // opacity during dismiss swipe
|
|
|
|
// notification
|
|
Rectangle {
|
|
id: rect
|
|
|
|
radius: 5
|
|
color: "white"
|
|
|
|
height: parent.height
|
|
width: parent.width
|
|
|
|
border.color: "#bdbdbd"
|
|
border.width: 1
|
|
ColumnLayout {
|
|
id: notifLayout
|
|
anchors {
|
|
left: parent.left
|
|
leftMargin: units.gridUnit * 0.5
|
|
right: parent.right
|
|
rightMargin: units.gridUnit * 0.5
|
|
verticalCenter: parent.verticalCenter
|
|
}
|
|
RowLayout {
|
|
Layout.fillWidth: true
|
|
spacing: units.smallSpacing / 2
|
|
// notif body
|
|
ColumnLayout {
|
|
id: textLayout
|
|
Layout.fillWidth: true
|
|
Layout.alignment: Qt.AlignVCenter | Qt.AlignLeft
|
|
spacing: units.gridUnit / 2
|
|
|
|
Label {
|
|
text: notification.summary
|
|
color: "#212121"
|
|
Layout.fillWidth: true
|
|
Layout.preferredHeight: implicitHeight
|
|
maximumLineCount: 3
|
|
wrapMode: Text.WordWrap
|
|
elide: Text.ElideRight
|
|
font.pointSize: 11
|
|
}
|
|
Label {
|
|
text: notification.body
|
|
color: "#616161"
|
|
Layout.fillWidth: true
|
|
wrapMode: Text.WordWrap
|
|
elide: Text.ElideRight
|
|
font.pointSize: 10
|
|
}
|
|
}
|
|
|
|
// notification icon
|
|
Item {
|
|
id: iconContainer
|
|
|
|
Layout.alignment: Qt.AlignVCenter | Qt.AlignRight
|
|
Layout.preferredWidth: units.iconSizes.large
|
|
Layout.preferredHeight: units.iconSizes.large
|
|
Layout.topMargin: units.smallSpacing
|
|
Layout.bottomMargin: units.smallSpacing
|
|
|
|
visible: iconItem.active || imageItem.active
|
|
|
|
PlasmaCore.IconItem {
|
|
id: iconItem
|
|
// don't show two identical icons
|
|
readonly property bool active: valid && source != notification.applicationIconSource
|
|
anchors.fill: parent
|
|
usesPlasmaTheme: false
|
|
smooth: true
|
|
source: {
|
|
let icon = notification.icon;
|
|
if (typeof icon !== "string") return "";
|
|
if (icon === "dialog-information") return "";
|
|
return icon;
|
|
}
|
|
visible: active
|
|
}
|
|
|
|
KQCAddons.QImageItem {
|
|
id: imageItem
|
|
readonly property bool active: !null && nativeWidth > 0
|
|
anchors.fill: parent
|
|
smooth: true
|
|
fillMode: KQCAddons.QImageItem.PreserveAspectFit
|
|
visible: active
|
|
image: typeof notification.icon === "object" ? notification.icon : undefined
|
|
}
|
|
}
|
|
}
|
|
|
|
Flow {
|
|
id: actionsflow
|
|
Layout.fillWidth: true
|
|
spacing: units.smallSpacing
|
|
layoutDirection: Qt.RightToLeft
|
|
Repeater {
|
|
id: actionRepeater
|
|
|
|
model: {
|
|
var buttons = [];
|
|
var actionNames = (notificationItem.notification.actionNames || []);
|
|
var actionLabels = (notificationItem.notification.actionLabels || []);
|
|
// HACK We want the actions to be right-aligned but Flow also reverses
|
|
for (var i = actionNames.length - 1; i >= 0; --i) {
|
|
buttons.push({
|
|
actionName: actionNames[i],
|
|
label: actionLabels[i]
|
|
});
|
|
}
|
|
|
|
return buttons;
|
|
}
|
|
|
|
PlasmaComponents3.ToolButton {
|
|
flat: false
|
|
// why does it spit "cannot assign undefined to string" when a notification becomes expired?
|
|
text: modelData.label || ""
|
|
|
|
onClicked: {
|
|
if (notificationItem.notification.category === "x-kde.incoming-call") {
|
|
notifModel.invokeAction(notificationItem.notification.notificationId, modelData.actionName);
|
|
} else {
|
|
notificationsRoot.pendingAction = {"notificationId": notificationItem.notification.notificationId,
|
|
"actionName":modelData.actionName};
|
|
root.askPassword();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// swipe gesture for dismissing notification (left/right)
|
|
MouseArea {
|
|
id: dismissSwipe
|
|
anchors.fill: parent
|
|
drag.axis: Drag.XAxis
|
|
drag.target: rect
|
|
onPressed: {
|
|
let pos = mapToItem(actionsflow, mouse.x, mouse.y);
|
|
if (actionsflow.childAt(pos.x, pos.y)) {
|
|
mouse.accepted = false;
|
|
}
|
|
}
|
|
onReleased: {
|
|
if (Math.abs(rect.x) > width / 2) { // dismiss notification when finished swipe
|
|
notifModel.close(notificationItem.notification.notificationId);
|
|
} else {
|
|
slideAnim.restart();
|
|
}
|
|
if (notificationItem.notification.hasDefaultAction && Math.abs(rect.x) < units.gridUnit) {
|
|
if (notificationItem.notification.category === "x-kde.incoming-call") {
|
|
notifModel.invokeDefaultAction(notificationItem.notification.notificationId);
|
|
} else {
|
|
notificationsRoot.pendingAction = {"notificationId": notificationItem.notification.notificationId,
|
|
"actionName": ""};
|
|
root.askPassword();
|
|
}
|
|
}
|
|
}
|
|
|
|
NumberAnimation {
|
|
id: slideAnim
|
|
target: rect
|
|
property: "x"
|
|
to: 0
|
|
duration: units.longDuration
|
|
}
|
|
}
|
|
}
|
|
|
|
DropShadow {
|
|
anchors.fill: rect
|
|
source: rect
|
|
horizontalOffset: 1
|
|
verticalOffset: 1
|
|
radius: 4
|
|
samples: 6
|
|
color: "#616161"
|
|
}
|
|
}
|