2022-05-27 17:31:54 +03:00
// SPDX-FileCopyrightText: 2022 Nheko Contributors
2023-01-02 06:25:33 +03:00
// SPDX-FileCopyrightText: 2023 Nheko Contributors
2022-05-27 17:31:54 +03:00
//
// SPDX-License-Identifier: GPL-3.0-or-later
import ".."
import "../components"
import QtQuick 2.12
import QtQuick . Controls 2.5
import QtQuick . Layouts 1.3
import im . nheko 1.0
ApplicationWindow {
id: plEditorW
property var roomSettings
property var editingModel: Nheko . editPowerlevels ( roomSettings . roomId )
modality: Qt . NonModal
flags: Qt . Dialog | Qt . WindowCloseButtonHint | Qt . WindowTitleHint
minimumWidth: 300
minimumHeight: 400
2022-06-26 01:09:00 +03:00
height: 600
2022-05-27 17:31:54 +03:00
title: qsTr ( "Permissions in %1" ) . arg ( roomSettings . roomName ) ;
2022-06-26 01:09:00 +03:00
// Shortcut {
// sequence: StandardKey.Cancel
// onActivated: dbb.rejected()
// }
2022-05-27 17:31:54 +03:00
ColumnLayout {
anchors.margins: Nheko . paddingMedium
anchors.fill: parent
spacing: 0
MatrixText {
text: qsTr ( "Be careful when editing permissions. You can't lower the permissions of people with a same or higher level than you. Be careful when promoting others." )
font.pixelSize: Math . floor ( fontMetrics . font . pixelSize * 1.1 )
Layout.fillWidth: true
Layout.fillHeight: false
color: Nheko . colors . text
Layout.bottomMargin: Nheko . paddingMedium
}
TabBar {
id: bar
width: parent . width
palette: Nheko . colors
component TabB : TabButton {
id: control
contentItem: Text {
text: control . text
font: control . font
opacity: enabled ? 1.0 : 0.3
color: control . down ? Nheko.colors.highlightedText : Nheko . colors . text
horizontalAlignment: Text . AlignHCenter
verticalAlignment: Text . AlignVCenter
elide: Text . ElideRight
}
background: Rectangle {
border.color: control . down ? Nheko.colors.highlight : Nheko . theme . separator
color: control . checked ? Nheko.colors.highlight : Nheko . colors . base
border.width: 1
radius: 2
}
}
TabB {
text: qsTr ( "Roles" )
}
TabB {
text: qsTr ( "Users" )
}
}
Rectangle {
Layout.fillWidth: true
Layout.fillHeight: true
color: Nheko . colors . alternateBase
border.width: 1
border.color: Nheko . theme . separator
StackLayout {
anchors.fill: parent
anchors.margins: Nheko . paddingMedium
currentIndex: bar . currentIndex
ColumnLayout {
spacing: Nheko . paddingMedium
MatrixText {
text: qsTr ( "Move permissions between roles to change them" )
font.pixelSize: Math . floor ( fontMetrics . font . pixelSize * 1.1 )
Layout.fillWidth: true
Layout.fillHeight: false
color: Nheko . colors . text
}
ReorderableListview {
Layout.fillWidth: true
Layout.fillHeight: true
model: editingModel . types
delegate: RowLayout {
Column {
Layout.fillWidth: true
Text { visible: model . isType ; text: model . displayName ; color: Nheko . colors . text }
Text {
visible: ! model . isType ;
text: {
if ( editingModel . adminLevel == model . powerlevel )
return qsTr ( "Administrator (%1)" ) . arg ( model . powerlevel )
else if ( editingModel . moderatorLevel == model . powerlevel )
return qsTr ( "Moderator (%1)" ) . arg ( model . powerlevel )
2022-06-26 01:09:00 +03:00
else if ( editingModel . defaultUserLevel == model . powerlevel )
return qsTr ( "User (%1)" ) . arg ( model . powerlevel )
2022-05-27 17:31:54 +03:00
else
return qsTr ( "Custom (%1)" ) . arg ( model . powerlevel )
}
color: Nheko . colors . text
}
}
ImageButton {
Layout.alignment: Qt . AlignRight
Layout.rightMargin: 2
image: model . isType ? ":/icons/icons/ui/dismiss.svg" : ":/icons/icons/ui/add-square-button.svg"
visible: ! model . isType || model . removeable
hoverEnabled: true
ToolTip.visible: hovered
ToolTip.text: model . isType ? qsTr ( "Remove event type" ) : qsTr ( "Add event type" )
onClicked: {
if ( model . isType ) {
editingModel . types . remove ( index ) ;
} else {
typeEntry . y = offset
typeEntry . visible = true
typeEntry . index = index ;
typeEntry . forceActiveFocus ( )
}
}
}
}
2022-06-26 01:09:00 +03:00
MatrixTextField {
id: typeEntry
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
property int index
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
width: parent . width
z: 5
visible: false
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
color: Nheko . colors . text
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
Keys.onPressed: {
if ( typeEntry . text . includes ( '.' ) && event . matches ( StandardKey . InsertParagraphSeparator ) ) {
editingModel . types . add ( typeEntry . index , typeEntry . text )
typeEntry . visible = false ;
typeEntry . clear ( ) ;
event . accepted = true ;
}
else if ( event . matches ( StandardKey . Cancel ) ) {
typeEntry . visible = false ;
typeEntry . clear ( ) ;
event . accepted = true ;
}
}
}
}
Button {
Layout.fillWidth: true
text: qsTr ( "Add new role" )
onClicked: newPLLay . visible = true
Rectangle {
id: newPLLay
anchors.fill: parent
visible: false
color: Nheko . colors . alternateBase
RowLayout {
spacing: Nheko . paddingMedium
anchors.fill: parent
SpinBox {
id: newPLVal
Layout.fillWidth: true
Layout.fillHeight: true
editable: true
//from: -9007199254740991
//to: 9007199254740991
// max qml values
from: - 2000000000
to: 2000000000
Keys.onPressed: {
if ( event . matches ( StandardKey . InsertParagraphSeparator ) ) {
editingModel . addRole ( newPLVal . value ) ;
newPLLay . visible = false ;
}
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
}
Button {
text: qsTr ( "Add" )
Layout.preferredWidth: 100
onClicked: {
editingModel . addRole ( newPLVal . value ) ;
newPLLay . visible = false ;
2022-05-27 17:31:54 +03:00
}
}
}
2022-06-26 01:09:00 +03:00
}
2022-05-27 17:31:54 +03:00
}
}
2022-06-26 01:09:00 +03:00
2022-05-27 17:31:54 +03:00
ColumnLayout {
spacing: Nheko . paddingMedium
MatrixText {
text: qsTr ( "Move users up or down to change their permissions" )
font.pixelSize: Math . floor ( fontMetrics . font . pixelSize * 1.1 )
Layout.fillWidth: true
Layout.fillHeight: false
}
ReorderableListview {
Layout.fillWidth: true
Layout.fillHeight: true
model: editingModel . users
2022-06-26 01:09:00 +03:00
Column {
id: userEntryCompleter
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
property int index: 0
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
visible: false
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
width: parent . width
spacing: 1
z: 5
2022-05-27 17:31:54 +03:00
MatrixTextField {
id: userEntry
width: parent . width
//font.pixelSize: Math.ceil(quickSwitcher.textHeight * 0.6)
color: Nheko . colors . text
onTextEdited: {
userCompleter . completer . searchString = text ;
}
Keys.onPressed: {
if ( event . key == Qt . Key_Up || event . key == Qt . Key_Backtab ) {
event . accepted = true ;
userCompleter . up ( ) ;
} else if ( event . key == Qt . Key_Down || event . key == Qt . Key_Tab ) {
event . accepted = true ;
if ( event . key == Qt . Key_Tab && ( event . modifiers & Qt . ShiftModifier ) )
userCompleter . up ( ) ;
else
userCompleter . down ( ) ;
} else if ( event . matches ( StandardKey . InsertParagraphSeparator ) ) {
2022-09-22 22:24:41 +03:00
if ( userCompleter . currentCompletion ( ) ) {
userCompleter . finishCompletion ( ) ;
} else if ( userEntry . text . startsWith ( "@" ) && userEntry . text . includes ( ":" ) ) {
userCompletionConnections . onCompletionSelected ( userEntry . text ) ;
}
2022-05-27 17:31:54 +03:00
event . accepted = true ;
} else if ( event . matches ( StandardKey . Cancel ) ) {
typeEntry . visible = false ;
typeEntry . clear ( ) ;
event . accepted = true ;
}
}
}
2022-06-26 01:09:00 +03:00
Completer {
id: userCompleter
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
visible: userEntry . text . length > 0
width: parent . width
roomId: plEditorW . roomSettings . roomId
completerName: "user"
bottomToTop: false
fullWidth: true
avatarHeight: Nheko . avatarSize / 2
avatarWidth: Nheko . avatarSize / 2
centerRowContent: false
rowMargin: 2
rowSpacing: 2
}
}
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
Connections {
function onCompletionSelected ( id ) {
console . log ( "selected: " + id ) ;
editingModel . users . add ( userEntryCompleter . index , id ) ;
userEntry . clear ( ) ;
userEntryCompleter . visible = false ;
}
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
function onCountChanged ( ) {
if ( userCompleter . count > 0 && ( userCompleter . currentIndex < 0 || userCompleter . currentIndex >= userCompleter . count ) )
userCompleter . currentIndex = 0 ;
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
target: userCompleter
2022-09-22 22:24:41 +03:00
id: userCompletionConnections
2022-06-26 01:09:00 +03:00
}
delegate: RowLayout {
//anchors { fill: parent; margins: 2 }
id: row
Avatar {
id: avatar
Layout.preferredHeight: Nheko . avatarSize / 2
Layout.preferredWidth: Nheko . avatarSize / 2
Layout.leftMargin: 2
userid: model . mxid
url: {
if ( model . isUser )
return model . avatarUrl . replace ( "mxc://" , "image://MxcImage/" )
else if ( editingModel . adminLevel >= model . powerlevel )
return "image://colorimage/:/icons/icons/ui/ribbon_star.svg?" + Nheko . colors . buttonText ;
else if ( editingModel . moderatorLevel >= model . powerlevel )
return "image://colorimage/:/icons/icons/ui/ribbon.svg?" + Nheko . colors . buttonText ;
else
return "image://colorimage/:/icons/icons/ui/person.svg?" + Nheko . colors . buttonText ;
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
displayName: model . displayName
enabled: false
}
Column {
Layout.fillWidth: true
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
Text { visible: model . isUser ; text: model . displayName ; color: Nheko . colors . text }
Text { visible: model . isUser ; text: model . mxid ; color: Nheko . colors . text }
Text {
visible: ! model . isUser ;
text: {
if ( editingModel . adminLevel == model . powerlevel )
return qsTr ( "Administrator (%1)" ) . arg ( model . powerlevel )
else if ( editingModel . moderatorLevel == model . powerlevel )
return qsTr ( "Moderator (%1)" ) . arg ( model . powerlevel )
else
return qsTr ( "Custom (%1)" ) . arg ( model . powerlevel )
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
color: Nheko . colors . text
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
}
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
ImageButton {
Layout.alignment: Qt . AlignRight
Layout.rightMargin: 2
image: model . isUser ? ":/icons/icons/ui/dismiss.svg" : ":/icons/icons/ui/add-square-button.svg"
visible: ! model . isUser || model . removeable
hoverEnabled: true
ToolTip.visible: hovered
ToolTip.text: model . isUser ? qsTr ( "Remove user" ) : qsTr ( "Add user" )
onClicked: {
if ( model . isUser ) {
editingModel . users . remove ( index ) ;
} else {
userEntryCompleter . y = offset
userEntryCompleter . visible = true
userEntryCompleter . index = index ;
userEntry . forceActiveFocus ( )
2022-05-27 17:31:54 +03:00
}
}
}
}
}
2022-06-26 01:09:00 +03:00
2022-05-27 17:31:54 +03:00
}
}
}
2022-06-26 01:09:00 +03:00
}
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
footer: DialogButtonBox {
id: dbb
2022-05-27 17:31:54 +03:00
2022-06-26 01:09:00 +03:00
standardButtons: DialogButtonBox . Ok | DialogButtonBox . Cancel
onAccepted: {
2022-09-28 03:09:04 +03:00
if ( editingModel . isSpace ) {
// TODO(Nico): Replace with showing a list of spaces to apply to
editingModel . updateSpacesModel ( ) ;
plEditorW . close ( ) ;
timelineRoot . showSpacePLApplyPrompt ( roomSettings , editingModel )
} else {
editingModel . commit ( ) ;
plEditorW . close ( ) ;
}
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
onRejected: plEditorW . close ( ) ;
2022-05-27 17:31:54 +03:00
}
2022-06-26 01:09:00 +03:00
}