1
0
mirror of https://github.com/mastodon/mastodon-ios.git synced 2025-01-23 22:11:22 +01:00
mastodon-app-ufficiale-ipho.../MastodonSDK/Sources/MastodonUI/Scene/ComposeContent/AutoComplete/View/AutoCompleteTopChevronView.swift
2023-09-27 15:08:12 +02:00

174 lines
5.3 KiB
Swift

//
// AutoCompleteTopChevronView.swift
// Mastodon
//
// Created by MainasuK Cirno on 2021-5-14.
//
import UIKit
import Combine
import MastodonCore
final class AutoCompleteTopChevronView: UIView {
var disposeBag = Set<AnyCancellable>()
static let chevronSize = CGSize(width: 20, height: 12)
private let shadowView = UIView()
private let shadowLayer = CAShapeLayer()
private let maskLayer = CAShapeLayer()
var chevronMinX: CGFloat = 0
var topViewBackgroundColor = SystemTheme.systemElevatedBackgroundColor
var bottomViewBackgroundColor = UIColor.systemBackground
override init(frame: CGRect) {
super.init(frame: frame)
_init()
}
required init?(coder: NSCoder) {
super.init(coder: coder)
_init()
}
}
extension AutoCompleteTopChevronView {
var standardizedChevronMinX: CGFloat {
min(max(chevronMinX, 0), bounds.width - AutoCompleteTopChevronView.chevronSize.width)
}
var edgePoints: [CGPoint] {
[
CGPoint(x: standardizedChevronMinX, y: bounds.maxY),
CGPoint(x: standardizedChevronMinX + 0.5 * AutoCompleteTopChevronView.chevronSize.width, y: bounds.maxY - AutoCompleteTopChevronView.chevronSize.height),
CGPoint(x: standardizedChevronMinX + AutoCompleteTopChevronView.chevronSize.width, y: bounds.maxY),
CGPoint(x: bounds.width, y: bounds.maxY),
]
}
}
extension AutoCompleteTopChevronView {
private func _init() {
clipsToBounds = false
backgroundColor = .clear
isUserInteractionEnabled = false
shadowView.translatesAutoresizingMaskIntoConstraints = false
addSubview(shadowView)
shadowView.pinToParent()
shadowLayer.fillColor = topViewBackgroundColor.cgColor
shadowView.layer.addSublayer(shadowLayer)
}
override func layoutSubviews() {
super.layoutSubviews()
// 1. setup shadow with chevron
shadowLayer.path = {
let path = UIBezierPath()
path.move(to: CGPoint(x: 0, y: bounds.maxY))
// bottom edge
for point in edgePoints {
path.addLine(to: point)
}
// up egde
path.addLine(to: CGPoint(x: bounds.maxX, y: 0))
path.addLine(to: CGPoint(x: 0, y: 0))
path.close()
return path.cgPath
}()
shadowLayer.fillColor = topViewBackgroundColor.cgColor
// 2. setup mask to clip shadow
maskLayer.path = {
let path = UIBezierPath()
path.move(to: CGPoint(x: 0, y: bounds.maxY))
// up edge
for point in edgePoints {
path.addLine(to: CGPoint(x: point.x, y: point.y - 2)) // move up 2pt
}
// bottom egde
path.addLine(to: CGPoint(x: bounds.maxX, y: 2 * bounds.maxY))
path.addLine(to: CGPoint(x: 0, y: 2 * bounds.maxY))
path.close()
return path.cgPath
}()
maskLayer.fillColor = UIColor.red.cgColor
shadowView.layer.setupShadow(color: UIColor.black.withAlphaComponent(0.25))
shadowView.layer.mask = maskLayer
layer.mask = maskLayer
}
}
extension AutoCompleteTopChevronView {
func invertMask(in rect: CGRect) -> CAShapeLayer {
let path = UIBezierPath()
path.move(to: CGPoint(x: 0, y: bounds.maxY))
// top edge
for point in edgePoints {
path.addLine(to: point)
}
// bottom edge
path.addLine(to: CGPoint(x: rect.maxX, y: rect.maxY))
path.addLine(to: CGPoint(x: 0, y: rect.maxY))
path.close()
let mask = CAShapeLayer()
mask.fillColor = UIColor.red.cgColor
mask.path = path.cgPath
return mask
}
}
#if canImport(SwiftUI) && DEBUG
import SwiftUI
struct AutoCompleteTopChevronView_Previews: PreviewProvider {
static var previews: some View {
Group {
UIViewPreview(width: 375) {
let view = AutoCompleteTopChevronView()
view.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
view.widthAnchor.constraint(equalToConstant: 375),
view.heightAnchor.constraint(equalToConstant: 100),
])
view.chevronMinX = 10
return view
}
.background(Color(SystemTheme.systemElevatedBackgroundColor))
.padding(20)
.previewLayout(.fixed(width: 375 + 40, height: 100 + 40))
UIViewPreview(width: 375) {
let view = AutoCompleteTopChevronView()
view.translatesAutoresizingMaskIntoConstraints = false
NSLayoutConstraint.activate([
view.widthAnchor.constraint(equalToConstant: 375),
view.heightAnchor.constraint(equalToConstant: 100),
])
view.chevronMinX = 10
return view
}
.background(Color(SystemTheme.systemElevatedBackgroundColor))
.preferredColorScheme(.dark)
.padding(20)
.previewLayout(.fixed(width: 375 + 40, height: 100 + 40))
}
}
}
#endif