2023-07-04 17:11:41 +02:00
|
|
|
import QtQuick 2.15
|
|
|
|
import QtQml.Models 2.15
|
|
|
|
import QtQuick.Controls 2.15
|
|
|
|
import QtQuick.Layouts 1.15
|
|
|
|
import QtGraphicalEffects 1.15
|
|
|
|
|
2024-04-01 16:24:18 +02:00
|
|
|
import StatusQ 0.1
|
2023-07-04 17:11:41 +02:00
|
|
|
import StatusQ.Core 0.1
|
|
|
|
import StatusQ.Components 0.1
|
|
|
|
import StatusQ.Controls 0.1
|
|
|
|
import StatusQ.Core.Theme 0.1
|
|
|
|
import StatusQ.Core.Utils 0.1
|
2023-08-02 19:39:42 +02:00
|
|
|
import StatusQ.Internal 0.1 as SQInternal
|
2023-07-04 17:11:41 +02:00
|
|
|
|
|
|
|
import SortFilterProxyModel 0.2
|
|
|
|
|
|
|
|
import AppLayouts.Profile.controls 1.0
|
|
|
|
import AppLayouts.Communities.controls 1.0
|
|
|
|
import AppLayouts.Communities.views 1.0
|
|
|
|
import AppLayouts.Communities.helpers 1.0
|
|
|
|
|
|
|
|
import utils 1.0
|
|
|
|
|
|
|
|
Control {
|
|
|
|
id: root
|
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
required property string componentUid
|
|
|
|
required property bool isEditMode
|
|
|
|
required property var selectedSharedAddressesMap // Map[address, [keyUid, selected, isAirdrop]
|
|
|
|
property var currentSharedAddressesMap // Map[address, [keyUid, selected, isAirdrop]
|
|
|
|
required property int totalNumOfAddressesForSharing
|
|
|
|
required property bool profileProvesOwnershipOfSelectedAddresses
|
|
|
|
required property bool allAddressesToRevealBelongToSingleNonProfileKeypair
|
2024-03-27 13:48:17 +01:00
|
|
|
required property int /*PermissionTypes.Type*/ eligibleToJoinAs
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-05-16 11:28:06 +02:00
|
|
|
property bool requirementsCheckPending
|
|
|
|
property bool checkingPermissionToJoinInProgress
|
|
|
|
property bool joinPermissionsCheckCompletedWithoutErrors
|
2023-08-22 14:04:58 -04:00
|
|
|
|
2024-03-29 00:05:59 +01:00
|
|
|
required property string communityId
|
2023-07-04 17:11:41 +02:00
|
|
|
required property string communityName
|
|
|
|
required property string communityIcon
|
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
required property var walletAssetsModel
|
2024-03-29 00:05:59 +01:00
|
|
|
required property var walletCollectiblesModel
|
|
|
|
|
2024-04-01 16:24:18 +02:00
|
|
|
required property var walletAccountsModel // name, address, emoji, colorId
|
2023-07-04 17:11:41 +02:00
|
|
|
required property var permissionsModel // id, key, permissionType, holdingsListModel, channelsListModel, isPrivate, tokenCriteriaMet
|
2024-03-29 00:05:59 +01:00
|
|
|
|
2023-07-04 17:11:41 +02:00
|
|
|
required property var assetsModel
|
|
|
|
required property var collectiblesModel
|
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property string title: isEditMode ? qsTr("Edit which addresses you share with %1").arg(root.communityName)
|
|
|
|
: qsTr("Select addresses to share with %1").arg(root.communityName)
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property var rightButtons: root.isEditMode? [d.cancelButton, d.saveButton] : [d.shareAddressesButton]
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-01-24 16:35:53 +00:00
|
|
|
property var getCurrencyAmount: function (balance, symbol){}
|
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
signal toggleAddressSelection(string keyUid, string address)
|
|
|
|
signal airdropAddressSelected (string address)
|
|
|
|
signal shareSelectedAddressesClicked()
|
2023-08-02 19:39:42 +02:00
|
|
|
signal close()
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2023-08-02 19:39:42 +02:00
|
|
|
padding: 0
|
2023-07-04 17:11:41 +02:00
|
|
|
spacing: Style.current.padding
|
|
|
|
|
|
|
|
QtObject {
|
|
|
|
id: d
|
|
|
|
|
|
|
|
// internal logic
|
2024-05-20 18:45:30 +03:00
|
|
|
readonly property bool hasPermissions: permissionsView.hasAnyVisiblePermission
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property int selectedSharedAddressesCount: root.selectedSharedAddressesMap.size
|
2023-08-02 19:39:42 +02:00
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property bool dirty: {
|
|
|
|
if (root.currentSharedAddressesMap.size !== root.selectedSharedAddressesMap.size) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
for (const [key, value] of root.currentSharedAddressesMap) {
|
|
|
|
const obj = root.selectedSharedAddressesMap.get(key)
|
|
|
|
if (!obj || value.selected !== obj.selected || value.isAirdrop !== obj.isAirdrop) {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
|
2024-04-01 16:24:18 +02:00
|
|
|
property var tokenCountMap: new Map()
|
|
|
|
function getTokenCount(address) {
|
|
|
|
if (d.tokenCountMap.has(address))
|
|
|
|
return d.tokenCountMap.get(address)
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
|
2023-08-02 19:39:42 +02:00
|
|
|
// warning states
|
|
|
|
readonly property bool lostCommunityPermission: root.isEditMode && permissionsView.lostPermissionToJoin
|
|
|
|
readonly property bool lostChannelPermissions: root.isEditMode && permissionsView.lostChannelPermissions
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property var cancelButton: StatusFlatButton {
|
|
|
|
visible: root.isEditMode
|
|
|
|
borderColor: Theme.palette.baseColor2
|
|
|
|
text: qsTr("Cancel")
|
|
|
|
onClicked: root.close()
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
|
2024-04-25 11:47:40 +02:00
|
|
|
readonly property string tooltipText: {
|
2024-05-16 11:28:06 +02:00
|
|
|
if (root.requirementsCheckPending || root.checkingPermissionToJoinInProgress)
|
2024-04-25 11:47:40 +02:00
|
|
|
return qsTr("Requirements check pending")
|
|
|
|
|
2024-05-16 11:28:06 +02:00
|
|
|
if (!root.joinPermissionsCheckCompletedWithoutErrors)
|
|
|
|
return qsTr("Checking permissions failed")
|
2024-04-25 11:47:40 +02:00
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
readonly property var saveButton: StatusButton {
|
|
|
|
visible: root.isEditMode
|
2024-05-16 11:28:06 +02:00
|
|
|
interactive: d.dirty && !root.requirementsCheckPending && root.joinPermissionsCheckCompletedWithoutErrors
|
2024-04-25 11:47:40 +02:00
|
|
|
loading: root.requirementsCheckPending
|
|
|
|
type: d.lostCommunityPermission || d.lostChannelPermissions ? StatusBaseButton.Type.Danger : StatusBaseButton.Type.Normal
|
|
|
|
tooltip.text: {
|
|
|
|
if (interactive)
|
|
|
|
return ""
|
2023-08-24 16:36:38 -04:00
|
|
|
|
2024-04-25 11:47:40 +02:00
|
|
|
return d.tooltipText
|
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
text: {
|
|
|
|
if (d.lostCommunityPermission) {
|
|
|
|
return qsTr("Save changes & leave %1").arg(root.communityName)
|
|
|
|
}
|
|
|
|
if (d.lostChannelPermissions) {
|
|
|
|
return qsTr("Save changes & update my permissions")
|
|
|
|
}
|
|
|
|
if (d.selectedSharedAddressesCount === root.totalNumOfAddressesForSharing) {
|
|
|
|
return qsTr("Reveal all addresses")
|
|
|
|
}
|
|
|
|
return qsTr("Reveal %n address(s)", "", d.selectedSharedAddressesCount)
|
|
|
|
}
|
2023-08-24 16:36:38 -04:00
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
icon.name: {
|
|
|
|
if (!d.lostCommunityPermission
|
|
|
|
&& !d.lostChannelPermissions
|
|
|
|
&& root.profileProvesOwnershipOfSelectedAddresses) {
|
|
|
|
if (userProfile.usingBiometricLogin) {
|
|
|
|
return "touch-id"
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userProfile.isKeycardUser) {
|
|
|
|
return "keycard"
|
|
|
|
}
|
|
|
|
|
|
|
|
return "password"
|
|
|
|
}
|
|
|
|
if (root.allAddressesToRevealBelongToSingleNonProfileKeypair) {
|
|
|
|
return "keycard"
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
onClicked: {
|
|
|
|
root.shareSelectedAddressesClicked()
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
|
|
|
|
readonly property var shareAddressesButton: StatusButton {
|
|
|
|
visible: !root.isEditMode
|
2024-05-16 11:28:06 +02:00
|
|
|
interactive: !root.checkingPermissionToJoinInProgress && root.eligibleToJoinAs !== PermissionTypes.Type.None && root.joinPermissionsCheckCompletedWithoutErrors
|
|
|
|
loading: root.checkingPermissionToJoinInProgress
|
2024-04-25 11:47:40 +02:00
|
|
|
tooltip.text: {
|
|
|
|
if (interactive)
|
|
|
|
return ""
|
|
|
|
|
|
|
|
return d.tooltipText
|
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
text: {
|
|
|
|
if (d.selectedSharedAddressesCount === root.totalNumOfAddressesForSharing) {
|
|
|
|
return qsTr("Share all addresses to join")
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
return qsTr("Share %n address(s) to join", "", d.selectedSharedAddressesCount)
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
icon.name: {
|
|
|
|
if (root.profileProvesOwnershipOfSelectedAddresses) {
|
|
|
|
if (userProfile.usingBiometricLogin) {
|
|
|
|
return "touch-id"
|
|
|
|
}
|
|
|
|
|
|
|
|
if (userProfile.isKeycardUser) {
|
|
|
|
return "keycard"
|
|
|
|
}
|
|
|
|
|
|
|
|
return "password"
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
if (root.allAddressesToRevealBelongToSingleNonProfileKeypair) {
|
|
|
|
return "keycard"
|
|
|
|
}
|
|
|
|
|
|
|
|
return ""
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
|
|
|
|
onClicked: {
|
|
|
|
root.shareSelectedAddressesClicked()
|
|
|
|
}
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2023-07-04 17:11:41 +02:00
|
|
|
|
2024-05-01 18:03:20 +02:00
|
|
|
contentItem: ColumnLayout {
|
2024-04-29 17:39:46 +02:00
|
|
|
anchors.fill: parent
|
2023-07-04 17:11:41 +02:00
|
|
|
spacing: 0
|
2023-08-02 19:39:42 +02:00
|
|
|
|
|
|
|
// warning panel
|
2024-04-12 11:52:53 +02:00
|
|
|
Rectangle {
|
2023-08-02 19:39:42 +02:00
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.preferredHeight: 32
|
2024-04-12 11:52:53 +02:00
|
|
|
color: Theme.palette.dangerColor1
|
|
|
|
visible: d.lostCommunityPermission || d.lostChannelPermissions
|
|
|
|
|
|
|
|
StatusBaseText {
|
|
|
|
width: parent.width
|
|
|
|
anchors.verticalCenter: parent.verticalCenter
|
|
|
|
horizontalAlignment: Qt.AlignHCenter
|
|
|
|
elide: Text.ElideRight
|
|
|
|
text: d.lostCommunityPermission ? qsTr("Selected addresses have insufficient tokens to maintain %1 membership").arg(root.communityName) :
|
|
|
|
d.lostChannelPermissions ? qsTr("By deselecting these addresses, you will lose channel permissions") :
|
|
|
|
""
|
|
|
|
font.pixelSize: Style.current.additionalTextSize
|
|
|
|
font.weight: Font.Medium
|
|
|
|
color: Theme.palette.indirectColor1
|
|
|
|
}
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
|
|
|
|
2023-07-04 17:11:41 +02:00
|
|
|
// addresses
|
|
|
|
SharedAddressesAccountSelector {
|
|
|
|
id: accountSelector
|
|
|
|
hasPermissions: d.hasPermissions
|
2024-03-29 00:05:59 +01:00
|
|
|
|
2023-07-04 17:11:41 +02:00
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.preferredHeight: contentHeight + topMargin + bottomMargin
|
|
|
|
Layout.maximumHeight: hasPermissions ? permissionsView.implicitHeight > root.availableHeight / 2 ? root.availableHeight / 2 : root.availableHeight : -1
|
|
|
|
Layout.fillHeight: !hasPermissions
|
2024-04-01 16:24:18 +02:00
|
|
|
implicitHeight: contentHeight
|
2024-03-29 00:05:59 +01:00
|
|
|
|
|
|
|
uniquePermissionAssetsKeys:
|
|
|
|
PermissionsHelpers.getUniquePermissionTokenKeys(
|
|
|
|
root.permissionsModel, Constants.TokenType.ERC20)
|
|
|
|
|
|
|
|
uniquePermissionCollectiblesKeys:
|
|
|
|
PermissionsHelpers.getUniquePermissionTokenKeys(
|
|
|
|
root.permissionsModel, Constants.TokenType.ERC721)
|
|
|
|
|
2024-04-01 16:24:18 +02:00
|
|
|
model: SortFilterProxyModel {
|
|
|
|
sourceModel: root.walletAccountsModel
|
|
|
|
proxyRoles: FastExpressionRole {
|
|
|
|
name: "tokenCount"
|
|
|
|
expression: {
|
|
|
|
d.tokenCountMap
|
|
|
|
return d.getTokenCount(model.address.toLowerCase())
|
|
|
|
}
|
|
|
|
expectedRoles: ["address"]
|
|
|
|
}
|
|
|
|
|
|
|
|
sorters: [
|
|
|
|
// FIXME add sort token-relevant accounts first; https://github.com/status-im/status-desktop/issues/14192
|
|
|
|
RoleSorter {
|
|
|
|
roleName: "tokenCount"
|
|
|
|
sortOrder: Qt.DescendingOrder
|
|
|
|
},
|
|
|
|
RoleSorter {
|
|
|
|
roleName: "name"
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}
|
2024-01-24 16:35:53 +00:00
|
|
|
walletAssetsModel: root.walletAssetsModel
|
2024-03-29 00:05:59 +01:00
|
|
|
walletCollectiblesModel: root.walletCollectiblesModel
|
|
|
|
|
|
|
|
communityId: root.communityId
|
|
|
|
communityCollectiblesModel: root.collectiblesModel
|
|
|
|
|
2024-03-19 09:41:41 +01:00
|
|
|
selectedSharedAddressesMap: root.selectedSharedAddressesMap
|
|
|
|
|
|
|
|
onToggleAddressSelection: {
|
|
|
|
root.toggleAddressSelection(keyUid, address)
|
|
|
|
}
|
|
|
|
|
|
|
|
onAirdropAddressSelected: {
|
|
|
|
root.airdropAddressSelected(address)
|
2023-08-02 19:39:42 +02:00
|
|
|
}
|
2024-03-19 09:41:41 +01:00
|
|
|
|
2024-03-27 13:48:17 +01:00
|
|
|
getCurrencyAmount: function (balance, symbol) {
|
2024-01-24 16:35:53 +00:00
|
|
|
return root.getCurrencyAmount(balance, symbol)
|
|
|
|
}
|
2024-04-01 16:24:18 +02:00
|
|
|
|
|
|
|
Component.onCompleted: {
|
|
|
|
const tmpTokenCountMap = new Map()
|
|
|
|
for (let i = 0; i < accountSelector.count; i++) {
|
|
|
|
const item = accountSelector.itemAtIndex(i)
|
|
|
|
if (!!item) {
|
|
|
|
tmpTokenCountMap.set(item.address.toLowerCase(), item.tokenCount)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
d.tokenCountMap = tmpTokenCountMap
|
|
|
|
}
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// divider with top rounded corners + drop shadow
|
|
|
|
Rectangle {
|
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.preferredHeight: Style.current.padding * 2
|
2024-03-27 13:48:17 +01:00
|
|
|
color: permissionsView.color
|
2023-07-04 17:11:41 +02:00
|
|
|
radius: Style.current.padding
|
|
|
|
border.width: 1
|
|
|
|
border.color: Theme.palette.baseColor3
|
2024-01-18 15:06:25 +01:00
|
|
|
visible: permissionsView.hasAnyVisiblePermission
|
2023-07-04 17:11:41 +02:00
|
|
|
|
|
|
|
layer.enabled: true
|
|
|
|
layer.effect: DropShadow {
|
|
|
|
horizontalOffset: 0
|
|
|
|
verticalOffset: -9
|
|
|
|
radius: 14
|
|
|
|
samples: 29
|
|
|
|
color: Qt.rgba(0, 0, 0, 0.04)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// permissions
|
|
|
|
SharedAddressesPermissionsPanel {
|
|
|
|
id: permissionsView
|
2023-08-02 19:39:42 +02:00
|
|
|
isEditMode: root.isEditMode
|
2024-04-12 11:52:53 +02:00
|
|
|
isDirty: d.dirty
|
2023-07-04 17:11:41 +02:00
|
|
|
permissionsModel: root.permissionsModel
|
|
|
|
assetsModel: root.assetsModel
|
|
|
|
collectiblesModel: root.collectiblesModel
|
2024-03-25 19:39:45 +01:00
|
|
|
requirementsCheckPending: root.requirementsCheckPending
|
2024-05-16 11:28:06 +02:00
|
|
|
checkingPermissionToJoinInProgress: root.checkingPermissionToJoinInProgress
|
|
|
|
joinPermissionsCheckCompletedWithoutErrors: root.joinPermissionsCheckCompletedWithoutErrors
|
2023-07-04 17:11:41 +02:00
|
|
|
communityName: root.communityName
|
|
|
|
communityIcon: root.communityIcon
|
2024-03-27 13:48:17 +01:00
|
|
|
eligibleToJoinAs: root.eligibleToJoinAs
|
2023-07-04 17:11:41 +02:00
|
|
|
|
|
|
|
Layout.fillHeight: true
|
|
|
|
Layout.fillWidth: true
|
|
|
|
Layout.topMargin: -Style.current.padding // compensate for the half-rounded divider above
|
2024-01-18 15:06:25 +01:00
|
|
|
visible: permissionsView.hasAnyVisiblePermission
|
2023-07-04 17:11:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|