final class MediaTextOverlayView : UIView { private let viewItem: ConversationViewItem private let albumViewWidth: CGFloat private let delegate: MessageCellDelegate var readMoreButton: UIButton? // MARK: Settings private static let maxHeight: CGFloat = 88; // MARK: Lifecycle init(viewItem: ConversationViewItem, albumViewWidth: CGFloat, delegate: MessageCellDelegate) { self.viewItem = viewItem self.albumViewWidth = albumViewWidth self.delegate = delegate super.init(frame: CGRect.zero) setUpViewHierarchy() } override init(frame: CGRect) { preconditionFailure("Use init(text:) instead.") } required init?(coder: NSCoder) { preconditionFailure("Use init(text:) instead.") } private func setUpViewHierarchy() { guard let message = viewItem.interaction as? TSMessage, let body = message.body, body.count > 0 else { return } // Shadow let shadowView = GradientView(from: .clear, to: UIColor.black.withAlphaComponent(0.7)) addSubview(shadowView) shadowView.pin(to: self) // Line let lineView = UIView() lineView.backgroundColor = Colors.accent lineView.set(.width, to: Values.accentLineThickness) // Body label let bodyLabel = UILabel() bodyLabel.numberOfLines = 0 bodyLabel.lineBreakMode = .byTruncatingTail bodyLabel.text = given(body) { MentionUtilities.highlightMentions(in: $0, threadID: viewItem.interaction.uniqueThreadId) } bodyLabel.textColor = .white bodyLabel.font = .systemFont(ofSize: Values.smallFontSize) // Content stack view let contentStackView = UIStackView(arrangedSubviews: [ lineView, bodyLabel ]) contentStackView.axis = .horizontal contentStackView.spacing = Values.smallSpacing addSubview(contentStackView) let inset = Values.mediumSpacing contentStackView.pin(.left, to: .left, of: self, withInset: inset) contentStackView.pin(.top, to: .top, of: self, withInset: 3 * inset) contentStackView.pin(.right, to: .right, of: self, withInset: -inset) // Max height bodyLabel.heightAnchor.constraint(lessThanOrEqualToConstant: MediaTextOverlayView.maxHeight).isActive = true // Overflow button let bodyLabelTargetSize = bodyLabel.sizeThatFits(CGSize(width: albumViewWidth - 2 * inset, height: .greatestFiniteMagnitude)) if bodyLabelTargetSize.height > MediaTextOverlayView.maxHeight { let readMoreButton = UIButton() self.readMoreButton = readMoreButton readMoreButton.setTitle("Read More", for: UIControl.State.normal) readMoreButton.titleLabel!.font = .boldSystemFont(ofSize: Values.smallFontSize) readMoreButton.setTitleColor(.white, for: UIControl.State.normal) readMoreButton.addTarget(self, action: #selector(readMore), for: UIControl.Event.touchUpInside) addSubview(readMoreButton) readMoreButton.pin(.left, to: .left, of: self, withInset: inset) readMoreButton.pin(.top, to: .bottom, of: contentStackView, withInset: Values.smallSpacing) readMoreButton.pin(.bottom, to: .bottom, of: self, withInset: -Values.smallSpacing) } else { contentStackView.pin(.bottom, to: .bottom, of: self, withInset: -inset) } } // MARK: Interaction @objc private func readMore() { delegate.showFullText(viewItem) } }