fix(drag-zone): make drag zone only work in correct chats

Fixes #4904

Moves drag zone to the AppMain so that we can check if the current chat accepts the images
Also disables the Input Connection if the channel is not active
This commit is contained in:
Jonathan Rainville 2022-02-28 15:14:04 -05:00 committed by Iuri Matias
parent cc55f811b1
commit 35e50d186b
6 changed files with 122 additions and 112 deletions

View File

@ -42,6 +42,7 @@ Item {
property bool isExtendedInput: isReply || isImage property bool isExtendedInput: isReply || isImage
property bool isConnected: false property bool isConnected: false
property string contactToRemove: "" property string contactToRemove: ""
property bool isSectionActive: mainModule.activeSection.id === parentModule.getMySectionId()
property string activeChatId: parentModule && parentModule.activeItem.id property string activeChatId: parentModule && parentModule.activeItem.id
property string activeSubItemId: parentModule && parentModule.activeItem.activeSubItem.id property string activeSubItemId: parentModule && parentModule.activeItem.activeSubItem.id
property string activeChatType: parentModule && parentModule.activeItem.type property string activeChatType: parentModule && parentModule.activeItem.type
@ -148,18 +149,7 @@ Item {
delegate: ChatContentView { delegate: ChatContentView {
width: parent.width width: parent.width
clip: true clip: true
height: { height: isActiveChannel ? parent.height : 0
// dynamically calculate the height of the view, if the active one is the current one
// then set the height to parent otherwise set it to 0
if(!chatContentModule)
return 0
let myChatId = chatContentModule.getMyChatId()
if(myChatId === root.activeChatId || myChatId === root.activeSubItemId)
return parent.height
return 0
}
rootStore: root.rootStore rootStore: root.rootStore
contactsStore: root.contactsStore contactsStore: root.contactsStore
sendTransactionNoEnsModal: cmpSendTransactionNoEns sendTransactionNoEnsModal: cmpSendTransactionNoEns
@ -167,6 +157,18 @@ Item {
sendTransactionWithEnsModal: cmpSendTransactionWithEns sendTransactionWithEnsModal: cmpSendTransactionWithEns
stickersLoaded: root.stickersLoaded stickersLoaded: root.stickersLoaded
isBlocked: model.blocked isBlocked: model.blocked
isActiveChannel: {
// dynamically calculate the height of the view, if the active one is the current one
// then set the height to parent otherwise set it to 0
if(!chatContentModule || !isSectionActive)
return false
let myChatId = chatContentModule.getMyChatId()
if(myChatId === root.activeChatId || myChatId === root.activeSubItemId)
return true
return false
}
Component.onCompleted: { Component.onCompleted: {
parentModule.prepareChatContentModuleForChatId(model.itemId) parentModule.prepareChatContentModuleForChatId(model.itemId)
chatContentModule = parentModule.getChatContentModule() chatContentModule = parentModule.getChatContentModule()
@ -178,18 +180,7 @@ Item {
delegate: ChatContentView { delegate: ChatContentView {
width: parent.width width: parent.width
clip: true clip: true
height: { height: isActiveChannel ? parent.height : 0
// dynamically calculate the height of the view, if the active one is the current one
// then set the height to parent otherwise set it to 0
if(!chatContentModule)
return 0
let myChatId = chatContentModule.getMyChatId()
if(myChatId === root.activeChatId || myChatId === root.activeSubItemId)
return parent.height
return 0
}
rootStore: root.rootStore rootStore: root.rootStore
contactsStore: root.contactsStore contactsStore: root.contactsStore
sendTransactionNoEnsModal: cmpSendTransactionNoEns sendTransactionNoEnsModal: cmpSendTransactionNoEns
@ -197,6 +188,18 @@ Item {
sendTransactionWithEnsModal: cmpSendTransactionWithEns sendTransactionWithEnsModal: cmpSendTransactionWithEns
stickersLoaded: root.stickersLoaded stickersLoaded: root.stickersLoaded
isBlocked: model.blocked isBlocked: model.blocked
isActiveChannel: {
// dynamically calculate the height of the view, if the active one is the current one
// then set the height to parent otherwise set it to 0
if(!chatContentModule || !isSectionActive)
return false
let myChatId = chatContentModule.getMyChatId()
if(myChatId === root.activeChatId || myChatId === root.activeSubItemId)
return true
return false
}
Component.onCompleted: { Component.onCompleted: {
parentModule.prepareChatContentModuleForChatId(itemId) parentModule.prepareChatContentModuleForChatId(itemId)
chatContentModule = parentModule.getChatContentModule() chatContentModule = parentModule.getChatContentModule()

View File

@ -31,6 +31,7 @@ ColumnLayout {
property var chatContentModule property var chatContentModule
property var rootStore property var rootStore
property var contactsStore property var contactsStore
property bool isActiveChannel: false
property UsersStore usersStore: UsersStore {} property UsersStore usersStore: UsersStore {}
onChatContentModuleChanged: { onChatContentModuleChanged: {
@ -416,6 +417,7 @@ ColumnLayout {
} }
messageContextMenu: contextmenu messageContextMenu: contextmenu
isContactBlocked: chatContentRoot.isBlocked isContactBlocked: chatContentRoot.isBlocked
isActiveChannel: chatContentRoot.isActiveChannel
chatInputPlaceholder: chatContentRoot.isBlocked ? chatInputPlaceholder: chatContentRoot.isBlocked ?
//% "This user has been blocked." //% "This user has been blocked."
qsTrId("this-user-has-been-blocked-") : qsTrId("this-user-has-been-blocked-") :

View File

@ -32,6 +32,7 @@ Item {
anchors.fill: parent anchors.fill: parent
property alias appLayout: appLayout property alias appLayout: appLayout
property alias dragAndDrop: dragTarget
property RootStore rootStore: RootStore { } property RootStore rootStore: RootStore { }
// set from main.qml // set from main.qml
property var sysPalette property var sysPalette
@ -645,6 +646,93 @@ Item {
} }
} }
DropArea {
id: dragTarget
signal droppedOnValidScreen(var drop)
property alias droppedUrls: rptDraggedPreviews.model
property var chatCommunitySectionModule: chatLayoutContainer.rootStore.chatCommunitySectionModule
property int activeChatType: chatCommunitySectionModule && chatCommunitySectionModule.activeItem.type
property bool enabled: !drag.source && !!loader.item && !!loader.item.appLayout
&& (
// in chat view
(mainModule.activeSection.sectionType === Constants.appSection.chat &&
(
// in a one-to-one chat
activeChatType === Constants.chatType.oneToOne ||
// in a private group chat
activeChatType === Constants.chatType.privateGroupChat
)
) ||
// In community section
mainModule.activeSection.sectionType === Constants.appSection.community
)
width: appMain.width
height: appMain.height
function cleanup() {
rptDraggedPreviews.model = []
}
onDropped: (drop) => {
if (enabled) {
droppedOnValidScreen(drop)
} else {
drop.accepted = false
}
cleanup()
}
onEntered: {
if (!enabled || !!drag.source) {
drag.accepted = false
return
}
// needed because drag.urls is not a normal js array
rptDraggedPreviews.model = drag.urls.filter(img => Utils.hasDragNDropImageExtension(img))
}
onPositionChanged: {
rptDraggedPreviews.x = drag.x
rptDraggedPreviews.y = drag.y
}
onExited: cleanup()
Rectangle {
id: dropRectangle
width: parent.width
height: parent.height
color: Style.current.transparent
opacity: 0.8
states: [
State {
when: dragTarget.enabled && dragTarget.containsDrag
PropertyChanges {
target: dropRectangle
color: Style.current.background
}
}
]
}
Repeater {
id: rptDraggedPreviews
Image {
source: modelData
width: 80
height: 80
sourceSize.width: 160
sourceSize.height: 160
fillMode: Image.PreserveAspectFit
x: index * 10 + rptDraggedPreviews.x
y: index * 10 + rptDraggedPreviews.y
z: 1
}
}
}
// Add SendModal here as it is used by the Wallet as well as the Browser // Add SendModal here as it is used by the Wallet as well as the Browser
Loader { Loader {
id: sendModal id: sendModal
@ -759,6 +847,7 @@ Item {
} }
Component.onCompleted: { Component.onCompleted: {
Global.appMain = this;
const whitelist = appMain.rootStore.privacyStore.getLinkPreviewWhitelist() const whitelist = appMain.rootStore.privacyStore.getLinkPreviewWhitelist()
try { try {
const whiteListedSites = JSON.parse(whitelist) const whiteListedSites = JSON.parse(whitelist)

View File

@ -40,6 +40,7 @@ Rectangle {
property bool isImage: false property bool isImage: false
property bool isEdit: false property bool isEdit: false
property bool isContactBlocked: false property bool isContactBlocked: false
property bool isActiveChannel: false
property var recentStickers property var recentStickers
property var stickerPackList property var stickerPackList
@ -591,7 +592,8 @@ Rectangle {
} }
Connections { Connections {
target: Global.applicationWindow.dragAndDrop enabled: control.isActiveChannel
target: Global.appMain.dragAndDrop
onDroppedOnValidScreen: (drop) => { onDroppedOnValidScreen: (drop) => {
let validImages = validateImages(drop.urls) let validImages = validateImages(drop.urls)
if (validImages.length > 0) { if (validImages.length > 0) {

View File

@ -7,6 +7,7 @@ QtObject {
id: root id: root
property var applicationWindow property var applicationWindow
property var appMain
property bool popupOpened: false property bool popupOpened: false
property int settingsSubsection: Constants.settingsSubsection.profile property int settingsSubsection: Constants.settingsSubsection.profile
property var errorSound property var errorSound

View File

@ -21,7 +21,6 @@ import "./app"
StatusWindow { StatusWindow {
property bool hasAccounts: startupModule.appState !== Constants.appState.onboarding property bool hasAccounts: startupModule.appState !== Constants.appState.onboarding
property alias dragAndDrop: dragTarget
property bool displayBeforeGetStartedModal: !hasAccounts property bool displayBeforeGetStartedModal: !hasAccounts
property bool appIsReady: false property bool appIsReady: false
@ -414,92 +413,6 @@ StatusWindow {
anchors.fill: parent anchors.fill: parent
} }
DropArea {
id: dragTarget
signal droppedOnValidScreen(var drop)
property alias droppedUrls: rptDraggedPreviews.model
property bool enabled: !drag.source && !!loader.item && !!loader.item.appLayout
// Not Refactored Yet
// && (
// // in chat view
// (loader.item.appLayout.appView.currentIndex === Constants.appViewStackIndex.chat &&
// (
// // in a one-to-one chat
// chatsModel.channelView.activeChannel.chatType === Constants.chatType.oneToOne ||
// // in a private group chat
// chatsModel.channelView.activeChannel.chatType === Constants.chatType.privateGroupChat
// )
// ) ||
// // In community section
// chatsModel.communities.activeCommunity.active
// )
width: applicationWindow.width
height: applicationWindow.height
function cleanup() {
rptDraggedPreviews.model = []
}
onDropped: (drop) => {
if (enabled) {
droppedOnValidScreen(drop)
} else {
drop.accepted = false
}
cleanup()
}
onEntered: {
if (!enabled || !!drag.source) {
drag.accepted = false
return
}
// needed because drag.urls is not a normal js array
rptDraggedPreviews.model = drag.urls.filter(img => Utils.hasDragNDropImageExtension(img))
}
onPositionChanged: {
rptDraggedPreviews.x = drag.x
rptDraggedPreviews.y = drag.y
}
onExited: cleanup()
Rectangle {
id: dropRectangle
width: parent.width
height: parent.height
color: Style.current.transparent
opacity: 0.8
states: [
State {
when: dragTarget.enabled && dragTarget.containsDrag
PropertyChanges {
target: dropRectangle
color: Style.current.background
}
}
]
}
Repeater {
id: rptDraggedPreviews
Image {
source: modelData
width: 80
height: 80
sourceSize.width: 160
sourceSize.height: 160
fillMode: Image.PreserveAspectFit
x: index * 10 + rptDraggedPreviews.x
y: index * 10 + rptDraggedPreviews.y
z: 1
}
}
}
Component { Component {
id: app id: app
AppMain { AppMain {