2019-12-04 03:44:00 +01:00
final class NewPrivateChatVC : UIViewController , UIPageViewControllerDataSource , UIPageViewControllerDelegate , OWSQRScannerDelegate {
private let pageVC = UIPageViewController ( transitionStyle : . scroll , navigationOrientation : . horizontal , options : nil )
private var pages : [ UIViewController ] = [ ]
private var targetVCIndex : Int ?
2019-12-04 06:54:46 +01:00
// MARK: S e t t i n g s
override var preferredStatusBarStyle : UIStatusBarStyle { return . lightContent }
2019-12-04 03:44:00 +01:00
// MARK: C o m p o n e n t s
private lazy var tabBar : TabBar = {
let tabs = [
2019-12-04 05:17:06 +01:00
TabBar . Tab ( title : NSLocalizedString ( " Enter Public Key " , comment : " " ) ) { [ weak self ] in
2019-12-04 03:44:00 +01:00
guard let self = self else { return }
self . pageVC . setViewControllers ( [ self . pages [ 0 ] ] , direction : . forward , animated : false , completion : nil )
} ,
2019-12-04 05:17:06 +01:00
TabBar . Tab ( title : NSLocalizedString ( " Scan QR Code " , comment : " " ) ) { [ weak self ] in
2019-12-04 03:44:00 +01:00
guard let self = self else { return }
self . pageVC . setViewControllers ( [ self . pages [ 1 ] ] , direction : . forward , animated : false , completion : nil )
}
]
return TabBar ( tabs : tabs )
} ( )
private lazy var enterPublicKeyVC : EnterPublicKeyVC = {
let result = EnterPublicKeyVC ( )
result . newPrivateChatVC = self
return result
} ( )
private lazy var scanQRCodePlaceholderVC : ScanQRCodePlaceholderVC = {
let result = ScanQRCodePlaceholderVC ( )
result . newPrivateChatVC = self
return result
} ( )
private lazy var scanQRCodeWrapperVC : ScanQRCodeWrapperVC = {
2019-12-04 05:17:06 +01:00
let message = NSLocalizedString ( " Users can share their QR code by going into their account settings and tapping \" Share QR Code \" . " , comment : " " )
2019-12-04 03:44:00 +01:00
let result = ScanQRCodeWrapperVC ( message : message )
result . delegate = self
return result
} ( )
// MARK: L i f e c y c l e
override func viewDidLoad ( ) {
// S e t g r a d i e n t b a c k g r o u n d
view . backgroundColor = . clear
let gradient = Gradients . defaultLokiBackground
view . setGradient ( gradient )
// S e t n a v i g a t i o n b a r b a c k g r o u n d c o l o r
let navigationBar = navigationController ! . navigationBar
navigationBar . setBackgroundImage ( UIImage ( ) , for : UIBarMetrics . default )
navigationBar . shadowImage = UIImage ( )
navigationBar . isTranslucent = false
navigationBar . barTintColor = Colors . navigationBarBackground
// S e t u p n a v i g a t i o n b a r b u t t o n s
let closeButton = UIBarButtonItem ( image : # imageLiteral ( resourceName : " X " ) , style : . plain , target : self , action : #selector ( close ) )
closeButton . tintColor = Colors . text
navigationItem . leftBarButtonItem = closeButton
// C u s t o m i z e t i t l e
let titleLabel = UILabel ( )
titleLabel . text = NSLocalizedString ( " New Conversation " , comment : " " )
titleLabel . textColor = Colors . text
2019-12-05 04:31:45 +01:00
titleLabel . font = . boldSystemFont ( ofSize : Values . veryLargeFontSize )
2019-12-04 03:44:00 +01:00
navigationItem . titleView = titleLabel
// S e t u p p a g e V C
let hasCameraAccess = ( AVCaptureDevice . authorizationStatus ( for : . video ) = = . authorized )
pages = [ enterPublicKeyVC , ( hasCameraAccess ? scanQRCodeWrapperVC : scanQRCodePlaceholderVC ) ]
pageVC . dataSource = self
pageVC . delegate = self
pageVC . setViewControllers ( [ enterPublicKeyVC ] , direction : . forward , animated : false , completion : nil )
// S e t u p t a b b a r
view . addSubview ( tabBar )
tabBar . pin ( . leading , to : . leading , of : view )
tabBar . pin ( . top , to : . top , of : view , withInset : navigationBar . height ( ) )
view . pin ( . trailing , to : . trailing , of : tabBar )
// S e t u p p a g e V C c o n s t r a i n t s
let pageVCView = pageVC . view !
view . addSubview ( pageVCView )
pageVCView . pin ( . leading , to : . leading , of : view )
pageVCView . pin ( . top , to : . bottom , of : tabBar )
view . pin ( . trailing , to : . trailing , of : pageVCView )
view . pin ( . bottom , to : . bottom , of : pageVCView )
let screen = UIScreen . main . bounds
pageVCView . set ( . width , to : screen . width )
let height = navigationController ! . view . bounds . height - navigationBar . height ( ) - Values . tabBarHeight
pageVCView . set ( . height , to : height )
enterPublicKeyVC . constrainHeight ( to : height )
scanQRCodePlaceholderVC . constrainHeight ( to : height )
}
// MARK: G e n e r a l
func pageViewController ( _ pageViewController : UIPageViewController , viewControllerBefore viewController : UIViewController ) -> UIViewController ? {
guard let index = pages . firstIndex ( of : viewController ) , index != 0 else { return nil }
return pages [ index - 1 ]
}
func pageViewController ( _ pageViewController : UIPageViewController , viewControllerAfter viewController : UIViewController ) -> UIViewController ? {
guard let index = pages . firstIndex ( of : viewController ) , index != ( pages . count - 1 ) else { return nil }
return pages [ index + 1 ]
}
fileprivate func handleCameraAccessGranted ( ) {
pages [ 1 ] = scanQRCodeWrapperVC
pageVC . setViewControllers ( [ scanQRCodeWrapperVC ] , direction : . forward , animated : false , completion : nil )
}
// MARK: U p d a t i n g
func pageViewController ( _ pageViewController : UIPageViewController , willTransitionTo pendingViewControllers : [ UIViewController ] ) {
guard let targetVC = pendingViewControllers . first , let index = pages . firstIndex ( of : targetVC ) else { return }
targetVCIndex = index
}
func pageViewController ( _ pageViewController : UIPageViewController , didFinishAnimating isFinished : Bool , previousViewControllers : [ UIViewController ] , transitionCompleted isCompleted : Bool ) {
guard isCompleted , let index = targetVCIndex else { return }
tabBar . selectTab ( at : index )
}
// MARK: I n t e r a c t i o n
@objc private func close ( ) {
dismiss ( animated : true , completion : nil )
}
func controller ( _ controller : OWSQRCodeScanningViewController , didDetectQRCodeWith string : String ) {
let hexEncodedPublicKey = string
startNewPrivateChatIfPossible ( with : hexEncodedPublicKey )
}
fileprivate func startNewPrivateChatIfPossible ( with hexEncodedPublicKey : String ) {
if ! ECKeyPair . isValidHexEncodedPublicKey ( candidate : hexEncodedPublicKey ) {
let alert = UIAlertController ( title : NSLocalizedString ( " Invalid Public Key " , comment : " " ) , message : NSLocalizedString ( " Please check the public key you entered and try again. " , comment : " " ) , preferredStyle : . alert )
alert . addAction ( UIAlertAction ( title : NSLocalizedString ( " OK " , comment : " " ) , style : . default , handler : nil ) )
presentAlert ( alert )
} else {
let thread = TSContactThread . getOrCreateThread ( contactId : hexEncodedPublicKey )
presentingViewController ? . dismiss ( animated : true , completion : nil )
SignalApp . shared ( ) . presentConversation ( for : thread , action : . compose , animated : false )
}
}
}
private final class EnterPublicKeyVC : UIViewController {
2019-12-04 05:17:06 +01:00
weak var newPrivateChatVC : NewPrivateChatVC !
2019-12-04 03:44:00 +01:00
private lazy var userHexEncodedPublicKey : String = {
2019-12-05 05:28:32 +01:00
if let masterHexEncodedPublicKey = UserDefaults . standard . string ( forKey : " masterDeviceHexEncodedPublicKey " ) {
2019-12-04 03:44:00 +01:00
return masterHexEncodedPublicKey
} else {
return OWSIdentityManager . shared ( ) . identityKeyPair ( ) ! . hexEncodedPublicKey
}
} ( )
// MARK: C o m p o n e n t s
private lazy var publicKeyTextField = TextField ( placeholder : NSLocalizedString ( " Enter public key of recipient " , comment : " " ) )
private lazy var copyButton : Button = {
2019-12-05 00:10:09 +01:00
let result = Button ( style : . unimportant , size : . medium )
2019-12-04 03:44:00 +01:00
result . setTitle ( NSLocalizedString ( " Copy " , comment : " " ) , for : UIControl . State . normal )
result . addTarget ( self , action : #selector ( copyPublicKey ) , for : UIControl . Event . touchUpInside )
return result
} ( )
// MARK: L i f e c y c l e
override func viewDidLoad ( ) {
// R e m o v e b a c k g r o u n d c o l o r
view . backgroundColor = . clear
// S e t u p e x p l a n a t i o n l a b e l
let explanationLabel = UILabel ( )
explanationLabel . textColor = Colors . text . withAlphaComponent ( Values . unimportantElementOpacity )
explanationLabel . font = . systemFont ( ofSize : Values . smallFontSize )
explanationLabel . text = NSLocalizedString ( " Users can share their public key by going into their account settings and tapping \" Share Public Key \" , or by sharing their QR code. " , comment : " " )
explanationLabel . numberOfLines = 0
explanationLabel . textAlignment = . center
explanationLabel . lineBreakMode = . byWordWrapping
// S e t u p s e p a r a t o r
let separator = Separator ( title : NSLocalizedString ( " Your Public Key " , comment : " " ) )
// S e t u p u s e r p u b l i c k e y l a b e l
let userPublicKeyLabel = UILabel ( )
userPublicKeyLabel . textColor = Colors . text
userPublicKeyLabel . font = Fonts . spaceMono ( ofSize : Values . mediumFontSize )
userPublicKeyLabel . numberOfLines = 0
userPublicKeyLabel . textAlignment = . center
userPublicKeyLabel . lineBreakMode = . byCharWrapping
userPublicKeyLabel . text = userHexEncodedPublicKey
// S e t u p s h a r e b u t t o n
2019-12-05 00:10:09 +01:00
let shareButton = Button ( style : . unimportant , size : . medium )
2019-12-04 03:44:00 +01:00
shareButton . setTitle ( NSLocalizedString ( " Share " , comment : " " ) , for : UIControl . State . normal )
shareButton . addTarget ( self , action : #selector ( sharePublicKey ) , for : UIControl . Event . touchUpInside )
// S e t u p b u t t o n c o n t a i n e r
let buttonContainer = UIStackView ( arrangedSubviews : [ copyButton , shareButton ] )
buttonContainer . axis = . horizontal
buttonContainer . spacing = Values . mediumSpacing
buttonContainer . distribution = . fillEqually
// N e x t b u t t o n
2019-12-06 04:42:43 +01:00
let nextButton = Button ( style : . prominentOutline , size : . large )
2019-12-04 03:44:00 +01:00
nextButton . setTitle ( NSLocalizedString ( " Next " , comment : " " ) , for : UIControl . State . normal )
nextButton . addTarget ( self , action : #selector ( startNewPrivateChatIfPossible ) , for : UIControl . Event . touchUpInside )
let nextButtonContainer = UIView ( )
nextButtonContainer . addSubview ( nextButton )
nextButton . pin ( . leading , to : . leading , of : nextButtonContainer , withInset : 80 )
nextButton . pin ( . top , to : . top , of : nextButtonContainer )
nextButtonContainer . pin ( . trailing , to : . trailing , of : nextButton , withInset : 80 )
nextButtonContainer . pin ( . bottom , to : . bottom , of : nextButton )
2019-12-05 05:28:32 +01:00
// S e t u p s t a c k v i e w
2019-12-05 00:10:09 +01:00
let stackView = UIStackView ( arrangedSubviews : [ publicKeyTextField , UIView . spacer ( withHeight : Values . smallSpacing ) , explanationLabel , UIView . spacer ( withHeight : Values . largeSpacing ) , separator , UIView . spacer ( withHeight : Values . veryLargeSpacing ) , userPublicKeyLabel , UIView . spacer ( withHeight : Values . veryLargeSpacing ) , buttonContainer , UIView . vStretchingSpacer ( ) , nextButtonContainer ] )
2019-12-04 03:44:00 +01:00
stackView . axis = . vertical
stackView . alignment = . fill
stackView . layoutMargins = UIEdgeInsets ( top : Values . mediumSpacing , left : Values . largeSpacing , bottom : Values . mediumSpacing , right : Values . largeSpacing )
stackView . isLayoutMarginsRelativeArrangement = true
view . addSubview ( stackView )
stackView . pin ( to : view )
// S e t u p w i d t h c o n s t r a i n t
view . set ( . width , to : UIScreen . main . bounds . width )
// D i s m i s s k e y b o a r d o n t a p
let tapGestureRecognizer = UITapGestureRecognizer ( target : self , action : #selector ( dismissKeyboard ) )
view . addGestureRecognizer ( tapGestureRecognizer )
}
// MARK: G e n e r a l
func constrainHeight ( to height : CGFloat ) {
view . set ( . height , to : height )
}
@objc private func dismissKeyboard ( ) {
publicKeyTextField . resignFirstResponder ( )
}
@objc private func enableCopyButton ( ) {
copyButton . isUserInteractionEnabled = true
UIView . transition ( with : copyButton , duration : 0.25 , options : . transitionCrossDissolve , animations : {
self . copyButton . setTitle ( NSLocalizedString ( " Copy " , comment : " " ) , for : UIControl . State . normal )
} , completion : nil )
}
// MARK: I n t e r a c t i o n
@objc private func copyPublicKey ( ) {
UIPasteboard . general . string = userHexEncodedPublicKey
copyButton . isUserInteractionEnabled = false
UIView . transition ( with : copyButton , duration : 0.25 , options : . transitionCrossDissolve , animations : {
self . copyButton . setTitle ( NSLocalizedString ( " Copied " , comment : " " ) , for : UIControl . State . normal )
} , completion : nil )
Timer . scheduledTimer ( timeInterval : 4 , target : self , selector : #selector ( enableCopyButton ) , userInfo : nil , repeats : false )
}
@objc private func sharePublicKey ( ) {
let shareVC = UIActivityViewController ( activityItems : [ userHexEncodedPublicKey ] , applicationActivities : nil )
2019-12-04 05:17:06 +01:00
newPrivateChatVC . navigationController ! . present ( shareVC , animated : true , completion : nil )
2019-12-04 03:44:00 +01:00
}
@objc private func startNewPrivateChatIfPossible ( ) {
let hexEncodedPublicKey = publicKeyTextField . text ? . trimmingCharacters ( in : . whitespaces ) ? ? " "
2019-12-04 05:17:06 +01:00
newPrivateChatVC . startNewPrivateChatIfPossible ( with : hexEncodedPublicKey )
2019-12-04 03:44:00 +01:00
}
}
private final class ScanQRCodePlaceholderVC : UIViewController {
2019-12-04 05:17:06 +01:00
weak var newPrivateChatVC : NewPrivateChatVC !
2019-12-04 03:44:00 +01:00
override func viewDidLoad ( ) {
// R e m o v e b a c k g r o u n d c o l o r
view . backgroundColor = . clear
// S e t u p e x p l a n a t i o n l a b e l
let explanationLabel = UILabel ( )
explanationLabel . textColor = Colors . text
explanationLabel . font = . systemFont ( ofSize : Values . smallFontSize )
explanationLabel . text = NSLocalizedString ( " Loki Messenger needs camera access to scan QR codes " , comment : " " )
explanationLabel . numberOfLines = 0
explanationLabel . textAlignment = . center
explanationLabel . lineBreakMode = . byWordWrapping
// S e t u p c a l l t o a c t i o n b u t t o n
let callToActionButton = UIButton ( )
callToActionButton . titleLabel ! . font = . boldSystemFont ( ofSize : Values . mediumFontSize )
callToActionButton . setTitleColor ( Colors . accent , for : UIControl . State . normal )
callToActionButton . setTitle ( NSLocalizedString ( " Enable Camera Access " , comment : " " ) , for : UIControl . State . normal )
callToActionButton . addTarget ( self , action : #selector ( requestCameraAccess ) , for : UIControl . Event . touchUpInside )
// S e t u p s t a c k v i e w
let stackView = UIStackView ( arrangedSubviews : [ explanationLabel , callToActionButton ] )
stackView . axis = . vertical
stackView . spacing = Values . mediumSpacing
stackView . alignment = . center
// S e t u p c o n s t r a i n t s
view . set ( . width , to : UIScreen . main . bounds . width )
view . addSubview ( stackView )
stackView . pin ( . leading , to : . leading , of : view , withInset : Values . massiveSpacing )
view . pin ( . trailing , to : . trailing , of : stackView , withInset : Values . massiveSpacing )
let verticalCenteringConstraint = stackView . center ( . vertical , in : view )
verticalCenteringConstraint . constant = - 16 // M a k e s t h i n g s a p p e a r c e n t e r e d v i s u a l l y
}
func constrainHeight ( to height : CGFloat ) {
view . set ( . height , to : height )
}
@objc private func requestCameraAccess ( ) {
ows_ask ( forCameraPermissions : { [ weak self ] hasCameraAccess in
if hasCameraAccess {
2019-12-04 05:17:06 +01:00
self ? . newPrivateChatVC . handleCameraAccessGranted ( )
2019-12-04 03:44:00 +01:00
} else {
// D o n o t h i n g
}
} )
}
}