2023-02-15 10:27:18 +02:00
import QtQuick 2.15
2020-06-17 15:18:31 -04:00
import QtQuick . Controls 2.13
import QtQuick . Layouts 1.13
2021-02-18 15:14:31 -04:00
import QtMultimedia 5.13
2022-03-04 00:50:53 +02:00
import Qt . labs . platform 1.1
2022-09-02 17:56:14 +03:00
import QtQml . Models 2.14
2023-02-27 13:11:51 +01:00
import QtQml 2.15
2022-03-04 00:50:53 +02:00
2022-03-08 13:49:33 -05:00
import AppLayouts . Wallet 1.0
import AppLayouts . Node 1.0
import AppLayouts . Browser 1.0
import AppLayouts . Chat 1.0
2022-05-13 11:27:26 -04:00
import AppLayouts . Chat . views 1.0
2022-03-08 13:49:33 -05:00
import AppLayouts . Profile 1.0
2023-06-23 08:17:04 +02:00
import AppLayouts . Communities 1.0
2021-09-28 18:04:06 +03:00
import utils 1.0
2021-10-28 00:27:49 +03:00
import shared 1.0
2022-03-17 12:15:38 -04:00
import shared . controls 1.0
2023-06-07 13:54:31 +03:00
import shared . controls . chat . menuItems 1.0
2021-10-28 00:27:49 +03:00
import shared . panels 1.0
import shared . popups 1.0
2022-09-13 12:03:25 +02:00
import shared . popups . keycard 1.0
2021-10-28 00:27:49 +03:00
import shared . status 1.0
2023-04-03 19:13:23 +02:00
import shared . stores 1.0
2023-09-05 17:27:30 +02:00
import shared . popups . send 1.0
2020-05-12 07:24:08 +10:00
2021-06-11 14:17:23 +02:00
import StatusQ . Core . Theme 0.1
2021-10-22 01:34:35 +02:00
import StatusQ . Components 0.1
2021-06-11 14:17:23 +02:00
import StatusQ . Controls 0.1
import StatusQ . Layout 0.1
import StatusQ . Popups 0.1
2022-09-02 17:56:14 +03:00
import StatusQ . Popups . Dialog 0.1
2021-10-22 01:34:35 +02:00
import StatusQ . Core 0.1
2021-06-11 14:17:23 +02:00
2022-05-16 18:02:03 +03:00
import AppLayouts . Browser . stores 1.0 as BrowserStores
2022-03-08 13:49:33 -05:00
import AppLayouts . stores 1.0
2023-02-09 22:49:36 +01:00
import AppLayouts . Chat . stores 1.0 as ChatStores
2023-06-23 08:17:04 +02:00
import AppLayouts . Communities . stores 1.0
2022-03-04 00:50:53 +02:00
2023-04-17 10:31:32 +02:00
import mainui . activitycenter . stores 1.0
import mainui . activitycenter . popups 1.0
2022-11-28 12:32:29 +01:00
import SortFilterProxyModel 0.2
2022-10-06 18:47:55 +03:00
import "panels"
2022-07-12 10:56:47 +02:00
2021-06-17 14:41:11 -04:00
Item {
2020-09-15 15:47:13 -04:00
id: appMain
2020-05-12 07:24:08 +10:00
2021-09-15 13:40:07 +02:00
property alias appLayout: appLayout
2022-09-23 18:02:54 +03:00
property RootStore rootStore: RootStore { }
2023-02-09 22:49:36 +01:00
property var rootChatStore: ChatStores . RootStore {
contactsStore: appMain . rootStore . contactStore
2023-04-11 10:09:01 +02:00
communityTokensStore: appMain . communityTokensStore
2023-02-09 22:49:36 +01:00
emojiReactionsModel: appMain . rootStore . emojiReactionsModel
openCreateChat: createChatView . opened
2023-04-04 13:31:04 +02:00
networkConnectionStore: appMain . networkConnectionStore
2023-02-09 22:49:36 +01:00
}
2023-05-04 14:36:35 -04:00
property var createChatPropertiesStore: ChatStores . CreateChatPropertiesStore { }
2022-10-26 20:00:20 +04:00
property ActivityCenterStore activityCenterStore: ActivityCenterStore { }
2023-04-03 19:13:23 +02:00
property NetworkConnectionStore networkConnectionStore: NetworkConnectionStore { }
2023-04-11 10:09:01 +02:00
property CommunityTokensStore communityTokensStore: CommunityTokensStore { }
2023-04-28 12:35:18 +02:00
property CommunitiesStore communitiesStore: CommunitiesStore { }
2021-12-09 14:28:02 +01:00
// set from main.qml
property var sysPalette
2021-10-20 11:50:50 +02:00
2022-03-17 17:24:50 +01:00
Connections {
target: rootStore . mainModuleInst
2022-09-13 12:03:25 +02:00
2023-01-18 11:25:36 +02:00
function onDisplayUserProfile ( publicKey: string ) {
2023-02-07 15:21:32 +01:00
popups . openProfilePopup ( publicKey )
2023-01-18 11:25:36 +02:00
}
2022-09-13 12:03:25 +02:00
2023-09-04 18:06:51 +02:00
function onDisplayKeycardSharedModuleForAuthentication ( ) {
keycardPopupForAuthentication . active = true
}
function onDestroyKeycardSharedModuleForAuthentication ( ) {
keycardPopupForAuthentication . active = false
}
2023-01-18 11:25:36 +02:00
function onDisplayKeycardSharedModuleFlow ( ) {
2023-06-30 11:24:08 +02:00
keycardPopup . active = true
2022-09-13 12:03:25 +02:00
}
2023-01-18 11:25:36 +02:00
function onDestroyKeycardSharedModuleFlow ( ) {
2023-06-30 11:24:08 +02:00
keycardPopup . active = false
2022-09-13 12:03:25 +02:00
}
2022-10-18 11:06:18 +02:00
2023-01-18 11:25:36 +02:00
function onMailserverWorking ( ) {
2023-01-18 17:55:23 +04:00
mailserverConnectionBanner . hide ( )
}
2023-01-18 11:25:36 +02:00
function onMailserverNotWorking ( ) {
2023-01-18 14:36:28 +04:00
mailserverConnectionBanner . show ( )
2022-10-18 11:06:18 +02:00
}
2023-01-18 11:25:36 +02:00
function onActiveSectionChanged ( ) {
2022-10-18 11:06:18 +02:00
createChatView . opened = false
}
2023-01-12 14:06:18 -05:00
2023-01-18 11:25:36 +02:00
function onOpenActivityCenter ( ) {
2023-04-17 10:31:32 +02:00
d . openActivityCenterPopup ( )
}
2023-07-17 22:06:34 +02:00
function onShowToastAccountAdded ( name: string ) {
Global . displayToastMessage (
qsTr ( "\"%1\" successfuly added" ) . arg ( name ) ,
"" ,
2023-08-07 20:47:16 +02:00
"checkmark-circle" ,
2023-07-17 22:06:34 +02:00
false ,
Constants . ephemeralNotificationType . success ,
""
)
}
function onShowToastKeypairRenamed ( oldName: string , newName: string ) {
Global . displayToastMessage (
qsTr ( "You successfully renamed your keypair\nfrom \"%1\" to \"%2\"" ) . arg ( oldName ) . arg ( newName ) ,
"" ,
2023-08-07 20:47:16 +02:00
"checkmark-circle" ,
2023-07-17 22:06:34 +02:00
false ,
Constants . ephemeralNotificationType . success ,
""
)
}
2023-08-07 20:47:16 +02:00
function onShowNetworkEndpointUpdated ( name: string , isTest: bool ) {
Global . displayToastMessage (
isTest ? qsTr ( "Test network settings for %1 updated" ) . arg ( name ) : qsTr ( "Live network settings for %1 updated" ) . arg ( name ) ,
"" ,
"checkmark-circle" ,
false ,
Constants . ephemeralNotificationType . success ,
""
)
}
function onShowIncludeWatchOnlyAccountUpdated ( includeWatchOnly: bool ) {
Global . displayToastMessage (
includeWatchOnly ? qsTr ( "Your wallet’ s total balance will now include balances of watched addresses" ) : qsTr ( "Your wallet’ s total balance will not include balances of watched addresses" ) ,
"" ,
"checkmark-circle" ,
false ,
Constants . ephemeralNotificationType . success ,
"" )
}
2023-08-08 21:01:19 +02:00
function onShowToastKeypairRemoved ( keypairName: string ) {
Global . displayToastMessage (
2023-08-23 14:01:26 +02:00
qsTr ( "“%1” keypair and its derived accounts were successfully removed from all devices" ) . arg ( keypairName ) ,
2023-08-08 21:01:19 +02:00
"" ,
"checkmark-circle" ,
false ,
Constants . ephemeralNotificationType . success ,
""
)
}
2023-08-21 12:58:21 +02:00
function onShowToastKeypairsImported ( keypairName: string , keypairsCount: int , error: string ) {
let notification = qsTr ( "Please re-generate QR code and try importing again" )
if ( error !== "" ) {
if ( error . startsWith ( "one or more expected keystore files are not found among the sent files" ) ) {
notification = qsTr ( "Make sure you're importing the exported keypair on paired device" )
}
}
else {
notification = qsTr ( "%1 keypair successfully imported" ) . arg ( keypairName )
if ( keypairsCount > 1 ) {
notification = qsTr ( "%n keypair(s) successfully imported" , "" , keypairsCount )
}
}
Global . displayToastMessage (
notification ,
"" ,
error !== "" ? "info" : "checkmark-circle" ,
false ,
error !== "" ? Constants.ephemeralNotificationType.normal : Constants . ephemeralNotificationType . success ,
""
)
}
2023-04-17 10:31:32 +02:00
}
QtObject {
id: d
property var activityCenterPopupObj: null
function openActivityCenterPopup ( ) {
if ( ! activityCenterPopupObj ) {
activityCenterPopupObj = activityCenterPopupComponent . createObject ( appMain )
}
if ( activityCenterPopupObj . opened ) {
activityCenterPopupObj . close ( )
} else {
activityCenterPopupObj . open ( )
}
2023-01-12 14:06:18 -05:00
}
2022-03-17 17:24:50 +01:00
}
2022-10-21 15:37:39 +02:00
Popups {
2023-02-07 15:21:32 +01:00
id: popups
popupParent: appMain
2022-10-21 15:37:39 +02:00
rootStore: appMain . rootStore
2023-04-28 12:35:18 +02:00
communitiesStore: appMain . communitiesStore
2023-06-26 13:44:56 +02:00
isDevBuild: ! production
2022-10-21 15:37:39 +02:00
}
2021-12-06 23:10:54 +02:00
Connections {
2023-02-07 15:21:32 +01:00
id: globalConns
2021-12-06 23:10:54 +02:00
target: Global
2023-02-07 15:21:32 +01:00
2023-01-18 11:25:36 +02:00
function onOpenLinkInBrowser ( link: string ) {
2023-01-10 14:19:02 +01:00
changeAppSectionBySectionId ( Constants . appSection . browser )
Qt . callLater ( ( ) = > browserLayoutContainer . item . openUrlInNewTab ( link ) ) ;
2021-12-06 23:10:54 +02:00
}
2022-10-18 11:06:18 +02:00
2023-01-18 11:25:36 +02:00
function onOpenCreateChatView ( ) {
2022-10-18 11:06:18 +02:00
createChatView . opened = true
}
2023-01-18 11:25:36 +02:00
function onCloseCreateChatView ( ) {
2022-10-18 11:06:18 +02:00
createChatView . opened = false
}
2023-02-07 15:21:32 +01:00
function onOpenActivityCenterPopupRequested ( ) {
2023-04-17 10:31:32 +02:00
d . openActivityCenterPopup ( )
2022-07-26 17:23:45 +03:00
}
2023-01-18 11:25:36 +02:00
function onDisplayToastMessage ( title: string , subTitle: string , icon: string , loading: bool , ephNotifType: int , url: string ) {
2023-02-07 15:21:32 +01:00
appMain . rootStore . mainModuleInst . displayEphemeralNotification ( title , subTitle , icon , loading , ephNotifType , url )
2022-12-14 16:40:50 +02:00
}
2023-01-18 11:25:36 +02:00
function onOpenLink ( link: string ) {
2023-07-22 02:08:44 +03:00
console . warn ( "opening external url without asking user" )
2022-12-14 16:40:50 +02:00
// Qt sometimes inserts random HTML tags; and this will break on invalid URL inside QDesktopServices::openUrl(link)
2023-02-07 15:21:32 +01:00
link = appMain . rootStore . plainText ( link )
2023-07-22 02:08:44 +03:00
2022-12-14 16:40:50 +02:00
if ( appMain . rootStore . showBrowserSelector ) {
2023-02-07 15:21:32 +01:00
popups . openChooseBrowserPopup ( link )
2022-12-14 16:40:50 +02:00
} else {
if ( appMain . rootStore . openLinksInStatus ) {
2023-02-07 15:21:32 +01:00
globalConns . onAppSectionBySectionTypeChanged ( Constants . appSection . browser )
globalConns . onOpenLinkInBrowser ( link )
2022-12-14 16:40:50 +02:00
} else {
2023-02-07 15:21:32 +01:00
Qt . openUrlExternally ( link )
2022-12-14 16:40:50 +02:00
}
}
}
2023-02-07 15:21:32 +01:00
function onPlaySendMessageSound ( ) {
sendMessageSound . stop ( )
sendMessageSound . play ( )
}
function onPlayNotificationSound ( ) {
notificationSound . stop ( )
notificationSound . play ( )
}
function onPlayErrorSound ( ) {
errorSound . stop ( )
errorSound . play ( )
}
2023-01-18 11:25:36 +02:00
function onSetNthEnabledSectionActive ( nthSection: int ) {
2022-12-14 16:40:50 +02:00
if ( ! appMain . rootStore . mainModuleInst )
return
appMain . rootStore . mainModuleInst . setNthEnabledSectionActive ( nthSection )
}
2023-01-18 11:25:36 +02:00
function onAppSectionBySectionTypeChanged ( sectionType: int , subsection: int ) {
2022-12-14 16:40:50 +02:00
if ( ! appMain . rootStore . mainModuleInst )
return
appMain . rootStore . mainModuleInst . setActiveSectionBySectionType ( sectionType )
if ( sectionType === Constants . appSection . profile ) {
Global . settingsSubsection = subsection ;
}
}
2023-02-28 16:00:10 +01:00
function onOpenSendModal ( address: string ) {
sendModal . open ( address )
}
function onSwitchToCommunity ( communityId: string ) {
2023-04-28 12:35:18 +02:00
appMain . communitiesStore . setActiveCommunity ( communityId )
2023-02-28 16:00:10 +01:00
}
2021-10-19 12:27:41 +02:00
}
2023-07-27 15:33:33 +03:00
Connections {
target: appMain . communitiesStore
function onImportingCommunityStateChanged ( communityId , state , errorMsg ) {
const community = appMain . communitiesStore . getCommunityDetailsAsJson ( communityId )
let title = ""
let subTitle = ""
let loading = false
let notificationType = Constants . ephemeralNotificationType . normal
let icon = ""
switch ( state )
{
case Constants.communityImported:
if ( community . isControlNode ) {
title = qsTr ( "This device is now the control node for the %1 Community" ) . arg ( community . name )
notificationType = Constants . ephemeralNotificationType . success
icon = "checkmark-circle"
} else {
title = qsTr ( "'%1' community imported" ) . arg ( community . name )
}
break
case Constants.communityImportingInProgress:
title = qsTr ( "Importing community is in progress" )
loading = true
break
case Constants.communityImportingError:
title = qsTr ( "Failed to import community '%1'" ) . arg ( community . name )
subTitle = errorMsg
break
default:
console . error ( "unknown state while importing community: %1" ) . arg ( state )
return
}
Global . displayToastMessage ( title ,
subTitle ,
icon ,
loading ,
notificationType ,
"" )
}
function onCommunityInfoAlreadyRequested ( ) {
Global . displayToastMessage ( qsTr ( "Community data not loaded yet." ) ,
qsTr ( "Please wait for the unfurl to show" ) ,
"" ,
true ,
Constants . ephemeralNotificationType . normal ,
"" )
}
function onCommunityPrivateKeyRemoved ( communityId ) {
const community = appMain . communitiesStore . getCommunityDetailsAsJson ( communityId )
Global . displayToastMessage ( qsTr ( "This device is no longer the control node for the %1 Community" ) . arg ( community . name ) ,
"" ,
"info" ,
false ,
Constants . ephemeralNotificationType . normal ,
"" )
}
}
2023-06-05 17:30:53 +02:00
Connections {
target: Global . applicationWindow
function onActiveChanged ( ) {
if ( Global . applicationWindow . active ) appMain . rootStore . windowActivated ( )
else appMain . rootStore . windowDeactivated ( )
}
}
2021-10-19 12:27:41 +02:00
function changeAppSectionBySectionId ( sectionId ) {
2022-10-18 11:06:18 +02:00
appMain . rootStore . mainModuleInst . setActiveSectionById ( sectionId )
2021-06-11 14:17:23 +02:00
}
2021-02-18 15:14:31 -04:00
Audio {
id: sendMessageSound
2021-12-08 23:20:43 +02:00
store: rootStore
2022-12-08 16:49:14 +01:00
source: "qrc:/imports/assets/audio/send_message.wav"
2021-02-18 15:14:31 -04:00
}
Audio {
id: notificationSound
2021-12-08 23:20:43 +02:00
store: rootStore
2022-12-08 16:49:14 +01:00
source: "qrc:/imports/assets/audio/notification.wav"
2022-01-12 00:16:17 +01:00
}
Audio {
id: errorSound
2022-12-08 16:49:14 +01:00
source: "qrc:/imports/assets/audio/error.mp3"
2022-01-12 00:16:17 +01:00
store: rootStore
2021-02-18 15:14:31 -04:00
}
2022-10-18 11:06:18 +02:00
Loader {
2021-11-10 09:09:31 +01:00
id: appSearch
2022-10-18 11:06:18 +02:00
active: false
asynchronous: true
function openSearchPopup ( ) {
if ( ! active )
active = true
item . openSearchPopup ( )
}
function closeSearchPopup ( ) {
if ( item )
item . closeSearchPopup ( )
2022-11-07 15:56:59 +01:00
2022-10-18 11:06:18 +02:00
active = false
}
sourceComponent: AppSearch {
store: appMain . rootStore . appSearchStore
2022-11-07 15:56:59 +01:00
onClosed: appSearch . active = false
2022-10-18 11:06:18 +02:00
}
2021-11-10 09:09:31 +01:00
}
2023-04-28 12:35:18 +02:00
Loader {
2022-03-07 09:56:05 -05:00
id: statusEmojiPopup
2023-04-28 12:35:18 +02:00
active: appMain . rootStore . mainModuleInst . sectionsLoaded
sourceComponent: StatusEmojiPopup {
width: 360
height: 440
}
2022-03-07 09:56:05 -05:00
}
2023-01-30 16:05:34 -05:00
Loader {
id: statusStickersPopupLoader
2023-03-16 15:05:05 -04:00
active: appMain . rootStore . mainModuleInst . sectionsLoaded
2023-01-30 16:05:34 -05:00
sourceComponent: StatusStickersPopup {
id: statusStickersPopup
2023-02-09 22:49:36 +01:00
store: appMain . rootChatStore
2023-01-30 16:05:34 -05:00
}
2022-11-14 15:21:00 -05:00
}
2022-08-09 18:08:39 +03:00
StatusMainLayout {
2021-09-15 13:40:07 +02:00
id: appLayout
2022-07-26 10:24:59 +02:00
anchors.fill: parent
2021-06-17 14:41:11 -04:00
2022-08-09 18:08:39 +03:00
leftPanel: StatusAppNavBar {
2022-11-28 12:32:29 +01:00
chatItemsModel: SortFilterProxyModel {
sourceModel: appMain . rootStore . mainModuleInst . sectionsModel
filters: [
ValueFilter {
roleName: "sectionType"
value: Constants . appSection . chat
} ,
ValueFilter {
roleName: "enabled"
value: true
}
]
2021-06-17 14:41:11 -04:00
}
2022-11-28 12:32:29 +01:00
chatItemDelegate: navbarButton
communityItemsModel: SortFilterProxyModel {
sourceModel: appMain . rootStore . mainModuleInst . sectionsModel
filters: [
ValueFilter {
roleName: "sectionType"
value: Constants . appSection . community
} ,
ValueFilter {
roleName: "enabled"
value: true
}
]
2021-10-19 12:27:41 +02:00
}
2022-11-28 12:32:29 +01:00
communityItemDelegate: StatusNavBarTabButton {
2022-08-11 14:24:49 +02:00
objectName: "CommunityNavBarButton"
2021-06-17 14:41:11 -04:00
anchors.horizontalCenter: parent . horizontalCenter
2021-10-19 12:27:41 +02:00
name: model . icon . length > 0 ? "" : model . name
icon.name: model . icon
icon.source: model . image
2022-10-12 17:22:23 +02:00
identicon.asset.color: ( hovered || identicon . highlighted || checked ) ? model.color : icon . color
2021-06-17 14:41:11 -04:00
tooltip.text: model . name
2021-10-19 12:27:41 +02:00
checked: model . active
badge.value: model . notificationsCount
badge.visible: model . hasNotification
2021-06-17 14:41:11 -04:00
badge.border.color: hovered ? Theme.palette.statusBadge.hoverBorderColor : Theme . palette . statusBadge . borderColor
badge.border.width: 2
2021-10-19 12:27:41 +02:00
onClicked: {
changeAppSectionBySectionId ( model . id )
}
2021-06-17 14:41:11 -04:00
2023-01-10 14:19:02 +01:00
popupMenu: Component {
StatusMenu {
id: communityContextMenu
property var chatCommunitySectionModule
2022-01-18 15:54:14 -05:00
2023-08-02 19:39:42 +02:00
readonly property bool isSpectator: model . spectated && ! model . joined
2023-01-10 14:19:02 +01:00
openHandler: function ( ) {
// we cannot return QVariant if we pass another parameter in a function call
// that's why we're using it this way
appMain . rootStore . mainModuleInst . prepareCommunitySectionModuleForCommunityId ( model . id )
communityContextMenu . chatCommunitySectionModule = appMain . rootStore . mainModuleInst . getCommunitySectionModule ( )
}
2021-06-17 14:41:11 -04:00
2023-01-10 14:19:02 +01:00
StatusAction {
text: qsTr ( "Invite People" )
icon.name: "share-ios"
enabled: model . canManageUsers
2023-09-12 10:55:40 +02:00
objectName: "invitePeople"
2023-01-10 14:19:02 +01:00
onTriggered: {
2023-02-07 15:21:32 +01:00
popups . openInviteFriendsToCommunityPopup ( model ,
2023-01-10 14:19:02 +01:00
communityContextMenu . chatCommunitySectionModule ,
null )
}
2022-09-29 00:07:18 +03:00
}
2021-06-17 14:41:11 -04:00
2023-01-10 14:19:02 +01:00
StatusAction {
text: qsTr ( "View Community" )
icon.name: "group-chat"
2023-02-07 15:21:32 +01:00
onTriggered: popups . openCommunityProfilePopup ( appMain . rootStore , model , communityContextMenu . chatCommunitySectionModule )
2023-01-10 14:19:02 +01:00
}
2021-06-17 14:41:11 -04:00
2023-06-14 10:42:52 +02:00
StatusMenuSeparator { }
2023-06-07 13:54:31 +03:00
MuteChatMenuItem {
enabled: ! model . muted
title: qsTr ( "Mute Community" )
onMuteTriggered: {
communityContextMenu . chatCommunitySectionModule . setCommunityMuted ( interval )
communityContextMenu . close ( )
}
}
2023-04-11 19:48:32 +03:00
StatusAction {
2023-06-07 13:54:31 +03:00
enabled: model . muted
text: qsTr ( "Unmute Community" )
2023-06-21 22:37:51 +02:00
icon.name: "notification"
2023-04-11 19:48:32 +03:00
onTriggered: {
2023-06-07 13:54:31 +03:00
communityContextMenu . chatCommunitySectionModule . setCommunityMuted ( Constants . MutingVariations . Unmuted )
2023-04-11 19:48:32 +03:00
}
}
2023-08-02 19:39:42 +02:00
StatusAction {
text: qsTr ( "Edit Shared Addresses" )
icon.name: "wallet"
enabled: {
if ( model . memberRole === Constants . memberRole . owner )
return false
if ( communityContextMenu . isSpectator && ! appMain . rootStore . isCommunityRequestPending ( model . id ) )
return false
return true
}
onTriggered: {
communityContextMenu . close ( )
Global . openEditSharedAddressesFlow ( model . id )
}
}
2023-06-14 10:42:52 +02:00
StatusMenuSeparator { visible: leaveCommunityMenuItem . enabled }
2021-06-17 14:41:11 -04:00
2023-01-10 14:19:02 +01:00
StatusAction {
2023-06-14 10:42:52 +02:00
id: leaveCommunityMenuItem
2023-06-19 16:16:35 +02:00
enabled: model . memberRole !== Constants . memberRole . owner
2023-06-14 10:42:52 +02:00
text: {
2023-08-02 19:39:42 +02:00
if ( communityContextMenu . isSpectator )
2023-06-14 10:42:52 +02:00
return qsTr ( "Close Community" )
return qsTr ( "Leave Community" )
}
2023-08-02 19:39:42 +02:00
icon.name: communityContextMenu . isSpectator ? "close-circle" : "arrow-left"
2023-01-10 14:19:02 +01:00
type: StatusAction . Type . Danger
2023-08-02 19:39:42 +02:00
onTriggered: communityContextMenu . isSpectator ? communityContextMenu . chatCommunitySectionModule . leaveCommunity ( )
: popups . openLeaveCommunityPopup ( model . name , model . id , model . outroMessage )
2023-01-10 14:19:02 +01:00
}
2021-06-17 14:41:11 -04:00
}
}
}
2022-11-28 12:32:29 +01:00
regularItemsModel: SortFilterProxyModel {
sourceModel: appMain . rootStore . mainModuleInst . sectionsModel
filters: [
RangeFilter {
roleName: "sectionType"
minimumValue: Constants . appSection . wallet
maximumValue: Constants . appSection . communitiesPortal
} ,
ValueFilter {
roleName: "enabled"
value: true
}
]
}
regularItemDelegate: navbarButton
delegateHeight: 40
profileComponent: StatusNavBarTabButton {
2021-06-17 14:41:11 -04:00
id: profileButton
2022-08-12 15:19:16 +02:00
objectName: "statusProfileNavBarTabButton"
2021-09-27 20:31:17 +10:00
property bool opened: false
2021-10-19 12:27:41 +02:00
2022-03-30 17:30:28 +02:00
name: appMain . rootStore . userProfileInst . name
2021-12-01 13:46:21 +01:00
icon.source: appMain . rootStore . userProfileInst . icon
2022-11-28 12:32:29 +01:00
implicitWidth: 32
implicitHeight: 32
2022-09-05 11:24:37 +02:00
identicon.asset.width: width
identicon.asset.height: height
2022-08-11 14:55:08 +03:00
identicon.asset.charactersLen: 2
identicon.asset.color: Utils . colorForPubkey ( appMain . rootStore . userProfileInst . pubKey )
2022-12-01 11:24:25 +01:00
identicon.ringSettings.ringSpecModel: Utils . getColorHashAsJson ( appMain . rootStore . userProfileInst . pubKey ,
2022-12-13 12:37:27 +03:00
appMain . rootStore . userProfileInst . preferredName )
2022-04-01 13:46:32 +02:00
2021-06-17 14:41:11 -04:00
badge.visible: true
2022-07-26 10:24:59 +02:00
badge . anchors {
left: undefined
top: undefined
right: profileButton . right
bottom: profileButton . bottom
margins: 0
rightMargin: - badge . border . width
bottomMargin: - badge . border . width
}
badge.implicitHeight: 12
badge.implicitWidth: 12
badge.border.width: 2
2021-06-17 14:41:11 -04:00
badge.border.color: hovered ? Theme.palette.statusBadge.hoverBorderColor : Theme . palette . statusAppNavBar . backgroundColor
2022-06-10 11:01:31 +02:00
badge.color: {
switch ( appMain . rootStore . userProfileInst . currentUserStatus ) {
case Constants.currentUserStatus.automatic:
case Constants.currentUserStatus.alwaysOnline:
return Style . current . green ;
default:
return Style . current . midGrey ;
}
}
2022-07-26 10:24:59 +02:00
onClicked: userStatusContextMenu . opened ? userStatusContextMenu . close ( ) : userStatusContextMenu . open ( )
2021-06-17 14:41:11 -04:00
UserStatusContextMenu {
id: userStatusContextMenu
2022-07-04 12:47:29 +02:00
y: profileButton . y - userStatusContextMenu . height + profileButton . height
x: profileButton . x + profileButton . width + 5
2021-11-05 01:38:57 +02:00
store: appMain . rootStore
2021-06-17 14:41:11 -04:00
}
2022-11-28 12:32:29 +01:00
}
Component {
id: navbarButton
StatusNavBarTabButton {
id: navbar
objectName: model . name + "-navbar"
anchors.horizontalCenter: parent . horizontalCenter
name: model . icon . length > 0 ? "" : model . name
icon.name: model . icon
icon.source: model . image
2022-12-12 13:39:25 +01:00
tooltip.text: Utils . translatedSectionName ( model . sectionType , model . name )
2022-11-28 12:32:29 +01:00
checked: model . active
badge.value: model . notificationsCount
badge.visible: model . hasNotification
badge.border.color: hovered ? Theme.palette.statusBadge.hoverBorderColor : Theme . palette . statusBadge . borderColor
badge.border.width: 2
onClicked: {
changeAppSectionBySectionId ( model . id )
}
}
2021-06-17 14:41:11 -04:00
}
2021-07-28 14:50:48 +02:00
}
2021-06-17 14:41:11 -04:00
2022-08-09 18:08:39 +03:00
rightPanel: ColumnLayout {
spacing: 0
2022-08-31 12:55:42 +02:00
objectName: "mainRightView"
2022-09-02 17:56:14 +03:00
ColumnLayout {
id: bannersLayout
2023-01-30 11:42:26 +01:00
enabled: ! localAppSettings . testEnvironment
visible: enabled
2022-09-02 17:56:14 +03:00
property var updateBanner: null
property var connectedBanner: null
2022-10-18 11:06:18 +02:00
readonly property bool isConnected: appMain . rootStore . mainModuleInst . isOnline
2022-09-02 17:56:14 +03:00
function processUpdateAvailable ( ) {
if ( ! updateBanner )
updateBanner = updateBannerComponent . createObject ( this )
2022-03-03 17:00:52 -04:00
}
2022-09-02 17:56:14 +03:00
function processConnected ( ) {
if ( ! connectedBanner )
connectedBanner = connectedBannerComponent . createObject ( this )
2021-10-19 12:27:41 +02:00
}
2022-03-17 12:15:38 -04:00
2022-09-02 17:56:14 +03:00
Layout.fillWidth: true
Layout.maximumHeight: implicitHeight
spacing: 1
onIsConnectedChanged: {
processConnected ( )
2023-01-11 17:06:31 +03:00
}
Component.onCompleted: {
if ( ! isConnected )
processConnected ( )
2022-03-17 12:15:38 -04:00
}
2022-09-02 17:56:14 +03:00
Connections {
target: rootStore . aboutModuleInst
2023-01-18 11:25:36 +02:00
function onAppVersionFetched ( available: bool , version: string , url: string ) {
2022-09-02 17:56:14 +03:00
rootStore . setLatestVersionInfo ( available , version , url ) ;
2023-06-12 13:54:19 -04:00
// TODO when we re-implement check for updates, uncomment this
// bannersLayout.processUpdateAvailable()
2022-09-02 17:56:14 +03:00
}
2022-03-17 12:15:38 -04:00
}
2022-09-02 17:56:14 +03:00
ModuleWarning {
id: testnetBanner
2022-09-29 16:30:25 +02:00
objectName: "testnetBanner"
2022-09-02 17:56:14 +03:00
Layout.fillWidth: true
2023-06-27 21:56:44 +02:00
text: qsTr ( "Testnet mode enabled. All balances, transactions and dApp interactions will be on testnets." )
2022-09-02 17:56:14 +03:00
buttonText: qsTr ( "Turn off" )
2023-06-27 21:56:44 +02:00
type: ModuleWarning . Warning
iconName: "warning"
2022-09-02 17:56:14 +03:00
active: appMain . rootStore . profileSectionStore . walletStore . areTestNetworksEnabled
2023-06-27 21:56:44 +02:00
onClicked: Global . openTestnetPopup ( )
2023-08-04 09:45:41 +02:00
onCloseClicked: hide ( )
2022-03-17 12:15:38 -04:00
}
2022-09-02 17:56:14 +03:00
ModuleWarning {
id: secureYourSeedPhrase
2022-09-29 16:30:25 +02:00
objectName: "secureYourSeedPhraseBanner"
2022-09-02 17:56:14 +03:00
Layout.fillWidth: true
2022-09-14 22:05:44 +03:00
active: ! appMain . rootStore . profileSectionStore . profileStore . userDeclinedBackupBanner
2022-09-02 17:56:14 +03:00
&& ! appMain . rootStore . profileSectionStore . profileStore . privacyStore . mnemonicBackedUp
type: ModuleWarning . Danger
text: qsTr ( "Secure your seed phrase" )
buttonText: qsTr ( "Back up now" )
2023-02-07 15:21:32 +01:00
onClicked: popups . openBackUpSeedPopup ( )
2022-09-02 17:56:14 +03:00
onCloseClicked: {
appMain . rootStore . profileSectionStore . profileStore . userDeclinedBackupBanner = true
}
2022-03-17 12:15:38 -04:00
}
2021-12-17 12:42:12 -05:00
2022-09-15 09:31:38 +02:00
ModuleWarning {
Layout.fillWidth: true
2023-04-28 12:35:18 +02:00
readonly property int progress: appMain . communitiesStore . discordImportProgress
readonly property bool inProgress: ( progress > 0 && progress < 100 ) || appMain . communitiesStore . discordImportInProgress
2022-09-15 09:31:38 +02:00
readonly property bool finished: progress >= 100
2023-04-28 12:35:18 +02:00
readonly property bool cancelled: appMain . communitiesStore . discordImportCancelled
readonly property bool stopped: appMain . communitiesStore . discordImportProgressStopped
readonly property int errors: appMain . communitiesStore . discordImportErrorsCount
readonly property int warnings: appMain . communitiesStore . discordImportWarningsCount
readonly property string communityId: appMain . communitiesStore . discordImportCommunityId
readonly property string communityName: appMain . communitiesStore . discordImportCommunityName
2023-09-01 09:58:48 +02:00
readonly property string channelId: appMain . communitiesStore . discordImportChannelId
readonly property string channelName: appMain . communitiesStore . discordImportChannelName
readonly property string channelOrCommunityName: channelName || communityName
2022-09-15 09:31:38 +02:00
active: ! cancelled && ( inProgress || finished || stopped )
type: errors ? ModuleWarning.Type.Danger : ModuleWarning . Type . Success
text: {
if ( finished || stopped ) {
if ( errors )
2023-09-01 09:58:48 +02:00
return qsTr ( "The import of ‘ %1’ from Discord to Status was stopped: <a href='#'>Critical issues found</a>" ) . arg ( channelOrCommunityName )
2022-09-15 09:31:38 +02:00
2023-09-01 09:58:48 +02:00
let result = qsTr ( "‘ %1’ was successfully imported from Discord to Status" ) . arg ( channelOrCommunityName ) + " <a href='#'>"
2022-09-15 09:31:38 +02:00
if ( warnings )
result += qsTr ( "Details (%1)" ) . arg ( qsTr ( "%n issue(s)" , "" , warnings ) )
else
result += qsTr ( "Details" )
result += "</a>"
return result
}
if ( inProgress ) {
2023-09-01 09:58:48 +02:00
let result = qsTr ( "Importing ‘ %1’ from Discord to Status" ) . arg ( channelOrCommunityName ) + " <a href='#'>"
2022-09-15 09:31:38 +02:00
if ( warnings )
result += qsTr ( "Check progress (%1)" ) . arg ( qsTr ( "%n issue(s)" , "" , warnings ) )
else
result += qsTr ( "Check progress" )
result += "</a>"
return result
}
2022-12-09 14:35:29 +03:00
return ""
2022-09-15 09:31:38 +02:00
}
2023-09-01 09:58:48 +02:00
onLinkActivated: popups . openDiscordImportProgressPopup ( ! ! channelId )
2022-09-15 09:31:38 +02:00
progressValue: progress
closeBtnVisible: finished || stopped
2023-09-01 09:58:48 +02:00
buttonText: finished && ! errors ? ! ! channelId ? qsTr ( "Visit your new channel" ) : qsTr ( "Visit your Community" ) : ""
2022-09-15 09:31:38 +02:00
onClicked: function ( ) {
2023-09-01 09:58:48 +02:00
if ( ! ! channelId )
rootStore . setActiveSectionChat ( communityId , channelId )
else
appMain . communitiesStore . setActiveCommunity ( communityId )
2022-09-15 09:31:38 +02:00
}
2023-09-01 09:58:48 +02:00
onCloseClicked: hide ( )
2022-09-15 09:31:38 +02:00
}
2022-12-01 16:10:51 +01:00
ModuleWarning {
id: downloadingArchivesBanner
Layout.fillWidth: true
2023-04-28 12:35:18 +02:00
active: appMain . communitiesStore . downloadingCommunityHistoryArchives
2022-12-01 16:10:51 +01:00
type: ModuleWarning . Danger
text: qsTr ( "Downloading message history archives, DO NOT CLOSE THE APP until this banner disappears." )
closeBtnVisible: false
}
2022-09-15 09:31:38 +02:00
2023-01-18 14:36:28 +04:00
ModuleWarning {
id: mailserverConnectionBanner
2023-01-18 20:37:36 +04:00
type: ModuleWarning . Warning
text: qsTr ( "Can not connect to store node. Retrying automatically" )
2023-01-18 14:36:28 +04:00
onCloseClicked: hide ( )
Layout.fillWidth: true
}
2022-09-02 17:56:14 +03:00
Component {
id: connectedBannerComponent
ModuleWarning {
2022-10-04 19:38:52 +03:00
id: connectedBanner
property bool isConnected: true
2022-09-02 17:56:14 +03:00
2022-09-29 16:30:25 +02:00
objectName: "connectionInfoBanner"
2022-09-02 17:56:14 +03:00
Layout.fillWidth: true
2023-03-01 21:59:08 +01:00
text: isConnected ? qsTr ( "You are back online" ) : qsTr ( "Internet connection lost. Reconnect to ensure everything is up to date." )
2022-09-02 17:56:14 +03:00
type: isConnected ? ModuleWarning.Success : ModuleWarning . Danger
function updateState ( ) {
if ( isConnected )
showFor ( )
else
show ( ) ;
}
Component.onCompleted: {
2022-10-04 19:38:52 +03:00
connectedBanner . isConnected = Qt . binding ( ( ) = > bannersLayout . isConnected ) ;
2022-09-02 17:56:14 +03:00
}
onIsConnectedChanged: {
updateState ( ) ;
}
onCloseClicked: {
hide ( ) ;
}
onHideFinished: {
destroy ( )
2023-01-30 11:42:26 +01:00
bannersLayout . connectedBanner = null
2022-09-02 17:56:14 +03:00
}
}
}
Component {
id: updateBannerComponent
ModuleWarning {
readonly property string version: appMain . rootStore . latestVersion
readonly property bool updateAvailable: appMain . rootStore . newVersionAvailable
2022-09-29 16:35:16 +02:00
objectName: "appVersionUpdateBanner"
2022-09-02 17:56:14 +03:00
Layout.fillWidth: true
type: ModuleWarning . Success
text: updateAvailable ? qsTr ( "A new version of Status (%1) is available" ) . arg ( version )
: qsTr ( "Your version is up to date" )
buttonText: updateAvailable ? qsTr ( "Update" )
: qsTr ( "Close" )
function updateState ( ) {
if ( updateAvailable )
show ( )
else
showFor ( 5000 )
}
Component.onCompleted: {
updateState ( )
}
onUpdateAvailableChanged: {
updateState ( ) ;
}
onClicked: {
if ( updateAvailable )
Global . openDownloadModal ( appMain . rootStore . newVersionAvailable ,
appMain . rootStore . latestVersion ,
appMain . rootStore . downloadURL )
else
close ( )
}
onCloseClicked: {
if ( updateAvailable )
appMain . rootStore . resetLastVersion ( ) ;
hide ( )
}
onHideFinished: {
destroy ( )
2023-01-30 11:42:26 +01:00
bannersLayout . updateBanner = null
2022-09-02 17:56:14 +03:00
}
}
}
2023-03-01 21:59:08 +01:00
ConnectionWarnings {
id: walletBlockchainConnectionBanner
objectName: "walletBlockchainConnectionBanner"
Layout.fillWidth: true
websiteDown: Constants . walletConnections . blockchains
2023-03-23 11:23:02 +01:00
withCache: networkConnectionStore . balanceCache
2023-04-03 19:13:23 +02:00
networkConnectionStore: appMain . networkConnectionStore
tooltipMessage: qsTr ( "Pocket Network (POKT) & Infura are currently both unavailable for %1. Balances for those chains are as of %2." ) . arg ( jointChainIdString ) . arg ( lastCheckedAt )
toastText: {
2023-03-01 21:59:08 +01:00
switch ( connectionState ) {
case Constants.ConnectionStatus.Success:
return qsTr ( "Pocket Network (POKT) connection successful" )
case Constants.ConnectionStatus.Failure:
if ( completelyDown ) {
if ( withCache )
2023-04-03 19:13:23 +02:00
return qsTr ( "POKT & Infura down. Token balances are as of %1." ) . arg ( lastCheckedAt )
2023-03-01 21:59:08 +01:00
else
2023-04-03 19:13:23 +02:00
return qsTr ( "POKT & Infura down. Token balances cannot be retrieved." )
2023-03-01 21:59:08 +01:00
}
else if ( chainIdsDown . length > 0 ) {
if ( chainIdsDown . length > 2 ) {
2023-04-03 19:13:23 +02:00
return qsTr ( "POKT & Infura down for <a href='#'>multiple chains </a>. Token balances for those chains cannot be retrieved." )
2023-03-01 21:59:08 +01:00
}
else if ( chainIdsDown . length === 1 ) {
2023-04-03 19:13:23 +02:00
return qsTr ( "POKT & Infura down for %1. %1 token balances are as of %2." ) . arg ( jointChainIdString ) . arg ( lastCheckedAt )
2023-03-01 21:59:08 +01:00
}
else {
2023-04-03 19:13:23 +02:00
return qsTr ( "POKT & Infura down for %1. %1 token balances cannot be retrieved." ) . arg ( jointChainIdString )
2023-03-01 21:59:08 +01:00
}
}
else
return ""
case Constants.ConnectionStatus.Retrying:
return qsTr ( "Retrying connection to Pocket Network (POKT)." )
default:
return ""
}
}
}
ConnectionWarnings {
id: walletCollectiblesConnectionBanner
objectName: "walletCollectiblesConnectionBanner"
Layout.fillWidth: true
websiteDown: Constants . walletConnections . collectibles
2023-03-23 11:23:02 +01:00
withCache: networkConnectionStore . collectiblesCache
2023-04-03 19:13:23 +02:00
networkConnectionStore: appMain . networkConnectionStore
toastText: {
2023-03-01 21:59:08 +01:00
switch ( connectionState ) {
case Constants.ConnectionStatus.Success:
return qsTr ( "Opensea connection successful" )
case Constants.ConnectionStatus.Failure:
if ( withCache ) {
2023-04-03 19:13:23 +02:00
return qsTr ( "Opensea down. Collectibles are as of %1." ) . arg ( lastCheckedAt )
2023-03-01 21:59:08 +01:00
}
else {
2023-04-03 19:13:23 +02:00
return qsTr ( "Opensea down." )
2023-03-01 21:59:08 +01:00
}
case Constants.ConnectionStatus.Retrying:
return qsTr ( "Retrying connection to Opensea..." )
default:
return ""
}
}
}
ConnectionWarnings {
id: walletMarketConnectionBanner
objectName: "walletMarketConnectionBanner"
Layout.fillWidth: true
websiteDown: Constants . walletConnections . market
2023-03-23 11:23:02 +01:00
withCache: networkConnectionStore . marketValuesCache
2023-04-03 19:13:23 +02:00
networkConnectionStore: appMain . networkConnectionStore
toastText: {
2023-03-01 21:59:08 +01:00
switch ( connectionState ) {
case Constants.ConnectionStatus.Success:
return qsTr ( "CryptoCompare and CoinGecko connection successful" )
case Constants.ConnectionStatus.Failure: {
if ( withCache ) {
2023-04-03 19:13:23 +02:00
return qsTr ( "CryptoCompare and CoinGecko down. Market values are as of %1." ) . arg ( lastCheckedAt )
2023-03-01 21:59:08 +01:00
}
else {
2023-04-03 19:13:23 +02:00
return qsTr ( "CryptoCompare and CoinGecko down. Market values cannot be retrieved." )
2023-03-01 21:59:08 +01:00
}
}
case Constants.ConnectionStatus.Retrying:
return qsTr ( "Retrying connection to CryptoCompare and CoinGecko..." )
2023-04-03 19:13:23 +02:00
default:
return ""
2023-03-01 21:59:08 +01:00
}
}
}
2022-09-02 17:56:14 +03:00
}
2022-07-12 10:56:47 +02:00
2022-05-13 11:27:26 -04:00
Item {
2022-06-23 18:28:30 +03:00
Layout.fillWidth: true
2021-12-17 12:42:12 -05:00
Layout.fillHeight: true
2022-05-13 11:27:26 -04:00
StackLayout {
id: appView
anchors.fill: parent
currentIndex: {
2022-10-18 11:06:18 +02:00
const activeSectionType = appMain . rootStore . mainModuleInst . activeSection . sectionType
if ( activeSectionType === Constants . appSection . chat )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . chat
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . community ) {
2023-01-10 14:19:02 +01:00
for ( let i = this . children . length - 1 ; i >= 0 ; i -- ) {
2022-10-18 11:06:18 +02:00
var obj = this . children [ i ]
2023-01-10 14:19:02 +01:00
if ( obj && obj . sectionId && obj . sectionId === appMain . rootStore . mainModuleInst . activeSection . id ) {
2022-05-13 11:27:26 -04:00
return i
}
2021-12-17 12:42:12 -05:00
}
2022-05-13 11:27:26 -04:00
// Should never be here, correct index must be returned from the for loop above
2023-01-10 14:19:02 +01:00
console . error ( "Wrong section type:" , appMain . rootStore . mainModuleInst . activeSection . sectionType ,
"or section id: " , appMain . rootStore . mainModuleInst . activeSection . id )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . community
}
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . communitiesPortal )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . communitiesPortal
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . wallet )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . wallet
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . browser )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . browser
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . profile )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . profile
2022-10-18 11:06:18 +02:00
if ( activeSectionType === Constants . appSection . node )
2022-05-13 11:27:26 -04:00
return Constants . appViewStackIndex . node
2021-12-17 12:42:12 -05:00
2022-05-13 11:27:26 -04:00
// We should never end up here
2022-10-18 11:06:18 +02:00
console . error ( "AppMain: Unknown section type" )
2021-11-01 21:10:50 +01:00
}
2021-10-19 12:27:41 +02:00
2022-05-13 11:27:26 -04:00
// NOTE:
// If we ever change stack layout component order we need to updade
// Constants.appViewStackIndex accordingly
2023-01-30 16:05:34 -05:00
Loader {
id: personalChatLayoutLoader
2023-02-07 15:21:32 +01:00
asynchronous: true
2023-04-17 22:45:17 +02:00
active: false
2023-01-30 16:05:34 -05:00
sourceComponent: {
if ( appMain . rootStore . mainModuleInst . chatsLoadingFailed ) {
return errorStateComponent
}
2023-03-16 15:05:05 -04:00
if ( appMain . rootStore . mainModuleInst . sectionsLoaded ) {
2023-01-30 16:05:34 -05:00
return personalChatLayoutComponent
}
return loadingStateComponent
}
2023-04-17 22:45:17 +02:00
// Do not unload section data from the memory in order not
// to reset scroll, not send text input and etc during the
// sections switching
Binding on active {
when: appView . currentIndex === Constants . appViewStackIndex . chat
value: true
restoreMode: Binding . RestoreNone
}
2023-06-07 13:37:44 +02:00
2023-01-30 16:05:34 -05:00
Component {
id: loadingStateComponent
Item {
anchors.fill: parent
Row {
anchors.centerIn: parent
spacing: 6
StatusBaseText {
2023-05-02 11:37:12 +02:00
anchors.verticalCenter: parent . verticalCenter
2023-03-16 15:05:05 -04:00
text: qsTr ( "Loading sections..." )
2023-01-30 16:05:34 -05:00
}
2023-04-28 12:35:18 +02:00
LoadingAnimation { anchors.verticalCenter: parent . verticalCenter }
2023-01-30 16:05:34 -05:00
}
}
}
2023-06-07 13:37:44 +02:00
2023-01-30 16:05:34 -05:00
Component {
id: errorStateComponent
Item {
anchors.fill: parent
StatusBaseText {
text: qsTr ( "Error loading chats, try closing the app and restarting" )
anchors.centerIn: parent
}
}
}
Component {
id: personalChatLayoutComponent
2021-06-17 14:41:11 -04:00
2023-01-30 16:05:34 -05:00
ChatLayout {
id: chatLayoutContainer
2021-12-17 12:42:12 -05:00
2023-02-27 13:11:51 +01:00
Binding {
target: rootDropAreaPanel
property: "enabled"
value: chatLayoutContainer . currentIndex === 0 // Meaning: Chats / channels view
when: visible
restoreMode: Binding . RestoreBindingOrValue
}
2023-02-09 22:49:36 +01:00
2023-03-17 19:58:07 +01:00
rootStore: ChatStores . RootStore {
contactsStore: appMain . rootStore . contactStore
2023-04-11 10:09:01 +02:00
communityTokensStore: appMain . communityTokensStore
2023-03-17 19:58:07 +01:00
emojiReactionsModel: appMain . rootStore . emojiReactionsModel
openCreateChat: createChatView . opened
chatCommunitySectionModule: appMain . rootStore . mainModuleInst . getChatSectionModule ( )
2023-04-04 13:31:04 +02:00
networkConnectionStore: appMain . networkConnectionStore
2023-03-17 19:58:07 +01:00
}
2023-05-04 14:36:35 -04:00
createChatPropertiesStore: appMain . createChatPropertiesStore
2023-04-28 12:35:18 +02:00
emojiPopup: statusEmojiPopup . item
2023-03-07 17:34:36 +01:00
stickersPopup: statusStickersPopupLoader . item
2021-12-21 10:26:13 +01:00
2023-03-07 17:34:36 +01:00
onProfileButtonClicked: {
2023-01-30 16:05:34 -05:00
Global . changeAppSectionBySectionType ( Constants . appSection . profile ) ;
}
2022-01-04 13:06:05 +01:00
2023-03-07 17:34:36 +01:00
onOpenAppSearch: {
2023-01-30 16:05:34 -05:00
appSearch . openSearchPopup ( )
}
}
2022-05-13 11:27:26 -04:00
}
2021-12-17 12:42:12 -05:00
}
2021-06-17 14:41:11 -04:00
2023-04-28 12:35:18 +02:00
Loader {
active: appView . currentIndex === Constants . appViewStackIndex . communitiesPortal
asynchronous: true
CommunitiesPortalLayout {
anchors.fill: parent
communitiesStore: appMain . communitiesStore
2023-06-16 14:38:32 -04:00
assetsModel: SortFilterProxyModel {
sourceModel: appMain . communitiesStore . communitiesModuleInst . tokenList
proxyRoles: ExpressionRole {
function tokenIcon ( symbol ) {
return Constants . tokenIcon ( symbol )
}
name: "iconSource"
expression: ! ! model . icon ? model.icon : tokenIcon ( model . symbol )
}
}
collectiblesModel: SortFilterProxyModel {
sourceModel: appMain . communitiesStore . communitiesModuleInst . collectiblesModel
proxyRoles: ExpressionRole {
function icon ( icon ) {
return ! ! icon ? icon : Style . png ( "tokens/DEFAULT-TOKEN" )
}
name: "iconSource"
expression: icon ( model . icon )
}
}
2023-04-28 12:35:18 +02:00
notificationCount: appMain . activityCenterStore . unreadNotificationsCount
hasUnseenNotifications: activityCenterStore . hasUnseenNotifications
}
2021-12-17 12:42:12 -05:00
}
2021-11-10 13:48:22 +01:00
2022-10-18 11:06:18 +02:00
Loader {
2023-01-10 14:19:02 +01:00
active: appView . currentIndex === Constants . appViewStackIndex . wallet
2022-10-18 11:06:18 +02:00
asynchronous: true
sourceComponent: WalletLayout {
store: appMain . rootStore
contactsStore: appMain . rootStore . profileSectionStore . contactsStore
2023-04-28 12:35:18 +02:00
emojiPopup: statusEmojiPopup . item
2022-10-17 12:17:25 +02:00
sendModalPopup: sendModal
2023-04-04 13:31:04 +02:00
networkConnectionStore: appMain . networkConnectionStore
2022-05-13 11:27:26 -04:00
}
2023-01-10 14:19:02 +01:00
onLoaded: item . showSigningPhrasePopup ( )
2022-05-13 11:27:26 -04:00
}
2022-05-23 15:11:30 +02:00
2022-05-13 11:27:26 -04:00
Loader {
id: browserLayoutContainer
2023-01-10 14:19:02 +01:00
active: appView . currentIndex === Constants . appViewStackIndex . browser
2022-10-18 11:06:18 +02:00
asynchronous: true
sourceComponent: BrowserLayout {
globalStore: appMain . rootStore
sendTransactionModal: sendModal
}
2022-05-13 11:27:26 -04:00
// Loaders do not have access to the context, so props need to be set
// Adding a "_" to avoid a binding loop
// Not Refactored Yet
// property var _chatsModel: chatsModel.messageView
// Not Refactored Yet
// property var _walletModel: walletModel
// Not Refactored Yet
// property var _utilsModel: utilsModel
2022-10-18 11:06:18 +02:00
// property var _web3Provider: BrowserStores.Web3ProviderStore.web3ProviderInst
2022-05-13 11:27:26 -04:00
}
2021-11-10 09:09:31 +01:00
2022-10-18 11:06:18 +02:00
Loader {
2023-01-10 14:19:02 +01:00
active: appView . currentIndex === Constants . appViewStackIndex . profile
2022-10-18 11:06:18 +02:00
asynchronous: true
sourceComponent: ProfileLayout {
store: appMain . rootStore . profileSectionStore
globalStore: appMain . rootStore
systemPalette: appMain . sysPalette
2023-04-28 12:35:18 +02:00
emojiPopup: statusEmojiPopup . item
2023-04-04 13:31:04 +02:00
networkConnectionStore: appMain . networkConnectionStore
2022-10-18 11:06:18 +02:00
}
2021-12-17 12:42:12 -05:00
}
2022-01-31 14:29:27 +01:00
2022-10-18 11:06:18 +02:00
Loader {
2023-01-10 14:19:02 +01:00
active: appView . currentIndex === Constants . appViewStackIndex . node
2022-10-18 11:06:18 +02:00
asynchronous: true
sourceComponent: NodeLayout { }
2022-05-13 11:27:26 -04:00
}
2021-06-17 14:41:11 -04:00
2022-05-13 11:27:26 -04:00
Repeater {
2023-04-17 22:45:17 +02:00
model: SortFilterProxyModel {
sourceModel: appMain . rootStore . mainModuleInst . sectionsModel
filters: ValueFilter {
roleName: "sectionType"
value: Constants . appSection . community
}
}
delegate: Loader {
id: communityLoader
readonly property string sectionId: model . id
Layout.fillWidth: true
Layout.alignment: Qt . AlignLeft | Qt . AlignTop
Layout.fillHeight: true
asynchronous: true
active: false
// Do not unload section data from the memory in order not
// to reset scroll, not send text input and etc during the
// sections switching
Binding on active {
when: sectionId === appMain . rootStore . mainModuleInst . activeSection . id
value: true
restoreMode: Binding . RestoreNone
}
sourceComponent: ChatLayout {
id: chatLayoutComponent
Binding {
target: rootDropAreaPanel
property: "enabled"
value: chatLayoutComponent . currentIndex === 0 // Meaning: Chats / channels view
when: visible
restoreMode: Binding . RestoreBindingOrValue
}
2023-06-21 22:37:51 +02:00
Connections {
target: Global
function onSwitchToCommunitySettings ( communityId: string ) {
if ( communityId !== model . id )
return
chatLayoutComponent . currentIndex = 1 // Settings
}
}
2023-04-17 22:45:17 +02:00
2023-04-28 12:35:18 +02:00
emojiPopup: statusEmojiPopup . item
2023-04-17 22:45:17 +02:00
stickersPopup: statusStickersPopupLoader . item
sectionItemModel: model
2023-08-02 19:39:42 +02:00
createChatPropertiesStore: appMain . createChatPropertiesStore
2023-09-01 09:58:48 +02:00
communitiesStore: appMain . communitiesStore
2023-07-24 13:31:31 +02:00
communitySettingsDisabled: production && appMain . rootStore . profileSectionStore . walletStore . areTestNetworksEnabled
2023-04-17 22:45:17 +02:00
rootStore: ChatStores . RootStore {
contactsStore: appMain . rootStore . contactStore
communityTokensStore: appMain . communityTokensStore
emojiReactionsModel: appMain . rootStore . emojiReactionsModel
openCreateChat: createChatView . opened
chatCommunitySectionModule: {
appMain . rootStore . mainModuleInst . prepareCommunitySectionModuleForCommunityId ( model . id )
return appMain . rootStore . mainModuleInst . getCommunitySectionModule ( )
2022-05-13 11:27:26 -04:00
}
}
2023-04-17 22:45:17 +02:00
onProfileButtonClicked: {
Global . changeAppSectionBySectionType ( Constants . appSection . profile ) ;
}
onOpenAppSearch: {
appSearch . openSearchPopup ( )
}
2022-05-13 11:27:26 -04:00
}
}
}
2021-12-17 12:42:12 -05:00
}
2021-06-17 14:41:11 -04:00
2022-10-18 11:06:18 +02:00
Loader {
2022-07-22 13:24:28 +02:00
id: createChatView
2022-05-13 11:27:26 -04:00
property bool opened: false
2023-03-16 15:05:05 -04:00
active: appMain . rootStore . mainModuleInst . sectionsLoaded && opened
2021-12-30 13:39:47 +01:00
2022-10-18 11:06:18 +02:00
asynchronous: true
2022-05-13 11:27:26 -04:00
anchors.top: parent . top
anchors.topMargin: 8
anchors.rightMargin: 8
anchors.bottom: parent . bottom
anchors.right: parent . right
2023-01-30 16:05:34 -05:00
width: active ?
parent . width - Constants . chatSectionLeftColumnWidth -
anchors . rightMargin - anchors.leftMargin : 0
2022-07-12 10:56:47 +02:00
2022-10-18 11:06:18 +02:00
sourceComponent: CreateChatView {
2023-03-17 19:58:07 +01:00
rootStore: ChatStores . RootStore {
contactsStore: appMain . rootStore . contactStore
2023-04-11 10:09:01 +02:00
communityTokensStore: appMain . communityTokensStore
2023-03-17 19:58:07 +01:00
emojiReactionsModel: appMain . rootStore . emojiReactionsModel
openCreateChat: createChatView . opened
chatCommunitySectionModule: appMain . rootStore . mainModuleInst . getChatSectionModule ( )
}
2023-05-04 14:36:35 -04:00
createChatPropertiesStore: appMain . createChatPropertiesStore
2023-04-28 12:35:18 +02:00
emojiPopup: statusEmojiPopup . item
2023-01-30 16:05:34 -05:00
stickersPopup: statusStickersPopupLoader . item
2022-06-01 17:49:57 +03:00
}
2022-05-17 16:09:22 +02:00
}
}
2022-06-01 17:49:57 +03:00
} // ColumnLayout
2021-06-17 14:41:11 -04:00
2022-07-26 17:23:45 +03:00
Component {
id: activityCenterPopupComponent
ActivityCenterPopup {
2022-11-04 17:23:33 +03:00
// TODO get screen size // Taken from old code top bar height was fixed there to 56
2023-04-17 10:31:32 +02:00
readonly property int _buttonSize : 56
2022-11-04 17:23:33 +03:00
x: parent . width - width - Style . current . smallPadding
y: parent . y + _buttonSize
height: appView . height - _buttonSize * 2
2023-03-22 14:05:31 -04:00
store: ChatStores . RootStore {
contactsStore: appMain . rootStore . contactStore
2023-04-11 10:09:01 +02:00
communityTokensStore: appMain . communityTokensStore
2023-03-22 14:05:31 -04:00
emojiReactionsModel: appMain . rootStore . emojiReactionsModel
openCreateChat: createChatView . opened
chatCommunitySectionModule: appMain . rootStore . mainModuleInst . getChatSectionModule ( )
}
2022-10-26 20:00:20 +04:00
activityCenterStore: appMain . activityCenterStore
2022-07-26 17:23:45 +03:00
}
}
2021-10-19 12:27:41 +02:00
// Add SendModal here as it is used by the Wallet as well as the Browser
2021-06-17 14:41:11 -04:00
Loader {
id: sendModal
active: false
2022-02-04 14:07:48 +01:00
2022-07-19 18:09:20 +03:00
function open ( address = "" ) {
2021-06-17 14:41:11 -04:00
this . active = true
2022-07-19 18:09:20 +03:00
this . item . addressText = address ;
2021-06-17 14:41:11 -04:00
this . item . open ( )
2020-11-03 21:29:56 +11:00
}
2021-06-17 14:41:11 -04:00
function closed ( ) {
// this.sourceComponent = undefined // kill an opened instance
this . active = false
2020-11-03 21:29:56 +11:00
}
2021-12-08 00:15:17 +01:00
property var selectedAccount
2023-09-11 07:20:36 -03:00
property string preSelectedHoldingID
2023-09-12 16:26:38 +02:00
property int preSelectedHoldingType
2023-08-31 12:27:15 +02:00
property int sendType: - 1
2021-06-17 14:41:11 -04:00
sourceComponent: SendModal {
2023-09-11 07:20:36 -03:00
onlyAssets: false
2021-06-17 14:41:11 -04:00
onClosed: {
sendModal . closed ( )
2023-08-31 12:27:15 +02:00
sendModal . sendType = - 1
2023-09-11 07:20:36 -03:00
sendModal . preSelectedHoldingID = ""
sendModal . preSelectedHoldingType = Constants . HoldingType . Unknown
2021-06-17 14:41:11 -04:00
}
2020-12-28 15:03:57 -05:00
}
2021-12-08 00:15:17 +01:00
onLoaded: {
2022-04-12 12:10:55 +03:00
if ( ! ! sendModal . selectedAccount ) {
2022-06-27 15:41:47 +02:00
item . selectedAccount = sendModal . selectedAccount
2021-12-08 00:15:17 +01:00
}
2023-09-12 16:26:38 +02:00
if ( sendModal . sendType >= 0 ) {
2023-08-31 12:27:15 +02:00
item . sendType = sendModal . sendType
2023-09-12 16:26:38 +02:00
}
2023-09-11 07:20:36 -03:00
if ( preSelectedHoldingType !== Constants . HoldingType . Unknown ) {
item . preSelectedHoldingID = sendModal . preSelectedHoldingID
item . preSelectedHoldingType = sendModal . preSelectedHoldingType
}
2021-12-08 00:15:17 +01:00
}
2020-12-28 15:03:57 -05:00
}
2021-06-17 14:41:11 -04:00
Action {
shortcut: "Ctrl+1"
2022-09-28 12:50:15 +02:00
onTriggered: {
Global . setNthEnabledSectionActive ( 0 )
}
2020-12-28 15:03:57 -05:00
}
2021-06-17 14:41:11 -04:00
Action {
shortcut: "Ctrl+2"
2022-09-28 12:50:15 +02:00
onTriggered: {
Global . setNthEnabledSectionActive ( 1 )
}
2020-12-28 15:03:57 -05:00
}
2021-06-17 14:41:11 -04:00
Action {
shortcut: "Ctrl+3"
2022-09-28 12:50:15 +02:00
onTriggered: {
Global . setNthEnabledSectionActive ( 2 )
}
2021-06-17 14:41:11 -04:00
}
Action {
2022-09-28 12:50:15 +02:00
shortcut: "Ctrl+4"
onTriggered: {
Global . setNthEnabledSectionActive ( 3 )
}
2021-06-17 14:41:11 -04:00
}
2022-09-28 12:50:15 +02:00
Action {
shortcut: "Ctrl+5"
onTriggered: {
Global . setNthEnabledSectionActive ( 4 )
}
}
Action {
shortcut: "Ctrl+6"
onTriggered: {
Global . setNthEnabledSectionActive ( 5 )
}
}
Action {
shortcut: "Ctrl+7"
onTriggered: {
Global . setNthEnabledSectionActive ( 6 )
}
}
Action {
shortcut: "Ctrl+8"
onTriggered: {
Global . setNthEnabledSectionActive ( 7 )
}
}
Action {
shortcut: "Ctrl+9"
onTriggered: {
Global . setNthEnabledSectionActive ( 8 )
}
}
2021-06-17 14:41:11 -04:00
Action {
shortcut: "Ctrl+K"
onTriggered: {
2022-02-16 11:13:45 -05:00
// FIXME the focus is no longer on the AppMain when the popup is opened, so this does not work to close
2022-10-18 11:06:18 +02:00
if ( ! channelPickerLoader . active )
channelPickerLoader . active = true
if ( channelPickerLoader . item . opened ) {
channelPickerLoader . item . close ( )
channelPickerLoader . active = false
2021-06-17 14:41:11 -04:00
} else {
2022-10-18 11:06:18 +02:00
channelPickerLoader . item . open ( )
2021-06-17 14:41:11 -04:00
}
}
}
2022-02-16 11:13:45 -05:00
Action {
shortcut: "Ctrl+F"
onTriggered: {
// FIXME the focus is no longer on the AppMain when the popup is opened, so this does not work to close
2022-10-18 11:06:18 +02:00
if ( appSearch . active ) {
2022-02-16 11:13:45 -05:00
appSearch . closeSearchPopup ( )
} else {
appSearch . openSearchPopup ( )
}
}
}
2021-06-17 14:41:11 -04:00
2022-10-18 11:06:18 +02:00
Loader {
id: channelPickerLoader
active: false
asynchronous: true
sourceComponent: StatusSearchListPopup {
searchBoxPlaceholder: qsTr ( "Where do you want to go?" )
model: rootStore . chatSearchModel
delegate: StatusListItem {
property var modelData
property bool isCurrentItem: true
function filterAccepts ( searchText ) {
2022-12-07 16:14:48 +01:00
const lowerCaseSearchText = searchText . toLowerCase ( )
return title . toLowerCase ( ) . includes ( lowerCaseSearchText ) || label . toLowerCase ( ) . includes ( lowerCaseSearchText )
2022-10-18 11:06:18 +02:00
}
2022-02-04 14:07:48 +01:00
2022-10-18 11:06:18 +02:00
title: modelData ? modelData.name : ""
label: modelData ? modelData.sectionName : ""
highlighted: isCurrentItem
sensor.hoverEnabled: false
statusListItemIcon {
name: modelData ? modelData.name : ""
active: true
}
asset.width: 30
asset.height: 30
2023-06-14 22:18:03 +02:00
asset.color: modelData ? modelData . color ? modelData.color : Utils . colorForColorId ( modelData . colorId ) : ""
2022-10-18 11:06:18 +02:00
asset.name: modelData ? modelData.icon : ""
2023-06-14 22:18:03 +02:00
asset.charactersLen: 2
asset.letterSize: asset . _twoLettersSize
ringSettings.ringSpecModel: modelData ? modelData.colorHash : undefined
2022-02-04 14:07:48 +01:00
}
2022-10-18 11:06:18 +02:00
onAboutToShow: rootStore . rebuildChatSearchModel ( )
onSelected: {
rootStore . setActiveSectionChat ( modelData . sectionId , modelData . chatId )
close ( )
2022-02-04 14:07:48 +01:00
}
2022-01-24 12:59:59 -05:00
}
2020-12-28 15:03:57 -05:00
}
}
2021-09-23 13:59:29 +02:00
2022-07-14 14:03:36 +03:00
StatusListView {
2022-05-05 12:28:54 +02:00
id: toastArea
2022-10-20 14:58:56 +02:00
objectName: "ephemeralNotificationList"
2022-05-05 12:28:54 +02:00
anchors.right: parent . right
anchors.rightMargin: 8
anchors.bottom: parent . bottom
anchors.bottomMargin: 60
2022-08-22 15:56:28 +03:00
width: 343
height: Math . min ( parent . height - 120 , toastArea . contentHeight )
2022-05-05 12:28:54 +02:00
spacing: 8
verticalLayoutDirection: ListView . BottomToTop
model: appMain . rootStore . mainModuleInst . ephemeralNotificationModel
2023-05-03 11:37:19 +03:00
clip: false
2022-08-22 15:56:28 +03:00
2022-05-05 12:28:54 +02:00
delegate: StatusToastMessage {
primaryText: model . title
secondaryText: model . subTitle
icon.name: model . icon
loading: model . loading
type: model . ephNotifType
linkUrl: model . url
duration: model . durationInMs
2022-06-21 12:42:27 +02:00
onClicked: {
2022-09-29 16:45:34 +03:00
appMain . rootStore . mainModuleInst . ephemeralNotificationClicked ( model . timestamp )
2022-06-21 12:42:27 +02:00
this . open = false
}
2022-05-05 12:28:54 +02:00
onLinkActivated: {
2023-02-28 16:00:10 +01:00
if ( link . startsWith ( "#" ) ) // internal link to section
globalConns . onAppSectionBySectionTypeChanged ( link . substring ( 1 ) )
else
Global . openLink ( link )
2022-05-05 12:28:54 +02:00
}
2022-08-22 15:56:28 +03:00
2022-05-05 12:28:54 +02:00
onClose: {
2022-09-29 16:45:34 +03:00
appMain . rootStore . mainModuleInst . removeEphemeralNotification ( model . timestamp )
2022-05-05 12:28:54 +02:00
}
}
}
2021-09-23 13:59:29 +02:00
Component.onCompleted: {
2022-03-07 15:34:59 -05:00
const whitelist = appMain . rootStore . messagingStore . getLinkPreviewWhitelist ( )
2022-02-10 15:02:02 -05:00
try {
const whiteListedSites = JSON . parse ( whitelist )
let settingsUpdated = false
2021-11-18 14:34:03 -05:00
2022-02-10 15:02:02 -05:00
// Add Status links to whitelist
whiteListedSites . push ( { title: "Status" , address: Constants . deepLinkPrefix , imageSite: false } )
2023-05-23 16:01:39 -04:00
whiteListedSites . push ( { title: "Status" , address: Constants . externalStatusLink , imageSite: false } )
2022-02-10 15:02:02 -05:00
let settings = localAccountSensitiveSettings . whitelistedUnfurlingSites
2021-12-13 15:24:21 +01:00
2022-02-10 15:02:02 -05:00
if ( ! settings ) {
settings = { }
}
2021-12-13 15:24:21 +01:00
2023-04-28 12:35:18 +02:00
// Set Status links as true. We intercept those URLs so it is privacy-safe
2023-05-23 16:01:39 -04:00
if ( ! settings [ Constants . deepLinkPrefix ] || ! settings [ Constants . externalStatusLink ] ) {
2022-02-10 15:02:02 -05:00
settings [ Constants . deepLinkPrefix ] = true
2023-05-23 16:01:39 -04:00
settings [ Constants . externalStatusLink ] = true
2022-02-10 15:02:02 -05:00
settingsUpdated = true
}
const whitelistedHostnames = [ ]
// Add whitelisted sites in to app settings that are not already there
whiteListedSites . forEach ( site = > {
if ( ! settings . hasOwnProperty ( site . address ) ) {
settings [ site . address ] = false
settingsUpdated = true
}
whitelistedHostnames . push ( site . address )
} )
// Remove any whitelisted sites from app settings that don't exist in the
// whitelist from status-go
Object . keys ( settings ) . forEach ( settingsHostname = > {
if ( ! whitelistedHostnames . includes ( settingsHostname ) ) {
delete settings [ settingsHostname ]
settingsUpdated = true
}
} )
if ( settingsUpdated ) {
localAccountSensitiveSettings . whitelistedUnfurlingSites = settings
}
} catch ( e ) {
console . error ( 'Could not parse the whitelist for sites' , e )
}
2022-10-18 11:06:18 +02:00
Global . settingsLoaded ( )
2021-09-23 13:59:29 +02:00
}
2022-09-13 12:03:25 +02:00
2023-09-04 18:06:51 +02:00
Loader {
id: keycardPopupForAuthentication
active: false
sourceComponent: KeycardPopup {
sharedKeycardModule: appMain . rootStore . mainModuleInst . keycardSharedModuleForAuthentication
}
onLoaded: {
keycardPopupForAuthentication . item . open ( )
}
}
2023-06-30 11:24:08 +02:00
Loader {
2022-09-13 12:03:25 +02:00
id: keycardPopup
2023-06-30 11:24:08 +02:00
active: false
sourceComponent: KeycardPopup {
2022-10-18 11:06:18 +02:00
sharedKeycardModule: appMain . rootStore . mainModuleInst . keycardSharedModule
2022-09-13 12:03:25 +02:00
}
2023-06-30 11:24:08 +02:00
onLoaded: {
keycardPopup . item . open ( )
}
2022-09-13 12:03:25 +02:00
}
2022-10-06 18:47:55 +03:00
DropAreaPanel {
2022-12-09 14:35:29 +03:00
id: rootDropAreaPanel
2023-02-27 13:11:51 +01:00
2022-10-06 18:47:55 +03:00
width: appMain . width
height: appMain . height
}
2023-06-14 22:06:13 +03:00
Loader {
id: userAgreementLoader
active: production && ! localAppSettings . testEnvironment
sourceComponent: UserAgreementPopup {
visible: appMain . visible
onClosed: userAgreementLoader . active = false
}
}
2020-05-12 07:24:08 +10:00
}