mirror of https://github.com/oxen-io/session-ios
Implement linkification
parent
aa027a28c5
commit
401a29344d
@ -1,70 +0,0 @@
|
|||||||
|
|
||||||
extension ConversationVC {
|
|
||||||
|
|
||||||
final class ScrollToBottomButton : UIView {
|
|
||||||
private let delegate: ScrollToBottomButtonDelegate
|
|
||||||
|
|
||||||
// MARK: Settings
|
|
||||||
private static let size: CGFloat = 40
|
|
||||||
private static let iconSize: CGFloat = 16
|
|
||||||
|
|
||||||
// MARK: Lifecycle
|
|
||||||
init(delegate: ScrollToBottomButtonDelegate) {
|
|
||||||
self.delegate = delegate
|
|
||||||
super.init(frame: CGRect.zero)
|
|
||||||
setUpViewHierarchy()
|
|
||||||
}
|
|
||||||
|
|
||||||
override init(frame: CGRect) {
|
|
||||||
preconditionFailure("Use init(delegate:) instead.")
|
|
||||||
}
|
|
||||||
|
|
||||||
required init?(coder: NSCoder) {
|
|
||||||
preconditionFailure("Use init(delegate:) instead.")
|
|
||||||
}
|
|
||||||
|
|
||||||
private func setUpViewHierarchy() {
|
|
||||||
// Background & blur
|
|
||||||
let backgroundView = UIView()
|
|
||||||
backgroundView.backgroundColor = isLightMode ? .white : .black
|
|
||||||
backgroundView.alpha = Values.lowOpacity
|
|
||||||
addSubview(backgroundView)
|
|
||||||
backgroundView.pin(to: self)
|
|
||||||
let blurView = UIVisualEffectView(effect: UIBlurEffect(style: .regular))
|
|
||||||
addSubview(blurView)
|
|
||||||
blurView.pin(to: self)
|
|
||||||
// Size & shape
|
|
||||||
let size = ScrollToBottomButton.size
|
|
||||||
set(.width, to: size)
|
|
||||||
set(.height, to: size)
|
|
||||||
layer.cornerRadius = size / 2
|
|
||||||
layer.masksToBounds = true
|
|
||||||
// Border
|
|
||||||
layer.borderWidth = 1
|
|
||||||
let borderColor = (isLightMode ? UIColor.black : UIColor.white).withAlphaComponent(Values.veryLowOpacity)
|
|
||||||
layer.borderColor = borderColor.cgColor
|
|
||||||
// Icon
|
|
||||||
let tint = isLightMode ? UIColor.black : UIColor.white
|
|
||||||
let icon = UIImage(named: "ic_chevron_down")!.withTint(tint)
|
|
||||||
let iconImageView = UIImageView(image: icon)
|
|
||||||
iconImageView.set(.width, to: ScrollToBottomButton.iconSize)
|
|
||||||
iconImageView.set(.height, to: ScrollToBottomButton.iconSize)
|
|
||||||
iconImageView.contentMode = .scaleAspectFit
|
|
||||||
addSubview(iconImageView)
|
|
||||||
iconImageView.center(in: self)
|
|
||||||
// Gesture recognizer
|
|
||||||
let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleTap))
|
|
||||||
addGestureRecognizer(tapGestureRecognizer)
|
|
||||||
}
|
|
||||||
|
|
||||||
// MARK: Interaction
|
|
||||||
@objc private func handleTap() {
|
|
||||||
delegate.handleScrollToBottomButtonTapped()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protocol ScrollToBottomButtonDelegate {
|
|
||||||
|
|
||||||
func handleScrollToBottomButtonTapped()
|
|
||||||
}
|
|
@ -0,0 +1,67 @@
|
|||||||
|
|
||||||
|
final class ScrollToBottomButton : UIView {
|
||||||
|
private let delegate: ScrollToBottomButtonDelegate
|
||||||
|
|
||||||
|
// MARK: Settings
|
||||||
|
private static let size: CGFloat = 40
|
||||||
|
private static let iconSize: CGFloat = 16
|
||||||
|
|
||||||
|
// MARK: Lifecycle
|
||||||
|
init(delegate: ScrollToBottomButtonDelegate) {
|
||||||
|
self.delegate = delegate
|
||||||
|
super.init(frame: CGRect.zero)
|
||||||
|
setUpViewHierarchy()
|
||||||
|
}
|
||||||
|
|
||||||
|
override init(frame: CGRect) {
|
||||||
|
preconditionFailure("Use init(delegate:) instead.")
|
||||||
|
}
|
||||||
|
|
||||||
|
required init?(coder: NSCoder) {
|
||||||
|
preconditionFailure("Use init(delegate:) instead.")
|
||||||
|
}
|
||||||
|
|
||||||
|
private func setUpViewHierarchy() {
|
||||||
|
// Background & blur
|
||||||
|
let backgroundView = UIView()
|
||||||
|
backgroundView.backgroundColor = isLightMode ? .white : .black
|
||||||
|
backgroundView.alpha = Values.lowOpacity
|
||||||
|
addSubview(backgroundView)
|
||||||
|
backgroundView.pin(to: self)
|
||||||
|
let blurView = UIVisualEffectView(effect: UIBlurEffect(style: .regular))
|
||||||
|
addSubview(blurView)
|
||||||
|
blurView.pin(to: self)
|
||||||
|
// Size & shape
|
||||||
|
let size = ScrollToBottomButton.size
|
||||||
|
set(.width, to: size)
|
||||||
|
set(.height, to: size)
|
||||||
|
layer.cornerRadius = size / 2
|
||||||
|
layer.masksToBounds = true
|
||||||
|
// Border
|
||||||
|
layer.borderWidth = 1
|
||||||
|
let borderColor = (isLightMode ? UIColor.black : UIColor.white).withAlphaComponent(Values.veryLowOpacity)
|
||||||
|
layer.borderColor = borderColor.cgColor
|
||||||
|
// Icon
|
||||||
|
let tint = isLightMode ? UIColor.black : UIColor.white
|
||||||
|
let icon = UIImage(named: "ic_chevron_down")!.withTint(tint)
|
||||||
|
let iconImageView = UIImageView(image: icon)
|
||||||
|
iconImageView.set(.width, to: ScrollToBottomButton.iconSize)
|
||||||
|
iconImageView.set(.height, to: ScrollToBottomButton.iconSize)
|
||||||
|
iconImageView.contentMode = .scaleAspectFit
|
||||||
|
addSubview(iconImageView)
|
||||||
|
iconImageView.center(in: self)
|
||||||
|
// Gesture recognizer
|
||||||
|
let tapGestureRecognizer = UITapGestureRecognizer(target: self, action: #selector(handleTap))
|
||||||
|
addGestureRecognizer(tapGestureRecognizer)
|
||||||
|
}
|
||||||
|
|
||||||
|
// MARK: Interaction
|
||||||
|
@objc private func handleTap() {
|
||||||
|
delegate.handleScrollToBottomButtonTapped()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protocol ScrollToBottomButtonDelegate {
|
||||||
|
|
||||||
|
func handleScrollToBottomButtonTapped()
|
||||||
|
}
|
@ -0,0 +1,69 @@
|
|||||||
|
|
||||||
|
final class URLModal : Modal {
|
||||||
|
private let url: URL
|
||||||
|
|
||||||
|
// MARK: Lifecycle
|
||||||
|
init(url: URL) {
|
||||||
|
self.url = url
|
||||||
|
super.init(nibName: nil, bundle: nil)
|
||||||
|
}
|
||||||
|
|
||||||
|
override init(nibName: String?, bundle: Bundle?) {
|
||||||
|
preconditionFailure("Use init(url:) instead.")
|
||||||
|
}
|
||||||
|
|
||||||
|
required init?(coder: NSCoder) {
|
||||||
|
preconditionFailure("Use init(url:) instead.")
|
||||||
|
}
|
||||||
|
|
||||||
|
override func populateContentView() {
|
||||||
|
// Title
|
||||||
|
let titleLabel = UILabel()
|
||||||
|
titleLabel.textColor = Colors.text
|
||||||
|
titleLabel.font = .boldSystemFont(ofSize: Values.largeFontSize)
|
||||||
|
titleLabel.text = "Open URL?"
|
||||||
|
titleLabel.textAlignment = .center
|
||||||
|
// Message
|
||||||
|
let messageLabel = UILabel()
|
||||||
|
messageLabel.textColor = Colors.text
|
||||||
|
messageLabel.font = .systemFont(ofSize: Values.smallFontSize)
|
||||||
|
let message = "Are you sure you want to open \(url.absoluteString)?"
|
||||||
|
let attributedMessage = NSMutableAttributedString(string: message)
|
||||||
|
attributedMessage.addAttributes([ .font : UIFont.boldSystemFont(ofSize: Values.smallFontSize) ], range: (message as NSString).range(of: url.absoluteString))
|
||||||
|
messageLabel.attributedText = attributedMessage
|
||||||
|
messageLabel.numberOfLines = 0
|
||||||
|
messageLabel.lineBreakMode = .byWordWrapping
|
||||||
|
messageLabel.textAlignment = .center
|
||||||
|
// Open button
|
||||||
|
let openButton = UIButton()
|
||||||
|
openButton.set(.height, to: Values.mediumButtonHeight)
|
||||||
|
openButton.layer.cornerRadius = Values.modalButtonCornerRadius
|
||||||
|
openButton.backgroundColor = Colors.buttonBackground
|
||||||
|
openButton.titleLabel!.font = .systemFont(ofSize: Values.smallFontSize)
|
||||||
|
openButton.setTitleColor(Colors.text, for: UIControl.State.normal)
|
||||||
|
openButton.setTitle("Open", for: UIControl.State.normal)
|
||||||
|
openButton.addTarget(self, action: #selector(openURL), for: UIControl.Event.touchUpInside)
|
||||||
|
// Button stack view
|
||||||
|
let buttonStackView = UIStackView(arrangedSubviews: [ cancelButton, openButton ])
|
||||||
|
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 openURL() {
|
||||||
|
let url = self.url
|
||||||
|
presentingViewController?.dismiss(animated: true, completion: {
|
||||||
|
UIApplication.shared.open(url, options: [:], completionHandler: nil)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue