mirror of
https://github.com/status-im/status-desktop.git
synced 2025-01-19 11:03:40 +00:00
481 lines
18 KiB
QML
481 lines
18 KiB
QML
import QtQuick 2.13
|
|
import QtQuick.Controls 2.13
|
|
import QtQuick.Layouts 1.13
|
|
import QtQml.Models 2.14
|
|
import QtGraphicalEffects 1.13
|
|
|
|
import utils 1.0
|
|
import shared 1.0
|
|
import shared.popups 1.0
|
|
import shared.stores 1.0
|
|
import shared.views 1.0 as SharedViews
|
|
import shared.controls.chat 1.0
|
|
import shared.panels 1.0
|
|
|
|
import StatusQ.Core 0.1
|
|
import StatusQ.Core.Theme 0.1
|
|
import StatusQ.Components 0.1
|
|
import StatusQ.Controls 0.1
|
|
import StatusQ.Popups.Dialog 0.1
|
|
|
|
StatusDialog {
|
|
id: popup
|
|
|
|
property Popup parentPopup
|
|
|
|
property var profileStore
|
|
property var contactsStore
|
|
|
|
property string userPublicKey: ""
|
|
property string userDisplayName: ""
|
|
property string userName: ""
|
|
property string userNickname: ""
|
|
property string userEnsName: ""
|
|
property string userIcon: ""
|
|
property string userBio: ""
|
|
property string userSocialLinks: ""
|
|
property int userTrustStatus: Constants.trustStatus.unknown
|
|
property int outgoingVerificationStatus: Constants.verificationStatus.unverified
|
|
property int incomingVerificationStatus: Constants.verificationStatus.unverified
|
|
property string text: ""
|
|
property string challenge: ""
|
|
property string response: ""
|
|
|
|
property bool userIsEnsVerified: false
|
|
property bool userIsBlocked: false
|
|
property bool userIsUntrustworthy: false
|
|
property bool userTrustIsUnknown: false
|
|
property bool isCurrentUser: false
|
|
property bool isAddedContact: false
|
|
property bool isContact: false
|
|
property bool isVerificationSent: false
|
|
property bool isVerified: false
|
|
property bool isTrusted: false
|
|
property bool hasReceivedVerificationRequest: false
|
|
|
|
property bool showRemoveVerified: false
|
|
property bool showVerifyIdentitySection: false
|
|
property bool showVerificationPendingSection: false
|
|
property bool showIdentityVerified: false
|
|
property bool showIdentityVerifiedUntrustworthy: false
|
|
|
|
property string verificationChallenge: ""
|
|
property string verificationResponse: ""
|
|
property string verificationResponseDisplayName: ""
|
|
property string verificationResponseIcon: ""
|
|
property string verificationRequestedAt: ""
|
|
property string verificationRepliedAt: ""
|
|
|
|
signal blockButtonClicked(name: string, address: string)
|
|
signal unblockButtonClicked(name: string, address: string)
|
|
signal removeButtonClicked(address: string)
|
|
|
|
signal contactUnblocked(publicKey: string)
|
|
signal contactBlocked(publicKey: string)
|
|
|
|
function openPopup(publicKey, state = "") {
|
|
// All this should be improved more, but for now we leave it like this.
|
|
const contactDetails = Utils.getContactDetailsAsJson(publicKey);
|
|
|
|
isCurrentUser = popup.profileStore.pubkey === publicKey;
|
|
|
|
userPublicKey = publicKey;
|
|
userDisplayName = isCurrentUser ? Qt.binding(() => { return popup.profileStore.displayName }) : contactDetails.displayName;
|
|
userName = contactDetails.alias;
|
|
userNickname = contactDetails.localNickname;
|
|
userEnsName = contactDetails.name;
|
|
userIcon = contactDetails.largeImage;
|
|
userBio = contactDetails.bio;
|
|
userSocialLinks = contactDetails.socialLinks;
|
|
userIsEnsVerified = contactDetails.ensVerified;
|
|
userIsBlocked = contactDetails.isBlocked;
|
|
isAddedContact = contactDetails.isAdded;
|
|
isContact = contactDetails.isContact
|
|
userTrustStatus = contactDetails.trustStatus
|
|
userTrustIsUnknown = contactDetails.trustStatus === Constants.trustStatus.unknown
|
|
userIsUntrustworthy = contactDetails.trustStatus === Constants.trustStatus.untrustworthy
|
|
outgoingVerificationStatus = contactDetails.verificationStatus
|
|
incomingVerificationStatus = contactDetails.incomingVerificationStatus
|
|
isVerificationSent = outgoingVerificationStatus !== Constants.verificationStatus.unverified
|
|
|
|
if (isContact && popup.contactsStore.hasReceivedVerificationRequestFrom(publicKey)) {
|
|
popup.hasReceivedVerificationRequest = true
|
|
}
|
|
|
|
if(isContact && isVerificationSent) {
|
|
let verificationDetails = popup.contactsStore.getSentVerificationDetailsAsJson(publicKey);
|
|
|
|
outgoingVerificationStatus = verificationDetails.requestStatus;
|
|
verificationChallenge = verificationDetails.challenge;
|
|
verificationResponse = verificationDetails.response;
|
|
verificationResponseDisplayName = verificationDetails.displayName;
|
|
verificationResponseIcon = verificationDetails.icon;
|
|
verificationRequestedAt = verificationDetails.requestedAt;
|
|
verificationRepliedAt = verificationDetails.repliedAt;
|
|
}
|
|
isTrusted = outgoingVerificationStatus === Constants.verificationStatus.trusted
|
|
|| incomingVerificationStatus === Constants.verificationStatus.trusted
|
|
isVerified = outgoingVerificationStatus === Constants.verificationStatus.verified
|
|
|
|
text = ""; // this is most likely unneeded
|
|
|
|
popup.open();
|
|
|
|
if (state === Constants.profilePopupStates.openNickname) {
|
|
profileView.nicknamePopup.open();
|
|
} else if (state === Constants.profilePopupStates.contactRequest) {
|
|
sendContactRequestModal.open()
|
|
} else if (state === Constants.profilePopupStates.blockUser) {
|
|
blockUser();
|
|
} else if (state === Constants.profilePopupStates.unblockUser) {
|
|
unblockUser();
|
|
} else if (state === Constants.profilePopupStates.verifyIdentity) {
|
|
showVerifyIdentitySection = true;
|
|
} else if (state === Constants.profilePopupStates.respondToPendingRequest) {
|
|
popup.openPendingRequestPopup()
|
|
} else if (state === Constants.profilePopupStates.showVerificationPendingSection) {
|
|
popup.showVerificationPendingSection = true
|
|
profileView.wizardAnimation.running = true
|
|
}
|
|
}
|
|
|
|
function blockUser() {
|
|
profileView.blockContactConfirmationDialog.contactName = userName;
|
|
profileView.blockContactConfirmationDialog.contactAddress = userPublicKey;
|
|
profileView.blockContactConfirmationDialog.open();
|
|
}
|
|
|
|
function unblockUser() {
|
|
profileView.unblockContactConfirmationDialog.contactName = userName;
|
|
profileView.unblockContactConfirmationDialog.contactAddress = userPublicKey;
|
|
profileView.unblockContactConfirmationDialog.open();
|
|
}
|
|
|
|
function openPendingRequestPopup() {
|
|
try {
|
|
let request = popup.contactsStore.getVerificationDetailsFromAsJson(popup.userPublicKey)
|
|
Global.openPopup(contactVerificationRequestPopupComponent, {
|
|
senderPublicKey: request.from,
|
|
senderDisplayName: request.displayName,
|
|
senderIcon: request.icon,
|
|
challengeText: request.challenge,
|
|
responseText: request.response,
|
|
messageTimestamp: request.requestedAt,
|
|
responseTimestamp: request.repliedAt
|
|
})
|
|
} catch (e) {
|
|
console.error("Error getting or parsing verification data", e)
|
|
}
|
|
}
|
|
|
|
width: 700
|
|
padding: 8
|
|
|
|
header: StatusDialogHeader {
|
|
id: dialogHeader
|
|
|
|
headline.title: {
|
|
if(showVerifyIdentitySection || showVerificationPendingSection){
|
|
return qsTr("Verify %1's Identity").arg(userIsEnsVerified ? userName : userDisplayName)
|
|
}
|
|
return popup.isCurrentUser ? qsTr("My Profile") :
|
|
qsTr("%1's Profile").arg(userIsEnsVerified ? userName : userDisplayName)
|
|
}
|
|
|
|
headline.subtitle: popup.isCurrentUser ? "" : userIsEnsVerified ? userName : Utils.getElidedCompressedPk(userPublicKey)
|
|
|
|
actions {
|
|
customButtons: ObjectModel {
|
|
StatusFlatRoundButton {
|
|
type: StatusFlatRoundButton.Type.Secondary
|
|
width: 32
|
|
height: 32
|
|
|
|
icon.width: 20
|
|
icon.height: 20
|
|
icon.name: "qr"
|
|
onClicked: profileView.qrCodePopup.open()
|
|
}
|
|
}
|
|
|
|
closeButton.onClicked: popup.close()
|
|
}
|
|
}
|
|
|
|
footer: StatusDialogFooter {
|
|
visible: !popup.isCurrentUser
|
|
|
|
leftButtons: ObjectModel {
|
|
StatusButton {
|
|
text: qsTr("Cancel verification")
|
|
visible: !isVerified && isContact && isVerificationSent && showVerificationPendingSection
|
|
onClicked: {
|
|
popup.contactsStore.cancelVerificationRequest(userPublicKey);
|
|
popup.close()
|
|
}
|
|
}
|
|
}
|
|
|
|
rightButtons: ObjectModel {
|
|
StatusFlatButton {
|
|
text: userIsBlocked ?
|
|
qsTr("Unblock User") :
|
|
qsTr("Block User")
|
|
type: StatusBaseButton.Type.Danger
|
|
visible: !isAddedContact
|
|
onClicked: userIsBlocked ? unblockUser() : blockUser()
|
|
}
|
|
|
|
StatusFlatButton {
|
|
visible: !showRemoveVerified && !showIdentityVerified && !showVerifyIdentitySection && !showVerificationPendingSection && !userIsBlocked && isAddedContact
|
|
type: StatusBaseButton.Type.Danger
|
|
text: qsTr('Remove Contact')
|
|
onClicked: {
|
|
profileView.removeContactConfirmationDialog.parentPopup = popup;
|
|
profileView.removeContactConfirmationDialog.open();
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Send Contact Request")
|
|
visible: !userIsBlocked && !isAddedContact
|
|
onClicked: sendContactRequestModal.open()
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Mark Untrustworthy")
|
|
visible: !showIdentityVerifiedUntrustworthy && !showIdentityVerified && !showVerifyIdentitySection && userTrustIsUnknown
|
|
enabled: !showVerificationPendingSection || verificationResponse !== ""
|
|
type: StatusBaseButton.Type.Danger
|
|
onClicked: {
|
|
if (showVerificationPendingSection) {
|
|
popup.showIdentityVerified = false;
|
|
popup.showIdentityVerifiedUntrustworthy = true;
|
|
popup.showVerificationPendingSection = false;
|
|
popup.showVerifyIdentitySection = false;
|
|
profileView.stepsListModel.setProperty(2, "stepCompleted", true);
|
|
popup.contactsStore.verifiedUntrustworthy(userPublicKey);
|
|
} else {
|
|
popup.contactsStore.markUntrustworthy(userPublicKey);
|
|
popup.close();
|
|
}
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Remove 'Identity Verified' status")
|
|
visible: isTrusted && !showIdentityVerified && !showRemoveVerified
|
|
type: StatusBaseButton.Type.Danger
|
|
onClicked: {
|
|
showRemoveVerified = true
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("No")
|
|
visible: showRemoveVerified
|
|
type: StatusBaseButton.Type.Danger
|
|
onClicked: {
|
|
showRemoveVerified = false
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Yes")
|
|
visible: showRemoveVerified
|
|
onClicked: {
|
|
popup.contactsStore.removeTrustStatus(userPublicKey);
|
|
popup.close();
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Remove Untrustworthy Mark")
|
|
visible: userIsUntrustworthy
|
|
onClicked: {
|
|
popup.contactsStore.removeTrustStatus(userPublicKey);
|
|
popup.close();
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Verify Identity")
|
|
visible: !showIdentityVerifiedUntrustworthy && !showIdentityVerified &&
|
|
!showVerifyIdentitySection && isContact && !isVerificationSent
|
|
&& !hasReceivedVerificationRequest
|
|
onClicked: {
|
|
popup.showVerifyIdentitySection = true
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Verify Identity pending...")
|
|
visible: (!showIdentityVerifiedUntrustworthy && !showIdentityVerified && !isTrusted
|
|
&& isContact && isVerificationSent && !showVerificationPendingSection) ||
|
|
(hasReceivedVerificationRequest && !isTrusted)
|
|
onClicked: {
|
|
if (hasReceivedVerificationRequest) {
|
|
popup.openPendingRequestPopup()
|
|
} else {
|
|
popup.showVerificationPendingSection = true
|
|
profileView.wizardAnimation.running = true
|
|
}
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Send verification request")
|
|
visible: showVerifyIdentitySection && isContact && !isVerificationSent
|
|
onClicked: {
|
|
popup.contactsStore.sendVerificationRequest(userPublicKey, Utils.escapeHtml(profileView.challengeTxt.input.text));
|
|
profileView.stepsListModel.setProperty(1, "stepCompleted", true);
|
|
Global.displayToastMessage(qsTr("Verification request sent"),
|
|
"",
|
|
"checkmark-circle",
|
|
false,
|
|
Constants.ephemeralNotificationType.normal,
|
|
"");
|
|
popup.close();
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
text: qsTr("Confirm Identity")
|
|
visible: isContact && isVerificationSent && !isTrusted && showVerificationPendingSection
|
|
enabled: verificationChallenge !== "" && verificationResponse !== ""
|
|
onClicked: {
|
|
popup.showIdentityVerified = true;
|
|
popup.showIdentityVerifiedUntrustworthy = false;
|
|
popup.showVerificationPendingSection = false;
|
|
popup.showVerifyIdentitySection = false;
|
|
profileView.stepsListModel.setProperty(2, "stepCompleted", true);
|
|
popup.contactsStore.verifiedTrusted(userPublicKey);
|
|
popup.isTrusted = true
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
visible: showIdentityVerified || showIdentityVerifiedUntrustworthy
|
|
text: qsTr("Rename")
|
|
onClicked: {
|
|
profileView.nicknamePopup.open()
|
|
}
|
|
}
|
|
|
|
StatusButton {
|
|
visible: showIdentityVerified || showIdentityVerifiedUntrustworthy
|
|
text: qsTr("Close")
|
|
onClicked: {
|
|
popup.close();
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
StatusScrollView {
|
|
id: scrollView
|
|
|
|
anchors.fill: parent
|
|
padding: 0
|
|
|
|
SharedViews.ProfileView {
|
|
id: profileView
|
|
|
|
width: scrollView.availableWidth
|
|
|
|
profileStore: popup.profileStore
|
|
contactsStore: popup.contactsStore
|
|
|
|
|
|
userPublicKey: popup.userPublicKey
|
|
userDisplayName: popup.userDisplayName
|
|
userName: popup.userName
|
|
userNickname: popup.userNickname
|
|
userEnsName: popup.userEnsName
|
|
userIcon: popup.userIcon
|
|
userBio: popup.userBio
|
|
userSocialLinks: popup.userSocialLinks
|
|
userIsEnsVerified: popup.userIsEnsVerified
|
|
userIsBlocked: popup.userIsBlocked
|
|
isAddedContact: popup.isAddedContact
|
|
isCurrentUser: popup.isCurrentUser
|
|
|
|
isContact: popup.isContact
|
|
isVerificationSent: popup.isVerificationSent
|
|
isVerified: popup.isVerified
|
|
isTrusted: popup.isTrusted
|
|
hasReceivedVerificationRequest: popup.hasReceivedVerificationRequest
|
|
|
|
userTrustStatus: popup.userTrustStatus
|
|
outgoingVerificationStatus: popup.outgoingVerificationStatus
|
|
|
|
showVerifyIdentitySection: popup.showVerifyIdentitySection
|
|
showVerificationPendingSection: popup.showVerificationPendingSection
|
|
showIdentityVerified: popup.showIdentityVerified
|
|
showIdentityVerifiedUntrustworthy: popup.showIdentityVerifiedUntrustworthy
|
|
|
|
challenge: popup.challenge
|
|
response: popup.response
|
|
|
|
userIsUntrustworthy: popup.userIsUntrustworthy
|
|
userTrustIsUnknown: popup.userTrustIsUnknown
|
|
|
|
verificationChallenge: popup.verificationChallenge
|
|
verificationResponse: popup.verificationResponse
|
|
verificationResponseDisplayName: popup.verificationResponseDisplayName
|
|
verificationResponseIcon: popup.verificationResponseIcon
|
|
verificationRequestedAt: popup.verificationRequestedAt
|
|
verificationRepliedAt: popup.verificationRepliedAt
|
|
|
|
onContactUnblocked: {
|
|
popup.close()
|
|
popup.contactUnblocked(publicKey)
|
|
}
|
|
|
|
onContactBlocked: {
|
|
popup.close()
|
|
popup.contactBlocked(publicKey)
|
|
}
|
|
|
|
onContactAdded: {
|
|
popup.close()
|
|
popup.contactAdded(publicKey)
|
|
}
|
|
|
|
onContactRemoved: {
|
|
popup.close()
|
|
}
|
|
|
|
onNicknameEdited: {
|
|
popup.close()
|
|
}
|
|
}
|
|
}
|
|
|
|
// TODO: replace with StatusStackModal
|
|
SendContactRequestModal {
|
|
id: sendContactRequestModal
|
|
anchors.centerIn: parent
|
|
width: popup.width
|
|
visible: false
|
|
header.title: qsTr("Send Contact Request to %1").arg(userDisplayName)
|
|
userPublicKey: popup.userPublicKey
|
|
userDisplayName: popup.userDisplayName
|
|
userIcon: popup.userIcon
|
|
onAccepted: popup.contactsStore.sendContactRequest(userPublicKey, message)
|
|
onClosed: popup.close()
|
|
}
|
|
|
|
Component {
|
|
id: contactVerificationRequestPopupComponent
|
|
ContactVerificationRequestPopup {
|
|
onResponseSent: {
|
|
popup.contactsStore.acceptVerificationRequest(senderPublicKey, response)
|
|
}
|
|
onVerificationRefused: {
|
|
popup.contactsStore.declineVerificationRequest(senderPublicKey)
|
|
}
|
|
}
|
|
}
|
|
}
|