mirror of https://github.com/oxen-io/session-ios
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.
123 lines
4.2 KiB
Swift
123 lines
4.2 KiB
Swift
// Copyright © 2022 Rangeproof Pty Ltd. All rights reserved.
|
|
|
|
import UIKit
|
|
import SessionUIKit
|
|
|
|
final class OptionView: UIView {
|
|
private let title: String
|
|
private let explanation: String
|
|
private let delegate: OptionViewDelegate
|
|
private let isRecommended: Bool
|
|
|
|
var isSelected = false { didSet { handleIsSelectedChanged() } }
|
|
|
|
private static let cornerRadius: CGFloat = 8
|
|
|
|
init(title: String, explanation: String, delegate: OptionViewDelegate, isRecommended: Bool = false) {
|
|
self.title = title
|
|
self.explanation = explanation
|
|
self.delegate = delegate
|
|
self.isRecommended = isRecommended
|
|
|
|
super.init(frame: CGRect.zero)
|
|
|
|
setUpViewHierarchy()
|
|
}
|
|
|
|
override init(frame: CGRect) {
|
|
preconditionFailure("Use init(string:explanation:) instead.")
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
preconditionFailure("Use init(string:explanation:) instead.")
|
|
}
|
|
|
|
private func setUpViewHierarchy() {
|
|
themeBackgroundColor = .backgroundSecondary
|
|
|
|
// Round corners
|
|
layer.cornerRadius = OptionView.cornerRadius
|
|
|
|
// Set up border
|
|
themeBorderColor = .borderSeparator
|
|
layer.borderWidth = 1
|
|
|
|
// Set up shadow
|
|
themeShadowColor = .black
|
|
layer.shadowOffset = CGSize(width: 0, height: 0.8)
|
|
|
|
ThemeManager.onThemeChange(observer: self) { [weak self] theme, _ in
|
|
switch theme.interfaceStyle {
|
|
case .light:
|
|
self?.layer.shadowOpacity = 0.16
|
|
self?.layer.shadowRadius = 4
|
|
|
|
default:
|
|
self?.layer.shadowOpacity = 1
|
|
self?.layer.shadowRadius = 6
|
|
}
|
|
}
|
|
|
|
// Set up title label
|
|
let titleLabel: UILabel = UILabel()
|
|
titleLabel.font = .boldSystemFont(ofSize: Values.mediumFontSize)
|
|
titleLabel.text = title
|
|
titleLabel.themeTextColor = .textPrimary
|
|
titleLabel.lineBreakMode = .byWordWrapping
|
|
titleLabel.numberOfLines = 0
|
|
|
|
// Set up explanation label
|
|
let explanationLabel: UILabel = UILabel()
|
|
explanationLabel.font = .systemFont(ofSize: Values.verySmallFontSize)
|
|
explanationLabel.text = explanation
|
|
explanationLabel.themeTextColor = .textPrimary
|
|
explanationLabel.lineBreakMode = .byWordWrapping
|
|
explanationLabel.numberOfLines = 0
|
|
|
|
// Set up stack view
|
|
let stackView = UIStackView(arrangedSubviews: [ titleLabel, explanationLabel ])
|
|
stackView.axis = .vertical
|
|
stackView.spacing = 4
|
|
stackView.alignment = .fill
|
|
addSubview(stackView)
|
|
|
|
stackView.pin(.leading, to: .leading, of: self, withInset: 12)
|
|
stackView.pin(.top, to: .top, of: self, withInset: 12)
|
|
self.pin(.trailing, to: .trailing, of: stackView, withInset: 12)
|
|
self.pin(.bottom, to: .bottom, of: stackView, withInset: 12)
|
|
|
|
// Set up recommended label if needed
|
|
if isRecommended {
|
|
let recommendedLabel: UILabel = UILabel()
|
|
recommendedLabel.font = .boldSystemFont(ofSize: Values.smallFontSize)
|
|
recommendedLabel.text = "recommended".localized()
|
|
recommendedLabel.themeTextColor = .primary
|
|
stackView.addArrangedSubview(recommendedLabel)
|
|
}
|
|
|
|
// Set up tap gesture recognizer
|
|
let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleTap))
|
|
addGestureRecognizer(tapGestureRecognizer)
|
|
}
|
|
|
|
@objc private func handleTap() {
|
|
isSelected = !isSelected
|
|
}
|
|
|
|
private func handleIsSelectedChanged() {
|
|
UIView.animate(withDuration: 0.3) { [weak self] in
|
|
self?.themeBorderColor = (self?.isSelected == true ? .primary : .borderSeparator)
|
|
self?.themeShadowColor = (self?.isSelected == true ? .primary : .black)
|
|
}
|
|
|
|
// Notify delegate
|
|
if isSelected { delegate.optionViewDidActivate(self) }
|
|
}
|
|
}
|
|
|
|
// MARK: - Option View Delegate
|
|
|
|
protocol OptionViewDelegate {
|
|
func optionViewDidActivate(_ optionView: OptionView)
|
|
}
|