mirror of
https://github.com/Swiftgram/Telegram-iOS.git
synced 2025-06-16 05:55:20 +00:00
283 lines
13 KiB
Swift
283 lines
13 KiB
Swift
import Foundation
|
|
import UIKit
|
|
import AsyncDisplayKit
|
|
|
|
open class ASButtonNode: ASControlNode {
|
|
public let titleNode: ImmediateTextNode
|
|
public let highlightedTitleNode: ImmediateTextNode
|
|
public let imageNode: ASImageNode
|
|
public let backgroundImageNode: ASImageNode
|
|
public let highlightedBackgroundImageNode: ASImageNode
|
|
|
|
public var contentEdgeInsets: UIEdgeInsets = UIEdgeInsets() {
|
|
didSet {
|
|
if self.contentEdgeInsets != oldValue {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
}
|
|
}
|
|
|
|
public var contentHorizontalAlignment: ASHorizontalAlignment = .middle {
|
|
didSet {
|
|
if self.contentHorizontalAlignment != oldValue {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
}
|
|
}
|
|
|
|
public var laysOutHorizontally: Bool = true {
|
|
didSet {
|
|
if self.laysOutHorizontally != oldValue {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
}
|
|
}
|
|
|
|
public var contentSpacing: CGFloat = 0.0 {
|
|
didSet {
|
|
if self.contentSpacing != oldValue {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
}
|
|
}
|
|
|
|
private var calculatedTitleSize: CGSize = CGSize()
|
|
private var calculatedHighlightedTitleSize: CGSize = CGSize()
|
|
|
|
override public init() {
|
|
self.titleNode = ImmediateTextNode()
|
|
self.titleNode.isUserInteractionEnabled = false
|
|
self.titleNode.displaysAsynchronously = false
|
|
|
|
self.highlightedTitleNode = ImmediateTextNode()
|
|
self.highlightedTitleNode.isUserInteractionEnabled = false
|
|
self.highlightedTitleNode.displaysAsynchronously = false
|
|
|
|
self.imageNode = ASImageNode()
|
|
self.imageNode.isUserInteractionEnabled = false
|
|
self.imageNode.displaysAsynchronously = false
|
|
self.imageNode.displayWithoutProcessing = true
|
|
|
|
self.backgroundImageNode = ASImageNode()
|
|
self.backgroundImageNode.isUserInteractionEnabled = false
|
|
self.backgroundImageNode.displaysAsynchronously = false
|
|
self.backgroundImageNode.displayWithoutProcessing = true
|
|
|
|
self.highlightedBackgroundImageNode = ASImageNode()
|
|
self.highlightedBackgroundImageNode.isUserInteractionEnabled = false
|
|
self.highlightedBackgroundImageNode.displaysAsynchronously = false
|
|
self.highlightedBackgroundImageNode.displayWithoutProcessing = true
|
|
|
|
super.init()
|
|
|
|
self.addSubnode(self.backgroundImageNode)
|
|
self.addSubnode(self.highlightedBackgroundImageNode)
|
|
self.highlightedBackgroundImageNode.isHidden = true
|
|
self.addSubnode(self.titleNode)
|
|
self.addSubnode(self.highlightedTitleNode)
|
|
self.highlightedTitleNode.isHidden = true
|
|
self.addSubnode(self.imageNode)
|
|
}
|
|
|
|
override open func calculateSizeThatFits(_ constrainedSize: CGSize) -> CGSize {
|
|
let horizontalInsets = self.contentEdgeInsets.left + self.contentEdgeInsets.right
|
|
let verticalInsets = self.contentEdgeInsets.top + self.contentEdgeInsets.bottom
|
|
|
|
let imageSize = self.imageNode.image?.size ?? CGSize()
|
|
|
|
let widthForTitle: CGFloat
|
|
if self.laysOutHorizontally {
|
|
widthForTitle = max(1.0, constrainedSize.width - horizontalInsets - imageSize.width - (imageSize.width.isZero ? 0.0 : self.contentSpacing))
|
|
} else {
|
|
widthForTitle = max(1.0, constrainedSize.width - horizontalInsets)
|
|
}
|
|
|
|
let normalTitleSize = self.titleNode.updateLayout(CGSize(width: widthForTitle, height: max(1.0, constrainedSize.height - verticalInsets)))
|
|
self.calculatedTitleSize = normalTitleSize
|
|
let highlightedTitleSize = self.highlightedTitleNode.updateLayout(CGSize(width: widthForTitle, height: max(1.0, constrainedSize.height - verticalInsets)))
|
|
self.calculatedHighlightedTitleSize = highlightedTitleSize
|
|
|
|
let titleSize = CGSize(width: max(normalTitleSize.width, highlightedTitleSize.width), height: max(normalTitleSize.height, highlightedTitleSize.height))
|
|
|
|
var contentSize: CGSize
|
|
if self.laysOutHorizontally {
|
|
contentSize = CGSize(width: titleSize.width + imageSize.width, height: max(titleSize.height, imageSize.height))
|
|
if !titleSize.width.isZero && !imageSize.width.isZero {
|
|
contentSize.width += self.contentSpacing
|
|
}
|
|
} else {
|
|
contentSize = CGSize(width: max(titleSize.width, imageSize.width), height: titleSize.height + imageSize.height)
|
|
if !titleSize.width.isZero && !imageSize.width.isZero {
|
|
contentSize.height += self.contentSpacing
|
|
}
|
|
}
|
|
|
|
return CGSize(width: min(constrainedSize.width, contentSize.width + self.contentEdgeInsets.left + self.contentEdgeInsets.right), height: min(constrainedSize.height, contentSize.height + self.contentEdgeInsets.top + self.contentEdgeInsets.bottom))
|
|
}
|
|
|
|
open func setAttributedTitle(_ title: NSAttributedString, for state: UIControl.State) {
|
|
if state == [] {
|
|
if let attributedText = self.titleNode.attributedText {
|
|
if !attributedText.isEqual(to: title) {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
} else {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
self.titleNode.attributedText = title
|
|
|
|
if let attributedText = self.highlightedTitleNode.attributedText {
|
|
if !attributedText.isEqual(to: title) {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
} else {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
self.highlightedTitleNode.attributedText = title
|
|
} else if state == .highlighted || state == .selected {
|
|
if let attributedText = self.highlightedTitleNode.attributedText {
|
|
if !attributedText.isEqual(to: title) {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
} else {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
self.highlightedTitleNode.attributedText = title
|
|
} else {
|
|
if let attributedText = self.titleNode.attributedText {
|
|
if !attributedText.isEqual(to: title) {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
} else {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
self.titleNode.attributedText = title
|
|
}
|
|
}
|
|
|
|
open func attributedTitle(for state: UIControl.State) -> NSAttributedString? {
|
|
if state == .highlighted || state == .selected {
|
|
return self.highlightedTitleNode.attributedText
|
|
} else {
|
|
return self.titleNode.attributedText
|
|
}
|
|
}
|
|
|
|
open func setTitle(_ title: String, with font: UIFont, with color: UIColor, for state: UIControl.State) {
|
|
self.setAttributedTitle(NSAttributedString(string: title, font: font, textColor: color), for: state)
|
|
}
|
|
|
|
open func setImage(_ image: UIImage?, for state: UIControl.State) {
|
|
if image?.size != self.imageNode.image?.size {
|
|
self.invalidateCalculatedLayout()
|
|
self.setNeedsLayout()
|
|
}
|
|
self.imageNode.image = image
|
|
}
|
|
|
|
open func setBackgroundImage(_ image: UIImage?, for state: UIControl.State) {
|
|
if state == [] {
|
|
self.backgroundImageNode.image = image
|
|
self.highlightedBackgroundImageNode.image = image
|
|
} else if state == .highlighted || state == .selected {
|
|
self.highlightedBackgroundImageNode.image = image
|
|
} else {
|
|
self.backgroundImageNode.image = image
|
|
}
|
|
}
|
|
|
|
open func image(for state: UIControl.State) -> UIImage? {
|
|
return self.imageNode.image
|
|
}
|
|
|
|
open func backgroundImage(for state: UIControl.State) -> UIImage? {
|
|
return self.backgroundImageNode.image
|
|
}
|
|
|
|
override open var isHighlighted: Bool {
|
|
didSet {
|
|
if self.isHighlighted != oldValue {
|
|
if self.isHighlighted {
|
|
if self.highlightedTitleNode.attributedText != nil {
|
|
self.highlightedTitleNode.isHidden = false
|
|
self.titleNode.isHidden = true
|
|
} else {
|
|
self.highlightedTitleNode.isHidden = true
|
|
self.titleNode.isHidden = false
|
|
}
|
|
if self.highlightedBackgroundImageNode.image != nil {
|
|
self.highlightedBackgroundImageNode.isHidden = false
|
|
self.backgroundImageNode.isHidden = true
|
|
} else {
|
|
self.highlightedBackgroundImageNode.isHidden = true
|
|
self.backgroundImageNode.isHidden = false
|
|
}
|
|
} else {
|
|
self.highlightedTitleNode.isHidden = true
|
|
self.titleNode.isHidden = false
|
|
|
|
self.highlightedBackgroundImageNode.isHidden = true
|
|
self.backgroundImageNode.isHidden = false
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
override open func layout() {
|
|
let size = self.bounds.size
|
|
|
|
let contentRect = CGRect(origin: CGPoint(x: self.contentEdgeInsets.left, y: self.contentEdgeInsets.top), size: CGSize(width: size.width - self.contentEdgeInsets.left - self.contentEdgeInsets.right, height: size.height - self.contentEdgeInsets.top - self.contentEdgeInsets.bottom))
|
|
|
|
let imageSize = self.imageNode.image?.size ?? CGSize()
|
|
|
|
let titleOrigin: CGPoint
|
|
let highlightedTitleOrigin: CGPoint
|
|
let imageOrigin: CGPoint
|
|
|
|
if self.laysOutHorizontally {
|
|
switch self.contentHorizontalAlignment {
|
|
case .left:
|
|
titleOrigin = CGPoint(x: contentRect.minX, y: contentRect.minY + floor((contentRect.height - self.calculatedTitleSize.height) / 2.0))
|
|
highlightedTitleOrigin = CGPoint(x: contentRect.minX, y: contentRect.minY + floor((contentRect.height - self.calculatedHighlightedTitleSize.height) / 2.0))
|
|
imageOrigin = CGPoint(x: titleOrigin.x + self.calculatedTitleSize.width + self.contentSpacing, y: contentRect.minY + floor((contentRect.height - imageSize.height) / 2.0))
|
|
case .right:
|
|
titleOrigin = CGPoint(x: contentRect.maxX - self.calculatedTitleSize.width, y: contentRect.minY + floor((contentRect.height - self.calculatedTitleSize.height) / 2.0))
|
|
highlightedTitleOrigin = CGPoint(x: contentRect.maxX - self.calculatedHighlightedTitleSize.width, y: contentRect.minY + floor((contentRect.height - self.calculatedHighlightedTitleSize.height) / 2.0))
|
|
imageOrigin = CGPoint(x: titleOrigin.x - self.contentSpacing - imageSize.width, y: contentRect.minY + floor((contentRect.height - imageSize.height) / 2.0))
|
|
default:
|
|
titleOrigin = CGPoint(x: contentRect.minY + floor((contentRect.width - self.calculatedTitleSize.width) / 2.0), y: contentRect.minY + floor((contentRect.height - self.calculatedTitleSize.height) / 2.0))
|
|
highlightedTitleOrigin = CGPoint(x: floor((contentRect.width - self.calculatedHighlightedTitleSize.width) / 2.0), y: contentRect.minY + floor((contentRect.height - self.calculatedHighlightedTitleSize.height) / 2.0))
|
|
imageOrigin = CGPoint(x: floor((contentRect.width - imageSize.width) / 2.0), y: contentRect.minY + floor((contentRect.height - imageSize.height) / 2.0))
|
|
}
|
|
} else {
|
|
var contentHeight: CGFloat = self.calculatedTitleSize.height
|
|
if !imageSize.height.isZero {
|
|
contentHeight += self.contentSpacing + imageSize.height
|
|
}
|
|
let contentY = contentRect.minY + floor((contentRect.height - contentHeight) / 2.0)
|
|
titleOrigin = CGPoint(x: contentRect.minX + floor((contentRect.width - self.calculatedTitleSize.width) / 2.0), y: contentY + contentHeight - self.calculatedTitleSize.height)
|
|
highlightedTitleOrigin = CGPoint(x: contentRect.minX + floor((contentRect.width - self.calculatedHighlightedTitleSize.width) / 2.0), y: contentY + contentHeight - self.calculatedHighlightedTitleSize.height)
|
|
imageOrigin = CGPoint(x: floor((contentRect.width - imageSize.width) / 2.0), y: contentY)
|
|
}
|
|
|
|
self.titleNode.frame = CGRect(origin: titleOrigin, size: self.calculatedTitleSize)
|
|
self.highlightedTitleNode.frame = CGRect(origin: highlightedTitleOrigin, size: self.calculatedHighlightedTitleSize)
|
|
self.imageNode.frame = CGRect(origin: imageOrigin, size: imageSize)
|
|
|
|
self.backgroundImageNode.frame = CGRect(origin: CGPoint(), size: size)
|
|
self.highlightedBackgroundImageNode.frame = CGRect(origin: CGPoint(), size: size)
|
|
}
|
|
}
|