mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00
160 lines
6.6 KiB
Swift
160 lines
6.6 KiB
Swift
import Foundation
|
|
import UIKit
|
|
import Display
|
|
import AsyncDisplayKit
|
|
import TelegramPresentationData
|
|
import ComponentFlow
|
|
import EmojiStatusComponent
|
|
import AccountContext
|
|
|
|
public struct WebAppTitle: Equatable {
|
|
public var title: String
|
|
public var counter: String
|
|
public var isVerified: Bool
|
|
|
|
public init(title: String, counter: String, isVerified: Bool) {
|
|
self.title = title
|
|
self.counter = counter
|
|
self.isVerified = isVerified
|
|
}
|
|
}
|
|
|
|
public final class WebAppTitleView: UIView {
|
|
private let context: AccountContext
|
|
|
|
private let titleNode: ImmediateTextNode
|
|
private let subtitleNode: ImmediateTextNode
|
|
private var titleCredibilityIconView: ComponentHostView<Empty>?
|
|
|
|
public var title: WebAppTitle = WebAppTitle(title: "", counter: "", isVerified: false) {
|
|
didSet {
|
|
if self.title != oldValue {
|
|
self.update()
|
|
}
|
|
}
|
|
}
|
|
|
|
public var theme: PresentationTheme {
|
|
didSet {
|
|
self.update()
|
|
}
|
|
}
|
|
|
|
private var primaryTextColor: UIColor?
|
|
private var secondaryTextColor: UIColor?
|
|
|
|
public func updateTextColors(primary: UIColor?, secondary: UIColor?, transition: ContainedViewLayoutTransition) {
|
|
self.primaryTextColor = primary
|
|
self.secondaryTextColor = secondary
|
|
|
|
if case let .animated(duration, curve) = transition {
|
|
if let snapshotView = self.snapshotContentTree() {
|
|
snapshotView.frame = self.bounds
|
|
self.addSubview(snapshotView)
|
|
|
|
snapshotView.layer.animateAlpha(from: 1.0, to: 0.0, duration: duration, timingFunction: curve.timingFunction, removeOnCompletion: false, completion: { _ in
|
|
snapshotView.removeFromSuperview()
|
|
})
|
|
self.titleNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: duration, timingFunction: curve.timingFunction)
|
|
self.subtitleNode.layer.animateAlpha(from: 0.0, to: 1.0, duration: duration, timingFunction: curve.timingFunction)
|
|
}
|
|
}
|
|
|
|
self.update()
|
|
}
|
|
|
|
private func update() {
|
|
let primaryTextColor = self.primaryTextColor ?? self.theme.rootController.navigationBar.primaryTextColor
|
|
let secondaryTextColor = self.secondaryTextColor ?? self.theme.rootController.navigationBar.secondaryTextColor
|
|
self.titleNode.attributedText = NSAttributedString(string: self.title.title, font: Font.semibold(17.0), textColor: primaryTextColor)
|
|
self.subtitleNode.attributedText = NSAttributedString(string: self.title.counter, font: Font.regular(13.0), textColor: secondaryTextColor)
|
|
|
|
self.accessibilityLabel = self.title.title
|
|
self.accessibilityValue = self.title.counter
|
|
|
|
self.setNeedsLayout()
|
|
}
|
|
|
|
public init(context: AccountContext, theme: PresentationTheme) {
|
|
self.context = context
|
|
self.theme = theme
|
|
|
|
self.titleNode = ImmediateTextNode()
|
|
self.titleNode.displaysAsynchronously = false
|
|
self.titleNode.maximumNumberOfLines = 1
|
|
self.titleNode.truncationType = .end
|
|
self.titleNode.isOpaque = false
|
|
|
|
self.subtitleNode = ImmediateTextNode()
|
|
self.subtitleNode.displaysAsynchronously = false
|
|
self.subtitleNode.maximumNumberOfLines = 1
|
|
self.subtitleNode.truncationType = .end
|
|
self.subtitleNode.isOpaque = false
|
|
|
|
super.init(frame: CGRect())
|
|
|
|
self.isAccessibilityElement = true
|
|
self.accessibilityTraits = .header
|
|
|
|
self.addSubnode(self.titleNode)
|
|
self.addSubnode(self.subtitleNode)
|
|
}
|
|
|
|
required public init?(coder aDecoder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
override public func layoutSubviews() {
|
|
super.layoutSubviews()
|
|
|
|
let size = self.bounds.size
|
|
let spacing: CGFloat = 0.0
|
|
|
|
let titleSize = self.titleNode.updateLayout(CGSize(width: max(1.0, size.width), height: size.height))
|
|
let subtitleSize = self.subtitleNode.updateLayout(CGSize(width: max(1.0, size.width), height: size.height))
|
|
let combinedHeight = titleSize.height + subtitleSize.height + spacing
|
|
|
|
|
|
var totalWidth = titleSize.width
|
|
|
|
if self.title.isVerified {
|
|
let statusContent: EmojiStatusComponent.Content = .verified(fillColor: self.theme.list.itemCheckColors.fillColor, foregroundColor: self.theme.list.itemCheckColors.foregroundColor, sizeType: .large)
|
|
let titleCredibilityIconTransition: ComponentTransition = .immediate
|
|
|
|
let titleCredibilityIconView: ComponentHostView<Empty>
|
|
if let current = self.titleCredibilityIconView {
|
|
titleCredibilityIconView = current
|
|
} else {
|
|
titleCredibilityIconView = ComponentHostView<Empty>()
|
|
self.titleCredibilityIconView = titleCredibilityIconView
|
|
self.addSubview(titleCredibilityIconView)
|
|
}
|
|
|
|
let titleIconSize = titleCredibilityIconView.update(
|
|
transition: titleCredibilityIconTransition,
|
|
component: AnyComponent(EmojiStatusComponent(
|
|
context: self.context,
|
|
animationCache: self.context.animationCache,
|
|
animationRenderer: self.context.animationRenderer,
|
|
content: statusContent,
|
|
isVisibleForAnimations: true,
|
|
action: {
|
|
}
|
|
)),
|
|
environment: {},
|
|
containerSize: CGSize(width: 20.0, height: 20.0)
|
|
)
|
|
|
|
totalWidth += titleIconSize.width + 2.0
|
|
|
|
titleCredibilityIconTransition.setFrame(view: titleCredibilityIconView, frame: CGRect(origin: CGPoint(x:floorToScreenPixels((size.width - totalWidth) / 2.0) + titleSize.width + 2.0, y: floorToScreenPixels(floorToScreenPixels((size.height - combinedHeight) / 2.0 + titleSize.height / 2.0) - titleIconSize.height / 2.0)), size: titleIconSize))
|
|
}
|
|
|
|
let titleFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((size.width - totalWidth) / 2.0), y: floorToScreenPixels((size.height - combinedHeight) / 2.0)), size: titleSize)
|
|
self.titleNode.frame = titleFrame
|
|
|
|
let subtitleFrame = CGRect(origin: CGPoint(x: floorToScreenPixels((size.width - subtitleSize.width) / 2.0), y: floorToScreenPixels((size.height - combinedHeight) / 2.0) + titleSize.height + spacing), size: subtitleSize)
|
|
self.subtitleNode.frame = subtitleFrame
|
|
}
|
|
}
|