812 lines
36 KiB
Swift
812 lines
36 KiB
Swift
//
|
|
// MainTabBarController.swift
|
|
// Mastodon
|
|
//
|
|
// Created by Cirno MainasuK on 2021-1-27.
|
|
//
|
|
|
|
import os.log
|
|
import UIKit
|
|
import Combine
|
|
import CoreDataStack
|
|
import SafariServices
|
|
import MastodonAsset
|
|
import MastodonCore
|
|
import MastodonLocalization
|
|
import MastodonUI
|
|
|
|
class MainTabBarController: UITabBarController {
|
|
|
|
let logger = Logger(subsystem: "MainTabBarController", category: "UI")
|
|
|
|
public var disposeBag = Set<AnyCancellable>()
|
|
|
|
weak var context: AppContext!
|
|
weak var coordinator: SceneCoordinator!
|
|
|
|
var authContext: AuthContext?
|
|
|
|
let composeButttonShadowBackgroundContainer = ShadowBackgroundContainer()
|
|
let composeButton: UIButton = {
|
|
let button = UIButton()
|
|
button.setImage(Asset.ObjectsAndTools.squareAndPencil.image.withRenderingMode(.alwaysTemplate), for: .normal)
|
|
button.setBackgroundImage(UIImage.placeholder(color: Asset.Colors.Label.primary.color), for: .normal)
|
|
button.setBackgroundImage(UIImage.placeholder(color: Asset.Colors.Label.primary.color.withAlphaComponent(0.8)), for: .highlighted)
|
|
button.tintColor = Asset.Colors.Label.primaryReverse.color
|
|
button.contentEdgeInsets = UIEdgeInsets(top: 6, left: 12, bottom: 6, right: 12)
|
|
button.layer.masksToBounds = true
|
|
button.layer.cornerCurve = .continuous
|
|
button.layer.cornerRadius = 8
|
|
button.isAccessibilityElement = false
|
|
return button
|
|
}()
|
|
|
|
static let avatarButtonSize = CGSize(width: 25, height: 25)
|
|
let avatarButton = CircleAvatarButton()
|
|
let accountSwitcherChevron = UIImageView(image: .chevronUpChevronDown)
|
|
|
|
@Published var currentTab: Tab = .home
|
|
|
|
enum Tab: Int, CaseIterable {
|
|
case home
|
|
case search
|
|
case compose
|
|
case notifications
|
|
case me
|
|
|
|
var tag: Int {
|
|
return rawValue
|
|
}
|
|
|
|
var title: String {
|
|
switch self {
|
|
case .home: return L10n.Common.Controls.Tabs.home
|
|
case .search: return L10n.Common.Controls.Tabs.searchAndExplore
|
|
case .compose: return L10n.Common.Controls.Actions.compose
|
|
case .notifications: return L10n.Common.Controls.Tabs.notifications
|
|
case .me: return L10n.Common.Controls.Tabs.profile
|
|
}
|
|
}
|
|
|
|
var image: UIImage {
|
|
switch self {
|
|
case .home: return Asset.ObjectsAndTools.house.image.withRenderingMode(.alwaysTemplate)
|
|
case .search: return Asset.ObjectsAndTools.magnifyingglass.image.withRenderingMode(.alwaysTemplate)
|
|
case .compose: return Asset.ObjectsAndTools.squareAndPencil.image.withRenderingMode(.alwaysTemplate)
|
|
case .notifications: return Asset.ObjectsAndTools.bell.image.withRenderingMode(.alwaysTemplate)
|
|
case .me: return UIImage(systemName: "person")!
|
|
}
|
|
}
|
|
|
|
var selectedImage: UIImage {
|
|
switch self {
|
|
case .home: return Asset.ObjectsAndTools.houseFill.image.withRenderingMode(.alwaysTemplate)
|
|
case .search: return Asset.ObjectsAndTools.magnifyingglassFill.image.withRenderingMode(.alwaysTemplate)
|
|
case .compose: return Asset.ObjectsAndTools.squareAndPencil.image.withRenderingMode(.alwaysTemplate)
|
|
case .notifications: return Asset.ObjectsAndTools.bellFill.image.withRenderingMode(.alwaysTemplate)
|
|
case .me: return UIImage(systemName: "person.fill")!
|
|
}
|
|
}
|
|
|
|
var largeImage: UIImage {
|
|
switch self {
|
|
case .home: return Asset.ObjectsAndTools.house.image.withRenderingMode(.alwaysTemplate).resized(size: CGSize(width: 80, height: 80))
|
|
case .search: return Asset.ObjectsAndTools.magnifyingglass.image.withRenderingMode(.alwaysTemplate).resized(size: CGSize(width: 80, height: 80))
|
|
case .compose: return Asset.ObjectsAndTools.squareAndPencil.image.withRenderingMode(.alwaysTemplate).resized(size: CGSize(width: 80, height: 80))
|
|
case .notifications: return Asset.ObjectsAndTools.bell.image.withRenderingMode(.alwaysTemplate).resized(size: CGSize(width: 80, height: 80))
|
|
case .me: return UIImage(systemName: "person", withConfiguration: UIImage.SymbolConfiguration(pointSize: 80))!
|
|
}
|
|
}
|
|
|
|
var sidebarImage: UIImage {
|
|
switch self {
|
|
case .home: return Asset.ObjectsAndTools.house.image.withRenderingMode(.alwaysTemplate)
|
|
case .search: return Asset.ObjectsAndTools.magnifyingglass.image.withRenderingMode(.alwaysTemplate)
|
|
case .compose: return Asset.ObjectsAndTools.squareAndPencil.image.withRenderingMode(.alwaysTemplate)
|
|
case .notifications: return Asset.ObjectsAndTools.bell.image.withRenderingMode(.alwaysTemplate)
|
|
case .me: return UIImage(systemName: "person")!
|
|
}
|
|
}
|
|
|
|
func viewController(context: AppContext, authContext: AuthContext?, coordinator: SceneCoordinator) -> UIViewController {
|
|
guard let authContext = authContext else {
|
|
return UITableViewController()
|
|
}
|
|
|
|
let viewController: UIViewController
|
|
switch self {
|
|
case .home:
|
|
let _viewController = HomeTimelineViewController()
|
|
_viewController.context = context
|
|
_viewController.coordinator = coordinator
|
|
_viewController.viewModel = .init(context: context, authContext: authContext)
|
|
viewController = _viewController
|
|
case .search:
|
|
let _viewController = SearchViewController()
|
|
_viewController.context = context
|
|
_viewController.coordinator = coordinator
|
|
_viewController.viewModel = .init(context: context, authContext: authContext)
|
|
viewController = _viewController
|
|
case .compose:
|
|
viewController = UIViewController()
|
|
case .notifications:
|
|
let _viewController = NotificationViewController()
|
|
_viewController.context = context
|
|
_viewController.coordinator = coordinator
|
|
_viewController.viewModel = .init(context: context, authContext: authContext)
|
|
viewController = _viewController
|
|
case .me:
|
|
let _viewController = ProfileViewController()
|
|
_viewController.context = context
|
|
_viewController.coordinator = coordinator
|
|
_viewController.viewModel = MeProfileViewModel(context: context, authContext: authContext)
|
|
viewController = _viewController
|
|
}
|
|
viewController.title = self.title
|
|
return AdaptiveStatusBarStyleNavigationController(rootViewController: viewController)
|
|
}
|
|
}
|
|
|
|
var _viewControllers: [UIViewController] = []
|
|
|
|
private(set) var isReadyForWizardAvatarButton = false
|
|
|
|
// output
|
|
var avatarURLObserver: AnyCancellable?
|
|
@Published var avatarURL: URL?
|
|
|
|
init(
|
|
context: AppContext,
|
|
coordinator: SceneCoordinator,
|
|
authContext: AuthContext?
|
|
) {
|
|
self.context = context
|
|
self.coordinator = coordinator
|
|
self.authContext = authContext
|
|
super.init(nibName: nil, bundle: nil)
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
}
|
|
|
|
extension MainTabBarController {
|
|
|
|
open override var childForStatusBarStyle: UIViewController? {
|
|
return selectedViewController
|
|
}
|
|
|
|
override func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
|
|
delegate = self
|
|
|
|
view.backgroundColor = ThemeService.shared.currentTheme.value.systemBackgroundColor
|
|
ThemeService.shared.currentTheme
|
|
.receive(on: RunLoop.main)
|
|
.sink { [weak self] theme in
|
|
guard let self = self else { return }
|
|
self.view.backgroundColor = theme.tabBarBackgroundColor
|
|
}
|
|
.store(in: &disposeBag)
|
|
|
|
// seealso: `ThemeService.apply(theme:)`
|
|
let tabs = Tab.allCases
|
|
let viewControllers: [UIViewController] = tabs.map { tab in
|
|
let viewController = tab.viewController(context: context, authContext: authContext, coordinator: coordinator)
|
|
viewController.tabBarItem.tag = tab.tag
|
|
viewController.tabBarItem.title = tab.title // needs for acessiblity large content label
|
|
viewController.tabBarItem.image = tab.image.imageWithoutBaseline()
|
|
viewController.tabBarItem.selectedImage = tab.selectedImage.imageWithoutBaseline()
|
|
viewController.tabBarItem.largeContentSizeImage = tab.largeImage.imageWithoutBaseline()
|
|
viewController.tabBarItem.accessibilityLabel = tab.title
|
|
viewController.tabBarItem.imageInsets = UIEdgeInsets(top: 6, left: 0, bottom: -6, right: 0)
|
|
return viewController
|
|
}
|
|
_viewControllers = viewControllers
|
|
setViewControllers(viewControllers, animated: false)
|
|
selectedIndex = 0
|
|
|
|
context.apiService.error
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] error in
|
|
guard let self = self, let coordinator = self.coordinator else { return }
|
|
switch error {
|
|
case .implicit:
|
|
break
|
|
case .explicit:
|
|
let alertController = UIAlertController(for: error, title: nil, preferredStyle: .alert)
|
|
let okAction = UIAlertAction(title: "OK", style: .default, handler: nil)
|
|
alertController.addAction(okAction)
|
|
_ = coordinator.present(
|
|
scene: .alertController(alertController: alertController),
|
|
from: nil,
|
|
transition: .alertController(animated: true, completion: nil)
|
|
)
|
|
}
|
|
}
|
|
.store(in: &disposeBag)
|
|
|
|
// handle post failure
|
|
// FIXME: refacotr
|
|
// context.statusPublishService
|
|
// .latestPublishingComposeViewModel
|
|
// .receive(on: DispatchQueue.main)
|
|
// .sink { [weak self] composeViewModel in
|
|
// guard let self = self else { return }
|
|
// guard let composeViewModel = composeViewModel else { return }
|
|
// guard let currentState = composeViewModel.publishStateMachine.currentState else { return }
|
|
// guard currentState is ComposeViewModel.PublishState.Fail else { return }
|
|
//
|
|
// let alertController = UIAlertController(title: L10n.Common.Alerts.PublishPostFailure.title, message: L10n.Common.Alerts.PublishPostFailure.message, preferredStyle: .alert)
|
|
// let discardAction = UIAlertAction(title: L10n.Common.Controls.Actions.discard, style: .destructive) { [weak self, weak composeViewModel] _ in
|
|
// guard let self = self else { return }
|
|
// guard let composeViewModel = composeViewModel else { return }
|
|
// self.context.statusPublishService.remove(composeViewModel: composeViewModel)
|
|
// }
|
|
// alertController.addAction(discardAction)
|
|
// let retryAction = UIAlertAction(title: L10n.Common.Controls.Actions.tryAgain, style: .default) { [weak composeViewModel] _ in
|
|
// guard let composeViewModel = composeViewModel else { return }
|
|
// composeViewModel.publishStateMachine.enter(ComposeViewModel.PublishState.Publishing.self)
|
|
// }
|
|
// alertController.addAction(retryAction)
|
|
// self.present(alertController, animated: true, completion: nil)
|
|
// }
|
|
// .store(in: &disposeBag)
|
|
|
|
// handle push notification.
|
|
// toggle entry when finish fetch latest notification
|
|
Publishers.CombineLatest(
|
|
context.notificationService.unreadNotificationCountDidUpdate,
|
|
$currentTab
|
|
)
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] authentication, currentTab in
|
|
guard let self = self else { return }
|
|
guard let notificationViewController = self.notificationViewController else { return }
|
|
|
|
let authentication = self.authContext?.mastodonAuthenticationBox.userAuthorization
|
|
let hasUnreadPushNotification: Bool = authentication.flatMap { authentication in
|
|
let count = UserDefaults.shared.getNotificationCountWithAccessToken(accessToken: authentication.accessToken)
|
|
return count > 0
|
|
} ?? false
|
|
|
|
let image: UIImage = {
|
|
if currentTab == .notifications {
|
|
return hasUnreadPushNotification ? Asset.ObjectsAndTools.bellBadgeFill.image.withRenderingMode(.alwaysTemplate) : Asset.ObjectsAndTools.bellFill.image.withRenderingMode(.alwaysTemplate)
|
|
} else {
|
|
return hasUnreadPushNotification ? Asset.ObjectsAndTools.bellBadge.image.withRenderingMode(.alwaysTemplate) : Asset.ObjectsAndTools.bell.image.withRenderingMode(.alwaysTemplate)
|
|
}
|
|
}()
|
|
notificationViewController.tabBarItem.image = image.imageWithoutBaseline()
|
|
notificationViewController.navigationController?.tabBarItem.image = image.imageWithoutBaseline()
|
|
}
|
|
.store(in: &disposeBag)
|
|
|
|
layoutComposeButton()
|
|
layoutAvatarButton()
|
|
|
|
$avatarURL
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] avatarURL in
|
|
guard let self = self else { return }
|
|
self.avatarButton.avatarImageView.setImage(
|
|
url: avatarURL,
|
|
placeholder: .placeholder(color: .systemFill),
|
|
scaleToSize: MainTabBarController.avatarButtonSize
|
|
)
|
|
}
|
|
.store(in: &disposeBag)
|
|
|
|
if let user = authContext?.mastodonAuthenticationBox.authenticationRecord.object(in: context.managedObjectContext)?.user {
|
|
self.avatarURLObserver = user.publisher(for: \.avatar)
|
|
.sink { [weak self, weak user] _ in
|
|
guard let self = self else { return }
|
|
guard let user = user else { return }
|
|
guard user.managedObjectContext != nil else { return }
|
|
self.avatarURL = user.avatarImageURL()
|
|
}
|
|
|
|
// a11y
|
|
let _profileTabItem = self.tabBar.items?.first { item in item.tag == Tab.me.tag }
|
|
guard let profileTabItem = _profileTabItem else { return }
|
|
profileTabItem.accessibilityHint = L10n.Scene.AccountList.tabBarHint(user.displayNameWithFallback)
|
|
|
|
context.authenticationService.updateActiveUserAccountPublisher
|
|
.sink { [weak self] in
|
|
self?.updateUserAccount()
|
|
}
|
|
.store(in: &disposeBag)
|
|
} else {
|
|
self.avatarURLObserver = nil
|
|
}
|
|
|
|
let tabBarLongPressGestureRecognizer = UILongPressGestureRecognizer()
|
|
tabBarLongPressGestureRecognizer.addTarget(self, action: #selector(MainTabBarController.tabBarLongPressGestureRecognizerHandler(_:)))
|
|
tabBar.addGestureRecognizer(tabBarLongPressGestureRecognizer)
|
|
|
|
// todo: reconsider the "double tap to change account" feature -> https://github.com/mastodon/mastodon-ios/issues/628
|
|
// let tabBarDoubleTapGestureRecognizer = UITapGestureRecognizer()
|
|
// tabBarDoubleTapGestureRecognizer.numberOfTapsRequired = 2
|
|
// tabBarDoubleTapGestureRecognizer.addTarget(self, action: #selector(MainTabBarController.tabBarDoubleTapGestureRecognizerHandler(_:)))
|
|
// tabBarDoubleTapGestureRecognizer.delaysTouchesEnded = false
|
|
// tabBar.addGestureRecognizer(tabBarDoubleTapGestureRecognizer)
|
|
|
|
self.isReadyForWizardAvatarButton = authContext != nil
|
|
|
|
$currentTab
|
|
.receive(on: DispatchQueue.main)
|
|
.sink { [weak self] tab in
|
|
guard let self = self else { return }
|
|
self.updateAvatarButtonAppearance()
|
|
}
|
|
.store(in: &disposeBag)
|
|
|
|
updateTabBarDisplay()
|
|
|
|
composeButton.addTarget(self, action: #selector(MainTabBarController.composeButtonDidPressed(_:)), for: .touchUpInside)
|
|
|
|
#if DEBUG
|
|
// Debug Register viewController
|
|
// Task { @MainActor in
|
|
// let _homeTimelineViewController = viewControllers
|
|
// .compactMap { $0 as? UINavigationController }
|
|
// .compactMap { $0.topViewController }
|
|
// .compactMap { $0 as? HomeTimelineViewController }
|
|
// .first
|
|
// try await _homeTimelineViewController?.showRegisterController()
|
|
// } // end Task
|
|
#endif
|
|
}
|
|
|
|
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
updateTabBarDisplay()
|
|
updateComposeButtonAppearance()
|
|
updateAvatarButtonAppearance()
|
|
}
|
|
|
|
}
|
|
|
|
extension MainTabBarController {
|
|
|
|
@objc private func composeButtonDidPressed(_ sender: Any) {
|
|
logger.log(level: .debug, "\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public)")
|
|
guard let authContext = self.authContext else { return }
|
|
let composeViewModel = ComposeViewModel(
|
|
context: context,
|
|
authContext: authContext,
|
|
destination: .topLevel
|
|
)
|
|
_ = coordinator.present(scene: .compose(viewModel: composeViewModel), from: nil, transition: .modal(animated: true, completion: nil))
|
|
}
|
|
|
|
private func touchedTab(by sender: UIGestureRecognizer) -> Tab? {
|
|
var _tab: Tab?
|
|
let location = sender.location(in: tabBar)
|
|
for item in tabBar.items ?? [] {
|
|
guard let tab = Tab(rawValue: item.tag) else { continue }
|
|
guard let view = item.value(forKey: "view") as? UIView else { continue }
|
|
guard view.frame.contains(location) else { continue}
|
|
|
|
_tab = tab
|
|
break
|
|
}
|
|
|
|
return _tab
|
|
}
|
|
|
|
@objc private func tabBarDoubleTapGestureRecognizerHandler(_ sender: UITapGestureRecognizer) {
|
|
guard sender.state == .ended else { return }
|
|
guard let tab = touchedTab(by: sender) else { return }
|
|
logger.debug("\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): double tap \(tab.title) tab")
|
|
|
|
switch tab {
|
|
case .me:
|
|
guard let authContext = authContext else { return }
|
|
assert(Thread.isMainThread)
|
|
|
|
guard let nextAccount = context.nextAccount(in: authContext) else { return }
|
|
|
|
Task { @MainActor in
|
|
let isActive = try await context.authenticationService.activeMastodonUser(domain: nextAccount.domain, userID: nextAccount.userID)
|
|
guard isActive else { return }
|
|
self.coordinator.setup()
|
|
}
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
|
|
@objc private func tabBarLongPressGestureRecognizerHandler(_ sender: UILongPressGestureRecognizer) {
|
|
guard sender.state == .began else { return }
|
|
guard let tab = touchedTab(by: sender) else { return }
|
|
logger.debug("\((#file as NSString).lastPathComponent, privacy: .public)[\(#line, privacy: .public)], \(#function, privacy: .public): long press \(tab.title) tab")
|
|
|
|
switch tab {
|
|
case .me:
|
|
guard let authContext = self.authContext else { return }
|
|
let accountListViewModel = AccountListViewModel(context: context, authContext: authContext)
|
|
_ = coordinator.present(scene: .accountList(viewModel: accountListViewModel), from: self, transition: .panModal)
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
extension MainTabBarController {
|
|
|
|
private func updateTabBarDisplay() {
|
|
switch traitCollection.horizontalSizeClass {
|
|
case .compact:
|
|
tabBar.isHidden = false
|
|
composeButttonShadowBackgroundContainer.isHidden = false
|
|
default:
|
|
tabBar.isHidden = true
|
|
composeButttonShadowBackgroundContainer.isHidden = true
|
|
}
|
|
}
|
|
|
|
private func layoutComposeButton() {
|
|
guard composeButton.superview == nil else { return }
|
|
|
|
let _composeTabItem = self.tabBar.items?.first { item in item.tag == Tab.compose.tag }
|
|
guard let composeTabItem = _composeTabItem else { return }
|
|
guard let view = composeTabItem.value(forKey: "view") as? UIView else {
|
|
return
|
|
}
|
|
|
|
let _anchorImageView = view.subviews.first { subview in subview is UIImageView } as? UIImageView
|
|
guard let anchorImageView = _anchorImageView else {
|
|
assertionFailure()
|
|
return
|
|
}
|
|
anchorImageView.alpha = 0
|
|
|
|
composeButttonShadowBackgroundContainer.translatesAutoresizingMaskIntoConstraints = false
|
|
self.view.addSubview(composeButttonShadowBackgroundContainer) // add to tabBar will crash on iPad when size class changing
|
|
NSLayoutConstraint.activate([
|
|
composeButttonShadowBackgroundContainer.centerXAnchor.constraint(equalTo: anchorImageView.centerXAnchor),
|
|
composeButttonShadowBackgroundContainer.centerYAnchor.constraint(equalTo: anchorImageView.centerYAnchor),
|
|
])
|
|
composeButttonShadowBackgroundContainer.cornerRadius = composeButton.layer.cornerRadius
|
|
|
|
composeButton.translatesAutoresizingMaskIntoConstraints = false
|
|
composeButttonShadowBackgroundContainer.addSubview(composeButton)
|
|
composeButton.pinToParent()
|
|
composeButton.setContentHuggingPriority(.required - 1, for: .horizontal)
|
|
composeButton.setContentHuggingPriority(.required - 1, for: .vertical)
|
|
}
|
|
|
|
private func updateComposeButtonAppearance() {
|
|
composeButton.setBackgroundImage(UIImage.placeholder(color: Asset.Colors.Label.primary.color), for: .normal)
|
|
composeButton.setBackgroundImage(UIImage.placeholder(color: Asset.Colors.Label.primary.color.withAlphaComponent(0.8)), for: .highlighted)
|
|
}
|
|
|
|
private func layoutAvatarButton() {
|
|
guard avatarButton.superview == nil else { return }
|
|
|
|
let _profileTabItem = self.tabBar.items?.first { item in item.tag == Tab.me.tag }
|
|
guard let profileTabItem = _profileTabItem else { return }
|
|
guard let view = profileTabItem.value(forKey: "view") as? UIView else {
|
|
return
|
|
}
|
|
|
|
let _anchorImageView = view.subviews.first { subview in subview is UIImageView } as? UIImageView
|
|
guard let anchorImageView = _anchorImageView else {
|
|
assertionFailure()
|
|
return
|
|
}
|
|
anchorImageView.alpha = 0
|
|
|
|
accountSwitcherChevron.translatesAutoresizingMaskIntoConstraints = false
|
|
view.addSubview(accountSwitcherChevron)
|
|
|
|
self.avatarButton.translatesAutoresizingMaskIntoConstraints = false
|
|
view.addSubview(self.avatarButton)
|
|
NSLayoutConstraint.activate([
|
|
self.avatarButton.centerXAnchor.constraint(equalTo: anchorImageView.centerXAnchor, constant: -16),
|
|
self.avatarButton.centerYAnchor.constraint(equalTo: anchorImageView.centerYAnchor),
|
|
self.avatarButton.widthAnchor.constraint(equalToConstant: MainTabBarController.avatarButtonSize.width).priority(.required - 1),
|
|
self.avatarButton.heightAnchor.constraint(equalToConstant: MainTabBarController.avatarButtonSize.height).priority(.required - 1),
|
|
accountSwitcherChevron.widthAnchor.constraint(equalToConstant: 10),
|
|
accountSwitcherChevron.heightAnchor.constraint(equalToConstant: 18),
|
|
accountSwitcherChevron.leadingAnchor.constraint(equalTo: avatarButton.trailingAnchor, constant: 8),
|
|
accountSwitcherChevron.centerYAnchor.constraint(equalTo: avatarButton.centerYAnchor)
|
|
])
|
|
self.avatarButton.setContentHuggingPriority(.required - 1, for: .horizontal)
|
|
self.avatarButton.setContentHuggingPriority(.required - 1, for: .vertical)
|
|
self.avatarButton.isUserInteractionEnabled = false
|
|
}
|
|
|
|
private func updateAvatarButtonAppearance() {
|
|
accountSwitcherChevron.tintColor = currentTab == .me ? .label : .secondaryLabel
|
|
avatarButton.borderColor = currentTab == .me ? .label : .systemFill
|
|
avatarButton.setNeedsLayout()
|
|
}
|
|
|
|
private func updateUserAccount() {
|
|
guard let authContext = authContext else { return }
|
|
|
|
Task { @MainActor in
|
|
let profileResponse = try await context.apiService.authenticatedUserInfo(
|
|
authenticationBox: authContext.mastodonAuthenticationBox
|
|
)
|
|
|
|
if let user = authContext.mastodonAuthenticationBox.authenticationRecord.object(
|
|
in: context.managedObjectContext
|
|
)?.user {
|
|
user.update(
|
|
property: .init(
|
|
entity: profileResponse.value,
|
|
domain: authContext.mastodonAuthenticationBox.domain
|
|
)
|
|
)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
extension MainTabBarController {
|
|
|
|
var notificationViewController: NotificationViewController? {
|
|
return viewController(of: NotificationViewController.self)
|
|
}
|
|
|
|
var searchViewController: SearchViewController? {
|
|
return viewController(of: SearchViewController.self)
|
|
}
|
|
|
|
}
|
|
|
|
// MARK: - UITabBarControllerDelegate
|
|
extension MainTabBarController: UITabBarControllerDelegate {
|
|
func tabBarController(_ tabBarController: UITabBarController, shouldSelect viewController: UIViewController) -> Bool {
|
|
if let tab = Tab(rawValue: viewController.tabBarItem.tag), tab == .compose {
|
|
composeButtonDidPressed(tabBarController)
|
|
return false
|
|
}
|
|
|
|
// Assert index is as same as the tab rawValue. This check needs to be done `shouldSelect`
|
|
// because the nav controller has already popped in `didSelect`.
|
|
if currentTab.rawValue == tabBarController.selectedIndex,
|
|
let navigationController = viewController as? UINavigationController,
|
|
navigationController.viewControllers.count == 1,
|
|
let scrollViewContainer = navigationController.topViewController as? ScrollViewContainer {
|
|
scrollViewContainer.scrollToTop(animated: true)
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
func tabBarController(_ tabBarController: UITabBarController, didSelect viewController: UIViewController) {
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: select %s", ((#file as NSString).lastPathComponent), #line, #function, viewController.debugDescription)
|
|
if let tab = Tab(rawValue: viewController.tabBarItem.tag) {
|
|
currentTab = tab
|
|
}
|
|
}
|
|
}
|
|
|
|
// MARK: - WizardViewControllerDelegate
|
|
extension MainTabBarController: WizardViewControllerDelegate {
|
|
func readyToLayoutItem(_ wizardViewController: WizardViewController, item: WizardViewController.Item) -> Bool {
|
|
switch item {
|
|
case .multipleAccountSwitch:
|
|
return isReadyForWizardAvatarButton
|
|
}
|
|
}
|
|
|
|
func layoutSpotlight(_ wizardViewController: WizardViewController, item: WizardViewController.Item) -> UIBezierPath {
|
|
switch item {
|
|
case .multipleAccountSwitch:
|
|
guard let avatarButtonFrameInView = avatarButtonFrameInWizardView(wizardView: wizardViewController.view) else {
|
|
return UIBezierPath()
|
|
}
|
|
return UIBezierPath(ovalIn: avatarButtonFrameInView)
|
|
}
|
|
}
|
|
|
|
func layoutWizardCard(_ wizardViewController: WizardViewController, item: WizardViewController.Item) {
|
|
switch item {
|
|
case .multipleAccountSwitch:
|
|
guard let avatarButtonFrameInView = avatarButtonFrameInWizardView(wizardView: wizardViewController.view) else {
|
|
return
|
|
}
|
|
let anchorView = UIView()
|
|
anchorView.frame = avatarButtonFrameInView
|
|
wizardViewController.backgroundView.addSubview(anchorView)
|
|
|
|
let wizardCardView = WizardCardView()
|
|
wizardCardView.arrowRectCorner = view.traitCollection.layoutDirection == .leftToRight ? .bottomRight : .bottomLeft
|
|
wizardCardView.titleLabel.text = item.title
|
|
wizardCardView.descriptionLabel.text = item.description
|
|
|
|
wizardCardView.translatesAutoresizingMaskIntoConstraints = false
|
|
wizardViewController.backgroundView.addSubview(wizardCardView)
|
|
NSLayoutConstraint.activate([
|
|
anchorView.topAnchor.constraint(equalTo: wizardCardView.bottomAnchor, constant: 13), // 13pt spacing
|
|
wizardCardView.trailingAnchor.constraint(equalTo: anchorView.centerXAnchor),
|
|
wizardCardView.widthAnchor.constraint(equalTo: wizardViewController.view.widthAnchor, multiplier: 2.0/3.0).priority(.required - 1),
|
|
])
|
|
wizardCardView.setContentHuggingPriority(.defaultLow, for: .vertical)
|
|
}
|
|
}
|
|
|
|
private func avatarButtonFrameInWizardView(wizardView: UIView) -> CGRect? {
|
|
guard let superview = avatarButton.superview else {
|
|
assertionFailure()
|
|
return nil
|
|
}
|
|
return superview.convert(avatarButton.frame, to: wizardView)
|
|
}
|
|
}
|
|
|
|
// HIG: keyboard UX
|
|
// https://developer.apple.com/design/human-interface-guidelines/macos/user-interaction/keyboard/
|
|
extension MainTabBarController {
|
|
|
|
var switchToTabKeyCommands: [UIKeyCommand] {
|
|
var commands: [UIKeyCommand] = []
|
|
let tabs: [Tab] = [
|
|
.home,
|
|
.search,
|
|
.notifications,
|
|
.me
|
|
]
|
|
for (i, tab) in tabs.enumerated() {
|
|
let title = L10n.Common.Controls.Keyboard.Common.switchToTab(tab.title)
|
|
let input = String(i + 1)
|
|
let command = UIKeyCommand(
|
|
title: title,
|
|
image: nil,
|
|
action: #selector(MainTabBarController.switchToTabKeyCommandHandler(_:)),
|
|
input: input,
|
|
modifierFlags: .command,
|
|
propertyList: tab.rawValue,
|
|
alternates: [],
|
|
discoverabilityTitle: nil,
|
|
attributes: [],
|
|
state: .off
|
|
)
|
|
commands.append(command)
|
|
}
|
|
return commands
|
|
}
|
|
|
|
var showFavoritesKeyCommand: UIKeyCommand {
|
|
UIKeyCommand(
|
|
title: L10n.Common.Controls.Keyboard.Common.showFavorites,
|
|
image: nil,
|
|
action: #selector(MainTabBarController.showFavoritesKeyCommandHandler(_:)),
|
|
input: "f",
|
|
modifierFlags: .command,
|
|
propertyList: nil,
|
|
alternates: [],
|
|
discoverabilityTitle: nil,
|
|
attributes: [],
|
|
state: .off
|
|
)
|
|
}
|
|
|
|
var openSettingsKeyCommand: UIKeyCommand {
|
|
UIKeyCommand(
|
|
title: L10n.Common.Controls.Keyboard.Common.openSettings,
|
|
image: nil,
|
|
action: #selector(MainTabBarController.openSettingsKeyCommandHandler(_:)),
|
|
input: ",",
|
|
modifierFlags: .command,
|
|
propertyList: nil,
|
|
alternates: [],
|
|
discoverabilityTitle: nil,
|
|
attributes: [],
|
|
state: .off
|
|
)
|
|
}
|
|
|
|
var composeNewPostKeyCommand: UIKeyCommand {
|
|
UIKeyCommand(
|
|
title: L10n.Common.Controls.Keyboard.Common.composeNewPost,
|
|
image: nil,
|
|
action: #selector(MainTabBarController.composeNewPostKeyCommandHandler(_:)),
|
|
input: "n",
|
|
modifierFlags: .command,
|
|
propertyList: nil,
|
|
alternates: [],
|
|
discoverabilityTitle: nil,
|
|
attributes: [],
|
|
state: .off
|
|
)
|
|
}
|
|
|
|
override var keyCommands: [UIKeyCommand]? {
|
|
guard let topMost = self.topMost else {
|
|
return []
|
|
}
|
|
|
|
var commands: [UIKeyCommand] = []
|
|
|
|
if topMost.isModal {
|
|
|
|
} else {
|
|
// switch tabs
|
|
commands.append(contentsOf: switchToTabKeyCommands)
|
|
|
|
// show compose
|
|
if !(self.topMost is ComposeViewController) {
|
|
commands.append(composeNewPostKeyCommand)
|
|
}
|
|
|
|
// show favorites
|
|
if !(self.topMost is FavoriteViewController) {
|
|
commands.append(showFavoritesKeyCommand)
|
|
}
|
|
|
|
// open settings
|
|
if context.settingService.currentSetting.value != nil {
|
|
commands.append(openSettingsKeyCommand)
|
|
}
|
|
}
|
|
|
|
return commands
|
|
}
|
|
|
|
@objc private func switchToTabKeyCommandHandler(_ sender: UIKeyCommand) {
|
|
guard let rawValue = sender.propertyList as? Int,
|
|
let tab = Tab(rawValue: rawValue) else { return }
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s: %s", ((#file as NSString).lastPathComponent), #line, #function, tab.title)
|
|
|
|
guard let index = Tab.allCases.firstIndex(of: tab) else { return }
|
|
let previousTab = Tab(rawValue: selectedIndex)
|
|
selectedIndex = index
|
|
if let tab = Tab(rawValue: index) {
|
|
currentTab = tab
|
|
}
|
|
|
|
if let previousTab = previousTab {
|
|
switch (tab, previousTab) {
|
|
case (.home, .home):
|
|
guard let navigationController = topMost?.navigationController else { return }
|
|
if navigationController.viewControllers.count > 1 {
|
|
// pop to top when previous tab position already is home
|
|
navigationController.popToRootViewController(animated: true)
|
|
} else if let homeTimelineViewController = topMost as? HomeTimelineViewController {
|
|
// trigger scrollToTop if topMost is home timeline
|
|
homeTimelineViewController.scrollToTop(animated: true)
|
|
}
|
|
default:
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
@objc private func showFavoritesKeyCommandHandler(_ sender: UIKeyCommand) {
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
guard let authContext = self.authContext else { return }
|
|
let favoriteViewModel = FavoriteViewModel(context: context, authContext: authContext)
|
|
_ = coordinator.present(scene: .favorite(viewModel: favoriteViewModel), from: nil, transition: .show)
|
|
}
|
|
|
|
@objc private func openSettingsKeyCommandHandler(_ sender: UIKeyCommand) {
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
guard let authContext = self.authContext else { return }
|
|
guard let setting = context.settingService.currentSetting.value else { return }
|
|
let settingsViewModel = SettingsViewModel(context: context, authContext: authContext, setting: setting)
|
|
_ = coordinator.present(scene: .settings(viewModel: settingsViewModel), from: nil, transition: .modal(animated: true, completion: nil))
|
|
}
|
|
|
|
@objc private func composeNewPostKeyCommandHandler(_ sender: UIKeyCommand) {
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
guard let authContext = self.authContext else { return }
|
|
let composeViewModel = ComposeViewModel(
|
|
context: context,
|
|
authContext: authContext,
|
|
destination: .topLevel
|
|
)
|
|
_ = coordinator.present(scene: .compose(viewModel: composeViewModel), from: nil, transition: .modal(animated: true, completion: nil))
|
|
}
|
|
|
|
}
|