2021-12-08 02:58:02 +00:00
|
|
|
//
|
|
|
|
// HomeViewController.swift
|
|
|
|
// Reader
|
|
|
|
//
|
|
|
|
// Created by Shadowfacts on 11/25/21.
|
|
|
|
//
|
|
|
|
|
|
|
|
import UIKit
|
2022-01-09 16:11:52 +00:00
|
|
|
import CoreData
|
2022-01-27 03:37:10 +00:00
|
|
|
import Combine
|
2022-06-19 18:56:56 +00:00
|
|
|
import Persistence
|
2021-12-08 02:58:02 +00:00
|
|
|
|
2022-01-12 18:48:52 +00:00
|
|
|
protocol HomeViewControllerDelegate: AnyObject {
|
|
|
|
func switchToAccount(_ account: LocalData.Account)
|
|
|
|
}
|
|
|
|
|
2022-01-09 22:13:30 +00:00
|
|
|
class HomeViewController: UIViewController {
|
2021-12-08 02:58:02 +00:00
|
|
|
|
2022-01-12 18:48:52 +00:00
|
|
|
weak var delegate: HomeViewControllerDelegate?
|
2022-01-15 19:09:30 +00:00
|
|
|
weak var itemsDelegate: ItemsViewControllerDelegate?
|
2022-01-12 18:48:52 +00:00
|
|
|
|
2021-12-08 02:58:02 +00:00
|
|
|
let fervorController: FervorController
|
|
|
|
|
2022-01-15 19:09:30 +00:00
|
|
|
var enableStretchyMenu = true
|
|
|
|
|
2022-01-09 16:11:52 +00:00
|
|
|
private var collectionView: UICollectionView!
|
2022-03-13 18:51:29 +00:00
|
|
|
private var dataSource: UICollectionViewDiffableDataSource<Section, ItemListType>!
|
2022-01-09 16:11:52 +00:00
|
|
|
private var groupResultsController: NSFetchedResultsController<Group>!
|
|
|
|
private var feedResultsController: NSFetchedResultsController<Feed>!
|
|
|
|
|
2022-01-27 03:37:10 +00:00
|
|
|
private var lastSyncState = FervorController.SyncState.done
|
2022-06-25 22:40:56 +00:00
|
|
|
// weak so that when it's removed from the superview, this becomes nil
|
|
|
|
private weak var syncStateView: SyncStateView?
|
2022-01-27 03:37:10 +00:00
|
|
|
private var cancellables = Set<AnyCancellable>()
|
|
|
|
|
2021-12-08 02:58:02 +00:00
|
|
|
init(fervorController: FervorController) {
|
|
|
|
self.fervorController = fervorController
|
|
|
|
|
|
|
|
super.init(nibName: nil, bundle: nil)
|
2022-06-25 22:40:56 +00:00
|
|
|
|
|
|
|
fervorController.syncState
|
|
|
|
.buffer(size: 25, prefetch: .byRequest, whenFull: .dropOldest)
|
|
|
|
.receive(on: RunLoop.main)
|
|
|
|
.sink { [unowned self] in
|
|
|
|
self.syncStateChanged($0)
|
|
|
|
}
|
|
|
|
.store(in: &cancellables)
|
2021-12-08 02:58:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
|
|
|
|
|
|
|
override func viewDidLoad() {
|
|
|
|
super.viewDidLoad()
|
|
|
|
|
2022-01-10 00:23:22 +00:00
|
|
|
// todo: account info
|
|
|
|
title = "Reader"
|
2021-12-08 02:58:02 +00:00
|
|
|
|
2022-01-15 19:09:30 +00:00
|
|
|
if enableStretchyMenu {
|
|
|
|
view.addInteraction(StretchyMenuInteraction(delegate: self))
|
|
|
|
}
|
2022-01-12 16:36:12 +00:00
|
|
|
|
2022-01-15 19:09:30 +00:00
|
|
|
if UIDevice.current.userInterfaceIdiom != .mac {
|
|
|
|
view.backgroundColor = .appBackground
|
|
|
|
}
|
2022-01-12 16:36:12 +00:00
|
|
|
|
2022-01-15 19:09:30 +00:00
|
|
|
var config = UICollectionLayoutListConfiguration(appearance: UIDevice.current.userInterfaceIdiom == .mac ? .sidebar : .grouped)
|
2022-01-10 00:23:22 +00:00
|
|
|
config.headerMode = .supplementary
|
2022-01-15 19:09:30 +00:00
|
|
|
config.backgroundColor = .clear
|
2022-01-10 00:23:22 +00:00
|
|
|
config.separatorConfiguration.topSeparatorVisibility = .visible
|
2022-01-14 23:17:00 +00:00
|
|
|
config.separatorConfiguration.topSeparatorInsets = NSDirectionalEdgeInsets(top: 0, leading: 20, bottom: 0, trailing: 0)
|
2022-01-10 00:23:22 +00:00
|
|
|
config.separatorConfiguration.bottomSeparatorVisibility = .hidden
|
|
|
|
config.itemSeparatorHandler = { indexPath, defaultConfig in
|
|
|
|
var config = defaultConfig
|
|
|
|
if indexPath.section == 0 && indexPath.row == 0 {
|
|
|
|
config.topSeparatorVisibility = .hidden
|
|
|
|
}
|
|
|
|
return config
|
|
|
|
}
|
|
|
|
let layout = UICollectionViewCompositionalLayout.list(using: config)
|
2022-01-09 16:11:52 +00:00
|
|
|
collectionView = UICollectionView(frame: view.bounds, collectionViewLayout: layout)
|
|
|
|
collectionView.autoresizingMask = [.flexibleWidth, .flexibleHeight]
|
|
|
|
collectionView.delegate = self
|
|
|
|
view.addSubview(collectionView)
|
|
|
|
|
|
|
|
dataSource = createDataSource()
|
|
|
|
|
2022-03-13 18:51:29 +00:00
|
|
|
var snapshot = NSDiffableDataSourceSnapshot<Section, ItemListType>()
|
2022-01-09 23:01:19 +00:00
|
|
|
snapshot.appendSections([.all, .groups, .feeds])
|
|
|
|
snapshot.appendItems([.unread, .all], toSection: .all)
|
2022-01-09 16:11:52 +00:00
|
|
|
dataSource.apply(snapshot, animatingDifferences: false)
|
|
|
|
|
|
|
|
let groupReq = Group.fetchRequest()
|
|
|
|
groupReq.sortDescriptors = [NSSortDescriptor(key: "title", ascending: true)]
|
|
|
|
groupResultsController = NSFetchedResultsController(fetchRequest: groupReq, managedObjectContext: fervorController.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)
|
|
|
|
groupResultsController.delegate = self
|
|
|
|
try! groupResultsController.performFetch()
|
|
|
|
|
|
|
|
let feedReq = Feed.fetchRequest()
|
|
|
|
feedReq.sortDescriptors = [NSSortDescriptor(key: "title", ascending: true)]
|
|
|
|
feedResultsController = NSFetchedResultsController(fetchRequest: feedReq, managedObjectContext: fervorController.persistentContainer.viewContext, sectionNameKeyPath: nil, cacheName: nil)
|
|
|
|
feedResultsController.delegate = self
|
|
|
|
try! feedResultsController.performFetch()
|
|
|
|
}
|
|
|
|
|
2022-01-09 23:01:19 +00:00
|
|
|
override func viewWillAppear(_ animated: Bool) {
|
|
|
|
super.viewWillAppear(animated)
|
|
|
|
|
|
|
|
if let indexPaths = collectionView.indexPathsForSelectedItems {
|
|
|
|
for indexPath in indexPaths {
|
|
|
|
collectionView.deselectItem(at: indexPath, animated: true)
|
|
|
|
}
|
|
|
|
}
|
2022-01-11 19:23:19 +00:00
|
|
|
|
|
|
|
var snapshot = dataSource.snapshot()
|
|
|
|
// reconfigure so that unread counts update
|
|
|
|
snapshot.reconfigureItems(snapshot.itemIdentifiers)
|
|
|
|
dataSource.apply(snapshot)
|
2022-01-09 23:01:19 +00:00
|
|
|
}
|
|
|
|
|
2022-03-13 18:51:29 +00:00
|
|
|
private func createDataSource() -> UICollectionViewDiffableDataSource<Section, ItemListType> {
|
2022-01-09 23:01:19 +00:00
|
|
|
let sectionHeaderCell = UICollectionView.SupplementaryRegistration<UICollectionViewListCell>(elementKind: UICollectionView.elementKindSectionHeader) { supplementaryView, elementKind, indexPath in
|
|
|
|
let section = self.dataSource.sectionIdentifier(for: indexPath.section)!
|
|
|
|
var config = supplementaryView.defaultContentConfiguration()
|
|
|
|
config.text = section.title
|
|
|
|
supplementaryView.contentConfiguration = config
|
|
|
|
}
|
2022-09-10 04:09:46 +00:00
|
|
|
let listCell = UICollectionView.CellRegistration<HomeCollectionViewCell, ItemListType> { [unowned self] cell, indexPath, item in
|
|
|
|
cell.updateUI(item: item, persistentContainer: self.fervorController.persistentContainer)
|
2022-01-09 16:11:52 +00:00
|
|
|
|
|
|
|
cell.accessories = [.disclosureIndicator()]
|
|
|
|
}
|
2022-03-13 18:51:29 +00:00
|
|
|
let dataSource = UICollectionViewDiffableDataSource<Section, ItemListType>(collectionView: collectionView) { collectionView, indexPath, item in
|
2022-01-09 16:11:52 +00:00
|
|
|
return collectionView.dequeueConfiguredReusableCell(using: listCell, for: indexPath, item: item)
|
|
|
|
}
|
2022-01-09 23:01:19 +00:00
|
|
|
dataSource.supplementaryViewProvider = { collectionView, elementKind, indexPath in
|
|
|
|
if elementKind == UICollectionView.elementKindSectionHeader {
|
|
|
|
return collectionView.dequeueConfiguredReusableSupplementary(using: sectionHeaderCell, for: indexPath)
|
|
|
|
} else {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2022-01-09 16:11:52 +00:00
|
|
|
return dataSource
|
2021-12-08 02:58:02 +00:00
|
|
|
}
|
|
|
|
|
2022-01-27 03:37:10 +00:00
|
|
|
private func syncStateChanged(_ newState: FervorController.SyncState) {
|
2022-06-25 22:40:56 +00:00
|
|
|
if case .done = newState {
|
2022-03-08 17:00:08 +00:00
|
|
|
// update unread counts for visible items
|
|
|
|
var snapshot = dataSource.snapshot()
|
|
|
|
snapshot.reconfigureItems(snapshot.itemIdentifiers)
|
|
|
|
dataSource.apply(snapshot, animatingDifferences: false)
|
|
|
|
|
|
|
|
if syncStateView == nil {
|
|
|
|
// no sync state view, nothing further to update
|
|
|
|
return
|
|
|
|
}
|
2022-01-27 03:37:10 +00:00
|
|
|
}
|
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
func updateView(_ syncStateView: SyncStateView, isFirstUpdate: Bool) {
|
2022-01-27 03:37:10 +00:00
|
|
|
switch newState {
|
|
|
|
case .groupsAndFeeds:
|
|
|
|
syncStateView.label.text = "Syncing groups and feeds"
|
|
|
|
case .items:
|
|
|
|
syncStateView.label.text = "Syncing items"
|
|
|
|
case .updateItems(current: let current, total: let total):
|
|
|
|
syncStateView.label.text = "Updating \(current + 1) of \(total) item\(total == 1 ? "" : "s")"
|
2022-06-25 22:40:56 +00:00
|
|
|
case .excerpts(current: let current, total: let total):
|
|
|
|
syncStateView.label.text = "Generating \(current) of \(total) excerpt\(total == 1 ? "" : "s")"
|
|
|
|
case .error(let error):
|
|
|
|
syncStateView.label.text = "Error syncing"
|
|
|
|
syncStateView.subtitleLabel.isHidden = false
|
|
|
|
syncStateView.subtitleLabel.text = error.localizedDescription
|
|
|
|
|
|
|
|
syncStateView.removeAfterDelay(delay: isFirstUpdate ? 2 : 1)
|
|
|
|
|
2022-01-27 03:37:10 +00:00
|
|
|
case .done:
|
|
|
|
syncStateView.label.text = "Done syncing"
|
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
syncStateView.removeAfterDelay(delay: isFirstUpdate ? 2 : 1)
|
2022-01-27 03:37:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let syncStateView = self.syncStateView {
|
2022-06-25 22:40:56 +00:00
|
|
|
updateView(syncStateView, isFirstUpdate: false)
|
2022-01-27 03:37:10 +00:00
|
|
|
} else {
|
|
|
|
let syncStateView = SyncStateView()
|
|
|
|
self.syncStateView = syncStateView
|
|
|
|
syncStateView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
view.addSubview(syncStateView)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
syncStateView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
|
|
syncStateView.trailingAnchor.constraint(equalTo: view.trailingAnchor),
|
|
|
|
syncStateView.bottomAnchor.constraint(equalTo: view.bottomAnchor),
|
|
|
|
])
|
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
updateView(syncStateView, isFirstUpdate: true)
|
|
|
|
|
2022-01-27 03:37:10 +00:00
|
|
|
view.layoutIfNeeded()
|
|
|
|
syncStateView.transform = CGAffineTransform(translationX: 0, y: syncStateView.bounds.height)
|
|
|
|
|
|
|
|
UIView.animate(withDuration: 0.25, delay: 0, options: .curveEaseOut) {
|
|
|
|
syncStateView.transform = .identity
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-13 18:51:29 +00:00
|
|
|
private func itemsViewController(for item: ItemListType) -> ItemsViewController {
|
|
|
|
let vc = ItemsViewController(type: item, fervorController: fervorController)
|
2022-03-08 02:54:41 +00:00
|
|
|
vc.delegate = itemsDelegate
|
|
|
|
return vc
|
|
|
|
}
|
|
|
|
|
2022-03-13 18:51:29 +00:00
|
|
|
func selectItem(_ item: ItemListType) {
|
|
|
|
guard let navigationController = navigationController else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if navigationController.viewControllers.count >= 2,
|
|
|
|
let second = navigationController.viewControllers[1] as? ItemsViewController,
|
|
|
|
second.type == item {
|
|
|
|
while navigationController.viewControllers.count > 2 {
|
|
|
|
navigationController.popViewController(animated: false)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
navigationController.popToRootViewController(animated: false)
|
|
|
|
navigationController.pushViewController(itemsViewController(for: item), animated: false)
|
|
|
|
}
|
2022-03-08 02:54:41 +00:00
|
|
|
}
|
|
|
|
|
2022-01-09 16:11:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extension HomeViewController {
|
|
|
|
enum Section: Hashable {
|
2022-01-09 23:01:19 +00:00
|
|
|
case all
|
2022-01-09 16:11:52 +00:00
|
|
|
case groups
|
|
|
|
case feeds
|
2022-01-09 23:01:19 +00:00
|
|
|
|
|
|
|
var title: String {
|
|
|
|
switch self {
|
|
|
|
case .all:
|
|
|
|
return ""
|
|
|
|
case .groups:
|
|
|
|
return "Groups"
|
|
|
|
case .feeds:
|
|
|
|
return "Feeds"
|
|
|
|
}
|
|
|
|
}
|
2022-01-09 16:11:52 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension HomeViewController: NSFetchedResultsControllerDelegate {
|
|
|
|
func controller(_ controller: NSFetchedResultsController<NSFetchRequestResult>, didChangeContentWith snapshot: NSDiffableDataSourceSnapshotReference) {
|
|
|
|
var snapshot = dataSource.snapshot()
|
|
|
|
if controller == groupResultsController {
|
|
|
|
if snapshot.sectionIdentifiers.contains(.groups) {
|
|
|
|
snapshot.deleteItems(snapshot.itemIdentifiers(inSection: .groups))
|
|
|
|
}
|
|
|
|
snapshot.appendItems(controller.fetchedObjects!.map { .group($0 as! Group) }, toSection: .groups)
|
|
|
|
} else if controller == feedResultsController {
|
|
|
|
if snapshot.sectionIdentifiers.contains(.feeds) {
|
|
|
|
snapshot.deleteItems(snapshot.itemIdentifiers(inSection: .feeds))
|
|
|
|
}
|
|
|
|
snapshot.appendItems(controller.fetchedObjects!.map { .feed($0 as! Feed) }, toSection: .feeds)
|
|
|
|
}
|
|
|
|
dataSource.apply(snapshot, animatingDifferences: false)
|
|
|
|
}
|
2021-12-08 02:58:02 +00:00
|
|
|
}
|
2022-01-09 22:13:30 +00:00
|
|
|
|
|
|
|
extension HomeViewController: UICollectionViewDelegate {
|
|
|
|
func collectionView(_ collectionView: UICollectionView, didSelectItemAt indexPath: IndexPath) {
|
|
|
|
guard let item = dataSource.itemIdentifier(for: indexPath) else {
|
|
|
|
return
|
|
|
|
}
|
2022-03-08 02:54:41 +00:00
|
|
|
show(itemsViewController(for: item), sender: nil)
|
2022-01-11 19:28:04 +00:00
|
|
|
UISelectionFeedbackGenerator().selectionChanged()
|
2022-01-09 22:13:30 +00:00
|
|
|
}
|
2022-01-11 16:37:41 +00:00
|
|
|
|
|
|
|
func collectionView(_ collectionView: UICollectionView, contextMenuConfigurationForItemAt indexPath: IndexPath, point: CGPoint) -> UIContextMenuConfiguration? {
|
|
|
|
guard let item = dataSource.itemIdentifier(for: indexPath) else {
|
|
|
|
return nil
|
|
|
|
}
|
2022-03-08 02:54:41 +00:00
|
|
|
return UIContextMenuConfiguration(identifier: nil, previewProvider: { [unowned self] in
|
|
|
|
return self.itemsViewController(for: item)
|
2022-01-11 16:37:41 +00:00
|
|
|
}, actionProvider: nil)
|
|
|
|
}
|
|
|
|
|
|
|
|
func collectionView(_ collectionView: UICollectionView, willPerformPreviewActionForMenuWith configuration: UIContextMenuConfiguration, animator: UIContextMenuInteractionCommitAnimating) {
|
|
|
|
guard let vc = animator.previewViewController else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
animator.preferredCommitStyle = .pop
|
|
|
|
animator.addCompletion {
|
|
|
|
self.show(vc, sender: nil)
|
|
|
|
}
|
|
|
|
}
|
2022-01-09 22:13:30 +00:00
|
|
|
}
|
2022-01-12 16:36:12 +00:00
|
|
|
|
|
|
|
extension HomeViewController: StretchyMenuInteractionDelegate {
|
|
|
|
func stretchyMenuTitle() -> String? {
|
|
|
|
return "Switch Accounts"
|
|
|
|
}
|
|
|
|
func stretchyMenuItems() -> [StretchyMenuItem] {
|
2022-01-12 18:48:52 +00:00
|
|
|
var items: [StretchyMenuItem] = LocalData.accounts.map { account in
|
|
|
|
var title = account.instanceURL.host!
|
|
|
|
if let port = account.instanceURL.port, port != 80 && port != 443 {
|
|
|
|
title += ":\(port)"
|
|
|
|
}
|
|
|
|
let subtitle = account.id == fervorController.account!.id ? "Currently logged in" : nil
|
2023-01-14 18:41:45 +00:00
|
|
|
let menu = UIMenu(children: [
|
|
|
|
UIAction(title: "Log Out", image: UIImage(systemName: "door.left.hand.open"), handler: { _ in
|
|
|
|
guard let index = LocalData.accounts.firstIndex(where: { $0.id == account.id }) else {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
LocalData.accounts.remove(at: index)
|
|
|
|
NotificationCenter.default.post(name: .logoutAccount, object: account)
|
|
|
|
})
|
|
|
|
])
|
|
|
|
return StretchyMenuItem(title: title, subtitle: subtitle, menu: menu) { [unowned self] in
|
2022-01-12 18:48:52 +00:00
|
|
|
guard account.id != self.fervorController.account!.id else { return }
|
|
|
|
self.delegate?.switchToAccount(account)
|
|
|
|
}
|
|
|
|
}
|
2023-01-14 18:41:45 +00:00
|
|
|
items.append(StretchyMenuItem(title: "Add Account", action: { [unowned self] in
|
2022-01-12 18:48:52 +00:00
|
|
|
let login = LoginViewController()
|
|
|
|
login.delegate = self
|
|
|
|
login.navigationItem.leftBarButtonItem = UIBarButtonItem(systemItem: .cancel, primaryAction: UIAction(handler: { (_) in
|
|
|
|
self.dismiss(animated: true)
|
|
|
|
}), menu: nil)
|
|
|
|
let nav = UINavigationController(rootViewController: login)
|
|
|
|
self.present(nav, animated: true)
|
|
|
|
}))
|
|
|
|
return items
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension HomeViewController: LoginViewControllerDelegate {
|
|
|
|
func didLogin(with controller: FervorController) {
|
|
|
|
self.dismiss(animated: true)
|
|
|
|
(view.window!.windowScene!.delegate as! SceneDelegate).didLogin(with: controller)
|
2022-01-12 16:36:12 +00:00
|
|
|
}
|
|
|
|
}
|
2022-01-27 03:37:10 +00:00
|
|
|
|
|
|
|
private class SyncStateView: UIView {
|
|
|
|
let label = UILabel()
|
2022-06-25 22:40:56 +00:00
|
|
|
let subtitleLabel = UILabel()
|
2022-01-27 03:37:10 +00:00
|
|
|
|
|
|
|
init() {
|
|
|
|
super.init(frame: .zero)
|
|
|
|
|
|
|
|
let blur = UIBlurEffect(style: .regular)
|
|
|
|
let blurView = UIVisualEffectView(effect: blur)
|
|
|
|
blurView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
addSubview(blurView)
|
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
label.font = .preferredFont(forTextStyle: .callout)
|
|
|
|
subtitleLabel.font = .preferredFont(forTextStyle: .caption1)
|
|
|
|
subtitleLabel.isHidden = true
|
|
|
|
subtitleLabel.numberOfLines = 2
|
|
|
|
|
|
|
|
let stack = UIStackView(arrangedSubviews: [
|
|
|
|
label,
|
|
|
|
subtitleLabel,
|
|
|
|
])
|
|
|
|
stack.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
stack.axis = .vertical
|
|
|
|
stack.alignment = .center
|
|
|
|
stack.spacing = 4
|
|
|
|
addSubview(stack)
|
2022-01-27 03:37:10 +00:00
|
|
|
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
blurView.leadingAnchor.constraint(equalTo: leadingAnchor),
|
|
|
|
blurView.trailingAnchor.constraint(equalTo: trailingAnchor),
|
|
|
|
blurView.topAnchor.constraint(equalTo: topAnchor),
|
|
|
|
blurView.bottomAnchor.constraint(equalTo: bottomAnchor),
|
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
stack.centerXAnchor.constraint(equalTo: safeAreaLayoutGuide.centerXAnchor),
|
|
|
|
stack.topAnchor.constraint(greaterThanOrEqualToSystemSpacingBelow: safeAreaLayoutGuide.topAnchor, multiplier: 1),
|
|
|
|
stack.bottomAnchor.constraint(lessThanOrEqualToSystemSpacingBelow: safeAreaLayoutGuide.bottomAnchor, multiplier: 1),
|
|
|
|
stack.centerYAnchor.constraint(equalTo: safeAreaLayoutGuide.centerYAnchor),
|
2022-01-27 03:37:10 +00:00
|
|
|
|
2022-06-25 22:40:56 +00:00
|
|
|
topAnchor.constraint(lessThanOrEqualTo: safeAreaLayoutGuide.bottomAnchor, constant: -50),
|
2022-01-27 03:37:10 +00:00
|
|
|
])
|
|
|
|
|
|
|
|
layer.shadowColor = UIColor.black.cgColor
|
|
|
|
layer.shadowOpacity = 0.1
|
|
|
|
layer.shadowRadius = 10
|
|
|
|
}
|
|
|
|
|
|
|
|
required init?(coder: NSCoder) {
|
|
|
|
fatalError("init(coder:) has not been implemented")
|
|
|
|
}
|
2022-06-25 22:40:56 +00:00
|
|
|
|
|
|
|
func removeAfterDelay(delay: Int) {
|
|
|
|
// can't use UIView.animate's delay b/c it may clash with the appearance animation
|
|
|
|
DispatchQueue.main.asyncAfter(deadline: .now() + .seconds(delay)) {
|
|
|
|
UIView.animate(withDuration: 0.25, delay: 0, options: .curveEaseIn) {
|
|
|
|
self.transform = CGAffineTransform(translationX: 0, y: self.bounds.height)
|
|
|
|
} completion: { _ in
|
|
|
|
self.removeFromSuperview()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-01-27 03:37:10 +00:00
|
|
|
}
|