* Initial implementation. * Add developer option for showing timeline item authenticity. * Refactor code to use new SendInfo.Status. --------- Co-authored-by: Doug <douglase@element.io>
773 lines
47 KiB
Swift
773 lines
47 KiB
Swift
//
|
|
// Copyright 2022 New Vector Ltd
|
|
//
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
//
|
|
|
|
import MatrixRustSDK
|
|
import UIKit
|
|
import UniformTypeIdentifiers
|
|
|
|
struct RoomTimelineItemFactory: RoomTimelineItemFactoryProtocol {
|
|
private let attributedStringBuilder: AttributedStringBuilderProtocol
|
|
private let stateEventStringBuilder: RoomStateEventStringBuilder
|
|
|
|
/// The Matrix ID of the current user.
|
|
private let userID: String
|
|
private let encryptionAuthenticityEnabled: Bool
|
|
|
|
init(userID: String,
|
|
encryptionAuthenticityEnabled: Bool,
|
|
attributedStringBuilder: AttributedStringBuilderProtocol,
|
|
stateEventStringBuilder: RoomStateEventStringBuilder) {
|
|
self.userID = userID
|
|
self.encryptionAuthenticityEnabled = encryptionAuthenticityEnabled
|
|
self.attributedStringBuilder = attributedStringBuilder
|
|
self.stateEventStringBuilder = stateEventStringBuilder
|
|
}
|
|
|
|
func buildTimelineItem(for eventItemProxy: EventTimelineItemProxy, isDM: Bool) -> RoomTimelineItemProtocol? {
|
|
let isOutgoing = eventItemProxy.isOwn
|
|
|
|
switch eventItemProxy.content.kind() {
|
|
case .unableToDecrypt(let encryptedMessage):
|
|
return buildEncryptedTimelineItem(eventItemProxy, encryptedMessage, isOutgoing)
|
|
case .redactedMessage:
|
|
return buildRedactedTimelineItem(eventItemProxy, isOutgoing)
|
|
case .sticker(let body, let imageInfo, let mediaSource):
|
|
guard let url = URL(string: mediaSource.url()) else {
|
|
MXLog.error("Invalid sticker url string: \(mediaSource.url())")
|
|
return buildUnsupportedTimelineItem(eventItemProxy, "m.sticker", "Invalid Sticker URL", isOutgoing)
|
|
}
|
|
|
|
return buildStickerTimelineItem(eventItemProxy, body, imageInfo, url, isOutgoing)
|
|
case .failedToParseMessageLike(let eventType, let error):
|
|
return buildUnsupportedTimelineItem(eventItemProxy, eventType, error, isOutgoing)
|
|
case .failedToParseState(let eventType, _, let error):
|
|
return buildUnsupportedTimelineItem(eventItemProxy, eventType, error, isOutgoing)
|
|
case .message:
|
|
return buildMessageTimelineItem(eventItemProxy, isOutgoing)
|
|
case .state(_, let content):
|
|
if isDM, content == .roomCreate {
|
|
return nil
|
|
}
|
|
return buildStateTimelineItem(for: eventItemProxy, state: content, isOutgoing: isOutgoing)
|
|
case .roomMembership(userId: let userID, let displayName, change: let change):
|
|
if isDM, change == .joined, userID == self.userID {
|
|
return nil
|
|
}
|
|
return buildStateMembershipChangeTimelineItem(for: eventItemProxy, memberUserID: userID, memberDisplayName: displayName, membershipChange: change, isOutgoing: isOutgoing)
|
|
case .profileChange(let displayName, let prevDisplayName, let avatarUrl, let prevAvatarUrl):
|
|
return buildStateProfileChangeTimelineItem(for: eventItemProxy,
|
|
displayName: displayName,
|
|
previousDisplayName: prevDisplayName,
|
|
avatarURLString: avatarUrl,
|
|
previousAvatarURLString: prevAvatarUrl,
|
|
isOutgoing: isOutgoing)
|
|
case .poll(question: let question, kind: let kind, maxSelections: let maxSelections, answers: let answers, votes: let votes, endTime: let endTime, let edited):
|
|
return buildPollTimelineItem(question, kind, maxSelections, answers, votes, endTime, eventItemProxy, isOutgoing, edited)
|
|
case .callInvite:
|
|
return buildCallInviteTimelineItem(for: eventItemProxy)
|
|
case .callNotify:
|
|
return buildCallNotificationTimelineItem(for: eventItemProxy)
|
|
}
|
|
}
|
|
|
|
// MARK: - Message Events
|
|
|
|
private func buildMessageTimelineItem(_ eventItemProxy: EventTimelineItemProxy, _ isOutgoing: Bool) -> RoomTimelineItemProtocol? {
|
|
guard let messageTimelineItem = eventItemProxy.content.asMessage() else {
|
|
fatalError("Invalid message timeline item: \(eventItemProxy)")
|
|
}
|
|
|
|
let isThreaded = messageTimelineItem.isThreaded()
|
|
switch messageTimelineItem.msgtype() {
|
|
case .text(content: let content):
|
|
return buildTextTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .image(content: let content):
|
|
return buildImageTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .video(let content):
|
|
return buildVideoTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .file(let content):
|
|
return buildFileTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .notice(content: let content):
|
|
return buildNoticeTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .emote(content: let content):
|
|
return buildEmoteTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .audio(let content):
|
|
if content.voice != nil {
|
|
return buildVoiceTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
} else {
|
|
return buildAudioTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
}
|
|
case .location(let content):
|
|
return buildLocationTimelineItem(for: eventItemProxy, messageTimelineItem, content, isOutgoing, isThreaded)
|
|
case .other:
|
|
return nil
|
|
}
|
|
}
|
|
|
|
private func buildUnsupportedTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
|
_ eventType: String,
|
|
_ error: String,
|
|
_ isOutgoing: Bool) -> RoomTimelineItemProtocol {
|
|
UnsupportedRoomTimelineItem(id: eventItemProxy.id,
|
|
body: L10n.commonUnsupportedEvent,
|
|
eventType: eventType,
|
|
error: error,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender,
|
|
properties: RoomTimelineItemProperties())
|
|
}
|
|
|
|
private func buildStickerTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
|
_ body: String,
|
|
_ imageInfo: ImageInfo,
|
|
_ imageURL: URL,
|
|
_ isOutgoing: Bool) -> RoomTimelineItemProtocol {
|
|
var aspectRatio: CGFloat?
|
|
let width = imageInfo.width.map(CGFloat.init)
|
|
let height = imageInfo.height.map(CGFloat.init)
|
|
if let width, let height, width > 0, height > 0 {
|
|
aspectRatio = width / height
|
|
}
|
|
|
|
return StickerRoomTimelineItem(id: eventItemProxy.id,
|
|
body: body,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender,
|
|
imageURL: imageURL,
|
|
width: width,
|
|
height: height,
|
|
aspectRatio: aspectRatio,
|
|
blurhash: imageInfo.blurhash,
|
|
properties: RoomTimelineItemProperties(reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildEncryptedTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
|
_ encryptedMessage: EncryptedMessage,
|
|
_ isOutgoing: Bool) -> RoomTimelineItemProtocol {
|
|
var encryptionType = EncryptedRoomTimelineItem.EncryptionType.unknown
|
|
var errorLabel = L10n.commonWaitingForDecryptionKey
|
|
switch encryptedMessage {
|
|
case .megolmV1AesSha2(let sessionID, let cause):
|
|
switch cause {
|
|
case .unknown:
|
|
encryptionType = .megolmV1AesSha2(sessionID: sessionID, cause: .unknown)
|
|
errorLabel = L10n.commonWaitingForDecryptionKey
|
|
case .membership:
|
|
encryptionType = .megolmV1AesSha2(sessionID: sessionID, cause: .membership)
|
|
errorLabel = L10n.commonUnableToDecryptNoAccess
|
|
}
|
|
case .olmV1Curve25519AesSha2(let senderKey):
|
|
encryptionType = .olmV1Curve25519AesSha2(senderKey: senderKey)
|
|
case .unknown:
|
|
break
|
|
}
|
|
|
|
return EncryptedRoomTimelineItem(id: eventItemProxy.id,
|
|
body: errorLabel,
|
|
encryptionType: encryptionType,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender,
|
|
properties: RoomTimelineItemProperties())
|
|
}
|
|
|
|
private func buildRedactedTimelineItem(_ eventItemProxy: EventTimelineItemProxy,
|
|
_ isOutgoing: Bool) -> RoomTimelineItemProtocol {
|
|
RedactedRoomTimelineItem(id: eventItemProxy.id,
|
|
body: L10n.commonMessageRemoved,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender,
|
|
properties: RoomTimelineItemProperties())
|
|
}
|
|
|
|
private func buildTextTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: TextMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
TextRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildTextTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildImageTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: ImageMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
ImageRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildImageTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildVideoTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: VideoMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
VideoRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildVideoTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildAudioTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: AudioMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
AudioRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildAudioTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildVoiceTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: AudioMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
VoiceMessageRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildAudioTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildFileTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: FileMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
FileRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildFileTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildNoticeTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: NoticeMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
NoticeRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildNoticeTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildEmoteTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: EmoteMessageContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
EmoteRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildEmoteTimelineItemContent(senderDisplayName: eventItemProxy.sender.displayName, senderID: eventItemProxy.sender.id, messageContent: messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildLocationTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
_ messageTimelineItem: Message,
|
|
_ messageContent: LocationContent,
|
|
_ isOutgoing: Bool,
|
|
_ isThreaded: Bool) -> RoomTimelineItemProtocol {
|
|
LocationRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
isThreaded: isThreaded,
|
|
sender: eventItemProxy.sender,
|
|
content: buildLocationTimelineItemContent(messageContent),
|
|
replyDetails: buildReplyToDetailsFromDetailsIfAvailable(details: messageTimelineItem.inReplyTo()),
|
|
properties: RoomTimelineItemProperties(isEdited: messageTimelineItem.isEdited(),
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
// swiftlint:disable:next function_parameter_count
|
|
private func buildPollTimelineItem(_ question: String,
|
|
_ pollKind: PollKind,
|
|
_ maxSelections: UInt64,
|
|
_ answers: [PollAnswer],
|
|
_ votes: [String: [String]],
|
|
_ endTime: UInt64?,
|
|
_ eventItemProxy: EventTimelineItemProxy,
|
|
_ isOutgoing: Bool,
|
|
_ edited: Bool) -> RoomTimelineItemProtocol {
|
|
let allVotes = votes.reduce(0) { count, pair in
|
|
count + pair.value.count
|
|
}
|
|
|
|
let maxOptionVotes = votes.map(\.value.count).max()
|
|
|
|
let options = answers.map { answer in
|
|
let optionVotesCount = votes[answer.id]?.count
|
|
|
|
return Poll.Option(id: answer.id,
|
|
text: answer.text,
|
|
votes: optionVotesCount ?? 0,
|
|
allVotes: allVotes,
|
|
isSelected: votes[answer.id]?.contains(userID) ?? false,
|
|
isWinning: optionVotesCount.map { $0 == maxOptionVotes } ?? false)
|
|
}
|
|
|
|
let poll = Poll(question: question,
|
|
kind: .init(pollKind: pollKind),
|
|
maxSelections: Int(maxSelections),
|
|
options: options,
|
|
votes: votes,
|
|
endDate: endTime.map { Date(timeIntervalSince1970: TimeInterval($0 / 1000)) },
|
|
createdByAccountOwner: eventItemProxy.sender.id == userID)
|
|
|
|
return PollRoomTimelineItem(id: eventItemProxy.id,
|
|
poll: poll,
|
|
body: poll.question,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender,
|
|
properties: RoomTimelineItemProperties(isEdited: edited,
|
|
reactions: aggregateReactions(eventItemProxy.reactions),
|
|
deliveryStatus: eventItemProxy.deliveryStatus,
|
|
orderedReadReceipts: orderReadReceipts(eventItemProxy.readReceipts),
|
|
encryptionAuthenticity: authenticity(eventItemProxy.shieldState)))
|
|
}
|
|
|
|
private func buildCallInviteTimelineItem(for eventItemProxy: EventTimelineItemProxy) -> RoomTimelineItemProtocol {
|
|
CallInviteRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender)
|
|
}
|
|
|
|
private func buildCallNotificationTimelineItem(for eventItemProxy: EventTimelineItemProxy) -> RoomTimelineItemProtocol {
|
|
CallNotificationRoomTimelineItem(id: eventItemProxy.id,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isEditable: eventItemProxy.isEditable,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender)
|
|
}
|
|
|
|
private func aggregateReactions(_ reactions: [Reaction]) -> [AggregatedReaction] {
|
|
reactions.map { reaction in
|
|
let senders = reaction.senders
|
|
.map { senderData in
|
|
ReactionSender(id: senderData.senderId, timestamp: Date(timeIntervalSince1970: TimeInterval(senderData.timestamp / 1000)))
|
|
}
|
|
.sorted { a, b in
|
|
// Sort reactions within an aggregation by timestamp descending.
|
|
// This puts the most recent at the top, useful in cases like the
|
|
// reaction summary view.
|
|
a.timestamp > b.timestamp
|
|
}
|
|
return AggregatedReaction(accountOwnerID: userID, key: reaction.key, senders: senders)
|
|
}
|
|
.sorted { a, b in
|
|
// Sort aggregated reactions by count and then timestamp ascending, using
|
|
// the most recent reaction in the aggregation(hence index 0).
|
|
// This appends new aggregations on the end of the reaction layout
|
|
// and the deterministic sort avoids reactions jumping around if the reactions timeline
|
|
// view reloads.
|
|
if a.count == b.count {
|
|
return a.senders[0].timestamp < b.senders[0].timestamp
|
|
}
|
|
return a.count > b.count
|
|
}
|
|
}
|
|
|
|
private func orderReadReceipts(_ receipts: [String: Receipt]) -> [ReadReceipt] {
|
|
receipts
|
|
.sorted { firstElement, secondElement in
|
|
// If there is no timestamp we order them as last
|
|
let firstTimestamp = firstElement.value.dateTimestamp ?? Date(timeIntervalSince1970: 0)
|
|
let secondTimestamp = secondElement.value.dateTimestamp ?? Date(timeIntervalSince1970: 0)
|
|
return firstTimestamp > secondTimestamp
|
|
}
|
|
.map { key, receipt in
|
|
ReadReceipt(userID: key, formattedTimestamp: receipt.dateTimestamp?.formattedMinimal())
|
|
}
|
|
}
|
|
|
|
private func authenticity(_ shieldState: ShieldState?) -> EncryptionAuthenticity? {
|
|
guard encryptionAuthenticityEnabled else { return nil }
|
|
return shieldState.flatMap(EncryptionAuthenticity.init)
|
|
}
|
|
|
|
// MARK: - Message events content
|
|
|
|
private func buildTextTimelineItemContent(_ messageContent: TextMessageContent) -> TextRoomTimelineItemContent {
|
|
let htmlBody = messageContent.formatted?.format == .html ? messageContent.formatted?.body : nil
|
|
let formattedBody = (htmlBody != nil ? attributedStringBuilder.fromHTML(htmlBody) : attributedStringBuilder.fromPlain(messageContent.body))
|
|
|
|
return .init(body: messageContent.body, formattedBody: formattedBody, formattedBodyHTMLString: htmlBody)
|
|
}
|
|
|
|
private func buildAudioTimelineItemContent(_ messageContent: AudioMessageContent) -> AudioRoomTimelineItemContent {
|
|
var waveform: EstimatedWaveform?
|
|
if let audioWaveform = messageContent.audio?.waveform {
|
|
waveform = EstimatedWaveform(data: audioWaveform)
|
|
}
|
|
|
|
return AudioRoomTimelineItemContent(body: messageContent.body,
|
|
duration: messageContent.audio?.duration ?? 0,
|
|
waveform: waveform,
|
|
source: MediaSourceProxy(source: messageContent.source, mimeType: messageContent.info?.mimetype),
|
|
contentType: UTType(mimeType: messageContent.info?.mimetype, fallbackFilename: messageContent.body))
|
|
}
|
|
|
|
private func buildImageTimelineItemContent(_ messageContent: ImageMessageContent) -> ImageRoomTimelineItemContent {
|
|
let thumbnailSource = messageContent.info?.thumbnailSource.map { MediaSourceProxy(source: $0, mimeType: messageContent.info?.thumbnailInfo?.mimetype) }
|
|
let width = messageContent.info?.width.map(CGFloat.init)
|
|
let height = messageContent.info?.height.map(CGFloat.init)
|
|
|
|
var aspectRatio: CGFloat?
|
|
if let width, let height, width > 0, height > 0 {
|
|
aspectRatio = width / height
|
|
}
|
|
|
|
return .init(body: messageContent.body,
|
|
source: MediaSourceProxy(source: messageContent.source, mimeType: messageContent.info?.mimetype),
|
|
thumbnailSource: thumbnailSource,
|
|
width: width,
|
|
height: height,
|
|
aspectRatio: aspectRatio,
|
|
blurhash: messageContent.info?.blurhash,
|
|
contentType: UTType(mimeType: messageContent.info?.mimetype, fallbackFilename: messageContent.body))
|
|
}
|
|
|
|
private func buildVideoTimelineItemContent(_ messageContent: VideoMessageContent) -> VideoRoomTimelineItemContent {
|
|
let thumbnailSource = messageContent.info?.thumbnailSource.map { MediaSourceProxy(source: $0, mimeType: messageContent.info?.thumbnailInfo?.mimetype) }
|
|
let width = messageContent.info?.width.map(CGFloat.init)
|
|
let height = messageContent.info?.height.map(CGFloat.init)
|
|
|
|
var aspectRatio: CGFloat?
|
|
if let width, let height, width > 0, height > 0 {
|
|
aspectRatio = width / height
|
|
}
|
|
|
|
return .init(body: messageContent.body,
|
|
duration: messageContent.info?.duration ?? 0,
|
|
source: MediaSourceProxy(source: messageContent.source, mimeType: messageContent.info?.mimetype),
|
|
thumbnailSource: thumbnailSource,
|
|
width: width,
|
|
height: height,
|
|
aspectRatio: aspectRatio,
|
|
blurhash: messageContent.info?.blurhash,
|
|
contentType: UTType(mimeType: messageContent.info?.mimetype, fallbackFilename: messageContent.body))
|
|
}
|
|
|
|
private func buildLocationTimelineItemContent(_ locationContent: LocationContent) -> LocationRoomTimelineItemContent {
|
|
LocationRoomTimelineItemContent(body: locationContent.body,
|
|
geoURI: .init(string: locationContent.geoUri),
|
|
description: locationContent.description)
|
|
}
|
|
|
|
private func buildFileTimelineItemContent(_ messageContent: FileMessageContent) -> FileRoomTimelineItemContent {
|
|
let thumbnailSource = messageContent.info?.thumbnailSource.map { MediaSourceProxy(source: $0, mimeType: messageContent.info?.thumbnailInfo?.mimetype) }
|
|
|
|
return .init(body: messageContent.body,
|
|
source: MediaSourceProxy(source: messageContent.source, mimeType: messageContent.info?.mimetype),
|
|
thumbnailSource: thumbnailSource,
|
|
contentType: UTType(mimeType: messageContent.info?.mimetype, fallbackFilename: messageContent.body))
|
|
}
|
|
|
|
private func buildNoticeTimelineItemContent(_ messageContent: NoticeMessageContent) -> NoticeRoomTimelineItemContent {
|
|
let htmlBody = messageContent.formatted?.format == .html ? messageContent.formatted?.body : nil
|
|
let formattedBody = (htmlBody != nil ? attributedStringBuilder.fromHTML(htmlBody) : attributedStringBuilder.fromPlain(messageContent.body))
|
|
|
|
return .init(body: messageContent.body, formattedBody: formattedBody)
|
|
}
|
|
|
|
private func buildEmoteTimelineItemContent(senderDisplayName: String?, senderID: String, messageContent: EmoteMessageContent) -> EmoteRoomTimelineItemContent {
|
|
let name = senderDisplayName ?? senderID
|
|
|
|
let htmlBody = messageContent.formatted?.format == .html ? messageContent.formatted?.body : nil
|
|
|
|
var formattedBody: AttributedString?
|
|
if let htmlBody {
|
|
formattedBody = buildEmoteFormattedBodyFromHTML(html: htmlBody, name: name)
|
|
} else {
|
|
formattedBody = attributedStringBuilder.fromPlain(L10n.commonEmote(name, messageContent.body))
|
|
}
|
|
|
|
return .init(body: messageContent.body, formattedBody: formattedBody, formattedBodyHTMLString: htmlBody)
|
|
}
|
|
|
|
// This fixes the issue of the name not belonging to the first <p> defined paragraph
|
|
private func buildEmoteFormattedBodyFromHTML(html: String, name: String) -> AttributedString? {
|
|
let htmlBodyPlaceholder = "{htmlBodyPlaceholder}"
|
|
var finalString = AttributedString(L10n.commonEmote(name, htmlBodyPlaceholder))
|
|
guard let htmlBodyString = attributedStringBuilder.fromHTML(html) else {
|
|
return nil
|
|
}
|
|
finalString.replace(htmlBodyPlaceholder, with: htmlBodyString)
|
|
return finalString
|
|
}
|
|
|
|
// MARK: - State Events
|
|
|
|
private func buildStateTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
state: OtherState,
|
|
isOutgoing: Bool) -> RoomTimelineItemProtocol? {
|
|
guard let text = stateEventStringBuilder.buildString(for: state, sender: eventItemProxy.sender, isOutgoing: isOutgoing) else { return nil }
|
|
return buildStateTimelineItem(for: eventItemProxy, text: text, isOutgoing: isOutgoing)
|
|
}
|
|
|
|
private func buildStateMembershipChangeTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
memberUserID: String,
|
|
memberDisplayName: String?,
|
|
membershipChange: MembershipChange?,
|
|
isOutgoing: Bool) -> RoomTimelineItemProtocol? {
|
|
guard let text = stateEventStringBuilder.buildString(for: membershipChange, memberUserID: memberUserID, memberDisplayName: memberDisplayName, sender: eventItemProxy.sender, isOutgoing: isOutgoing) else { return nil }
|
|
return buildStateTimelineItem(for: eventItemProxy, text: text, isOutgoing: isOutgoing)
|
|
}
|
|
|
|
// swiftlint:disable:next function_parameter_count
|
|
private func buildStateProfileChangeTimelineItem(for eventItemProxy: EventTimelineItemProxy,
|
|
displayName: String?,
|
|
previousDisplayName: String?,
|
|
avatarURLString: String?,
|
|
previousAvatarURLString: String?,
|
|
isOutgoing: Bool) -> RoomTimelineItemProtocol? {
|
|
guard let text = stateEventStringBuilder.buildProfileChangeString(displayName: displayName,
|
|
previousDisplayName: previousDisplayName,
|
|
avatarURLString: avatarURLString,
|
|
previousAvatarURLString: previousAvatarURLString,
|
|
member: eventItemProxy.sender.id,
|
|
memberIsYou: isOutgoing) else { return nil }
|
|
return buildStateTimelineItem(for: eventItemProxy, text: text, isOutgoing: isOutgoing)
|
|
}
|
|
|
|
private func buildStateTimelineItem(for eventItemProxy: EventTimelineItemProxy, text: String, isOutgoing: Bool) -> RoomTimelineItemProtocol {
|
|
StateRoomTimelineItem(id: eventItemProxy.id,
|
|
body: text,
|
|
timestamp: eventItemProxy.timestamp.formatted(date: .omitted, time: .shortened),
|
|
isOutgoing: isOutgoing,
|
|
isEditable: false,
|
|
canBeRepliedTo: eventItemProxy.canBeRepliedTo,
|
|
sender: eventItemProxy.sender)
|
|
}
|
|
|
|
// MARK: - Reply details
|
|
|
|
func buildReply(details: InReplyToDetails) -> TimelineItemReply {
|
|
let isThreaded = details.event.isThreaded
|
|
switch details.event {
|
|
case .unavailable:
|
|
return .init(details: .notLoaded(eventID: details.eventId), isThreaded: isThreaded)
|
|
case .pending:
|
|
return .init(details: .loading(eventID: details.eventId), isThreaded: isThreaded)
|
|
case let .ready(timelineItem, senderID, senderProfile):
|
|
let sender: TimelineItemSender
|
|
switch senderProfile {
|
|
case let .ready(displayName, isDisplayNameAmbiguous, avatarUrl):
|
|
sender = TimelineItemSender(id: senderID,
|
|
displayName: displayName,
|
|
isDisplayNameAmbiguous: isDisplayNameAmbiguous,
|
|
avatarURL: avatarUrl.flatMap(URL.init(string:)))
|
|
default:
|
|
sender = TimelineItemSender(id: senderID,
|
|
displayName: nil,
|
|
isDisplayNameAmbiguous: false,
|
|
avatarURL: nil)
|
|
}
|
|
|
|
let replyContent: TimelineEventContent
|
|
|
|
switch timelineItem.kind() {
|
|
case .message:
|
|
return .init(details: timelineItemReplyDetails(sender: sender, eventID: details.eventId, messageType: timelineItem.asMessage()?.msgtype()), isThreaded: isThreaded)
|
|
case .poll(let question, _, _, _, _, _, _):
|
|
replyContent = .poll(question: question)
|
|
case .sticker(let body, _, _):
|
|
replyContent = .message(.text(.init(body: body)))
|
|
case .redactedMessage:
|
|
replyContent = .redacted
|
|
default:
|
|
replyContent = .message(.text(.init(body: L10n.commonUnsupportedEvent)))
|
|
}
|
|
|
|
return .init(details: .loaded(sender: sender, eventID: details.eventId, eventContent: replyContent), isThreaded: isThreaded)
|
|
case let .error(message):
|
|
return .init(details: .error(eventID: details.eventId, message: message), isThreaded: isThreaded)
|
|
}
|
|
}
|
|
|
|
private func buildReplyToDetailsFromDetailsIfAvailable(details: InReplyToDetails?) -> TimelineItemReplyDetails? {
|
|
guard let details else {
|
|
return nil
|
|
}
|
|
|
|
return buildReply(details: details).details
|
|
}
|
|
|
|
private func timelineItemReplyDetails(sender: TimelineItemSender, eventID: String, messageType: MessageType?) -> TimelineItemReplyDetails {
|
|
let replyContent: EventBasedMessageTimelineItemContentType
|
|
|
|
switch messageType {
|
|
case .audio(let content):
|
|
if content.voice != nil {
|
|
replyContent = .voice(buildAudioTimelineItemContent(content))
|
|
} else {
|
|
replyContent = .audio(buildAudioTimelineItemContent(content))
|
|
}
|
|
case .emote(let content):
|
|
replyContent = .emote(buildEmoteTimelineItemContent(senderDisplayName: sender.displayName, senderID: sender.id, messageContent: content))
|
|
case .file(let content):
|
|
replyContent = .file(buildFileTimelineItemContent(content))
|
|
case .image(let content):
|
|
replyContent = .image(buildImageTimelineItemContent(content))
|
|
case .notice(let content):
|
|
replyContent = .notice(buildNoticeTimelineItemContent(content))
|
|
case .text(let content):
|
|
replyContent = .text(buildTextTimelineItemContent(content))
|
|
case .video(let content):
|
|
replyContent = .video(buildVideoTimelineItemContent(content))
|
|
case .location(let content):
|
|
replyContent = .location(buildLocationTimelineItemContent(content))
|
|
case .other, .none:
|
|
replyContent = .text(.init(body: L10n.commonUnsupportedEvent))
|
|
}
|
|
|
|
return .loaded(sender: sender,
|
|
eventID: eventID,
|
|
eventContent: .message(replyContent))
|
|
}
|
|
}
|
|
|
|
extension Poll.Kind {
|
|
init(pollKind: MatrixRustSDK.PollKind) {
|
|
switch pollKind {
|
|
case .disclosed:
|
|
self = .disclosed
|
|
case .undisclosed:
|
|
self = .undisclosed
|
|
}
|
|
}
|
|
}
|
|
|
|
private extension RepliedToEventDetails {
|
|
var isThreaded: Bool {
|
|
switch self {
|
|
case .ready(let content, _, _):
|
|
return content.asMessage()?.isThreaded() ?? false
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
}
|