Skip to content
Snippets Groups Projects
  • q234rty's avatar
    7b07548b
    HiDPI Fixes · 7b07548b
    q234rty authored
    Fix various downscaled icons by removing undeeded multiplications by devicePixelRatio in sourceSize.
    
    Fix downscaled PL indicator in the timeline by using the actual size as sourceSize.
    
    Fix various blurry icons by specifying sourceSize.
    HiDPI Fixes
    q234rty authored
    Fix various downscaled icons by removing undeeded multiplications by devicePixelRatio in sourceSize.
    
    Fix downscaled PL indicator in the timeline by using the actual size as sourceSize.
    
    Fix various blurry icons by specifying sourceSize.
Code owners
Assign users and groups as approvers for specific file changes. Learn more.
UserProfile.qml 20.29 KiB
// SPDX-FileCopyrightText: Nheko Contributors
//
// SPDX-License-Identifier: GPL-3.0-or-later

import ".."
import "../ui"
import "../components"
import QtQuick 2.15
import QtQuick.Controls 2.15
import QtQuick.Layouts 1.2
import QtQuick.Window 2.13
import QtQml.Models 2.2
import im.nheko 1.0

ApplicationWindow {
    id: userProfileDialog

    property var profile

    height: 650
    width: 420
    minimumWidth: 150
    minimumHeight: 150
    color: palette.window
    title: profile.isGlobalUserProfile ? qsTr("Global User Profile") : qsTr("Room User Profile")
    modality: Qt.NonModal
    flags: Qt.Dialog | Qt.WindowCloseButtonHint | Qt.WindowTitleHint

    Shortcut {
        sequence: StandardKey.Cancel
        onActivated: userProfileDialog.close()
    }

    ListView {
        id: devicelist

        property int selectedTab: 0

        Layout.fillHeight: true
        Layout.fillWidth: true
        clip: true
        spacing: 8
        boundsBehavior: Flickable.StopAtBounds
        anchors.fill: parent
        anchors.margins: 10
        footerPositioning: ListView.OverlayFooter

        header: ColumnLayout {
            id: contentL

            width: devicelist.width
            spacing: Nheko.paddingMedium

            Avatar {
                id: displayAvatar

                url: profile.avatarUrl.replace("mxc://", "image://MxcImage/")
                Layout.preferredHeight: 130
                Layout.preferredWidth: 130
                displayName: profile.displayName
                userid: profile.userid
                Layout.alignment: Qt.AlignHCenter
                onClicked: TimelineManager.openImageOverlay(null, profile.avatarUrl, "", 0, 0)

                ImageButton {
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: profile.isGlobalUserProfile ? qsTr("Change avatar globally.") : qsTr("Change avatar. Will only apply to this room.")
                    anchors.left: displayAvatar.left
                    anchors.top: displayAvatar.top
                    anchors.leftMargin: Nheko.paddingMedium
                    anchors.topMargin: Nheko.paddingMedium
                    visible: profile.isSelf
                    image: ":/icons/icons/ui/edit.svg"
                    onClicked: profile.changeAvatar()
                }

            }

            Spinner {
                Layout.alignment: Qt.AlignHCenter
                running: profile.isLoading
                visible: profile.isLoading
                foreground: palette.mid
            }

            Text {
                id: errorText

                color: "red"
                visible: opacity > 0
                opacity: 0
                Layout.alignment: Qt.AlignHCenter
            }

            SequentialAnimation {
                id: hideErrorAnimation

                running: false

                PauseAnimation {
                    duration: 4000
                }

                NumberAnimation {
                    target: errorText
                    property: 'opacity'
                    to: 0
                    duration: 1000
                }

            }

            Connections {
                function onDisplayError(errorMessage) {
                    errorText.text = errorMessage;
                    errorText.opacity = 1;
                    hideErrorAnimation.restart();
                }

                target: profile
            }

            TextInput {
                id: displayUsername

                property bool isUsernameEditingAllowed

                readOnly: !isUsernameEditingAllowed
                text: profile.displayName
                font.pixelSize: 20
                color: TimelineManager.userColor(profile.userid, palette.window)
                font.bold: true
                Layout.alignment: Qt.AlignHCenter
                Layout.maximumWidth: parent.width - (Nheko.paddingSmall * 2) - usernameChangeButton.anchors.leftMargin - (usernameChangeButton.width * 2)
                horizontalAlignment: TextInput.AlignHCenter
                wrapMode: TextInput.Wrap
                selectByMouse: true
                onAccepted: {
                    profile.changeUsername(displayUsername.text);
                    displayUsername.isUsernameEditingAllowed = false;
                }

                ImageButton {
                    id: usernameChangeButton
                    visible: profile.isSelf
                    anchors.leftMargin: Nheko.paddingSmall
                    anchors.left: displayUsername.right
                    anchors.verticalCenter: displayUsername.verticalCenter
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: profile.isGlobalUserProfile ? qsTr("Change display name globally.") : qsTr("Change display name. Will only apply to this room.")
                    image: displayUsername.isUsernameEditingAllowed ? ":/icons/icons/ui/checkmark.svg" : ":/icons/icons/ui/edit.svg"
                    onClicked: {
                        if (displayUsername.isUsernameEditingAllowed) {
                            profile.changeUsername(displayUsername.text);
                            displayUsername.isUsernameEditingAllowed = false;
                        } else {
                            displayUsername.isUsernameEditingAllowed = true;
                            displayUsername.focus = true;
                            displayUsername.selectAll();
                        }
                    }
                }

            }

            MatrixText {
                text: profile.userid
                Layout.alignment: Qt.AlignHCenter
            }

            MatrixText {
                id: statusMsg
                text: qsTr("<i><b>Status:</b> %1</i>").arg(userStatus)
                visible: userStatus != ""
                Layout.fillWidth: true
                horizontalAlignment: TextEdit.AlignHCenter
                Layout.leftMargin: Nheko.paddingMedium
                Layout.rightMargin: Nheko.paddingMedium
                font.pointSize: Math.floor(fontMetrics.font.pointSize * 0.9)

                property string userStatus: Presence.userStatus(profile.userid)
                Connections {
                    target: Presence
                    function onPresenceChanged(id) {
                        if (id == profile.userid) statusMsg.userStatus = Presence.userStatus(profile.userid);
                    }
                }
            }

            RowLayout {
                visible: !profile.isGlobalUserProfile
                Layout.alignment: Qt.AlignHCenter
                spacing: Nheko.paddingSmall

                MatrixText {
                    id: displayRoomname

                    text: qsTr("Room: %1").arg(profile.room ? profile.room.roomName : "")
                    ToolTip.text: qsTr("This is a room-specific profile. The user's name and avatar may be different from their global versions.")
                    ToolTip.visible: ma.hovered
                    Layout.maximumWidth: parent.parent.width - (parent.spacing * 3) - 16
                    horizontalAlignment: TextEdit.AlignHCenter

                    HoverHandler {
                        id: ma
                    }

                }

                ImageButton {
                    image: ":/icons/icons/ui/world.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: qsTr("Open the global profile for this user.")
                    onClicked: profile.openGlobalProfile()
                }

            }

            Button {
                id: verifyUserButton

                text: qsTr("Verify")
                Layout.alignment: Qt.AlignHCenter
                enabled: profile.userVerified != Crypto.Verified
                visible: profile.userVerified != Crypto.Verified && !profile.isSelf && profile.userVerificationEnabled
                onClicked: profile.verify()
            }

            EncryptionIndicator {
                Layout.preferredHeight: 32
                Layout.preferredWidth: 32
                sourceSize.width: width
                sourceSize.height: height
                encrypted: profile.userVerificationEnabled
                trust: profile.userVerified
                Layout.alignment: Qt.AlignHCenter
                ToolTip.visible: false
            }

            RowLayout {
                // ImageButton{
                //     image:":/icons/icons/ui/volume-off-indicator.svg"
                //     Layout.margins: {
                //         left: 5
                //         right: 5
                //     }
                //     ToolTip.visible: hovered
                //     ToolTip.text: qsTr("Ignore messages from this user.")
                //     onClicked : {
                //         profile.ignoreUser()
                //     }
                // }

                Layout.alignment: Qt.AlignHCenter
                Layout.bottomMargin: 10
                spacing: Nheko.paddingSmall

                ImageButton {
                    Layout.preferredHeight: 24
                    Layout.preferredWidth: 24
                    image: ":/icons/icons/ui/chat.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: qsTr("Start a private chat.")
                    onClicked: profile.startChat()
                }

                ImageButton {
                    Layout.preferredHeight: 24
                    Layout.preferredWidth: 24
                    image: ":/icons/icons/ui/round-remove-button.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: qsTr("Kick the user.")
                    onClicked: profile.kickUser()
                    visible: !profile.isGlobalUserProfile && profile.room.permissions.canKick()
                }

                ImageButton {
                    Layout.preferredHeight: 24
                    Layout.preferredWidth: 24
                    image: ":/icons/icons/ui/ban.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: qsTr("Ban the user.")
                    onClicked: profile.banUser()
                    visible: !profile.isGlobalUserProfile && profile.room.permissions.canBan()
                }

                ImageButton {
                    Layout.preferredHeight: 24
                    Layout.preferredWidth: 24
                    image: ":/icons/icons/ui/volume-off-indicator.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: profile.ignored ? qsTr("Unignore the user.") : qsTr("Ignore the user.")
                    buttonTextColor: profile.ignored ? Nheko.theme.red : palette.buttonText
                    onClicked: profile.ignored = !profile.ignored
                    visible: !profile.isSelf
                }

                ImageButton {
                    Layout.preferredHeight: 24
                    Layout.preferredWidth: 24
                    image: ":/icons/icons/ui/refresh.svg"
                    hoverEnabled: true
                    ToolTip.visible: hovered
                    ToolTip.text: qsTr("Refresh device list.")
                    onClicked: profile.refreshDevices()
                }
            }

            TabBar {
                id: tabbar
                visible: !profile.isSelf
                Layout.fillWidth: true

                onCurrentIndexChanged: devicelist.selectedTab = currentIndex


                NhekoTabButton {
                    text: qsTr("Devices")
                }
                NhekoTabButton {
                    text: qsTr("Shared Rooms")
                }

                Layout.bottomMargin: Nheko.paddingMedium
            }
        }

        model: (selectedTab == 0) ? devicesModel : sharedRoomsModel

        DelegateModel {
            id: devicesModel
            model: profile.deviceList
            delegate: RowLayout {
                required property int verificationStatus
                required property string deviceId
                required property string deviceName
                required property string lastIp
                required property var lastTs

                width: devicelist.width
                spacing: 4

                ColumnLayout {
                    spacing: 0

                    Layout.leftMargin: Nheko.paddingMedium
                    Layout.rightMargin: Nheko.paddingMedium
                    RowLayout {
                        Text {
                            Layout.fillWidth: true
                            Layout.alignment: Qt.AlignLeft
                            elide: Text.ElideRight
                            font.bold: true
                            color: palette.text
                            text: deviceId
                        }

                        Image {
                            Layout.preferredHeight: 16
                            Layout.preferredWidth: 16
                            visible: profile.isSelf && verificationStatus != VerificationStatus.NOT_APPLICABLE
                            sourceSize.height: height
                            sourceSize.width: width
                            source: {
                                switch (verificationStatus) {
                                    case VerificationStatus.VERIFIED:
                                    return "image://colorimage/:/icons/icons/ui/shield-filled-checkmark.svg?" + Nheko.theme.green;
                                    case VerificationStatus.UNVERIFIED:
                                    return "image://colorimage/:/icons/icons/ui/shield-filled-exclamation-mark.svg?" + Nheko.theme.orange;
                                    case VerificationStatus.SELF:
                                    return "image://colorimage/:/icons/icons/ui/checkmark.svg?" + Nheko.theme.green;
                                    default:
                                    return "image://colorimage/:/icons/icons/ui/shield-filled-cross.svg?" + Nheko.theme.orange;
                                }
                            }
                        }

                        ImageButton {
                            Layout.alignment: Qt.AlignTop
                            image: ":/icons/icons/ui/power-off.svg"
                            hoverEnabled: true
                            ToolTip.visible: hovered
                            ToolTip.text: qsTr("Sign out this device.")
                            onClicked: profile.signOutDevice(deviceId)
                            visible: profile.isSelf
                        }

                    }

                    RowLayout {
                        id: deviceNameRow

                        property bool isEditingAllowed

                        TextInput {
                            id: deviceNameField

                            readOnly: !deviceNameRow.isEditingAllowed
                            text: deviceName
                            color: palette.text
                            Layout.alignment: Qt.AlignLeft
                            Layout.fillWidth: true
                            selectByMouse: true
                            onAccepted: {
                                profile.changeDeviceName(deviceId, deviceNameField.text);
                                deviceNameRow.isEditingAllowed = false;
                            }
                        }

                        ImageButton {
                            visible: profile.isSelf
                            hoverEnabled: true
                            ToolTip.visible: hovered
                            ToolTip.text: qsTr("Change device name.")
                            image: deviceNameRow.isEditingAllowed ? ":/icons/icons/ui/checkmark.svg" : ":/icons/icons/ui/edit.svg"
                            onClicked: {
                                if (deviceNameRow.isEditingAllowed) {
                                    profile.changeDeviceName(deviceId, deviceNameField.text);
                                    deviceNameRow.isEditingAllowed = false;
                                } else {
                                    deviceNameRow.isEditingAllowed = true;
                                    deviceNameField.focus = true;
                                    deviceNameField.selectAll();
                                }
                            }
                        }

                    }

                    Text {
                        visible: profile.isSelf
                        Layout.fillWidth: true
                        Layout.alignment: Qt.AlignLeft
                        elide: Text.ElideRight
                        color: palette.text
                        text: qsTr("Last seen %1 from %2").arg(new Date(lastTs).toLocaleString(Locale.ShortFormat)).arg(lastIp ? lastIp : "???")
                    }

                }

                Image {
                    Layout.preferredHeight: 16
                    Layout.preferredWidth: 16
                    sourceSize.height: height
                    sourceSize.width: width
                    visible: !profile.isSelf && verificationStatus != VerificationStatus.NOT_APPLICABLE
                    source: {
                        switch (verificationStatus) {
                            case VerificationStatus.VERIFIED:
                            return "image://colorimage/:/icons/icons/ui/shield-filled-checkmark.svg?" + Nheko.theme.green;
                            case VerificationStatus.UNVERIFIED:
                            return "image://colorimage/:/icons/icons/ui/shield-filled-exclamation-mark.svg?" + Nheko.theme.orange;
                            case VerificationStatus.SELF:
                            return "image://colorimage/:/icons/icons/ui/checkmark.svg?" + Nheko.theme.green;
                            default:
                            return "image://colorimage/:/icons/icons/ui/shield-filled.svg?" + Nheko.theme.red;
                        }
                    }
                }

                Button {
                    id: verifyButton

                    visible: verificationStatus == VerificationStatus.UNVERIFIED && (profile.isSelf || !profile.userVerificationEnabled)
                    text: (verificationStatus != VerificationStatus.VERIFIED) ? qsTr("Verify") : qsTr("Unverify")
                    onClicked: {
                        if (verificationStatus == VerificationStatus.VERIFIED)
                        profile.unverify(deviceId);
                        else
                        profile.verify(deviceId);
                    }
                }

            }
        }

        DelegateModel {
            id: sharedRoomsModel
            model: profile.sharedRooms
            delegate: RowLayout {
                required property string roomId
                required property string roomName
                required property string avatarUrl

                width: devicelist.width
                spacing: 4


                Avatar {
                    id: avatar

                    enabled: false
                    Layout.alignment: Qt.AlignVCenter
                    Layout.leftMargin: Nheko.paddingMedium

                    property int avatarSize: Math.ceil(fontMetrics.lineSpacing * 1.6)
                    Layout.preferredHeight: avatarSize
                    Layout.preferredWidth: avatarSize
                    url: avatarUrl.replace("mxc://", "image://MxcImage/")
                    roomid: roomId
                    displayName: roomName
                }

                ElidedLabel {
                    Layout.alignment: Qt.AlignVCenter
                    color: palette.text
                    Layout.fillWidth: true
                    elideWidth: width
                    fullText: roomName
                    textFormat: Text.PlainText
                    Layout.rightMargin: Nheko.paddingMedium
                }

                Item {
                    Layout.fillWidth: true
                }
            }
        }

        footer: DialogButtonBox {
            z: 2
            width: devicelist.width
            alignment: Qt.AlignRight
            standardButtons: DialogButtonBox.Ok
            onAccepted: userProfileDialog.close()

            background: Rectangle {
                anchors.fill: parent
                color: palette.window
            }

        }

    }

}