2021-02-05 10:53:00 +01:00
|
|
|
//
|
|
|
|
// MastodonRegisterViewController.swift
|
|
|
|
// Mastodon
|
|
|
|
//
|
|
|
|
// Created by MainasuK Cirno on 2021-2-5.
|
|
|
|
//
|
|
|
|
|
|
|
|
import Combine
|
|
|
|
import MastodonSDK
|
2021-02-20 06:55:06 +01:00
|
|
|
import os.log
|
2021-03-02 11:21:54 +01:00
|
|
|
import PhotosUI
|
2021-02-20 06:55:06 +01:00
|
|
|
import UIKit
|
2021-02-05 10:53:00 +01:00
|
|
|
import UITextField_Shake
|
|
|
|
|
2021-02-25 08:39:48 +01:00
|
|
|
final class MastodonRegisterViewController: UIViewController, NeedsDependency, OnboardingViewControllerAppearance {
|
2021-02-05 10:53:00 +01:00
|
|
|
var disposeBag = Set<AnyCancellable>()
|
|
|
|
|
|
|
|
weak var context: AppContext! { willSet { precondition(!isViewLoaded) } }
|
|
|
|
weak var coordinator: SceneCoordinator! { willSet { precondition(!isViewLoaded) } }
|
|
|
|
|
|
|
|
var viewModel: MastodonRegisterViewModel!
|
2021-02-20 06:55:06 +01:00
|
|
|
|
2021-03-02 10:24:04 +01:00
|
|
|
lazy var imagePicker: PHPickerViewController = {
|
|
|
|
var configuration = PHPickerConfiguration()
|
|
|
|
configuration.filter = .images
|
|
|
|
|
|
|
|
let imagePicker = PHPickerViewController(configuration: configuration)
|
|
|
|
imagePicker.delegate = self
|
|
|
|
return imagePicker
|
|
|
|
}()
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let tapGestureRecognizer = UITapGestureRecognizer.singleTapGestureRecognizer
|
|
|
|
|
2021-02-22 09:20:44 +01:00
|
|
|
let scrollView: UIScrollView = {
|
2021-02-20 06:55:06 +01:00
|
|
|
let scrollview = UIScrollView()
|
|
|
|
scrollview.showsVerticalScrollIndicator = false
|
2021-02-20 12:54:08 +01:00
|
|
|
scrollview.keyboardDismissMode = .interactive
|
2021-02-26 09:43:59 +01:00
|
|
|
scrollview.alwaysBounceVertical = true
|
2021-03-02 11:21:54 +01:00
|
|
|
scrollview.clipsToBounds = false // make content could display over bleeding
|
2021-02-26 09:43:59 +01:00
|
|
|
scrollview.translatesAutoresizingMaskIntoConstraints = false
|
2021-02-20 06:55:06 +01:00
|
|
|
return scrollview
|
|
|
|
}()
|
2021-02-05 10:53:00 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let largeTitleLabel: UILabel = {
|
|
|
|
let label = UILabel()
|
2021-02-20 11:24:23 +01:00
|
|
|
label.font = UIFontMetrics(forTextStyle: .largeTitle).scaledFont(for: UIFont.boldSystemFont(ofSize: 34))
|
2021-02-23 08:16:55 +01:00
|
|
|
label.textColor = .black
|
2021-02-22 09:20:44 +01:00
|
|
|
label.text = L10n.Scene.Register.title
|
2021-02-20 06:55:06 +01:00
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
let photoView: UIView = {
|
|
|
|
let view = UIView()
|
|
|
|
view.backgroundColor = .clear
|
|
|
|
return view
|
|
|
|
}()
|
|
|
|
|
|
|
|
let photoButton: UIButton = {
|
|
|
|
let button = UIButton(type: .custom)
|
|
|
|
let boldFont = UIFont.systemFont(ofSize: 42)
|
|
|
|
let configuration = UIImage.SymbolConfiguration(font: boldFont)
|
|
|
|
let image = UIImage(systemName: "person.fill.viewfinder", withConfiguration: configuration)
|
|
|
|
|
|
|
|
button.setImage(image?.withRenderingMode(UIImage.RenderingMode.alwaysTemplate), for: UIControl.State.normal)
|
|
|
|
button.imageView?.tintColor = Asset.Colors.Icon.photo.color
|
|
|
|
button.backgroundColor = .white
|
|
|
|
button.layer.cornerRadius = 45
|
|
|
|
button.clipsToBounds = true
|
2021-03-02 09:19:20 +01:00
|
|
|
|
|
|
|
button.addTarget(self, action: #selector(MastodonRegisterViewController.avatarButtonPressed(_:)), for: .touchUpInside)
|
2021-02-20 06:55:06 +01:00
|
|
|
return button
|
|
|
|
}()
|
|
|
|
|
2021-02-20 11:24:23 +01:00
|
|
|
let plusIconBackground: UIImageView = {
|
|
|
|
let icon = UIImageView()
|
|
|
|
let boldFont = UIFont.systemFont(ofSize: 24)
|
|
|
|
let configuration = UIImage.SymbolConfiguration(font: boldFont)
|
|
|
|
let image = UIImage(systemName: "plus.circle", withConfiguration: configuration)
|
|
|
|
icon.image = image
|
|
|
|
icon.tintColor = .white
|
|
|
|
return icon
|
|
|
|
}()
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let plusIcon: UIImageView = {
|
|
|
|
let icon = UIImageView()
|
|
|
|
let boldFont = UIFont.systemFont(ofSize: 24)
|
|
|
|
let configuration = UIImage.SymbolConfiguration(font: boldFont)
|
|
|
|
let image = UIImage(systemName: "plus.circle.fill", withConfiguration: configuration)
|
|
|
|
icon.image = image
|
|
|
|
icon.tintColor = Asset.Colors.Icon.plus.color
|
|
|
|
return icon
|
|
|
|
}()
|
|
|
|
|
|
|
|
let domainLabel: UILabel = {
|
2021-02-05 10:53:00 +01:00
|
|
|
let label = UILabel()
|
|
|
|
label.font = .preferredFont(forTextStyle: .headline)
|
2021-02-23 08:16:55 +01:00
|
|
|
label.textColor = .black
|
2021-02-05 10:53:00 +01:00
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
let usernameTextField: UITextField = {
|
|
|
|
let textField = UITextField()
|
2021-02-20 06:55:06 +01:00
|
|
|
|
2021-02-05 10:53:00 +01:00
|
|
|
textField.autocapitalizationType = .none
|
|
|
|
textField.autocorrectionType = .no
|
2021-02-20 06:55:06 +01:00
|
|
|
textField.backgroundColor = .white
|
2021-02-26 11:27:47 +01:00
|
|
|
textField.textColor = Asset.Colors.Label.primary.color
|
2021-02-22 09:20:44 +01:00
|
|
|
textField.attributedPlaceholder = NSAttributedString(string: L10n.Scene.Register.Input.Username.placeholder,
|
2021-02-26 11:27:47 +01:00
|
|
|
attributes: [NSAttributedString.Key.foregroundColor: Asset.Colors.Label.secondary.color,
|
2021-02-20 06:55:06 +01:00
|
|
|
NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .headline)])
|
|
|
|
textField.borderStyle = UITextField.BorderStyle.roundedRect
|
|
|
|
let paddingView = UIView(frame: CGRect(x: 0, y: 0, width: 5, height: textField.frame.height))
|
|
|
|
textField.leftView = paddingView
|
|
|
|
textField.leftViewMode = .always
|
2021-02-05 10:53:00 +01:00
|
|
|
return textField
|
|
|
|
}()
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let usernameIsTakenLabel: UILabel = {
|
2021-02-05 10:53:00 +01:00
|
|
|
let label = UILabel()
|
2021-03-01 09:27:24 +01:00
|
|
|
let color = Asset.Colors.lightDangerRed.color
|
|
|
|
let font = UIFont.preferredFont(forTextStyle: .caption1)
|
|
|
|
let attributeString = NSMutableAttributedString()
|
|
|
|
|
|
|
|
let errorImage = NSTextAttachment()
|
|
|
|
let configuration = UIImage.SymbolConfiguration(font: font)
|
|
|
|
errorImage.image = UIImage(systemName: "xmark.octagon.fill", withConfiguration: configuration)?.withTintColor(color)
|
|
|
|
let errorImageAttachment = NSAttributedString(attachment: errorImage)
|
|
|
|
attributeString.append(errorImageAttachment)
|
|
|
|
|
|
|
|
let errorString = NSAttributedString(string: L10n.Common.Errors.Item.username + " " + L10n.Common.Errors.errTaken, attributes: [NSAttributedString.Key.font: font, NSAttributedString.Key.foregroundColor: color])
|
|
|
|
attributeString.append(errorString)
|
|
|
|
label.attributedText = attributeString
|
|
|
|
|
2021-02-05 10:53:00 +01:00
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let displayNameTextField: UITextField = {
|
|
|
|
let textField = UITextField()
|
|
|
|
textField.autocapitalizationType = .none
|
|
|
|
textField.autocorrectionType = .no
|
|
|
|
textField.backgroundColor = .white
|
2021-02-26 11:27:47 +01:00
|
|
|
textField.textColor = Asset.Colors.Label.primary.color
|
2021-02-22 09:20:44 +01:00
|
|
|
textField.attributedPlaceholder = NSAttributedString(string: L10n.Scene.Register.Input.DisplayName.placeholder,
|
2021-02-26 11:27:47 +01:00
|
|
|
attributes: [NSAttributedString.Key.foregroundColor: Asset.Colors.Label.secondary.color,
|
2021-02-20 06:55:06 +01:00
|
|
|
NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .headline)])
|
|
|
|
textField.borderStyle = UITextField.BorderStyle.roundedRect
|
|
|
|
let paddingView = UIView(frame: CGRect(x: 0, y: 0, width: 5, height: textField.frame.height))
|
|
|
|
textField.leftView = paddingView
|
|
|
|
textField.leftViewMode = .always
|
|
|
|
return textField
|
|
|
|
}()
|
|
|
|
|
2021-02-05 10:53:00 +01:00
|
|
|
let emailTextField: UITextField = {
|
|
|
|
let textField = UITextField()
|
|
|
|
textField.autocapitalizationType = .none
|
|
|
|
textField.autocorrectionType = .no
|
|
|
|
textField.keyboardType = .emailAddress
|
2021-02-20 06:55:06 +01:00
|
|
|
textField.backgroundColor = .white
|
2021-02-26 11:27:47 +01:00
|
|
|
textField.textColor = Asset.Colors.Label.primary.color
|
2021-02-22 09:20:44 +01:00
|
|
|
textField.attributedPlaceholder = NSAttributedString(string: L10n.Scene.Register.Input.Email.placeholder,
|
2021-02-26 11:27:47 +01:00
|
|
|
attributes: [NSAttributedString.Key.foregroundColor: Asset.Colors.Label.secondary.color,
|
2021-02-20 06:55:06 +01:00
|
|
|
NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .headline)])
|
|
|
|
textField.borderStyle = UITextField.BorderStyle.roundedRect
|
|
|
|
let paddingView = UIView(frame: CGRect(x: 0, y: 0, width: 5, height: textField.frame.height))
|
|
|
|
textField.leftView = paddingView
|
|
|
|
textField.leftViewMode = .always
|
2021-02-05 10:53:00 +01:00
|
|
|
return textField
|
|
|
|
}()
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
let passwordCheckLabel: UILabel = {
|
2021-02-05 10:53:00 +01:00
|
|
|
let label = UILabel()
|
2021-02-22 05:51:35 +01:00
|
|
|
label.numberOfLines = 0
|
2021-02-05 10:53:00 +01:00
|
|
|
return label
|
|
|
|
}()
|
|
|
|
|
|
|
|
let passwordTextField: UITextField = {
|
|
|
|
let textField = UITextField()
|
|
|
|
textField.autocapitalizationType = .none
|
|
|
|
textField.autocorrectionType = .no
|
|
|
|
textField.keyboardType = .asciiCapable
|
|
|
|
textField.isSecureTextEntry = true
|
2021-02-20 06:55:06 +01:00
|
|
|
textField.backgroundColor = .white
|
2021-02-26 11:27:47 +01:00
|
|
|
textField.textColor = Asset.Colors.Label.primary.color
|
2021-02-22 09:20:44 +01:00
|
|
|
textField.attributedPlaceholder = NSAttributedString(string: L10n.Scene.Register.Input.Password.placeholder,
|
2021-02-26 11:27:47 +01:00
|
|
|
attributes: [NSAttributedString.Key.foregroundColor: Asset.Colors.Label.secondary.color,
|
2021-02-20 06:55:06 +01:00
|
|
|
NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .headline)])
|
|
|
|
textField.borderStyle = UITextField.BorderStyle.roundedRect
|
|
|
|
let paddingView = UIView(frame: CGRect(x: 0, y: 0, width: 5, height: textField.frame.height))
|
|
|
|
textField.leftView = paddingView
|
|
|
|
textField.leftViewMode = .always
|
2021-02-05 10:53:00 +01:00
|
|
|
return textField
|
|
|
|
}()
|
|
|
|
|
2021-02-26 05:52:37 +01:00
|
|
|
lazy var inviteTextField: UITextField = {
|
|
|
|
let textField = UITextField()
|
|
|
|
textField.autocapitalizationType = .none
|
|
|
|
textField.autocorrectionType = .no
|
|
|
|
textField.backgroundColor = .white
|
2021-02-26 11:27:47 +01:00
|
|
|
textField.textColor = Asset.Colors.Label.primary.color
|
2021-02-26 05:52:37 +01:00
|
|
|
textField.attributedPlaceholder = NSAttributedString(string: L10n.Scene.Register.Input.Invite.registrationUserInviteRequest,
|
2021-02-26 11:27:47 +01:00
|
|
|
attributes: [NSAttributedString.Key.foregroundColor: Asset.Colors.Label.secondary.color,
|
2021-02-26 05:52:37 +01:00
|
|
|
NSAttributedString.Key.font: UIFont.preferredFont(forTextStyle: .headline)])
|
|
|
|
textField.borderStyle = UITextField.BorderStyle.roundedRect
|
|
|
|
let paddingView = UIView(frame: CGRect(x: 0, y: 0, width: 5, height: textField.frame.height))
|
|
|
|
textField.leftView = paddingView
|
|
|
|
textField.leftViewMode = .always
|
|
|
|
return textField
|
|
|
|
}()
|
|
|
|
|
2021-02-26 11:27:47 +01:00
|
|
|
let buttonContainer = UIView()
|
|
|
|
let signUpButton: PrimaryActionButton = {
|
2021-02-26 09:43:59 +01:00
|
|
|
let button = PrimaryActionButton()
|
2021-02-20 06:55:06 +01:00
|
|
|
button.isEnabled = false
|
2021-02-22 09:20:44 +01:00
|
|
|
button.setTitle(L10n.Common.Controls.Actions.continue, for: .normal)
|
2021-02-05 10:53:00 +01:00
|
|
|
return button
|
|
|
|
}()
|
|
|
|
|
2021-02-26 11:27:47 +01:00
|
|
|
deinit {
|
2021-03-02 11:21:54 +01:00
|
|
|
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
|
|
|
}
|
2021-02-05 10:53:00 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonRegisterViewController {
|
|
|
|
override func viewDidLoad() {
|
|
|
|
super.viewDidLoad()
|
|
|
|
|
2021-02-26 09:43:59 +01:00
|
|
|
setupOnboardingAppearance()
|
|
|
|
defer { setupNavigationBarBackgroundView() }
|
2021-02-25 08:39:48 +01:00
|
|
|
|
2021-03-02 11:21:54 +01:00
|
|
|
NSObject.KeyValueObservingPublisher<UIButton, Bool>(object: photoButton, keyPath: \.isHighlighted, options: NSKeyValueObservingOptions.new)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] isHighlighted in
|
|
|
|
guard let self = self else { return }
|
|
|
|
let alpha: CGFloat = isHighlighted ? 0.8 : 1
|
|
|
|
self.plusIcon.alpha = alpha
|
|
|
|
self.plusIconBackground.alpha = alpha
|
|
|
|
self.photoButton.alpha = alpha
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
domainLabel.text = "@" + viewModel.domain + " "
|
|
|
|
domainLabel.sizeToFit()
|
|
|
|
passwordCheckLabel.attributedText = viewModel.attributeStringForPassword()
|
|
|
|
usernameTextField.rightView = domainLabel
|
|
|
|
usernameTextField.rightViewMode = .always
|
|
|
|
usernameTextField.delegate = self
|
|
|
|
displayNameTextField.delegate = self
|
|
|
|
emailTextField.delegate = self
|
|
|
|
passwordTextField.delegate = self
|
2021-02-05 10:53:00 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
// gesture
|
|
|
|
view.addGestureRecognizer(tapGestureRecognizer)
|
2021-02-22 10:48:44 +01:00
|
|
|
tapGestureRecognizer.addTarget(self, action: #selector(tapGestureRecognizerHandler))
|
2021-02-05 10:53:00 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
// stackview
|
|
|
|
let stackView = UIStackView()
|
2021-02-05 10:53:00 +01:00
|
|
|
stackView.axis = .vertical
|
2021-02-20 06:55:06 +01:00
|
|
|
stackView.distribution = .fill
|
|
|
|
stackView.spacing = 40
|
2021-02-22 09:26:50 +01:00
|
|
|
stackView.layoutMargins = UIEdgeInsets(top: 20, left: 0, bottom: 26, right: 0)
|
2021-02-20 06:55:06 +01:00
|
|
|
stackView.isLayoutMarginsRelativeArrangement = true
|
|
|
|
stackView.addArrangedSubview(largeTitleLabel)
|
|
|
|
stackView.addArrangedSubview(photoView)
|
2021-02-05 10:53:00 +01:00
|
|
|
stackView.addArrangedSubview(usernameTextField)
|
2021-03-01 10:45:02 +01:00
|
|
|
stackView.addArrangedSubview(usernameIsTakenLabel)
|
2021-02-20 06:55:06 +01:00
|
|
|
stackView.addArrangedSubview(displayNameTextField)
|
2021-02-05 10:53:00 +01:00
|
|
|
stackView.addArrangedSubview(emailTextField)
|
|
|
|
stackView.addArrangedSubview(passwordTextField)
|
2021-02-20 06:55:06 +01:00
|
|
|
stackView.addArrangedSubview(passwordCheckLabel)
|
2021-03-01 10:45:02 +01:00
|
|
|
if viewModel.approvalRequired {
|
2021-02-26 05:52:37 +01:00
|
|
|
stackView.addArrangedSubview(inviteTextField)
|
|
|
|
}
|
2021-02-22 09:20:44 +01:00
|
|
|
// scrollView
|
|
|
|
view.addSubview(scrollView)
|
2021-02-20 06:55:06 +01:00
|
|
|
NSLayoutConstraint.activate([
|
2021-02-22 09:20:44 +01:00
|
|
|
scrollView.frameLayoutGuide.topAnchor.constraint(equalTo: view.layoutMarginsGuide.topAnchor),
|
|
|
|
scrollView.frameLayoutGuide.leadingAnchor.constraint(equalTo: view.readableContentGuide.leadingAnchor),
|
|
|
|
view.readableContentGuide.trailingAnchor.constraint(equalTo: scrollView.frameLayoutGuide.trailingAnchor),
|
|
|
|
scrollView.frameLayoutGuide.bottomAnchor.constraint(equalTo: view.layoutMarginsGuide.bottomAnchor),
|
|
|
|
scrollView.frameLayoutGuide.widthAnchor.constraint(equalTo: scrollView.contentLayoutGuide.widthAnchor),
|
2021-02-20 06:55:06 +01:00
|
|
|
])
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
// stackview
|
2021-02-22 09:20:44 +01:00
|
|
|
scrollView.addSubview(stackView)
|
2021-02-20 06:55:06 +01:00
|
|
|
stackView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
NSLayoutConstraint.activate([
|
2021-02-22 09:20:44 +01:00
|
|
|
stackView.topAnchor.constraint(equalTo: scrollView.contentLayoutGuide.topAnchor),
|
|
|
|
stackView.leadingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.leadingAnchor),
|
|
|
|
stackView.trailingAnchor.constraint(equalTo: scrollView.contentLayoutGuide.trailingAnchor),
|
|
|
|
stackView.widthAnchor.constraint(equalTo: scrollView.contentLayoutGuide.widthAnchor),
|
|
|
|
scrollView.contentLayoutGuide.bottomAnchor.constraint(equalTo: stackView.bottomAnchor),
|
|
|
|
])
|
2021-02-20 06:55:06 +01:00
|
|
|
|
|
|
|
// photoview
|
|
|
|
photoView.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
photoView.addSubview(photoButton)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
photoView.heightAnchor.constraint(equalToConstant: 90).priority(.defaultHigh),
|
|
|
|
])
|
|
|
|
photoButton.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
photoButton.heightAnchor.constraint(equalToConstant: 90).priority(.defaultHigh),
|
|
|
|
photoButton.widthAnchor.constraint(equalToConstant: 90).priority(.defaultHigh),
|
|
|
|
photoButton.centerXAnchor.constraint(equalTo: photoView.centerXAnchor),
|
|
|
|
photoButton.centerYAnchor.constraint(equalTo: photoView.centerYAnchor),
|
|
|
|
])
|
2021-02-20 11:24:23 +01:00
|
|
|
plusIconBackground.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
photoView.addSubview(plusIconBackground)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
plusIconBackground.trailingAnchor.constraint(equalTo: photoButton.trailingAnchor),
|
|
|
|
plusIconBackground.bottomAnchor.constraint(equalTo: photoButton.bottomAnchor),
|
|
|
|
])
|
2021-02-20 06:55:06 +01:00
|
|
|
plusIcon.translatesAutoresizingMaskIntoConstraints = false
|
|
|
|
photoView.addSubview(plusIcon)
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
plusIcon.trailingAnchor.constraint(equalTo: photoButton.trailingAnchor),
|
|
|
|
plusIcon.bottomAnchor.constraint(equalTo: photoButton.bottomAnchor),
|
|
|
|
])
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
// textfield
|
|
|
|
NSLayoutConstraint.activate([
|
|
|
|
usernameTextField.heightAnchor.constraint(equalToConstant: 50).priority(.defaultHigh),
|
|
|
|
displayNameTextField.heightAnchor.constraint(equalToConstant: 50).priority(.defaultHigh),
|
|
|
|
emailTextField.heightAnchor.constraint(equalToConstant: 50).priority(.defaultHigh),
|
|
|
|
passwordTextField.heightAnchor.constraint(equalToConstant: 50).priority(.defaultHigh),
|
|
|
|
])
|
|
|
|
|
|
|
|
// password
|
|
|
|
stackView.setCustomSpacing(6, after: passwordTextField)
|
|
|
|
stackView.setCustomSpacing(32, after: passwordCheckLabel)
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
// button
|
2021-02-26 11:27:47 +01:00
|
|
|
stackView.addArrangedSubview(buttonContainer)
|
2021-02-05 10:53:00 +01:00
|
|
|
signUpButton.translatesAutoresizingMaskIntoConstraints = false
|
2021-02-26 11:27:47 +01:00
|
|
|
buttonContainer.addSubview(signUpButton)
|
2021-02-05 10:53:00 +01:00
|
|
|
NSLayoutConstraint.activate([
|
2021-02-26 11:27:47 +01:00
|
|
|
signUpButton.topAnchor.constraint(equalTo: buttonContainer.topAnchor),
|
|
|
|
signUpButton.leadingAnchor.constraint(equalTo: buttonContainer.leadingAnchor, constant: MastodonRegisterViewController.actionButtonMargin),
|
|
|
|
buttonContainer.trailingAnchor.constraint(equalTo: signUpButton.trailingAnchor, constant: MastodonRegisterViewController.actionButtonMargin),
|
|
|
|
buttonContainer.bottomAnchor.constraint(equalTo: signUpButton.bottomAnchor),
|
2021-02-26 09:43:59 +01:00
|
|
|
signUpButton.heightAnchor.constraint(equalToConstant: MastodonRegisterViewController.actionButtonHeight).priority(.defaultHigh),
|
2021-02-05 10:53:00 +01:00
|
|
|
])
|
2021-02-26 11:27:47 +01:00
|
|
|
|
2021-02-22 05:51:35 +01:00
|
|
|
Publishers.CombineLatest(
|
2021-02-20 12:54:08 +01:00
|
|
|
KeyboardResponderService.shared.state.eraseToAnyPublisher(),
|
2021-02-22 05:51:35 +01:00
|
|
|
KeyboardResponderService.shared.willEndFrame.eraseToAnyPublisher()
|
2021-02-20 12:54:08 +01:00
|
|
|
)
|
2021-02-22 05:51:35 +01:00
|
|
|
.sink(receiveValue: { [weak self] state, endFrame in
|
2021-02-20 12:54:08 +01:00
|
|
|
guard let self = self else { return }
|
|
|
|
|
2021-02-22 05:51:35 +01:00
|
|
|
guard state == .dock else {
|
2021-02-22 09:20:44 +01:00
|
|
|
self.scrollView.contentInset.bottom = 0.0
|
|
|
|
self.scrollView.verticalScrollIndicatorInsets.bottom = 0.0
|
2021-02-20 12:54:08 +01:00
|
|
|
return
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-22 09:20:44 +01:00
|
|
|
let contentFrame = self.view.convert(self.scrollView.frame, to: nil)
|
2021-02-20 12:54:08 +01:00
|
|
|
let padding = contentFrame.maxY - endFrame.minY
|
|
|
|
guard padding > 0 else {
|
2021-02-22 09:20:44 +01:00
|
|
|
self.scrollView.contentInset.bottom = 0.0
|
|
|
|
self.scrollView.verticalScrollIndicatorInsets.bottom = 0.0
|
2021-02-20 12:54:08 +01:00
|
|
|
return
|
|
|
|
}
|
2021-02-22 09:37:13 +01:00
|
|
|
|
2021-02-22 09:20:44 +01:00
|
|
|
self.scrollView.contentInset.bottom = padding + 16
|
|
|
|
self.scrollView.verticalScrollIndicatorInsets.bottom = padding + 16
|
2021-02-22 09:37:13 +01:00
|
|
|
|
|
|
|
if self.passwordTextField.isFirstResponder {
|
2021-02-26 11:27:47 +01:00
|
|
|
let contentFrame = self.buttonContainer.convert(self.signUpButton.frame, to: nil)
|
2021-02-22 09:37:13 +01:00
|
|
|
let labelPadding = contentFrame.maxY - endFrame.minY
|
|
|
|
let contentOffsetY = self.scrollView.contentOffset.y
|
|
|
|
DispatchQueue.main.async {
|
|
|
|
self.scrollView.setContentOffset(CGPoint(x: 0, y: contentOffsetY + labelPadding + 16.0), animated: true)
|
|
|
|
}
|
|
|
|
}
|
2021-02-20 12:54:08 +01:00
|
|
|
})
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
|
2021-02-05 10:53:00 +01:00
|
|
|
viewModel.isRegistering
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] isRegistering in
|
|
|
|
guard let self = self else { return }
|
2021-02-26 11:27:47 +01:00
|
|
|
isRegistering ? self.signUpButton.showLoading() : self.signUpButton.stopLoading()
|
2021-02-05 10:53:00 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.usernameValidateState
|
2021-02-20 10:13:16 +01:00
|
|
|
.receive(on: DispatchQueue.main)
|
2021-02-22 10:48:44 +01:00
|
|
|
.sink { [weak self] validateState in
|
2021-02-20 10:13:16 +01:00
|
|
|
guard let self = self else { return }
|
2021-02-22 10:48:44 +01:00
|
|
|
self.setTextFieldValidAppearance(self.usernameTextField, validateState: validateState)
|
2021-02-20 10:13:16 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.displayNameValidateState
|
2021-02-20 11:24:23 +01:00
|
|
|
.receive(on: DispatchQueue.main)
|
2021-02-22 10:48:44 +01:00
|
|
|
.sink { [weak self] validateState in
|
2021-02-20 11:24:23 +01:00
|
|
|
guard let self = self else { return }
|
2021-02-22 10:48:44 +01:00
|
|
|
self.setTextFieldValidAppearance(self.displayNameTextField, validateState: validateState)
|
2021-02-20 11:24:23 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.emailValidateState
|
2021-02-20 11:24:23 +01:00
|
|
|
.receive(on: DispatchQueue.main)
|
2021-02-22 10:48:44 +01:00
|
|
|
.sink { [weak self] validateState in
|
2021-02-20 11:24:23 +01:00
|
|
|
guard let self = self else { return }
|
2021-02-22 10:48:44 +01:00
|
|
|
self.setTextFieldValidAppearance(self.emailTextField, validateState: validateState)
|
2021-02-20 11:24:23 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.passwordValidateState
|
2021-02-20 11:24:23 +01:00
|
|
|
.receive(on: DispatchQueue.main)
|
2021-02-22 10:48:44 +01:00
|
|
|
.sink { [weak self] validateState in
|
2021-02-20 11:24:23 +01:00
|
|
|
guard let self = self else { return }
|
2021-02-22 10:48:44 +01:00
|
|
|
self.setTextFieldValidAppearance(self.passwordTextField, validateState: validateState)
|
|
|
|
self.passwordCheckLabel.attributedText = self.viewModel.attributeStringForPassword(eightCharacters: validateState == .valid)
|
2021-02-20 11:24:23 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-20 10:13:16 +01:00
|
|
|
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.isAllValid
|
2021-03-01 10:45:02 +01:00
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] isAllValid in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.signUpButton.isEnabled = isAllValid
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-03-01 10:45:02 +01:00
|
|
|
viewModel.isUsernameTaken
|
|
|
|
.receive(on: DispatchQueue.main)
|
2021-03-02 11:21:54 +01:00
|
|
|
.sink { [weak self] isUsernameTaken in
|
2021-03-01 10:45:02 +01:00
|
|
|
guard let self = self else { return }
|
|
|
|
if isUsernameTaken {
|
|
|
|
self.usernameIsTakenLabel.isHidden = false
|
|
|
|
stackView.setCustomSpacing(6, after: self.usernameTextField)
|
|
|
|
stackView.setCustomSpacing(16, after: self.usernameIsTakenLabel)
|
|
|
|
} else {
|
|
|
|
self.usernameIsTakenLabel.isHidden = true
|
|
|
|
stackView.setCustomSpacing(40, after: self.usernameTextField)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-05 10:53:00 +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-03-01 10:45:02 +01:00
|
|
|
switch error.mastodonError {
|
|
|
|
case .generic(let mastodonEntityError):
|
|
|
|
if let usernameTakenError = mastodonEntityError.details?.username {
|
|
|
|
let isUsernameAvaliable = usernameTakenError.filter { errorDetailReason -> Bool in
|
|
|
|
errorDetailReason.error == .ERR_TAKEN
|
|
|
|
}.isEmpty
|
|
|
|
self.viewModel.isUsernameTaken.value = !isUsernameAvaliable
|
|
|
|
}
|
|
|
|
default:
|
|
|
|
break
|
|
|
|
}
|
2021-02-24 12:08:30 +01:00
|
|
|
let alertController = UIAlertController(for: error, title: "Sign Up Failure", preferredStyle: .alert)
|
2021-02-22 09:20:44 +01:00
|
|
|
let okAction = UIAlertAction(title: L10n.Common.Controls.Actions.ok, style: .default, handler: nil)
|
2021-02-05 10:53:00 +01:00
|
|
|
alertController.addAction(okAction)
|
|
|
|
self.coordinator.present(
|
|
|
|
scene: .alertController(alertController: alertController),
|
|
|
|
from: nil,
|
|
|
|
transition: .alertController(animated: true, completion: nil)
|
|
|
|
)
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-02-22 10:48:44 +01:00
|
|
|
NotificationCenter.default
|
|
|
|
.publisher(for: UITextField.textDidChangeNotification, object: usernameTextField)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.viewModel.username.value = self.usernameTextField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
|
|
|
|
NotificationCenter.default
|
|
|
|
.publisher(for: UITextField.textDidChangeNotification, object: displayNameTextField)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.viewModel.displayName.value = self.displayNameTextField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
|
|
|
|
NotificationCenter.default
|
|
|
|
.publisher(for: UITextField.textDidChangeNotification, object: emailTextField)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.viewModel.email.value = self.emailTextField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
NotificationCenter.default
|
|
|
|
.publisher(for: UITextField.textDidChangeNotification, object: passwordTextField)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
guard let self = self else { return }
|
2021-02-22 10:48:44 +01:00
|
|
|
self.viewModel.password.value = self.passwordTextField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-02-20 12:54:08 +01:00
|
|
|
|
2021-03-01 10:45:02 +01:00
|
|
|
if viewModel.approvalRequired {
|
2021-02-26 05:52:37 +01:00
|
|
|
inviteTextField.delegate = self
|
|
|
|
NSLayoutConstraint.activate([
|
2021-03-02 11:21:54 +01:00
|
|
|
inviteTextField.heightAnchor.constraint(equalToConstant: 50).priority(.defaultHigh),
|
2021-02-26 05:52:37 +01:00
|
|
|
])
|
|
|
|
|
|
|
|
viewModel.inviteValidateState
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] validateState in
|
|
|
|
guard let self = self else { return }
|
|
|
|
self.setTextFieldValidAppearance(self.inviteTextField, validateState: validateState)
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
NotificationCenter.default
|
|
|
|
.publisher(for: UITextField.textDidChangeNotification, object: inviteTextField)
|
|
|
|
.receive(on: DispatchQueue.main)
|
|
|
|
.sink { [weak self] _ in
|
|
|
|
guard let self = self else { return }
|
2021-02-26 11:27:47 +01:00
|
|
|
self.viewModel.reason.value = self.inviteTextField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
2021-02-26 05:52:37 +01:00
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
|
|
|
}
|
|
|
|
|
2021-02-05 10:53:00 +01:00
|
|
|
signUpButton.addTarget(self, action: #selector(MastodonRegisterViewController.signUpButtonPressed(_:)), for: .touchUpInside)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-20 06:55:06 +01:00
|
|
|
extension MastodonRegisterViewController: UITextFieldDelegate {
|
2021-02-22 10:48:44 +01:00
|
|
|
func textFieldDidBeginEditing(_ textField: UITextField) {
|
|
|
|
let text = textField.text?.trimmingCharacters(in: .whitespacesAndNewlines) ?? ""
|
|
|
|
|
2021-02-20 10:13:16 +01:00
|
|
|
switch textField {
|
|
|
|
case usernameTextField:
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.username.value = text
|
2021-02-20 11:24:23 +01:00
|
|
|
case displayNameTextField:
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.displayName.value = text
|
2021-02-20 11:24:23 +01:00
|
|
|
case emailTextField:
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.email.value = text
|
2021-02-20 11:24:23 +01:00
|
|
|
case passwordTextField:
|
2021-02-22 10:48:44 +01:00
|
|
|
viewModel.password.value = text
|
2021-02-26 05:52:37 +01:00
|
|
|
case inviteTextField:
|
2021-02-26 11:27:47 +01:00
|
|
|
viewModel.reason.value = text
|
2021-02-22 10:48:44 +01:00
|
|
|
default:
|
|
|
|
break
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func showShadowWithColor(color: UIColor, textField: UITextField) {
|
|
|
|
// To apply Shadow
|
|
|
|
textField.layer.shadowOpacity = 1
|
|
|
|
textField.layer.shadowRadius = 2.0
|
2021-02-20 11:24:23 +01:00
|
|
|
textField.layer.shadowOffset = CGSize.zero
|
2021-02-20 06:55:06 +01:00
|
|
|
textField.layer.shadowColor = color.cgColor
|
2021-02-20 10:13:16 +01:00
|
|
|
textField.layer.shadowPath = UIBezierPath(roundedRect: textField.bounds, byRoundingCorners: .allCorners, cornerRadii: CGSize(width: 2.0, height: 2.0)).cgPath
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
2021-02-20 11:24:23 +01:00
|
|
|
|
2021-02-22 10:48:44 +01:00
|
|
|
private func setTextFieldValidAppearance(_ textField: UITextField, validateState: MastodonRegisterViewModel.ValidateState) {
|
|
|
|
switch validateState {
|
|
|
|
case .empty:
|
|
|
|
showShadowWithColor(color: textField.isFirstResponder ? Asset.Colors.TextField.highlight.color : .clear, textField: textField)
|
|
|
|
case .valid:
|
|
|
|
showShadowWithColor(color: Asset.Colors.TextField.valid.color, textField: textField)
|
|
|
|
case .invalid:
|
|
|
|
showShadowWithColor(color: Asset.Colors.TextField.invalid.color, textField: textField)
|
2021-02-20 10:13:16 +01:00
|
|
|
}
|
|
|
|
}
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
extension MastodonRegisterViewController {
|
2021-02-22 10:48:44 +01:00
|
|
|
@objc private func tapGestureRecognizerHandler(_ sender: UITapGestureRecognizer) {
|
|
|
|
view.endEditing(true)
|
2021-02-20 06:55:06 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
@objc private func signUpButtonPressed(_ sender: UIButton) {
|
|
|
|
os_log(.info, log: .debug, "%{public}s[%{public}ld], %{public}s", (#file as NSString).lastPathComponent, #line, #function)
|
2021-02-22 10:48:44 +01:00
|
|
|
guard viewModel.isAllValid.value else { return }
|
2021-02-05 10:53:00 +01:00
|
|
|
|
|
|
|
guard !viewModel.isRegistering.value else { return }
|
|
|
|
viewModel.isRegistering.value = true
|
2021-02-22 10:48:44 +01:00
|
|
|
|
|
|
|
let username = viewModel.username.value
|
|
|
|
let email = viewModel.email.value
|
|
|
|
let password = viewModel.password.value
|
2021-02-26 11:27:47 +01:00
|
|
|
let query = Mastodon.API.Account.RegisterQuery(
|
|
|
|
reason: viewModel.reason.value,
|
|
|
|
username: username,
|
|
|
|
email: email,
|
|
|
|
password: password,
|
|
|
|
agreement: true, // TODO:
|
|
|
|
locale: "en" // TODO:
|
|
|
|
)
|
2021-02-05 10:53:00 +01:00
|
|
|
|
2021-03-01 11:06:37 +01:00
|
|
|
// register without show server rules
|
|
|
|
context.apiService.accountRegister(
|
|
|
|
domain: viewModel.domain,
|
|
|
|
query: query,
|
|
|
|
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
|
2021-02-26 11:27:47 +01:00
|
|
|
}
|
2021-03-01 11:06:37 +01:00
|
|
|
} 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)
|
2021-02-22 09:20:44 +01:00
|
|
|
}
|
2021-03-01 11:06:37 +01:00
|
|
|
.store(in: &disposeBag)
|
2021-02-05 10:53:00 +01:00
|
|
|
}
|
|
|
|
}
|