From 23e50341d2935054c2b7bf80b9abc6a4c3840c21 Mon Sep 17 00:00:00 2001 From: Igor Sirotin Date: Sat, 20 Aug 2022 03:01:28 +0300 Subject: [PATCH] fix(StatusMessage): Design update and minor improvements (#752) --- .../sandbox/demoapp/ChatChannelView.qml | 69 +- ui/StatusQ/sandbox/demoapp/data/Models.qml | 872 ++++++----- .../Components/StatusDateGroupLabel.qml | 54 + .../src/StatusQ/Components/StatusMessage.qml | 377 +++-- .../Components/StatusMessageDetails.qml | 19 +- .../Components/StatusMessageSenderDetails.qml | 33 + .../Components/StatusSmartIdenticon.qml | 4 + .../statusMessage/StatusEditMessage.qml | 36 +- .../statusMessage/StatusImageMessage.qml | 8 +- .../StatusMessageEmojiReactions.qml | 227 +++ .../statusMessage/StatusMessageHeader.qml | 59 +- .../StatusMessageQuickActions.qml | 26 +- .../statusMessage/StatusMessageReply.qml | 29 +- .../statusMessage/StatusPinMessageDetails.qml | 37 +- .../statusMessage/StatusTextMessage.qml | 45 +- ui/StatusQ/src/StatusQ/Components/qmldir | 2 + .../Core/StatusProfileImageSettings.qml | 40 + .../StatusQ/Core/Theme/StatusDarkTheme.qml | 7 + .../StatusQ/Core/Theme/StatusLightTheme.qml | 7 + .../src/StatusQ/Core/Theme/ThemePalette.qml | 3 +- ui/StatusQ/src/StatusQ/Core/Utils/Utils.qml | 71 + ui/StatusQ/src/StatusQ/Core/Utils/qmldir | 2 + ui/StatusQ/src/StatusQ/Core/Utils/xss.js | 1318 +++++++++++++++++ ui/StatusQ/src/StatusQ/Core/qmldir | 1 + 24 files changed, 2800 insertions(+), 546 deletions(-) create mode 100644 ui/StatusQ/src/StatusQ/Components/StatusDateGroupLabel.qml create mode 100644 ui/StatusQ/src/StatusQ/Components/StatusMessageSenderDetails.qml create mode 100644 ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageEmojiReactions.qml create mode 100644 ui/StatusQ/src/StatusQ/Core/StatusProfileImageSettings.qml create mode 100644 ui/StatusQ/src/StatusQ/Core/Utils/xss.js diff --git a/ui/StatusQ/sandbox/demoapp/ChatChannelView.qml b/ui/StatusQ/sandbox/demoapp/ChatChannelView.qml index c076993daf..9da543380b 100644 --- a/ui/StatusQ/sandbox/demoapp/ChatChannelView.qml +++ b/ui/StatusQ/sandbox/demoapp/ChatChannelView.qml @@ -14,9 +14,10 @@ ListView { anchors.fill: parent anchors.margins: 15 clip: true + delegate: StatusMessage { id: delegate - width: parent.width + width: ListView.view.width audioMessageInfoText: "Audio Message" cancelButtonText: "Cancel" @@ -26,44 +27,70 @@ ListView { resendText: "Resend" pinnedMsgInfoText: "Pinned by" + timestamp: model.timestamp + isAReply: model.isReply + hasMention: model.hasMention + isPinned: model.isPinned + pinnedBy: model.pinnedBy + hasExpired: model.hasExpired + reactionsModel: model.reactions || [] + messageDetails: StatusMessageDetails { contentType: model.contentType messageContent: model.messageContent amISender: model.amIsender - displayName: model.userName - secondaryName: model.localName !== "" && model.ensName.startsWith("@") ? model.ensName: "" - chatID: model.chatKey - profileImage: StatusImageSettings { + sender.id: model.senderId + sender.userName: model.userName + sender.localName: model.localName + sender.ensName: model.ensName + sender.isContact: model.isContact + sender.trustIndicator: model.trustIndicator + sender.profileImage { width: 40 height: 40 - source: model.profileImage - isIdenticon: model.isIdenticon + pubkey: model.senderId + source: model.profileImage || "" + colorId: 1 + colorHash: ListModel { + ListElement { colorId: 13; segmentLength: 5 } + ListElement { colorId: 31; segmentLength: 5 } + ListElement { colorId: 10; segmentLength: 1 } + ListElement { colorId: 2; segmentLength: 5 } + ListElement { colorId: 26; segmentLength: 2 } + ListElement { colorId: 19; segmentLength: 4 } + ListElement { colorId: 28; segmentLength: 3 } + } } + messageText: model.message - hasMention: model.hasMention - isContact: model.isContact - trustIndicator: model.trustIndicator - isPinned: model.isPinned - pinnedBy: model.pinnedBy - hasExpired: model.hasExpired } - timestamp.text: "10:00 am" - timestamp.tooltip.text: "10:01 am" - // reply related data - isAReply: model.isReply + replyDetails: StatusMessageDetails { - amISender: model.isReply ? model.replyAmISender : "" - displayName: model.isReply ? model.replySenderName: "" - profileImage: StatusImageSettings { + amISender: model.isReply && model.replyAmISender + sender.id: model.replySenderId || "" + sender.userName: model.isReply ? model.replySenderName: "" + sender.ensName: model.isReply ? model.replySenderEnsName : "" + sender.profileImage { width: 20 height: 20 + pubkey: model.replySenderId source: model.isReply ? model.replyProfileImage: "" - isIdenticon: model.isReply ? model.replyIsIdenticon: "" + colorId: 1 + colorHash: ListModel { + ListElement { colorId: 13; segmentLength: 5 } + ListElement { colorId: 31; segmentLength: 5 } + ListElement { colorId: 10; segmentLength: 1 } + ListElement { colorId: 2; segmentLength: 5 } + ListElement { colorId: 26; segmentLength: 2 } + ListElement { colorId: 19; segmentLength: 4 } + ListElement { colorId: 28; segmentLength: 3 } + } } messageText: model.isReply ? model.replyMessageText: "" contentType: model.replyContentType messageContent: model.replyMessageContent } + quickActions: [ StatusFlatRoundButton { id: emojiBtn diff --git a/ui/StatusQ/sandbox/demoapp/data/Models.qml b/ui/StatusQ/sandbox/demoapp/data/Models.qml index 7a179b650b..7d4c9b8987 100644 --- a/ui/StatusQ/sandbox/demoapp/data/Models.qml +++ b/ui/StatusQ/sandbox/demoapp/data/Models.qml @@ -433,384 +433,279 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I property var chatMessagesModel: ListModel { id: messageData ListElement { + timestamp: "1656937930" amIsender: false + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" userName: "Ferocious Herringbone Sinewave" ensName: "" localName: "" - message: '

Adapted from "The Colors of Animals" by Sir John Lubbock in A Book of Natural History (1902, ed. David Starr Jordan)

The color of animals is by no means a matter of chance; it depends on many considerations, but in the majority of cases tends to protect the animal from danger by rendering it less conspicuous. Perhaps it may be said that if coloring is mainly protective, there ought to be but few brightly colored animals. There are, however, not a few cases in which vivid colors are themselves protective. The kingfisher itself, though so brightly colored, is by no means easy to see. The blue harmonizes with the water, and the bird as it darts along the stream looks almost like a flash of sunlight.

Desert animals are generally the color of the desert. Thus, for instance, the lion, the antelope, and the wild donkey are all sand-colored. “Indeed,” says Canon Tristram, “in the desert, where neither trees, brushwood, nor even undulation of the surface afford the slightest protection to its foes, a modification of color assimilated to that of the surrounding country is absolutely necessary. Hence, without exception, the upper plumage of every bird, and also the fur of all the smaller mammals and the skin of all the snakes and lizards, is of one uniform sand color.”

The next point is the color of the mature caterpillars, some of which are brown. This probably makes the caterpillar even more conspicuous among the green leaves than would otherwise be the case. Let us see, then, whether the habits of the insect will throw any light upon the riddle. What would you do if you were a big caterpillar? Why, like most other defenseless creatures, you would feed by night, and lie concealed by day. So do these caterpillars. When the morning light comes, they creep down the stem of the food plant, and lie concealed among the thick herbage and dry sticks and leaves, near the ground, and it is obvious that under such circumstances the brown color really becomes a protection. It might indeed be argued that the caterpillars, having become brown, concealed themselves on the ground, and that we were reversing the state of things. But this is not so, because, while we may say as a general rule that large caterpillars feed by night and lie concealed by day, it is by no means always the case that they are brown; some of them still retaining the green color. We may then conclude that the habit of concealing themselves by day came first, and that the brown color is a later adaptation.

The example of the mature caterpillar in the third paragraph is primarily intended to demonstrate _____________.

' - sticker: "" - contentType: 1 - messageContent: "" - repeatMessageInfo: true profileImage: " CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f5789" + contentType: StatusMessage.ContentType.Text + message: '

Long message
Adapted from "The Colors of Animals" by Sir John Lubbock in A Book of Natural History (1902, ed. David Starr Jordan)

The color of animals is by no means a matter of chance; it depends on many considerations, but in the majority of cases tends to protect the animal from danger by rendering it less conspicuous. Perhaps it may be said that if coloring is mainly protective, there ought to be but few brightly colored animals. There are, however, not a few cases in which vivid colors are themselves protective. The kingfisher itself, though so brightly colored, is by no means easy to see. The blue harmonizes with the water, and the bird as it darts along the stream looks almost like a flash of sunlight.

Desert animals are generally the color of the desert. Thus, for instance, the lion, the antelope, and the wild donkey are all sand-colored. “Indeed,” says Canon Tristram, “in the desert, where neither trees, brushwood, nor even undulation of the surface afford the slightest protection to its foes, a modification of color assimilated to that of the surrounding country is absolutely necessary. Hence, without exception, the upper plumage of every bird, and also the fur of all the smaller mammals and the skin of all the snakes and lizards, is of one uniform sand color.”

The next point is the color of the mature caterpillars, some of which are brown. This probably makes the caterpillar even more conspicuous among the green leaves than would otherwise be the case. Let us see, then, whether the habits of the insect will throw any light upon the riddle. What would you do if you were a big caterpillar? Why, like most other defenseless creatures, you would feed by night, and lie concealed by day. So do these caterpillars. When the morning light comes, they creep down the stem of the food plant, and lie concealed among the thick herbage and dry sticks and leaves, near the ground, and it is obvious that under such circumstances the brown color really becomes a protection. It might indeed be argued that the caterpillars, having become brown, concealed themselves on the ground, and that we were reversing the state of things. But this is not so, because, while we may say as a general rule that large caterpillars feed by night and lie concealed by day, it is by no means always the case that they are brown; some of them still retaining the green color. We may then conclude that the habit of concealing themselves by day came first, and that the brown color is a later adaptation.

The example of the mature caterpillar in the third paragraph is primarily intended to demonstrate _____________.

' + messageContent: "" isContact: true trustIndicator: StatusContactVerificationIcons.TrustedType.None hasMention: false editMode: false isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Teenage Mutant Turtle" - ensName: "@turtle.statusofus.eth" - localName: "" - message: "You're tearing me apart, Lisa!" - sticker: "" - contentType: 1 - messageContent: "" - repeatMessageInfo: true - profileImage: " - CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2ImYgiNITTlTdG1nUZ5a92VITQxITFiJmIIjSE0htAYQrMHAAD//+wwFVpz+yqXAAAAAElFTkSuQmCC" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f2333" - isContact: true - trustIndicator: StatusContactVerificationIcons.TrustedType.Verified - hasMention: false - editMode: false - isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Teenage Mutant Turtle" - ensName: "@turtle.statusofus.eth" - localName: "" - message: "It's bullshit, I did not hit her.\nI did nooot." - sticker: "" - contentType: 1 - messageContent: "" - repeatMessageInfo: false - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f2333" - isContact: true - trustIndicator: StatusContactVerificationIcons.TrustedType.Verified - hasMention: false - editMode: false - isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Boo" - ensName: "@turtle.statusofus.eth" - localName: "Boo" - message: "Oh hi, Mark!" - contentType: 1 - messageContent: "" - sticker: "Qme8vJtyrEHxABcSVGPF95PtozDgUyfr1xGjePmFdZgk9v" - repeatMessageInfo: false - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f6789" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: true - userName: "You" - ensName: "@ghd.statusofus.eth" - localName: "" - message: "Hi Johnny" - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: true - trustIndicator: StatusContactVerificationIcons.TrustedType.None - hasMention: false - editMode: false - isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: true - } - ListElement { - amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" - localName: "" - message: '

Do you have a Bitcoin wallet or Coinbase wallet?
You can earn up to 0.06021BTC every 3 hours with your phone or PC...
Without referrals nor registration fee...
If you are interested ask me “HOW”
https://t.me/Markstones455

' - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: false - replyProfileImage: "" - replyIsIdenticon: false - replyMessageText: "" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: true - pinnedBy: "Teenage Mutant Turtle" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" - localName: "" - message: '

Uniform Dark Pike

' - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: true - editMode: false - isReply: false + replySenderId: "" replySenderName: "" replyProfileImage: "" - replyIsIdenticon: true replyMessageText: "" replyAmISender: false - replyContentType: 1 + replyContentType: StatusMessage.ContentType.Text replyMessageContent: "" isPinned: false pinnedBy: "" hasExpired: false } ListElement { + timestamp: "1657937930" amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + userName: "Teenage Mutant Turtle" + ensName: "" localName: "" - message: "Replying to text message" - isCurrentUser: true - contentType: 1 + profileImage: "" + contentType: StatusMessage.ContentType.Text + message: 'Simple text message' messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: true - replySenderName: "You" - replyProfileImage: "" - replyIsIdenticon: true - replyMessageText: "Hi Johnny" - replyAmISender: false - replyContentType: 1 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" - localName: "" - message: "Replying to a Image Message" - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: true - replySenderName: "You" - replyProfileImage: "" - replyIsIdenticon: true - replyMessageText: "" - replyAmISender: false - replyContentType: 3 - replyMessageContent: "" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" - localName: "" - message: "Replying to a sticker message" - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: true - replySenderName: "You" - replyProfileImage: "" - replyIsIdenticon: true - replyMessageText: "" - replyAmISender: false - replyContentType: 4 - replyMessageContent: "https://ipfs.infura.io/ipfs/QmW4rVW3BXYHiDHzD6cDwVZtuvEa6aPyb1bbEnitEA6Hhg" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Pompie" - ensName: "@ghd.statusofus.eth" - localName: "" - message: "Replying to a Audio message" - isCurrentUser: true - contentType: 1 - messageContent: "" - sticker: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy - hasMention: false - editMode: false - isReply: true - replySenderName: "You" - replyProfileImage: "" - replyIsIdenticon: true - replyMessageText: "" - replyAmISender: false - replyContentType: StatusMessage.ContentType.Audio - replyMessageContent: "/home/khushboo/Music/SymphonyNo6.mp3" - isPinned: false - pinnedBy: "" - hasExpired: false - } - ListElement { - amIsender: false - userName: "Pumba" - ensName: "@quite.statusofus.eth" - localName: "" - message: "This is me" - isCurrentUser: true - contentType: StatusMessage.ContentType.Sticker - messageContent: "https://ipfs.infura.io/ipfs/QmW4rVW3BXYHiDHzD6cDwVZtuvEa6aPyb1bbEnitEA6Hhg" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" - isContact: false - trustIndicator: StatusContactVerificationIcons.TrustedType.Verified + trustIndicator: StatusContactVerificationIcons.TrustedType.None hasMention: false editMode: false isReply: false + replySenderId: "" + replySenderName: "" replyProfileImage: "" - replyIsIdenticon: false replyMessageText: "" replyAmISender: false - replyContentType: 1 + replyContentType: StatusMessage.ContentType.Text replyMessageContent: "" isPinned: false pinnedBy: "" hasExpired: false } ListElement { - amIsender: true - userName: "You" - ensName: "@ghd.statusofus.eth" - localName: "" - message: "" - isCurrentUser: true - contentType: StatusMessage.ContentType.Image - messageContent: "" - repeatMessageInfo: true - profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + timestamp: "1657937930" + amIsender: false + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + userName: "Teenage Mutant Turtle" + ensName: "" + localName: "Bro from work" + profileImage: "" + contentType: StatusMessage.ContentType.Text + message: 'Renamed, contact' + messageContent: "" isContact: true trustIndicator: StatusContactVerificationIcons.TrustedType.None hasMention: false editMode: false isReply: false + replySenderId: "" + replySenderName: "" replyProfileImage: "" - replyIsIdenticon: false replyMessageText: "" replyAmISender: false - replyContentType: 1 + replyContentType: StatusMessage.ContentType.Text replyMessageContent: "" isPinned: false pinnedBy: "" hasExpired: false } ListElement { + timestamp: "1657937930" + amIsender: false + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + userName: "Teenage Mutant Turtle" + ensName: "@turtle.statusofus.eth" + localName: "Bro from work" + profileImage: "" + contentType: StatusMessage.ContentType.Text + message: 'ENS, Renamed, Contact, Untrustworthy' + messageContent: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: false + userName: "Teenage Mutant Turtle" + ensName: "@turtle.statusofus.eth" + localName: "Bro from work" + profileImage: "" + message: 'ENS, renamed, contact, verified' + contentType: StatusMessage.ContentType.Text + messageContent: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.Verified + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: false + userName: "Teenage Mutant Turtle" + ensName: "" + localName: "Bro from work" + profileImage: "" + message: 'With profile image, no ENS' + contentType: StatusMessage.ContentType.Text + messageContent: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.Verified + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: false + userName: "Teenage Mutant Turtle" + ensName: "@turtle.statusofus.eth" + localName: "Bro from work" + profileImage: "" + message: 'With profile image and ENS' + contentType: StatusMessage.ContentType.Text + messageContent: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.Verified + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: true + userName: "You" + ensName: "@ghd.statusofus.eth" + localName: "" + profileImage: "" + message: 'Message with image' + contentType: StatusMessage.ContentType.Image + messageContent: "https://placekitten.com/400/400" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: true + userName: "You" + ensName: "@ghd.statusofus.eth" + localName: "" + profileImage: "" + message: '👍' + contentType: StatusMessage.ContentType.Emoji + messageContent: "👍" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1658937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + amIsender: true + userName: "You" + ensName: "@ghd.statusofus.eth" + localName: "" + profileImage: "" + message: 'Message with sticker' + contentType: StatusMessage.ContentType.Sticker + messageContent: "https://ipfs.infura.io/ipfs/QmW4rVW3BXYHiDHzD6cDwVZtuvEa6aPyb1bbEnitEA6Hhg" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" amIsender: true userName: "You" ensName: "@ghd.statusofus.eth" @@ -821,23 +716,338 @@ CExPynn1gWf9bx498P7/nzPcxEzGExhBdJGYihtAYQlO+tUZvqrPbqeudo5iJGEJjCE15a3VtodH3q2I messageContent: "/home/khushboo/Music/SymphonyNo6.mp3" repeatMessageInfo: true profileImage: "" - isIdenticon: true - chatKey: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" isContact: true trustIndicator: StatusContactVerificationIcons.TrustedType.None hasMention: false editMode: false isReply: false + replySenderId: "" + replySenderName: "" replyProfileImage: "" - replyIsIdenticon: false replyMessageText: "" replyAmISender: false - replyContentType: 1 + replyContentType: StatusMessage.ContentType.Text replyMessageContent: "" isPinned: false pinnedBy: "" hasExpired: false } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: true + userName: "You" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "Hi Johnny" + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: true + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: '

Do you have a Bitcoin wallet or Coinbase wallet?
You can earn up to 0.06021BTC every 3 hours with your phone or PC...
Without referrals nor registration fee...
If you are interested ask me “HOW”
https://t.me/Markstones455

' + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: true + pinnedBy: "Teenage Mutant Turtle" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: '

Uniform Dark Pike

' + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: true + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "Replying to text message" + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: true + replySenderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486dsfkjghyu2cf04" + replySenderName: "You" + replySenderEnsName: "" + replyProfileImage: "" + replyMessageText: "Hi Johnny" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "Replying to a Image Message" + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: true + replySenderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486dsfkjghyu2cf04" + replySenderName: "You" + replySenderEnsName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Image + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "Replying to a sticker message" + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: true + replySenderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486dsfkjghyu2cf04" + replySenderName: "You" + replySenderEnsName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Sticker + replyMessageContent: "https://ipfs.infura.io/ipfs/QmW4rVW3BXYHiDHzD6cDwVZtuvEa6aPyb1bbEnitEA6Hhg" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pompie" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "Replying to a Audio message" + isCurrentUser: true + contentType: StatusMessage.ContentType.Text + messageContent: "" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Untrustworthy + hasMention: false + editMode: false + isReply: true + replySenderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486dsfkjghyu2cf04" + replySenderName: "You" + replySenderEnsName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Audio + replyMessageContent: "/home/khushboo/Music/SymphonyNo6.mp3" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: false + userName: "Pumba" + ensName: "@quite.statusofus.eth" + localName: "" + message: "This is me" + isCurrentUser: true + contentType: StatusMessage.ContentType.Sticker + messageContent: "https://ipfs.infura.io/ipfs/QmW4rVW3BXYHiDHzD6cDwVZtuvEa6aPyb1bbEnitEA6Hhg" + repeatMessageInfo: true + profileImage: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.Verified + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1660937930" + senderId: "0x04d1bed19c523299cbd07ddec7b8949ad7dd923819a68e0b733c9c0bc38cde276bd256f098e755f8f028395c2c91f438a22adaff6caded060b7cc0ef3f470f1234" + amIsender: true + userName: "You" + ensName: "@ghd.statusofus.eth" + localName: "" + message: "" + isCurrentUser: true + contentType: StatusMessage.ContentType.Image + messageContent: "https://placekitten.com/600/400" + repeatMessageInfo: true + profileImage: "" + isContact: true + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + } + ListElement { + timestamp: "1657937930" + amIsender: false + senderId: "0x043a7ed0e8752236a4688563652fd0296453cef00a5dcddbe252dc74f72cc1caa97a2b65e4a1a52d9c30a84c9966beaaaf6b333d659cbdd2e486b443ed1012cf04" + userName: "Teenage Mutant Turtle" + ensName: "" + localName: "" + profileImage: "" + contentType: StatusMessage.ContentType.Text + message: 'Simple text message with reactions' + messageContent: "" + isContact: false + trustIndicator: StatusContactVerificationIcons.TrustedType.None + hasMention: false + editMode: false + isReply: false + replySenderId: "" + replySenderName: "" + replyProfileImage: "" + replyMessageText: "" + replyAmISender: false + replyContentType: StatusMessage.ContentType.Text + replyMessageContent: "" + isPinned: false + pinnedBy: "" + hasExpired: false + reactions: [ + ListElement { + numberOfReactions: 2 + didIReactWithThisEmoji: false + jsonArrayOfUsersReactedWithThisEmoji: '["User 1", "User 3"]' +// jsonArrayOfUsersReactedWithThisEmoji: [ +// ListElement { text: "User 1" }, +// ListElement { text: "User 3" } +// ] + emojiId: 1 + }, + ListElement { + numberOfReactions: 3 + didIReactWithThisEmoji: true + jsonArrayOfUsersReactedWithThisEmoji: '["Teenage Mutant Turtle", "User 1", "User 3"]' + emojiId: 3 + }, + ListElement { + numberOfReactions: 1 + didIReactWithThisEmoji: false + jsonArrayOfUsersReactedWithThisEmoji: '["User 3"]' + emojiId: 4 + } + ] + } } property var membersListModel: ListModel { diff --git a/ui/StatusQ/src/StatusQ/Components/StatusDateGroupLabel.qml b/ui/StatusQ/src/StatusQ/Components/StatusDateGroupLabel.qml new file mode 100644 index 0000000000..e933792cdf --- /dev/null +++ b/ui/StatusQ/src/StatusQ/Components/StatusDateGroupLabel.qml @@ -0,0 +1,54 @@ +import QtQuick 2.14 + +import StatusQ.Core 0.1 +import StatusQ.Core.Theme 0.1 + +StatusBaseText { + id: root + + property int previousMessageIndex: -1 + property double previousMessageTimestamp + property double messageTimestamp + + font.pixelSize: 13 + color: Theme.palette.baseColor1 + horizontalAlignment: Text.AlignHCenter + + text: { + if (previousMessageIndex === -1) + return ""; + + const now = new Date() + const yesterday = new Date() + yesterday.setDate(now.getDate()-1) + + const currentMsgDate = new Date(messageTimestamp); + const prevMsgDate = new Date(previousMessageTimestamp); + + if (!!prevMsgDate && currentMsgDate.getDay() === prevMsgDate.getDay()) + return ""; + + if (now == currentMsgDate) + return qsTr("Today"); + + if (yesterday == currentMsgDate) + return qsTr("Yesterday"); + + const monthNames = [ + qsTr("January"), + qsTr("February"), + qsTr("March"), + qsTr("April"), + qsTr("May"), + qsTr("June"), + qsTr("July"), + qsTr("August"), + qsTr("September"), + qsTr("October"), + qsTr("November"), + qsTr("December") + ]; + + return monthNames[currentMsgDate.getMonth()] + ", " + currentMsgDate.getDate(); + } +} diff --git a/ui/StatusQ/src/StatusQ/Components/StatusMessage.qml b/ui/StatusQ/src/StatusQ/Components/StatusMessage.qml index ec4ea894d8..fe8ddf3520 100644 --- a/ui/StatusQ/src/StatusQ/Components/StatusMessage.qml +++ b/ui/StatusQ/src/StatusQ/Components/StatusMessage.qml @@ -1,14 +1,16 @@ import QtQuick 2.14 import QtQuick.Layouts 1.14 +import QtQuick.Controls 2.14 import StatusQ.Core 0.1 import StatusQ.Core.Theme 0.1 +import StatusQ.Core.Utils 0.1 import StatusQ.Controls 0.1 import "./private/statusMessage" Rectangle { - id: statusMessage + id: root enum ContentType { Unknown = 0, @@ -21,12 +23,12 @@ Rectangle { Invitation = 7 } - property alias messageHeader: messageHeader - property alias quickActions:quickActionsPanel.quickActions + property alias quickActions: quickActionsPanel.items property alias statusChatInput: editComponent.inputComponent property alias linksComponent: linksLoader.sourceComponent - property alias footerComponent: footer.sourceComponent - property alias timestamp: messageHeader.timestamp + property alias transcationComponent: transactionBubbleLoader.sourceComponent + property alias invitationComponent: invitationBubbleLoader.sourceComponent + property alias mouseArea: mouseArea property string resendText: "" property string cancelButtonText: "" @@ -35,155 +37,360 @@ Rectangle { property string errorLoadingImageText: "" property string audioMessageInfoText: "" property string pinnedMsgInfoText: "" + property var reactionIcons: [ + Emoji.iconSource("❤"), + Emoji.iconSource("👍"), + Emoji.iconSource("👎"), + Emoji.iconSource("🤣"), + Emoji.iconSource("😥"), + Emoji.iconSource("😠") + ] + property string messageId: "" property bool isAppWindowActive: false property bool editMode: false property bool isAReply: false + property bool isEdited: false + property bool isChatBlocked: false + + property bool hasMention: false + property bool isPinned: false + property string pinnedBy: "" + property bool hasExpired: false + property double timestamp: 0 + property var reactionsModel: [] + + readonly property bool dateGroupVisible: dateGroupLabel.visible + property bool showHeader: true + property bool isActiveMessage: false + property bool disableHover: false + property bool hideQuickActions: false + property color overrideBackgroundColor: "transparent" + property bool overrideBackground: false + + property alias previousMessageIndex: dateGroupLabel.previousMessageIndex + property alias previousMessageTimestamp: dateGroupLabel.previousMessageTimestamp + property StatusMessageDetails messageDetails: StatusMessageDetails {} property StatusMessageDetails replyDetails: StatusMessageDetails {} - signal profilePictureClicked() - signal senderNameClicked() - signal editCompleted(var newMsgText) - signal replyProfileClicked() - signal stickerLoaded() - signal imageClicked(var imageSource) + property string timestampString: Qt.formatTime(new Date(timestamp), "hh:mm"); + property string timestampTooltipString: Qt.formatTime(new Date(timestamp), "dddd, MMMM d, yyyy hh:mm:ss t"); + + signal clicked(var sender, var mouse) + signal profilePictureClicked(var sender, var mouse) + signal senderNameClicked(var sender, var mouse) + signal replyProfileClicked(var sender, var mouse) + + signal addReactionClicked(var sender, var mouse) + signal toggleReactionClicked(int emojiId) + signal imageClicked(var image, var mouse, var imageSource) + signal stickerClicked() signal resendClicked() - height: childrenRect.height - color: hoverHandler.hovered ? (messageDetails.hasMention ? Theme.palette.mentionColor3 : messageDetails.isPinned ? Theme.palette.pinColor2 : Theme.palette.baseColor2) : messageDetails.hasMention ? Theme.palette.mentionColor4 : messageDetails.isPinned ? Theme.palette.pinColor3 : "transparent" + signal editCompleted(var newMsgText) + signal editCancelled() + signal stickerLoaded() + signal linkActivated(string link) + + signal hoverChanged(string messageId, bool hovered) + signal activeChanged(string messageId, bool active) + + function startMessageFoundAnimation() { + messageFoundAnimation.start(); + } + + implicitWidth: messageLayout.implicitWidth + + messageLayout.anchors.leftMargin + + messageLayout.anchors.rightMargin + + implicitHeight: messageLayout.implicitHeight + + messageLayout.anchors.topMargin + + messageLayout.anchors.bottomMargin + + color: { + if (root.overrideBackground) + return root.overrideBackgroundColor; + + if (root.editMode) + return Theme.palette.baseColor2; + + if (hoverHandler.hovered || root.isActiveMessage) { + if (root.hasMention) + return Theme.palette.mentionColor3; + if (root.isPinned) + return Theme.palette.pinColor2; + return Theme.palette.baseColor2; + } + + if (root.hasMention) + return Theme.palette.mentionColor4; + if (root.isPinned) + return Theme.palette.pinColor3; + return "transparent"; + } + + Rectangle { + anchors { + top: parent.top + bottom: parent.bottom + left: parent.left + } + width: 2 + visible: root.isPinned + color: Theme.palette.pinColor1 + } + + Rectangle { + anchors { + top: parent.top + bottom: parent.bottom + left: parent.left + } + width: 2 + visible: root.hasMention + color: Theme.palette.mentionColor1 + } + + SequentialAnimation { + id: messageFoundAnimation + + PauseAnimation { + duration: 600 + } + NumberAnimation { + target: highlightRect + property: "opacity" + to: 1.0 + duration: 1500 + } + PauseAnimation { + duration: 1000 + } + NumberAnimation { + target: highlightRect + property: "opacity" + to: 0.0 + duration: 1500 + } + } + + Rectangle { + id: highlightRect + anchors.fill: parent + opacity: 0 + visible: opacity > 0.001 + color: Theme.palette.baseColor2 + } + + MouseArea { + id: mouseArea + anchors.fill: parent + } HoverHandler { id: hoverHandler + enabled: !root.isActiveMessage && !root.disableHover } ColumnLayout { id: messageLayout - width: parent.width - StatusMessageReply { + anchors.fill: parent + anchors.topMargin: 8 + anchors.bottomMargin: 8 + + StatusDateGroupLabel { + id: dateGroupLabel Layout.fillWidth: true - visible: isAReply - replyDetails: statusMessage.replyDetails - onReplyProfileClicked: statusMessage.replyProfileClicked() - audioMessageInfoText: statusMessage.audioMessageInfoText + Layout.topMargin: 20 + messageTimestamp: root.timestamp + visible: text !== "" } - RowLayout { - spacing: 8 + + Loader { Layout.fillWidth: true - StatusSmartIdenticon { - id: profileImage + active: isAReply + visible: active + sourceComponent: StatusMessageReply { + replyDetails: root.replyDetails + onReplyProfileClicked: root.replyProfileClicked(sender, mouse) + audioMessageInfoText: root.audioMessageInfoText + } + } + + RowLayout { + Layout.fillWidth: true + Layout.leftMargin: 16 + Layout.rightMargin: 16 + spacing: 8 + + Item { Layout.alignment: Qt.AlignTop - Layout.topMargin: 10 - Layout.leftMargin: 16 - image: messageDetails.profileImage - name: messageHeader.displayName - MouseArea { - cursorShape: Qt.PointingHandCursor - acceptedButtons: Qt.LeftButton | Qt.RightButton - anchors.fill: parent - onClicked: statusMessage.profilePictureClicked() + + implicitWidth: profileImage.effectiveSize.width + implicitHeight: profileImage.visible ? profileImage.effectiveSize.height : 0 + + StatusSmartIdenticon { + id: profileImage + + active: root.showHeader + visible: active + + name: root.messageDetails.sender.userName + image: root.messageDetails.sender.profileImage.imageSettings + icon: root.messageDetails.sender.profileImage.iconSettings + ringSettings: root.messageDetails.sender.profileImage.ringSettings + + MouseArea { + cursorShape: Qt.PointingHandCursor + acceptedButtons: Qt.LeftButton | Qt.RightButton + anchors.fill: parent + onClicked: root.profilePictureClicked(this, mouse) + } } } - Column { + + ColumnLayout { spacing: 4 Layout.alignment: Qt.AlignTop - Layout.topMargin: 10 Layout.fillWidth: true - StatusPinMessageDetails { - visible: messageDetails.isPinned && !editMode - pinnedMsgInfoText: statusMessage.pinnedMsgInfoText - pinnedBy: messageDetails.pinnedBy + + Loader { + active: root.isPinned && !editMode + visible: active + sourceComponent: StatusPinMessageDetails { + pinnedMsgInfoText: root.pinnedMsgInfoText + pinnedBy: root.pinnedBy + } } StatusMessageHeader { - id: messageHeader - width: parent.width - displayName: messageDetails.displayName - secondaryName: messageDetails.secondaryName - tertiaryDetail: messageDetails.chatID - isContact: messageDetails.isContact - trustIndicator: messageDetails.trustIndicator - resendText: statusMessage.resendText - showResendButton: messageDetails.hasExpired && messageDetails.amISender - onClicked: statusMessage.senderNameClicked() - onResendClicked: statusMessage.resendClicked() - visible: !editMode + Layout.fillWidth: true + sender: root.messageDetails.sender + amISender: root.messageDetails.amISender + resendText: root.resendText + showResendButton: root.hasExpired && root.messageDetails.amISender + onClicked: root.senderNameClicked(sender, mouse) + onResendClicked: root.resendClicked() + visible: root.showHeader && !editMode + timestamp.text: root.timestampString + timestamp.tooltip.text: root.timestampTooltipString } Loader { - active: !editMode && !!messageDetails.messageText - width: parent.width + Layout.fillWidth: true + active: !editMode && !!root.messageDetails.messageText visible: active sourceComponent: StatusTextMessage { - width: parent.width - textField.text: messageDetails.messageText + textField.text: { + if (root.messageDetails.contentType === StatusMessage.ContentType.Sticker) + return ""; + + const formattedMessage = Utils.linkifyAndXSS(root.messageDetails.messageText); + + if (root.messageDetails.contentType === StatusMessage.ContentType.Emoji) + return Emoji.parse(formattedMessage, Emoji.size.middle, Emoji.format.png); + + if (root.isEdited) { + const index = formattedMessage.endsWith("code>") ? formattedMessage.length : formattedMessage.length - 4; + const editedMessage = formattedMessage.slice(0, index) + + ` ` + qsTr("(edited)") + `` + + formattedMessage.slice(index); + return Utils.getMessageWithStyle(Emoji.parse(editedMessage), textField.hoveredLink) + } + + return Utils.getMessageWithStyle(Emoji.parse(formattedMessage), textField.hoveredLink) + } + onLinkActivated: { + root.linkActivated(link); + } } + } + Loader { - active: messageDetails.contentType === StatusMessage.ContentType.Image && !editMode + active: root.messageDetails.contentType === StatusMessage.ContentType.Image && !editMode visible: active sourceComponent: StatusImageMessage { - source: messageDetails.contentType === StatusMessage.ContentType.Image ? messageDetails.messageContent : "" - onClicked: statusMessage.imageClicked() - shapeType: messageDetails.amISender ? StatusImageMessage.ShapeType.RIGHT_ROUNDED : StatusImageMessage.ShapeType.LEFT_ROUNDED + source: root.messageDetails.contentType === StatusMessage.ContentType.Image ? root.messageDetails.messageContent : "" + onClicked: root.imageClicked(image, mouse, imageSource) + shapeType: root.messageDetails.amISender ? StatusImageMessage.ShapeType.RIGHT_ROUNDED : StatusImageMessage.ShapeType.LEFT_ROUNDED } } - StatusSticker { - visible: messageDetails.contentType === StatusMessage.ContentType.Sticker && !editMode - image.source: messageDetails.messageContent - onLoaded: statusMessage.stickerLoaded() + Loader { + active: root.messageDetails.contentType === StatusMessage.ContentType.Sticker && !editMode + visible: active + sourceComponent: StatusSticker { + image.source: root.messageDetails.messageContent + onLoaded: root.stickerLoaded() + onClicked: { + root.stickerClicked() + } + } } Loader { - active: messageDetails.contentType === StatusMessage.ContentType.Audio && !editMode + active: root.messageDetails.contentType === StatusMessage.ContentType.Audio && !editMode visible: active sourceComponent: StatusAudioMessage { - audioSource: messageDetails.messageContent + audioSource: root.messageDetails.messageContent hovered: hoverHandler.hovered - audioMessageInfoText: statusMessage.audioMessageInfoText + audioMessageInfoText: root.audioMessageInfoText } } Loader { id: linksLoader - active: !!linksLoader.sourceComponent + active: !root.editMode visible: active } Loader { id: transactionBubbleLoader - active: messageDetails.contentType === StatusMessage.ContentType.Transaction && !editMode + active: root.messageDetails.contentType === StatusMessage.ContentType.Transaction && !editMode visible: active } Loader { id: invitationBubbleLoader - active: messageDetails.contentType === StatusMessage.ContentType.Invitation && !editMode + active: root.messageDetails.contentType === StatusMessage.ContentType.Invitation && !editMode visible: active } StatusEditMessage { id: editComponent - width: parent.width - msgText: messageDetails.messageText - visible: editMode - saveButtonText: statusMessage.saveButtonText - cancelButtonText: statusMessage.cancelButtonText - onCancelEditClicked: editMode = false - onEditCompleted: { - editMode = false - statusMessage.editCompleted(newMsgText) - } + Layout.fillWidth: true + Layout.rightMargin: 16 + active: root.editMode + visible: active + msgText: root.messageDetails.messageText + saveButtonText: root.saveButtonText + cancelButtonText: root.cancelButtonText + onEditCancelled: root.editCancelled() + onEditCompleted: root.editCompleted(newMsgText) } StatusBaseText { - id: retryLbl color: Theme.palette.dangerColor1 - text: statusMessage.resendText + text: root.resendText font.pixelSize: 12 - visible: messageDetails.hasExpired && messageDetails.amISender && !messageDetails.timestamp && !editMode + visible: root.hasExpired && root.messageDetails.amISender && !root.timestamp && !editMode MouseArea { cursorShape: Qt.PointingHandCursor anchors.fill: parent - onClicked: statusMessage.resendClicked() + onClicked: root.resendClicked() } } Loader { - id: footer - active: sourceComponent && !editMode + active: root.reactionsModel.count > 0 visible: active + sourceComponent: StatusMessageEmojiReactions { + id: emojiReactionsPanel + + emojiReactionsModel: root.reactionsModel + store: root.messageStore + icons: root.reactionIcons + + onHoverChanged: { + root.hoverChanged(messageId, hovered) + } + + isCurrentUser: root.messageDetails.amISender + onAddEmojiClicked: root.addReactionClicked(sender, mouse) + onToggleReaction: root.toggleReactionClicked(emojiID) + } } } } @@ -195,6 +402,6 @@ Rectangle { anchors.rightMargin: 20 anchors.top: parent.top anchors.topMargin: -8 - visible: hoverHandler.hovered && !editMode + visible: hoverHandler.hovered && !root.hideQuickActions } } diff --git a/ui/StatusQ/src/StatusQ/Components/StatusMessageDetails.qml b/ui/StatusQ/src/StatusQ/Components/StatusMessageDetails.qml index 428ef6b293..df1cf5e56e 100644 --- a/ui/StatusQ/src/StatusQ/Components/StatusMessageDetails.qml +++ b/ui/StatusQ/src/StatusQ/Components/StatusMessageDetails.qml @@ -6,24 +6,13 @@ QtObject { id: msgDetails property bool amISender: false - property string displayName: "" - property string secondaryName: "" - property string chatID: "" - property StatusImageSettings profileImage: StatusImageSettings { - width: 40 - height: 40 - } + + property StatusMessageSenderDetails sender: StatusMessageSenderDetails { } + property bool isEdited: false - property string messageText: "" property int contentType: 0 + property string messageText: "" property string messageContent: "" - property bool isContact: false - property var trustIndicator: StatusContactVerificationIcons.TrustedType.None - property bool hasMention: false - property bool isPinned: false - property string pinnedBy: "" - property bool hasExpired: false - property string timestamp: "" } diff --git a/ui/StatusQ/src/StatusQ/Components/StatusMessageSenderDetails.qml b/ui/StatusQ/src/StatusQ/Components/StatusMessageSenderDetails.qml new file mode 100644 index 0000000000..7d680fef1e --- /dev/null +++ b/ui/StatusQ/src/StatusQ/Components/StatusMessageSenderDetails.qml @@ -0,0 +1,33 @@ +import QtQuick 2.0 +import StatusQ.Core 0.1 + +QtObject { + id: root + + property string id: "" + property string userName: "" + property string ensName: "" + property string localName: "" + + property bool isContact: false + property int trustIndicator: StatusContactVerificationIcons.TrustedType.None + + property StatusProfileImageSettings profileImage: StatusProfileImageSettings { + pubkey: root.id + showRing: !root.ensName + width: 40 + height: 40 + } + + readonly property string displayName: root.localName !== "" + ? root.localName + : root.ensName !== "" + ? root.ensName + : root.userName + + readonly property string secondaryName: root.localName === "" + ? "" + : root.ensName !== "" + ? root.ensName + : root.userName +} diff --git a/ui/StatusQ/src/StatusQ/Components/StatusSmartIdenticon.qml b/ui/StatusQ/src/StatusQ/Components/StatusSmartIdenticon.qml index dc4e8a4427..8da1d17e7a 100644 --- a/ui/StatusQ/src/StatusQ/Components/StatusSmartIdenticon.qml +++ b/ui/StatusQ/src/StatusQ/Components/StatusSmartIdenticon.qml @@ -28,6 +28,10 @@ Loader { distinctiveColors: Theme.palette.identiconRingColors } + readonly property size effectiveSize: !!statusSmartIdenticon.image.source.toString() + ? Qt.size(statusSmartIdenticon.image.width, statusSmartIdenticon.image.width) + : Qt.size(statusSmartIdenticon.icon.width, statusSmartIdenticon.icon.height) + sourceComponent: statusSmartIdenticon.icon.isLetterIdenticon ? letterIdenticon : !!statusSmartIdenticon.image.source.toString() ? roundedImage : !!statusSmartIdenticon.icon.name.toString() ? roundedIcon : letterIdenticon diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusEditMessage.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusEditMessage.qml index fe0bec630f..f2555ab30b 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusEditMessage.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusEditMessage.qml @@ -6,45 +6,63 @@ import StatusQ.Core.Theme 0.1 import StatusQ.Controls 0.1 Item { - id: editText + id: root property alias inputComponent: chatInputLoader.sourceComponent + property alias active: chatInputLoader.active property string cancelButtonText: "" property string saveButtonText: "" property string msgText: "" - signal cancelEditClicked() + signal editCancelled() signal editCompleted(var newMsgText) - height: childrenRect.height + implicitHeight: layout.implicitHeight + implicitWidth: layout.implicitWidth ColumnLayout { + id: layout + + anchors.fill: parent spacing: 4 + Loader { id: chatInputLoader - // To-Do: Move to StatusChatInput once its moved to StatusQ + Layout.fillWidth: true + + /* + NOTE: sourceComponent must have `messageText` property + TODO: Replace with StatusChatInput once its moved to StatusQ. + */ + sourceComponent: StatusInput { - width: editText.width - placeholderText: "" + readonly property string messageText: input.text + width: parent.width + input.placeholderText: "" input.text: msgText maximumHeight: 40 } } + RowLayout { spacing: 4 StatusFlatButton { id: cancelBtn text: cancelButtonText size: StatusBaseButton.Size.Small - onClicked: cancelEditClicked() + onClicked: { + editCancelled() + } } StatusButton { id: saveBtn text: saveButtonText size: StatusBaseButton.Size.Small - enabled: chatInputLoader.item.input.text.trim().length > 0 - onClicked: editCompleted(chatInputLoader.item.input.text) + enabled: !!chatInputLoader.item && chatInputLoader.item.messageText.trim().length > 0 + onClicked: { + editCompleted(!chatInputLoader.item ? "" : chatInputLoader.item.messageText) + } } } } diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusImageMessage.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusImageMessage.qml index 9e28a6bf45..5f8d13ab03 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusImageMessage.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusImageMessage.qml @@ -25,10 +25,10 @@ Item { property string loadingImageText: "" property string errorLoadingImageText: "" - signal clicked(var image, var mouse) + signal clicked(var image, var mouse, var imageSource) - width: loadingImage.visible ? loadingImage.width : imageMessage.width - height: loadingImage.visible ? loadingImage.height : imageMessage.paintedHeight + implicitWidth: loadingImage.visible ? loadingImage.width : imageMessage.width + implicitHeight: loadingImage.visible ? loadingImage.height : imageMessage.paintedHeight QtObject { id: _internal @@ -87,7 +87,7 @@ Item { _internal.pausePlaying = ! _internal.pausePlaying return } - imageContainer.clicked(imageMessage, mouse) + imageContainer.clicked(imageMessage, mouse, imageMessage.source) } } } diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageEmojiReactions.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageEmojiReactions.qml new file mode 100644 index 0000000000..58e888cee5 --- /dev/null +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageEmojiReactions.qml @@ -0,0 +1,227 @@ +import QtQuick 2.3 +import QtQuick.Controls 2.13 +import QtGraphicalEffects 1.13 + +import StatusQ.Core 0.1 +import StatusQ.Core.Theme 0.1 +import StatusQ.Core.Utils 0.1 +import StatusQ.Controls 0.1 +import StatusQ.Components 0.1 + +Item { + id: root + + implicitHeight: 22 + implicitWidth: childrenRect.width + + property int imageMargin: 4 + signal addEmojiClicked(var sender, var mouse) + signal hoverChanged(bool hovered) + signal toggleReaction(int emojiID) + + property var store + property bool isCurrentUser + property var emojiReactionsModel + + property var icons: [] + + QtObject { + id: d + + function lastTwoItems(nodes) { + return nodes.join(qsTr(" and ")); + } + + function showReactionAuthors(jsonArrayOfUsersReactedWithThisEmoji, emojiId) { + const listOfUsers = JSON.parse(jsonArrayOfUsersReactedWithThisEmoji) + if (listOfUsers.error) { + console.error("error parsing users who reacted to a message, error: ", obj.error) + return + } + + let author; + if (listOfUsers.length === 1) { + author = listOfUsers[0] + } else if (listOfUsers.length === 2) { + author = lastTwoItems(listOfUsers); + } else { + var leftNode = []; + var rightNode = []; + const maxReactions = 12 + let maximum = Math.min(maxReactions, listOfUsers.length) + + if (listOfUsers.length > maxReactions) { + leftNode = listOfUsers.slice(0, maxReactions); + rightNode = listOfUsers.slice(maxReactions, listOfUsers.length); + return (rightNode.length === 1) ? + lastTwoItems([leftNode.join(", "), rightNode[0]]) : + lastTwoItems([leftNode.join(", "), qsTr("%1 more").arg(rightNode.length)]); + } + + leftNode = listOfUsers.slice(0, maximum - 1); + rightNode = listOfUsers.slice(maximum - 1, listOfUsers.length); + author = lastTwoItems([leftNode.join(", "), rightNode[0]]) + } + return qsTr("%1 reacted with %2") + .arg(author) + .arg(Emoji.getEmojiFromId(emojiId)); + } + } + + Row { + spacing: root.imageMargin + + Repeater { + id: reactionRepeater + width: childrenRect.width + model: root.emojiReactionsModel + + Rectangle { + id: emojiContainer + + readonly property bool isHovered: mouseArea.containsMouse + + width: emojiImage.width + emojiCount.width + (root.imageMargin * 2) + + 8 + height: 20 + radius: 10 + color: model.didIReactWithThisEmoji ? + (isHovered ? Theme.palette.statusMessage.emojiReactionActiveBackgroundHovered : Theme.palette.statusMessage.emojiReactionActiveBackground) : + (isHovered ? Theme.palette.statusMessage.emojiReactionBackgroundHovered : Theme.palette.statusMessage.emojiReactionBackground) + + StatusToolTip { + visible: mouseArea.containsMouse + maxWidth: 400 + text: d.showReactionAuthors(model.jsonArrayOfUsersReactedWithThisEmoji, model.emojiId) + } + + // Rounded corner to cover one corner + Rectangle { + color: parent.color + width: 10 + height: 10 + anchors.top: parent.top + anchors.left: !root.isCurrentUser ? parent.left : undefined + anchors.leftMargin: 0 + anchors.right: !root.isCurrentUser ? undefined : parent.right + anchors.rightMargin: 0 + radius: 2 + z: -1 + } + + // This is a workaround to get a "border" around the rectangle including the weird rectangle + Loader { + active: model.didIReactWithThisEmoji + anchors.top: parent.top + anchors.topMargin: -1 + anchors.left: parent.left + anchors.leftMargin: -1 + z: -2 + + sourceComponent: Component { + Rectangle { + width: emojiContainer.width + 2 + height: emojiContainer.height + 2 + radius: emojiContainer.radius + color: Theme.palette.primaryColor1 + + Rectangle { + color: parent.color + width: 10 + height: 10 + anchors.top: parent.top + anchors.left: !root.isCurrentUser ? parent.left : undefined + anchors.leftMargin: 0 + anchors.right: !root.isCurrentUser ? undefined : parent.right + anchors.rightMargin: 0 + radius: 2 + z: -1 + } + } + } + } + + // TODO: Use Row + + StatusEmoji { + id: emojiImage + + anchors.verticalCenter: parent.verticalCenter + anchors.left: parent.left + anchors.leftMargin: root.imageMargin + + width: 15 + height: 15 + + source: { + if (model.emojiId >= 1 && model.emojiId <= root.icons.length) + return root.icons[model.emojiId - 1]; + return ""; + } + } + + StatusBaseText { + id: emojiCount + text: model.numberOfReactions + anchors.verticalCenter: parent.verticalCenter + anchors.left: emojiImage.right + anchors.leftMargin: root.imageMargin + font.pixelSize: 12 + color: model.didIReactWithThisEmoji ? Theme.palette.primaryColor1 : Theme.palette.directColor1 + } + + MouseArea { + id: mouseArea + anchors.fill: parent + hoverEnabled: true + cursorShape: Qt.PointingHandCursor + onEntered: { + root.hoverChanged(true) + } + onExited: { + root.hoverChanged(false) + } + onClicked: { + root.toggleReaction(model.emojiId) + } + } + } + } + + Item { + width: addEmojiButton.width + addEmojiButton.anchors.leftMargin // there is more margin between the button and the emojis than between each emoji + height: addEmojiButton.height + + StatusIcon { + id: addEmojiButton + + property bool isHovered: false // TODO: Replace with mouseArea.containsMouse + + anchors.left: parent.left + anchors.leftMargin: 2.5 + + icon: "reaction-b" + width: 16.5 + height: 16.5 + + color: addEmojiButton.isHovered ? Theme.palette.primaryColor1 : Theme.palette.baseColor1 + } + + MouseArea { + id: addEmojiButtonMouseArea + anchors.fill: addEmojiButton + cursorShape: Qt.PointingHandCursor + hoverEnabled: true + onEntered: addEmojiButton.isHovered = true + onExited: addEmojiButton.isHovered = false + onClicked: { + root.addEmojiClicked(this, mouse); + } + } + + StatusToolTip { + visible: addEmojiButton.isHovered + text: qsTr("Add reaction") + } + } + } +} diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageHeader.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageHeader.qml index 1cd904c71f..bede3c8b20 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageHeader.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageHeader.qml @@ -3,36 +3,39 @@ import QtQuick.Layouts 1.14 import StatusQ.Core 0.1 import StatusQ.Core.Theme 0.1 +import StatusQ.Core.Utils 0.1 import StatusQ.Components 0.1 import StatusQ.Controls 0.1 Item { - id: statusMessageHeader + id: root + + property StatusMessageSenderDetails sender: StatusMessageSenderDetails { } property alias displayNameLabel: primaryDisplayName property alias secondaryNameLabel: secondaryDisplayName property alias tertiaryDetailsLabel: tertiaryDetailText property alias timestamp: timestampText - property string displayName: "" - property string secondaryName: "" - property string tertiaryDetail: "" + property string tertiaryDetail: sender.id property string resendText: "" property bool showResendButton: false - property bool isContact: false - property var trustIndicator: StatusContactVerificationIcons.TrustedType.None + property bool isContact: sender.isContact + property int trustIndicator: sender.trustIndicator + property bool amISender: false - signal clicked() + signal clicked(var sender, var mouse) signal resendClicked() - height: childrenRect.height - width: primaryDisplayName.width + (secondaryDisplayName.visible ? secondaryDisplayName.width + header.spacing : 0) + implicitHeight: layout.implicitHeight + implicitWidth: layout.implicitWidth RowLayout { - id: header + id: layout spacing: 4 TextEdit { id: primaryDisplayName + Layout.alignment: Qt.AlignBottom font.family: Theme.palette.baseFont.name font.weight: Font.Medium font.pixelSize: 15 @@ -41,7 +44,7 @@ Item { wrapMode: Text.WordWrap selectByMouse: true color: Theme.palette.primaryColor1 - text: displayName + text: root.amISender ? qsTr("You") : root.sender.displayName MouseArea { id: mouseArea anchors.fill: parent @@ -49,47 +52,45 @@ Item { acceptedButtons: Qt.LeftButton | Qt.RightButton hoverEnabled: true onClicked: { - statusMessageHeader.clicked() + root.clicked(this, mouse) } } - Layout.alignment: Qt.AlignBottom } StatusContactVerificationIcons { - isContact: statusMessageHeader.isContact - trustIndicator: statusMessageHeader.trustIndicator + visible: !root.amISender + isContact: root.isContact + trustIndicator: root.trustIndicator } StatusBaseText { id: secondaryDisplayName Layout.alignment: Qt.AlignVCenter + visible: !root.amISender && !!root.sender.secondaryName color: Theme.palette.baseColor1 font.pixelSize: 10 - text: secondaryName - visible: !!text + text: `(${root.sender.secondaryName})` } StatusBaseText { - id: dotSeparator1 - Layout.fillHeight: true + Layout.alignment: Qt.AlignVCenter + visible: secondaryDisplayName.visible font.pixelSize: 10 color: Theme.palette.baseColor1 - text: "." - visible: secondaryDisplayName.visible + text: "•" } StatusBaseText { id: tertiaryDetailText + visible: !root.amISender Layout.alignment: Qt.AlignVCenter - Layout.maximumWidth: 58 font.pixelSize: 10 elide: Text.ElideMiddle color: Theme.palette.baseColor1 - text: tertiaryDetail + text: Utils.elideText(tertiaryDetail, 5, 3) } StatusBaseText { - id: dotSeparator2 - Layout.fillHeight: true + Layout.alignment: Qt.AlignVCenter + visible: tertiaryDetailText.visible font.pixelSize: 10 color: Theme.palette.baseColor1 - text: "." - visible: tertiaryDetailText.visible + text: "•" } StatusTimeStampLabel { id: timestampText @@ -98,12 +99,12 @@ Item { Layout.alignment: Qt.AlignVCenter color: Theme.palette.dangerColor1 font.pixelSize: 12 - text: statusMessageHeader.resendText + text: root.resendText visible: showResendButton && !!timestampText.text MouseArea { cursorShape: Qt.PointingHandCursor anchors.fill: parent - onClicked: statusMessageHeader.resendClicked() + onClicked: root.resendClicked() } } } diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageQuickActions.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageQuickActions.qml index ee2bd6e373..dcd0e58b38 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageQuickActions.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageQuickActions.qml @@ -6,29 +6,29 @@ import StatusQ.Controls 0.1 import StatusQ.Core.Theme 0.1 Rectangle { - id: buttonsContainer + id: root - property list quickActions + property list items QtObject { id: _internal readonly property int containerMargin: 2 } - width: buttonRow.width + _internal.containerMargin * 2 - height: 36 + implicitWidth: buttonRow.width + _internal.containerMargin * 2 + implicitHeight: 36 radius: 8 color: Theme.palette.statusSelect.menuItemBackgroundColor layer.enabled: true layer.effect: DropShadow { - width: buttonsContainer.width - height: buttonsContainer.height - x: buttonsContainer.x - y: buttonsContainer.y + 10 + width: root.width + height: root.height + x: root.x + y: root.y + 10 horizontalOffset: 0 verticalOffset: 2 - source: buttonsContainer + source: root radius: 10 samples: 15 color: Theme.palette.dropShadow @@ -39,13 +39,13 @@ Rectangle { spacing: _internal.containerMargin anchors.left: parent.left anchors.leftMargin: _internal.containerMargin - anchors.verticalCenter: buttonsContainer.verticalCenter + anchors.verticalCenter: root.verticalCenter height: parent.height - 2 * _internal.containerMargin } - onQuickActionsChanged: { - for (let idx in quickActions) { - quickActions[idx].parent = buttonRow + onItemsChanged: { + for (let idx in items) { + items[idx].parent = buttonRow } } } diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageReply.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageReply.qml index ef250133c1..9c1d5c78e8 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageReply.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusMessageReply.qml @@ -7,18 +7,19 @@ import StatusQ.Core 0.1 import StatusQ.Core.Theme 0.1 import StatusQ.Components 0.1 -Loader { - id: chatReply +Item { + id: root property StatusMessageDetails replyDetails property string audioMessageInfoText: "" - signal replyProfileClicked() + signal replyProfileClicked(var sender, var mouse) - active: visible + implicitHeight: layout.implicitHeight + implicitWidth: layout.implicitWidth - sourceComponent: RowLayout { - id: replyLayout + RowLayout { + id: layout spacing: 8 Shape { id: replyCorner @@ -56,13 +57,16 @@ Loader { StatusSmartIdenticon { id: profileImage Layout.alignment: Qt.AlignTop - image: replyDetails.profileImage - name: replyDetails.displayName + name: replyDetails.sender.userName + image: replyDetails.sender.profileImage.imageSettings + icon: replyDetails.sender.profileImage.iconSettings + ringSettings: replyDetails.sender.profileImage.ringSettings + MouseArea { cursorShape: Qt.PointingHandCursor acceptedButtons: Qt.LeftButton | Qt.RightButton anchors.fill: parent - onClicked: replyProfileClicked() + onClicked: replyProfileClicked(this, mouse) } } TextEdit { @@ -74,7 +78,7 @@ Loader { font.weight: Font.Medium selectByMouse: true readOnly: true - text: replyDetails.displayName + text: replyDetails.amISender ? qsTr("You") : replyDetails.sender.displayName } } StatusTextMessage { @@ -85,13 +89,14 @@ Loader { textField.height: 18 clip: true visible: !!replyDetails.messageText + allowShowMore: false } StatusImageMessage { Layout.fillWidth: true Layout.preferredHeight: imageAlias.paintedHeight imageWidth: 56 source: replyDetails.contentType === StatusMessage.ContentType.Image ? replyDetails.messageContent : "" - visible: replyDetails.contentType === StatusMessage.ContentType.Image +// visible: replyDetails.contentType === StatusMessage.ContentType.Image shapeType: StatusImageMessage.ShapeType.ROUNDED } Item { @@ -116,7 +121,7 @@ Loader { height: 22 isPreview: true audioSource: replyDetails.messageContent - audioMessageInfoText: chatReply.audioMessageInfoText + audioMessageInfoText: root.audioMessageInfoText } } } diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusPinMessageDetails.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusPinMessageDetails.qml index 93b89762a3..4a2e56d062 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusPinMessageDetails.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusPinMessageDetails.qml @@ -1,4 +1,5 @@ import QtQuick 2.13 +import QtQuick.Controls 2.14 import QtQuick.Layouts 1.14 import QtGraphicalEffects 1.13 @@ -11,14 +12,34 @@ Loader { active: visible - sourceComponent: Rectangle { - height: 24 - width: layout.width + 16 - color: Theme.palette.pinColor2 - radius: 12 - RowLayout { - id: layout - anchors.centerIn: parent + sourceComponent: Control { + verticalPadding: 3 + leftPadding: 2 + rightPadding: 6 + + background: Rectangle { + readonly property color translucentColor: Theme.palette.pinColor2 + + implicitWidth: 24 + implicitHeight: 24 + color: Qt.rgba(translucentColor.r, + translucentColor.g, + translucentColor.b, 1) + opacity: translucentColor.a + layer.enabled: true + radius: 12 + + Rectangle { + anchors.bottom: parent.bottom + anchors.left: parent.left + width: parent.width / 2 + height: parent.height / 2 + color: parent.color + radius: 4 + } + } + + contentItem: RowLayout { StatusIcon { Layout.alignment: Qt.AlignVCenter Layout.preferredWidth: 16 diff --git a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusTextMessage.qml b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusTextMessage.qml index b9ed294f57..d1e2994177 100644 --- a/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusTextMessage.qml +++ b/ui/StatusQ/src/StatusQ/Components/private/statusMessage/StatusTextMessage.qml @@ -6,24 +6,33 @@ import StatusQ.Controls 0.1 import StatusQ.Core.Theme 0.1 Item { - id: textMessage + id: root - property int contentType: 0 property alias textField: chatText + property bool allowShowMore: true - signal linkActivated(url link) + signal linkActivated(string link) - implicitHeight: showMoreLoader.active ? childrenRect.height : chatText.height + implicitWidth: chatText.implicitWidth + implicitHeight: chatText.effectiveHeight + d.showMoreHeight QtObject { - id: _internal + id: d property bool readMore: false - property bool veryLongChatText: chatText.length > 1000 + readonly property bool veryLongChatText: chatText.length > 1000 + readonly property int showMoreHeight: showMoreLoader.visible ? showMoreLoader.height : 0 } TextEdit { id: chatText - visible: !showMoreLoader.active || _internal.readMore + + readonly property int effectiveHeight: d.veryLongChatText && !d.readMore ? Math.min(chatText.implicitHeight, 200) + : chatText.implicitHeight + + width: parent.width + height: effectiveHeight + d.showMoreHeight / 2 + visible: !opMask.active + clip: true selectedTextColor: Theme.palette.directColor1 selectionColor: Theme.palette.primaryColor3 color: Theme.palette.directColor1 @@ -33,12 +42,12 @@ Item { wrapMode: Text.Wrap readOnly: true selectByMouse: true - height: _internal.veryLongChatText && !_internal.readMore ? Math.min(implicitHeight, 200) : implicitHeight - width: parent.width - clip: height < implicitHeight - onLinkActivated: textMessage.linkActivated(link) + onLinkActivated: { + root.linkActivated(link); + } onLinkHovered: { - cursorShape: Qt.PointingHandCursor + // Strange thing. Without this empty stub the cursorShape + // is not changed to pointingHandCursor. } } @@ -60,7 +69,7 @@ Item { Loader { id: opMask - active: showMoreLoader.active && !_internal.readMore + active: showMoreLoader.active && !d.readMore anchors.fill: chatText sourceComponent: OpacityMask { source: chatText @@ -70,17 +79,17 @@ Item { Loader { id: showMoreLoader - active: _internal.veryLongChatText - anchors.top: chatText.bottom - anchors.topMargin: -10 + active: root.allowShowMore && d.veryLongChatText + visible: active + anchors.verticalCenter: chatText.bottom anchors.horizontalCenter: parent.horizontalCenter sourceComponent: StatusRoundButton { implicitWidth: 24 implicitHeight: 24 type: StatusRoundButton.Type.Secondary - icon.name: _internal.readMore ? "chevron-up": "chevron-down" + icon.name: d.readMore ? "chevron-up": "chevron-down" onClicked: { - _internal.readMore = !_internal.readMore + d.readMore = !d.readMore } } } diff --git a/ui/StatusQ/src/StatusQ/Components/qmldir b/ui/StatusQ/src/StatusQ/Components/qmldir index 2a7cdc3ad4..eee86b5f98 100644 --- a/ui/StatusQ/src/StatusQ/Components/qmldir +++ b/ui/StatusQ/src/StatusQ/Components/qmldir @@ -13,6 +13,7 @@ StatusChatToolBar 0.1 StatusChatToolBar.qml StatusContactRequestsIndicatorListItem 0.1 StatusContactRequestsIndicatorListItem.qml StatusEmoji 0.1 StatusEmoji.qml StatusContactVerificationIcons 0.1 StatusContactVerificationIcons.qml +StatusDateGroupLabel 0.1 StatusDateGroupLabel.qml StatusDescriptionListItem 0.1 StatusDescriptionListItem.qml StatusLetterIdenticon 0.1 StatusLetterIdenticon.qml StatusListItem 0.1 StatusListItem.qml @@ -31,6 +32,7 @@ StatusExpandableItem 0.1 StatusExpandableItem.qml StatusSmartIdenticon 0.1 StatusSmartIdenticon.qml StatusMessage 0.1 StatusMessage.qml StatusMessageDetails 0.1 StatusMessageDetails.qml +StatusMessageSenderDetails 0.1 StatusMessageSenderDetails.qml StatusTagSelector 0.1 StatusTagSelector.qml StatusToastMessage 0.1 StatusToastMessage.qml StatusWizardStepper 0.1 StatusWizardStepper.qml diff --git a/ui/StatusQ/src/StatusQ/Core/StatusProfileImageSettings.qml b/ui/StatusQ/src/StatusQ/Core/StatusProfileImageSettings.qml new file mode 100644 index 0000000000..5da49f68aa --- /dev/null +++ b/ui/StatusQ/src/StatusQ/Core/StatusProfileImageSettings.qml @@ -0,0 +1,40 @@ +import QtQuick 2.0 +import StatusQ.Core.Theme 0.1 + +QtObject { + id: root + + property url source + property int width + property int height + property bool isIdenticon: false + + property string name + property string pubkey + property string image + property bool showRing: true + property bool interactive: true + + property int colorId // TODO: default value Utils.colorIdForPubkey(pubkey) + property var colorHash // TODO: default value Utils.getColorHashAsJson(pubkey) + + property StatusImageSettings imageSettings: StatusImageSettings { + width: root.width + height: root.height + source: root.source + } + + readonly property StatusIconSettings iconSettings: StatusIconSettings { + width: root.width + height: root.height + color: Theme.palette.userCustomizationColors[root.colorId] + charactersLen: 2 + } + + readonly property StatusIdenticonRingSettings ringSettings: StatusIdenticonRingSettings { + initalAngleRad: 0 + ringPxSize: Math.max(1.5, root.width / 24.0) + ringSpecModel: root.showRing ? root.colorHash : undefined + distinctiveColors: Theme.palette.identiconRingColors + } +} diff --git a/ui/StatusQ/src/StatusQ/Core/Theme/StatusDarkTheme.qml b/ui/StatusQ/src/StatusQ/Core/Theme/StatusDarkTheme.qml index 76062b19e8..efbd1a8d45 100644 --- a/ui/StatusQ/src/StatusQ/Core/Theme/StatusDarkTheme.qml +++ b/ui/StatusQ/src/StatusQ/Core/Theme/StatusDarkTheme.qml @@ -154,5 +154,12 @@ ThemePalette { property color menuItemBackgroundColor: baseColor2 property color menuItemHoverBackgroundColor: directColor7 } + + property QtObject statusMessage: QtObject { + property color emojiReactionBackground: "#2d2823" + property color emojiReactionBackgroundHovered: "#3a3632" + property color emojiReactionActiveBackground: "#353a4d" + property color emojiReactionActiveBackgroundHovered: "#cbd5f1" + } } diff --git a/ui/StatusQ/src/StatusQ/Core/Theme/StatusLightTheme.qml b/ui/StatusQ/src/StatusQ/Core/Theme/StatusLightTheme.qml index 7e143dc04b..e058c309c5 100644 --- a/ui/StatusQ/src/StatusQ/Core/Theme/StatusLightTheme.qml +++ b/ui/StatusQ/src/StatusQ/Core/Theme/StatusLightTheme.qml @@ -152,5 +152,12 @@ ThemePalette { property color menuItemBackgroundColor: white property color menuItemHoverBackgroundColor: baseColor2 } + + property QtObject statusMessage: QtObject { + property color emojiReactionBackground: "#e2e6e9" + property color emojiReactionBackgroundHovered: "#d7dadd" + property color emojiReactionActiveBackground: getColor('blue6') + property color emojiReactionActiveBackgroundHovered: "#cbd5f1" + } } diff --git a/ui/StatusQ/src/StatusQ/Core/Theme/ThemePalette.qml b/ui/StatusQ/src/StatusQ/Core/Theme/ThemePalette.qml index c3cbea55eb..a6963dedcd 100644 --- a/ui/StatusQ/src/StatusQ/Core/Theme/ThemePalette.qml +++ b/ui/StatusQ/src/StatusQ/Core/Theme/ThemePalette.qml @@ -243,7 +243,8 @@ QtObject { } function getColor(name, alpha) { - return !!alpha ? alphaColor(StatusColors.colors[name], alpha) : StatusColors.colors[name] + return !!alpha ? alphaColor(StatusColors.colors[name], alpha) + : StatusColors.colors[name] } } diff --git a/ui/StatusQ/src/StatusQ/Core/Utils/Utils.qml b/ui/StatusQ/src/StatusQ/Core/Utils/Utils.qml index 270a5b9d0e..e8f8317dd1 100644 --- a/ui/StatusQ/src/StatusQ/Core/Utils/Utils.qml +++ b/ui/StatusQ/src/StatusQ/Core/Utils/Utils.qml @@ -1,6 +1,8 @@ pragma Singleton import QtQuick 2.13 +import StatusQ.Core.Theme 0.1 +import "./xss.js" as XSS QtObject { @@ -151,6 +153,71 @@ QtObject { } } + function linkifyAndXSS(inputText) { + //URLs starting with http://, https://, or ftp:// + var replacePattern1 = /(\b(https?|ftp|statusim):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/gim; + var replacedText = inputText.replace(replacePattern1, "$1"); + + //URLs starting with "www." (without // before it, or it'd re-link the ones done above). + var replacePattern2 = /(^|[^\/])(www\.[\S]+(\b|$))/gim; + replacedText = replacedText.replace(replacePattern2, "$1$2"); + + return XSS.filterXSS(replacedText) + } + + function filterXSS(inputText) { + return XSS.filterXSS(inputText) + } + + function getMessageWithStyle(msg, hoveredLink = "") { + return `` + + `${msg}` + } + function delegateModelSort(srcGroup, dstGroup, lessThan) { const insertPosition = (lessThan, item) => { let lower = 0 @@ -173,6 +240,10 @@ QtObject { dstGroup.move(item.itemsIndex, index) } } + + function elideText(text, leftCharsCount, rightCharsCount = leftCharsCount) { + return text.substr(0, leftCharsCount) + "..." + text.substr(text.length - rightCharsCount) + } } diff --git a/ui/StatusQ/src/StatusQ/Core/Utils/qmldir b/ui/StatusQ/src/StatusQ/Core/Utils/qmldir index e45fcce791..cf96ab85d9 100644 --- a/ui/StatusQ/src/StatusQ/Core/Utils/qmldir +++ b/ui/StatusQ/src/StatusQ/Core/Utils/qmldir @@ -1,4 +1,6 @@ module StatusQ.Core.Utils +EmojiJSON 1.0 emojiList.js +XSS 1.0 xss.js singleton Utils 0.1 Utils.qml singleton Emoji 0.1 Emoji.qml diff --git a/ui/StatusQ/src/StatusQ/Core/Utils/xss.js b/ui/StatusQ/src/StatusQ/Core/Utils/xss.js new file mode 100644 index 0000000000..f502efa36f --- /dev/null +++ b/ui/StatusQ/src/StatusQ/Core/Utils/xss.js @@ -0,0 +1,1318 @@ +/** + * NOTICE: + * + * Most this code was copied from https://github.com/leizongmin/js-xss and slightly modifed + * to work with JavaScript resource loading in QML. + */ + +var defaultCSSFilter = new FilterCSS(); + +var DEFAULT = { + whiteList: getDefaultWhiteList(), + getDefaultWhiteList, + onTag, + onIgnoreTag, + onTagAttr, + onIgnoreTagAttr, + safeAttrValue, + escapeHtml, + escapeQuote, + unescapeQuote, + escapeHtmlEntities, + escapeDangerHtml5Entities, + clearNonPrintableCharacter, + friendlyAttrValue, + escapeAttrValue, + onIgnoreTagStripAll, + StripTagBody, + stripCommentTag , + stripBlankChar, + cssFilter: defaultCSSFilter, + getDefaultCSSWhiteList, +} + +/** + * shallow copy + * + * @param {Object} obj + * @return {Object} + */ +function shallowCopyObject(obj) { + var ret = {}; + for (var i in obj) { + ret[i] = obj[i]; + } + return ret; +} + + +var _ = { + indexOf: function(arr, item) { + var i, j; + if (Array.prototype.indexOf) { + return arr.indexOf(item); + } + for (i = 0, j = arr.length; i < j; i++) { + if (arr[i] === item) { + return i; + } + } + return -1; + }, + forEach: function(arr, fn, scope) { + var i, j; + if (Array.prototype.forEach) { + return arr.forEach(fn, scope); + } + for (i = 0, j = arr.length; i < j; i++) { + fn.call(scope, arr[i], i, arr); + } + }, + trim: function(str) { + if (String.prototype.trim) { + return str.trim(); + } + return str.replace(/(^\s*)|(\s*$)/g, ""); + }, + spaceIndex: function(str) { + var reg = /\s|\n|\t/; + var match = reg.exec(str); + return match ? match.index : -1; + } +}; + +function filterXSS(html, options) { + var xss = new FilterXSS(options); + return xss.process(html); +} + +function onAttr (name, value, options) { + // do nothing +} + +function onIgnoreAttr (name, value, options) { + // do nothing +} + +/** + * FilterXSS class + * + * @param {Object} options + * whiteList, onTag, onTagAttr, onIgnoreTag, + * onIgnoreTagAttr, safeAttrValue, escapeHtml + * stripIgnoreTagBody, allowCommentTag, stripBlankChar + * css{whiteList, onAttr, onIgnoreAttr} `css=false` means don't use `cssfilter` + */ +function FilterXSS(options) { + options = shallowCopyObject(options || {}); + + if (options.stripIgnoreTag) { + if (options.onIgnoreTag) { + console.error( + 'Notes: cannot use these two options "stripIgnoreTag" and "onIgnoreTag" at the same time' + ); + } + options.onIgnoreTag = onIgnoreTagStripAll; + } + + options.whiteList = options.whiteList || getDefaultWhiteList(); + options.onTag = options.onTag || onTag; + options.onTagAttr = options.onTagAttr || onTagAttr; + options.onIgnoreTag = options.onIgnoreTag || onIgnoreTag; + options.onIgnoreTagAttr = options.onIgnoreTagAttr || onIgnoreTagAttr; + options.safeAttrValue = options.safeAttrValue || safeAttrValue; + options.escapeHtml = options.escapeHtml || escapeHtml; + this.options = options; + + if (options.css === false) { + this.cssFilter = false; + } else { + options.css = options.css || {}; + this.cssFilter = new FilterCSS(options.css); + } +} + +/** + * start process and returns result + * + * @param {String} html + * @return {String} + */ +FilterXSS.prototype.process = function(html) { + // compatible with the input + html = html || ""; + html = html.toString(); + if (!html) return ""; + + var me = this; + var options = me.options; + var whiteList = options.whiteList; + var onTag = options.onTag; + var onIgnoreTag = options.onIgnoreTag; + var onTagAttr = options.onTagAttr; + var onIgnoreTagAttr = options.onIgnoreTagAttr; + var safeAttrValue = options.safeAttrValue; + var escapeHtml = options.escapeHtml; + var cssFilter = me.cssFilter; + + // remove invisible characters + if (options.stripBlankChar) { + html = stripBlankChar(html); + } + + // remove html comments + if (!options.allowCommentTag) { + html = stripCommentTag(html); + } + + // if enable stripIgnoreTagBody + var stripIgnoreTagBody = false; + if (options.stripIgnoreTagBody) { + var stripIgnoreTagBody = StripTagBody( + options.stripIgnoreTagBody, + onIgnoreTag + ); + onIgnoreTag = stripIgnoreTagBody.onIgnoreTag; + } + + var retHtml = parseTag( + html, + function(sourcePosition, position, tag, html, isClosing) { + var info = { + sourcePosition: sourcePosition, + position: position, + isClosing: isClosing, + isWhite: whiteList.hasOwnProperty(tag) + }; + + // call `onTag()` + var ret = onTag(tag, html, info); + if (!isNull(ret)) return ret; + + if (info.isWhite) { + if (info.isClosing) { + return ""; + } + + var attrs = getAttrs(html); + var whiteAttrList = whiteList[tag]; + var attrsHtml = parseAttr(attrs.html, function(name, value) { + // call `onTagAttr()` + var isWhiteAttr = _.indexOf(whiteAttrList, name) !== -1; + var ret = onTagAttr(tag, name, value, isWhiteAttr); + if (!isNull(ret)) return ret; + + if (isWhiteAttr) { + // call `safeAttrValue()` + value = safeAttrValue(tag, name, value, cssFilter); + if (value) { + return name + '="' + value + '"'; + } else { + return name; + } + } else { + // call `onIgnoreTagAttr()` + var ret = onIgnoreTagAttr(tag, name, value, isWhiteAttr); + if (!isNull(ret)) return ret; + return; + } + }); + + // build new tag html + var html = "<" + tag; + if (attrsHtml) html += " " + attrsHtml; + if (attrs.closing) html += " /"; + html += ">"; + return html; + } else { + // call `onIgnoreTag()` + var ret = onIgnoreTag(tag, html, info); + if (!isNull(ret)) return ret; + return escapeHtml(html); + } + }, + escapeHtml + ); + + // if enable stripIgnoreTagBody + if (stripIgnoreTagBody) { + retHtml = stripIgnoreTagBody.remove(retHtml); + } + + return retHtml; +}; + + +function getDefaultWhiteList() { + return { + a: ["target", "href", "title", "class"], + abbr: ["title"], + address: [], + area: ["shape", "coords", "href", "alt"], + article: [], + aside: [], + audio: ["autoplay", "controls", "loop", "preload", "src"], + b: [], + bdi: ["dir"], + bdo: ["dir"], + big: [], + blockquote: ["cite"], + br: [], + caption: [], + center: [], + cite: [], + code: [], + col: ["align", "valign", "span", "width"], + colgroup: ["align", "valign", "span", "width"], + dd: [], + del: ["datetime"], + details: ["open"], + div: [], + dl: [], + dt: [], + em: [], + font: ["color", "size", "face"], + footer: [], + h1: [], + h2: [], + h3: [], + h4: [], + h5: [], + h6: [], + header: [], + hr: [], + i: [], + img: ["src", "alt", "title", "width", "height"], + ins: ["datetime"], + li: [], + mark: [], + nav: [], + ol: [], + p: [], + pre: [], + s: [], + section: [], + small: [], + span: [], + sub: [], + sup: [], + strong: [], + table: ["width", "height", "border", "bgcolor", "cellspacing", "cellpadding", "class"], + tbody: ["align", "valign"], + td: ["width", "bgcolor", "rowspan", "colspan", "align", "valign", "class"], + tfoot: ["align", "valign"], + th: ["width", "rowspan", "colspan", "align", "valign"], + thead: ["align", "valign"], + tr: ["rowspan", "align", "valign"], + tt: [], + u: [], + ul: [], + video: ["autoplay", "controls", "loop", "preload", "src", "height", "width"] + }; +} + +function getDefaultCSSWhiteList () { + // 白名单值说明: + // true: 允许该属性 + // Function: function (val) { } 返回true表示允许该属性,其他值均表示不允许 + // RegExp: regexp.test(val) 返回true表示允许该属性,其他值均表示不允许 + // 除上面列出的值外均表示不允许 + var whiteList = {}; + + whiteList['align-content'] = false; // default: auto + whiteList['align-items'] = false; // default: auto + whiteList['align-self'] = false; // default: auto + whiteList['alignment-adjust'] = false; // default: auto + whiteList['alignment-baseline'] = false; // default: baseline + whiteList['all'] = false; // default: depending on individual properties + whiteList['anchor-point'] = false; // default: none + whiteList['animation'] = false; // default: depending on individual properties + whiteList['animation-delay'] = false; // default: 0 + whiteList['animation-direction'] = false; // default: normal + whiteList['animation-duration'] = false; // default: 0 + whiteList['animation-fill-mode'] = false; // default: none + whiteList['animation-iteration-count'] = false; // default: 1 + whiteList['animation-name'] = false; // default: none + whiteList['animation-play-state'] = false; // default: running + whiteList['animation-timing-function'] = false; // default: ease + whiteList['azimuth'] = false; // default: center + whiteList['backface-visibility'] = false; // default: visible + whiteList['background'] = true; // default: depending on individual properties + whiteList['background-attachment'] = true; // default: scroll + whiteList['background-clip'] = true; // default: border-box + whiteList['background-color'] = true; // default: transparent + whiteList['background-image'] = true; // default: none + whiteList['background-origin'] = true; // default: padding-box + whiteList['background-position'] = true; // default: 0% 0% + whiteList['background-repeat'] = true; // default: repeat + whiteList['background-size'] = true; // default: auto + whiteList['baseline-shift'] = false; // default: baseline + whiteList['binding'] = false; // default: none + whiteList['bleed'] = false; // default: 6pt + whiteList['bookmark-label'] = false; // default: content() + whiteList['bookmark-level'] = false; // default: none + whiteList['bookmark-state'] = false; // default: open + whiteList['border'] = true; // default: depending on individual properties + whiteList['border-bottom'] = true; // default: depending on individual properties + whiteList['border-bottom-color'] = true; // default: current color + whiteList['border-bottom-left-radius'] = true; // default: 0 + whiteList['border-bottom-right-radius'] = true; // default: 0 + whiteList['border-bottom-style'] = true; // default: none + whiteList['border-bottom-width'] = true; // default: medium + whiteList['border-collapse'] = true; // default: separate + whiteList['border-color'] = true; // default: depending on individual properties + whiteList['border-image'] = true; // default: none + whiteList['border-image-outset'] = true; // default: 0 + whiteList['border-image-repeat'] = true; // default: stretch + whiteList['border-image-slice'] = true; // default: 100% + whiteList['border-image-source'] = true; // default: none + whiteList['border-image-width'] = true; // default: 1 + whiteList['border-left'] = true; // default: depending on individual properties + whiteList['border-left-color'] = true; // default: current color + whiteList['border-left-style'] = true; // default: none + whiteList['border-left-width'] = true; // default: medium + whiteList['border-radius'] = true; // default: 0 + whiteList['border-right'] = true; // default: depending on individual properties + whiteList['border-right-color'] = true; // default: current color + whiteList['border-right-style'] = true; // default: none + whiteList['border-right-width'] = true; // default: medium + whiteList['border-spacing'] = true; // default: 0 + whiteList['border-style'] = true; // default: depending on individual properties + whiteList['border-top'] = true; // default: depending on individual properties + whiteList['border-top-color'] = true; // default: current color + whiteList['border-top-left-radius'] = true; // default: 0 + whiteList['border-top-right-radius'] = true; // default: 0 + whiteList['border-top-style'] = true; // default: none + whiteList['border-top-width'] = true; // default: medium + whiteList['border-width'] = true; // default: depending on individual properties + whiteList['bottom'] = false; // default: auto + whiteList['box-decoration-break'] = true; // default: slice + whiteList['box-shadow'] = true; // default: none + whiteList['box-sizing'] = true; // default: content-box + whiteList['box-snap'] = true; // default: none + whiteList['box-suppress'] = true; // default: show + whiteList['break-after'] = true; // default: auto + whiteList['break-before'] = true; // default: auto + whiteList['break-inside'] = true; // default: auto + whiteList['caption-side'] = false; // default: top + whiteList['chains'] = false; // default: none + whiteList['clear'] = true; // default: none + whiteList['clip'] = false; // default: auto + whiteList['clip-path'] = false; // default: none + whiteList['clip-rule'] = false; // default: nonzero + whiteList['color'] = true; // default: implementation dependent + whiteList['color-interpolation-filters'] = true; // default: auto + whiteList['column-count'] = false; // default: auto + whiteList['column-fill'] = false; // default: balance + whiteList['column-gap'] = false; // default: normal + whiteList['column-rule'] = false; // default: depending on individual properties + whiteList['column-rule-color'] = false; // default: current color + whiteList['column-rule-style'] = false; // default: medium + whiteList['column-rule-width'] = false; // default: medium + whiteList['column-span'] = false; // default: none + whiteList['column-width'] = false; // default: auto + whiteList['columns'] = false; // default: depending on individual properties + whiteList['contain'] = false; // default: none + whiteList['content'] = false; // default: normal + whiteList['counter-increment'] = false; // default: none + whiteList['counter-reset'] = false; // default: none + whiteList['counter-set'] = false; // default: none + whiteList['crop'] = false; // default: auto + whiteList['cue'] = false; // default: depending on individual properties + whiteList['cue-after'] = false; // default: none + whiteList['cue-before'] = false; // default: none + whiteList['cursor'] = false; // default: auto + whiteList['direction'] = false; // default: ltr + whiteList['display'] = true; // default: depending on individual properties + whiteList['display-inside'] = true; // default: auto + whiteList['display-list'] = true; // default: none + whiteList['display-outside'] = true; // default: inline-level + whiteList['dominant-baseline'] = false; // default: auto + whiteList['elevation'] = false; // default: level + whiteList['empty-cells'] = false; // default: show + whiteList['filter'] = false; // default: none + whiteList['flex'] = false; // default: depending on individual properties + whiteList['flex-basis'] = false; // default: auto + whiteList['flex-direction'] = false; // default: row + whiteList['flex-flow'] = false; // default: depending on individual properties + whiteList['flex-grow'] = false; // default: 0 + whiteList['flex-shrink'] = false; // default: 1 + whiteList['flex-wrap'] = false; // default: nowrap + whiteList['float'] = false; // default: none + whiteList['float-offset'] = false; // default: 0 0 + whiteList['flood-color'] = false; // default: black + whiteList['flood-opacity'] = false; // default: 1 + whiteList['flow-from'] = false; // default: none + whiteList['flow-into'] = false; // default: none + whiteList['font'] = true; // default: depending on individual properties + whiteList['font-family'] = true; // default: implementation dependent + whiteList['font-feature-settings'] = true; // default: normal + whiteList['font-kerning'] = true; // default: auto + whiteList['font-language-override'] = true; // default: normal + whiteList['font-size'] = true; // default: medium + whiteList['font-size-adjust'] = true; // default: none + whiteList['font-stretch'] = true; // default: normal + whiteList['font-style'] = true; // default: normal + whiteList['font-synthesis'] = true; // default: weight style + whiteList['font-variant'] = true; // default: normal + whiteList['font-variant-alternates'] = true; // default: normal + whiteList['font-variant-caps'] = true; // default: normal + whiteList['font-variant-east-asian'] = true; // default: normal + whiteList['font-variant-ligatures'] = true; // default: normal + whiteList['font-variant-numeric'] = true; // default: normal + whiteList['font-variant-position'] = true; // default: normal + whiteList['font-weight'] = true; // default: normal + whiteList['grid'] = false; // default: depending on individual properties + whiteList['grid-area'] = false; // default: depending on individual properties + whiteList['grid-auto-columns'] = false; // default: auto + whiteList['grid-auto-flow'] = false; // default: none + whiteList['grid-auto-rows'] = false; // default: auto + whiteList['grid-column'] = false; // default: depending on individual properties + whiteList['grid-column-end'] = false; // default: auto + whiteList['grid-column-start'] = false; // default: auto + whiteList['grid-row'] = false; // default: depending on individual properties + whiteList['grid-row-end'] = false; // default: auto + whiteList['grid-row-start'] = false; // default: auto + whiteList['grid-template'] = false; // default: depending on individual properties + whiteList['grid-template-areas'] = false; // default: none + whiteList['grid-template-columns'] = false; // default: none + whiteList['grid-template-rows'] = false; // default: none + whiteList['hanging-punctuation'] = false; // default: none + whiteList['height'] = true; // default: auto + whiteList['hyphens'] = false; // default: manual + whiteList['icon'] = false; // default: auto + whiteList['image-orientation'] = false; // default: auto + whiteList['image-resolution'] = false; // default: normal + whiteList['ime-mode'] = false; // default: auto + whiteList['initial-letters'] = false; // default: normal + whiteList['inline-box-align'] = false; // default: last + whiteList['justify-content'] = false; // default: auto + whiteList['justify-items'] = false; // default: auto + whiteList['justify-self'] = false; // default: auto + whiteList['left'] = false; // default: auto + whiteList['letter-spacing'] = true; // default: normal + whiteList['lighting-color'] = true; // default: white + whiteList['line-box-contain'] = false; // default: block inline replaced + whiteList['line-break'] = false; // default: auto + whiteList['line-grid'] = false; // default: match-parent + whiteList['line-height'] = false; // default: normal + whiteList['line-snap'] = false; // default: none + whiteList['line-stacking'] = false; // default: depending on individual properties + whiteList['line-stacking-ruby'] = false; // default: exclude-ruby + whiteList['line-stacking-shift'] = false; // default: consider-shifts + whiteList['line-stacking-strategy'] = false; // default: inline-line-height + whiteList['list-style'] = true; // default: depending on individual properties + whiteList['list-style-image'] = true; // default: none + whiteList['list-style-position'] = true; // default: outside + whiteList['list-style-type'] = true; // default: disc + whiteList['margin'] = true; // default: depending on individual properties + whiteList['margin-bottom'] = true; // default: 0 + whiteList['margin-left'] = true; // default: 0 + whiteList['margin-right'] = true; // default: 0 + whiteList['margin-top'] = true; // default: 0 + whiteList['marker-offset'] = false; // default: auto + whiteList['marker-side'] = false; // default: list-item + whiteList['marks'] = false; // default: none + whiteList['mask'] = false; // default: border-box + whiteList['mask-box'] = false; // default: see individual properties + whiteList['mask-box-outset'] = false; // default: 0 + whiteList['mask-box-repeat'] = false; // default: stretch + whiteList['mask-box-slice'] = false; // default: 0 fill + whiteList['mask-box-source'] = false; // default: none + whiteList['mask-box-width'] = false; // default: auto + whiteList['mask-clip'] = false; // default: border-box + whiteList['mask-image'] = false; // default: none + whiteList['mask-origin'] = false; // default: border-box + whiteList['mask-position'] = false; // default: center + whiteList['mask-repeat'] = false; // default: no-repeat + whiteList['mask-size'] = false; // default: border-box + whiteList['mask-source-type'] = false; // default: auto + whiteList['mask-type'] = false; // default: luminance + whiteList['max-height'] = true; // default: none + whiteList['max-lines'] = false; // default: none + whiteList['max-width'] = true; // default: none + whiteList['min-height'] = true; // default: 0 + whiteList['min-width'] = true; // default: 0 + whiteList['move-to'] = false; // default: normal + whiteList['nav-down'] = false; // default: auto + whiteList['nav-index'] = false; // default: auto + whiteList['nav-left'] = false; // default: auto + whiteList['nav-right'] = false; // default: auto + whiteList['nav-up'] = false; // default: auto + whiteList['object-fit'] = false; // default: fill + whiteList['object-position'] = false; // default: 50% 50% + whiteList['opacity'] = false; // default: 1 + whiteList['order'] = false; // default: 0 + whiteList['orphans'] = false; // default: 2 + whiteList['outline'] = false; // default: depending on individual properties + whiteList['outline-color'] = false; // default: invert + whiteList['outline-offset'] = false; // default: 0 + whiteList['outline-style'] = false; // default: none + whiteList['outline-width'] = false; // default: medium + whiteList['overflow'] = false; // default: depending on individual properties + whiteList['overflow-wrap'] = false; // default: normal + whiteList['overflow-x'] = false; // default: visible + whiteList['overflow-y'] = false; // default: visible + whiteList['padding'] = true; // default: depending on individual properties + whiteList['padding-bottom'] = true; // default: 0 + whiteList['padding-left'] = true; // default: 0 + whiteList['padding-right'] = true; // default: 0 + whiteList['padding-top'] = true; // default: 0 + whiteList['page'] = false; // default: auto + whiteList['page-break-after'] = false; // default: auto + whiteList['page-break-before'] = false; // default: auto + whiteList['page-break-inside'] = false; // default: auto + whiteList['page-policy'] = false; // default: start + whiteList['pause'] = false; // default: implementation dependent + whiteList['pause-after'] = false; // default: implementation dependent + whiteList['pause-before'] = false; // default: implementation dependent + whiteList['perspective'] = false; // default: none + whiteList['perspective-origin'] = false; // default: 50% 50% + whiteList['pitch'] = false; // default: medium + whiteList['pitch-range'] = false; // default: 50 + whiteList['play-during'] = false; // default: auto + whiteList['position'] = false; // default: static + whiteList['presentation-level'] = false; // default: 0 + whiteList['quotes'] = false; // default: text + whiteList['region-fragment'] = false; // default: auto + whiteList['resize'] = false; // default: none + whiteList['rest'] = false; // default: depending on individual properties + whiteList['rest-after'] = false; // default: none + whiteList['rest-before'] = false; // default: none + whiteList['richness'] = false; // default: 50 + whiteList['right'] = false; // default: auto + whiteList['rotation'] = false; // default: 0 + whiteList['rotation-point'] = false; // default: 50% 50% + whiteList['ruby-align'] = false; // default: auto + whiteList['ruby-merge'] = false; // default: separate + whiteList['ruby-position'] = false; // default: before + whiteList['shape-image-threshold'] = false; // default: 0.0 + whiteList['shape-outside'] = false; // default: none + whiteList['shape-margin'] = false; // default: 0 + whiteList['size'] = false; // default: auto + whiteList['speak'] = false; // default: auto + whiteList['speak-as'] = false; // default: normal + whiteList['speak-header'] = false; // default: once + whiteList['speak-numeral'] = false; // default: continuous + whiteList['speak-punctuation'] = false; // default: none + whiteList['speech-rate'] = false; // default: medium + whiteList['stress'] = false; // default: 50 + whiteList['string-set'] = false; // default: none + whiteList['tab-size'] = false; // default: 8 + whiteList['table-layout'] = false; // default: auto + whiteList['text-align'] = true; // default: start + whiteList['text-align-last'] = true; // default: auto + whiteList['text-combine-upright'] = true; // default: none + whiteList['text-decoration'] = true; // default: none + whiteList['text-decoration-color'] = true; // default: currentColor + whiteList['text-decoration-line'] = true; // default: none + whiteList['text-decoration-skip'] = true; // default: objects + whiteList['text-decoration-style'] = true; // default: solid + whiteList['text-emphasis'] = true; // default: depending on individual properties + whiteList['text-emphasis-color'] = true; // default: currentColor + whiteList['text-emphasis-position'] = true; // default: over right + whiteList['text-emphasis-style'] = true; // default: none + whiteList['text-height'] = true; // default: auto + whiteList['text-indent'] = true; // default: 0 + whiteList['text-justify'] = true; // default: auto + whiteList['text-orientation'] = true; // default: mixed + whiteList['text-overflow'] = true; // default: clip + whiteList['text-shadow'] = true; // default: none + whiteList['text-space-collapse'] = true; // default: collapse + whiteList['text-transform'] = true; // default: none + whiteList['text-underline-position'] = true; // default: auto + whiteList['text-wrap'] = true; // default: normal + whiteList['top'] = false; // default: auto + whiteList['transform'] = false; // default: none + whiteList['transform-origin'] = false; // default: 50% 50% 0 + whiteList['transform-style'] = false; // default: flat + whiteList['transition'] = false; // default: depending on individual properties + whiteList['transition-delay'] = false; // default: 0s + whiteList['transition-duration'] = false; // default: 0s + whiteList['transition-property'] = false; // default: all + whiteList['transition-timing-function'] = false; // default: ease + whiteList['unicode-bidi'] = false; // default: normal + whiteList['vertical-align'] = false; // default: baseline + whiteList['visibility'] = false; // default: visible + whiteList['voice-balance'] = false; // default: center + whiteList['voice-duration'] = false; // default: auto + whiteList['voice-family'] = false; // default: implementation dependent + whiteList['voice-pitch'] = false; // default: medium + whiteList['voice-range'] = false; // default: medium + whiteList['voice-rate'] = false; // default: normal + whiteList['voice-stress'] = false; // default: normal + whiteList['voice-volume'] = false; // default: medium + whiteList['volume'] = false; // default: medium + whiteList['white-space'] = false; // default: normal + whiteList['widows'] = false; // default: 2 + whiteList['width'] = true; // default: auto + whiteList['will-change'] = false; // default: auto + whiteList['word-break'] = true; // default: normal + whiteList['word-spacing'] = true; // default: normal + whiteList['word-wrap'] = true; // default: normal + whiteList['wrap-flow'] = false; // default: auto + whiteList['wrap-through'] = false; // default: wrap + whiteList['writing-mode'] = false; // default: horizontal-tb + whiteList['z-index'] = false; // default: auto + + return whiteList; +} + + +/** + * 创建CSS过滤器 + * + * @param {Object} options + * - {Object} whiteList + * - {Function} onAttr + * - {Function} onIgnoreAttr + * - {Function} safeAttrValue + */ +function FilterCSS (options) { + options = shallowCopyObject(options || {}); + options.whiteList = options.whiteList || getDefaultWhiteList() + options.onAttr = options.onAttr || onAttr; + options.onIgnoreAttr = options.onIgnoreAttr || onIgnoreAttr; + options.safeAttrValue = options.safeAttrValue || safeAttrValue; + this.options = options; +} + +FilterCSS.prototype.process = function (css) { + // 兼容各种奇葩输入 + css = css || ''; + css = css.toString(); + if (!css) return ''; + + var me = this; + var options = me.options; + var whiteList = options.whiteList; + var onAttr = options.onAttr; + var onIgnoreAttr = options.onIgnoreAttr; + var safeAttrValue = options.safeAttrValue; + + var retCSS = parseStyle(css, function (sourcePosition, position, name, value, source) { + + var check = whiteList[name]; + var isWhite = false; + if (check === true) isWhite = check; + else if (typeof check === 'function') isWhite = check(value); + else if (check instanceof RegExp) isWhite = check.test(value); + if (isWhite !== true) isWhite = false; + + // 如果过滤后 value 为空则直接忽略 + value = safeAttrValue(name, value); + if (!value) return; + + var opts = { + position: position, + sourcePosition: sourcePosition, + source: source, + isWhite: isWhite + }; + + if (isWhite) { + + var ret = onAttr(name, value, opts); + if (isNull(ret)) { + return name + ':' + value; + } else { + return ret; + } + + } else { + + var ret = onIgnoreAttr(name, value, opts); + if (!isNull(ret)) { + return ret; + } + + } + }); + + return retCSS; +}; + +/** + * default onTag function + * + * @param {String} tag + * @param {String} html + * @param {Object} options + * @return {String} + */ +function onTag(tag, html, options) { + // do nothing +} + +/** + * default onIgnoreTag function + * + * @param {String} tag + * @param {String} html + * @param {Object} options + * @return {String} + */ +function onIgnoreTag(tag, html, options) { + // do nothing +} + +/** + * default onTagAttr function + * + * @param {String} tag + * @param {String} name + * @param {String} value + * @return {String} + */ +function onTagAttr(tag, name, value) { + // do nothing +} + +/** + * default onIgnoreTagAttr function + * + * @param {String} tag + * @param {String} name + * @param {String} value + * @return {String} + */ +function onIgnoreTagAttr(tag, name, value) { + // do nothing +} + +/** + * default escapeHtml function + * + * @param {String} html + */ +function escapeHtml(html) { + return html.replace(REGEXP_LT, "<").replace(REGEXP_GT, ">"); +} + +/** + * default safeAttrValue function + * + * @param {String} tag + * @param {String} name + * @param {String} value + * @param {Object} cssFilter + * @return {String} + */ +function safeAttrValue(tag, name, value, cssFilter) { + // unescape attribute value firstly + value = friendlyAttrValue(value); + + if (name === "href" || name === "src") { + // filter `href` and `src` attribute + // only allow the value that starts with `http://` | `https://` | `mailto:` | `/` | `#` + value = _.trim(value); + if (value === "#") return "#"; + if ( + !( + value.substr(0, 7) === "http://" || + value.substr(0, 8) === "https://" || + value.substr(0, 7) === "mailto:" || + value.substr(0, 4) === "tel:" || + value.substr(0, 11) === "data:image/" || + value.substr(0, 6) === "ftp://" || + value.substr(0, 2) === "./" || + value.substr(0, 3) === "../" || + value[0] === "#" || + value[0] === "/" + ) + ) { + return ""; + } + } else if (name === "background") { + // filter `background` attribute (maybe no use) + // `javascript:` + REGEXP_DEFAULT_ON_TAG_ATTR_4.lastIndex = 0; + if (REGEXP_DEFAULT_ON_TAG_ATTR_4.test(value)) { + return ""; + } + } else if (name === "style") { + // `expression()` + REGEXP_DEFAULT_ON_TAG_ATTR_7.lastIndex = 0; + if (REGEXP_DEFAULT_ON_TAG_ATTR_7.test(value)) { + return ""; + } + // `url()` + REGEXP_DEFAULT_ON_TAG_ATTR_8.lastIndex = 0; + if (REGEXP_DEFAULT_ON_TAG_ATTR_8.test(value)) { + REGEXP_DEFAULT_ON_TAG_ATTR_4.lastIndex = 0; + if (REGEXP_DEFAULT_ON_TAG_ATTR_4.test(value)) { + return ""; + } + } + if (cssFilter !== false) { + cssFilter = cssFilter || defaultCSSFilter; + value = cssFilter.process(value); + } + } + + // escape `<>"` before returns + value = escapeAttrValue(value); + return value; +} + +// RegExp list +var REGEXP_LT = //g; +var REGEXP_QUOTE = /"/g; +var REGEXP_QUOTE_2 = /"/g; +var REGEXP_ATTR_VALUE_1 = /&#([a-zA-Z0-9]*);?/gim; +var REGEXP_ATTR_VALUE_COLON = /:?/gim; +var REGEXP_ATTR_VALUE_NEWLINE = /&newline;?/gim; +var REGEXP_DEFAULT_ON_TAG_ATTR_3 = /\/\*|\*\//gm; +var REGEXP_DEFAULT_ON_TAG_ATTR_4 = /((j\s*a\s*v\s*a|v\s*b|l\s*i\s*v\s*e)\s*s\s*c\s*r\s*i\s*p\s*t\s*|m\s*o\s*c\s*h\s*a)\:/gi; +var REGEXP_DEFAULT_ON_TAG_ATTR_5 = /^[\s"'`]*(d\s*a\s*t\s*a\s*)\:/gi; +var REGEXP_DEFAULT_ON_TAG_ATTR_6 = /^[\s"'`]*(d\s*a\s*t\s*a\s*)\:\s*image\//gi; +var REGEXP_DEFAULT_ON_TAG_ATTR_7 = /e\s*x\s*p\s*r\s*e\s*s\s*s\s*i\s*o\s*n\s*\(.*/gi; +var REGEXP_DEFAULT_ON_TAG_ATTR_8 = /u\s*r\s*l\s*\(.*/gi; + +/** + * escape doube quote + * + * @param {String} str + * @return {String} str + */ +function escapeQuote(str) { + return str.replace(REGEXP_QUOTE, """); +} + +/** + * unescape double quote + * + * @param {String} str + * @return {String} str + */ +function unescapeQuote(str) { + return str.replace(REGEXP_QUOTE_2, '"'); +} + +/** + * escape html entities + * + * @param {String} str + * @return {String} + */ +function escapeHtmlEntities(str) { + return str.replace(REGEXP_ATTR_VALUE_1, function replaceUnicode(str, code) { + return code[0] === "x" || code[0] === "X" + ? String.fromCharCode(parseInt(code.substr(1), 16)) + : String.fromCharCode(parseInt(code, 10)); + }); +} + +/** + * escape html5 new danger entities + * + * @param {String} str + * @return {String} + */ +function escapeDangerHtml5Entities(str) { + return str + .replace(REGEXP_ATTR_VALUE_COLON, ":") + .replace(REGEXP_ATTR_VALUE_NEWLINE, " "); +} + +/** + * clear nonprintable characters + * + * @param {String} str + * @return {String} + */ +function clearNonPrintableCharacter(str) { + var str2 = ""; + for (var i = 0, len = str.length; i < len; i++) { + str2 += str.charCodeAt(i) < 32 ? " " : str.charAt(i); + } + return _.trim(str2); +} + +/** + * get friendly attribute value + * + * @param {String} str + * @return {String} + */ +function friendlyAttrValue(str) { + str = unescapeQuote(str); + str = escapeHtmlEntities(str); + str = escapeDangerHtml5Entities(str); + str = clearNonPrintableCharacter(str); + return str; +} + +/** + * unescape attribute value + * + * @param {String} str + * @return {String} + */ +function escapeAttrValue(str) { + str = escapeQuote(str); + str = escapeHtml(str); + return str; +} + +/** + * `onIgnoreTag` function for removing all the tags that are not in whitelist + */ +function onIgnoreTagStripAll() { + return ""; +} + +/** + * remove tag body + * specify a `tags` list, if the tag is not in the `tags` list then process by the specify function (optional) + * + * @param {array} tags + * @param {function} next + */ +function StripTagBody(tags, next) { + if (typeof next !== "function") { + next = function() {}; + } + + var isRemoveAllTag = !Array.isArray(tags); + function isRemoveTag(tag) { + if (isRemoveAllTag) return true; + return _.indexOf(tags, tag) !== -1; + } + + var removeList = []; + var posStart = false; + + return { + onIgnoreTag: function(tag, html, options) { + if (isRemoveTag(tag)) { + if (options.isClosing) { + var ret = "[/removed]"; + var end = options.position + ret.length; + removeList.push([ + posStart !== false ? posStart : options.position, + end + ]); + posStart = false; + return ret; + } else { + if (!posStart) { + posStart = options.position; + } + return "[removed]"; + } + } else { + return next(tag, html, options); + } + }, + remove: function(html) { + var rethtml = ""; + var lastPos = 0; + _.forEach(removeList, function(pos) { + rethtml += html.slice(lastPos, pos[0]); + lastPos = pos[1]; + }); + rethtml += html.slice(lastPos); + return rethtml; + } + }; +} + +/** + * remove html comments + * + * @param {String} html + * @return {String} + */ +function stripCommentTag(html) { + return html.replace(STRIP_COMMENT_TAG_REGEXP, ""); +} +var STRIP_COMMENT_TAG_REGEXP = //g; + +/** + * remove invisible characters + * + * @param {String} html + * @return {String} + */ +function stripBlankChar(html) { + var chars = html.split(""); + chars = chars.filter(function(char) { + var c = char.charCodeAt(0); + if (c === 127) return false; + if (c <= 31) { + if (c === 10 || c === 13) return true; + return false; + } + return true; + }); + return chars.join(""); +} + + +/** + * get tag name + * + * @param {String} html e.g. '' + * @return {String} + */ +function getTagName(html) { + var i = _.spaceIndex(html); + if (i === -1) { + var tagName = html.slice(1, -1); + } else { + var tagName = html.slice(1, i + 1); + } + tagName = _.trim(tagName).toLowerCase(); + if (tagName.slice(0, 1) === "/") tagName = tagName.slice(1); + if (tagName.slice(-1) === "/") tagName = tagName.slice(0, -1); + return tagName; +} + +/** + * is close tag? + * + * @param {String} html 如:'' + * @return {Boolean} + */ +function isClosing(html) { + return html.slice(0, 2) === "") { + rethtml += escapeHtml(html.slice(lastPos, tagStart)); + currentHtml = html.slice(tagStart, currentPos + 1); + currentTagName = getTagName(currentHtml); + rethtml += onTag( + tagStart, + rethtml.length, + currentTagName, + currentHtml, + isClosing(currentHtml) + ); + lastPos = currentPos + 1; + tagStart = false; + continue; + } + if ((c === '"' || c === "'") && html.charAt(currentPos - 1) === "=") { + quoteStart = c; + continue; + } + } else { + if (c === quoteStart) { + quoteStart = false; + continue; + } + } + } + } + if (lastPos < html.length) { + rethtml += escapeHtml(html.substr(lastPos)); + } + + return rethtml; +} + +var REGEXP_ILLEGAL_ATTR_NAME = /[^a-zA-Z0-9_:\.\-]/gim; + +/** + * parse input attributes and returns processed attributes + * + * @param {String} html e.g. `href="#" target="_blank"` + * @param {Function} onAttr e.g. `function (name, value)` + * @return {String} + */ +function parseAttr(html, onAttr) { + "use strict"; + + var lastPos = 0; + var retAttrs = []; + var tmpName = false; + var len = html.length; + + function addAttr(name, value) { + name = _.trim(name); + name = name.replace(REGEXP_ILLEGAL_ATTR_NAME, "").toLowerCase(); + if (name.length < 1) return; + var ret = onAttr(name, value || ""); + if (ret) retAttrs.push(ret); + } + + // 逐个分析字符 + for (var i = 0; i < len; i++) { + var c = html.charAt(i); + var v, j; + if (tmpName === false && c === "=") { + tmpName = html.slice(lastPos, i); + lastPos = i + 1; + continue; + } + if (tmpName !== false) { + if ( + i === lastPos && + (c === '"' || c === "'") && + html.charAt(i - 1) === "=" + ) { + j = html.indexOf(c, i + 1); + if (j === -1) { + break; + } else { + v = _.trim(html.slice(lastPos + 1, j)); + addAttr(tmpName, v); + tmpName = false; + i = j; + lastPos = i + 1; + continue; + } + } + } + if (/\s|\n|\t/.test(c)) { + html = html.replace(/\s|\n|\t/g, " "); + if (tmpName === false) { + j = findNextEqual(html, i); + if (j === -1) { + v = _.trim(html.slice(lastPos, i)); + addAttr(v); + tmpName = false; + lastPos = i + 1; + continue; + } else { + i = j - 1; + continue; + } + } else { + j = findBeforeEqual(html, i - 1); + if (j === -1) { + v = _.trim(html.slice(lastPos, i)); + v = stripQuoteWrap(v); + addAttr(tmpName, v); + tmpName = false; + lastPos = i + 1; + continue; + } else { + continue; + } + } + } + } + + if (lastPos < html.length) { + if (tmpName === false) { + addAttr(html.slice(lastPos)); + } else { + addAttr(tmpName, stripQuoteWrap(_.trim(html.slice(lastPos)))); + } + } + + return _.trim(retAttrs.join(" ")); +} + +function findNextEqual(str, i) { + for (; i < str.length; i++) { + var c = str[i]; + if (c === " ") continue; + if (c === "=") return i; + return -1; + } +} + +function findBeforeEqual(str, i) { + for (; i > 0; i--) { + var c = str[i]; + if (c === " ") continue; + if (c === "=") return i; + return -1; + } +} + +function isQuoteWrapString(text) { + if ( + (text[0] === '"' && text[text.length - 1] === '"') || + (text[0] === "'" && text[text.length - 1] === "'") + ) { + return true; + } else { + return false; + } +} + +function stripQuoteWrap(text) { + if (isQuoteWrapString(text)) { + return text.substr(1, text.length - 2); + } else { + return text; + } +} + + +/** + * returns `true` if the input value is `undefined` or `null` + * + * @param {Object} obj + * @return {Boolean} + */ +function isNull(obj) { + return obj === undefined || obj === null; +} + +/** + * get attributes for a tag + * + * @param {String} html + * @return {Object} + * - {String} html + * - {Boolean} closing + */ +function getAttrs(html) { + var i = _.spaceIndex(html); + if (i === -1) { + return { + html: "", + closing: html[html.length - 2] === "/" + }; + } + html = _.trim(html.slice(i + 1, -1)); + var isClosing = html[html.length - 1] === "/"; + if (isClosing) html = _.trim(html.slice(0, -1)); + return { + html: html, + closing: isClosing + }; +} + diff --git a/ui/StatusQ/src/StatusQ/Core/qmldir b/ui/StatusQ/src/StatusQ/Core/qmldir index d0da9f06a6..02a5ff1182 100644 --- a/ui/StatusQ/src/StatusQ/Core/qmldir +++ b/ui/StatusQ/src/StatusQ/Core/qmldir @@ -14,3 +14,4 @@ StatusAnimatedStack 0.1 StatusAnimatedStack.qml StatusScrollView 0.1 StatusScrollView.qml StatusListView 0.1 StatusListView.qml StatusGridView 0.1 StatusGridView.qml +StatusProfileImageSettings 0.1 StatusProfileImageSettings.qml