mirror of
https://github.com/mastodon/mastodon-ios.git
synced 2025-01-07 06:50:34 +01:00
338 lines
15 KiB
Swift
338 lines
15 KiB
Swift
//
|
|
// MediaGridContainerView.swift
|
|
// MediaGridContainerView
|
|
//
|
|
// Created by Cirno MainasuK on 2021-8-23.
|
|
// Copyright © 2021 Twidere. All rights reserved.
|
|
//
|
|
|
|
import os.log
|
|
import UIKit
|
|
import func AVFoundation.AVMakeRect
|
|
|
|
public protocol MediaGridContainerViewDelegate: AnyObject {
|
|
func mediaGridContainerView(_ container: MediaGridContainerView, didTapMediaView mediaView: MediaView, at index: Int)
|
|
func mediaGridContainerView(_ container: MediaGridContainerView, toggleContentWarningOverlayViewDisplay contentWarningOverlayView: ContentWarningOverlayView)
|
|
}
|
|
|
|
public final class MediaGridContainerView: UIView {
|
|
|
|
public static let maxCount = 9
|
|
|
|
let logger = Logger(subsystem: "MediaGridContainerView", category: "UI")
|
|
|
|
public weak var delegate: MediaGridContainerViewDelegate?
|
|
public private(set) lazy var viewModel: ViewModel = {
|
|
let viewModel = ViewModel()
|
|
viewModel.bind(view: self)
|
|
return viewModel
|
|
}()
|
|
|
|
// lazy var is required here to setup gesture recognizer target-action
|
|
// Swift not doesn't emit compiler error if without `lazy` here
|
|
private(set) lazy var _mediaViews: [MediaView] = {
|
|
var mediaViews: [MediaView] = []
|
|
for i in 0..<MediaGridContainerView.maxCount {
|
|
// init media view
|
|
let mediaView = MediaView()
|
|
mediaView.tag = i
|
|
mediaViews.append(mediaView)
|
|
|
|
// add gesture recognizer
|
|
let tapGesture = UITapGestureRecognizer.singleTapGestureRecognizer
|
|
tapGesture.addTarget(self, action: #selector(MediaGridContainerView.mediaViewTapGestureRecognizerHandler(_:)))
|
|
mediaView.container.addGestureRecognizer(tapGesture)
|
|
mediaView.container.isUserInteractionEnabled = true
|
|
}
|
|
return mediaViews
|
|
}()
|
|
|
|
|
|
let sensitiveToggleButtonBlurVisualEffectView: UIVisualEffectView = {
|
|
let visualEffectView = UIVisualEffectView(effect: ContentWarningOverlayView.blurVisualEffect)
|
|
visualEffectView.layer.masksToBounds = true
|
|
visualEffectView.layer.cornerRadius = 6
|
|
visualEffectView.layer.cornerCurve = .continuous
|
|
return visualEffectView
|
|
}()
|
|
let sensitiveToggleButtonVibrancyVisualEffectView = UIVisualEffectView(effect: UIVibrancyEffect(blurEffect: ContentWarningOverlayView.blurVisualEffect))
|
|
let sensitiveToggleButton: HitTestExpandedButton = {
|
|
let button = HitTestExpandedButton(type: .system)
|
|
button.setImage(UIImage(systemName: "eye.slash.fill"), for: .normal)
|
|
return button
|
|
}()
|
|
|
|
public let contentWarningOverlayView: ContentWarningOverlayView = {
|
|
let overlay = ContentWarningOverlayView()
|
|
overlay.layer.masksToBounds = true
|
|
overlay.layer.cornerRadius = MediaView.cornerRadius
|
|
overlay.layer.cornerCurve = .continuous
|
|
return overlay
|
|
}()
|
|
|
|
public override init(frame: CGRect) {
|
|
super.init(frame: frame)
|
|
_init()
|
|
}
|
|
|
|
public required init?(coder: NSCoder) {
|
|
super.init(coder: coder)
|
|
_init()
|
|
}
|
|
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
private func _init() {
|
|
sensitiveToggleButton.addTarget(self, action: #selector(MediaGridContainerView.sensitiveToggleButtonDidPressed(_:)), for: .touchUpInside)
|
|
contentWarningOverlayView.delegate = self
|
|
}
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
@objc private func mediaViewTapGestureRecognizerHandler(_ sender: UITapGestureRecognizer) {
|
|
guard let index = _mediaViews.firstIndex(where: { $0.container === sender.view }) else { return }
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): \(index)")
|
|
let mediaView = _mediaViews[index]
|
|
delegate?.mediaGridContainerView(self, didTapMediaView: mediaView, at: index)
|
|
}
|
|
|
|
@objc private func sensitiveToggleButtonDidPressed(_ sender: UIButton) {
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public)")
|
|
delegate?.mediaGridContainerView(self, toggleContentWarningOverlayViewDisplay: contentWarningOverlayView)
|
|
}
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
|
|
public func dequeueMediaView(adaptiveLayout layout: AdaptiveLayout) -> MediaView {
|
|
prepareForReuse()
|
|
|
|
let mediaView = _mediaViews[0]
|
|
layout.layout(in: self, mediaView: mediaView)
|
|
|
|
layoutSensitiveToggleButton()
|
|
bringSubviewToFront(sensitiveToggleButtonBlurVisualEffectView)
|
|
|
|
layoutContentOverlayView(on: mediaView)
|
|
bringSubviewToFront(contentWarningOverlayView)
|
|
|
|
return mediaView
|
|
}
|
|
|
|
public func dequeueMediaView(gridLayout layout: GridLayout) -> [MediaView] {
|
|
prepareForReuse()
|
|
|
|
let mediaViews = Array(_mediaViews[0..<layout.count])
|
|
layout.layout(in: self, mediaViews: mediaViews)
|
|
|
|
layoutSensitiveToggleButton()
|
|
bringSubviewToFront(sensitiveToggleButtonBlurVisualEffectView)
|
|
|
|
layoutContentOverlayView(on: self)
|
|
bringSubviewToFront(contentWarningOverlayView)
|
|
|
|
return mediaViews
|
|
}
|
|
|
|
public func prepareForReuse() {
|
|
_mediaViews.forEach { view in
|
|
view.removeFromSuperview()
|
|
view.removeConstraints(view.constraints)
|
|
view.prepareForReuse()
|
|
}
|
|
|
|
subviews.forEach { view in
|
|
view.removeFromSuperview()
|
|
}
|
|
|
|
removeConstraints(constraints)
|
|
}
|
|
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
private func layoutSensitiveToggleButton() {
|
|
sensitiveToggleButtonBlurVisualEffectView.translatesAutoresizingMaskIntoConstraints = false
|
|
addSubview(sensitiveToggleButtonBlurVisualEffectView)
|
|
NSLayoutConstraint.activate([
|
|
sensitiveToggleButtonBlurVisualEffectView.topAnchor.constraint(equalTo: topAnchor, constant: 8),
|
|
sensitiveToggleButtonBlurVisualEffectView.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 8),
|
|
])
|
|
|
|
sensitiveToggleButtonVibrancyVisualEffectView.translatesAutoresizingMaskIntoConstraints = false
|
|
sensitiveToggleButtonBlurVisualEffectView.contentView.addSubview(sensitiveToggleButtonVibrancyVisualEffectView)
|
|
NSLayoutConstraint.activate([
|
|
sensitiveToggleButtonVibrancyVisualEffectView.topAnchor.constraint(equalTo: sensitiveToggleButtonBlurVisualEffectView.contentView.topAnchor),
|
|
sensitiveToggleButtonVibrancyVisualEffectView.leadingAnchor.constraint(equalTo: sensitiveToggleButtonBlurVisualEffectView.contentView.leadingAnchor),
|
|
sensitiveToggleButtonVibrancyVisualEffectView.trailingAnchor.constraint(equalTo: sensitiveToggleButtonBlurVisualEffectView.contentView.trailingAnchor),
|
|
sensitiveToggleButtonVibrancyVisualEffectView.bottomAnchor.constraint(equalTo: sensitiveToggleButtonBlurVisualEffectView.contentView.bottomAnchor),
|
|
])
|
|
|
|
sensitiveToggleButton.translatesAutoresizingMaskIntoConstraints = false
|
|
sensitiveToggleButtonVibrancyVisualEffectView.contentView.addSubview(sensitiveToggleButton)
|
|
NSLayoutConstraint.activate([
|
|
sensitiveToggleButton.topAnchor.constraint(equalTo: sensitiveToggleButtonVibrancyVisualEffectView.contentView.topAnchor, constant: 4),
|
|
sensitiveToggleButton.leadingAnchor.constraint(equalTo: sensitiveToggleButtonVibrancyVisualEffectView.contentView.leadingAnchor, constant: 4),
|
|
sensitiveToggleButtonVibrancyVisualEffectView.contentView.trailingAnchor.constraint(equalTo: sensitiveToggleButton.trailingAnchor, constant: 4),
|
|
sensitiveToggleButtonVibrancyVisualEffectView.contentView.bottomAnchor.constraint(equalTo: sensitiveToggleButton.bottomAnchor, constant: 4),
|
|
])
|
|
}
|
|
|
|
private func layoutContentOverlayView(on view: UIView) {
|
|
contentWarningOverlayView.translatesAutoresizingMaskIntoConstraints = false
|
|
addSubview(contentWarningOverlayView) // should add to container
|
|
NSLayoutConstraint.activate([
|
|
contentWarningOverlayView.topAnchor.constraint(equalTo: view.topAnchor),
|
|
contentWarningOverlayView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
contentWarningOverlayView.trailingAnchor.constraint(equalTo: view.trailingAnchor),
|
|
contentWarningOverlayView.bottomAnchor.constraint(equalTo: view.bottomAnchor),
|
|
])
|
|
}
|
|
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
|
|
public var mediaViews: [MediaView] {
|
|
_mediaViews.filter { $0.superview != nil }
|
|
}
|
|
|
|
public func setAlpha(_ alpha: CGFloat) {
|
|
_mediaViews.forEach { $0.alpha = alpha }
|
|
}
|
|
|
|
public func setAlpha(_ alpha: CGFloat, index: Int) {
|
|
if index < _mediaViews.count {
|
|
_mediaViews[index].alpha = alpha
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
extension MediaGridContainerView {
|
|
public struct AdaptiveLayout {
|
|
let aspectRatio: CGSize
|
|
let maxSize: CGSize
|
|
|
|
func layout(in view: UIView, mediaView: MediaView) {
|
|
let imageViewSize = AVMakeRect(aspectRatio: aspectRatio, insideRect: CGRect(origin: .zero, size: maxSize)).size
|
|
mediaView.translatesAutoresizingMaskIntoConstraints = false
|
|
view.addSubview(mediaView)
|
|
NSLayoutConstraint.activate([
|
|
mediaView.topAnchor.constraint(equalTo: view.topAnchor),
|
|
mediaView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
mediaView.trailingAnchor.constraint(equalTo: view.trailingAnchor).priority(.defaultLow),
|
|
mediaView.bottomAnchor.constraint(equalTo: view.bottomAnchor),
|
|
mediaView.widthAnchor.constraint(equalToConstant: imageViewSize.width).priority(.required - 1),
|
|
mediaView.heightAnchor.constraint(equalToConstant: imageViewSize.height).priority(.required - 1),
|
|
])
|
|
}
|
|
}
|
|
|
|
public struct GridLayout {
|
|
static let spacing: CGFloat = 1
|
|
|
|
let count: Int
|
|
let maxSize: CGSize
|
|
|
|
init(count: Int, maxSize: CGSize) {
|
|
self.count = min(count, 9)
|
|
self.maxSize = maxSize
|
|
|
|
}
|
|
|
|
private func createStackView(axis: NSLayoutConstraint.Axis) -> UIStackView {
|
|
let stackView = UIStackView()
|
|
stackView.axis = axis
|
|
stackView.semanticContentAttribute = .forceLeftToRight
|
|
stackView.spacing = GridLayout.spacing
|
|
stackView.distribution = .fillEqually
|
|
return stackView
|
|
}
|
|
|
|
public func layout(in view: UIView, mediaViews: [MediaView]) {
|
|
let containerVerticalStackView = createStackView(axis: .vertical)
|
|
containerVerticalStackView.translatesAutoresizingMaskIntoConstraints = false
|
|
view.addSubview(containerVerticalStackView)
|
|
NSLayoutConstraint.activate([
|
|
containerVerticalStackView.topAnchor.constraint(equalTo: view.topAnchor),
|
|
containerVerticalStackView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
containerVerticalStackView.trailingAnchor.constraint(equalTo: view.trailingAnchor),
|
|
containerVerticalStackView.bottomAnchor.constraint(equalTo: view.bottomAnchor),
|
|
])
|
|
|
|
let count = mediaViews.count
|
|
switch count {
|
|
case 1:
|
|
assertionFailure("should use Adaptive Layout")
|
|
containerVerticalStackView.addArrangedSubview(mediaViews[0])
|
|
case 2:
|
|
let horizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(horizontalStackView)
|
|
horizontalStackView.addArrangedSubview(mediaViews[0])
|
|
horizontalStackView.addArrangedSubview(mediaViews[1])
|
|
case 3:
|
|
let horizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(horizontalStackView)
|
|
horizontalStackView.addArrangedSubview(mediaViews[0])
|
|
|
|
let verticalStackView = createStackView(axis: .vertical)
|
|
horizontalStackView.addArrangedSubview(verticalStackView)
|
|
verticalStackView.addArrangedSubview(mediaViews[1])
|
|
verticalStackView.addArrangedSubview(mediaViews[2])
|
|
case 4:
|
|
let topHorizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(topHorizontalStackView)
|
|
topHorizontalStackView.addArrangedSubview(mediaViews[0])
|
|
topHorizontalStackView.addArrangedSubview(mediaViews[1])
|
|
|
|
let bottomHorizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(bottomHorizontalStackView)
|
|
bottomHorizontalStackView.addArrangedSubview(mediaViews[2])
|
|
bottomHorizontalStackView.addArrangedSubview(mediaViews[3])
|
|
case 5...9:
|
|
let topHorizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(topHorizontalStackView)
|
|
topHorizontalStackView.addArrangedSubview(mediaViews[0])
|
|
topHorizontalStackView.addArrangedSubview(mediaViews[1])
|
|
topHorizontalStackView.addArrangedSubview(mediaViews[2])
|
|
|
|
func mediaViewOrPlaceholderView(at index: Int) -> UIView {
|
|
return index < mediaViews.count ? mediaViews[index] : UIView()
|
|
}
|
|
let middleHorizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(middleHorizontalStackView)
|
|
middleHorizontalStackView.addArrangedSubview(mediaViews[3])
|
|
middleHorizontalStackView.addArrangedSubview(mediaViews[4])
|
|
middleHorizontalStackView.addArrangedSubview(mediaViewOrPlaceholderView(at: 5))
|
|
|
|
if count > 6 {
|
|
let bottomHorizontalStackView = createStackView(axis: .horizontal)
|
|
containerVerticalStackView.addArrangedSubview(bottomHorizontalStackView)
|
|
bottomHorizontalStackView.addArrangedSubview(mediaViewOrPlaceholderView(at: 6))
|
|
bottomHorizontalStackView.addArrangedSubview(mediaViewOrPlaceholderView(at: 7))
|
|
bottomHorizontalStackView.addArrangedSubview(mediaViewOrPlaceholderView(at: 8))
|
|
}
|
|
default:
|
|
assertionFailure()
|
|
return
|
|
}
|
|
|
|
let containerWidth = maxSize.width
|
|
let containerHeight = count > 6 ? containerWidth : containerWidth * 2 / 3
|
|
NSLayoutConstraint.activate([
|
|
view.widthAnchor.constraint(equalToConstant: containerWidth).priority(.required - 1),
|
|
view.heightAnchor.constraint(equalToConstant: containerHeight).priority(.required - 1),
|
|
])
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - ContentWarningOverlayViewDelegate
|
|
extension MediaGridContainerView: ContentWarningOverlayViewDelegate {
|
|
public func contentWarningOverlayViewDidPressed(_ contentWarningOverlayView: ContentWarningOverlayView) {
|
|
delegate?.mediaGridContainerView(self, toggleContentWarningOverlayViewDisplay: contentWarningOverlayView)
|
|
}
|
|
}
|