mirror of
https://github.com/mastodon/mastodon-ios.git
synced 2025-01-19 04:01:51 +01:00
300 lines
14 KiB
Swift
300 lines
14 KiB
Swift
//
|
|
// ProfileCardView.swift
|
|
//
|
|
//
|
|
// Created by MainasuK on 2022-4-14.
|
|
//
|
|
|
|
import os.log
|
|
import UIKit
|
|
import Combine
|
|
import MetaTextKit
|
|
import MastodonAsset
|
|
|
|
public protocol ProfileCardViewDelegate: AnyObject {
|
|
func profileCardView(_ profileCardView: ProfileCardView, relationshipButtonDidPressed button: ProfileRelationshipActionButton)
|
|
func profileCardView(_ profileCardView: ProfileCardView, familiarFollowersDashboardViewDidPressed view: FamiliarFollowersDashboardView)
|
|
}
|
|
|
|
public final class ProfileCardView: UIView {
|
|
|
|
let logger = Logger(subsystem: "ProfileCardView", category: "View")
|
|
|
|
static let avatarSize = CGSize(width: 56, height: 56)
|
|
static let friendshipActionButtonSize = CGSize(width: 108, height: 34)
|
|
static let contentMargin: CGFloat = 16
|
|
|
|
weak var delegate: ProfileCardViewDelegate?
|
|
private var _disposeBag = Set<AnyCancellable>()
|
|
var disposeBag = Set<AnyCancellable>()
|
|
|
|
let container = UIStackView()
|
|
|
|
let bannerImageView: UIImageView = {
|
|
let imageView = UIImageView()
|
|
imageView.contentMode = .scaleAspectFill
|
|
imageView.layer.masksToBounds = true
|
|
imageView.layer.cornerRadius = 3
|
|
imageView.layer.cornerCurve = .continuous
|
|
return imageView
|
|
}()
|
|
|
|
// avatar
|
|
public let avatarButtonBackgroundView = UIView()
|
|
public let avatarButton = AvatarButton()
|
|
|
|
// author name
|
|
public let authorNameLabel = MetaLabel(style: .profileCardName)
|
|
|
|
// author username
|
|
public let authorUsernameLabel = MetaLabel(style: .profileCardUsername)
|
|
|
|
// bio
|
|
let bioMetaTextAdaptiveMarginContainerView = AdaptiveMarginContainerView()
|
|
let bioMetaText: MetaText = {
|
|
let metaText = MetaText()
|
|
metaText.textView.backgroundColor = .clear
|
|
metaText.textView.isEditable = false
|
|
metaText.textView.isSelectable = true
|
|
metaText.textView.isScrollEnabled = false
|
|
//metaText.textView.textContainer.lineFragmentPadding = 0
|
|
//metaText.textView.textContainerInset = .zero
|
|
metaText.textView.layer.masksToBounds = false
|
|
metaText.textView.textDragInteraction?.isEnabled = false // disable drag for link and attachment
|
|
|
|
metaText.textView.layer.masksToBounds = true
|
|
metaText.textView.layer.cornerCurve = .continuous
|
|
metaText.textView.layer.cornerRadius = 10
|
|
|
|
metaText.paragraphStyle = {
|
|
let style = NSMutableParagraphStyle()
|
|
style.lineSpacing = 5
|
|
style.paragraphSpacing = 8
|
|
return style
|
|
}()
|
|
metaText.textAttributes = [
|
|
.font: UIFont.preferredFont(forTextStyle: .body),
|
|
.foregroundColor: Asset.Colors.Label.primary.color,
|
|
]
|
|
metaText.linkAttributes = [
|
|
.font: UIFont.preferredFont(forTextStyle: .body),
|
|
.foregroundColor: Asset.Colors.brand.color,
|
|
]
|
|
return metaText
|
|
}()
|
|
|
|
let infoContainerAdaptiveMarginContainerView = AdaptiveMarginContainerView()
|
|
let infoContainer = UIStackView()
|
|
|
|
let statusDashboardView = ProfileStatusDashboardView()
|
|
|
|
let relationshipActionButtonShadowContainer = ShadowBackgroundContainer()
|
|
let relationshipActionButton: ProfileRelationshipActionButton = {
|
|
let button = ProfileRelationshipActionButton()
|
|
button.titleLabel?.font = .systemFont(ofSize: 17, weight: .semibold)
|
|
button.titleLabel?.adjustsFontSizeToFitWidth = true
|
|
button.titleLabel?.minimumScaleFactor = 0.5
|
|
return button
|
|
}()
|
|
|
|
let familiarFollowersDashboardViewAdaptiveMarginContainerView = AdaptiveMarginContainerView()
|
|
let familiarFollowersDashboardView = FamiliarFollowersDashboardView()
|
|
|
|
public private(set) lazy var viewModel: ViewModel = {
|
|
let viewModel = ViewModel()
|
|
viewModel.bind(view: self)
|
|
return viewModel
|
|
}()
|
|
|
|
public func prepareForReuse() {
|
|
disposeBag.removeAll()
|
|
bannerImageView.af.cancelImageRequest()
|
|
bannerImageView.image = nil
|
|
}
|
|
|
|
override init(frame: CGRect) {
|
|
super.init(frame: frame)
|
|
_init()
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
super.init(coder: coder)
|
|
_init()
|
|
}
|
|
|
|
}
|
|
|
|
extension ProfileCardView {
|
|
private func _init() {
|
|
avatarButton.isUserInteractionEnabled = false
|
|
authorNameLabel.isUserInteractionEnabled = false
|
|
authorUsernameLabel.isUserInteractionEnabled = false
|
|
bioMetaText.textView.isUserInteractionEnabled = false
|
|
statusDashboardView.isUserInteractionEnabled = false
|
|
|
|
// container: V - [ bannerContainer | authorContainer | bioMetaText | infoContainer | familiarFollowersDashboardView ]
|
|
container.axis = .vertical
|
|
container.spacing = 8
|
|
container.translatesAutoresizingMaskIntoConstraints = false
|
|
addSubview(container)
|
|
container.pinToParent()
|
|
|
|
// bannerContainer
|
|
let bannerContainer = UIView()
|
|
bannerContainer.translatesAutoresizingMaskIntoConstraints = false
|
|
container.addArrangedSubview(bannerContainer)
|
|
container.setCustomSpacing(6, after: bannerContainer)
|
|
|
|
// bannerImageView
|
|
bannerImageView.translatesAutoresizingMaskIntoConstraints = false
|
|
bannerContainer.addSubview(bannerImageView)
|
|
NSLayoutConstraint.activate([
|
|
bannerImageView.topAnchor.constraint(equalTo: bannerContainer.topAnchor, constant: 4),
|
|
bannerImageView.leadingAnchor.constraint(equalTo: bannerContainer.leadingAnchor, constant: 4),
|
|
bannerContainer.trailingAnchor.constraint(equalTo: bannerImageView.trailingAnchor, constant: 4),
|
|
bannerImageView.bottomAnchor.constraint(equalTo: bannerContainer.bottomAnchor),
|
|
bannerImageView.widthAnchor.constraint(equalTo: bannerImageView.heightAnchor, multiplier: 335.0/128.0).priority(.required - 1),
|
|
])
|
|
|
|
// authorContainer: H - [ avatarPlaceholder | authorInfoContainer ]
|
|
let authorContainer = UIStackView()
|
|
authorContainer.axis = .horizontal
|
|
authorContainer.spacing = 16
|
|
let authorContainerAdaptiveMarginContainerView = AdaptiveMarginContainerView()
|
|
authorContainerAdaptiveMarginContainerView.contentView = authorContainer
|
|
authorContainerAdaptiveMarginContainerView.margin = ProfileCardView.contentMargin
|
|
container.addArrangedSubview(authorContainerAdaptiveMarginContainerView)
|
|
container.setCustomSpacing(6, after: bannerContainer)
|
|
|
|
// avatarPlaceholder
|
|
let avatarPlaceholder = UIView()
|
|
avatarPlaceholder.translatesAutoresizingMaskIntoConstraints = false
|
|
authorContainer.addArrangedSubview(avatarPlaceholder)
|
|
NSLayoutConstraint.activate([
|
|
avatarPlaceholder.widthAnchor.constraint(equalToConstant: ProfileCardView.avatarSize.width).priority(.required - 1),
|
|
avatarPlaceholder.heightAnchor.constraint(equalToConstant: ProfileCardView.avatarSize.height - 14).priority(.required - 1),
|
|
])
|
|
|
|
avatarButton.translatesAutoresizingMaskIntoConstraints = false
|
|
authorContainer.addSubview(avatarButton)
|
|
NSLayoutConstraint.activate([
|
|
avatarButton.leadingAnchor.constraint(equalTo: avatarPlaceholder.leadingAnchor),
|
|
avatarButton.trailingAnchor.constraint(equalTo: avatarPlaceholder.trailingAnchor),
|
|
avatarButton.bottomAnchor.constraint(equalTo: avatarPlaceholder.bottomAnchor),
|
|
avatarButton.heightAnchor.constraint(equalToConstant: ProfileCardView.avatarSize.height).priority(.required - 1),
|
|
])
|
|
|
|
avatarButtonBackgroundView.layer.masksToBounds = true
|
|
avatarButtonBackgroundView.layer.cornerCurve = .continuous
|
|
avatarButtonBackgroundView.layer.cornerRadius = 12 + 1
|
|
avatarButtonBackgroundView.translatesAutoresizingMaskIntoConstraints = false
|
|
authorContainer.insertSubview(avatarButtonBackgroundView, belowSubview: avatarButton)
|
|
NSLayoutConstraint.activate([
|
|
avatarButtonBackgroundView.centerXAnchor.constraint(equalTo: avatarButton.centerXAnchor),
|
|
avatarButtonBackgroundView.centerYAnchor.constraint(equalTo: avatarButton.centerYAnchor),
|
|
avatarButtonBackgroundView.widthAnchor.constraint(equalToConstant: ProfileCardView.avatarSize.width + 4).priority(.required - 1),
|
|
avatarButtonBackgroundView.heightAnchor.constraint(equalToConstant: ProfileCardView.avatarSize.height + 4).priority(.required - 1),
|
|
])
|
|
|
|
// authorInfoContainer: V - [ authorNameLabel | authorUsernameLabel ]
|
|
let authorInfoContainer = UIStackView()
|
|
authorInfoContainer.axis = .vertical
|
|
// authorInfoContainer.spacing = 2
|
|
authorContainer.addArrangedSubview(authorInfoContainer)
|
|
|
|
authorInfoContainer.addArrangedSubview(authorNameLabel)
|
|
authorInfoContainer.addArrangedSubview(authorUsernameLabel)
|
|
|
|
// bioMetaText
|
|
bioMetaTextAdaptiveMarginContainerView.contentView = bioMetaText.textView
|
|
bioMetaTextAdaptiveMarginContainerView.margin = ProfileCardView.contentMargin
|
|
bioMetaText.textView.setContentHuggingPriority(.required - 1, for: .vertical)
|
|
bioMetaText.textView.setContentCompressionResistancePriority(.required - 1, for: .vertical)
|
|
container.addArrangedSubview(bioMetaTextAdaptiveMarginContainerView)
|
|
container.setCustomSpacing(16, after: bioMetaTextAdaptiveMarginContainerView)
|
|
|
|
// infoContainer: H - [ statusDashboardView | (spacer) | relationshipActionButton]
|
|
infoContainer.axis = .horizontal
|
|
infoContainer.spacing = 8
|
|
infoContainerAdaptiveMarginContainerView.contentView = infoContainer
|
|
infoContainerAdaptiveMarginContainerView.margin = ProfileCardView.contentMargin
|
|
container.addArrangedSubview(infoContainerAdaptiveMarginContainerView)
|
|
container.setCustomSpacing(16, after: infoContainerAdaptiveMarginContainerView)
|
|
|
|
infoContainer.addArrangedSubview(statusDashboardView)
|
|
let infoContainerSpacer = UIView()
|
|
infoContainer.addArrangedSubview(UIView())
|
|
infoContainerSpacer.setContentHuggingPriority(.defaultLow - 100, for: .vertical)
|
|
infoContainerSpacer.setContentHuggingPriority(.defaultLow - 100, for: .horizontal)
|
|
let relationshipActionButtonShadowContainer = ShadowBackgroundContainer()
|
|
relationshipActionButtonShadowContainer.translatesAutoresizingMaskIntoConstraints = false
|
|
infoContainer.addArrangedSubview(relationshipActionButtonShadowContainer)
|
|
|
|
relationshipActionButton.translatesAutoresizingMaskIntoConstraints = false
|
|
relationshipActionButtonShadowContainer.addSubview(relationshipActionButton)
|
|
relationshipActionButton.pinToParent()
|
|
NSLayoutConstraint.activate([
|
|
relationshipActionButtonShadowContainer.widthAnchor.constraint(greaterThanOrEqualToConstant: ProfileCardView.friendshipActionButtonSize.width).priority(.required - 1),
|
|
relationshipActionButtonShadowContainer.heightAnchor.constraint(equalToConstant: ProfileCardView.friendshipActionButtonSize.height).priority(.required - 1),
|
|
])
|
|
|
|
familiarFollowersDashboardViewAdaptiveMarginContainerView.contentView = familiarFollowersDashboardView
|
|
familiarFollowersDashboardViewAdaptiveMarginContainerView.margin = ProfileCardView.contentMargin
|
|
container.addArrangedSubview(familiarFollowersDashboardViewAdaptiveMarginContainerView)
|
|
|
|
let bottomPadding = UIView()
|
|
bottomPadding.translatesAutoresizingMaskIntoConstraints = false
|
|
container.addArrangedSubview(bottomPadding)
|
|
NSLayoutConstraint.activate([
|
|
bottomPadding.heightAnchor.constraint(equalToConstant: 8).priority(.required - 10),
|
|
])
|
|
|
|
relationshipActionButton.addTarget(self, action: #selector(ProfileCardView.relationshipActionButtonDidPressed(_:)), for: .touchUpInside)
|
|
|
|
let familiarFollowersDashboardViewTapGestureRecognizer = UITapGestureRecognizer.singleTapGestureRecognizer
|
|
familiarFollowersDashboardViewTapGestureRecognizer.addTarget(self, action: #selector(ProfileCardView.familiarFollowersDashboardViewDidPressed(_:)))
|
|
familiarFollowersDashboardView.addGestureRecognizer(familiarFollowersDashboardViewTapGestureRecognizer)
|
|
}
|
|
|
|
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
viewModel.userInterfaceStyle = traitCollection.userInterfaceStyle
|
|
}
|
|
|
|
public override func layoutSubviews() {
|
|
updateInfoContainerLayout()
|
|
super.layoutSubviews()
|
|
}
|
|
|
|
}
|
|
|
|
extension ProfileCardView {
|
|
public func setupLayoutFrame(_ rect: CGRect) {
|
|
frame.size.width = rect.width
|
|
bioMetaTextAdaptiveMarginContainerView.frame.size.width = frame.width
|
|
bioMetaTextAdaptiveMarginContainerView.contentView?.frame.size.width = frame.width - 2 * bioMetaTextAdaptiveMarginContainerView.margin
|
|
infoContainerAdaptiveMarginContainerView.frame.size.width = frame.width
|
|
infoContainerAdaptiveMarginContainerView.contentView?.frame.size.width = frame.width - 2 * infoContainerAdaptiveMarginContainerView.margin
|
|
}
|
|
|
|
private func updateInfoContainerLayout() {
|
|
let isCompactAdaptive = bounds.width < 350
|
|
infoContainer.axis = isCompactAdaptive ? .vertical : .horizontal
|
|
}
|
|
}
|
|
|
|
extension ProfileCardView {
|
|
@objc private func relationshipActionButtonDidPressed(_ sender: UIButton) {
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public)")
|
|
assert(sender === relationshipActionButton)
|
|
delegate?.profileCardView(self, relationshipButtonDidPressed: relationshipActionButton)
|
|
}
|
|
|
|
@objc private func familiarFollowersDashboardViewDidPressed(_ sender: UITapGestureRecognizer) {
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public)")
|
|
assert(sender.view === familiarFollowersDashboardView)
|
|
delegate?.profileCardView(self, familiarFollowersDashboardViewDidPressed: familiarFollowersDashboardView)
|
|
}
|
|
}
|