status-desktop/ui/app/AppLayouts/Profile/views/NotificationsView.qml
Sale Djenic 03d7632f32 fix(@desktop/notifications): revamp notifications settings & behaviour
- signal for adding notification to the Activity Center introduced
and emitted, but that part is not handled within this commit since
there are some details needed to be discussed in the PR#5334
- signal for displaying ephemeral app notification is introduced
and emitted, but that part is not handled within this commit since
we don't have yet ephemeral notificaiton in place.

Fixes #4902
2022-05-13 14:58:39 -04:00

552 lines
28 KiB
QML

import QtQuick 2.13
import QtQuick.Controls 2.13
import QtQuick.Layouts 1.13
import StatusQ.Core 0.1
import StatusQ.Core.Theme 0.1
import StatusQ.Controls 0.1
import StatusQ.Components 0.1
import StatusQ.Popups 0.1
import utils 1.0
import shared.panels 1.0
import shared.controls 1.0
import "../stores"
import "../controls"
import "../panels"
import "../popups"
SettingsContentBase {
id: root
property NotificationsStore notificationsStore
property DevicesStore devicesStore
ColumnLayout {
id: contentColumn
spacing: Constants.settingsSection.itemSpacing
ButtonGroup {
id: messageSetting
}
Loader {
id: exemptionNotificationsModal
active: false
function open(item) {
active = true
exemptionNotificationsModal.item.item = item
exemptionNotificationsModal.item.open()
}
function close() {
active = false
}
sourceComponent: ExemptionNotificationsModal {
anchors.centerIn: parent
notificationsStore: root.notificationsStore
onClosed: {
exemptionNotificationsModal.close();
}
}
}
Component {
id: exemptionDelegateComponent
StatusListItem {
property string lowerCaseSearchString: searchBox.text.toLowerCase()
width: parent.width
height: visible ? implicitHeight : 0
visible: lowerCaseSearchString === "" ||
model.itemId.toLowerCase().includes(lowerCaseSearchString) ||
model.name.toLowerCase().includes(lowerCaseSearchString)
title: model.name
subTitle: {
if(model.type === Constants.settingsSection.exemptions.community)
return qsTr("Community")
else if(model.type === Constants.settingsSection.exemptions.oneToOneChat)
return qsTr("1:1 Chat")
else if(model.type === Constants.settingsSection.exemptions.groupChat)
return qsTr("Group Chat")
else
return ""
}
label: {
if(!model.customized)
return ""
let l = ""
if(model.muteAllMessages)
l += qsTr("Muted")
else {
let nbOfChanges = 0
if(model.personalMentions !== Constants.settingsSection.notifications.sendAlertsValue)
{
nbOfChanges++
let valueText = model.personalMentions === Constants.settingsSection.notifications.turnOffValue?
qsTr("Off") :
qsTr("Quiet")
l = qsTr("Personal @ Mentions %1").arg(valueText)
}
if(model.globalMentions !== Constants.settingsSection.notifications.sendAlertsValue)
{
nbOfChanges++
let valueText = model.globalMentions === Constants.settingsSection.notifications.turnOffValue?
qsTr("Off") :
qsTr("Quiet")
l = qsTr("Global @ Mentions %1").arg(valueText)
}
if(model.otherMessages !== Constants.settingsSection.notifications.turnOffValue)
{
nbOfChanges++
let valueText = model.otherMessages === Constants.settingsSection.notifications.sendAlertsValue?
qsTr("Alerts") :
qsTr("Quiet")
l = qsTr("Other Messages %1").arg(valueText)
}
if(nbOfChanges > 1)
l = qsTr("Multiple Exemptions")
}
return l
}
// Maybe we need to redo `StatusListItem` to display identicon ring, but that's not in Figma design for now.
image.source: model.image
ringSettings.ringSpecModel: Utils.getColorHashAsJson(model.itemId)
icon: StatusIconSettings {
color: model.type === Constants.settingsSection.exemptions.oneToOneChat?
Theme.palette.userCustomizationColors[Utils.colorIdForPubkey(model.itemId)] :
model.color
charactersLen: model.type === Constants.settingsSection.exemptions.oneToOneChat? 2 : 1
isLetterIdenticon: model.image === ""
height: isLetterIdenticon ? 40 : 20
width: isLetterIdenticon ? 40 : 20
}
components: [
StatusIcon {
visible: model.customized
icon: "chevron-down"
rotation: 270
color: Theme.palette.baseColor1
MouseArea {
anchors.fill: parent
cursorShape: Qt.PointingHandCursor
onClicked: {
exemptionNotificationsModal.open(model)
}
}
},
StatusIcon {
visible: !model.customized
icon: "add"
rotation: 270
color: Theme.palette.primaryColor1
MouseArea {
anchors.fill: parent
cursorShape: Qt.PointingHandCursor
onClicked: {
exemptionNotificationsModal.open(model)
}
}
}]
}
}
Rectangle {
Layout.preferredWidth: root.contentWidth
implicitHeight: col1.height + 2 * Style.current.padding
visible: Qt.platform.os == "osx"
radius: Constants.settingsSection.radius
color: Theme.palette.primaryColor3
ColumnLayout {
id: col1
anchors.margins: Style.current.padding
anchors.left: parent.left
anchors.right: parent.right
anchors.verticalCenter: parent.verticalCenter
spacing: Constants.settingsSection.infoSpacing
StatusBaseText {
Layout.preferredWidth: parent.width
text: qsTr("Enable Notifications in macOS Settings")
font.pixelSize: Constants.settingsSection.infoFontSize
lineHeight: Constants.settingsSection.infoLineHeight
lineHeightMode: Text.FixedHeight
color: Theme.palette.primaryColor1
}
StatusBaseText {
Layout.preferredWidth: parent.width
text: qsTr("To receive Status notifications, make sure you've enabled them in" +
" your computer's settings under <b>System Preferences > Notifications</b>")
font.pixelSize: Constants.settingsSection.infoFontSize
lineHeight: Constants.settingsSection.infoLineHeight
lineHeightMode: Text.FixedHeight
color: Theme.palette.baseColor1
wrapMode: Text.WordWrap
}
}
}
Rectangle {
Layout.preferredWidth: root.contentWidth
implicitHeight: row1.height + 2 * Style.current.padding
radius: Constants.settingsSection.radius
color: Theme.palette.pinColor2
RowLayout {
id: row1
anchors.margins: Style.current.padding
anchors.left: parent.left
anchors.right: parent.right
anchors.verticalCenter: parent.verticalCenter
visible: root.devicesStore.devicesModel.count > 0
StatusBaseText {
Layout.fillWidth: true
text: qsTr("Sync your devices to share notifications preferences")
font.pixelSize: Constants.settingsSection.infoFontSize
lineHeight: Constants.settingsSection.infoLineHeight
lineHeightMode: Text.FixedHeight
color: Theme.palette.pinColor1
}
StatusBaseText {
text: qsTr("Syncing >")
font.pixelSize: Constants.settingsSection.infoFontSize
lineHeight: Constants.settingsSection.infoLineHeight
lineHeightMode: Text.FixedHeight
color: Theme.palette.pinColor1
MouseArea {
anchors.fill: parent
cursorShape: Qt.PointingHandCursor
onClicked: {
root.devicesStore.syncAll()
}
}
}
}
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Allow Notifications")
components: [
StatusSwitch {
id: allowNotifSwitch
checked: localAccountSensitiveSettings.notifSettingAllowNotifications
onClicked: {
localAccountSensitiveSettings.notifSettingAllowNotifications = !localAccountSensitiveSettings.notifSettingAllowNotifications
}
}
]
sensor.onClicked: {
allowNotifSwitch.clicked()
}
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Messages")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.baseColor1
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("1:1 Chats")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingOneToOneChats
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingOneToOneChats = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingOneToOneChats = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingOneToOneChats = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Group Chats")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingGroupChats
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingGroupChats = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingGroupChats = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingGroupChats = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Personal @ Mentions")
tertiaryTitle: qsTr("Messages containing @%1").arg(userProfile.name)
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingPersonalMentions
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingPersonalMentions = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingPersonalMentions = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingPersonalMentions = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Global @ Mentions")
tertiaryTitle: qsTr("Messages containing @here and @channel")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingGlobalMentions
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingGlobalMentions = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingGlobalMentions = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingGlobalMentions = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("All Messages")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingAllMessages
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingAllMessages = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingAllMessages = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingAllMessages = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Others")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.baseColor1
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Contact Requests")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingContactRequests
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingContactRequests = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingContactRequests = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingContactRequests = Constants.settingsSection.notifications.turnOffValue
}
]
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Identity Verification Requests")
components: [
NotificationSelect {
selected: localAccountSensitiveSettings.notifSettingIdentityVerificationRequests
onSendAlertsClicked: localAccountSensitiveSettings.notifSettingIdentityVerificationRequests = Constants.settingsSection.notifications.sendAlertsValue
onDeliverQuietlyClicked: localAccountSensitiveSettings.notifSettingIdentityVerificationRequests = Constants.settingsSection.notifications.deliverQuietlyValue
onTurnOffClicked: localAccountSensitiveSettings.notifSettingIdentityVerificationRequests = Constants.settingsSection.notifications.turnOffValue
}
]
}
Separator {
Layout.preferredWidth: root.contentWidth
Layout.preferredHeight: Style.current.bigPadding
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Notification Content")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.directColor1
}
NotificationAppearancePreviewPanel {
id: notifNameAndMsg
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
name: qsTr("Show Name and Message")
notificationTitle: "Vitalik Buterin"
notificationMessage: qsTr("Hi there! So EIP-1559 will defini...")
buttonGroup: messageSetting
checked: localAccountSensitiveSettings.notificationMessagePreviewSetting === Constants.settingsSection.notificationsBubble.previewNameAndMessage
onRadioCheckedChanged: {
if (checked) {
localAccountSensitiveSettings.notificationMessagePreviewSetting = Constants.settingsSection.notificationsBubble.previewNameAndMessage
}
}
}
NotificationAppearancePreviewPanel {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
name: qsTr("Name Only")
notificationTitle: "Vitalik Buterin"
notificationMessage: qsTr("You have a new message")
buttonGroup: messageSetting
checked: localAccountSensitiveSettings.notificationMessagePreviewSetting === Constants.settingsSection.notificationsBubble.previewNameOnly
onRadioCheckedChanged: {
if (checked) {
localAccountSensitiveSettings.notificationMessagePreviewSetting = Constants.settingsSection.notificationsBubble.previewNameOnly
}
}
}
NotificationAppearancePreviewPanel {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
name: qsTr("Anonymous")
notificationTitle: "Status"
notificationMessage: qsTr("You have a new message")
buttonGroup: messageSetting
checked: localAccountSensitiveSettings.notificationMessagePreviewSetting === Constants.settingsSection.notificationsBubble.previewAnonymous
onRadioCheckedChanged: {
if (checked) {
localAccountSensitiveSettings.notificationMessagePreviewSetting = Constants.settingsSection.notificationsBubble.previewAnonymous
}
}
}
StatusListItem {
Layout.preferredWidth: root.contentWidth
title: qsTr("Play a Sound When Receiving a Notification")
components: [
StatusSwitch {
id: soundSwitch
checked: localAccountSensitiveSettings.notificationSoundsEnabled
onClicked: {
localAccountSensitiveSettings.notificationSoundsEnabled = !localAccountSensitiveSettings.notificationSoundsEnabled
}
}
]
sensor.onClicked: {
soundSwitch.clicked()
}
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Volume")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.directColor1
}
Item {
Layout.preferredWidth: root.contentWidth
Layout.preferredHeight: Constants.settingsSection.itemHeight + Style.current.padding
StatusSlider {
id: volumeSlider
anchors.top: parent.top
anchors.left: parent.left
anchors.right: parent.right
anchors.topMargin: Style.current.bigPadding
anchors.leftMargin: Style.current.padding
anchors.rightMargin: Style.current.padding
from: 0
to: 100
stepSize: 1
onValueChanged: {
localAccountSensitiveSettings.volume = value
}
Component.onCompleted: {
value = localAccountSensitiveSettings.volume
}
}
RowLayout {
anchors.top: volumeSlider.bottom
anchors.left: volumeSlider.left
anchors.topMargin: Style.current.halfPadding
width: volumeSlider.width
StatusBaseText {
font.pixelSize: 15
text: volumeSlider.from
Layout.preferredWidth: volumeSlider.width/2
color: Theme.palette.baseColor1
}
StatusBaseText {
font.pixelSize: 15
text: volumeSlider.to
Layout.alignment: Qt.AlignRight
color: Theme.palette.baseColor1
}
}
}
StatusButton {
Layout.leftMargin: Style.current.padding
text: qsTr("Send a Test Notification")
onClicked: {
root.notificationsStore.sendTestNotification(notifNameAndMsg.notificationTitle,
notifNameAndMsg.notificationMessage)
}
}
Separator {
Layout.preferredWidth: root.contentWidth
Layout.preferredHeight: Style.current.bigPadding
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Exemptions")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.directColor1
}
SearchBox {
id: searchBox
Layout.preferredWidth: root.contentWidth - 2 * Style.current.padding
Layout.leftMargin: Style.current.padding
Layout.rightMargin: Style.current.padding
input.implicitHeight: 44
input.placeholderText: qsTr("Search Communities, Group Chats and 1:1 Chats")
}
StatusBaseText {
Layout.preferredWidth: root.contentWidth
Layout.leftMargin: Style.current.padding
text: qsTr("Most recent")
font.pixelSize: Constants.settingsSection.subHeaderFontSize
color: Theme.palette.baseColor1
}
ListView {
Layout.preferredWidth: root.contentWidth
Layout.preferredHeight: 400
visible: root.notificationsStore.exemptionsModel.count > 0
clip: true
model: root.notificationsStore.exemptionsModel
delegate: exemptionDelegateComponent
}
}
//}
// }
//}
}