feat(MessageContextMenu): Allow Copy and Download Image

Added copy and download Image functionality on right click of an image or its preview

closes #2672
This commit is contained in:
Khushboo Mehta 2021-08-16 11:11:43 +02:00 committed by Iuri Matias
parent 98f2a68209
commit 8caf95aace
10 changed files with 88 additions and 12 deletions

View File

@ -274,6 +274,12 @@ QtObject:
proc copyToClipboard*(self: ChatsView, content: string) {.slot.} = proc copyToClipboard*(self: ChatsView, content: string) {.slot.} =
setClipBoardText(content) setClipBoardText(content)
proc copyImageToClipboard*(self: ChatsView, content: string) {.slot} =
setClipBoardImage(content)
proc downloadImage*(self: ChatsView, content: string, path: string) {.slot} =
downloadImage(content, path)
proc linkPreviewDataWasReceived*(self: ChatsView, previewData: string) {.signal.} proc linkPreviewDataWasReceived*(self: ChatsView, previewData: string) {.signal.}
proc linkPreviewDataReceived(self: ChatsView, previewData: string) {.slot.} = proc linkPreviewDataReceived(self: ChatsView, previewData: string) {.slot.} =

View File

@ -172,6 +172,17 @@ Item {
StatusImageModal { StatusImageModal {
id: imagePopup id: imagePopup
onClicked: {
if (button === Qt.LeftButton) {
imagePopup.close()
}
else if(button === Qt.RightButton) {
messageContextMenu.imageSource = imagePopup.imageSource
messageContextMenu.hideEmojiPicker = true
messageContextMenu.isRightClickOnImage = true;
messageContextMenu.show()
}
}
} }
ColumnLayout { ColumnLayout {

View File

@ -186,12 +186,12 @@ Item {
} }
} }
property var clickMessage: function(isProfileClick, isSticker = false, isImage = false, image = null, emojiOnly = false, hideEmojiPicker = false, isReply = false) { property var clickMessage: function(isProfileClick, isSticker = false, isImage = false, image = null, emojiOnly = false, hideEmojiPicker = false, isReply = false, isRightClickOnImage = false, imageSource = "") {
if (placeholderMessage || activityCenterMessage) { if (placeholderMessage || activityCenterMessage) {
return return
} }
if (isImage) { if (isImage && !isRightClickOnImage) {
imageClick(image); imageClick(image);
return; return;
} }
@ -212,6 +212,8 @@ Item {
messageContextMenu.hideEmojiPicker = hideEmojiPicker; messageContextMenu.hideEmojiPicker = hideEmojiPicker;
messageContextMenu.pinnedMessage = pinnedMessage; messageContextMenu.pinnedMessage = pinnedMessage;
messageContextMenu.isCurrentUser = isCurrentUser; messageContextMenu.isCurrentUser = isCurrentUser;
messageContextMenu.isRightClickOnImage = isRightClickOnImage
messageContextMenu.imageSource = imageSource
if (isReply) { if (isReply) {
let nickname = appMain.getUserNickname(repliedMessageAuthor) let nickname = appMain.getUserNickname(repliedMessageAuthor)

View File

@ -317,7 +317,14 @@ Item {
ChatImage { ChatImage {
imageSource: image imageSource: image
imageWidth: 200 imageWidth: 200
onClicked: root.clickMessage(false, false, true, image) onClicked: {
if (button === Qt.LeftButton) {
root.clickMessage(false, false, true, image)
}
else if (button === Qt.RightButton) {
root.clickMessage(false, false, true, image, false, true, false, true, imageSource)
}
}
container: root.container container: root.container
} }
} }

View File

@ -10,7 +10,7 @@ Item {
property url source property url source
property bool playing: true property bool playing: true
property bool isAnimated: !!source && source.toString().endsWith('.gif') property bool isAnimated: !!source && source.toString().endsWith('.gif')
signal clicked(var image) signal clicked(var image, var button)
property var container property var container
property alias imageAlias: imageMessage property alias imageAlias: imageMessage
property bool allCornersRounded: false property bool allCornersRounded: false
@ -104,6 +104,7 @@ Item {
MouseArea { MouseArea {
cursorShape: Qt.PointingHandCursor cursorShape: Qt.PointingHandCursor
acceptedButtons: Qt.LeftButton | Qt.RightButton
anchors.fill: parent anchors.fill: parent
onClicked: { onClicked: {
if (imageContainer.isAnimated) { if (imageContainer.isAnimated) {
@ -112,7 +113,7 @@ Item {
imageContainer.playing = !imageContainer.playing imageContainer.playing = !imageContainer.playing
return return
} }
imageContainer.clicked(imageMessage) imageContainer.clicked(imageMessage, mouse.button)
} }
} }
} }

View File

@ -2,6 +2,7 @@ import QtQuick 2.12
import QtQuick.Controls 2.3 import QtQuick.Controls 2.3
import QtQuick.Layouts 1.3 import QtQuick.Layouts 1.3
import QtQml.Models 2.3 import QtQml.Models 2.3
import QtQuick.Dialogs 1.0
import StatusQ.Popups 0.1 import StatusQ.Popups 0.1
@ -24,6 +25,7 @@ StatusPopupMenu {
property bool pinnedPopup: false property bool pinnedPopup: false
property bool isText: false property bool isText: false
property bool isCurrentUser: false property bool isCurrentUser: false
property bool isRightClickOnImage: false
property string linkUrls: "" property string linkUrls: ""
property alias emojiContainer: emojiContainer property alias emojiContainer: emojiContainer
property var identicon: "" property var identicon: ""
@ -34,6 +36,7 @@ StatusPopupMenu {
property var emojiReactionsReactedByUser: [] property var emojiReactionsReactedByUser: []
property var onClickEdit: function(){} property var onClickEdit: function(){}
property var reactionModel property var reactionModel
property string imageSource: ""
property bool canPin: { property bool canPin: {
const nbPinnedMessages = chatsModel.messageView.pinnedMessagesList.count const nbPinnedMessages = chatsModel.messageView.pinnedMessagesList.count
return nbPinnedMessages < Constants.maxNumberOfPins return nbPinnedMessages < Constants.maxNumberOfPins
@ -187,6 +190,28 @@ StatusPopupMenu {
} }
*/ */
StatusMenuItem {
id: copyImageAction
text: qsTr("Copy image")
onTriggered: {
chatsModel.copyImageToClipboard(imageSource ? imageSource : "")
messageContextMenu.close()
}
icon.name: "copy"
enabled: isRightClickOnImage
}
StatusMenuItem {
id: downloadImageAction
text: qsTr("Download image")
onTriggered: {
fileDialog.open()
messageContextMenu.close()
}
icon.name: "download"
enabled: isRightClickOnImage
}
StatusMenuItem { StatusMenuItem {
id: viewProfileAction id: viewProfileAction
//% "View Profile" //% "View Profile"
@ -217,7 +242,7 @@ StatusPopupMenu {
messageContextMenu.close() messageContextMenu.close()
} }
icon.name: "chat" icon.name: "chat"
enabled: isProfile || (!hideEmojiPicker && !emojiOnly && !isProfile) enabled: isProfile || (!hideEmojiPicker && !emojiOnly && !isProfile && !isRightClickOnImage)
} }
StatusMenuItem { StatusMenuItem {
@ -228,7 +253,7 @@ StatusPopupMenu {
onClickEdit(); onClickEdit();
} }
icon.name: "edit" icon.name: "edit"
enabled: isCurrentUser && !hideEmojiPicker && !emojiOnly && !isProfile enabled: isCurrentUser && !hideEmojiPicker && !emojiOnly && !isProfile && !isRightClickOnImage
} }
StatusMenuItem { StatusMenuItem {
@ -259,7 +284,7 @@ StatusPopupMenu {
} }
icon.name: "pin" icon.name: "pin"
enabled: { enabled: {
if(isProfile || emojiOnly) if(isProfile || emojiOnly || isRightClickOnImage)
return false return false
switch (chatsModel.channelView.activeChannel.chatType) { switch (chatsModel.channelView.activeChannel.chatType) {
@ -280,7 +305,7 @@ StatusPopupMenu {
StatusMenuItem { StatusMenuItem {
id: deleteMessageAction id: deleteMessageAction
enabled: isCurrentUser && !isProfile && !emojiOnly && !pinnedPopup && enabled: isCurrentUser && !isProfile && !emojiOnly && !pinnedPopup && !isRightClickOnImage &&
(contentType === Constants.messageType || (contentType === Constants.messageType ||
contentType === Constants.stickerType || contentType === Constants.stickerType ||
contentType === Constants.emojiType || contentType === Constants.emojiType ||
@ -324,4 +349,18 @@ StatusPopupMenu {
} }
icon.name: "up" icon.name: "up"
} }
FileDialog {
id: fileDialog
title: qsTr("Please choose a directory")
selectFolder: true
modality: Qt.NonModal
onAccepted: {
chatsModel.downloadImage(imageSource ? imageSource : "", fileDialog.fileUrls)
fileDialog.close()
}
onRejected: {
fileDialog.close()
}
}
} }

View File

@ -42,6 +42,9 @@ ScrollView {
StatusImageModal { StatusImageModal {
id: imagePopup id: imagePopup
onClicked: {
root.close()
}
} }
EmojiReactions { EmojiReactions {

View File

@ -8,6 +8,10 @@ import "../../shared"
Popup { Popup {
id: root id: root
signal clicked(var button)
property string imageSource: messageImage.source
modal: true modal: true
Overlay.modal: Rectangle { Overlay.modal: Rectangle {
color: "#40000000" color: "#40000000"
@ -51,8 +55,11 @@ Popup {
smooth: false smooth: false
MouseArea { MouseArea {
onClicked: root.close()
anchors.fill: parent anchors.fill: parent
acceptedButtons: Qt.LeftButton | Qt.RightButton
onClicked: {
root.clicked(mouse.button)
}
} }
} }
} }

2
vendor/DOtherSide vendored

@ -1 +1 @@
Subproject commit 37254b04de06d9c1461a1ba6ef3c6ab8ec012547 Subproject commit 142968d0854a60f40e20e4577e31002b8b07904e

2
vendor/nimqml vendored

@ -1 +1 @@
Subproject commit b3309ae7c99a5b20cfdcf4c7d9d916d99e8d8feb Subproject commit 03c8072f896a4751acf347a38d403754c99e182b