mirror of https://github.com/oxen-io/session-ios
You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1272 lines
58 KiB
Swift
1272 lines
58 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
import SessionUIKit
|
|
import SignalUtilitiesKit
|
|
import SessionUtilitiesKit
|
|
import SessionMessagingKit
|
|
|
|
final class VisibleMessageCell: MessageCell, TappableLabelDelegate {
|
|
private var isHandlingLongPress: Bool = false
|
|
private var unloadContent: (() -> Void)?
|
|
private var previousX: CGFloat = 0
|
|
|
|
var albumView: MediaAlbumView?
|
|
var quoteView: QuoteView?
|
|
var linkPreviewView: LinkPreviewView?
|
|
var documentView: DocumentView?
|
|
var bodyTappableLabel: TappableLabel?
|
|
var voiceMessageView: VoiceMessageView?
|
|
var audioStateChanged: ((TimeInterval, Bool) -> ())?
|
|
|
|
override var contextSnapshotView: UIView? { return snContentView }
|
|
|
|
// Constraints
|
|
private lazy var authorLabelTopConstraint = authorLabel.pin(.top, to: .top, of: self)
|
|
private lazy var authorLabelHeightConstraint = authorLabel.set(.height, to: 0)
|
|
private lazy var profilePictureViewLeadingConstraint = profilePictureView.pin(.leading, to: .leading, of: self, withInset: VisibleMessageCell.groupThreadHSpacing)
|
|
private lazy var contentViewLeadingConstraint1 = snContentView.pin(.leading, to: .trailing, of: profilePictureView, withInset: VisibleMessageCell.groupThreadHSpacing)
|
|
private lazy var contentViewLeadingConstraint2 = snContentView.leadingAnchor.constraint(greaterThanOrEqualTo: leadingAnchor, constant: VisibleMessageCell.gutterSize)
|
|
private lazy var contentViewTopConstraint = snContentView.pin(.top, to: .bottom, of: authorLabel, withInset: VisibleMessageCell.authorLabelBottomSpacing)
|
|
private lazy var contentViewTrailingConstraint1 = snContentView.pin(.trailing, to: .trailing, of: self, withInset: -VisibleMessageCell.contactThreadHSpacing)
|
|
private lazy var contentViewTrailingConstraint2 = snContentView.trailingAnchor.constraint(lessThanOrEqualTo: trailingAnchor, constant: -VisibleMessageCell.gutterSize)
|
|
private lazy var contentBottomConstraint = snContentView.bottomAnchor
|
|
.constraint(lessThanOrEqualTo: self.bottomAnchor, constant: -1)
|
|
|
|
private lazy var underBubbleStackViewIncomingLeadingConstraint: NSLayoutConstraint = underBubbleStackView.pin(.leading, to: .leading, of: snContentView)
|
|
private lazy var underBubbleStackViewIncomingTrailingConstraint: NSLayoutConstraint = underBubbleStackView.pin(.trailing, to: .trailing, of: self, withInset: -VisibleMessageCell.contactThreadHSpacing)
|
|
private lazy var underBubbleStackViewOutgoingLeadingConstraint: NSLayoutConstraint = underBubbleStackView.pin(.leading, to: .leading, of: self, withInset: VisibleMessageCell.contactThreadHSpacing)
|
|
private lazy var underBubbleStackViewOutgoingTrailingConstraint: NSLayoutConstraint = underBubbleStackView.pin(.trailing, to: .trailing, of: snContentView)
|
|
private lazy var underBubbleStackViewNoHeightConstraint: NSLayoutConstraint = underBubbleStackView.set(.height, to: 0)
|
|
|
|
private lazy var timerViewOutgoingMessageConstraint = timerView.pin(.trailing, to: .trailing, of: messageStatusContainerView)
|
|
private lazy var timerViewIncomingMessageConstraint = timerView.pin(.leading, to: .leading, of: messageStatusContainerView)
|
|
private lazy var messageStatusLabelOutgoingMessageConstraint = messageStatusLabel.pin(.trailing, to: .leading, of: timerView, withInset: -2)
|
|
private lazy var messageStatusLabelIncomingMessageConstraint = messageStatusLabel.pin(.leading, to: .trailing, of: timerView, withInset: 2)
|
|
|
|
private lazy var panGestureRecognizer: UIPanGestureRecognizer = {
|
|
let result = UIPanGestureRecognizer(target: self, action: #selector(handlePan))
|
|
result.delegate = self
|
|
return result
|
|
}()
|
|
|
|
// MARK: - UI Components
|
|
|
|
private lazy var viewsToMoveForReply: [UIView] = [
|
|
snContentView,
|
|
profilePictureView,
|
|
replyButton,
|
|
timerView,
|
|
messageStatusContainerView,
|
|
reactionContainerView
|
|
]
|
|
|
|
private lazy var profilePictureView: ProfilePictureView = ProfilePictureView(size: .message)
|
|
|
|
lazy var bubbleBackgroundView: UIView = {
|
|
let result = UIView()
|
|
result.layer.cornerRadius = VisibleMessageCell.largeCornerRadius
|
|
return result
|
|
}()
|
|
|
|
lazy var bubbleView: UIView = {
|
|
let result = UIView()
|
|
result.clipsToBounds = true
|
|
result.layer.cornerRadius = VisibleMessageCell.largeCornerRadius
|
|
result.set(.width, greaterThanOrEqualTo: VisibleMessageCell.largeCornerRadius * 2)
|
|
return result
|
|
}()
|
|
|
|
private lazy var authorLabel: UILabel = {
|
|
let result = UILabel()
|
|
result.font = .boldSystemFont(ofSize: Values.smallFontSize)
|
|
return result
|
|
}()
|
|
|
|
lazy var snContentView: UIStackView = {
|
|
let result = UIStackView(arrangedSubviews: [])
|
|
result.axis = .vertical
|
|
result.spacing = Values.verySmallSpacing
|
|
result.alignment = .leading
|
|
return result
|
|
}()
|
|
|
|
private lazy var replyButton: UIView = {
|
|
let result = UIView()
|
|
let size = VisibleMessageCell.replyButtonSize + 8
|
|
result.set(.width, to: size)
|
|
result.set(.height, to: size)
|
|
result.themeBorderColor = .textPrimary
|
|
result.layer.borderWidth = 1
|
|
result.layer.cornerRadius = (size / 2)
|
|
result.layer.masksToBounds = true
|
|
result.alpha = 0
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var replyIconImageView: UIImageView = {
|
|
let result = UIImageView()
|
|
let size = VisibleMessageCell.replyButtonSize
|
|
result.set(.width, to: size)
|
|
result.set(.height, to: size)
|
|
result.image = UIImage(named: "ic_reply")?.withRenderingMode(.alwaysTemplate)
|
|
result.themeTintColor = .textPrimary
|
|
|
|
// Flip horizontally for RTL languages
|
|
result.transform = CGAffineTransform.identity
|
|
.scaledBy(
|
|
x: (Singleton.hasAppContext && Singleton.appContext.isRTL ? -1 : 1),
|
|
y: 1
|
|
)
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var timerView: DisappearingMessageTimerView = DisappearingMessageTimerView()
|
|
|
|
lazy var underBubbleStackView: UIStackView = {
|
|
let result = UIStackView(arrangedSubviews: [])
|
|
result.setContentHuggingPriority(.required, for: .vertical)
|
|
result.setContentCompressionResistancePriority(.required, for: .vertical)
|
|
result.axis = .vertical
|
|
result.spacing = Values.verySmallSpacing
|
|
result.alignment = .trailing
|
|
|
|
return result
|
|
}()
|
|
|
|
private lazy var reactionContainerView = ReactionContainerView()
|
|
|
|
internal lazy var messageStatusContainerView: UIView = {
|
|
let result = UIView()
|
|
|
|
return result
|
|
}()
|
|
|
|
internal lazy var messageStatusLabel: UILabel = {
|
|
let result = UILabel()
|
|
result.accessibilityIdentifier = "Message sent status"
|
|
result.accessibilityLabel = "Message sent status"
|
|
result.font = .systemFont(ofSize: Values.verySmallFontSize)
|
|
result.themeTextColor = .messageBubble_deliveryStatus
|
|
|
|
return result
|
|
}()
|
|
|
|
internal lazy var messageStatusImageView: UIImageView = {
|
|
let result = UIImageView()
|
|
result.accessibilityIdentifier = "Message sent status tick"
|
|
result.accessibilityLabel = "Message sent status tick"
|
|
result.contentMode = .scaleAspectFit
|
|
result.themeTintColor = .messageBubble_deliveryStatus
|
|
|
|
return result
|
|
}()
|
|
|
|
internal lazy var messageStatusLabelPaddingView: UIView = UIView()
|
|
|
|
// MARK: - Settings
|
|
|
|
private static let messageStatusImageViewSize: CGFloat = 12
|
|
private static let authorLabelBottomSpacing: CGFloat = 4
|
|
private static let groupThreadHSpacing: CGFloat = 12
|
|
private static let authorLabelInset: CGFloat = 12
|
|
private static let replyButtonSize: CGFloat = 24
|
|
private static let maxBubbleTranslationX: CGFloat = 40
|
|
private static let swipeToReplyThreshold: CGFloat = 110
|
|
static let smallCornerRadius: CGFloat = 4
|
|
static let largeCornerRadius: CGFloat = 18
|
|
static let contactThreadHSpacing = Values.mediumSpacing
|
|
|
|
static var gutterSize: CGFloat = {
|
|
var result = groupThreadHSpacing + ProfilePictureView.Size.message.viewSize + groupThreadHSpacing
|
|
|
|
if UIDevice.current.isIPad {
|
|
result += 168
|
|
}
|
|
|
|
return result
|
|
}()
|
|
|
|
static var leftGutterSize: CGFloat { groupThreadHSpacing + ProfilePictureView.Size.message.viewSize + groupThreadHSpacing }
|
|
|
|
// MARK: Direction & Position
|
|
|
|
enum Direction { case incoming, outgoing }
|
|
|
|
// MARK: - Lifecycle
|
|
|
|
override func setUpViewHierarchy() {
|
|
super.setUpViewHierarchy()
|
|
|
|
// Author label
|
|
addSubview(authorLabel)
|
|
authorLabelTopConstraint.isActive = true
|
|
authorLabelHeightConstraint.isActive = true
|
|
|
|
// Profile picture view
|
|
addSubview(profilePictureView)
|
|
profilePictureViewLeadingConstraint.isActive = true
|
|
|
|
// Content view
|
|
addSubview(snContentView)
|
|
contentViewLeadingConstraint1.isActive = true
|
|
contentViewTopConstraint.isActive = true
|
|
contentViewTrailingConstraint1.isActive = true
|
|
snContentView.pin(.bottom, to: .bottom, of: profilePictureView)
|
|
|
|
// Bubble background view
|
|
bubbleBackgroundView.addSubview(bubbleView)
|
|
bubbleView.pin(to: bubbleBackgroundView)
|
|
|
|
// Reply button
|
|
addSubview(replyButton)
|
|
replyButton.addSubview(replyIconImageView)
|
|
replyIconImageView.center(in: replyButton)
|
|
replyButton.pin(.leading, to: .trailing, of: snContentView, withInset: Values.smallSpacing)
|
|
replyButton.center(.vertical, in: snContentView)
|
|
|
|
// Remaining constraints
|
|
authorLabel.pin(.leading, to: .leading, of: snContentView, withInset: VisibleMessageCell.authorLabelInset)
|
|
authorLabel.pin(.trailing, to: .trailing, of: self, withInset: -Values.mediumSpacing)
|
|
|
|
// Under bubble content
|
|
addSubview(underBubbleStackView)
|
|
underBubbleStackView.pin(.top, to: .bottom, of: snContentView, withInset: Values.verySmallSpacing)
|
|
underBubbleStackView.pin(.bottom, to: .bottom, of: self)
|
|
|
|
underBubbleStackView.addArrangedSubview(reactionContainerView)
|
|
underBubbleStackView.addArrangedSubview(messageStatusContainerView)
|
|
underBubbleStackView.addArrangedSubview(messageStatusLabelPaddingView)
|
|
|
|
messageStatusContainerView.addSubview(messageStatusLabel)
|
|
messageStatusContainerView.addSubview(messageStatusImageView)
|
|
messageStatusContainerView.addSubview(timerView)
|
|
|
|
reactionContainerView.widthAnchor
|
|
.constraint(lessThanOrEqualTo: underBubbleStackView.widthAnchor)
|
|
.isActive = true
|
|
messageStatusImageView.pin(.top, to: .top, of: messageStatusContainerView)
|
|
messageStatusImageView.pin(.bottom, to: .bottom, of: messageStatusContainerView)
|
|
messageStatusImageView.pin(.trailing, to: .trailing, of: messageStatusContainerView)
|
|
messageStatusImageView.set(.width, to: VisibleMessageCell.messageStatusImageViewSize)
|
|
messageStatusImageView.set(.height, to: VisibleMessageCell.messageStatusImageViewSize)
|
|
timerView.pin(.top, to: .top, of: messageStatusContainerView)
|
|
timerView.pin(.bottom, to: .bottom, of: messageStatusContainerView)
|
|
timerView.set(.width, to: VisibleMessageCell.messageStatusImageViewSize)
|
|
timerView.set(.height, to: VisibleMessageCell.messageStatusImageViewSize)
|
|
messageStatusLabel.center(.vertical, in: messageStatusContainerView)
|
|
messageStatusLabelPaddingView.pin(.leading, to: .leading, of: messageStatusContainerView)
|
|
messageStatusLabelPaddingView.pin(.trailing, to: .trailing, of: messageStatusContainerView)
|
|
}
|
|
|
|
override func setUpGestureRecognizers() {
|
|
let longPressRecognizer = UILongPressGestureRecognizer(target: self, action: #selector(handleLongPress))
|
|
addGestureRecognizer(longPressRecognizer)
|
|
|
|
let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleTap))
|
|
tapGestureRecognizer.numberOfTapsRequired = 1
|
|
addGestureRecognizer(tapGestureRecognizer)
|
|
|
|
let doubleTapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleDoubleTap))
|
|
doubleTapGestureRecognizer.numberOfTapsRequired = 2
|
|
addGestureRecognizer(doubleTapGestureRecognizer)
|
|
tapGestureRecognizer.require(toFail: doubleTapGestureRecognizer)
|
|
}
|
|
|
|
// MARK: - Updating
|
|
|
|
override func update(
|
|
with cellViewModel: MessageViewModel,
|
|
mediaCache: NSCache<NSString, AnyObject>,
|
|
playbackInfo: ConversationViewModel.PlaybackInfo?,
|
|
showExpandedReactions: Bool,
|
|
lastSearchText: String?
|
|
) {
|
|
self.viewModel = cellViewModel
|
|
|
|
// We want to add spacing between "clusters" of messages to indicate that time has
|
|
// passed (even if there wasn't enough time to warrant showing a date header)
|
|
let shouldAddTopInset: Bool = (
|
|
!cellViewModel.shouldShowDateHeader &&
|
|
cellViewModel.previousVariant?.isInfoMessage != true && (
|
|
cellViewModel.positionInCluster == .top ||
|
|
cellViewModel.isOnlyMessageInCluster
|
|
)
|
|
)
|
|
let isGroupThread: Bool = (
|
|
cellViewModel.threadVariant == .community ||
|
|
cellViewModel.threadVariant == .legacyGroup ||
|
|
cellViewModel.threadVariant == .group
|
|
)
|
|
|
|
// Profile picture view (should always be handled as a standard 'contact' profile picture)
|
|
let profileShouldBeVisible: Bool = (
|
|
cellViewModel.canHaveProfile &&
|
|
cellViewModel.shouldShowProfile &&
|
|
cellViewModel.profile != nil
|
|
)
|
|
profilePictureViewLeadingConstraint.constant = (isGroupThread ? VisibleMessageCell.groupThreadHSpacing : 0)
|
|
profilePictureView.isHidden = !cellViewModel.canHaveProfile
|
|
profilePictureView.alpha = (profileShouldBeVisible ? 1 : 0)
|
|
profilePictureView.update(
|
|
publicKey: cellViewModel.authorId,
|
|
threadVariant: .contact, // Always show the display picture in 'contact' mode
|
|
customImageData: nil,
|
|
profile: cellViewModel.profile,
|
|
profileIcon: (cellViewModel.isSenderOpenGroupModerator ? .crown : .none)
|
|
)
|
|
|
|
// Bubble view
|
|
contentViewLeadingConstraint1.isActive = (
|
|
cellViewModel.variant == .standardIncoming ||
|
|
cellViewModel.variant == .standardIncomingDeleted
|
|
)
|
|
contentViewLeadingConstraint1.constant = (isGroupThread ? VisibleMessageCell.groupThreadHSpacing : VisibleMessageCell.contactThreadHSpacing)
|
|
contentViewLeadingConstraint2.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
contentViewTopConstraint.constant = (cellViewModel.senderName == nil ? 0 : VisibleMessageCell.authorLabelBottomSpacing)
|
|
contentViewTrailingConstraint1.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
contentViewTrailingConstraint2.isActive = (
|
|
cellViewModel.variant == .standardIncoming ||
|
|
cellViewModel.variant == .standardIncomingDeleted
|
|
)
|
|
|
|
let bubbleBackgroundColor: ThemeValue = ((
|
|
cellViewModel.variant == .standardIncoming ||
|
|
cellViewModel.variant == .standardIncomingDeleted
|
|
) ? .messageBubble_incomingBackground : .messageBubble_outgoingBackground)
|
|
bubbleView.themeBackgroundColor = bubbleBackgroundColor
|
|
bubbleBackgroundView.themeBackgroundColor = bubbleBackgroundColor
|
|
updateBubbleViewCorners()
|
|
|
|
// Content view
|
|
populateContentView(
|
|
for: cellViewModel,
|
|
mediaCache: mediaCache,
|
|
playbackInfo: playbackInfo,
|
|
lastSearchText: lastSearchText
|
|
)
|
|
|
|
bubbleView.accessibilityIdentifier = "Message body"
|
|
bubbleView.accessibilityLabel = bodyTappableLabel?.attributedText?.string
|
|
bubbleView.isAccessibilityElement = true
|
|
|
|
// Author label
|
|
authorLabelTopConstraint.constant = (shouldAddTopInset ? Values.mediumSpacing : 0)
|
|
authorLabel.isHidden = (cellViewModel.senderName == nil)
|
|
authorLabel.text = cellViewModel.senderName
|
|
authorLabel.themeTextColor = .textPrimary
|
|
|
|
let authorLabelAvailableWidth: CGFloat = (VisibleMessageCell.getMaxWidth(for: cellViewModel) - 2 * VisibleMessageCell.authorLabelInset)
|
|
let authorLabelAvailableSpace = CGSize(width: authorLabelAvailableWidth, height: .greatestFiniteMagnitude)
|
|
let authorLabelSize = authorLabel.sizeThatFits(authorLabelAvailableSpace)
|
|
authorLabelHeightConstraint.constant = (cellViewModel.senderName != nil ? authorLabelSize.height : 0)
|
|
|
|
// Swipe to reply
|
|
if ContextMenuVC.viewModelCanReply(cellViewModel) {
|
|
addGestureRecognizer(panGestureRecognizer)
|
|
}
|
|
else {
|
|
removeGestureRecognizer(panGestureRecognizer)
|
|
}
|
|
|
|
// Under bubble content
|
|
underBubbleStackView.alignment = (cellViewModel.variant == .standardOutgoing ?
|
|
.trailing :
|
|
.leading
|
|
)
|
|
underBubbleStackViewIncomingLeadingConstraint.isActive = (cellViewModel.variant != .standardOutgoing)
|
|
underBubbleStackViewIncomingTrailingConstraint.isActive = (cellViewModel.variant != .standardOutgoing)
|
|
underBubbleStackViewOutgoingLeadingConstraint.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
underBubbleStackViewOutgoingTrailingConstraint.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
|
|
// Reaction view
|
|
reactionContainerView.isHidden = (cellViewModel.reactionInfo?.isEmpty != false)
|
|
populateReaction(
|
|
for: cellViewModel,
|
|
maxWidth: VisibleMessageCell.getMaxWidth(
|
|
for: cellViewModel,
|
|
includingOppositeGutter: false
|
|
),
|
|
showExpandedReactions: showExpandedReactions
|
|
)
|
|
|
|
// Message status image view
|
|
let (image, statusText, tintColor) = cellViewModel.state.statusIconInfo(
|
|
variant: cellViewModel.variant,
|
|
hasAtLeastOneReadReceipt: cellViewModel.hasAtLeastOneReadReceipt,
|
|
hasAttachments: (cellViewModel.attachments?.isEmpty == false)
|
|
)
|
|
messageStatusLabel.text = statusText
|
|
messageStatusLabel.themeTextColor = tintColor
|
|
messageStatusImageView.image = image
|
|
messageStatusLabel.accessibilityIdentifier = "Message sent status: \(statusText ?? "invalid")"
|
|
messageStatusImageView.themeTintColor = tintColor
|
|
messageStatusContainerView.isHidden = (
|
|
(cellViewModel.expiresInSeconds ?? 0) == 0 && (
|
|
cellViewModel.variant != .standardOutgoing ||
|
|
cellViewModel.variant == .infoCall ||
|
|
(
|
|
cellViewModel.state == .sent &&
|
|
!cellViewModel.isLastOutgoing
|
|
)
|
|
)
|
|
)
|
|
messageStatusLabelPaddingView.isHidden = (
|
|
messageStatusContainerView.isHidden ||
|
|
cellViewModel.isLast
|
|
)
|
|
|
|
// Timer
|
|
if
|
|
let expiresStartedAtMs: Double = cellViewModel.expiresStartedAtMs,
|
|
let expiresInSeconds: TimeInterval = cellViewModel.expiresInSeconds
|
|
{
|
|
let expirationTimestampMs: Double = (expiresStartedAtMs + (expiresInSeconds * 1000))
|
|
|
|
timerView.configure(
|
|
expirationTimestampMs: expirationTimestampMs,
|
|
initialDurationSeconds: expiresInSeconds
|
|
)
|
|
timerView.themeTintColor = tintColor
|
|
timerView.isHidden = false
|
|
messageStatusImageView.isHidden = true
|
|
}
|
|
else {
|
|
timerView.isHidden = true
|
|
messageStatusImageView.isHidden = false
|
|
}
|
|
|
|
timerViewOutgoingMessageConstraint.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
timerViewIncomingMessageConstraint.isActive = (
|
|
cellViewModel.variant == .standardIncoming ||
|
|
cellViewModel.variant == .standardIncomingDeleted
|
|
)
|
|
messageStatusLabelOutgoingMessageConstraint.isActive = (cellViewModel.variant == .standardOutgoing)
|
|
messageStatusLabelIncomingMessageConstraint.isActive = (
|
|
cellViewModel.variant == .standardIncoming ||
|
|
cellViewModel.variant == .standardIncomingDeleted
|
|
)
|
|
|
|
// Set the height of the underBubbleStackView to 0 if it has no content (need to do this
|
|
// otherwise it can randomly stretch)
|
|
underBubbleStackViewNoHeightConstraint.isActive = underBubbleStackView.arrangedSubviews
|
|
.filter { !$0.isHidden }
|
|
.isEmpty
|
|
}
|
|
|
|
private func populateContentView(
|
|
for cellViewModel: MessageViewModel,
|
|
mediaCache: NSCache<NSString, AnyObject>,
|
|
playbackInfo: ConversationViewModel.PlaybackInfo?,
|
|
lastSearchText: String?
|
|
) {
|
|
let bodyLabelTextColor: ThemeValue = (cellViewModel.variant == .standardOutgoing ?
|
|
.messageBubble_outgoingText :
|
|
.messageBubble_incomingText
|
|
)
|
|
|
|
snContentView.alignment = (cellViewModel.variant == .standardOutgoing ?
|
|
.trailing :
|
|
.leading
|
|
)
|
|
|
|
for subview in snContentView.arrangedSubviews {
|
|
snContentView.removeArrangedSubview(subview)
|
|
subview.removeFromSuperview()
|
|
}
|
|
for subview in bubbleView.subviews {
|
|
subview.removeFromSuperview()
|
|
}
|
|
albumView = nil
|
|
quoteView = nil
|
|
linkPreviewView = nil
|
|
documentView = nil
|
|
bodyTappableLabel = nil
|
|
|
|
// Handle the deleted state first (it's much simpler than the others)
|
|
guard cellViewModel.variant != .standardIncomingDeleted else {
|
|
let deletedMessageView: DeletedMessageView = DeletedMessageView(textColor: bodyLabelTextColor)
|
|
bubbleView.addSubview(deletedMessageView)
|
|
deletedMessageView.pin(to: bubbleView)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
return
|
|
}
|
|
|
|
// If it's an incoming media message and the thread isn't trusted then show the placeholder view
|
|
if cellViewModel.cellType != .textOnlyMessage && cellViewModel.variant == .standardIncoming && !cellViewModel.threadIsTrusted {
|
|
let mediaPlaceholderView = MediaPlaceholderView(cellViewModel: cellViewModel, textColor: bodyLabelTextColor)
|
|
bubbleView.addSubview(mediaPlaceholderView)
|
|
mediaPlaceholderView.pin(to: bubbleView)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
return
|
|
}
|
|
|
|
switch cellViewModel.cellType {
|
|
case .typingIndicator, .dateHeader, .unreadMarker: break
|
|
|
|
case .textOnlyMessage:
|
|
let inset: CGFloat = 12
|
|
let maxWidth: CGFloat = (VisibleMessageCell.getMaxWidth(for: cellViewModel) - 2 * inset)
|
|
|
|
if let linkPreview: LinkPreview = cellViewModel.linkPreview {
|
|
switch linkPreview.variant {
|
|
case .standard:
|
|
let linkPreviewView: LinkPreviewView = LinkPreviewView(maxWidth: maxWidth)
|
|
linkPreviewView.update(
|
|
with: LinkPreview.SentState(
|
|
linkPreview: linkPreview,
|
|
imageAttachment: cellViewModel.linkPreviewAttachment
|
|
),
|
|
isOutgoing: (cellViewModel.variant == .standardOutgoing),
|
|
delegate: self,
|
|
cellViewModel: cellViewModel,
|
|
bodyLabelTextColor: bodyLabelTextColor,
|
|
lastSearchText: lastSearchText
|
|
)
|
|
self.linkPreviewView = linkPreviewView
|
|
bubbleView.addSubview(linkPreviewView)
|
|
linkPreviewView.pin(to: bubbleView, withInset: 0)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
self.bodyTappableLabel = linkPreviewView.bodyTappableLabel
|
|
|
|
case .openGroupInvitation:
|
|
let openGroupInvitationView: OpenGroupInvitationView = OpenGroupInvitationView(
|
|
name: (linkPreview.title ?? ""),
|
|
url: linkPreview.url,
|
|
textColor: bodyLabelTextColor,
|
|
isOutgoing: (cellViewModel.variant == .standardOutgoing)
|
|
)
|
|
openGroupInvitationView.isAccessibilityElement = true
|
|
openGroupInvitationView.accessibilityIdentifier = "Community invitation"
|
|
openGroupInvitationView.accessibilityLabel = cellViewModel.linkPreview?.title
|
|
bubbleView.addSubview(openGroupInvitationView)
|
|
bubbleView.pin(to: openGroupInvitationView)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
}
|
|
}
|
|
else {
|
|
// Stack view
|
|
let stackView = UIStackView(arrangedSubviews: [])
|
|
stackView.axis = .vertical
|
|
stackView.spacing = 2
|
|
|
|
// Quote view
|
|
if let quote: Quote = cellViewModel.quote {
|
|
let hInset: CGFloat = 2
|
|
let quoteView: QuoteView = QuoteView(
|
|
for: .regular,
|
|
authorId: quote.authorId,
|
|
quotedText: quote.body,
|
|
threadVariant: cellViewModel.threadVariant,
|
|
currentUserPublicKey: cellViewModel.currentUserPublicKey,
|
|
currentUserBlinded15PublicKey: cellViewModel.currentUserBlinded15PublicKey,
|
|
currentUserBlinded25PublicKey: cellViewModel.currentUserBlinded25PublicKey,
|
|
direction: (cellViewModel.variant == .standardOutgoing ?
|
|
.outgoing :
|
|
.incoming
|
|
),
|
|
attachment: cellViewModel.quoteAttachment
|
|
)
|
|
self.quoteView = quoteView
|
|
let quoteViewContainer = UIView(wrapping: quoteView, withInsets: UIEdgeInsets(top: 0, leading: hInset, bottom: 0, trailing: hInset))
|
|
stackView.addArrangedSubview(quoteViewContainer)
|
|
}
|
|
|
|
// Body text view
|
|
let bodyTappableLabel = VisibleMessageCell.getBodyTappableLabel(
|
|
for: cellViewModel,
|
|
with: maxWidth,
|
|
textColor: bodyLabelTextColor,
|
|
searchText: lastSearchText,
|
|
delegate: self
|
|
)
|
|
self.bodyTappableLabel = bodyTappableLabel
|
|
stackView.addArrangedSubview(bodyTappableLabel)
|
|
|
|
// Constraints
|
|
bubbleView.addSubview(stackView)
|
|
stackView.pin(to: bubbleView, withInset: inset)
|
|
stackView.widthAnchor.constraint(lessThanOrEqualToConstant: maxWidth).isActive = true
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
}
|
|
|
|
case .mediaMessage:
|
|
// Body text view
|
|
if let body: String = cellViewModel.body, !body.isEmpty {
|
|
let inset: CGFloat = 12
|
|
let maxWidth: CGFloat = (VisibleMessageCell.getMaxWidth(for: cellViewModel) - 2 * inset)
|
|
let bodyTappableLabel = VisibleMessageCell.getBodyTappableLabel(
|
|
for: cellViewModel,
|
|
with: maxWidth,
|
|
textColor: bodyLabelTextColor,
|
|
searchText: lastSearchText,
|
|
delegate: self
|
|
)
|
|
|
|
self.bodyTappableLabel = bodyTappableLabel
|
|
bubbleView.addSubview(bodyTappableLabel)
|
|
bodyTappableLabel.pin(to: bubbleView, withInset: inset)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
}
|
|
|
|
// Album view
|
|
let maxMessageWidth: CGFloat = VisibleMessageCell.getMaxWidth(for: cellViewModel)
|
|
let albumView = MediaAlbumView(
|
|
mediaCache: mediaCache,
|
|
items: (cellViewModel.attachments?
|
|
.filter { $0.isVisualMedia })
|
|
.defaulting(to: []),
|
|
isOutgoing: (cellViewModel.variant == .standardOutgoing),
|
|
maxMessageWidth: maxMessageWidth
|
|
)
|
|
self.albumView = albumView
|
|
let size = getSize(for: cellViewModel)
|
|
albumView.set(.width, to: size.width)
|
|
albumView.set(.height, to: size.height)
|
|
albumView.loadMedia()
|
|
albumView.accessibilityLabel = "contentDescriptionMediaMessage".localized()
|
|
snContentView.addArrangedSubview(albumView)
|
|
|
|
unloadContent = { albumView.unloadMedia() }
|
|
|
|
case .voiceMessage:
|
|
guard let attachment: Attachment = cellViewModel.attachments?.first(where: { $0.isAudio }) else {
|
|
return
|
|
}
|
|
|
|
let voiceMessageView: VoiceMessageView = VoiceMessageView()
|
|
voiceMessageView.update(
|
|
with: attachment,
|
|
isPlaying: (playbackInfo?.state == .playing),
|
|
progress: (playbackInfo?.progress ?? 0),
|
|
playbackRate: (playbackInfo?.playbackRate ?? 1),
|
|
oldPlaybackRate: (playbackInfo?.oldPlaybackRate ?? 1)
|
|
)
|
|
|
|
bubbleView.addSubview(voiceMessageView)
|
|
voiceMessageView.pin(to: bubbleView)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
self.voiceMessageView = voiceMessageView
|
|
|
|
case .audio, .genericAttachment:
|
|
guard let attachment: Attachment = cellViewModel.attachments?.first else { preconditionFailure() }
|
|
|
|
let inset: CGFloat = 12
|
|
let maxWidth = (VisibleMessageCell.getMaxWidth(for: cellViewModel) - 2 * inset)
|
|
|
|
// Stack view
|
|
let stackView = UIStackView(arrangedSubviews: [])
|
|
stackView.axis = .vertical
|
|
stackView.spacing = Values.smallSpacing
|
|
|
|
// Document view
|
|
let documentView = DocumentView(attachment: attachment, textColor: bodyLabelTextColor)
|
|
self.documentView = documentView
|
|
stackView.addArrangedSubview(documentView)
|
|
|
|
// Body text view
|
|
if let body: String = cellViewModel.body, !body.isEmpty { // delegate should always be set at this point
|
|
let bodyContainerView: UIView = UIView()
|
|
let bodyTappableLabel = VisibleMessageCell.getBodyTappableLabel(
|
|
for: cellViewModel,
|
|
with: maxWidth,
|
|
textColor: bodyLabelTextColor,
|
|
searchText: lastSearchText,
|
|
delegate: self
|
|
)
|
|
|
|
self.bodyTappableLabel = bodyTappableLabel
|
|
bodyContainerView.addSubview(bodyTappableLabel)
|
|
bodyTappableLabel.pin(.top, to: .top, of: bodyContainerView)
|
|
bodyTappableLabel.pin(.leading, to: .leading, of: bodyContainerView, withInset: 12)
|
|
bodyTappableLabel.pin(.trailing, to: .trailing, of: bodyContainerView, withInset: -12)
|
|
bodyTappableLabel.pin(.bottom, to: .bottom, of: bodyContainerView, withInset: -12)
|
|
stackView.addArrangedSubview(bodyContainerView)
|
|
}
|
|
|
|
bubbleView.addSubview(stackView)
|
|
stackView.pin(to: bubbleView)
|
|
snContentView.addArrangedSubview(bubbleBackgroundView)
|
|
}
|
|
}
|
|
|
|
private func populateReaction(
|
|
for cellViewModel: MessageViewModel,
|
|
maxWidth: CGFloat,
|
|
showExpandedReactions: Bool
|
|
) {
|
|
let reactions: OrderedDictionary<EmojiWithSkinTones, ReactionViewModel> = (cellViewModel.reactionInfo ?? [])
|
|
.reduce(into: OrderedDictionary()) { result, reactionInfo in
|
|
guard let emoji: EmojiWithSkinTones = EmojiWithSkinTones(rawValue: reactionInfo.reaction.emoji) else {
|
|
return
|
|
}
|
|
|
|
let isSelfSend: Bool = (reactionInfo.reaction.authorId == cellViewModel.currentUserPublicKey)
|
|
|
|
if let value: ReactionViewModel = result.value(forKey: emoji) {
|
|
result.replace(
|
|
key: emoji,
|
|
value: ReactionViewModel(
|
|
emoji: emoji,
|
|
number: (value.number + Int(reactionInfo.reaction.count)),
|
|
showBorder: (value.showBorder || isSelfSend)
|
|
)
|
|
)
|
|
}
|
|
else {
|
|
result.append(
|
|
key: emoji,
|
|
value: ReactionViewModel(
|
|
emoji: emoji,
|
|
number: Int(reactionInfo.reaction.count),
|
|
showBorder: isSelfSend
|
|
)
|
|
)
|
|
}
|
|
}
|
|
|
|
reactionContainerView.update(
|
|
reactions.orderedValues,
|
|
maxWidth: maxWidth,
|
|
showingAllReactions: showExpandedReactions,
|
|
showNumbers: (
|
|
cellViewModel.threadVariant == .legacyGroup ||
|
|
cellViewModel.threadVariant == .group ||
|
|
cellViewModel.threadVariant == .community
|
|
)
|
|
)
|
|
}
|
|
|
|
override func layoutSubviews() {
|
|
super.layoutSubviews()
|
|
updateBubbleViewCorners()
|
|
}
|
|
|
|
private func updateBubbleViewCorners() {
|
|
let cornersToRound: UIRectCorner = .allCorners
|
|
|
|
bubbleBackgroundView.layer.cornerRadius = VisibleMessageCell.largeCornerRadius
|
|
bubbleBackgroundView.layer.maskedCorners = getCornerMask(from: cornersToRound)
|
|
bubbleView.layer.cornerRadius = VisibleMessageCell.largeCornerRadius
|
|
bubbleView.layer.maskedCorners = getCornerMask(from: cornersToRound)
|
|
}
|
|
|
|
override func dynamicUpdate(with cellViewModel: MessageViewModel, playbackInfo: ConversationViewModel.PlaybackInfo?) {
|
|
guard cellViewModel.variant != .standardIncomingDeleted else { return }
|
|
|
|
// If it's an incoming media message and the thread isn't trusted then show the placeholder view
|
|
if cellViewModel.cellType != .textOnlyMessage && cellViewModel.variant == .standardIncoming && !cellViewModel.threadIsTrusted {
|
|
return
|
|
}
|
|
|
|
switch cellViewModel.cellType {
|
|
case .voiceMessage:
|
|
guard let attachment: Attachment = cellViewModel.attachments?.first(where: { $0.isAudio }) else {
|
|
return
|
|
}
|
|
|
|
self.voiceMessageView?.update(
|
|
with: attachment,
|
|
isPlaying: (playbackInfo?.state == .playing),
|
|
progress: (playbackInfo?.progress ?? 0),
|
|
playbackRate: (playbackInfo?.playbackRate ?? 1),
|
|
oldPlaybackRate: (playbackInfo?.oldPlaybackRate ?? 1)
|
|
)
|
|
|
|
default: break
|
|
}
|
|
}
|
|
|
|
override func prepareForReuse() {
|
|
super.prepareForReuse()
|
|
|
|
unloadContent?()
|
|
viewsToMoveForReply.forEach { $0.transform = .identity }
|
|
replyButton.alpha = 0
|
|
timerView.prepareForReuse()
|
|
}
|
|
|
|
// MARK: - Interaction
|
|
|
|
override func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool {
|
|
return true // Needed for the pan gesture recognizer to work with the table view's pan gesture recognizer
|
|
}
|
|
|
|
override func gestureRecognizerShouldBegin(_ gestureRecognizer: UIGestureRecognizer) -> Bool {
|
|
if gestureRecognizer == panGestureRecognizer {
|
|
let v = panGestureRecognizer.velocity(in: self)
|
|
// Only allow swipes to the left; allowing swipes to the right gets in the way of
|
|
// the default iOS swipe to go back gesture
|
|
guard
|
|
(Singleton.hasAppContext && Singleton.appContext.isRTL && v.x > 0) ||
|
|
(!Singleton.hasAppContext || !Singleton.appContext.isRTL && v.x < 0)
|
|
else { return false }
|
|
|
|
return abs(v.x) > abs(v.y) // It has to be more horizontal than vertical
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func highlight() {
|
|
let shadowColor: ThemeValue = (ThemeManager.currentTheme.interfaceStyle == .light ?
|
|
.black :
|
|
.primary
|
|
)
|
|
let opacity: Float = (ThemeManager.currentTheme.interfaceStyle == .light ?
|
|
0.5 :
|
|
1
|
|
)
|
|
|
|
DispatchQueue.main.async { [weak self] in
|
|
let oldMasksToBounds: Bool = (self?.layer.masksToBounds ?? false)
|
|
self?.layer.masksToBounds = false
|
|
self?.bubbleBackgroundView.setShadow(radius: 10, opacity: opacity, offset: .zero, color: shadowColor)
|
|
|
|
UIView.animate(
|
|
withDuration: 1.6,
|
|
delay: 0,
|
|
options: .curveEaseInOut,
|
|
animations: {
|
|
self?.bubbleBackgroundView.setShadow(radius: 0, opacity: 0, offset: .zero, color: .clear)
|
|
},
|
|
completion: { _ in
|
|
self?.layer.masksToBounds = oldMasksToBounds
|
|
}
|
|
)
|
|
}
|
|
}
|
|
|
|
@objc func handleLongPress(_ gestureRecognizer: UITapGestureRecognizer) {
|
|
if [ .ended, .cancelled, .failed ].contains(gestureRecognizer.state) {
|
|
isHandlingLongPress = false
|
|
return
|
|
}
|
|
guard !isHandlingLongPress, let cellViewModel: MessageViewModel = self.viewModel else { return }
|
|
|
|
let location = gestureRecognizer.location(in: self)
|
|
|
|
if reactionContainerView.bounds.contains(reactionContainerView.convert(location, from: self)) {
|
|
let convertedLocation = reactionContainerView.convert(location, from: self)
|
|
|
|
for reactionView in reactionContainerView.reactionViews {
|
|
if reactionContainerView.convert(reactionView.frame, from: reactionView.superview).contains(convertedLocation) {
|
|
delegate?.showReactionList(cellViewModel, selectedReaction: reactionView.viewModel.emoji)
|
|
break
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
delegate?.handleItemLongPressed(cellViewModel)
|
|
}
|
|
|
|
isHandlingLongPress = true
|
|
}
|
|
|
|
@objc private func handleTap(_ gestureRecognizer: UITapGestureRecognizer) { onTap(gestureRecognizer) }
|
|
|
|
private func onTap(_ gestureRecognizer: UITapGestureRecognizer, using dependencies: Dependencies = Dependencies()) {
|
|
guard let cellViewModel: MessageViewModel = self.viewModel else { return }
|
|
|
|
let location = gestureRecognizer.location(in: self)
|
|
|
|
if profilePictureView.bounds.contains(profilePictureView.convert(location, from: self)), cellViewModel.shouldShowProfile {
|
|
// For open groups only attempt to start a conversation if the author has a blinded id
|
|
guard cellViewModel.threadVariant != .community else {
|
|
// FIXME: Add in support for opening a conversation with a 'blinded25' id
|
|
guard (try? SessionId.Prefix(from: cellViewModel.authorId)) == .blinded15 else { return }
|
|
|
|
delegate?.startThread(
|
|
with: cellViewModel.authorId,
|
|
openGroupServer: cellViewModel.threadOpenGroupServer,
|
|
openGroupPublicKey: cellViewModel.threadOpenGroupPublicKey
|
|
)
|
|
return
|
|
}
|
|
|
|
delegate?.startThread(
|
|
with: cellViewModel.authorId,
|
|
openGroupServer: nil,
|
|
openGroupPublicKey: nil
|
|
)
|
|
}
|
|
else if replyButton.alpha > 0 && replyButton.bounds.contains(replyButton.convert(location, from: self)) {
|
|
UIImpactFeedbackGenerator(style: .heavy).impactOccurred()
|
|
reply()
|
|
}
|
|
else if reactionContainerView.bounds.contains(reactionContainerView.convert(location, from: self)) {
|
|
let convertedLocation = reactionContainerView.convert(location, from: self)
|
|
|
|
for reactionView in reactionContainerView.reactionViews {
|
|
if reactionContainerView.convert(reactionView.frame, from: reactionView.superview).contains(convertedLocation) {
|
|
|
|
if reactionView.viewModel.showBorder {
|
|
delegate?.removeReact(cellViewModel, for: reactionView.viewModel.emoji, using: dependencies)
|
|
}
|
|
else {
|
|
delegate?.react(cellViewModel, with: reactionView.viewModel.emoji, using: dependencies)
|
|
}
|
|
return
|
|
}
|
|
}
|
|
|
|
if let expandButton = reactionContainerView.expandButton, expandButton.bounds.contains(expandButton.convert(location, from: self)) {
|
|
reactionContainerView.showAllEmojis()
|
|
delegate?.needsLayout(for: cellViewModel, expandingReactions: true)
|
|
}
|
|
|
|
if reactionContainerView.collapseButton.frame.contains(convertedLocation) {
|
|
reactionContainerView.showLessEmojis()
|
|
delegate?.needsLayout(for: cellViewModel, expandingReactions: false)
|
|
}
|
|
}
|
|
else if snContentView.bounds.contains(snContentView.convert(location, from: self)) {
|
|
delegate?.handleItemTapped(cellViewModel, cell: self, cellLocation: location, using: dependencies)
|
|
}
|
|
}
|
|
|
|
@objc private func handleDoubleTap() {
|
|
guard let cellViewModel: MessageViewModel = self.viewModel else { return }
|
|
|
|
delegate?.handleItemDoubleTapped(cellViewModel)
|
|
}
|
|
|
|
@objc private func handlePan(_ gestureRecognizer: UIPanGestureRecognizer) {
|
|
guard let cellViewModel: MessageViewModel = self.viewModel else { return }
|
|
|
|
let translationX = gestureRecognizer
|
|
.translation(in: self)
|
|
.x
|
|
.clamp(
|
|
(Singleton.hasAppContext && Singleton.appContext.isRTL ? 0 : -CGFloat.greatestFiniteMagnitude),
|
|
(Singleton.hasAppContext && Singleton.appContext.isRTL ? CGFloat.greatestFiniteMagnitude : 0)
|
|
)
|
|
|
|
switch gestureRecognizer.state {
|
|
case .began: delegate?.handleItemSwiped(cellViewModel, state: .began)
|
|
|
|
case .changed:
|
|
// The idea here is to asymptotically approach a maximum drag distance
|
|
let damping: CGFloat = 20
|
|
let sign: CGFloat = (Singleton.hasAppContext && Singleton.appContext.isRTL ? 1 : -1)
|
|
let x = (damping * (sqrt(abs(translationX)) / sqrt(damping))) * sign
|
|
viewsToMoveForReply.forEach { $0.transform = CGAffineTransform(translationX: x, y: 0) }
|
|
|
|
if timerView.isHidden {
|
|
replyButton.alpha = abs(translationX) / VisibleMessageCell.maxBubbleTranslationX
|
|
}
|
|
else {
|
|
replyButton.alpha = 0 // Always hide the reply button if the timer view is showing, otherwise they can overlap
|
|
}
|
|
|
|
if abs(translationX) > VisibleMessageCell.swipeToReplyThreshold && abs(previousX) < VisibleMessageCell.swipeToReplyThreshold {
|
|
UIImpactFeedbackGenerator(style: .heavy).impactOccurred() // Let the user know when they've hit the swipe to reply threshold
|
|
}
|
|
previousX = translationX
|
|
|
|
case .ended, .cancelled:
|
|
if abs(translationX) > VisibleMessageCell.swipeToReplyThreshold {
|
|
delegate?.handleItemSwiped(cellViewModel, state: .ended)
|
|
reply()
|
|
}
|
|
else {
|
|
delegate?.handleItemSwiped(cellViewModel, state: .cancelled)
|
|
resetReply()
|
|
}
|
|
|
|
default: break
|
|
}
|
|
}
|
|
|
|
func tapableLabel(_ label: TappableLabel, didTapUrl url: String, atRange range: NSRange) {
|
|
delegate?.openUrl(url)
|
|
}
|
|
|
|
private func resetReply() {
|
|
UIView.animate(withDuration: 0.25) { [weak self] in
|
|
self?.viewsToMoveForReply.forEach { $0.transform = .identity }
|
|
self?.replyButton.alpha = 0
|
|
}
|
|
}
|
|
|
|
private func reply(using dependencies: Dependencies = Dependencies()) {
|
|
guard let cellViewModel: MessageViewModel = self.viewModel else { return }
|
|
|
|
resetReply()
|
|
delegate?.handleReplyButtonTapped(for: cellViewModel, using: dependencies)
|
|
}
|
|
|
|
// MARK: - Convenience
|
|
|
|
private func getCornerMask(from rectCorner: UIRectCorner) -> CACornerMask {
|
|
guard !rectCorner.contains(.allCorners) else {
|
|
return [ .layerMaxXMinYCorner, .layerMinXMinYCorner, .layerMaxXMaxYCorner, .layerMinXMaxYCorner]
|
|
}
|
|
|
|
var cornerMask = CACornerMask()
|
|
if rectCorner.contains(.topRight) { cornerMask.insert(.layerMaxXMinYCorner) }
|
|
if rectCorner.contains(.topLeft) { cornerMask.insert(.layerMinXMinYCorner) }
|
|
if rectCorner.contains(.bottomRight) { cornerMask.insert(.layerMaxXMaxYCorner) }
|
|
if rectCorner.contains(.bottomLeft) { cornerMask.insert(.layerMinXMaxYCorner) }
|
|
return cornerMask
|
|
}
|
|
|
|
private static func getFontSize(for cellViewModel: MessageViewModel) -> CGFloat {
|
|
let baselineFontSize = Values.mediumFontSize
|
|
|
|
guard cellViewModel.containsOnlyEmoji == true else { return baselineFontSize }
|
|
|
|
switch (cellViewModel.glyphCount ?? 0) {
|
|
case 1: return baselineFontSize + 30
|
|
case 2: return baselineFontSize + 24
|
|
case 3, 4, 5: return baselineFontSize + 18
|
|
default: return baselineFontSize
|
|
}
|
|
}
|
|
|
|
private func getSize(for cellViewModel: MessageViewModel) -> CGSize {
|
|
guard let mediaAttachments: [Attachment] = cellViewModel.attachments?.filter({ $0.isVisualMedia }) else {
|
|
preconditionFailure()
|
|
}
|
|
|
|
let maxMessageWidth = VisibleMessageCell.getMaxWidth(for: cellViewModel)
|
|
let defaultSize = MediaAlbumView.layoutSize(forMaxMessageWidth: maxMessageWidth, items: mediaAttachments)
|
|
|
|
guard
|
|
let firstAttachment: Attachment = mediaAttachments.first,
|
|
var width: CGFloat = firstAttachment.width.map({ CGFloat($0) }),
|
|
var height: CGFloat = firstAttachment.height.map({ CGFloat($0) }),
|
|
mediaAttachments.count == 1,
|
|
width > 0,
|
|
height > 0
|
|
else { return defaultSize }
|
|
|
|
// Honor the content aspect ratio for single media
|
|
let size: CGSize = CGSize(width: width, height: height)
|
|
var aspectRatio = (size.width / size.height)
|
|
// Clamp the aspect ratio so that very thin/wide content still looks alright
|
|
let minAspectRatio: CGFloat = 0.35
|
|
let maxAspectRatio = 1 / minAspectRatio
|
|
let maxSize = CGSize(width: maxMessageWidth, height: maxMessageWidth)
|
|
aspectRatio = aspectRatio.clamp(minAspectRatio, maxAspectRatio)
|
|
|
|
if aspectRatio > 1 {
|
|
width = maxSize.width
|
|
height = width / aspectRatio
|
|
}
|
|
else {
|
|
height = maxSize.height
|
|
width = height * aspectRatio
|
|
}
|
|
|
|
// Don't blow up small images unnecessarily
|
|
let minSize: CGFloat = 150
|
|
let shortSourceDimension = min(size.width, size.height)
|
|
let shortDestinationDimension = min(width, height)
|
|
|
|
if shortDestinationDimension > minSize && shortDestinationDimension > shortSourceDimension {
|
|
let factor = minSize / shortDestinationDimension
|
|
width *= factor; height *= factor
|
|
}
|
|
|
|
return CGSize(width: width, height: height)
|
|
}
|
|
|
|
static func getMaxWidth(for cellViewModel: MessageViewModel, includingOppositeGutter: Bool = true) -> CGFloat {
|
|
let screen: CGRect = UIScreen.main.bounds
|
|
let width: CGFloat = UIDevice.current.isIPad ? screen.width * 0.75 : screen.width
|
|
let oppositeEdgePadding: CGFloat = (includingOppositeGutter ? gutterSize : contactThreadHSpacing)
|
|
|
|
switch cellViewModel.variant {
|
|
case .standardOutgoing:
|
|
return (width - contactThreadHSpacing - oppositeEdgePadding)
|
|
|
|
case .standardIncoming, .standardIncomingDeleted:
|
|
let isGroupThread = (
|
|
cellViewModel.threadVariant == .community ||
|
|
cellViewModel.threadVariant == .legacyGroup ||
|
|
cellViewModel.threadVariant == .group
|
|
)
|
|
let leftGutterSize = (isGroupThread ? leftGutterSize : contactThreadHSpacing)
|
|
|
|
return (width - leftGutterSize - oppositeEdgePadding)
|
|
|
|
default: preconditionFailure()
|
|
}
|
|
}
|
|
|
|
static func getBodyAttributedText(
|
|
for cellViewModel: MessageViewModel,
|
|
theme: Theme,
|
|
primaryColor: Theme.PrimaryColor,
|
|
textColor: ThemeValue,
|
|
searchText: String?
|
|
) -> NSMutableAttributedString?
|
|
{
|
|
guard
|
|
let body: String = cellViewModel.body,
|
|
!body.isEmpty,
|
|
let actualTextColor: UIColor = theme.color(for: textColor),
|
|
let backgroundPrimaryColor: UIColor = theme.color(for: .backgroundPrimary),
|
|
let textPrimaryColor: UIColor = theme.color(for: .textPrimary)
|
|
else { return nil }
|
|
|
|
let isOutgoing: Bool = (cellViewModel.variant == .standardOutgoing)
|
|
|
|
let attributedText: NSMutableAttributedString = NSMutableAttributedString(
|
|
attributedString: MentionUtilities.highlightMentions(
|
|
in: body,
|
|
threadVariant: cellViewModel.threadVariant,
|
|
currentUserPublicKey: cellViewModel.currentUserPublicKey,
|
|
currentUserBlinded15PublicKey: cellViewModel.currentUserBlinded15PublicKey,
|
|
currentUserBlinded25PublicKey: cellViewModel.currentUserBlinded25PublicKey,
|
|
location: (isOutgoing ? .outgoingMessage : .incomingMessage),
|
|
textColor: actualTextColor,
|
|
theme: theme,
|
|
primaryColor: primaryColor,
|
|
attributes: [
|
|
.foregroundColor: actualTextColor,
|
|
.font: UIFont.systemFont(ofSize: getFontSize(for: cellViewModel))
|
|
]
|
|
)
|
|
)
|
|
|
|
// Custom handle links
|
|
let links: [URL: NSRange] = {
|
|
guard let detector: NSDataDetector = try? NSDataDetector(types: NSTextCheckingResult.CheckingType.link.rawValue) else {
|
|
return [:]
|
|
}
|
|
|
|
// Note: The 'String.count' value is based on actual character counts whereas
|
|
// NSAttributedString and NSRange are both based on UTF-16 encoded lengths, so
|
|
// in order to avoid strings which contain emojis breaking strings which end
|
|
// with URLs we need to use the 'String.utf16.count' value when creating the range
|
|
return detector
|
|
.matches(
|
|
in: attributedText.string,
|
|
options: [],
|
|
range: NSRange(location: 0, length: attributedText.string.utf16.count)
|
|
)
|
|
.reduce(into: [:]) { result, match in
|
|
guard
|
|
let matchUrl: URL = match.url,
|
|
let originalRange: Range = Range(match.range, in: attributedText.string)
|
|
else { return }
|
|
|
|
/// If the URL entered didn't have a scheme it will default to 'http', we want to catch this and
|
|
/// set the scheme to 'https' instead as we don't load previews for 'http' so this will result
|
|
/// in more previews actually getting loaded without forcing the user to enter 'https://' before
|
|
/// every URL they enter
|
|
let originalString: String = String(attributedText.string[originalRange])
|
|
|
|
guard matchUrl.absoluteString != "http://\(originalString)" else {
|
|
guard let httpsUrl: URL = URL(string: "https://\(originalString)") else {
|
|
return
|
|
}
|
|
|
|
result[httpsUrl] = match.range
|
|
return
|
|
}
|
|
|
|
result[matchUrl] = match.range
|
|
}
|
|
}()
|
|
|
|
for (linkUrl, urlRange) in links {
|
|
attributedText.addAttributes(
|
|
[
|
|
.font: UIFont.systemFont(ofSize: getFontSize(for: cellViewModel)),
|
|
.foregroundColor: actualTextColor,
|
|
.underlineColor: actualTextColor,
|
|
.underlineStyle: NSUnderlineStyle.single.rawValue,
|
|
.attachment: linkUrl
|
|
],
|
|
range: urlRange
|
|
)
|
|
}
|
|
|
|
// If there is a valid search term then highlight each part that matched
|
|
if let searchText = searchText, searchText.count >= ConversationSearchController.minimumSearchTextLength {
|
|
let normalizedBody: String = attributedText.string.lowercased()
|
|
|
|
SessionThreadViewModel.searchTermParts(searchText)
|
|
.map { part -> String in
|
|
guard part.hasPrefix("\"") && part.hasSuffix("\"") else { return part }
|
|
|
|
let partRange = (part.index(after: part.startIndex)..<part.index(before: part.endIndex))
|
|
return String(part[partRange])
|
|
}
|
|
.forEach { part in
|
|
// Highlight all ranges of the text (Note: The search logic only finds
|
|
// results that start with the term so we use the regex below to ensure
|
|
// we only highlight those cases)
|
|
normalizedBody
|
|
.ranges(
|
|
of: (Singleton.appContext.isRTL ?
|
|
"(\(part.lowercased()))(^|[^a-zA-Z0-9])" :
|
|
"(^|[^a-zA-Z0-9])(\(part.lowercased()))"
|
|
),
|
|
options: [.regularExpression]
|
|
)
|
|
.forEach { range in
|
|
let targetRange: Range<String.Index> = {
|
|
let term: String = String(normalizedBody[range])
|
|
|
|
// If the matched term doesn't actually match the "part" value then it means
|
|
// we've matched a term after a non-alphanumeric character so need to shift
|
|
// the range over by 1
|
|
guard term.starts(with: part.lowercased()) else {
|
|
return (normalizedBody.index(after: range.lowerBound)..<range.upperBound)
|
|
}
|
|
|
|
return range
|
|
}()
|
|
|
|
let legacyRange: NSRange = NSRange(targetRange, in: normalizedBody)
|
|
attributedText.addThemeAttribute(.background(backgroundPrimaryColor), range: legacyRange)
|
|
attributedText.addThemeAttribute(.foreground(textPrimaryColor), range: legacyRange)
|
|
}
|
|
}
|
|
}
|
|
|
|
return attributedText
|
|
}
|
|
|
|
static func getBodyTappableLabel(
|
|
for cellViewModel: MessageViewModel,
|
|
with availableWidth: CGFloat,
|
|
textColor: ThemeValue,
|
|
searchText: String?,
|
|
delegate: TappableLabelDelegate?
|
|
) -> TappableLabel {
|
|
let result: TappableLabel = TappableLabel()
|
|
result.setContentCompressionResistancePriority(.required, for: .vertical)
|
|
result.themeBackgroundColor = .clear
|
|
result.isOpaque = false
|
|
result.isUserInteractionEnabled = true
|
|
result.delegate = delegate
|
|
|
|
ThemeManager.onThemeChange(observer: result) { [weak result] theme, primaryColor in
|
|
let hasPreviousSetText: Bool = ((result?.attributedText?.length ?? 0) > 0)
|
|
|
|
result?.attributedText = Self.getBodyAttributedText(
|
|
for: cellViewModel,
|
|
theme: theme,
|
|
primaryColor: primaryColor,
|
|
textColor: textColor,
|
|
searchText: searchText
|
|
)
|
|
|
|
if let result: TappableLabel = result, !hasPreviousSetText {
|
|
let availableSpace = CGSize(width: availableWidth, height: .greatestFiniteMagnitude)
|
|
let size = result.sizeThatFits(availableSpace)
|
|
result.set(.height, to: size.height)
|
|
}
|
|
}
|
|
|
|
return result
|
|
}
|
|
}
|