|  |  |  | 
 | 
					
						
							|  |  |  | final class PermissionMissingModal : Modal { | 
					
						
							|  |  |  |     private let permission: String | 
					
						
							|  |  |  |     private let onCancel: () -> Void | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Lifecycle | 
					
						
							|  |  |  |     init(permission: String, onCancel: @escaping () -> Void) { | 
					
						
							|  |  |  |         self.permission = permission | 
					
						
							|  |  |  |         self.onCancel = onCancel | 
					
						
							|  |  |  |         super.init(nibName: nil, bundle: nil) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override init(nibName: String?, bundle: Bundle?) { | 
					
						
							|  |  |  |         preconditionFailure("Use init(permission:) instead.") | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     required init?(coder: NSCoder) { | 
					
						
							|  |  |  |         preconditionFailure("Use init(permission:) instead.") | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override func populateContentView() { | 
					
						
							|  |  |  |         // Title | 
					
						
							|  |  |  |         let titleLabel = UILabel() | 
					
						
							|  |  |  |         titleLabel.textColor = Colors.text | 
					
						
							|  |  |  |         titleLabel.font = .boldSystemFont(ofSize: Values.largeFontSize) | 
					
						
							|  |  |  |         titleLabel.text = "Session" | 
					
						
							|  |  |  |         titleLabel.textAlignment = .center | 
					
						
							|  |  |  |         // Message | 
					
						
							|  |  |  |         let messageLabel = UILabel() | 
					
						
							|  |  |  |         messageLabel.textColor = Colors.text | 
					
						
							|  |  |  |         messageLabel.font = .systemFont(ofSize: Values.smallFontSize) | 
					
						
							|  |  |  |         let message = "Session needs \(permission) access to continue. You can enable access in the iOS settings." | 
					
						
							|  |  |  |         let attributedMessage = NSMutableAttributedString(string: message) | 
					
						
							|  |  |  |         attributedMessage.addAttributes([ .font : UIFont.boldSystemFont(ofSize: Values.smallFontSize) ], range: (message as NSString).range(of: permission)) | 
					
						
							|  |  |  |         messageLabel.attributedText = attributedMessage | 
					
						
							|  |  |  |         messageLabel.numberOfLines = 0 | 
					
						
							|  |  |  |         messageLabel.lineBreakMode = .byWordWrapping | 
					
						
							|  |  |  |         messageLabel.textAlignment = .center | 
					
						
							|  |  |  |         // Settings button | 
					
						
							|  |  |  |         let settingsButton = UIButton() | 
					
						
							|  |  |  |         settingsButton.set(.height, to: Values.mediumButtonHeight) | 
					
						
							|  |  |  |         settingsButton.layer.cornerRadius = Modal.buttonCornerRadius | 
					
						
							|  |  |  |         settingsButton.backgroundColor = Colors.buttonBackground | 
					
						
							|  |  |  |         settingsButton.titleLabel!.font = .systemFont(ofSize: Values.smallFontSize) | 
					
						
							|  |  |  |         settingsButton.setTitleColor(Colors.text, for: UIControl.State.normal) | 
					
						
							|  |  |  |         settingsButton.setTitle("Settings", for: UIControl.State.normal) | 
					
						
							|  |  |  |         settingsButton.addTarget(self, action: #selector(goToSettings), for: UIControl.Event.touchUpInside) | 
					
						
							|  |  |  |         // Button stack view | 
					
						
							|  |  |  |         let buttonStackView = UIStackView(arrangedSubviews: [ cancelButton, settingsButton ]) | 
					
						
							|  |  |  |         buttonStackView.axis = .horizontal | 
					
						
							|  |  |  |         buttonStackView.spacing = Values.mediumSpacing | 
					
						
							|  |  |  |         buttonStackView.distribution = .fillEqually | 
					
						
							|  |  |  |         // Main stack view | 
					
						
							|  |  |  |         let mainStackView = UIStackView(arrangedSubviews: [ titleLabel, messageLabel, buttonStackView ]) | 
					
						
							|  |  |  |         mainStackView.axis = .vertical | 
					
						
							|  |  |  |         mainStackView.spacing = Values.largeSpacing | 
					
						
							|  |  |  |         contentView.addSubview(mainStackView) | 
					
						
							|  |  |  |         mainStackView.pin(.leading, to: .leading, of: contentView, withInset: Values.largeSpacing) | 
					
						
							|  |  |  |         mainStackView.pin(.top, to: .top, of: contentView, withInset: Values.largeSpacing) | 
					
						
							|  |  |  |         contentView.pin(.trailing, to: .trailing, of: mainStackView, withInset: Values.largeSpacing) | 
					
						
							|  |  |  |         contentView.pin(.bottom, to: .bottom, of: mainStackView, withInset: Values.largeSpacing) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // MARK: Interaction | 
					
						
							|  |  |  |     @objc private func goToSettings() { | 
					
						
							|  |  |  |         presentingViewController?.dismiss(animated: true, completion: { | 
					
						
							|  |  |  |             UIApplication.shared.open(URL(string: UIApplication.openSettingsURLString)!) | 
					
						
							|  |  |  |         }) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     override func close() { | 
					
						
							|  |  |  |         super.close() | 
					
						
							|  |  |  |         onCancel() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |