kdeconnect-kde/plasmoid/package/contents/ui/DeviceDelegate.qml

495 lines
16 KiB
QML
Raw Permalink Normal View History

/**
* SPDX-FileCopyrightText: 2013 Albert Vaca <albertvaka@gmail.com>
2024-06-30 23:03:57 +01:00
* SPDX-FileCopyrightText: 2024 ivan tkachenko <me@ratijas.tk>
*
* SPDX-License-Identifier: GPL-2.0-only OR GPL-3.0-only OR LicenseRef-KDE-Accepted-GPL
*/
pragma ComponentBehavior: Bound
2024-06-30 23:03:57 +01:00
import QtCore
import QtQuick
2024-06-30 23:03:57 +01:00
import QtQuick.Dialogs as QtDialogs
import QtQuick.Layouts
2024-06-30 23:03:57 +01:00
import org.kde.kdeconnect as KDEConnect
import org.kde.kirigami as Kirigami
2024-06-30 23:03:57 +01:00
import org.kde.plasma.components as PlasmaComponents
import org.kde.plasma.core as PlasmaCore
import org.kde.plasma.extras as PlasmaExtras
2024-06-30 23:03:57 +01:00
PlasmaComponents.ItemDelegate {
id: root
2024-06-30 23:03:57 +01:00
required property int index
required property var model
2024-06-30 23:03:57 +01:00
readonly property KDEConnect.DeviceDbusInterface device: KDEConnect.DeviceDbusInterfaceFactory.create(model.deviceId)
hoverEnabled: false
down: false
Battery {
id: battery
device: root.device
}
Clipboard {
id: clipboard
device: root.device
}
Connectivity {
id: connectivity
device: root.device
}
FindMyPhone {
id: findmyphone
device: root.device
}
RemoteCommands {
id: remoteCommands
device: root.device
}
Sftp {
id: sftp
device: root.device
}
Share {
id: share
device: root.device
}
SMS {
id: sms
device: root.device
}
VirtualMonitor {
id: virtualmonitor
device: root.device
}
Kirigami.PromptDialog {
id: prompt
visible: false
showCloseButton: true
standardButtons: Kirigami.Dialog.NoButton
title: i18n("Virtual Monitor is not available")
}
QtDialogs.FileDialog {
id: fileDialog
title: i18n("Please choose a file")
currentFolder: StandardPaths.writableLocation(StandardPaths.HomeLocation)
fileMode: QtDialogs.FileDialog.OpenFiles
onAccepted: {
selectedFiles.forEach(url => share.plugin.shareUrl(url));
}
}
PlasmaExtras.Menu {
id: menu
visualParent: overflowMenu
placement: PlasmaExtras.Menu.BottomPosedLeftAlignedPopup
// Share
PlasmaExtras.MenuItem {
icon: "document-share"
visible: share.available
text: i18n("Share file")
onClicked: fileDialog.open()
}
// Clipboard
PlasmaExtras.MenuItem {
icon: "klipper"
visible: clipboard.clipboard?.isAutoShareDisabled ?? false
text: i18n("Send Clipboard")
onClicked: {
clipboard.sendClipboard()
}
}
// Find my phone
PlasmaExtras.MenuItem {
icon: "irc-voice"
visible: findmyphone.available
text: i18n("Ring my phone")
onClicked: {
findmyphone.ring()
}
}
// SFTP
PlasmaExtras.MenuItem {
icon: "document-open-folder"
visible: sftp.available
text: i18n("Browse this device")
onClicked: {
sftp.browse()
}
}
// SMS
PlasmaExtras.MenuItem {
icon: "message-new"
visible: sms.available
text: i18n("SMS Messages")
onClicked: {
sms.plugin.launchApp()
}
}
}
DropArea {
id: fileDropArea
anchors.fill: parent
2024-06-30 23:03:57 +01:00
onDropped: drop => {
if (drop.hasUrls) {
2024-06-30 23:03:57 +01:00
const urls = new Set(drop.urls.map(url => url.toString()));
urls.forEach(url => share.plugin.shareUrl(url));
}
drop.accepted = true;
}
PlasmaCore.ToolTipArea {
id: dropAreaToolTip
anchors.fill: parent
active: true
mainText: i18n("File Transfer")
subText: i18n("Drop a file to transfer it onto your phone.")
}
}
contentItem: ColumnLayout {
spacing: Kirigami.Units.smallSpacing
2024-06-30 23:03:57 +01:00
RowLayout {
width: parent.width
spacing: Kirigami.Units.smallSpacing
2014-01-27 16:52:29 +00:00
PlasmaComponents.Label {
id: deviceName
elide: Text.ElideRight
text: root.model.name
Layout.fillWidth: true
textFormat: Text.PlainText
}
PlasmaComponents.ToolButton {
icon.name: "krdc"
visible: virtualmonitor.available
text: i18n("Virtual Display")
onClicked: {
let err = "";
if (virtualmonitor?.plugin?.hasRemoteVncClient === false) {
err = i18n("Remote device does not have a VNC client (eg. krdc) installed.");
}
if (virtualmonitor?.plugin?.isVirtualMonitorAvailable === false) {
err = (err ? err + "\n\n" : "")
+ i18n("The krfb package is required on the local device.");
}
if (err) {
prompt.subtitle = err;
prompt.visible = true;
} else if (!virtualmonitor.plugin.requestVirtualMonitor()) {
prompt.subtitle = i18n("Failed to create the virtual monitor.");
prompt.visible = true;
}
}
}
2024-06-30 23:03:57 +01:00
RowLayout {
id: connectionInformation
2024-06-30 23:03:57 +01:00
visible: connectivity.available
spacing: Kirigami.Units.smallSpacing
// TODO: In the future, when the Connectivity Report plugin supports more than one
// subscription, add more signal strength icons to represent all the available
// connections.
Kirigami.Icon {
id: celluarConnectionStrengthIcon
source: connectivity.iconName
Layout.preferredHeight: connectivityText.height
Layout.preferredWidth: Layout.preferredHeight
Layout.alignment: Qt.AlignCenter
visible: valid
}
PlasmaComponents.Label {
// Fallback plain-text label. Only show this if the icon doesn't work.
id: connectivityText
text: connectivity.displayString
textFormat: Text.PlainText
visible: !celluarConnectionStrengthIcon.visible
}
}
2024-06-30 23:03:57 +01:00
RowLayout {
id: batteryInformation
2024-06-30 23:03:57 +01:00
visible: battery.available && battery.charge > -1
spacing: Kirigami.Units.smallSpacing
Kirigami.Icon {
id: batteryIcon
source: battery.iconName
// Make the icon the same size as the text so that it doesn't dominate
Layout.preferredHeight: batteryPercent.height
Layout.preferredWidth: Layout.preferredHeight
Layout.alignment: Qt.AlignCenter
}
PlasmaComponents.Label {
id: batteryPercent
text: i18nc("Battery charge percentage", "%1%", battery.charge)
textFormat: Text.PlainText
}
}
PlasmaComponents.ToolButton {
id: overflowMenu
2024-06-30 23:03:57 +01:00
icon.name: "application-menu"
checked: menu.status === PlasmaExtras.Menu.Open
onPressed: menu.openRelative()
}
}
2014-02-14 16:11:41 +00:00
2024-06-30 23:03:57 +01:00
// RemoteKeyboard
PlasmaComponents.ItemDelegate {
visible: remoteKeyboard.remoteState
Layout.fillWidth: true
contentItem: RowLayout {
width: parent.width
spacing: 5
PlasmaComponents.Label {
id: remoteKeyboardLabel
text: i18n("Remote Keyboard")
}
2024-06-30 23:03:57 +01:00
KDEConnect.RemoteKeyboard {
id: remoteKeyboard
device: root.device
Layout.fillWidth: true
}
}
}
2024-06-30 23:03:57 +01:00
// Notifications
PlasmaComponents.ItemDelegate {
2024-06-30 23:03:57 +01:00
visible: notificationsModel.count > 0
enabled: true
Layout.fillWidth: true
contentItem: RowLayout {
spacing: Kirigami.Units.smallSpacing
PlasmaComponents.Label {
text: i18n("Notifications:")
}
PlasmaComponents.ToolButton {
enabled: true
visible: notificationsModel.isAnyDimissable;
Layout.alignment: Qt.AlignRight
icon.name: "edit-clear-history"
2024-06-30 23:03:57 +01:00
PlasmaComponents.ToolTip.text: i18n("Dismiss all notifications")
onClicked: notificationsModel.dismissAll();
}
}
}
2024-06-30 23:03:57 +01:00
Repeater {
id: notificationsView
2024-06-30 23:03:57 +01:00
model: KDEConnect.NotificationsModel {
id: notificationsModel
deviceId: root.model.deviceId
}
2024-06-30 23:03:57 +01:00
delegate: PlasmaComponents.ItemDelegate {
id: listitem
2024-06-30 23:03:57 +01:00
required property int index
required property var model
enabled: true
onClicked: checked = !checked
Layout.fillWidth: true
property bool replying: false
contentItem: ColumnLayout {
spacing: Kirigami.Units.smallSpacing
RowLayout {
spacing: Kirigami.Units.smallSpacing
Kirigami.Icon {
id: notificationIcon
source: listitem.model.appIcon
width: (valid && listitem.model.appIcon !== "") ? dismissButton.width : 0
height: width
Layout.alignment: Qt.AlignLeft
}
PlasmaComponents.Label {
id: notificationLabel
text: {
const { appName, notitext, title } = listitem.model;
const description = title !== "" ? (appName === title ? notitext : `${title}: ${notitext}`) : notitext;
return `${appName}: ${description}`;
}
elide: listitem.checked ? Text.ElideNone : Text.ElideRight
maximumLineCount: listitem.checked ? 0 : 1
wrapMode: Text.Wrap
Layout.fillWidth: true
}
PlasmaComponents.ToolButton {
id: replyButton
visible: listitem.model.repliable
enabled: listitem.model.repliable && !listitem.replying
icon.name: "mail-reply-sender"
2024-06-30 23:03:57 +01:00
PlasmaComponents.ToolTip.text: i18n("Reply")
onClicked: {
listitem.replying = true;
replyTextField.forceActiveFocus();
}
}
PlasmaComponents.ToolButton {
id: dismissButton
visible: notificationsModel.isAnyDimissable;
enabled: listitem.model.dismissable
Layout.alignment: Qt.AlignRight
icon.name: "window-close"
2024-06-30 23:03:57 +01:00
PlasmaComponents.ToolTip.text: i18n("Dismiss")
onClicked: listitem.model.dbusInterface.dismiss();
}
}
RowLayout {
visible: listitem.replying
width: notificationLabel.width + replyButton.width + dismissButton.width + Kirigami.Units.smallSpacing * 2
spacing: Kirigami.Units.smallSpacing
PlasmaComponents.Button {
id: replyCancelButton
Layout.alignment: Qt.AlignBottom
text: i18n("Cancel")
display: PlasmaComponents.AbstractButton.IconOnly
PlasmaComponents.ToolTip {
text: replyCancelButton.text
}
icon.name: "dialog-cancel"
onClicked: {
replyTextField.text = "";
listitem.replying = false;
}
}
PlasmaComponents.TextArea {
id: replyTextField
placeholderText: i18nc("@info:placeholder", "Reply to %1…", listitem.model.appName)
wrapMode: TextEdit.Wrap
Layout.fillWidth: true
2024-06-30 23:03:57 +01:00
Keys.onPressed: event => {
if ((event.key === Qt.Key_Return || event.key === Qt.Key_Enter) && !(event.modifiers & Qt.ShiftModifier)) {
replySendButton.clicked();
event.accepted = true;
}
2024-06-30 23:03:57 +01:00
if (event.key === Qt.Key_Escape) {
replyCancelButton.clicked();
event.accepted = true;
}
}
}
PlasmaComponents.Button {
Layout.alignment: Qt.AlignBottom
id: replySendButton
text: i18n("Send")
icon.name: "document-send"
enabled: replyTextField.text !== ""
onClicked: {
listitem.model.dbusInterface.sendReply(replyTextField.text);
replyTextField.text = "";
listitem.replying = false;
}
}
}
}
}
}
// Commands
2018-11-02 14:38:52 +00:00
RowLayout {
2024-06-30 23:03:57 +01:00
visible: remoteCommands.available
2018-11-02 14:38:52 +00:00
width: parent.width
spacing: Kirigami.Units.smallSpacing
2018-11-02 14:38:52 +00:00
PlasmaComponents.Label {
text: i18n("Run command")
2018-11-02 14:38:52 +00:00
Layout.fillWidth: true
}
2018-11-02 14:38:52 +00:00
2024-06-30 23:03:57 +01:00
PlasmaComponents.Button {
id: addCommandButton
icon.name: "list-add"
2024-06-30 23:03:57 +01:00
PlasmaComponents.ToolTip.text: i18n("Add command")
onClicked: remoteCommands.plugin.editCommands()
visible: remoteCommands.plugin?.canAddCommand ?? false
}
}
2024-06-30 23:03:57 +01:00
Repeater {
id: commandsView
2024-06-30 23:03:57 +01:00
visible: remoteCommands.available
model: KDEConnect.RemoteCommandsModel {
id: commandsModel
deviceId: root.model.deviceId
}
2024-06-30 23:03:57 +01:00
delegate: PlasmaComponents.ItemDelegate {
id: commandDelegate
required property int index
required property var model
enabled: true
onClicked: {
remoteCommands.plugin?.triggerCommand(commandDelegate.model.key);
}
Layout.fillWidth: true
contentItem: PlasmaComponents.Label {
text: `${commandDelegate.model.name}\n${commandDelegate.model.command}`
}
}
}
}
}