You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
session-ios/Session/Shared/SessionHostingViewControlle...

249 lines
9.2 KiB
Swift

// Copyright © 2023 Rangeproof Pty Ltd. All rights reserved.
import SwiftUI
import SessionUIKit
import SessionUtilitiesKit
public class HostWrapper: ObservableObject {
public weak var controller: UIViewController?
}
public enum NavigationItemPosition {
case left
case right
}
public class SessionHostingViewController<Content>: UIHostingController<ModifiedContent<Content, _EnvironmentKeyWritingModifier<HostWrapper?>>>, ThemedNavigation where Content : View {
public override var preferredStatusBarStyle: UIStatusBarStyle {
return ThemeManager.currentTheme.statusBarStyle
}
public var navigationBackground: ThemeValue? { customizedNavigationBackground }
private let customizedNavigationBackground: ThemeValue?
private let shouldHideNavigationBar: Bool
lazy var navBarTitleLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.veryLargeFontSize)
result.themeTextColor = .textPrimary
result.textAlignment = .center
result.alpha = 1
return result
}()
lazy var crossfadeLabel: UILabel = {
let result = UILabel()
result.font = .boldSystemFont(ofSize: Values.veryLargeFontSize)
result.themeTextColor = .textPrimary
result.textAlignment = .center
result.alpha = 0
return result
}()
public init(rootView:Content, customizedNavigationBackground: ThemeValue? = nil, shouldHideNavigationBar: Bool = false) {
self.customizedNavigationBackground = customizedNavigationBackground
self.shouldHideNavigationBar = shouldHideNavigationBar
let container = HostWrapper()
let modified = rootView.environmentObject(container) as! ModifiedContent<Content, _EnvironmentKeyWritingModifier<HostWrapper?>>
super.init(rootView: modified)
container.controller = self
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
public override func viewDidLoad() {
super.viewDidLoad()
navigationItem.backButtonTitle = ""
view.themeBackgroundColor = .backgroundPrimary
ThemeManager.applyNavigationStylingIfNeeded(to: self)
setNeedsStatusBarAppearanceUpdate()
}
public override func viewWillAppear(_ animated: Bool) {
if shouldHideNavigationBar {
self.navigationController?.setNavigationBarHidden(true, animated: animated)
}
super.viewWillAppear(animated)
}
public override func viewWillDisappear(_ animated: Bool) {
if shouldHideNavigationBar {
self.navigationController?.setNavigationBarHidden(false, animated: animated)
}
super.viewWillDisappear(animated)
}
internal func setNavBarTitle(_ title: String, customFontSize: CGFloat? = nil) {
let container = UIView()
navBarTitleLabel.text = title
crossfadeLabel.text = title
if let customFontSize = customFontSize {
navBarTitleLabel.font = .boldSystemFont(ofSize: customFontSize)
crossfadeLabel.font = .boldSystemFont(ofSize: customFontSize)
}
container.addSubview(navBarTitleLabel)
container.addSubview(crossfadeLabel)
navBarTitleLabel.pin(to: container)
crossfadeLabel.pin(to: container)
navigationItem.titleView = container
}
internal func setUpNavBarSessionHeading() {
let headingImageView = UIImageView(
image: UIImage(named: "SessionHeading")?
.withRenderingMode(.alwaysTemplate)
)
headingImageView.themeTintColor = .textPrimary
headingImageView.contentMode = .scaleAspectFit
headingImageView.set(.width, to: 150)
headingImageView.set(.height, to: Values.mediumFontSize)
navigationItem.titleView = headingImageView
}
internal func setUpNavBarSessionIcon() {
let logoImageView = UIImageView()
logoImageView.image = #imageLiteral(resourceName: "SessionGreen32")
logoImageView.contentMode = .scaleAspectFit
logoImageView.set(.width, to: 32)
logoImageView.set(.height, to: 32)
navigationItem.titleView = logoImageView
}
internal func setUpClearDataBackButton(flow: Onboarding.Flow) {
if #available(iOS 16.0, *) {
navigationItem.backAction = UIAction() { [weak self] action in
switch flow {
case .register:
self?.clearDataForAccountCreation()
case .recover:
self?.clearDataForLoadAccount()
}
}
} else {
let action: Selector = {
switch flow {
case .register:
return #selector(clearDataForAccountCreation)
case .recover:
return #selector(clearDataForLoadAccount)
}
}()
let clearDataBackButton = UIBarButtonItem(
image: UIImage(
systemName: "chevron.backward",
withConfiguration: UIImage.SymbolConfiguration(textStyle: .headline, scale: .large)
),
style: .plain,
target: self,
action: action
)
clearDataBackButton.imageInsets = .init(top: 0, leading: -8, bottom: 0, trailing: 8)
clearDataBackButton.themeTintColor = .textPrimary
navigationItem.leftBarButtonItem = clearDataBackButton
}
}
@objc private func clearDataForAccountCreation() {
let modal: ConfirmationModal = ConfirmationModal(
targetView: self.view,
info: ConfirmationModal.Info(
title: "warning".localized(),
body: .text(
"onboardingBackAccountCreation"
.put(key: "app_name", value: Constants.app_name)
.localized()
),
confirmTitle: "quitButton".localized(),
confirmAccessibility: Accessibility(identifier: "Quit"),
confirmStyle: .danger,
cancelStyle: .textPrimary,
onConfirm: { [weak self] confirmationModal in
self?.deleteAllLocalData()
}
)
)
self.present(modal, animated: true)
}
@objc private func clearDataForLoadAccount() {
let modal: ConfirmationModal = ConfirmationModal(
targetView: self.view,
info: ConfirmationModal.Info(
title: "warning".localized(),
body: .text(
"onboardingBackLoadAccount"
.put(key: "app_name", value: Constants.app_name)
.localized()
),
confirmTitle: "quitButton".localized(),
confirmStyle: .danger,
cancelStyle: .textPrimary,
onConfirm: { [weak self] confirmationModal in
self?.deleteAllLocalData()
}
)
)
self.present(modal, animated: true)
}
private func deleteAllLocalData(using dependencies: Dependencies = Dependencies()) {
/// Stop and cancel all current jobs (don't want to inadvertantly have a job store data after it's table has already been cleared)
///
/// **Note:** This is file as long as this process kills the app, if it doesn't then we need an alternate mechanism to flag that
/// the `JobRunner` is allowed to start it's queues again
JobRunner.stopAndClearPendingJobs(using: dependencies)
// Clear the app badge and notifications
AppEnvironment.shared.notificationPresenter.clearAllNotifications()
UIApplication.shared.applicationIconBadgeNumber = 0
// Clear out the user defaults
UserDefaults.removeAll()
// Remove the cached key so it gets re-cached on next access
dependencies.caches.mutate(cache: .general) {
$0.encodedPublicKey = nil
$0.recentReactionTimestamps = []
}
// Stop any pollers
(UIApplication.shared.delegate as? AppDelegate)?.stopPollers()
// Call through to the SessionApp's "resetAppData" which will wipe out logs, database and
// profile storage
let wasUnlinked: Bool = UserDefaults.standard[.wasUnlinked]
SessionApp.resetAppData(using: dependencies) {
// Resetting the data clears the old user defaults. We need to restore the unlink default.
UserDefaults.standard[.wasUnlinked] = wasUnlinked
}
}
internal func setUpDismissingButton(on postion: NavigationItemPosition) {
let closeButton = UIBarButtonItem(image: #imageLiteral(resourceName: "X"), style: .plain, target: self, action: #selector(close))
closeButton.themeTintColor = .textPrimary
switch postion {
case .left:
navigationItem.leftBarButtonItem = closeButton
case .right:
navigationItem.rightBarButtonItem = closeButton
}
}
@objc private func close() {
dismiss(animated: true, completion: nil)
}
}