2022-01-27 21:23:39 +08:00
|
|
|
//
|
|
|
|
// ActionToolBarContainer.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by sxiaojian on 2021/2/1.
|
|
|
|
//
|
|
|
|
|
|
|
|
import os.log
|
|
|
|
import UIKit
|
|
|
|
import MastodonAsset
|
|
|
|
import MastodonLocalization
|
2022-12-28 11:52:44 +01:00
|
|
|
import MastodonExtension
|
2022-01-27 21:23:39 +08:00
|
|
|
|
|
|
|
public protocol ActionToolbarContainerDelegate: AnyObject {
|
|
|
|
func actionToolbarContainer(_ actionToolbarContainer: ActionToolbarContainer, buttonDidPressed button: UIButton, action: ActionToolbarContainer.Action)
|
2022-10-31 12:40:18 -04:00
|
|
|
func actionToolbarContainer(_ actionToolbarContainer: ActionToolbarContainer, showReblogs action: UIAccessibilityCustomAction)
|
|
|
|
func actionToolbarContainer(_ actionToolbarContainer: ActionToolbarContainer, showFavorites action: UIAccessibilityCustomAction)
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public final class ActionToolbarContainer: UIView {
|
|
|
|
|
|
|
|
let logger = Logger(subsystem: "ActionToolbarContainer", category: "Control")
|
|
|
|
|
2022-02-11 15:27:29 +08:00
|
|
|
static let replyImage = Asset.Communication.bubbleLeftAndBubbleRight.image.withRenderingMode(.alwaysTemplate)
|
|
|
|
static let reblogImage = Asset.Arrow.repeat.image.withRenderingMode(.alwaysTemplate)
|
|
|
|
static let starImage = Asset.ObjectsAndTools.star.image.withRenderingMode(.alwaysTemplate)
|
|
|
|
static let starFillImage = Asset.ObjectsAndTools.starFill.image.withRenderingMode(.alwaysTemplate)
|
2022-09-15 21:28:40 +08:00
|
|
|
static let shareImage = Asset.Arrow.squareAndArrowUp.image.withRenderingMode(.alwaysTemplate)
|
2022-01-27 21:23:39 +08:00
|
|
|
|
|
|
|
public let replyButton = HighlightDimmableButton()
|
|
|
|
public let reblogButton = HighlightDimmableButton()
|
|
|
|
public let favoriteButton = HighlightDimmableButton()
|
|
|
|
public let shareButton = HighlightDimmableButton()
|
|
|
|
|
|
|
|
public weak var delegate: ActionToolbarContainerDelegate?
|
|
|
|
|
|
|
|
private let container = UIStackView()
|
2023-03-20 03:39:11 -04:00
|
|
|
private let firstContainer = UIStackView()
|
|
|
|
private let secondContainer = UIStackView()
|
|
|
|
|
|
|
|
private var isAccessibilityCategory: Bool?
|
|
|
|
private var shareButtonWidthConstraint: NSLayoutConstraint?
|
|
|
|
|
2022-01-27 21:23:39 +08:00
|
|
|
public override init(frame: CGRect) {
|
|
|
|
super.init(frame: frame)
|
|
|
|
_init()
|
|
|
|
}
|
|
|
|
|
|
|
|
public required init?(coder: NSCoder) {
|
|
|
|
super.init(coder: coder)
|
|
|
|
_init()
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
|
2022-02-18 17:49:20 +08:00
|
|
|
private func _init() {
|
2022-01-27 21:23:39 +08:00
|
|
|
container.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
addSubview(container)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
container.topAnchor.constraint(equalTo: topAnchor),
|
|
|
|
container.leadingAnchor.constraint(equalTo: leadingAnchor),
|
|
|
|
trailingAnchor.constraint(equalTo: container.trailingAnchor),
|
|
|
|
bottomAnchor.constraint(equalTo: container.bottomAnchor),
|
|
|
|
])
|
2023-03-20 03:39:11 -04:00
|
|
|
|
2022-01-27 21:23:39 +08:00
|
|
|
replyButton.addTarget(self, action: #selector(ActionToolbarContainer.buttonDidPressed(_:)), for: .touchUpInside)
|
|
|
|
reblogButton.addTarget(self, action: #selector(ActionToolbarContainer.buttonDidPressed(_:)), for: .touchUpInside)
|
|
|
|
favoriteButton.addTarget(self, action: #selector(ActionToolbarContainer.buttonDidPressed(_:)), for: .touchUpInside)
|
|
|
|
shareButton.addTarget(self, action: #selector(ActionToolbarContainer.buttonDidPressed(_:)), for: .touchUpInside)
|
2023-03-20 03:39:11 -04:00
|
|
|
|
2022-01-27 21:23:39 +08:00
|
|
|
let buttons = [replyButton, reblogButton, favoriteButton, shareButton]
|
|
|
|
buttons.forEach { button in
|
|
|
|
button.tintColor = Asset.Colors.Button.actionToolbar.color
|
2023-03-20 03:39:11 -04:00
|
|
|
button.titleLabel?.font = UIFontMetrics(forTextStyle: .caption1)
|
|
|
|
.scaledFont(for: .monospacedDigitSystemFont(ofSize: 12, weight: .regular))
|
|
|
|
button.titleLabel?.adjustsFontForContentSizeCategory = true
|
2022-01-27 21:23:39 +08:00
|
|
|
button.setTitle("", for: .normal)
|
|
|
|
button.setTitleColor(.secondaryLabel, for: .normal)
|
|
|
|
button.expandEdgeInsets = UIEdgeInsets(top: -10, left: -10, bottom: -10, right: -10)
|
2023-03-20 03:39:11 -04:00
|
|
|
button.setInsets(forContentPadding: .zero, imageTitlePadding: 4)
|
|
|
|
button.adjustsImageSizeForAccessibilityContentSizeCategory = true
|
|
|
|
button.setContentCompressionResistancePriority(.defaultHigh + 100, for: .horizontal)
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
// add more expand for menu button
|
|
|
|
shareButton.expandEdgeInsets = UIEdgeInsets(top: -10, left: -20, bottom: -10, right: -20)
|
|
|
|
|
|
|
|
replyButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.reply
|
|
|
|
reblogButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.reblog // needs update to follow state
|
|
|
|
favoriteButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.favorite // needs update to follow state
|
2022-02-14 19:34:22 +08:00
|
|
|
shareButton.accessibilityLabel = L10n.Common.Controls.Actions.share
|
2022-01-27 21:23:39 +08:00
|
|
|
|
2023-03-20 03:39:11 -04:00
|
|
|
buttons.forEach { button in
|
|
|
|
button.contentHorizontalAlignment = .leading
|
|
|
|
}
|
|
|
|
replyButton.setImage(ActionToolbarContainer.replyImage, for: .normal)
|
|
|
|
reblogButton.setImage(ActionToolbarContainer.reblogImage, for: .normal)
|
|
|
|
favoriteButton.setImage(ActionToolbarContainer.starImage, for: .normal)
|
|
|
|
shareButton.setImage(ActionToolbarContainer.shareImage, for: .normal)
|
|
|
|
|
|
|
|
container.axis = .horizontal
|
|
|
|
container.distribution = .equalSpacing
|
|
|
|
|
|
|
|
replyButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
reblogButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
favoriteButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
shareButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
|
|
|
|
firstContainer.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
firstContainer.axis = .horizontal
|
|
|
|
firstContainer.distribution = .equalSpacing
|
|
|
|
|
|
|
|
secondContainer.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
secondContainer.axis = .horizontal
|
|
|
|
secondContainer.distribution = .equalSpacing
|
|
|
|
|
|
|
|
shareButton.setContentHuggingPriority(.defaultHigh, for: .horizontal)
|
|
|
|
shareButton.setContentCompressionResistancePriority(.defaultHigh, for: .horizontal)
|
|
|
|
|
|
|
|
shareButtonWidthConstraint = replyButton.widthAnchor.constraint(equalTo: shareButton.widthAnchor)
|
|
|
|
|
|
|
|
traitCollectionDidChange(nil)
|
|
|
|
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
replyButton.heightAnchor.constraint(equalToConstant: 36).priority(.defaultHigh),
|
|
|
|
replyButton.heightAnchor.constraint(equalTo: reblogButton.heightAnchor).priority(.defaultHigh),
|
|
|
|
replyButton.heightAnchor.constraint(equalTo: favoriteButton.heightAnchor).priority(.defaultHigh),
|
|
|
|
replyButton.heightAnchor.constraint(equalTo: shareButton.heightAnchor).priority(.defaultHigh),
|
|
|
|
replyButton.widthAnchor.constraint(equalTo: reblogButton.widthAnchor).priority(.defaultHigh),
|
|
|
|
replyButton.widthAnchor.constraint(equalTo: favoriteButton.widthAnchor).priority(.defaultHigh),
|
|
|
|
])
|
|
|
|
}
|
|
|
|
|
|
|
|
public override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
|
|
|
|
let isAccessibilityCategory = traitCollection.preferredContentSizeCategory.isAccessibilityCategory
|
|
|
|
guard isAccessibilityCategory != self.isAccessibilityCategory else { return }
|
|
|
|
self.isAccessibilityCategory = isAccessibilityCategory
|
|
|
|
|
|
|
|
if isAccessibilityCategory {
|
|
|
|
container.axis = .vertical
|
|
|
|
container.spacing = 12
|
|
|
|
|
|
|
|
firstContainer.addArrangedSubview(replyButton)
|
|
|
|
firstContainer.addArrangedSubview(reblogButton)
|
|
|
|
container.addArrangedSubview(firstContainer)
|
|
|
|
|
|
|
|
secondContainer.addArrangedSubview(favoriteButton)
|
|
|
|
secondContainer.addArrangedSubview(shareButton)
|
|
|
|
container.addArrangedSubview(secondContainer)
|
|
|
|
} else {
|
2022-01-27 21:23:39 +08:00
|
|
|
container.axis = .horizontal
|
2023-03-20 03:39:11 -04:00
|
|
|
container.spacing = 0
|
|
|
|
|
2022-01-27 21:23:39 +08:00
|
|
|
container.addArrangedSubview(replyButton)
|
|
|
|
container.addArrangedSubview(reblogButton)
|
|
|
|
container.addArrangedSubview(favoriteButton)
|
|
|
|
container.addArrangedSubview(shareButton)
|
2023-03-20 03:39:11 -04:00
|
|
|
|
|
|
|
firstContainer.removeFromSuperview()
|
|
|
|
secondContainer.removeFromSuperview()
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
2023-03-20 03:39:11 -04:00
|
|
|
shareButtonWidthConstraint!.isActive = isAccessibilityCategory
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
|
|
|
|
public enum Action: String, CaseIterable {
|
|
|
|
case reply
|
|
|
|
case reblog
|
|
|
|
case like
|
|
|
|
case share
|
|
|
|
}
|
|
|
|
|
|
|
|
private func isReblogButtonHighlightStateDidChange(to isHighlight: Bool) {
|
|
|
|
let tintColor = isHighlight ? Asset.Colors.successGreen.color : Asset.Colors.Button.actionToolbar.color
|
|
|
|
reblogButton.tintColor = tintColor
|
|
|
|
reblogButton.setTitleColor(tintColor, for: .normal)
|
|
|
|
reblogButton.setTitleColor(tintColor, for: .highlighted)
|
|
|
|
}
|
|
|
|
|
|
|
|
private func isFavoriteButtonHighlightStateDidChange(to isHighlight: Bool) {
|
|
|
|
let tintColor = isHighlight ? Asset.Colors.systemOrange.color : Asset.Colors.Button.actionToolbar.color
|
|
|
|
favoriteButton.tintColor = tintColor
|
|
|
|
favoriteButton.setTitleColor(tintColor, for: .normal)
|
|
|
|
favoriteButton.setTitleColor(tintColor, for: .highlighted)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
|
|
|
|
@objc private func buttonDidPressed(_ sender: UIButton) {
|
|
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public)")
|
|
|
|
|
|
|
|
let _action: Action?
|
|
|
|
switch sender {
|
|
|
|
case replyButton: _action = .reply
|
|
|
|
case reblogButton: _action = .reblog
|
|
|
|
case favoriteButton: _action = .like
|
|
|
|
case shareButton: _action = .share
|
|
|
|
default: _action = nil
|
|
|
|
}
|
|
|
|
|
|
|
|
guard let action = _action else {
|
|
|
|
assertionFailure()
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): \(action.rawValue) button pressed")
|
|
|
|
delegate?.actionToolbarContainer(self, buttonDidPressed: sender, action: action)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
|
|
|
|
public func configureReply(count: Int, isEnabled: Bool) {
|
|
|
|
let title = ActionToolbarContainer.title(from: count)
|
|
|
|
replyButton.setTitle(title, for: .normal)
|
2022-10-31 15:07:27 -04:00
|
|
|
replyButton.accessibilityLabel = L10n.Common.Controls.Actions.reply
|
|
|
|
replyButton.accessibilityValue = L10n.Plural.Count.reply(count)
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public func configureReblog(count: Int, isEnabled: Bool, isHighlighted: Bool) {
|
|
|
|
let title = ActionToolbarContainer.title(from: count)
|
|
|
|
reblogButton.setTitle(title, for: .normal)
|
2023-03-15 14:58:25 +01:00
|
|
|
reblogButton.accessibilityValue = L10n.Plural.Count.reblogA11y(count)
|
2022-01-27 21:23:39 +08:00
|
|
|
reblogButton.isEnabled = isEnabled
|
|
|
|
reblogButton.setImage(ActionToolbarContainer.reblogImage, for: .normal)
|
|
|
|
let tintColor = isHighlighted ? Asset.Colors.successGreen.color : Asset.Colors.Button.actionToolbar.color
|
|
|
|
reblogButton.tintColor = tintColor
|
|
|
|
reblogButton.setTitleColor(tintColor, for: .normal)
|
|
|
|
reblogButton.setTitleColor(tintColor, for: .highlighted)
|
2022-02-14 19:34:22 +08:00
|
|
|
|
|
|
|
if isHighlighted {
|
|
|
|
reblogButton.accessibilityTraits.insert(.selected)
|
2023-03-27 21:29:38 +02:00
|
|
|
reblogButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.A11YLabels.unreblog
|
2022-02-14 19:34:22 +08:00
|
|
|
} else {
|
|
|
|
reblogButton.accessibilityTraits.remove(.selected)
|
2023-03-27 21:29:38 +02:00
|
|
|
reblogButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.A11YLabels.reblog
|
2022-02-14 19:34:22 +08:00
|
|
|
}
|
2022-10-31 12:40:18 -04:00
|
|
|
reblogButton.accessibilityCustomActions = [
|
|
|
|
UIAccessibilityCustomAction(name: "Show All Reblogs") { [weak self] action in
|
|
|
|
guard let self = self else { return false }
|
|
|
|
self.delegate?.actionToolbarContainer(self, showReblogs: action)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
]
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
public func configureFavorite(count: Int, isEnabled: Bool, isHighlighted: Bool) {
|
|
|
|
let title = ActionToolbarContainer.title(from: count)
|
|
|
|
favoriteButton.setTitle(title, for: .normal)
|
2022-10-31 12:40:18 -04:00
|
|
|
favoriteButton.accessibilityValue = L10n.Plural.Count.favorite(count)
|
2022-01-27 21:23:39 +08:00
|
|
|
favoriteButton.isEnabled = isEnabled
|
|
|
|
let image = isHighlighted ? ActionToolbarContainer.starFillImage : ActionToolbarContainer.starImage
|
|
|
|
favoriteButton.setImage(image, for: .normal)
|
|
|
|
let tintColor = isHighlighted ? Asset.Colors.systemOrange.color : Asset.Colors.Button.actionToolbar.color
|
|
|
|
favoriteButton.tintColor = tintColor
|
|
|
|
favoriteButton.setTitleColor(tintColor, for: .normal)
|
|
|
|
favoriteButton.setTitleColor(tintColor, for: .highlighted)
|
2022-02-14 19:34:22 +08:00
|
|
|
|
|
|
|
if isHighlighted {
|
|
|
|
favoriteButton.accessibilityTraits.insert(.selected)
|
2022-10-31 12:40:18 -04:00
|
|
|
favoriteButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.unfavorite
|
2022-02-14 19:34:22 +08:00
|
|
|
} else {
|
|
|
|
favoriteButton.accessibilityTraits.remove(.selected)
|
2022-10-31 12:40:18 -04:00
|
|
|
favoriteButton.accessibilityLabel = L10n.Common.Controls.Status.Actions.favorite
|
2022-02-14 19:34:22 +08:00
|
|
|
}
|
2022-10-31 12:40:18 -04:00
|
|
|
favoriteButton.accessibilityCustomActions = [
|
|
|
|
UIAccessibilityCustomAction(name: "Show All Favorites") { [weak self] action in
|
|
|
|
guard let self = self else { return false }
|
|
|
|
self.delegate?.actionToolbarContainer(self, showFavorites: action)
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
]
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
private static func title(from number: Int?) -> String {
|
|
|
|
guard let number = number, number > 0 else { return "" }
|
2022-12-28 11:52:44 +01:00
|
|
|
return number.asAbbreviatedCountString()
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension ActionToolbarContainer {
|
|
|
|
public override var accessibilityElements: [Any]? {
|
|
|
|
get { [replyButton, reblogButton, favoriteButton, shareButton] }
|
|
|
|
set { }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
import SwiftUI
|
|
|
|
|
|
|
|
struct ActionToolbarContainer_Previews: PreviewProvider {
|
|
|
|
static var previews: some View {
|
|
|
|
Group {
|
|
|
|
UIViewPreview(width: 300) {
|
2023-03-20 03:39:11 -04:00
|
|
|
ActionToolbarContainer()
|
2022-01-27 21:23:39 +08:00
|
|
|
}
|
|
|
|
.previewLayout(.fixed(width: 300, height: 44))
|
|
|
|
.previewDisplayName("Inline")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|