Swiftgram/submodules/InstantPageUI/Sources/InstantPageDetailsItem.swift
2022-12-17 00:17:31 +04:00

116 lines
5.1 KiB
Swift

import Foundation
import UIKit
import Postbox
import TelegramCore
import AsyncDisplayKit
import Display
import TelegramPresentationData
import TelegramUIPreferences
import AccountContext
import ContextUI
final class InstantPageDetailsItem: InstantPageItem {
var frame: CGRect
let wantsNode: Bool = true
let separatesTiles: Bool = true
var medias: [InstantPageMedia] {
var result: [InstantPageMedia] = []
for item in self.items {
result.append(contentsOf: item.medias)
}
return result
}
let titleItems: [InstantPageItem]
let titleHeight: CGFloat
let items: [InstantPageItem]
let safeInset: CGFloat
let rtl: Bool
let initiallyExpanded: Bool
let index: Int
init(frame: CGRect, titleItems: [InstantPageItem], titleHeight: CGFloat, items: [InstantPageItem], safeInset: CGFloat, rtl: Bool, initiallyExpanded: Bool, index: Int) {
self.frame = frame
self.titleItems = titleItems
self.titleHeight = titleHeight
self.items = items
self.safeInset = safeInset
self.rtl = rtl
self.initiallyExpanded = initiallyExpanded
self.index = index
}
func node(context: AccountContext, strings: PresentationStrings, nameDisplayOrder: PresentationPersonNameOrder, theme: InstantPageTheme, sourceLocation: InstantPageSourceLocation, openMedia: @escaping (InstantPageMedia) -> Void, longPressMedia: @escaping (InstantPageMedia) -> Void, activatePinchPreview: ((PinchSourceContainerNode) -> Void)?, pinchPreviewFinished: ((InstantPageNode) -> Void)?, openPeer: @escaping (EnginePeer) -> Void, openUrl: @escaping (InstantPageUrlItem) -> Void, updateWebEmbedHeight: @escaping (CGFloat) -> Void, updateDetailsExpanded: @escaping (Bool) -> Void, currentExpandedDetails: [Int : Bool]?) -> InstantPageNode? {
var expanded: Bool?
if let expandedDetails = currentExpandedDetails, let currentlyExpanded = expandedDetails[self.index] {
expanded = currentlyExpanded
}
return InstantPageDetailsNode(context: context, sourceLocation: sourceLocation, strings: strings, nameDisplayOrder: nameDisplayOrder, theme: theme, item: self, openMedia: openMedia, longPressMedia: longPressMedia, openPeer: openPeer, openUrl: openUrl, currentlyExpanded: expanded, updateDetailsExpanded: updateDetailsExpanded)
}
func matchesAnchor(_ anchor: String) -> Bool {
return false
}
func matchesNode(_ node: InstantPageNode) -> Bool {
if let node = node as? InstantPageDetailsNode {
return self === node.item
} else {
return false
}
}
func distanceThresholdGroup() -> Int? {
return 8
}
func distanceThresholdWithGroupCount(_ count: Int) -> CGFloat {
return CGFloat.greatestFiniteMagnitude
}
func drawInTile(context: CGContext) {
}
func linkSelectionRects(at point: CGPoint) -> [CGRect] {
if point.y < self.titleHeight {
for item in self.titleItems {
if item.frame.contains(point) {
let rects = item.linkSelectionRects(at: point.offsetBy(dx: -item.frame.minX, dy: -item.frame.minY))
return rects.map { $0.offsetBy(dx: item.frame.minX, dy: item.frame.minY) }
}
}
} else {
let convertedPoint = point.offsetBy(dx: 0.0, dy: -self.titleHeight)
for item in self.items {
if item.frame.contains(convertedPoint) {
let rects = item.linkSelectionRects(at: convertedPoint.offsetBy(dx: -item.frame.minX, dy: -item.frame.minY))
if !rects.isEmpty {
return rects.map { $0.offsetBy(dx: item.frame.minX, dy: item.frame.minY + self.titleHeight) }
}
}
}
}
return []
}
}
func layoutDetailsItem(theme: InstantPageTheme, title: NSAttributedString, boundingWidth: CGFloat, items: [InstantPageItem], contentSize: CGSize, safeInset: CGFloat, rtl: Bool, initiallyExpanded: Bool, index: Int) -> InstantPageDetailsItem {
let detailsInset: CGFloat = 17.0 + safeInset
let titleInset: CGFloat = 22.0
let (_, titleItems, titleSize) = layoutTextItemWithString(title, boundingWidth: boundingWidth - detailsInset * 2.0 - titleInset, offset: CGPoint(x: detailsInset + titleInset, y: 0.0))
let titleHeight = max(44.0, titleSize.height + 26.0)
var offset: CGFloat?
for var item in titleItems {
var itemOffset = floorToScreenPixels((titleHeight - item.frame.height) / 2.0)
if item is InstantPageTextItem {
offset = itemOffset
} else if let offset = offset {
itemOffset = offset
}
item.frame = item.frame.offsetBy(dx: 0.0, dy: itemOffset)
}
return InstantPageDetailsItem(frame: CGRect(x: 0.0, y: 0.0, width: boundingWidth, height: contentSize.height + titleHeight), titleItems: titleItems, titleHeight: titleHeight, items: items, safeInset: safeInset, rtl: rtl, initiallyExpanded: initiallyExpanded, index: index)
}