feat: add profile pic support
This commit is contained in:
parent
43d9d5184d
commit
d01c9fef79
|
@ -15,6 +15,8 @@ import web3/[conversions, ethtypes]
|
|||
import ../../status/threads
|
||||
import views/[channels_list, message_list, chat_item, suggestions_list, reactions, stickers, groups, transactions]
|
||||
import json_serialization
|
||||
import ../../status/libstatus/utils
|
||||
import ../utils/image_utils
|
||||
|
||||
logScope:
|
||||
topics = "chats-view"
|
||||
|
@ -149,10 +151,7 @@ QtObject:
|
|||
proc sendImage*(self: ChatsView, imagePath: string): string {.slot.} =
|
||||
result = ""
|
||||
try:
|
||||
var image: string = replace(imagePath, "file://", "")
|
||||
if defined(windows):
|
||||
# Windows doesn't work with paths starting with a slash
|
||||
image.removePrefix('/')
|
||||
var image = image_utils.formatImagePath(imagePath)
|
||||
let tmpImagePath = image_resizer(image, 2000, TMPDIR)
|
||||
self.status.chat.sendImage(self.activeChannel.id, tmpImagePath)
|
||||
removeFile(tmpImagePath)
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import NimQml, Tables, json, nimcrypto, strformat, json_serialization
|
||||
import NimQml, Tables, json, nimcrypto, strformat, json_serialization, chronicles
|
||||
import ../../status/signals/types
|
||||
import ../../status/libstatus/types as status_types
|
||||
import ../../status/libstatus/accounts as status_accounts
|
||||
|
@ -9,9 +9,11 @@ import core
|
|||
|
||||
type
|
||||
AccountRoles {.pure.} = enum
|
||||
Username = UserRole + 1,
|
||||
Identicon = UserRole + 2,
|
||||
Username = UserRole + 1
|
||||
Identicon = UserRole + 2
|
||||
Address = UserRole + 3
|
||||
ThumbnailImage = UserRole + 4
|
||||
LargeImage = UserRole + 5
|
||||
|
||||
QtObject:
|
||||
type LoginView* = ref object of QAbstractListModel
|
||||
|
@ -41,7 +43,12 @@ QtObject:
|
|||
|
||||
proc setCurrentAccount*(self: LoginView, selectedAccountIdx: int) {.slot.} =
|
||||
let currNodeAcct = self.accounts[selectedAccountIdx]
|
||||
self.currentAccount.setAccount(GeneratedAccount(name: currNodeAcct.name, photoPath: currNodeAcct.photoPath, address: currNodeAcct.keyUid))
|
||||
self.currentAccount.setAccount(GeneratedAccount(
|
||||
name: currNodeAcct.name,
|
||||
identicon: currNodeAcct.identicon,
|
||||
address: currNodeAcct.keyUid,
|
||||
identityImage: currNodeAcct.identityImage
|
||||
))
|
||||
|
||||
QtProperty[QVariant] currentAccount:
|
||||
read = getCurrentAccount
|
||||
|
@ -72,13 +79,25 @@ QtObject:
|
|||
let assetRole = role.AccountRoles
|
||||
case assetRole:
|
||||
of AccountRoles.Username: result = newQVariant(asset.name)
|
||||
of AccountRoles.Identicon: result = newQVariant(asset.photoPath)
|
||||
of AccountRoles.Identicon: result = newQVariant(asset.identicon)
|
||||
of AccountRoles.Address: result = newQVariant(asset.keyUid)
|
||||
of AccountRoles.ThumbnailImage:
|
||||
if (not asset.identityImage.isNil):
|
||||
result = newQVariant(asset.identityImage.thumbnail)
|
||||
else:
|
||||
result = newQVariant(asset.identicon)
|
||||
of AccountRoles.LargeImage:
|
||||
if (not asset.identityImage.isNil):
|
||||
result = newQVariant(asset.identityImage.large)
|
||||
else:
|
||||
result = newQVariant(asset.identicon)
|
||||
|
||||
method roleNames(self: LoginView): Table[int, string] =
|
||||
{ AccountRoles.Username.int:"username",
|
||||
AccountRoles.Identicon.int:"identicon",
|
||||
AccountRoles.Address.int:"address" }.toTable
|
||||
AccountRoles.Address.int:"address",
|
||||
AccountRoles.ThumbnailImage.int:"thumbnailImage",
|
||||
AccountRoles.LargeImage.int:"largeImage" }.toTable
|
||||
|
||||
proc login(self: LoginView, password: string): string {.slot.} =
|
||||
var currentAccountId = 0
|
||||
|
|
|
@ -57,7 +57,7 @@ QtObject:
|
|||
let assetRole = role.AccountRoles
|
||||
case assetRole:
|
||||
of AccountRoles.Username: result = newQVariant(asset.name)
|
||||
of AccountRoles.Identicon: result = newQVariant(asset.photoPath)
|
||||
of AccountRoles.Identicon: result = newQVariant(asset.identicon)
|
||||
of AccountRoles.Address: result = newQVariant(asset.keyUid)
|
||||
|
||||
method roleNames(self: OnboardingView): Table[int, string] =
|
||||
|
|
|
@ -28,11 +28,29 @@ QtObject:
|
|||
read = username
|
||||
notify = accountChanged
|
||||
|
||||
proc identicon*(self: AccountInfoView): string {.slot.} = result = ?.self.account.photoPath
|
||||
proc identicon*(self: AccountInfoView): string {.slot.} = result = ?.self.account.identicon
|
||||
QtProperty[string] identicon:
|
||||
read = identicon
|
||||
notify = accountChanged
|
||||
|
||||
proc thumbnailImage*(self: AccountInfoView): string {.slot.} =
|
||||
if (not self.account.identityImage.isNil):
|
||||
result = self.account.identityImage.thumbnail
|
||||
else:
|
||||
result = ?.self.account.identicon
|
||||
QtProperty[string] thumbnailImage:
|
||||
read = thumbnailImage
|
||||
notify = identityImageChanged
|
||||
|
||||
proc largeImage*(self: AccountInfoView): string {.slot.} =
|
||||
if (not self.account.identityImage.isNil):
|
||||
result = self.account.identityImage.large
|
||||
else:
|
||||
result = ?.self.account.identicon
|
||||
QtProperty[string] largeImage:
|
||||
read = largeImage
|
||||
notify = identityImageChanged
|
||||
|
||||
proc address*(self: AccountInfoView): string {.slot.} = result = ?.self.account.address
|
||||
QtProperty[string] address:
|
||||
read = address
|
||||
|
|
|
@ -43,6 +43,11 @@ proc init*(self: ProfileController, account: Account) =
|
|||
profile.id = pubKey
|
||||
profile.address = account.keyUid
|
||||
|
||||
let identityImage = self.status.profile.getIdentityImage(profile.address)
|
||||
|
||||
if (identityImage.thumbnail != ""):
|
||||
profile.identityImage = identityImage
|
||||
|
||||
self.view.devices.addDevices(status_devices.getAllDevices())
|
||||
self.view.devices.setDeviceSetup(status_devices.isDeviceSetup())
|
||||
self.view.setNewProfile(profile)
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import NimQml, sequtils, strutils, sugar, os, json
|
||||
import NimQml, sequtils, strutils, sugar, os, json, chronicles
|
||||
import views/[mailservers_list, ens_manager, contacts, devices, mailservers, mnemonic, network, fleets, profile_info, device_list, dapp_list]
|
||||
import ../chat/views/channels_list
|
||||
import ../../status/profile/profile
|
||||
|
@ -13,6 +13,7 @@ import ../../status/threads
|
|||
import ../../status/libstatus/types
|
||||
import ../../status/libstatus/accounts/constants as accountConstants
|
||||
import qrcode/qrcode
|
||||
import ../utils/image_utils
|
||||
|
||||
QtObject:
|
||||
type ProfileView* = ref object of QObject
|
||||
|
@ -209,3 +210,24 @@ QtObject:
|
|||
|
||||
QtProperty[QVariant] network:
|
||||
read = getNetwork
|
||||
|
||||
proc uploadNewProfilePic*(self: ProfileView, imageUrl: string, aX: int, aY: int, bX: int, bY: int): string {.slot.} =
|
||||
var image = image_utils.formatImagePath(imageUrl)
|
||||
# FIXME the function to get the file size is messed up
|
||||
# var size = image_getFileSize(image)
|
||||
# TODO find a way to i18n this (maybe send just a code and then QML sets the right string)
|
||||
# return "Max file size is 20MB"
|
||||
|
||||
try:
|
||||
# TODO add crop tool for the image
|
||||
let identityImage = self.status.profile.storeIdentityImage(self.profile.address, image, aX, aY, bX, bY)
|
||||
self.profile.setIdentityImage(identityImage)
|
||||
result = ""
|
||||
except Exception as e:
|
||||
error "Error storing identity image", msg=e.msg
|
||||
result = "Error storing identity image: " & e.msg
|
||||
|
||||
proc deleteProfilePic*(self: ProfileView): string {.slot.} =
|
||||
result = self.status.profile.deleteIdentityImage(self.profile.address)
|
||||
if (result == ""):
|
||||
self.profile.removeIdentityImage()
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import NimQml
|
||||
import NimQml, chronicles
|
||||
import Tables
|
||||
import ../../../status/profile/profile
|
||||
from ../../../status/ens import nil
|
||||
|
@ -14,6 +14,8 @@ type
|
|||
Alias = UserRole + 7
|
||||
EnsVerified = UserRole + 8
|
||||
LocalNickname = UserRole + 9
|
||||
ThumbnailImage = UserRole + 10
|
||||
LargeImage = UserRole + 11
|
||||
|
||||
QtObject:
|
||||
type ContactList* = ref object of QAbstractListModel
|
||||
|
@ -42,6 +44,14 @@ QtObject:
|
|||
return ens.userNameOrAlias(contact)
|
||||
return defaultValue
|
||||
|
||||
proc getContactIndexByPubkey(self: ContactList, pubkey: string): int {.slot.} =
|
||||
var i = 0
|
||||
for contact in self.contacts:
|
||||
if (contact.id == pubkey):
|
||||
return i
|
||||
i = i + 1
|
||||
return -1
|
||||
|
||||
proc rowData(self: ContactList, index: int, column: string): string {.slot.} =
|
||||
let contact = self.contacts[index]
|
||||
case column:
|
||||
|
@ -54,6 +64,8 @@ QtObject:
|
|||
of "alias": result = contact.alias
|
||||
of "ensVerified": result = $contact.ensVerified
|
||||
of "localNickname": result = $contact.localNickname
|
||||
of "thumbnailImage": result = $contact.identityImage.thumbnail
|
||||
of "largeImage": result = $contact.identityImage.large
|
||||
|
||||
method data(self: ContactList, index: QModelIndex, role: int): QVariant =
|
||||
if not index.isValid:
|
||||
|
@ -71,6 +83,8 @@ QtObject:
|
|||
of ContactRoles.Alias: result = newQVariant(contact.alias)
|
||||
of ContactRoles.EnsVerified: result = newQVariant(contact.ensVerified)
|
||||
of ContactRoles.LocalNickname: result = newQVariant(contact.localNickname)
|
||||
of ContactRoles.ThumbnailImage: result = newQVariant(contact.identityImage.thumbnail)
|
||||
of ContactRoles.LargeImage: result = newQVariant(contact.identityImage.large)
|
||||
|
||||
method roleNames(self: ContactList): Table[int, string] =
|
||||
{
|
||||
|
@ -82,7 +96,9 @@ QtObject:
|
|||
ContactRoles.IsBlocked.int:"isBlocked",
|
||||
ContactRoles.Alias.int:"alias",
|
||||
ContactRoles.LocalNickname.int:"localNickname",
|
||||
ContactRoles.EnsVerified.int:"ensVerified"
|
||||
ContactRoles.EnsVerified.int:"ensVerified",
|
||||
ContactRoles.ThumbnailImage.int:"thumbnailImage",
|
||||
ContactRoles.LargeImage.int:"largeImage"
|
||||
}.toTable
|
||||
|
||||
proc addContactToList*(self: ContactList, contact: Profile) =
|
||||
|
@ -95,6 +111,8 @@ QtObject:
|
|||
if(c.isContact()): return true
|
||||
return false
|
||||
|
||||
proc contactChanged*(self: ContactList, pubkey: string) {.signal.}
|
||||
|
||||
proc updateContact*(self: ContactList, contact: Profile) =
|
||||
var found = false
|
||||
let topLeft = self.createIndex(0, 0, nil)
|
||||
|
@ -104,11 +122,13 @@ QtObject:
|
|||
found = true
|
||||
c.ensName = contact.ensName
|
||||
c.ensVerified = contact.ensVerified
|
||||
c.identityImage = contact.identityImage
|
||||
|
||||
if not found:
|
||||
self.addContactToList(contact)
|
||||
else:
|
||||
self.dataChanged(topLeft, bottomRight, @[ContactRoles.Name.int])
|
||||
self.contactChanged(contact.id)
|
||||
|
||||
proc setNewData*(self: ContactList, contactList: seq[Profile]) =
|
||||
self.beginResetModel()
|
||||
|
|
|
@ -1,12 +1,15 @@
|
|||
import NimQml
|
||||
import chronicles
|
||||
import ../../../status/profile/profile
|
||||
import ../../../status/libstatus/types
|
||||
import std/wrapnils
|
||||
|
||||
QtObject:
|
||||
type ProfileInfoView* = ref object of QObject
|
||||
username*: string
|
||||
identicon*: string
|
||||
address*: string
|
||||
identityImage*: IdentityImage
|
||||
pubKey*: string
|
||||
appearance*: int
|
||||
ensVerified*: bool
|
||||
|
@ -24,11 +27,14 @@ QtObject:
|
|||
result.username = ""
|
||||
result.identicon = ""
|
||||
result.appearance = 0
|
||||
result.identityImage = IdentityImage()
|
||||
result.ensVerified = false
|
||||
result.setup
|
||||
|
||||
proc profileChanged*(self: ProfileInfoView) {.signal.}
|
||||
|
||||
proc identityImageChanged*(self: ProfileInfoView) {.signal.}
|
||||
|
||||
proc setProfile*(self: ProfileInfoView, profile: Profile) =
|
||||
self.username = profile.username
|
||||
self.identicon = profile.identicon
|
||||
|
@ -36,8 +42,17 @@ QtObject:
|
|||
self.pubKey = profile.id
|
||||
self.address = profile.address
|
||||
self.ensVerified = profile.ensVerified
|
||||
self.identityImage = profile.identityImage
|
||||
self.profileChanged()
|
||||
|
||||
proc setIdentityImage*(self: ProfileInfoView, identityImage: IdentityImage) =
|
||||
self.identityImage = identityImage
|
||||
self.identityImageChanged()
|
||||
|
||||
proc removeIdentityImage*(self: ProfileInfoView) =
|
||||
self.identityImage = IdentityImage()
|
||||
self.identityImageChanged()
|
||||
|
||||
proc username*(self: ProfileInfoView): string {.slot.} = result = self.username
|
||||
QtProperty[string] username:
|
||||
read = username
|
||||
|
@ -59,6 +74,30 @@ QtObject:
|
|||
read = identicon
|
||||
notify = profileChanged
|
||||
|
||||
proc thumbnailImage*(self: ProfileInfoView): string {.slot.} =
|
||||
if (?.self.identityImage.thumbnail != ""):
|
||||
result = self.identityImage.thumbnail
|
||||
else:
|
||||
result = self.identicon
|
||||
QtProperty[string] thumbnailImage:
|
||||
read = thumbnailImage
|
||||
notify = identityImageChanged
|
||||
|
||||
proc largeImage*(self: ProfileInfoView): string {.slot.} =
|
||||
if (?.self.identityImage.large != ""):
|
||||
result = self.identityImage.large
|
||||
else:
|
||||
result = self.identicon
|
||||
QtProperty[string] largeImage:
|
||||
read = largeImage
|
||||
notify = identityImageChanged
|
||||
|
||||
proc hasIdentityImage*(self: ProfileInfoView): bool {.slot.} =
|
||||
result = (?.self.identityImage.thumbnail != "")
|
||||
QtProperty[bool] hasIdentityImage:
|
||||
read = hasIdentityImage
|
||||
notify = identityImageChanged
|
||||
|
||||
proc pubKey*(self: ProfileInfoView): string {.slot.} = self.pubKey
|
||||
|
||||
QtProperty[string] pubKey:
|
||||
|
|
|
@ -0,0 +1,8 @@
|
|||
import strutils, os
|
||||
|
||||
proc formatImagePath*(imagePath: string): string =
|
||||
var image: string = replace(imagePath, "file://", "")
|
||||
if defined(windows):
|
||||
# Windows doesn't work with paths starting with a slash
|
||||
image.removePrefix('/')
|
||||
return image
|
|
@ -19,7 +19,7 @@ proc generateAddresses*(self: AccountModel): seq[GeneratedAccount] =
|
|||
var accounts = status_accounts.generateAddresses()
|
||||
for account in accounts.mitems:
|
||||
account.name = status_accounts.generateAlias(account.derived.whisper.publicKey)
|
||||
account.photoPath = status_accounts.generateIdenticon(account.derived.whisper.publicKey)
|
||||
account.identicon = status_accounts.generateIdenticon(account.derived.whisper.publicKey)
|
||||
self.generatedAddresses.add(account)
|
||||
result = self.generatedAddresses
|
||||
|
||||
|
@ -41,7 +41,7 @@ proc importMnemonic*(self: AccountModel, mnemonic: string): GeneratedAccount =
|
|||
let importedAccount = status_accounts.multiAccountImportMnemonic(mnemonic)
|
||||
importedAccount.derived = status_accounts.deriveAccounts(importedAccount.id)
|
||||
importedAccount.name = status_accounts.generateAlias(importedAccount.derived.whisper.publicKey)
|
||||
importedAccount.photoPath = status_accounts.generateIdenticon(importedAccount.derived.whisper.publicKey)
|
||||
importedAccount.identicon = status_accounts.generateIdenticon(importedAccount.derived.whisper.publicKey)
|
||||
result = importedAccount
|
||||
|
||||
proc reset*(self: AccountModel) =
|
||||
|
|
|
@ -67,9 +67,34 @@ proc initNode*() =
|
|||
|
||||
discard $nim_status.initKeystore(KEYSTOREDIR)
|
||||
|
||||
proc parseIdentityImage*(images: JsonNode): IdentityImage =
|
||||
result = IdentityImage()
|
||||
if (images.kind != JNull):
|
||||
for image in images:
|
||||
if (image["type"].getStr == "thumbnail"):
|
||||
# TODO check if this can be url or if it's always uri
|
||||
result.thumbnail = image["uri"].getStr
|
||||
elif (image["type"].getStr == "large"):
|
||||
result.large = image["uri"].getStr
|
||||
|
||||
proc openAccounts*(): seq[NodeAccount] =
|
||||
let strNodeAccounts = $nim_status.openAccounts(DATADIR)
|
||||
result = Json.decode(strNodeAccounts, seq[NodeAccount])
|
||||
let strNodeAccounts = nim_status.openAccounts(DATADIR).parseJson
|
||||
# FIXME fix serialization
|
||||
result = @[]
|
||||
if (strNodeAccounts.kind != JNull):
|
||||
for account in strNodeAccounts:
|
||||
let nodeAccount = NodeAccount(
|
||||
name: account["name"].getStr,
|
||||
timestamp: account["timestamp"].getInt,
|
||||
keyUid: account["key-uid"].getStr,
|
||||
identicon: account["identicon"].getStr,
|
||||
keycardPairing: account["keycard-pairing"].getStr
|
||||
)
|
||||
if (account{"images"}.kind != JNull):
|
||||
nodeAccount.identityImage = parseIdentityImage(account["images"])
|
||||
|
||||
result.add(nodeAccount)
|
||||
|
||||
|
||||
proc saveAccountAndLogin*(
|
||||
account: GeneratedAccount,
|
||||
|
@ -91,7 +116,7 @@ proc saveAccountAndLogin*(
|
|||
"public-key": account.derived.whisper.publicKey,
|
||||
"address": account.derived.whisper.address,
|
||||
"name": account.name,
|
||||
"photo-path": account.photoPath,
|
||||
"identicon": account.identicon,
|
||||
"path": constants.PATH_WHISPER,
|
||||
"chat": true
|
||||
}
|
||||
|
@ -127,7 +152,7 @@ proc getAccountData*(account: GeneratedAccount): JsonNode =
|
|||
result = %* {
|
||||
"name": account.name,
|
||||
"address": account.address,
|
||||
"photo-path": account.photoPath,
|
||||
"identicon": account.identicon,
|
||||
"key-uid": account.keyUid,
|
||||
"keycard-pairing": nil
|
||||
}
|
||||
|
@ -148,7 +173,7 @@ proc getAccountSettings*(account: GeneratedAccount, defaultNetworks: JsonNode, i
|
|||
"latest-derived-path": 0,
|
||||
"networks/networks": defaultNetworks,
|
||||
"currency": "usd",
|
||||
"photo-path": account.photoPath,
|
||||
"identicon": account.identicon,
|
||||
"waku-enabled": true,
|
||||
"wallet/visible-tokens": {
|
||||
"mainnet": ["SNT"]
|
||||
|
@ -336,3 +361,22 @@ proc deriveAccounts*(accountId: string): MultiAccounts =
|
|||
|
||||
proc logout*(): StatusGoError =
|
||||
result = Json.decode($nim_status.logout(), StatusGoError)
|
||||
|
||||
proc storeIdentityImage*(keyUID: string, imagePath: string, aX, aY, bX, bY: int): IdentityImage =
|
||||
let response = callPrivateRPC("multiaccounts_storeIdentityImage", %* [keyUID, imagePath, aX, aY, bX, bY]).parseJson
|
||||
result = parseIdentityImage(response{"result"})
|
||||
|
||||
proc getIdentityImage*(keyUID: string): IdentityImage =
|
||||
try:
|
||||
let response = callPrivateRPC("multiaccounts_getIdentityImages", %* [keyUID]).parseJson
|
||||
result = parseIdentityImage(response{"result"})
|
||||
except Exception as e:
|
||||
error "Error getting identity image", msg=e.msg
|
||||
|
||||
proc deleteIdentityImage*(keyUID: string): string =
|
||||
try:
|
||||
let response = callPrivateRPC("multiaccounts_deleteIdentityImage", %* [keyUID]).parseJson
|
||||
result = ""
|
||||
except Exception as e:
|
||||
error "Error getting identity image", msg=e.msg
|
||||
result = e.msg
|
||||
|
|
|
@ -15,6 +15,7 @@ proc getOurInstallations*(useCached: bool = true): JsonNode =
|
|||
result = installations
|
||||
|
||||
proc syncDevices*(preferredName: string): string =
|
||||
# TODO change this to identicon when status-go is updated
|
||||
let photoPath = ""
|
||||
result = callPrivateRPC("syncDevices".prefix, %* [preferredName, photoPath])
|
||||
|
||||
|
|
|
@ -42,12 +42,17 @@ type MultiAccounts* = object
|
|||
defaultWallet* {.serializedFieldName(PATH_DEFAULT_WALLET).}: DerivedAccount
|
||||
eip1581* {.serializedFieldName(PATH_EIP_1581).}: DerivedAccount
|
||||
|
||||
type
|
||||
IdentityImage* = ref object
|
||||
thumbnail*: string
|
||||
large*: string
|
||||
|
||||
type
|
||||
Account* = ref object of RootObj
|
||||
name*: string
|
||||
keyUid* {.serializedFieldName("key-uid").}: string
|
||||
photoPath* {.serializedFieldName("photo-path").}: string
|
||||
identityImage*: IdentityImage
|
||||
identicon*: string
|
||||
|
||||
type
|
||||
NodeAccount* = ref object of Account
|
||||
|
@ -66,7 +71,8 @@ type
|
|||
# serializedFieldName pragma would need to be different
|
||||
name*: string
|
||||
keyUid*: string
|
||||
photoPath*: string
|
||||
identicon*: string
|
||||
identityImage*: IdentityImage
|
||||
|
||||
type RpcError* = ref object
|
||||
code*: int
|
||||
|
@ -88,10 +94,10 @@ type
|
|||
error*: RpcError
|
||||
|
||||
proc toAccount*(account: GeneratedAccount): Account =
|
||||
result = Account(name: account.name, photoPath: account.photoPath, keyUid: account.address)
|
||||
result = Account(name: account.name, identityImage: account.identityImage, identicon: account.identicon, keyUid: account.address)
|
||||
|
||||
proc toAccount*(account: NodeAccount): Account =
|
||||
result = Account(name: account.name, photoPath: account.photoPath, keyUid: account.keyUid)
|
||||
result = Account(name: account.name, identityImage: account.identityImage, identicon: account.identicon, keyUid: account.keyUid)
|
||||
|
||||
type AccountArgs* = ref object of Args
|
||||
account*: Account
|
||||
|
|
|
@ -17,3 +17,12 @@ proc logout*(self: ProfileModel) =
|
|||
|
||||
proc getLinkPreviewWhitelist*(self: ProfileModel): JsonNode =
|
||||
result = status_settings.getLinkPreviewWhitelist()
|
||||
|
||||
proc storeIdentityImage*(self: ProfileModel, keyUID: string, imagePath: string, aX, aY, bX, bY: int): IdentityImage =
|
||||
result = status_accounts.storeIdentityImage(keyUID, imagePath, aX, aY, bX, bY)
|
||||
|
||||
proc getIdentityImage*(self: ProfileModel, keyUID: string): IdentityImage =
|
||||
result = status_accounts.getIdentityImage(keyUID)
|
||||
|
||||
proc deleteIdentityImage*(self: ProfileModel, keyUID: string): string =
|
||||
result = status_accounts.deleteIdentityImage(keyUID)
|
||||
|
|
|
@ -4,6 +4,7 @@ import ../libstatus/types
|
|||
type Profile* = ref object
|
||||
id*, alias*, username*, identicon*, address*, ensName*, localNickname*: string
|
||||
ensVerified*: bool
|
||||
identityImage*: IdentityImage
|
||||
ensVerifiedAt*, ensVerificationRetries*, appearance*: int
|
||||
systemTags*: seq[string]
|
||||
|
||||
|
@ -17,7 +18,7 @@ proc toProfileModel*(account: Account): Profile =
|
|||
result = Profile(
|
||||
id: "",
|
||||
username: account.name,
|
||||
identicon: account.photoPath,
|
||||
identicon: account.identicon,
|
||||
alias: account.name,
|
||||
ensName: "",
|
||||
ensVerified: false,
|
||||
|
@ -36,6 +37,7 @@ proc toProfileModel*(profile: JsonNode): Profile =
|
|||
id: profile["id"].str,
|
||||
username: profile["alias"].str,
|
||||
identicon: profile["identicon"].str,
|
||||
identityImage: IdentityImage(),
|
||||
address: profile["id"].str,
|
||||
alias: profile["alias"].str,
|
||||
ensName: "",
|
||||
|
@ -51,3 +53,8 @@ proc toProfileModel*(profile: JsonNode): Profile =
|
|||
|
||||
if profile.hasKey("localNickname"):
|
||||
result.localNickname = profile["localNickname"].str
|
||||
if profile.hasKey("images"):
|
||||
if profile["images"].hasKey("thumbnail"):
|
||||
result.identityImage.thumbnail = profile["images"]["thumbnail"]["uri"].str
|
||||
if profile["images"].hasKey("large"):
|
||||
result.identityImage.large = profile["images"]["large"]["uri"].str
|
||||
|
|
|
@ -41,7 +41,7 @@ Item {
|
|||
onLinkActivated: function (linkClicked) {
|
||||
switch (linkClicked) {
|
||||
case "shareKey":
|
||||
openProfilePopup(profileModel.profile.username, profileModel.profile.pubKey, profileModel.profile.identicon);
|
||||
openProfilePopup(profileModel.profile.username, profileModel.profile.pubKey, profileModel.profile.thumbnailImage);
|
||||
break;
|
||||
case "invite": inviteFriendsPopup.open(); break;
|
||||
default: //no idea what was clicked
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
import QtQuick 2.3
|
||||
import QtQuick 2.13
|
||||
import "../../../../shared"
|
||||
import "../../../../imports"
|
||||
import "./MessageComponents"
|
||||
|
@ -25,6 +25,7 @@ Item {
|
|||
property bool timeout: false
|
||||
property string linkUrls: ""
|
||||
property string imageUrls: ""
|
||||
property bool placeholderMessage: false
|
||||
|
||||
property string authorCurrentMsg: "authorCurrentMsg"
|
||||
property string authorPrevMsg: "authorPrevMsg"
|
||||
|
@ -48,6 +49,25 @@ Item {
|
|||
|
||||
property var imageClick: function () {}
|
||||
property var scrollToBottom: function () {}
|
||||
property string userPubKey: {
|
||||
if (contentType === Constants.chatIdentifier) {
|
||||
return chatId
|
||||
}
|
||||
return fromAuthor
|
||||
}
|
||||
property bool useLargeImage: contentType === Constants.chatIdentifier
|
||||
|
||||
property string profileImageSource: !placeholderMessage && chatView.getProfileImage(userPubKey, isCurrentUser, useLargeImage) || ""
|
||||
|
||||
Connections {
|
||||
enabled: !placeholderMessage
|
||||
target: profileModel.contacts.list
|
||||
onContactChanged: {
|
||||
if (pubkey === fromAuthor) {
|
||||
profileImageSource = chatView.getProfileImage(userPubKey, isCurrentUser, useLargeImage)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
id: root
|
||||
width: parent.width
|
||||
|
@ -82,7 +102,7 @@ Item {
|
|||
messageContextMenu.isProfile = !!isProfileClick
|
||||
messageContextMenu.isSticker = isSticker
|
||||
messageContextMenu.emojiOnly = emojiOnly
|
||||
messageContextMenu.show(userName, fromAuthor, identicon, "", nickname)
|
||||
messageContextMenu.show(userName, fromAuthor, root.profileImageSource || identicon, "", nickname)
|
||||
// Position the center of the menu where the mouse is
|
||||
messageContextMenu.x = messageContextMenu.x - messageContextMenu.width / 2
|
||||
}
|
||||
|
@ -165,6 +185,7 @@ Item {
|
|||
id: channelIdentifierComponent
|
||||
ChannelIdentifier {
|
||||
authorCurrentMsg: root.authorCurrentMsg
|
||||
profileImage: profileImageSource
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -6,8 +6,10 @@ Item {
|
|||
property string authorCurrentMsg: "authorCurrentMsg"
|
||||
property int verticalMargin: 50
|
||||
|
||||
property string profileImage
|
||||
|
||||
id: channelIdentifier
|
||||
visible: authorCurrentMsg == ""
|
||||
visible: authorCurrentMsg === ""
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
anchors.top: parent.top
|
||||
anchors.topMargin: this.visible ? verticalMargin : 0
|
||||
|
@ -28,15 +30,13 @@ Item {
|
|||
return chatsModel.activeChannel.color
|
||||
}
|
||||
|
||||
Image {
|
||||
RoundedImage {
|
||||
visible: chatsModel.activeChannel.chatType === Constants.chatTypeOneToOne
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
width: 120
|
||||
height: 120
|
||||
fillMode: Image.PreserveAspectFit
|
||||
source: chatsModel.activeChannel.identicon
|
||||
mipmap: true
|
||||
source: channelIdentifier.profileImage || chatsModel.activeChannel.identicon
|
||||
smooth: false
|
||||
antialiasing: true
|
||||
}
|
||||
|
@ -71,7 +71,7 @@ Item {
|
|||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
}
|
||||
|
||||
Rectangle {
|
||||
Item {
|
||||
id: channelDescription
|
||||
visible: descText.visible
|
||||
width: visible ? 330 : 0
|
||||
|
@ -79,7 +79,6 @@ Item {
|
|||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
anchors.top: channelName.bottom
|
||||
anchors.topMargin: 16
|
||||
color: Style.current.transparent
|
||||
|
||||
StyledText {
|
||||
id: descText
|
||||
|
|
|
@ -50,7 +50,8 @@ Item {
|
|||
|
||||
if (link.startsWith('//')) {
|
||||
let pk = link.replace("//", "");
|
||||
openProfilePopup(chatsModel.userNameOrAlias(pk), pk, utilsModel.generateIdenticon(pk))
|
||||
const userProfileImage = chatView.getProfileImage(pk)
|
||||
openProfilePopup(chatsModel.userNameOrAlias(pk), pk, userProfileImage || utilsModel.generateIdenticon(pk))
|
||||
return;
|
||||
}
|
||||
|
||||
|
|
|
@ -7,21 +7,24 @@ Loader {
|
|||
height: active ? item.height : 0
|
||||
|
||||
sourceComponent: Component {
|
||||
Rectangle {
|
||||
Item {
|
||||
id: chatImage
|
||||
width: identiconImage.width
|
||||
height: identiconImage.height
|
||||
border.width: 1
|
||||
border.color: Style.current.border
|
||||
radius: 50
|
||||
|
||||
Image {
|
||||
RoundedImage {
|
||||
id: identiconImage
|
||||
width: 36
|
||||
height: 36
|
||||
fillMode: Image.PreserveAspectFit
|
||||
source: !isCurrentUser ? identicon : profileModel.profile.identicon
|
||||
mipmap: true
|
||||
border.width: 1
|
||||
border.color: Style.current.border
|
||||
source: {
|
||||
if (profileImageSource) {
|
||||
return profileImageSource
|
||||
}
|
||||
|
||||
return !isCurrentUser ? identicon : profileModel.profile.identicon
|
||||
}
|
||||
smooth: false
|
||||
antialiasing: true
|
||||
|
||||
|
|
|
@ -40,7 +40,8 @@ Rectangle {
|
|||
groupInfoPopup.open()
|
||||
break;
|
||||
case Constants.chatTypeOneToOne:
|
||||
openProfilePopup(chatsModel.activeChannel.name, chatsModel.activeChannel.id, chatsModel.activeChannel.identicon)
|
||||
const profileImage = chatView.getProfileImage(chatsModel.activeChannel.id)
|
||||
openProfilePopup(chatsModel.activeChannel.name, chatsModel.activeChannel.id, profileImage || chatsModel.activeChannel.identicon)
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -37,6 +37,19 @@ SplitView {
|
|||
chatGroupsListViewCount: contactsColumn.chatGroupsListViewCount
|
||||
}
|
||||
|
||||
function getProfileImage(pubkey, isCurrentUser, useLargeImage) {
|
||||
if (isCurrentUser || (isCurrentUser === undefined && pubkey === profileModel.profile.pubKey)) {
|
||||
return profileModel.profile.thumbnailImage
|
||||
}
|
||||
|
||||
const index = profileModel.contacts.list.getContactIndexByPubkey(pubkey)
|
||||
if (index === -1) {
|
||||
return
|
||||
}
|
||||
|
||||
return profileModel.contacts.list.rowData(index, useLargeImage ? "largeImage" : "thumbnailImage")
|
||||
}
|
||||
|
||||
function openProfilePopup(userNameParam, fromAuthorParam, identiconParam, textParam, nicknameParam, parentPopup){
|
||||
var popup = profilePopupComponent.createObject(chatView);
|
||||
if(parentPopup){
|
||||
|
|
|
@ -20,6 +20,18 @@ Rectangle {
|
|||
property bool muted: false
|
||||
property bool hovered: false
|
||||
|
||||
property string profileImage: chatType === Constants.chatTypeOneToOne ? chatView.getProfileImage(chatId) || "" : ""
|
||||
|
||||
Connections {
|
||||
enabled: chatType === Constants.chatTypeOneToOne
|
||||
target: profileModel.contacts.list
|
||||
onContactChanged: {
|
||||
if (pubkey === wrapper.chatId) {
|
||||
wrapper.profileImage = chatView.getProfileImage(wrapper.chatId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
id: wrapper
|
||||
color: {
|
||||
if (ListView.isCurrentItem || wrapper.hovered) {
|
||||
|
@ -42,7 +54,7 @@ Rectangle {
|
|||
width: !isCompact ? 40 : 20
|
||||
chatName: wrapper.name
|
||||
chatType: wrapper.chatType
|
||||
identicon: wrapper.identicon
|
||||
identicon: wrapper.profileImage || wrapper.identicon
|
||||
anchors.left: parent.left
|
||||
anchors.leftMargin: !isCompact ? Style.current.padding : Style.current.smallPadding
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
|
|
|
@ -156,7 +156,8 @@ ScrollView {
|
|||
chatsModel.setActiveChannelByIndex(channelContextMenu.channelIndex)
|
||||
chatGroupsListView.currentIndex = channelContextMenu.channelIndex
|
||||
if (channelContextMenu.chatType === Constants.chatTypeOneToOne) {
|
||||
return openProfilePopup(channelContextMenu.chatName, channelContextMenu.chatId, channelContextMenu.chatIdenticon)
|
||||
const userProfileImage = chatView.getProfileImage(channelContextMenu.chatId)
|
||||
return openProfilePopup(channelContextMenu.chatName, channelContextMenu.chatId, userProfileImage || channelContextMenu.chatIdenticon)
|
||||
}
|
||||
if (channelContextMenu.chatType === Constants.chatTypePrivateGroupChat) {
|
||||
return groupInfoPopup.open()
|
||||
|
|
|
@ -37,6 +37,7 @@ ModalPopup {
|
|||
pubKey: profileModel.contacts.list.rowData(i, "pubKey"),
|
||||
address: profileModel.contacts.list.rowData(i, "address"),
|
||||
identicon: profileModel.contacts.list.rowData(i, "identicon"),
|
||||
thumbnailImage: profileModel.contacts.list.rowData(i, "thumbnailImage"),
|
||||
isUser: false,
|
||||
isContact: profileModel.contacts.list.rowData(i, "isContact") !== "false"
|
||||
});
|
||||
|
@ -47,6 +48,7 @@ ModalPopup {
|
|||
pubKey: profileModel.profile.pubKey,
|
||||
address: "",
|
||||
identicon: profileModel.profile.identicon,
|
||||
thumbnailImage: profileModel.profile.thumbnailImage,
|
||||
isUser: true
|
||||
});
|
||||
noContactsRect.visible = !profileModel.contacts.list.hasAddedContacts();
|
||||
|
@ -186,7 +188,7 @@ ModalPopup {
|
|||
name: !model.name.endsWith(".eth") && !!model.localNickname ?
|
||||
model.localNickname : Utils.removeStatusEns(model.name)
|
||||
address: model.address
|
||||
identicon: model.identicon
|
||||
identicon: model.thumbnailImage || model.identicon
|
||||
onItemChecked: function(pubKey, itemChecked){
|
||||
var idx = pubKeys.indexOf(pubKey)
|
||||
if(itemChecked){
|
||||
|
|
|
@ -30,6 +30,7 @@ ModalPopup {
|
|||
pubKey: profileModel.contacts.list.rowData(i, "pubKey"),
|
||||
address: profileModel.contacts.list.rowData(i, "address"),
|
||||
identicon: profileModel.contacts.list.rowData(i, "identicon"),
|
||||
thumbnailImage: profileModel.contacts.list.rowData(i, "thumbnailImage"),
|
||||
isUser: false
|
||||
});
|
||||
}
|
||||
|
@ -211,7 +212,7 @@ ModalPopup {
|
|||
name: model.name.endsWith(".eth") || !model.localNickname ?
|
||||
Utils.removeStatusEns(model.name) : model.localNickname
|
||||
address: model.address
|
||||
identicon: model.identicon
|
||||
identicon: model.thumbnailImage || model.identicon
|
||||
onItemChecked: function(pubKey, itemChecked){
|
||||
var idx = pubKeys.indexOf(pubKey)
|
||||
if(itemChecked){
|
||||
|
@ -278,7 +279,7 @@ ModalPopup {
|
|||
StatusImageIdenticon {
|
||||
id: identicon
|
||||
anchors.left: parent.left
|
||||
source: model.identicon
|
||||
source: chatView.getProfileImage(model.pubKey)|| model.identicon
|
||||
}
|
||||
|
||||
StyledText {
|
||||
|
@ -293,7 +294,10 @@ ModalPopup {
|
|||
MouseArea {
|
||||
anchors.fill: parent
|
||||
cursorShape: Qt.PointingHandCursor
|
||||
onClicked: openProfilePopup(model.userName, model.pubKey, model.identicon, '', contactRow.nickname)
|
||||
onClicked: {
|
||||
const userProfileImage = chatView.getProfileImage(model.pubKey)
|
||||
openProfilePopup(model.userName, model.pubKey, userProfileImage || model.identicon, '', contactRow.nickname)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -138,7 +138,7 @@ ModalPopup {
|
|||
isUser: false
|
||||
name: model.name
|
||||
address: model.address
|
||||
identicon: model.identicon
|
||||
identicon: model.thumbnailImage || model.identicon
|
||||
showListSelector: true
|
||||
onItemChecked: function(pubKey, itemChecked){
|
||||
chatsModel.joinChat(pubKey, Constants.chatTypeOneToOne);
|
||||
|
|
|
@ -49,22 +49,15 @@ ModalPopup {
|
|||
header: Item {
|
||||
height: children[0].height
|
||||
width: parent.width
|
||||
Rectangle {
|
||||
RoundedImage {
|
||||
id: profilePic
|
||||
width: 40
|
||||
height: 40
|
||||
radius: 30
|
||||
border.color: "#10000000"
|
||||
border.color: Style.current.border
|
||||
border.width: 1
|
||||
color: Style.current.transparent
|
||||
anchors.top: parent.top
|
||||
SVGImage {
|
||||
width: parent.width
|
||||
height: parent.height
|
||||
fillMode: Image.PreserveAspectFit
|
||||
source: identicon
|
||||
}
|
||||
}
|
||||
|
||||
StyledTextEdit {
|
||||
id: profileName
|
||||
|
|
|
@ -45,12 +45,7 @@ SplitView {
|
|||
|
||||
// This list needs to match LeftTab/constants.js
|
||||
// Would be better if we could make them match automatically
|
||||
MyProfileContainer {
|
||||
username: profileModel.profile.username
|
||||
ensName: profileModel.ens.preferredUsername
|
||||
identicon: profileModel.profile.identicon
|
||||
pubkey: profileModel.profile.pubKey
|
||||
}
|
||||
MyProfileContainer {}
|
||||
|
||||
onCurrentIndexChanged: {
|
||||
if(visibleChildren[0] === ensContainer){
|
||||
|
|
|
@ -81,7 +81,7 @@ ScrollView {
|
|||
identicon: ""
|
||||
message: qsTr("Blockchains will drop search costs, causing a kind of decomposition that allows you to have markets of entities that are horizontally segregated and vertically segregated.")
|
||||
contentType: Constants.messageType
|
||||
|
||||
placeholderMessage: true
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,150 @@
|
|||
import QtQuick 2.13
|
||||
import QtQuick.Dialogs 1.3
|
||||
import "../../../../imports"
|
||||
import "../../../../shared"
|
||||
import "../../../../shared/status"
|
||||
|
||||
ModalPopup {
|
||||
property string selectedImage // selectedImage is for us to be able to analyze it before setting it as current
|
||||
property string uploadError
|
||||
|
||||
id: popup
|
||||
|
||||
title: qsTr("Profile picture")
|
||||
|
||||
onClosed: {
|
||||
destroy()
|
||||
}
|
||||
|
||||
onSelectedImageChanged: {
|
||||
if (!selectedImage) {
|
||||
return
|
||||
}
|
||||
|
||||
cropImageModal.open()
|
||||
}
|
||||
|
||||
Item {
|
||||
anchors.fill: parent
|
||||
|
||||
RoundedImage {
|
||||
id: profilePic
|
||||
source: profileModel.profile.largeImage
|
||||
width: 160
|
||||
height: 160
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
border.width: 1
|
||||
border.color: Style.current.border
|
||||
onClicked: imageDialog.open()
|
||||
}
|
||||
|
||||
StyledText {
|
||||
visible: !!uploadError
|
||||
text: uploadError
|
||||
anchors.left: parent.left
|
||||
anchors.right: parent.right
|
||||
anchors.top: profilePic.bottom
|
||||
horizontalAlignment: Text.AlignHCenter
|
||||
font.pixelSize: 13
|
||||
wrapMode: Text.WordWrap
|
||||
anchors.topMargin: 13
|
||||
font.weight: Font.Thin
|
||||
color: Style.current.danger
|
||||
}
|
||||
|
||||
ModalPopup {
|
||||
id: cropImageModal
|
||||
width: image.width + 50
|
||||
height: image.height + 170
|
||||
title: qsTr("Crop your image (optionnal)")
|
||||
|
||||
Image {
|
||||
id: image
|
||||
width: 400
|
||||
source: popup.selectedImage
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
fillMode: Image.PreserveAspectFit
|
||||
}
|
||||
|
||||
ImageCropper {
|
||||
id: imageCropper
|
||||
x: image.x
|
||||
y: image.y
|
||||
image: image
|
||||
}
|
||||
|
||||
footer: StatusButton {
|
||||
id: doUploadBtn
|
||||
text: qsTr("Finish")
|
||||
anchors.right: parent.right
|
||||
anchors.bottom: parent.bottom
|
||||
onClicked: {
|
||||
const aXPercent = imageCropper.selectorRectangle.x / image.width
|
||||
const aYPercent = imageCropper.selectorRectangle.y / image.height
|
||||
const bXPercent = (imageCropper.selectorRectangle.x + imageCropper.selectorRectangle.width) / image.width
|
||||
const bYPercent = (imageCropper.selectorRectangle.y + imageCropper.selectorRectangle.height) / image.height
|
||||
|
||||
|
||||
const aX = Math.round(aXPercent * image.sourceSize.width)
|
||||
const aY = Math.round(aYPercent * image.sourceSize.height)
|
||||
|
||||
const bX = Math.round(bXPercent * image.sourceSize.width)
|
||||
const bY = Math.round(bYPercent * image.sourceSize.height)
|
||||
|
||||
uploadError = profileModel.uploadNewProfilePic(selectedImage, aX, aY, bX, bY)
|
||||
cropImageModal.close()
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
footer: Item {
|
||||
anchors.fill:parent
|
||||
|
||||
StatusButton {
|
||||
visible: profileModel.profile.hasIdentityImage
|
||||
type: "secondary"
|
||||
flat: true
|
||||
color: Style.current.danger
|
||||
text: qsTr("Remove")
|
||||
anchors.right: uploadBtn.left
|
||||
anchors.rightMargin: Style.current.padding
|
||||
anchors.bottom: parent.bottom
|
||||
onClicked: {
|
||||
uploadError = profileModel.deleteProfilePic()
|
||||
}
|
||||
}
|
||||
|
||||
StatusButton {
|
||||
id: uploadBtn
|
||||
text: qsTr("Upload")
|
||||
anchors.right: parent.right
|
||||
anchors.bottom: parent.bottom
|
||||
onClicked: {
|
||||
imageDialog.open()
|
||||
}
|
||||
|
||||
FileDialog {
|
||||
id: imageDialog
|
||||
//% "Please choose an image"
|
||||
title: qsTrId("please-choose-an-image")
|
||||
folder: shortcuts.pictures
|
||||
nameFilters: [
|
||||
//% "Image files (*.jpg *.jpeg *.png)"
|
||||
qsTrId("image-files----jpg---jpeg---png-")
|
||||
]
|
||||
onAccepted: {
|
||||
selectedImage = imageDialog.fileUrls[0]
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/*##^##
|
||||
Designer {
|
||||
D{i:0;autoSize:true;height:480;width:640}
|
||||
}
|
||||
##^##*/
|
|
@ -25,7 +25,7 @@ ListView {
|
|||
name: Utils.removeStatusEns(model.name)
|
||||
address: model.address
|
||||
localNickname: model.localNickname
|
||||
identicon: model.identicon
|
||||
identicon: model.thumbnailImage || model.identicon
|
||||
isContact: model.isContact
|
||||
isBlocked: model.isBlocked
|
||||
selectable: contactList.selectable
|
||||
|
|
|
@ -6,15 +6,19 @@ import "../../../../shared"
|
|||
import "../../../../shared/status"
|
||||
|
||||
Item {
|
||||
property string username: "Jotaro Kujo"
|
||||
property string identicon: ""
|
||||
property string pubkey: "0x04d8c07dd137bd1b73a6f51df148b4f77ddaa11209d36e43d8344c0a7d6db1cad6085f27cfb75dd3ae21d86ceffebe4cf8a35b9ce8d26baa19dc264efe6d8f221b"
|
||||
property string ensName: "joestar.eth"
|
||||
property string ensName: profileModel.profile.preferredUsername || ""
|
||||
property string username: profileModel.profile.username
|
||||
property string pubkey: profileModel.profile.pubKey
|
||||
|
||||
id: profileHeaderContent
|
||||
height: parent.height
|
||||
Layout.fillWidth: true
|
||||
|
||||
Component {
|
||||
id: changeProfileModalComponent
|
||||
ChangeProfilePicModal {}
|
||||
}
|
||||
|
||||
Item {
|
||||
id: profileImgNameContainer
|
||||
anchors.top: parent.top
|
||||
|
@ -26,43 +30,58 @@ Item {
|
|||
|
||||
height: this.childrenRect.height
|
||||
|
||||
Rectangle {
|
||||
Item {
|
||||
id: profileImgContainer
|
||||
width: profileImg.width
|
||||
height: profileImg.height
|
||||
|
||||
RoundedImage {
|
||||
id: profileImg
|
||||
width: identiconImage.width
|
||||
height: identiconImage.height
|
||||
width: 64
|
||||
height: 64
|
||||
border.width: 1
|
||||
border.color: Style.current.border
|
||||
radius: 50
|
||||
color: Style.current.background
|
||||
|
||||
Image {
|
||||
id: identiconImage
|
||||
width: 60
|
||||
height: 60
|
||||
fillMode: Image.PreserveAspectFit
|
||||
source: identicon
|
||||
mipmap: true
|
||||
source: profileModel.profile.thumbnailImage || ""
|
||||
smooth: false
|
||||
antialiasing: true
|
||||
}
|
||||
|
||||
RoundedIcon {
|
||||
source: "../../../img/pencil.svg"
|
||||
anchors.bottom: parent.bottom
|
||||
anchors.bottomMargin: -3
|
||||
anchors.right: parent.right
|
||||
anchors.rightMargin: -3
|
||||
width: 24
|
||||
height: 24
|
||||
border.width: 1
|
||||
border.color: Style.current.background
|
||||
}
|
||||
|
||||
MouseArea {
|
||||
anchors.fill: parent
|
||||
cursorShape: Qt.PointingHandCursor
|
||||
onClicked: {
|
||||
const popup = changeProfileModalComponent.createObject(profileHeaderContent);
|
||||
popup.open()
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
StyledText {
|
||||
id: profileName
|
||||
text: ensName !== "" ? ensName : username
|
||||
anchors.left: profileImg.right
|
||||
anchors.left: profileImgContainer.right
|
||||
anchors.leftMargin: 8
|
||||
anchors.top: profileImg.top
|
||||
anchors.topMargin: 4
|
||||
font.family: "Inter"
|
||||
font.weight: Font.Bold
|
||||
font.pixelSize: 20
|
||||
anchors.top: profileImgContainer.top
|
||||
font.weight: Font.Medium
|
||||
font.pixelSize: 15
|
||||
}
|
||||
|
||||
Address {
|
||||
id: pubkeyText
|
||||
text: ensName !== "" ? username : pubkey
|
||||
anchors.bottom: profileImg.bottom
|
||||
anchors.bottom: profileImgContainer.bottom
|
||||
anchors.left: profileName.left
|
||||
anchors.bottomMargin: 4
|
||||
width: 200
|
||||
|
@ -85,7 +104,9 @@ Item {
|
|||
|
||||
Separator {
|
||||
id: lineSeparator
|
||||
anchors.top: profileImg.bottom
|
||||
anchors.top: profileImgContainer.bottom
|
||||
anchors.topMargin: 36
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
|
|
|
@ -0,0 +1,4 @@
|
|||
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||
<path d="M2.10144 13.5959L2.79885 10.8062C2.83108 10.6773 2.87171 10.5512 2.92034 10.4285C2.9822 10.2724 3.18194 10.2423 3.30066 10.361L5.63945 12.6998C5.75817 12.8185 5.72809 13.0183 5.572 13.0801C5.44932 13.1288 5.32315 13.1694 5.19424 13.2016L2.40461 13.899C2.22152 13.9448 2.05567 13.779 2.10144 13.5959Z" fill="white"/>
|
||||
<path d="M6.32361 11.2627C6.71413 11.6532 7.3473 11.6532 7.73782 11.2627L14.2933 4.7072C14.6838 4.31668 14.6838 3.68351 14.2933 3.29299L12.7075 1.7072C12.317 1.31668 11.6838 1.31668 11.2933 1.7072L4.73782 8.26266C4.3473 8.65318 4.3473 9.28635 4.73782 9.67687L6.32361 11.2627Z" fill="white"/>
|
||||
</svg>
|
After Width: | Height: | Size: 720 B |
|
@ -182,6 +182,7 @@ DISTFILES += \
|
|||
app/AppLayouts/Profile/Sections/AppearanceContainer.qml \
|
||||
app/AppLayouts/Profile/Sections/BackupSeedModal.qml \
|
||||
app/AppLayouts/Profile/Sections/BrowserContainer.qml \
|
||||
app/AppLayouts/Profile/Sections/ChangeProfilePicModal.qml \
|
||||
app/AppLayouts/Profile/Sections/MyProfileContainer.qml \
|
||||
app/AppLayouts/Profile/Sections/SoundsContainer.qml \
|
||||
app/AppLayouts/UIComponents/UIComponents.qml \
|
||||
|
@ -364,8 +365,10 @@ DISTFILES += \
|
|||
shared/AccountSelector.qml \
|
||||
shared/AddButton.qml \
|
||||
shared/Address.qml \
|
||||
shared/CropCornerRectangle.qml \
|
||||
shared/FormGroup.qml \
|
||||
shared/IconButton.qml \
|
||||
shared/ImageCropper.qml \
|
||||
shared/Input.qml \
|
||||
shared/LabelValueRow.qml \
|
||||
shared/ModalPopup.qml \
|
||||
|
|
|
@ -35,7 +35,7 @@ Item {
|
|||
StatusImageIdenticon {
|
||||
id: userImage
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
source: loginModel.currentAccount.identicon
|
||||
source: loginModel.currentAccount.thumbnailImage
|
||||
}
|
||||
|
||||
StyledText {
|
||||
|
|
|
@ -17,7 +17,7 @@ ListView {
|
|||
delegate: AddressView {
|
||||
username: model.username
|
||||
address: model.address
|
||||
identicon: model.identicon
|
||||
identicon: model.thumbnailImage
|
||||
isSelected: function (index, address) {
|
||||
return addressesView.isSelected(index, address)
|
||||
}
|
||||
|
|
|
@ -0,0 +1,23 @@
|
|||
import QtQuick 2.13
|
||||
import "../imports"
|
||||
|
||||
Rectangle {
|
||||
id: root
|
||||
width: 25
|
||||
height: 25
|
||||
border.width: 2
|
||||
border.color: Style.current.orange
|
||||
color: Utils.setColorAlpha(Style.current.orange, 0.5)
|
||||
|
||||
Drag.active: dragArea.drag.active
|
||||
|
||||
MouseArea {
|
||||
id: dragArea
|
||||
property int oldX
|
||||
property int oldY
|
||||
|
||||
anchors.fill: parent
|
||||
drag.target: parent
|
||||
cursorShape: Qt.PointingHandCursor
|
||||
}
|
||||
}
|
|
@ -0,0 +1,107 @@
|
|||
import QtQuick 2.13
|
||||
import "."
|
||||
import "../imports"
|
||||
|
||||
Item {
|
||||
property var image
|
||||
property alias selectorRectangle: selectorRectangle
|
||||
|
||||
width: image.width
|
||||
height: image.height
|
||||
|
||||
Rectangle {
|
||||
id: selectorRectangle
|
||||
visible: false
|
||||
x: 0
|
||||
y: 0
|
||||
border.width: 2
|
||||
border.color: Style.current.orange
|
||||
color: Style.current.transparent
|
||||
|
||||
function initialSetup() {
|
||||
topLeftCorner.x = 0
|
||||
topLeftCorner.y = 0
|
||||
topRightCorner.x = image.width - topRightCorner.width
|
||||
topRightCorner.y = 0
|
||||
bottomLeftCorner.x = 0
|
||||
bottomLeftCorner.y = image.height - topRightCorner.height
|
||||
bottomRightCorner.x = image.width - topRightCorner.width
|
||||
bottomRightCorner.y = image.height - topRightCorner.height
|
||||
|
||||
selectorRectangle.width = image.width
|
||||
selectorRectangle.height = image.height
|
||||
}
|
||||
|
||||
function adjustRectangleSize() {
|
||||
selectorRectangle.width = bottomRightCorner.x + bottomRightCorner.width - topLeftCorner.x
|
||||
selectorRectangle.height = bottomRightCorner.y + bottomRightCorner.height - topLeftCorner.y
|
||||
selectorRectangle.x = topLeftCorner.x
|
||||
selectorRectangle.y = topLeftCorner.y
|
||||
}
|
||||
|
||||
Connections {
|
||||
target: image
|
||||
onStatusChanged: {
|
||||
if (image.status === Image.Ready) {
|
||||
selectorRectangle.initialSetup()
|
||||
selectorRectangle.visible = true
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Size calculations are only done on top-left and bottom-right, because the other two corners follow them
|
||||
CropCornerRectangle {
|
||||
id: topLeftCorner
|
||||
onXChanged: {
|
||||
if (x < 0) x = 0
|
||||
if (x > topRightCorner.x - width) x = topRightCorner.x - width
|
||||
|
||||
bottomLeftCorner.x = x
|
||||
selectorRectangle.adjustRectangleSize()
|
||||
}
|
||||
onYChanged: {
|
||||
if (y < 0) y = 0
|
||||
if (y > bottomRightCorner.y - height) y = bottomRightCorner.y - height
|
||||
|
||||
topRightCorner.y = y
|
||||
selectorRectangle.adjustRectangleSize()
|
||||
}
|
||||
}
|
||||
|
||||
CropCornerRectangle {
|
||||
id: topRightCorner
|
||||
onXChanged: {
|
||||
bottomRightCorner.x = x
|
||||
}
|
||||
onYChanged: {
|
||||
topLeftCorner.y = y
|
||||
}
|
||||
}
|
||||
|
||||
CropCornerRectangle {
|
||||
id: bottomLeftCorner
|
||||
onXChanged: {
|
||||
topLeftCorner.x = x
|
||||
}
|
||||
onYChanged: {
|
||||
bottomRightCorner.y = y
|
||||
}
|
||||
}
|
||||
|
||||
CropCornerRectangle {
|
||||
id: bottomRightCorner
|
||||
onXChanged: {
|
||||
if (x < topLeftCorner.x + topLeftCorner.width) x = topLeftCorner.x + topLeftCorner.width
|
||||
if (x > image.width - width) x = image.width - width
|
||||
topRightCorner.x = x
|
||||
selectorRectangle.adjustRectangleSize()
|
||||
}
|
||||
onYChanged: {
|
||||
if (y < topRightCorner.y + topRightCorner.height) y = topRightCorner.y + topRightCorner.height
|
||||
if (y > image.height - height) y = image.height - height
|
||||
bottomLeftCorner.y = y
|
||||
selectorRectangle.adjustRectangleSize()
|
||||
}
|
||||
}
|
||||
}
|
|
@ -4,6 +4,7 @@ import "../imports"
|
|||
|
||||
Rectangle {
|
||||
id: root
|
||||
property bool noHover: false
|
||||
property alias source: image.source
|
||||
property alias fillMode: image.fillMode
|
||||
signal loaded
|
||||
|
@ -138,7 +139,7 @@ Rectangle {
|
|||
}
|
||||
}
|
||||
MouseArea {
|
||||
cursorShape: Qt.PointingHandCursor
|
||||
cursorShape: noHover ? Qt.ArrowCursor : Qt.PointingHandCursor
|
||||
anchors.fill: parent
|
||||
onClicked: {
|
||||
root.clicked()
|
||||
|
|
|
@ -4,6 +4,7 @@ import QtGraphicalEffects 1.0
|
|||
Rectangle {
|
||||
id: root
|
||||
signal clicked
|
||||
property bool noHover: false
|
||||
property alias source: imgStickerPackThumb.source
|
||||
property alias fillMode: imgStickerPackThumb.fillMode
|
||||
|
||||
|
@ -25,6 +26,7 @@ Rectangle {
|
|||
|
||||
ImageLoader {
|
||||
id: imgStickerPackThumb
|
||||
noHover: root.noHover
|
||||
opacity: 1
|
||||
smooth: false
|
||||
radius: root.radius
|
||||
|
|
|
@ -14,14 +14,26 @@ Item {
|
|||
property int identiconSize: 40
|
||||
property bool isCompact: false
|
||||
|
||||
property string profileImage: chatType === Constants.chatTypeOneToOne ? chatView.getProfileImage(chatId) || "" : ""
|
||||
|
||||
height: 48
|
||||
width: nameAndInfo.width + chatIdenticon.width + Style.current.smallPadding
|
||||
|
||||
Connections {
|
||||
enabled: chatType === Constants.chatTypeOneToOne
|
||||
target: profileModel.contacts.list
|
||||
onContactChanged: {
|
||||
if (pubkey === root.chatId) {
|
||||
root.profileImage = chatView.getProfileImage(root.chatId)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
StatusIdenticon {
|
||||
id: chatIdenticon
|
||||
chatType: root.chatType
|
||||
chatName: root.chatName
|
||||
identicon: root.identicon
|
||||
identicon: root.profileImage || root.identicon
|
||||
width: root.isCompact ? 20 : root.identiconSize
|
||||
height: root.isCompact ? 20 : root.identiconSize
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
|
|
|
@ -24,7 +24,12 @@ Button {
|
|||
chatId: control.chatId
|
||||
chatName: control.chatName
|
||||
chatType: control.chatType
|
||||
identicon: control.identicon
|
||||
identicon: {
|
||||
if (control.chatType === Constants.chatTypeOneToOne) {
|
||||
return chatView.getProfileImage(control.chatId) || control.identicon
|
||||
}
|
||||
return control.identicon
|
||||
}
|
||||
identiconSize: control.identiconSize
|
||||
isCompact: control.isCompact
|
||||
}
|
||||
|
|
|
@ -2,25 +2,14 @@ import QtQuick 2.13
|
|||
import "../../imports"
|
||||
import "../../shared"
|
||||
|
||||
Rectangle {
|
||||
RoundedImage {
|
||||
id: root
|
||||
property url source:""
|
||||
noHover: true
|
||||
source:""
|
||||
width: 40
|
||||
height: 40
|
||||
color: Style.current.background
|
||||
radius: width / 2
|
||||
border.width: 1
|
||||
border.color: Style.current.borderSecondary
|
||||
|
||||
Image {
|
||||
width: parent.width
|
||||
height: parent.height
|
||||
fillMode: Image.PreserveAspectFit
|
||||
anchors.horizontalCenter: parent.horizontalCenter
|
||||
anchors.verticalCenter: parent.verticalCenter
|
||||
source: root.source
|
||||
mipmap: true
|
||||
smooth: false
|
||||
antialiasing: true
|
||||
}
|
||||
}
|
||||
|
|
|
@ -1 +1 @@
|
|||
Subproject commit a474e0e1fedd213fa9590811faca198761660857
|
||||
Subproject commit 5c34c8e3563b2d10a2f3d44ac0ee11da13f9a916
|
|
@ -1 +1 @@
|
|||
Subproject commit b1ba8cba3c75de574046071de824031cde6e580b
|
||||
Subproject commit db7042d4902dc19e5a286416ffa293e43b6249c5
|
|
@ -1 +1 @@
|
|||
Subproject commit d8307a60cfc49998a48dcb7026b3a1ca17d55985
|
||||
Subproject commit 55a08e9e4d259d77637420ec318cfbafeb8f9f6e
|
Loading…
Reference in New Issue