2021-02-22 09:20:44 +01:00
|
|
|
//
|
|
|
|
// MastodonServerRulesViewController.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by MainasuK Cirno on 2021-2-22.
|
|
|
|
//
|
|
|
|
|
|
|
|
import os.log
|
|
|
|
import UIKit
|
2021-02-26 11:27:47 +01:00
|
|
|
import Combine
|
2021-03-01 09:27:24 +01:00
|
|
|
import MastodonSDK
|
2021-02-22 09:20:44 +01:00
|
|
|
|
2021-02-26 09:43:59 +01:00
|
|
|
final class MastodonServerRulesViewController: UIViewController, NeedsDependency {
|
2021-02-22 09:20:44 +01:00
|
|
|
|
2021-02-26 11:27:47 +01:00
|
|
|
var disposeBag = Set<AnyCancellable>()
|
|
|
|
|
2021-02-22 09:20:44 +01:00
|
|
|
weak var context: AppContext! { willSet { precondition(!isViewLoaded) } }
|
|
|
|
weak var coordinator: SceneCoordinator! { willSet { precondition(!isViewLoaded) } }
|
|
|
|
|
|
|
|
var viewModel: MastodonServerRulesViewModel!
|
|
|
|
|
|
|
|
let largeTitleLabel: UILabel = {
|
|
|
|
let label = UILabel()
|
2021-02-26 09:43:59 +01:00
|
|
|
label.font = UIFontMetrics(forTextStyle: .largeTitle).scaledFont(for: .systemFont(ofSize: 34, weight: .bold))
|
2021-02-22 09:20:44 +01:00
|
|
|
label.textColor = .label
|
|
|
|
label.text = L10n.Scene.ServerRules.title
|
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
private(set) lazy var subtitleLabel: UILabel = {
|
|
|
|
let label = UILabel()
|
|
|
|
label.font = UIFontMetrics(forTextStyle: .title1).scaledFont(for: UIFont.systemFont(ofSize: 20))
|
|
|
|
label.textColor = .secondaryLabel
|
|
|
|
label.text = L10n.Scene.ServerRules.subtitle(viewModel.domain)
|
|
|
|
label.numberOfLines = 0
|
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
let rulesLabel: UILabel = {
|
|
|
|
let label = UILabel()
|
|
|
|
label.font = .preferredFont(forTextStyle: .body)
|
2021-02-23 08:16:55 +01:00
|
|
|
label.textColor = .black
|
2021-02-22 09:20:44 +01:00
|
|
|
label.text = "Rules"
|
|
|
|
label.numberOfLines = 0
|
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
let bottonContainerView: UIView = {
|
|
|
|
let view = UIView()
|
|
|
|
view.backgroundColor = Asset.Colors.Background.onboardingBackground.color
|
|
|
|
return view
|
|
|
|
}()
|
|
|
|
|
|
|
|
private(set) lazy var bottomPromptLabel: UILabel = {
|
|
|
|
let label = UILabel()
|
|
|
|
label.font = .preferredFont(forTextStyle: .body)
|
|
|
|
label.textColor = .label
|
|
|
|
label.text = L10n.Scene.ServerRules.prompt(viewModel.domain)
|
|
|
|
label.numberOfLines = 0
|
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
2021-02-26 11:27:47 +01:00
|
|
|
let confirmButton: PrimaryActionButton = {
|
2021-02-26 09:43:59 +01:00
|
|
|
let button = PrimaryActionButton()
|
2021-02-22 09:20:44 +01:00
|
|
|
button.titleLabel?.font = .preferredFont(forTextStyle: .headline)
|
2021-02-26 09:43:59 +01:00
|
|
|
button.setTitleColor(.white, for: .normal)
|
2021-02-22 09:20:44 +01:00
|
|
|
button.setTitle(L10n.Scene.ServerRules.Button.confirm, for: .normal)
|
|
|
|
return button
|
|
|
|
}()
|
|
|
|
|
2021-02-26 09:43:59 +01:00
|
|
|
let scrollView: UIScrollView = {
|
|
|
|
let scrollView = UIScrollView()
|
|
|
|
scrollView.alwaysBounceVertical = true
|
2021-02-26 12:52:58 +01:00
|
|
|
scrollView.showsVerticalScrollIndicator = false
|
2021-02-26 09:43:59 +01:00
|
|
|
return scrollView
|
|
|
|
}()
|
2021-02-26 11:27:47 +01:00
|
|
|
|
|
|
|
deinit {
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
|
|
|
}
|
2021-02-22 09:20:44 +01:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonServerRulesViewController {
|
|
|
|
|
|
|
|
override func viewDidLoad() {
|
|
|
|
super.viewDidLoad()
|
|
|
|
|
2021-02-26 12:52:58 +01:00
|
|
|
setupOnboardingAppearance()
|
|
|
|
defer { setupNavigationBarBackgroundView() }
|
2021-02-22 09:20:44 +01:00
|
|
|
|
|
|
|
bottonContainerView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
view.addSubview(bottonContainerView)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
view.bottomAnchor.constraint(equalTo: bottonContainerView.bottomAnchor),
|
|
|
|
bottonContainerView.leadingAnchor.constraint(equalTo: view.leadingAnchor),
|
|
|
|
bottonContainerView.trailingAnchor.constraint(equalTo: view.trailingAnchor),
|
|
|
|
])
|
|
|
|
bottonContainerView.preservesSuperviewLayoutMargins = true
|
2021-02-26 12:52:58 +01:00
|
|
|
defer {
|
|
|
|
view.bringSubviewToFront(bottonContainerView)
|
|
|
|
}
|
2021-02-22 09:20:44 +01:00
|
|
|
|
|
|
|
confirmButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
bottonContainerView.addSubview(confirmButton)
|
|
|
|
NSLayoutConstraint.activate([
|
2021-02-26 09:43:59 +01:00
|
|
|
bottonContainerView.layoutMarginsGuide.bottomAnchor.constraint(equalTo: confirmButton.bottomAnchor, constant: MastodonServerRulesViewController.viewBottomPaddingHeight),
|
2021-02-26 12:52:58 +01:00
|
|
|
confirmButton.leadingAnchor.constraint(equalTo: bottonContainerView.readableContentGuide.leadingAnchor, constant: MastodonServerRulesViewController.actionButtonMargin),
|
|
|
|
bottonContainerView.readableContentGuide.trailingAnchor.constraint(equalTo: confirmButton.trailingAnchor, constant: MastodonServerRulesViewController.actionButtonMargin),
|
|
|
|
confirmButton.heightAnchor.constraint(equalToConstant: MastodonServerRulesViewController.actionButtonHeight).priority(.defaultHigh),
|
2021-02-22 09:20:44 +01:00
|
|
|
])
|
|
|
|
|
|
|
|
bottomPromptLabel.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
bottonContainerView.addSubview(bottomPromptLabel)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
bottomPromptLabel.topAnchor.constraint(equalTo: bottonContainerView.topAnchor, constant: 20),
|
|
|
|
bottomPromptLabel.leadingAnchor.constraint(equalTo: bottonContainerView.readableContentGuide.leadingAnchor),
|
|
|
|
bottomPromptLabel.trailingAnchor.constraint(equalTo: bottonContainerView.readableContentGuide.trailingAnchor),
|
|
|
|
confirmButton.topAnchor.constraint(equalTo: bottomPromptLabel.bottomAnchor, constant: 20),
|
|
|
|
])
|
|
|
|
|
|
|
|
scrollView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
view.addSubview(scrollView)
|
|
|
|
NSLayoutConstraint.activate([
|
2021-02-26 12:52:58 +01:00
|
|
|
scrollView.frameLayoutGuide.topAnchor.constraint(equalTo: view.topAnchor),
|
2021-02-22 09:20:44 +01:00
|
|
|
scrollView.frameLayoutGuide.leadingAnchor.constraint(equalTo: view.readableContentGuide.leadingAnchor),
|
2021-02-26 12:52:58 +01:00
|
|
|
scrollView.frameLayoutGuide.trailingAnchor.constraint(equalTo: view.readableContentGuide.trailingAnchor),
|
|
|
|
scrollView.frameLayoutGuide.bottomAnchor.constraint(equalTo: view.bottomAnchor),
|
2021-02-22 09:20:44 +01:00
|
|
|
scrollView.frameLayoutGuide.widthAnchor.constraint(equalTo: scrollView.contentLayoutGuide.widthAnchor),
|
|
|
|
])
|
|
|
|
|
|
|
|
let stackView = UIStackView()
|
|
|
|
stackView.axis = .vertical
|
|
|
|
stackView.distribution = .fill
|
|
|
|
stackView.spacing = 10
|
2021-02-26 09:43:59 +01:00
|
|
|
stackView.layoutMargins = UIEdgeInsets(top: 20, left: 0, bottom: 20, right: 0)
|
2021-02-22 09:20:44 +01:00
|
|
|
stackView.addArrangedSubview(largeTitleLabel)
|
|
|
|
stackView.addArrangedSubview(subtitleLabel)
|
|
|
|
stackView.addArrangedSubview(rulesLabel)
|
|
|
|
|
|
|
|
stackView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
scrollView.addSubview(stackView)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
stackView.topAnchor.constraint(equalTo: scrollView.contentLayoutGuide.topAnchor),
|
|
|
|
stackView.leadingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.leadingAnchor),
|
|
|
|
stackView.trailingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.trailingAnchor),
|
|
|
|
scrollView.contentLayoutGuide.bottomAnchor.constraint(equalTo: stackView.bottomAnchor),
|
|
|
|
])
|
|
|
|
|
|
|
|
rulesLabel.attributedText = viewModel.rulesAttributedString
|
|
|
|
confirmButton.addTarget(self, action: #selector(MastodonServerRulesViewController.confirmButtonPressed(_:)), for: .touchUpInside)
|
2021-02-26 11:27:47 +01:00
|
|
|
|
|
|
|
viewModel.isRegistering
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] isRegistering in
|
|
|
|
guard let self = self else { return }
|
|
|
|
isRegistering ? self.confirmButton.showLoading() : self.confirmButton.stopLoading()
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-26 12:52:58 +01:00
|
|
|
|
|
|
|
viewModel.error
|
|
|
|
.compactMap { $0 }
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] error in
|
|
|
|
guard let self = self else { return }
|
2021-03-01 09:27:24 +01:00
|
|
|
guard let error = error as? Mastodon.API.Error else { return }
|
2021-02-26 12:52:58 +01:00
|
|
|
let alertController = UIAlertController(for: error, title: "Sign Up Failure", preferredStyle: .alert)
|
|
|
|
let okAction = UIAlertAction(title: L10n.Common.Controls.Actions.ok, style: .default, handler: nil)
|
|
|
|
alertController.addAction(okAction)
|
|
|
|
self.coordinator.present(
|
|
|
|
scene: .alertController(alertController: alertController),
|
|
|
|
from: nil,
|
|
|
|
transition: .alertController(animated: true, completion: nil)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
}
|
|
|
|
|
|
|
|
override func viewDidLayoutSubviews() {
|
|
|
|
super.viewDidLayoutSubviews()
|
|
|
|
updateScrollViewContentInset()
|
2021-02-22 09:20:44 +01:00
|
|
|
}
|
|
|
|
|
2021-02-26 12:52:58 +01:00
|
|
|
override func viewSafeAreaInsetsDidChange() {
|
|
|
|
super.viewSafeAreaInsetsDidChange()
|
|
|
|
updateScrollViewContentInset()
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonServerRulesViewController {
|
|
|
|
func updateScrollViewContentInset() {
|
|
|
|
view.layoutIfNeeded()
|
|
|
|
scrollView.contentInset.bottom = bottonContainerView.frame.height
|
|
|
|
scrollView.verticalScrollIndicatorInsets.bottom = bottonContainerView.frame.height
|
|
|
|
}
|
2021-02-22 09:20:44 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonServerRulesViewController {
|
|
|
|
@objc private func confirmButtonPressed(_ sender: UIButton) {
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", ((#file as NSString).lastPathComponent), #line, #function)
|
2021-02-26 11:27:47 +01:00
|
|
|
|
|
|
|
let email = viewModel.registerQuery.email
|
|
|
|
|
|
|
|
context.apiService.accountRegister(
|
|
|
|
domain: viewModel.domain,
|
|
|
|
query: viewModel.registerQuery,
|
|
|
|
authorization: viewModel.applicationAuthorization
|
|
|
|
)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] completion in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.viewModel.isRegistering.value = false
|
|
|
|
switch completion {
|
|
|
|
case .failure(let error):
|
|
|
|
self.viewModel.error.send(error)
|
|
|
|
case .finished:
|
|
|
|
break
|
|
|
|
}
|
|
|
|
} receiveValue: { [weak self] response in
|
|
|
|
guard let self = self else { return }
|
|
|
|
let userToken = response.value
|
|
|
|
let viewModel = MastodonConfirmEmailViewModel(context: self.context, email: email, authenticateInfo: self.viewModel.authenticateInfo, userToken: userToken)
|
|
|
|
self.coordinator.present(scene: .mastodonConfirmEmail(viewModel: viewModel), from: self, transition: .show)
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-22 09:20:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-26 09:43:59 +01:00
|
|
|
// MARK: - OnboardingViewControllerAppearance
|
|
|
|
extension MastodonServerRulesViewController: OnboardingViewControllerAppearance { }
|
|
|
|
|
2021-02-22 09:20:44 +01:00
|
|
|
#if canImport(SwiftUI) && DEBUG
|
|
|
|
import SwiftUI
|
|
|
|
|
|
|
|
struct ServerRulesViewController_Previews: PreviewProvider {
|
|
|
|
|
|
|
|
static var previews: some View {
|
|
|
|
UIViewControllerPreview {
|
|
|
|
let viewController = MastodonServerRulesViewController()
|
|
|
|
return viewController
|
|
|
|
}
|
|
|
|
.previewLayout(.fixed(width: 375, height: 800))
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|