session-ios/Session/Conversations/Message Cells/Content Views/LinkPreviewView.swift

171 lines
6.7 KiB
Swift
Raw Normal View History

2021-02-15 03:51:26 +01:00
import NVActivityIndicatorView
2021-02-19 06:02:19 +01:00
final class LinkPreviewView : UIView {
private let viewItem: ConversationViewItem?
2021-02-11 04:24:38 +01:00
private let maxWidth: CGFloat
2021-02-19 06:02:19 +01:00
private let delegate: LinkPreviewViewDelegate
var linkPreviewState: LinkPreviewState? { didSet { update() } }
2021-02-15 03:51:26 +01:00
private lazy var imageViewContainerWidthConstraint = imageView.set(.width, to: 100)
private lazy var imageViewContainerHeightConstraint = imageView.set(.height, to: 100)
2021-02-15 03:51:26 +01:00
private lazy var sentLinkPreviewTextColor: UIColor = {
let isOutgoing = (viewItem!.interaction.interactionType() == .outgoingMessage)
switch (isOutgoing, AppModeManager.shared.currentAppMode) {
case (true, .dark), (false, .light): return .black
2022-03-17 00:43:11 +01:00
case (true, .light): return Colors.grey
default: return .white
}
2021-02-15 03:51:26 +01:00
}()
// MARK: UI Components
private lazy var imageView: UIImageView = {
let result = UIImageView()
result.contentMode = .scaleAspectFill
return result
}()
2021-02-15 03:51:26 +01:00
private lazy var imageViewContainer: UIView = {
let result = UIView()
result.clipsToBounds = true
return result
}()
private lazy var loader: NVActivityIndicatorView = {
let color: UIColor = isLightMode ? .black : .white
return NVActivityIndicatorView(frame: CGRect.zero, type: .circleStrokeSpin, color: color, padding: nil)
}()
private lazy var titleLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.smallFontSize)
result.numberOfLines = 0
return result
}()
private lazy var bodyTextViewContainer = UIView()
2021-02-15 03:51:26 +01:00
private lazy var hStackViewContainer = UIView()
2021-02-15 04:45:46 +01:00
private lazy var hStackView = UIStackView()
private lazy var cancelButton: UIButton = {
let result = UIButton(type: .custom)
let tint: UIColor = isLightMode ? .black : .white
result.setImage(UIImage(named: "X")?.withTint(tint), for: UIControl.State.normal)
2021-02-19 06:02:19 +01:00
let cancelButtonSize = LinkPreviewView.cancelButtonSize
2021-02-15 04:45:46 +01:00
result.set(.width, to: cancelButtonSize)
result.set(.height, to: cancelButtonSize)
result.addTarget(self, action: #selector(cancel), for: UIControl.Event.touchUpInside)
return result
}()
var bodyTextView: UITextView?
2021-02-15 04:45:46 +01:00
// MARK: Settings
2021-02-15 03:51:26 +01:00
private static let loaderSize: CGFloat = 24
2021-02-15 04:45:46 +01:00
private static let cancelButtonSize: CGFloat = 45
// MARK: Lifecycle
2021-02-19 06:02:19 +01:00
init(for viewItem: ConversationViewItem?, maxWidth: CGFloat, delegate: LinkPreviewViewDelegate) {
self.viewItem = viewItem
2021-02-11 04:24:38 +01:00
self.maxWidth = maxWidth
self.delegate = delegate
super.init(frame: CGRect.zero)
setUpViewHierarchy()
}
override init(frame: CGRect) {
2021-02-11 04:24:38 +01:00
preconditionFailure("Use init(for:maxWidth:delegate:) instead.")
}
required init?(coder: NSCoder) {
2021-02-11 04:24:38 +01:00
preconditionFailure("Use init(for:maxWidth:delegate:) instead.")
}
private func setUpViewHierarchy() {
// Image view
2021-02-15 03:51:26 +01:00
imageViewContainerWidthConstraint.isActive = true
imageViewContainerHeightConstraint.isActive = true
imageViewContainer.addSubview(imageView)
imageView.pin(to: imageViewContainer)
// Title label
let titleLabelContainer = UIView()
titleLabelContainer.addSubview(titleLabel)
titleLabel.pin(to: titleLabelContainer, withInset: Values.smallSpacing)
// Horizontal stack view
2021-02-15 04:45:46 +01:00
hStackView.addArrangedSubview(imageViewContainer)
hStackView.addArrangedSubview(titleLabelContainer)
hStackView.axis = .horizontal
hStackView.alignment = .center
hStackViewContainer.addSubview(hStackView)
hStackView.pin(to: hStackViewContainer)
// Vertical stack view
let vStackView = UIStackView(arrangedSubviews: [ hStackViewContainer, bodyTextViewContainer ])
vStackView.axis = .vertical
addSubview(vStackView)
vStackView.pin(to: self)
2021-02-15 03:51:26 +01:00
// Loader
addSubview(loader)
2021-02-19 06:02:19 +01:00
let loaderSize = LinkPreviewView.loaderSize
2021-02-15 03:51:26 +01:00
loader.set(.width, to: loaderSize)
loader.set(.height, to: loaderSize)
loader.center(in: self)
}
// MARK: Updating
private func update() {
2021-02-15 04:45:46 +01:00
cancelButton.removeFromSuperview()
guard let linkPreviewState = linkPreviewState else { return }
2021-02-15 06:16:22 +01:00
var image = linkPreviewState.image()
if image == nil && (linkPreviewState is LinkPreviewDraft || linkPreviewState is LinkPreviewSent) {
image = UIImage(named: "Link")?.withTint(isLightMode ? .black : .white)
}
// Image view
2021-02-15 03:51:26 +01:00
let imageViewContainerSize: CGFloat = (linkPreviewState is LinkPreviewSent) ? 100 : 80
imageViewContainerWidthConstraint.constant = imageViewContainerSize
imageViewContainerHeightConstraint.constant = imageViewContainerSize
imageViewContainer.layer.cornerRadius = (linkPreviewState is LinkPreviewSent) ? 0 : 8
if linkPreviewState is LinkPreviewLoading {
imageViewContainer.backgroundColor = .clear
} else {
imageViewContainer.backgroundColor = isDarkMode ? .black : UIColor.black.withAlphaComponent(0.06)
}
2021-02-15 06:16:22 +01:00
imageView.image = image
imageView.contentMode = (linkPreviewState.image() == nil) ? .center : .scaleAspectFill
2021-02-15 03:51:26 +01:00
// Loader
2021-02-15 06:16:22 +01:00
loader.alpha = (image != nil) ? 0 : 1
if image != nil { loader.stopAnimating() } else { loader.startAnimating() }
// Title
2022-03-17 00:43:11 +01:00
titleLabel.textColor = sentLinkPreviewTextColor
titleLabel.text = linkPreviewState.title()
2021-02-15 03:51:26 +01:00
// Horizontal stack view
switch linkPreviewState {
case is LinkPreviewSent: hStackViewContainer.backgroundColor = isDarkMode ? .black : UIColor.black.withAlphaComponent(0.06)
default: hStackViewContainer.backgroundColor = nil
}
// Body text view
bodyTextViewContainer.subviews.forEach { $0.removeFromSuperview() }
if let viewItem = viewItem {
2021-02-19 03:25:31 +01:00
let bodyTextView = VisibleMessageCell.getBodyTextView(for: viewItem, with: maxWidth, textColor: sentLinkPreviewTextColor, searchText: delegate.lastSearchedText, delegate: delegate)
self.bodyTextView = bodyTextView
bodyTextViewContainer.addSubview(bodyTextView)
bodyTextView.pin(to: bodyTextViewContainer, withInset: 12)
}
2021-02-15 04:45:46 +01:00
if linkPreviewState is LinkPreviewDraft {
hStackView.addArrangedSubview(cancelButton)
}
}
// MARK: Interaction
@objc private func cancel() {
delegate.handleLinkPreviewCanceled()
}
}
2021-02-15 04:45:46 +01:00
// MARK: Delegate
2021-02-19 06:02:19 +01:00
protocol LinkPreviewViewDelegate : UITextViewDelegate & BodyTextViewDelegate {
2021-02-19 03:25:31 +01:00
var lastSearchedText: String? { get }
2021-02-15 04:45:46 +01:00
func handleLinkPreviewCanceled()
}