364 lines
14 KiB
Swift
364 lines
14 KiB
Swift
//
|
|
// MainTabBarViewController.swift
|
|
// Tusker
|
|
//
|
|
// Created by Shadowfacts on 8/21/18.
|
|
// Copyright © 2018 Shadowfacts. All rights reserved.
|
|
//
|
|
|
|
import UIKit
|
|
|
|
class MainTabBarViewController: UITabBarController, UITabBarControllerDelegate {
|
|
|
|
weak var mastodonController: MastodonController!
|
|
|
|
private var composePlaceholder: UIViewController!
|
|
|
|
private var fastAccountSwitcher: FastAccountSwitcherViewController!
|
|
private var fastSwitcherIndicator: FastAccountSwitcherIndicatorView!
|
|
private var fastSwitcherConstraints: [NSLayoutConstraint] = []
|
|
|
|
@available(iOS, obsoleted: 16.0)
|
|
private var draftToPresentOnAppear: Draft?
|
|
|
|
var selectedTab: Tab {
|
|
return Tab(rawValue: selectedIndex)!
|
|
}
|
|
|
|
override var supportedInterfaceOrientations: UIInterfaceOrientationMask {
|
|
if UIDevice.current.userInterfaceIdiom == .phone {
|
|
return .portrait
|
|
} else {
|
|
return .all
|
|
}
|
|
}
|
|
|
|
init(mastodonController: MastodonController) {
|
|
self.mastodonController = mastodonController
|
|
|
|
super.init(nibName: nil, bundle: nil)
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
override func viewDidLoad() {
|
|
super.viewDidLoad()
|
|
|
|
stateRestorationLogger.info("MainTabBarViewController: viewDidLoad, selectedIndex=\(self.selectedIndex, privacy: .public)")
|
|
|
|
self.delegate = self
|
|
|
|
composePlaceholder = UIViewController()
|
|
composePlaceholder.title = "Compose"
|
|
composePlaceholder.tabBarItem.image = UIImage(systemName: "pencil")
|
|
|
|
viewControllers = [
|
|
embedInNavigationController(Tab.timelines.createViewController(mastodonController)),
|
|
embedInNavigationController(Tab.notifications.createViewController(mastodonController)),
|
|
composePlaceholder,
|
|
embedInNavigationController(Tab.explore.createViewController(mastodonController)),
|
|
embedInNavigationController(Tab.myProfile.createViewController(mastodonController)),
|
|
]
|
|
|
|
fastAccountSwitcher = FastAccountSwitcherViewController()
|
|
fastAccountSwitcher.delegate = self
|
|
fastAccountSwitcher.view.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
tabBar.addGestureRecognizer(fastAccountSwitcher.createSwitcherGesture())
|
|
let tapRecognizer = UITapGestureRecognizer(target: self, action: #selector(tabBarTapped))
|
|
tapRecognizer.cancelsTouchesInView = false
|
|
tabBar.addGestureRecognizer(tapRecognizer)
|
|
|
|
if findMyProfileTabBarButton() != nil {
|
|
fastSwitcherIndicator = FastAccountSwitcherIndicatorView()
|
|
fastSwitcherIndicator.translatesAutoresizingMaskIntoConstraints = false
|
|
view.addSubview(fastSwitcherIndicator)
|
|
}
|
|
|
|
tabBar.isSpringLoaded = true
|
|
}
|
|
|
|
override func viewWillAppear(_ animated: Bool) {
|
|
super.viewWillAppear(animated)
|
|
stateRestorationLogger.info("MainTabBarViewController: viewWillAppear, selectedIndex=\(self.selectedIndex, privacy: .public)")
|
|
}
|
|
|
|
override func viewDidAppear(_ animated: Bool) {
|
|
super.viewDidAppear(animated)
|
|
stateRestorationLogger.info("MainTabBarViewController: viewDidAppear, selectedIndex=\(self.selectedIndex, privacy: .public)")
|
|
|
|
if let draftToPresentOnAppear {
|
|
self.draftToPresentOnAppear = nil
|
|
compose(editing: draftToPresentOnAppear, animated: true)
|
|
}
|
|
}
|
|
|
|
override func viewDidLayoutSubviews() {
|
|
super.viewDidLayoutSubviews()
|
|
|
|
// i hate that we have to do this so often :S
|
|
// but doing it only in viewWillAppear makes it not appear initially
|
|
// doing it in viewWillAppear inside a DispatchQueue.main.async works initially but then it disappears when long-pressed
|
|
repositionFastSwitcherIndicator()
|
|
}
|
|
|
|
override func traitCollectionDidChange(_ previousTraitCollection: UITraitCollection?) {
|
|
super.traitCollectionDidChange(previousTraitCollection)
|
|
|
|
repositionFastSwitcherIndicator()
|
|
}
|
|
|
|
func select(tab: Tab) {
|
|
if tab == .compose {
|
|
compose(editing: nil)
|
|
} else {
|
|
// when switching tabs, dismiss the currently presented VC
|
|
// otherwise the selected tab changes behind the presented VC
|
|
if presentedViewController != nil {
|
|
dismiss(animated: true) {
|
|
self.selectedIndex = tab.rawValue
|
|
}
|
|
} else {
|
|
stateRestorationLogger.info("MainTabBarViewController: selecting \(String(describing: tab), privacy: .public)")
|
|
selectedIndex = tab.rawValue
|
|
}
|
|
}
|
|
}
|
|
|
|
override func show(_ vc: UIViewController, sender: Any?) {
|
|
if let nav = selectedViewController as? UINavigationController {
|
|
nav.pushViewController(vc, animated: true)
|
|
} else {
|
|
present(vc, animated: true)
|
|
}
|
|
}
|
|
|
|
private func repositionFastSwitcherIndicator() {
|
|
guard let myProfileButton = findMyProfileTabBarButton() else {
|
|
return
|
|
}
|
|
NSLayoutConstraint.deactivate(fastSwitcherConstraints)
|
|
// using interfaceOrientation isn't ideal, but UITabBar buttons may lay out horizontally even in the compact size class
|
|
if traitCollection.horizontalSizeClass == .compact && interfaceOrientation.isPortrait {
|
|
fastSwitcherConstraints = [
|
|
fastSwitcherIndicator.centerYAnchor.constraint(equalTo: myProfileButton.centerYAnchor, constant: -4),
|
|
// tab bar button image width is 30
|
|
fastSwitcherIndicator.leftAnchor.constraint(equalTo: myProfileButton.centerXAnchor, constant: 15 + 2),
|
|
]
|
|
} else {
|
|
fastSwitcherConstraints = [
|
|
fastSwitcherIndicator.centerYAnchor.constraint(equalTo: myProfileButton.centerYAnchor),
|
|
fastSwitcherIndicator.trailingAnchor.constraint(equalTo: myProfileButton.trailingAnchor),
|
|
]
|
|
}
|
|
NSLayoutConstraint.activate(fastSwitcherConstraints)
|
|
}
|
|
|
|
private func findMyProfileTabBarButton() -> UIView? {
|
|
let tabBarButtons = tabBar.subviews.filter { String(describing: type(of: $0)).lowercased().contains("button") }
|
|
// sanity check that there is 1 button per VC
|
|
guard tabBarButtons.count == viewControllers!.count,
|
|
let myProfileButton = tabBarButtons.last else {
|
|
return nil
|
|
}
|
|
return myProfileButton
|
|
}
|
|
|
|
@objc private func tabBarTapped(_ recognizer: UITapGestureRecognizer) {
|
|
fastAccountSwitcher.hide()
|
|
}
|
|
|
|
@objc func handleComposeKeyCommand() {
|
|
compose(editing: nil)
|
|
}
|
|
|
|
func embedInNavigationController(_ vc: UIViewController) -> UINavigationController {
|
|
if let vc = vc as? UINavigationController {
|
|
return vc
|
|
} else {
|
|
let nav = EnhancedNavigationViewController(rootViewController: vc)
|
|
// nav.useBrowserStyleNavigation = true
|
|
return nav
|
|
}
|
|
}
|
|
|
|
func tabBarController(_ tabBarController: UITabBarController, shouldSelect viewController: UIViewController) -> Bool {
|
|
if viewController == composePlaceholder {
|
|
compose(editing: nil)
|
|
return false
|
|
}
|
|
if viewController == viewControllers![selectedIndex],
|
|
let nav = viewController as? UINavigationController,
|
|
nav.viewControllers.count == 1,
|
|
let scrollableVC = nav.viewControllers.first as? TabBarScrollableViewController {
|
|
scrollableVC.tabBarScrollToTop()
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
func setViewController(_ viewController: UIViewController, forTab tab: Tab) {
|
|
viewControllers![tab.rawValue] = viewController
|
|
}
|
|
|
|
func viewController(for tab: Tab) -> UIViewController {
|
|
return viewControllers![tab.rawValue]
|
|
}
|
|
}
|
|
|
|
extension MainTabBarViewController {
|
|
enum Tab: Int, Hashable, CaseIterable {
|
|
case timelines
|
|
case notifications
|
|
case compose
|
|
case explore
|
|
case myProfile
|
|
|
|
func createViewController(_ mastodonController: MastodonController) -> UIViewController {
|
|
switch self {
|
|
case .timelines:
|
|
return TimelinesPageViewController(mastodonController: mastodonController)
|
|
case .notifications:
|
|
return NotificationsPageViewController(mastodonController: mastodonController)
|
|
case .compose:
|
|
return ComposeHostingController(mastodonController: mastodonController)
|
|
case .explore:
|
|
return ExploreViewController(mastodonController: mastodonController)
|
|
case .myProfile:
|
|
return MyProfileViewController(mastodonController: mastodonController)
|
|
}
|
|
}
|
|
}
|
|
|
|
func getTabController(tab: Tab) -> UIViewController? {
|
|
if tab == .compose {
|
|
return nil
|
|
} else {
|
|
// viewWControllers array is setup in viewDidLoad
|
|
loadViewIfNeeded()
|
|
return viewControllers![tab.rawValue]
|
|
}
|
|
}
|
|
}
|
|
|
|
extension MainTabBarViewController: FastAccountSwitcherViewControllerDelegate {
|
|
func fastAccountSwitcherAddToViewHierarchy(_ fastAccountSwitcher: FastAccountSwitcherViewController) {
|
|
view.addSubview(fastAccountSwitcher.view)
|
|
NSLayoutConstraint.activate([
|
|
fastAccountSwitcher.accountsStack.bottomAnchor.constraint(equalTo: fastAccountSwitcher.view.bottomAnchor),
|
|
|
|
fastAccountSwitcher.view.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
fastAccountSwitcher.view.trailingAnchor.constraint(equalTo: view.trailingAnchor),
|
|
fastAccountSwitcher.view.topAnchor.constraint(equalTo: view.topAnchor),
|
|
fastAccountSwitcher.view.bottomAnchor.constraint(equalTo: tabBar.topAnchor),
|
|
])
|
|
}
|
|
|
|
func fastAccountSwitcher(_ fastAccountSwitcher: FastAccountSwitcherViewController, triggerZoneContains point: CGPoint) -> Bool {
|
|
guard let myProfileButton = findMyProfileTabBarButton() else {
|
|
return false
|
|
}
|
|
let locationInButton = myProfileButton.convert(point, from: tabBar)
|
|
return myProfileButton.bounds.contains(locationInButton)
|
|
}
|
|
}
|
|
|
|
extension MainTabBarViewController: TuskerNavigationDelegate {
|
|
var apiController: MastodonController! { mastodonController }
|
|
}
|
|
|
|
extension MainTabBarViewController: StateRestorableViewController {
|
|
func stateRestorationActivity() -> NSUserActivity? {
|
|
var activity: NSUserActivity?
|
|
if let presentedNav = presentedViewController as? UINavigationController,
|
|
let compose = presentedNav.viewControllers.first as? ComposeHostingController {
|
|
activity = UserActivityManager.editDraftActivity(id: compose.draft.id, accountID: compose.draft.accountID)
|
|
} else if let vc = (selectedViewController as! UINavigationController).topViewController as? StateRestorableViewController {
|
|
activity = vc.stateRestorationActivity()
|
|
}
|
|
if activity == nil {
|
|
stateRestorationLogger.fault("MainTabBarViewController: Unable to create state restoration activity, couldn't find StateRestorableViewController")
|
|
}
|
|
return activity
|
|
}
|
|
}
|
|
|
|
extension MainTabBarViewController: TuskerRootViewController {
|
|
func select(route: TuskerRoute, animated: Bool) {
|
|
switch route {
|
|
case .timelines:
|
|
select(tab: .timelines)
|
|
case .notifications:
|
|
select(tab: .notifications)
|
|
case .myProfile:
|
|
select(tab: .myProfile)
|
|
case .explore:
|
|
select(tab: .explore)
|
|
case .bookmarks:
|
|
select(tab: .explore)
|
|
getNavigationController().pushViewController(BookmarksViewController(mastodonController: mastodonController), animated: animated)
|
|
case .list(id: let id):
|
|
select(tab: .explore)
|
|
if let list = mastodonController.getCachedList(id: id) {
|
|
let nav = getNavigationController()
|
|
_ = nav.popToRootViewController(animated: animated)
|
|
nav.pushViewController(ListTimelineViewController(for: list, mastodonController: mastodonController), animated: animated)
|
|
}
|
|
}
|
|
}
|
|
|
|
func getNavigationDelegate() -> TuskerNavigationDelegate? {
|
|
return self
|
|
}
|
|
|
|
func getNavigationController() -> NavigationControllerProtocol {
|
|
return (selectedViewController as! UINavigationController)
|
|
}
|
|
|
|
func performSearch(query: String) {
|
|
guard let exploreNavController = getTabController(tab: .explore) as? UINavigationController,
|
|
let exploreController = exploreNavController.viewControllers.first as? ExploreViewController else {
|
|
return
|
|
}
|
|
|
|
select(tab: .explore)
|
|
exploreNavController.popToRootViewController(animated: false)
|
|
|
|
// setting searchController.isActive directly doesn't work until the view has loaded/appeared for the first time
|
|
if exploreController.isViewLoaded {
|
|
exploreController.searchController.isActive = true
|
|
} else {
|
|
exploreController.searchControllerStatusOnAppearance = true
|
|
// we still need to load the view so that we can setup the search query
|
|
exploreController.loadViewIfNeeded()
|
|
}
|
|
|
|
exploreController.searchController.searchBar.text = query
|
|
exploreController.resultsController.performSearch(query: query)
|
|
}
|
|
|
|
func presentPreferences(completion: (() -> Void)?) {
|
|
present(PreferencesNavigationController(mastodonController: mastodonController), animated: true, completion: completion)
|
|
}
|
|
|
|
func handleStatusBarTapped(xPosition: CGFloat) -> StatusBarTapActionResult {
|
|
guard presentedViewController == nil else {
|
|
return .stop
|
|
}
|
|
guard let vc = viewController(for: selectedTab) as? StatusBarTappableViewController else {
|
|
return .continue
|
|
}
|
|
return vc.handleStatusBarTapped(xPosition: xPosition)
|
|
}
|
|
}
|
|
|
|
extension MainTabBarViewController: BackgroundableViewController {
|
|
func sceneDidEnterBackground() {
|
|
if let selectedVC = selectedViewController as? BackgroundableViewController {
|
|
selectedVC.sceneDidEnterBackground()
|
|
}
|
|
}
|
|
}
|