session-ios/Session/Shared/ConversationCell.swift

398 lines
19 KiB
Swift
Raw Normal View History

2020-11-09 06:03:59 +01:00
import UIKit
import SessionUIKit
2019-11-28 06:42:07 +01:00
final class ConversationCell : UITableViewCell {
2022-01-17 06:56:51 +01:00
var isShowingGlobalSearchResult = false
var threadViewModel: ThreadViewModel! {
didSet {
isShowingGlobalSearchResult ? updateForSearchResult() : update()
}
}
2019-11-28 06:42:07 +01:00
2019-11-29 06:30:01 +01:00
static let reuseIdentifier = "ConversationCell"
2019-11-28 06:42:07 +01:00
2021-03-01 04:04:54 +01:00
// MARK: UI Components
private let accentLineView = UIView()
2019-11-28 06:42:07 +01:00
private lazy var profilePictureView = ProfilePictureView()
private lazy var displayNameLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.mediumFontSize)
result.textColor = Colors.text
result.lineBreakMode = .byTruncatingTail
return result
}()
2021-03-01 04:04:54 +01:00
private lazy var unreadCountView: UIView = {
let result = UIView()
result.backgroundColor = Colors.text.withAlphaComponent(Values.veryLowOpacity)
let size = ConversationCell.unreadCountViewSize
result.set(.width, greaterThanOrEqualTo: size)
2021-03-01 04:04:54 +01:00
result.set(.height, to: size)
result.layer.masksToBounds = true
result.layer.cornerRadius = size / 2
return result
}()
private lazy var unreadCountLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.verySmallFontSize)
result.textColor = Colors.text
result.textAlignment = .center
return result
}()
private lazy var hasMentionView: UIView = {
let result = UIView()
result.backgroundColor = Colors.accent
let size = ConversationCell.unreadCountViewSize
result.set(.width, to: size)
result.set(.height, to: size)
result.layer.masksToBounds = true
result.layer.cornerRadius = size / 2
return result
}()
private lazy var hasMentionLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.verySmallFontSize)
result.textColor = Colors.text
result.text = "@"
result.textAlignment = .center
return result
}()
2021-11-17 05:51:53 +01:00
private lazy var isPinnedIcon: UIImageView = {
let result = UIImageView(image: UIImage(named: "Pin")!.withRenderingMode(.alwaysTemplate))
result.contentMode = .scaleAspectFit
let size = ConversationCell.unreadCountViewSize
result.set(.width, to: size)
result.set(.height, to: size)
2021-11-30 03:52:31 +01:00
result.tintColor = Colors.pinIcon
2021-11-17 05:51:53 +01:00
result.layer.masksToBounds = true
return result
}()
2019-11-29 06:30:01 +01:00
private lazy var timestampLabel: UILabel = {
let result = UILabel()
result.font = .systemFont(ofSize: Values.smallFontSize)
result.textColor = Colors.text
result.lineBreakMode = .byTruncatingTail
2021-01-29 01:46:32 +01:00
result.alpha = Values.lowOpacity
2019-11-29 06:30:01 +01:00
return result
}()
2019-11-28 06:42:07 +01:00
private lazy var snippetLabel: UILabel = {
let result = UILabel()
result.font = .systemFont(ofSize: Values.smallFontSize)
result.textColor = Colors.text
result.lineBreakMode = .byTruncatingTail
return result
}()
private lazy var typingIndicatorView = TypingIndicatorView()
2019-11-29 06:30:01 +01:00
private lazy var statusIndicatorView: UIImageView = {
let result = UIImageView()
result.contentMode = .scaleAspectFit
2021-03-01 04:04:54 +01:00
result.layer.cornerRadius = ConversationCell.statusIndicatorSize / 2
result.layer.masksToBounds = true
2019-11-29 06:30:01 +01:00
return result
}()
private lazy var topLabelStackView: UIStackView = {
let result = UIStackView()
result.axis = .horizontal
result.alignment = .center
result.spacing = Values.smallSpacing / 2 // Effectively Values.smallSpacing because there'll be spacing before and after the invisible spacer
return result
}()
private lazy var bottomLabelStackView: UIStackView = {
let result = UIStackView()
result.axis = .horizontal
result.alignment = .center
result.spacing = Values.smallSpacing / 2 // Effectively Values.smallSpacing because there'll be spacing before and after the invisible spacer
return result
}()
2021-01-29 01:46:32 +01:00
// MARK: Settings
public static let unreadCountViewSize: CGFloat = 20
2021-03-01 04:04:54 +01:00
private static let statusIndicatorSize: CGFloat = 14
2021-01-29 01:46:32 +01:00
2019-11-28 06:42:07 +01:00
// MARK: Initialization
override init(style: UITableViewCell.CellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
setUpViewHierarchy()
}
required init?(coder: NSCoder) {
super.init(coder: coder)
setUpViewHierarchy()
}
private func setUpViewHierarchy() {
2020-03-20 00:06:33 +01:00
let cellHeight: CGFloat = 68
2021-03-01 04:04:54 +01:00
// Background color
backgroundColor = Colors.cellBackground
2021-03-01 04:04:54 +01:00
// Highlight color
2019-11-29 06:30:01 +01:00
let selectedBackgroundView = UIView()
selectedBackgroundView.backgroundColor = Colors.cellSelected
2019-11-29 06:30:01 +01:00
self.selectedBackgroundView = selectedBackgroundView
2021-03-01 04:04:54 +01:00
// Accent line view
accentLineView.set(.width, to: Values.accentLineThickness)
accentLineView.set(.height, to: cellHeight)
// Profile picture view
2019-11-28 06:42:07 +01:00
let profilePictureViewSize = Values.mediumProfilePictureSize
profilePictureView.set(.width, to: profilePictureViewSize)
profilePictureView.set(.height, to: profilePictureViewSize)
profilePictureView.size = profilePictureViewSize
2021-03-01 04:04:54 +01:00
// Unread count view
unreadCountView.addSubview(unreadCountLabel)
unreadCountLabel.pin([ VerticalEdge.top, VerticalEdge.bottom ], to: unreadCountView)
unreadCountView.pin(.leading, to: .leading, of: unreadCountLabel, withInset: -4)
unreadCountView.pin(.trailing, to: .trailing, of: unreadCountLabel, withInset: 4)
// Has mention view
hasMentionView.addSubview(hasMentionLabel)
hasMentionLabel.pin(to: hasMentionView)
2021-03-01 04:04:54 +01:00
// Label stack view
2020-01-23 02:27:58 +01:00
let topLabelSpacer = UIView.hStretchingSpacer()
[ displayNameLabel, isPinnedIcon, unreadCountView, hasMentionView, topLabelSpacer, timestampLabel ].forEach{ view in
topLabelStackView.addArrangedSubview(view)
}
2019-11-28 06:42:07 +01:00
let snippetLabelContainer = UIView()
snippetLabelContainer.addSubview(snippetLabel)
snippetLabelContainer.addSubview(typingIndicatorView)
2020-01-23 02:27:58 +01:00
let bottomLabelSpacer = UIView.hStretchingSpacer()
[ snippetLabelContainer, bottomLabelSpacer, statusIndicatorView ].forEach{ view in
bottomLabelStackView.addArrangedSubview(view)
}
let labelContainerView = UIStackView(arrangedSubviews: [ topLabelStackView, bottomLabelStackView ])
labelContainerView.axis = .vertical
labelContainerView.alignment = .leading
labelContainerView.spacing = 6
labelContainerView.isUserInteractionEnabled = false
2021-03-01 04:04:54 +01:00
// Main stack view
let stackView = UIStackView(arrangedSubviews: [ accentLineView, profilePictureView, labelContainerView ])
2019-11-28 06:42:07 +01:00
stackView.axis = .horizontal
stackView.alignment = .center
stackView.spacing = Values.mediumSpacing
contentView.addSubview(stackView)
2021-03-01 04:04:54 +01:00
// Constraints
accentLineView.pin(.top, to: .top, of: contentView)
accentLineView.pin(.bottom, to: .bottom, of: contentView)
timestampLabel.setContentCompressionResistancePriority(.required, for: NSLayoutConstraint.Axis.horizontal)
// HACK: The six lines below are part of a workaround for a weird layout bug
topLabelStackView.set(.width, to: UIScreen.main.bounds.width - Values.accentLineThickness - profilePictureViewSize - 3 * Values.mediumSpacing)
2020-03-19 23:20:14 +01:00
topLabelStackView.set(.height, to: 20)
topLabelSpacer.set(.height, to: 20)
2021-03-01 04:04:54 +01:00
bottomLabelStackView.set(.width, to: UIScreen.main.bounds.width - Values.accentLineThickness - profilePictureViewSize - 3 * Values.mediumSpacing)
2020-03-19 23:20:14 +01:00
bottomLabelStackView.set(.height, to: 18)
bottomLabelSpacer.set(.height, to: 18)
2021-03-01 04:04:54 +01:00
statusIndicatorView.set(.width, to: ConversationCell.statusIndicatorSize)
statusIndicatorView.set(.height, to: ConversationCell.statusIndicatorSize)
2019-11-28 06:42:07 +01:00
snippetLabel.pin(to: snippetLabelContainer)
typingIndicatorView.pin(.leading, to: .leading, of: snippetLabelContainer)
typingIndicatorView.centerYAnchor.constraint(equalTo: snippetLabel.centerYAnchor).isActive = true
stackView.pin(.leading, to: .leading, of: contentView)
stackView.pin(.top, to: .top, of: contentView)
2021-03-01 04:04:54 +01:00
// HACK: The two lines below are part of a workaround for a weird layout bug
2020-01-23 02:27:58 +01:00
stackView.set(.width, to: UIScreen.main.bounds.width - Values.mediumSpacing)
2020-01-30 10:09:02 +01:00
stackView.set(.height, to: cellHeight)
2019-11-28 06:42:07 +01:00
}
2022-01-17 06:56:51 +01:00
// MARK: Updating for search results
private func updateForSearchResult() {
AssertIsOnMainThread()
guard let thread = threadViewModel?.threadRecord else { return }
profilePictureView.update(for: thread)
isPinnedIcon.isHidden = true
unreadCountView.isHidden = true
hasMentionView.isHidden = true
}
public func configureForRecent() {
displayNameLabel.attributedText = NSMutableAttributedString(string: getDisplayName(), attributes: [.foregroundColor:Colors.text])
bottomLabelStackView.isHidden = false
let snippet = String(format: NSLocalizedString("RECENT_SEARCH_LAST_MESSAGE_DATETIME", comment: ""), DateUtil.formatDate(forDisplay: threadViewModel.lastMessageDate))
snippetLabel.attributedText = NSMutableAttributedString(string: snippet, attributes: [.foregroundColor:Colors.text.withAlphaComponent(Values.lowOpacity)])
timestampLabel.isHidden = true
}
2022-02-23 05:49:19 +01:00
public func configure(snippet: String?, searchText: String, message: TSMessage? = nil) {
2022-01-20 05:33:31 +01:00
let normalizedSearchText = searchText.lowercased()
2022-02-23 05:49:19 +01:00
if let messageTimestamp = message?.timestamp, let snippet = snippet {
// Message
2022-02-23 05:49:19 +01:00
let messageDate = NSDate.ows_date(withMillisecondsSince1970: messageTimestamp)
displayNameLabel.attributedText = NSMutableAttributedString(string: getDisplayName(), attributes: [.foregroundColor:Colors.text])
timestampLabel.isHidden = false
2022-01-17 06:56:51 +01:00
timestampLabel.text = DateUtil.formatDate(forDisplay: messageDate)
bottomLabelStackView.isHidden = false
2022-01-27 06:39:22 +01:00
var rawSnippet = snippet
if let message = message, let name = getMessageAuthorName(message: message) {
rawSnippet = "\(name): \(snippet)"
}
snippetLabel.attributedText = getHighlightedSnippet(snippet: rawSnippet, searchText: normalizedSearchText, fontSize: Values.smallFontSize)
2022-01-17 06:56:51 +01:00
} else {
// Contact
2022-01-20 05:33:31 +01:00
if threadViewModel.isGroupThread, let thread = threadViewModel.threadRecord as? TSGroupThread {
2022-01-20 06:21:32 +01:00
displayNameLabel.attributedText = getHighlightedSnippet(snippet: getDisplayName(), searchText: normalizedSearchText, fontSize: Values.mediumFontSize)
2022-01-20 05:33:31 +01:00
bottomLabelStackView.isHidden = false
let context: Contact.Context = thread.isOpenGroup ? .openGroup : .regular
var rawSnippet: String = ""
thread.groupModel.groupMemberIds.forEach{ id in
if let displayName = Storage.shared.getContact(with: id)?.displayName(for: context) {
if !rawSnippet.isEmpty {
rawSnippet += ", \(displayName)"
}
if displayName.lowercased().contains(normalizedSearchText) {
rawSnippet = displayName
}
}
}
snippetLabel.attributedText = getHighlightedSnippet(snippet: rawSnippet, searchText: normalizedSearchText, fontSize: Values.smallFontSize)
} else {
2022-01-20 06:21:32 +01:00
displayNameLabel.attributedText = getHighlightedSnippet(snippet: getDisplayNameForSearch(threadViewModel.contactSessionID!), searchText: normalizedSearchText, fontSize: Values.mediumFontSize)
2022-01-20 05:33:31 +01:00
bottomLabelStackView.isHidden = true
}
timestampLabel.isHidden = true
2022-01-17 06:56:51 +01:00
}
}
private func getHighlightedSnippet(snippet: String, searchText: String, fontSize: CGFloat) -> NSMutableAttributedString {
2022-01-19 00:59:49 +01:00
guard snippet != NSLocalizedString("NOTE_TO_SELF", comment: "") else {
return NSMutableAttributedString(string: snippet, attributes: [.foregroundColor:Colors.text])
}
let result = NSMutableAttributedString(string: snippet, attributes: [.foregroundColor:Colors.text.withAlphaComponent(Values.lowOpacity)])
let normalizedSnippet = snippet.lowercased() as NSString
2022-01-20 05:33:31 +01:00
guard normalizedSnippet.contains(searchText) else { return result }
2022-01-20 05:33:31 +01:00
let range = normalizedSnippet.range(of: searchText)
result.addAttribute(.foregroundColor, value: Colors.text, range: range)
result.addAttribute(.font, value: UIFont.boldSystemFont(ofSize: fontSize), range: range)
return result
}
2019-11-28 06:42:07 +01:00
// MARK: Updating
private func update() {
2020-06-15 05:50:56 +02:00
AssertIsOnMainThread()
2021-08-03 01:37:48 +02:00
guard let thread = threadViewModel?.threadRecord else { return }
2021-11-30 03:52:31 +01:00
backgroundColor = threadViewModel.isPinned ? Colors.cellPinned : Colors.cellBackground
2020-07-21 05:49:41 +02:00
let isBlocked: Bool
2020-10-28 03:30:48 +01:00
if let thread = thread as? TSContactThread {
isBlocked = SSKEnvironment.shared.blockingManager.isRecipientIdBlocked(thread.contactSessionID())
2020-07-21 05:49:41 +02:00
} else {
isBlocked = false
}
if isBlocked {
2021-03-01 04:04:54 +01:00
accentLineView.backgroundColor = Colors.destructive
accentLineView.alpha = 1
2020-07-21 05:49:41 +02:00
} else {
2021-03-01 04:04:54 +01:00
accentLineView.backgroundColor = Colors.accent
accentLineView.alpha = threadViewModel.hasUnreadMessages ? 1 : 0.0001 // Setting the alpha to exactly 0 causes an issue on iOS 12
2020-07-21 05:49:41 +02:00
}
2021-11-17 05:51:53 +01:00
isPinnedIcon.isHidden = !threadViewModel.isPinned
2021-03-01 04:04:54 +01:00
unreadCountView.isHidden = !threadViewModel.hasUnreadMessages
let unreadCount = threadViewModel.unreadCount
unreadCountLabel.text = unreadCount < 10000 ? "\(unreadCount)" : "9999+"
let fontSize = (unreadCount < 10000) ? Values.verySmallFontSize : 8
2021-03-01 04:04:54 +01:00
unreadCountLabel.font = .boldSystemFont(ofSize: fontSize)
hasMentionView.isHidden = !(threadViewModel.hasUnreadMentions && thread.isGroupThread())
2020-10-28 03:30:48 +01:00
profilePictureView.update(for: thread)
2019-11-28 06:42:07 +01:00
displayNameLabel.text = getDisplayName()
timestampLabel.text = DateUtil.formatDate(forDisplay: threadViewModel.lastMessageDate)
2020-10-28 03:30:48 +01:00
if SSKEnvironment.shared.typingIndicators.typingRecipientId(forThread: thread) != nil {
2019-11-28 06:42:07 +01:00
snippetLabel.text = ""
typingIndicatorView.isHidden = false
typingIndicatorView.startAnimation()
} else {
snippetLabel.attributedText = getSnippet()
typingIndicatorView.isHidden = true
typingIndicatorView.stopAnimation()
}
statusIndicatorView.backgroundColor = nil
2019-11-29 06:30:01 +01:00
let lastMessage = threadViewModel.lastMessageForInbox
if let lastMessage = lastMessage as? TSOutgoingMessage {
let image: UIImage
let status = MessageRecipientStatusUtils.recipientStatus(outgoingMessage: lastMessage)
switch status {
2020-11-19 00:37:18 +01:00
case .uploading, .sending: image = #imageLiteral(resourceName: "CircleDotDotDot").asTintedImage(color: Colors.text)!
2020-03-17 06:18:53 +01:00
case .sent, .skipped, .delivered: image = #imageLiteral(resourceName: "CircleCheck").asTintedImage(color: Colors.text)!
case .read:
2020-03-17 06:18:53 +01:00
statusIndicatorView.backgroundColor = isLightMode ? .black : .white
image = isLightMode ? #imageLiteral(resourceName: "FilledCircleCheckLightMode") : #imageLiteral(resourceName: "FilledCircleCheckDarkMode")
2020-02-02 09:08:46 +01:00
case .failed: image = #imageLiteral(resourceName: "message_status_failed").asTintedImage(color: Colors.text)!
2019-11-29 06:30:01 +01:00
}
statusIndicatorView.image = image
statusIndicatorView.isHidden = false
} else {
statusIndicatorView.isHidden = true
}
2019-11-28 06:42:07 +01:00
}
2022-01-27 06:39:22 +01:00
private func getMessageAuthorName(message: TSMessage) -> String? {
guard threadViewModel.isGroupThread else { return nil }
if let incomingMessage = message as? TSIncomingMessage {
return Storage.shared.getContact(with: incomingMessage.authorId)?.displayName(for: .regular) ?? "Anonymous"
}
return nil
}
2022-01-20 06:21:32 +01:00
private func getDisplayNameForSearch(_ sessionID: String) -> String {
2022-01-27 05:29:10 +01:00
if threadViewModel.threadRecord.isNoteToSelf() {
return NSLocalizedString("NOTE_TO_SELF", comment: "")
} else {
var result = sessionID
if let contact = Storage.shared.getContact(with: sessionID), let name = contact.name {
result = name
if let nickname = contact.nickname { result += "(\(nickname))"}
}
return result
2022-01-20 06:21:32 +01:00
}
}
2019-11-28 06:42:07 +01:00
private func getDisplayName() -> String {
if threadViewModel.isGroupThread {
2020-01-30 10:09:02 +01:00
if threadViewModel.name.isEmpty {
2021-02-26 05:56:41 +01:00
return "Unknown Group"
2019-11-28 06:42:07 +01:00
} else {
return threadViewModel.name
}
} else {
if threadViewModel.threadRecord.isNoteToSelf() {
return NSLocalizedString("NOTE_TO_SELF", comment: "")
2019-11-28 06:42:07 +01:00
} else {
let hexEncodedPublicKey = threadViewModel.contactSessionID!
2021-02-26 05:56:41 +01:00
return Storage.shared.getContact(with: hexEncodedPublicKey)?.displayName(for: .regular) ?? hexEncodedPublicKey
2019-11-28 06:42:07 +01:00
}
}
}
private func getSnippet() -> NSMutableAttributedString {
let result = NSMutableAttributedString()
if threadViewModel.isMuted {
2019-11-29 06:30:01 +01:00
result.append(NSAttributedString(string: "\u{e067} ", attributes: [ .font : UIFont.ows_elegantIconsFont(10), .foregroundColor : Colors.unimportant ]))
2021-07-29 02:14:06 +02:00
} else if threadViewModel.isOnlyNotifyingForMentions {
let imageAttachment = NSTextAttachment()
imageAttachment.image = UIImage(named: "NotifyMentions.png")?.asTintedImage(color: Colors.unimportant)
imageAttachment.bounds = CGRect(x: 0, y: -2, width: Values.smallFontSize, height: Values.smallFontSize)
let imageString = NSAttributedString(attachment: imageAttachment)
result.append(imageString)
result.append(NSAttributedString(string: " ", attributes: [ .font : UIFont.ows_elegantIconsFont(10), .foregroundColor : Colors.unimportant ]))
2019-11-28 06:42:07 +01:00
}
2022-01-27 06:39:22 +01:00
let font = threadViewModel.hasUnreadMessages ? UIFont.boldSystemFont(ofSize: Values.smallFontSize) : UIFont.systemFont(ofSize: Values.smallFontSize)
if threadViewModel.isGroupThread, let message = threadViewModel.lastMessageForInbox as? TSMessage, let name = getMessageAuthorName(message: message) {
result.append(NSAttributedString(string: "\(name): ", attributes: [ .font : font, .foregroundColor : Colors.text ]))
}
2019-11-28 06:42:07 +01:00
if let rawSnippet = threadViewModel.lastMessageText {
let snippet = MentionUtilities.highlightMentions(in: rawSnippet, threadID: threadViewModel.threadRecord.uniqueId!)
2019-11-29 06:30:01 +01:00
result.append(NSAttributedString(string: snippet, attributes: [ .font : font, .foregroundColor : Colors.text ]))
2019-11-28 06:42:07 +01:00
}
return result
}
}