2022-06-09 17:27:14 +02:00
|
|
|
|
import QtQuick 2.14
|
|
|
|
|
import QtQuick.Layouts 1.14
|
|
|
|
|
|
|
|
|
|
import StatusQ.Core 0.1
|
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
|
import StatusQ.Components 0.1
|
|
|
|
|
import StatusQ.Controls 0.1
|
2022-09-07 11:40:25 +02:00
|
|
|
|
import StatusQ.Core.Utils 0.1 as SQ
|
2022-06-09 17:27:14 +02:00
|
|
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
|
import shared.panels 1.0
|
|
|
|
|
|
2022-09-07 11:40:25 +02:00
|
|
|
|
import SortFilterProxyModel 0.2
|
|
|
|
|
|
2022-06-09 17:27:14 +02:00
|
|
|
|
import "../../../Chat/controls/community"
|
2022-08-23 10:46:37 +02:00
|
|
|
|
import "../../stores"
|
2022-06-09 17:27:14 +02:00
|
|
|
|
|
|
|
|
|
Flickable {
|
|
|
|
|
id: root
|
|
|
|
|
|
2022-08-23 10:46:37 +02:00
|
|
|
|
property var store: CommunitiesStore {}
|
|
|
|
|
|
|
|
|
|
|
2022-06-09 17:27:14 +02:00
|
|
|
|
signal createPermission()
|
|
|
|
|
|
2022-08-23 10:46:37 +02:00
|
|
|
|
// TODO: Call this when permissions are stored in backend to start a new permissions flow
|
|
|
|
|
function clearPermissions() {
|
|
|
|
|
d.permissions.clear()
|
|
|
|
|
}
|
|
|
|
|
|
2022-06-09 17:27:14 +02:00
|
|
|
|
QtObject {
|
|
|
|
|
id: d
|
|
|
|
|
property bool isPrivate: false
|
2022-08-23 10:46:37 +02:00
|
|
|
|
property ListModel permissions: ListModel{}
|
2022-09-10 00:09:37 +02:00
|
|
|
|
property int permissionType: PermissionTypes.Type.None
|
2022-06-09 17:27:14 +02:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
contentWidth: mainLayout.width
|
|
|
|
|
contentHeight: mainLayout.height
|
|
|
|
|
clip: true
|
|
|
|
|
flickableDirection: Flickable.AutoFlickIfNeeded
|
|
|
|
|
|
|
|
|
|
ColumnLayout {
|
|
|
|
|
id: mainLayout
|
|
|
|
|
width: 560 // by design
|
|
|
|
|
spacing: 0
|
|
|
|
|
CurveSeparatorWithText {
|
|
|
|
|
Layout.alignment: Qt.AlignLeft
|
|
|
|
|
Layout.leftMargin: 14
|
|
|
|
|
text: qsTr("Anyone")
|
|
|
|
|
}
|
|
|
|
|
StatusItemSelector {
|
|
|
|
|
id: tokensSelector
|
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
icon: Style.svg("contact_verified")
|
|
|
|
|
title: qsTr("Who holds")
|
|
|
|
|
defaultItemText: qsTr("Example: 10 SNT")
|
|
|
|
|
andOperatorText: qsTr("and")
|
|
|
|
|
orOperatorText: qsTr("or")
|
2022-09-07 11:40:25 +02:00
|
|
|
|
|
|
|
|
|
// roles: type, key, name, amount, imageSource, operator
|
|
|
|
|
ListModel {
|
|
|
|
|
id: holdingsModel
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
property int editedIndex
|
|
|
|
|
|
|
|
|
|
function getText(type, name, amount) {
|
|
|
|
|
switch (type) {
|
|
|
|
|
case HoldingTypes.Type.Token:
|
|
|
|
|
case HoldingTypes.Type.Collectible:
|
2022-09-15 12:12:38 +02:00
|
|
|
|
return `${LocaleUtils.numberToLocaleString(amount)} ${name}`
|
2022-09-07 11:40:25 +02:00
|
|
|
|
case HoldingTypes.Type.Ens:
|
|
|
|
|
if (name)
|
|
|
|
|
return qsTr("ENS username on '%1' domain").arg(name)
|
|
|
|
|
else
|
|
|
|
|
return qsTr("Any ENS username")
|
|
|
|
|
default:
|
|
|
|
|
return ""
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
itemsModel: SortFilterProxyModel {
|
|
|
|
|
sourceModel: holdingsModel
|
|
|
|
|
|
|
|
|
|
proxyRoles: ExpressionRole {
|
|
|
|
|
name: "text"
|
|
|
|
|
expression: tokensSelector.getText(model.type, model.name, model.amount)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
HoldingsDropdown {
|
2022-06-09 17:27:14 +02:00
|
|
|
|
id: dropdown
|
2022-08-23 10:46:37 +02:00
|
|
|
|
store: root.store
|
2022-09-07 11:40:25 +02:00
|
|
|
|
|
|
|
|
|
function addItem(type, item, amount, operator) {
|
|
|
|
|
const key = item.key
|
|
|
|
|
const name = item.name
|
|
|
|
|
const imageSource = item.iconSource.toString()
|
|
|
|
|
|
|
|
|
|
holdingsModel.append({ type, key, name, amount, imageSource, operator })
|
|
|
|
|
d.permissions.append([{ key }, { operator }])
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onAddToken: {
|
|
|
|
|
const modelItem = store.getTokenByKey(key)
|
|
|
|
|
addItem(HoldingTypes.Type.Token, modelItem, amount, operator)
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onAddCollectible: {
|
|
|
|
|
const modelItem = store.getCollectibleByKey(key)
|
|
|
|
|
addItem(HoldingTypes.Type.Collectible, modelItem, amount, operator)
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onAddEns: {
|
|
|
|
|
const key = any ? "EnsAny" : "EnsCustom"
|
|
|
|
|
const name = any ? "" : customDomain
|
2022-09-10 00:09:37 +02:00
|
|
|
|
const icon = Style.svg("ensUsernames")
|
2022-09-07 11:40:25 +02:00
|
|
|
|
|
|
|
|
|
holdingsModel.append({type: HoldingTypes.Type.Ens, key, name, amount: 1, imageSource: icon, operator })
|
|
|
|
|
d.permissions.append([{ key }, { operator }])
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onUpdateToken: {
|
|
|
|
|
const modelItem = store.getTokenByKey(key)
|
|
|
|
|
const name = modelItem.name
|
|
|
|
|
const imageSource = modelItem.iconSource.toString()
|
|
|
|
|
|
|
|
|
|
holdingsModel.set(tokensSelector.editedIndex, { type: HoldingTypes.Type.Token, key, name, amount, imageSource })
|
2022-06-09 17:27:14 +02:00
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
2022-09-07 11:40:25 +02:00
|
|
|
|
|
|
|
|
|
onUpdateCollectible: {
|
|
|
|
|
const modelItem = store.getCollectibleByKey(key)
|
|
|
|
|
const name = modelItem.name
|
|
|
|
|
const imageSource = modelItem.iconSource.toString()
|
|
|
|
|
|
|
|
|
|
holdingsModel.set(tokensSelector.editedIndex, { type: HoldingTypes.Type.Collectible, key, name, amount, imageSource })
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onUpdateEns: {
|
|
|
|
|
const key = any ? "EnsAny" : "EnsCustom"
|
|
|
|
|
const name = any ? "" : customDomain
|
2022-09-10 00:09:37 +02:00
|
|
|
|
const icon = Style.svg("ensUsernames")
|
2022-09-07 11:40:25 +02:00
|
|
|
|
|
|
|
|
|
holdingsModel.set(tokensSelector.editedIndex, { type: HoldingTypes.Type.Ens, key, name: name, amount: 1, imageSource: icon })
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onRemoveClicked: {
|
|
|
|
|
holdingsModel.remove(tokensSelector.editedIndex)
|
|
|
|
|
|
|
|
|
|
if (holdingsModel.count) {
|
|
|
|
|
holdingsModel.set(0, { operator: SQ.Utils.Operators.None})
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
dropdown.close()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addButton.onClicked: {
|
|
|
|
|
dropdown.parent = tokensSelector.addButton
|
|
|
|
|
dropdown.x = tokensSelector.addButton.width + 4
|
|
|
|
|
dropdown.y = 0
|
|
|
|
|
|
2022-09-10 00:09:37 +02:00
|
|
|
|
if (holdingsModel.count === 0)
|
2022-09-07 11:40:25 +02:00
|
|
|
|
dropdown.openFlow(HoldingsDropdown.FlowType.Add)
|
|
|
|
|
else
|
|
|
|
|
dropdown.openFlow(HoldingsDropdown.FlowType.AddWithOperators)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onItemClicked: {
|
|
|
|
|
if (mouse.button !== Qt.LeftButton)
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
dropdown.parent = item
|
|
|
|
|
dropdown.x = mouse.x + 4
|
|
|
|
|
dropdown.y = 1
|
|
|
|
|
|
|
|
|
|
const modelItem = tokensSelector.itemsModel.get(index)
|
|
|
|
|
|
|
|
|
|
switch(modelItem.type) {
|
|
|
|
|
case HoldingTypes.Type.Token:
|
|
|
|
|
dropdown.tokenKey = modelItem.key
|
|
|
|
|
dropdown.tokenAmount = modelItem.amount
|
|
|
|
|
break
|
|
|
|
|
case HoldingTypes.Type.Collectible:
|
|
|
|
|
dropdown.collectibleKey = modelItem.key
|
|
|
|
|
dropdown.collectibleAmount = modelItem.amount
|
|
|
|
|
dropdown.collectiblesSpecificAmount = modelItem.amount !== 1
|
|
|
|
|
break
|
|
|
|
|
case HoldingTypes.Type.Ens:
|
|
|
|
|
dropdown.ensType = modelItem.name ? EnsPanel.EnsType.CustomSubdomain
|
|
|
|
|
: EnsPanel.EnsType.Any
|
|
|
|
|
dropdown.ensDomainName = modelItem.name
|
|
|
|
|
break
|
|
|
|
|
default:
|
|
|
|
|
console.warn("Unsupported holdings type.")
|
|
|
|
|
}
|
|
|
|
|
|
2022-09-15 12:12:38 +02:00
|
|
|
|
dropdown.openFlow(HoldingsDropdown.FlowType.Update)
|
|
|
|
|
dropdown.setActiveTab(modelItem.type)
|
|
|
|
|
|
2022-09-07 11:40:25 +02:00
|
|
|
|
editedIndex = index
|
2022-06-09 17:27:14 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Rectangle {
|
|
|
|
|
Layout.leftMargin: 16
|
|
|
|
|
Layout.preferredWidth: 2
|
|
|
|
|
Layout.preferredHeight: 24
|
|
|
|
|
color: Style.current.separator
|
|
|
|
|
}
|
|
|
|
|
StatusItemSelector {
|
2022-09-10 00:09:37 +02:00
|
|
|
|
id: permissionsSelector
|
|
|
|
|
|
2022-06-09 17:27:14 +02:00
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
icon: Style.svg("profile/security")
|
|
|
|
|
iconSize: 24
|
2022-09-10 00:09:37 +02:00
|
|
|
|
useIcons: true
|
2022-06-09 17:27:14 +02:00
|
|
|
|
title: qsTr("Is allowed to")
|
|
|
|
|
defaultItemText: qsTr("Example: View and post")
|
2022-09-10 00:09:37 +02:00
|
|
|
|
|
|
|
|
|
Binding on itemsModel {
|
|
|
|
|
when: d.permissionType !== PermissionTypes.Type.None
|
|
|
|
|
value: QtObject {
|
|
|
|
|
id: permissionsListObjectModel
|
|
|
|
|
|
|
|
|
|
readonly property int operator: SQ.Utils.Operators.None
|
|
|
|
|
property string text: ""
|
|
|
|
|
property string imageSource: ""
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addButton.visible: d.permissionType === PermissionTypes.Type.None
|
|
|
|
|
|
|
|
|
|
PermissionsDropdown {
|
|
|
|
|
id: permissionsDropdown
|
|
|
|
|
|
|
|
|
|
initialPermissionType: d.permissionType
|
|
|
|
|
|
|
|
|
|
onDone: {
|
|
|
|
|
d.permissionType = permissionType
|
|
|
|
|
permissionsListObjectModel.text = title
|
|
|
|
|
permissionsListObjectModel.imageSource = asset
|
|
|
|
|
permissionsDropdown.close()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
addButton.onClicked: {
|
|
|
|
|
permissionsDropdown.mode = PermissionsDropdown.Mode.Add
|
|
|
|
|
permissionsDropdown.parent = permissionsSelector.addButton
|
|
|
|
|
permissionsDropdown.x = permissionsSelector.addButton.width + 4
|
|
|
|
|
permissionsDropdown.y = 0
|
|
|
|
|
permissionsDropdown.open()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onItemClicked: {
|
|
|
|
|
if (mouse.button !== Qt.LeftButton)
|
|
|
|
|
return
|
|
|
|
|
|
|
|
|
|
permissionsDropdown.mode = PermissionsDropdown.Mode.Update
|
|
|
|
|
permissionsDropdown.parent = item
|
|
|
|
|
permissionsDropdown.x = mouse.x + 4
|
|
|
|
|
permissionsDropdown.y = 1
|
|
|
|
|
permissionsDropdown.open()
|
|
|
|
|
}
|
2022-06-09 17:27:14 +02:00
|
|
|
|
}
|
|
|
|
|
Rectangle {
|
|
|
|
|
Layout.leftMargin: 16
|
|
|
|
|
Layout.preferredWidth: 2
|
|
|
|
|
Layout.preferredHeight: 24
|
|
|
|
|
color: Style.current.separator
|
|
|
|
|
}
|
|
|
|
|
StatusItemSelector {
|
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
icon: Style.svg("create-category")
|
|
|
|
|
iconSize: 24
|
|
|
|
|
title: qsTr("In")
|
|
|
|
|
defaultItemText: qsTr("Example: `#general` channel")
|
|
|
|
|
}
|
|
|
|
|
Separator {
|
|
|
|
|
Layout.topMargin: 24
|
|
|
|
|
}
|
|
|
|
|
RowLayout {
|
|
|
|
|
Layout.topMargin: 12
|
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
Layout.leftMargin: 16
|
|
|
|
|
Layout.rightMargin: Layout.leftMargin
|
|
|
|
|
spacing: 16
|
|
|
|
|
StatusRoundIcon {
|
2022-08-11 14:55:08 +03:00
|
|
|
|
asset.name: "hide"
|
2022-06-09 17:27:14 +02:00
|
|
|
|
}
|
|
|
|
|
ColumnLayout {
|
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
StatusBaseText {
|
|
|
|
|
text: qsTr("Private")
|
|
|
|
|
color: Theme.palette.directColor1
|
|
|
|
|
font.pixelSize: 15
|
|
|
|
|
}
|
|
|
|
|
StatusBaseText {
|
|
|
|
|
Layout.fillWidth: true
|
|
|
|
|
Layout.fillHeight: true
|
|
|
|
|
text: qsTr("Make this permission private to hide it from members who don’t meet it’s requirements")
|
|
|
|
|
color: Theme.palette.baseColor1
|
|
|
|
|
font.pixelSize: 15
|
|
|
|
|
lineHeight: 1.2
|
|
|
|
|
wrapMode: Text.WordWrap
|
|
|
|
|
elide: Text.ElideRight
|
|
|
|
|
clip: true
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
StatusSwitch {
|
|
|
|
|
checked: d.isPrivate
|
|
|
|
|
onToggled: { d.isPrivate = checked }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
StatusButton {
|
|
|
|
|
Layout.topMargin: 24
|
|
|
|
|
text: qsTr("Create permission")
|
2022-08-23 10:46:37 +02:00
|
|
|
|
enabled: d.permissions.count > 0
|
2022-06-09 17:27:14 +02:00
|
|
|
|
Layout.preferredHeight: 44
|
|
|
|
|
Layout.alignment: Qt.AlignHCenter
|
|
|
|
|
Layout.fillWidth: true
|
2022-08-23 10:46:37 +02:00
|
|
|
|
onClicked: {
|
|
|
|
|
root.store.createPermissions(d.permissions)
|
|
|
|
|
root.clearPermissions()
|
|
|
|
|
}
|
2022-06-09 17:27:14 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|