2020-05-08 17:49:14 +10:00
import UIKit
import Lottie
2020-06-19 17:43:33 +10:00
import KeychainSwift
2020-05-08 17:49:14 +10:00
import SafariServices
2020-08-18 10:52:17 +10:00
import Reachability
2020-05-08 17:49:14 +10:00
class HomeViewController : UIViewController {
private var observer : NSObjectProtocol ?
2020-11-09 16:49:53 -08:00
private let reauthenticationNeededKey = " ReauthenticationNeededKey "
2020-05-08 17:49:14 +10:00
@IBOutlet weak var bluetoothStatusOffView : UIView !
@IBOutlet weak var bluetoothPermissionOffView : UIView !
2020-11-09 16:49:53 -08:00
@IBOutlet weak var inactiveSettingsContent : UIView !
@IBOutlet weak var inactiveTokenExpiredView : UIView !
2020-05-08 17:49:14 +10:00
@IBOutlet weak var shareView : UIView !
2020-09-14 11:23:11 +10:00
@IBOutlet weak var inactiveAppSectionView : UIView !
@IBOutlet weak var activeAppSectionView : UIView !
@IBOutlet weak var covidInactiveLabel : UILabel !
@IBOutlet weak var covidActiveLabel : UILabel !
2020-05-08 17:49:14 +10:00
@IBOutlet weak var animatedBluetoothHeader : UIView !
@IBOutlet weak var versionNumberLabel : UILabel !
@IBOutlet weak var versionView : UIView !
@IBOutlet weak var uploadView : UIView !
2020-09-14 11:23:11 +10:00
@IBOutlet weak var uploadDateView : UIView !
2020-05-26 17:13:26 +10:00
@IBOutlet weak var uploadDateLabel : UILabel !
2020-06-05 10:26:40 +10:00
@IBOutlet weak var pairingRequestsLabel : UILabel !
2020-09-14 11:23:11 +10:00
@IBOutlet weak var appActiveSubtitleLabel : UILabel !
@IBOutlet weak var uploadDataContentLabel : UILabel !
@IBOutlet weak var uploadDataTitleLabel : UILabel !
2020-11-09 16:49:53 -08:00
@IBOutlet weak var covidStatisticsSection : UIView !
2020-09-14 11:23:11 +10:00
@IBOutlet weak var covidStatisticsContainer : UIView !
@IBOutlet weak var contentStackView : UIStackView !
@IBOutlet weak var statisticsSectionHeight : NSLayoutConstraint !
@IBOutlet weak var scrollView : UIScrollView !
var appActiveSubtitleLabelInitialColor : UIColor ?
2020-07-21 15:42:48 +10:00
2020-05-08 17:49:14 +10:00
var lottieBluetoothView : AnimationView !
2020-09-14 11:23:11 +10:00
let covidStatisticsViewController : CovidStatisticsViewController = CovidStatisticsViewController ( nibName : " CovidStatisticsView " , bundle : nil )
2020-05-08 17:49:14 +10:00
var allPermissionOn = true
2020-11-09 16:49:53 -08:00
var registrationNeeded : Bool {
return UserDefaults . standard . bool ( forKey : reauthenticationNeededKey )
}
2020-05-08 17:49:14 +10:00
var bluetoothStatusOn = true
var bluetoothPermissionOn = true
var pushNotificationOn = true
2020-09-14 11:23:11 +10:00
var shouldShowUpdateApp = false
2020-05-08 17:49:14 +10:00
var didUploadData : Bool {
let uploadTimestamp = UserDefaults . standard . double ( forKey : " uploadDataDate " )
let lastUpload = Date ( timeIntervalSince1970 : uploadTimestamp )
2020-05-26 17:13:26 +10:00
return Date ( ) . timeIntervalSince ( lastUpload ) < 86400 * 14
2020-05-08 17:49:14 +10:00
}
2020-05-26 17:13:26 +10:00
var dataUploadedAttributedString : NSAttributedString ? {
let uploadTimestamp = UserDefaults . standard . double ( forKey : " uploadDataDate " )
if ( uploadTimestamp > 0 ) {
let lastUpload = Date ( timeIntervalSince1970 : uploadTimestamp )
let dateFormatterPrint = DateFormatter ( )
dateFormatterPrint . dateFormat = " dd MMM yyyy "
let formattedDate = dateFormatterPrint . string ( from : lastUpload )
2020-06-05 10:26:40 +10:00
let newAttributedString = NSMutableAttributedString (
string : String . localizedStringWithFormat (
2020-06-19 17:43:33 +10:00
" InformationUploaded " . localizedString ( comment : " Information uploaded template " ) ,
2020-06-05 10:26:40 +10:00
formattedDate )
)
2020-05-26 17:13:26 +10:00
guard let dateRange = newAttributedString . string . range ( of : formattedDate ) else { return nil }
let nsRange = NSRange ( dateRange , in : newAttributedString . string )
newAttributedString . addAttribute ( . font ,
2020-09-14 11:23:11 +10:00
value : UIFont . preferredFont ( for : . body , weight : . bold ) ,
2020-05-26 17:13:26 +10:00
range : nsRange )
return newAttributedString
}
return nil
}
var shouldShowUploadDate : Bool {
let uploadTimestamp = UserDefaults . standard . double ( forKey : " uploadDataDate " )
2020-05-08 17:49:14 +10:00
if ( uploadTimestamp > 0 ) {
let lastUpload = Date ( timeIntervalSince1970 : uploadTimestamp )
2020-05-26 17:13:26 +10:00
return Date ( ) . timeIntervalSince ( lastUpload ) <= 86400 * 14
2020-05-08 17:49:14 +10:00
}
return false
}
var _preferredScreenEdgesDeferringSystemGestures : UIRectEdge = [ ]
override var preferredScreenEdgesDeferringSystemGestures : UIRectEdge {
return _preferredScreenEdgesDeferringSystemGestures
}
2020-08-18 10:52:17 +10:00
private let reachability = try ! Reachability ( )
2020-05-08 17:49:14 +10:00
override func viewDidLoad ( ) {
super . viewDidLoad ( )
2020-09-14 11:23:11 +10:00
scrollView . refreshControl = UIRefreshControl ( )
scrollView . refreshControl ? . addTarget ( self , action : #selector ( refreshControlEvent ) , for : . valueChanged )
2020-06-19 17:43:33 +10:00
// t h i s i s t o s h o w t h e s e t t i n g s p r o m p t i n i t i a l l y i f b l u e t o o t h i s o f f
if ! BluetraceManager . shared . isBluetoothOn ( ) {
BluetraceManager . shared . turnOn ( )
}
2020-05-08 17:49:14 +10:00
observer = NotificationCenter . default . addObserver ( forName : UIApplication . didBecomeActiveNotification , object : nil , queue : . main ) { [ unowned self ] notification in
2020-09-14 11:23:11 +10:00
self . refreshView ( )
2020-05-08 17:49:14 +10:00
}
updateAnimationViewWithAnimationName ( name : " Spinner_home " )
NotificationCenter . default . addObserver ( self , selector : #selector ( enableDeferringSystemGestures ( _ : ) ) , name : . enableDeferringSystemGestures , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( disableDeferringSystemGestures ( _ : ) ) , name : . disableDeferringSystemGestures , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( disableUserInteraction ( _ : ) ) , name : . disableUserInteraction , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( enableUserInteraction ( _ : ) ) , name : . enableUserInteraction , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( appWillResignActive ( _ : ) ) , name : UIApplication . willResignActiveNotification , object : nil )
NotificationCenter . default . addObserver ( self , selector : #selector ( appWillEnterForeground ( _ : ) ) , name : UIApplication . willEnterForegroundNotification , object : nil )
2020-05-26 17:13:26 +10:00
2020-05-08 17:49:14 +10:00
if let versionNumber = Bundle . main . versionShort , let buildNumber = Bundle . main . version {
2020-06-05 10:26:40 +10:00
self . versionNumberLabel . text = String . localizedStringWithFormat (
2020-07-03 14:26:13 +10:00
" home_version_number_ios " . localizedString ( comment : " Version number template " ) ,
2020-06-05 10:26:40 +10:00
versionNumber , buildNumber
)
2020-05-08 17:49:14 +10:00
} else {
toggleViewVisibility ( view : versionView , isVisible : false )
}
2020-07-03 14:26:13 +10:00
// S o m e t r a n s l a t o r s a r e a d d i n g t h e * * f o r t h i s l i n k , j u s t c l e a n i n g t h a t u p .
let pairingRequestString = NSLocalizedString ( " PairingRequestsInfo " , comment : " Text explaining COVIDSafe does not send pairing requests " ) . replacingOccurrences ( of : " * " , with : " " )
2020-06-05 10:26:40 +10:00
let pairingRequestText = NSMutableAttributedString ( string : pairingRequestString ,
attributes : [ . font : UIFont . preferredFont ( forTextStyle : . body ) ] )
let pairingRequestUnderlinedString = NSLocalizedString ( " PairingRequestsInfoUnderline " , comment : " section of text that should be underlined from the PairingRequestsInfo text " )
2020-07-03 14:26:13 +10:00
if let requestsRange = pairingRequestText . string . range ( of : pairingRequestUnderlinedString ) {
let nsRange = NSRange ( requestsRange , in : pairingRequestText . string )
pairingRequestText . addAttribute ( . underlineStyle , value : NSUnderlineStyle . single . rawValue , range : nsRange )
pairingRequestsLabel . attributedText = pairingRequestText
}
2020-09-14 11:23:11 +10:00
uploadDataTitleLabel . font = UIFont . preferredFont ( for : . title3 , weight : . bold )
uploadDataContentLabel . font = UIFont . preferredFont ( for : . callout , weight : . bold )
covidInactiveLabel . font = UIFont . preferredFont ( for : . title3 , weight : . bold )
appActiveSubtitleLabelInitialColor = appActiveSubtitleLabel . textColor
setupStatisticsView ( )
getStatistics ( )
2020-05-08 17:49:14 +10:00
}
deinit {
if let observer = observer {
NotificationCenter . default . removeObserver ( observer )
}
2020-08-18 10:52:17 +10:00
reachability . stopNotifier ( )
NotificationCenter . default . removeObserver ( self , name : . reachabilityChanged , object : reachability )
2020-05-08 17:49:14 +10:00
}
override func viewWillAppear ( _ animated : Bool ) {
super . viewWillAppear ( animated )
2020-08-18 10:52:17 +10:00
NotificationCenter . default . addObserver ( self , selector : #selector ( reachabilityChanged ( note : ) ) , name : . reachabilityChanged , object : reachability )
do {
try reachability . startNotifier ( )
} catch {
DLog ( " Could not start reachability notifier " )
}
2020-05-08 17:49:14 +10:00
self . toggleViews ( )
2020-09-14 11:23:11 +10:00
if ! UserDefaults . standard . bool ( forKey : " PerformHealthChecks " ) {
DispatchQueue . global ( qos : . background ) . async {
self . getMessagesFromServer ( )
}
}
performHealthCheck ( )
covidStatisticsViewController . statisticsDelegate = self
2020-05-08 17:49:14 +10:00
}
override func viewDidAppear ( _ animated : Bool ) {
super . viewDidAppear ( animated )
self . lottieBluetoothView ? . play ( )
self . becomeFirstResponder ( )
2020-06-19 17:43:33 +10:00
self . updateJWTKeychainAccess ( )
2020-10-16 15:45:07 +11:00
if shouldShowPolicyUpdateMessage ( ) {
showPolicyUpdateMessage ( )
}
2020-05-08 17:49:14 +10:00
}
override func viewWillDisappear ( _ animated : Bool ) {
super . viewWillDisappear ( animated )
self . lottieBluetoothView ? . stop ( )
2020-08-18 10:52:17 +10:00
reachability . stopNotifier ( )
NotificationCenter . default . removeObserver ( self , name : . reachabilityChanged , object : reachability )
2020-05-08 17:49:14 +10:00
}
override var canBecomeFirstResponder : Bool {
return true
}
override func motionEnded ( _ motion : UIEvent . EventSubtype , with event : UIEvent ? ) {
#if DEBUG
guard let allowDebug = PlistHelper . getBoolFromInfoPlist ( withKey : " Allow_Debug " , plistName : " CovidSafe-config " ) else {
return
}
if allowDebug = = true && event ? . subtype = = . motionShake {
guard let debugVC = UIStoryboard ( name : " Debug " , bundle : nil ) . instantiateInitialViewController ( ) else {
return
}
debugVC . modalTransitionStyle = . coverVertical
debugVC . modalPresentationStyle = . fullScreen
present ( debugVC , animated : true , completion : nil )
}
#endif
}
2020-06-19 17:43:33 +10:00
func updateJWTKeychainAccess ( ) {
let hasUpdatedKeychainAccess = UserDefaults . standard . bool ( forKey : " HasUpdatedKeychainAccess " )
if ( ! hasUpdatedKeychainAccess ) {
let keychain = KeychainSwift ( )
if let jwt = keychain . get ( " JWT_TOKEN " ) {
if ( keychain . set ( jwt , forKey : " JWT_TOKEN " , withAccess : . accessibleAfterFirstUnlock ) ) {
DLog ( " Updated access class on JWT " )
UserDefaults . standard . set ( true , forKey : " HasUpdatedKeychainAccess " )
}
}
}
}
2020-09-14 11:23:11 +10:00
fileprivate func refreshView ( ) {
toggleViews ( )
performHealthCheck ( )
getStatistics ( )
}
fileprivate func setupStatisticsView ( ) {
addChild ( covidStatisticsViewController )
covidStatisticsViewController . view . translatesAutoresizingMaskIntoConstraints = false
covidStatisticsContainer . addSubview ( covidStatisticsViewController . view )
2020-07-21 15:42:48 +10:00
2020-09-14 11:23:11 +10:00
NSLayoutConstraint . activate ( [
covidStatisticsViewController . view . leadingAnchor . constraint ( equalTo : covidStatisticsContainer . leadingAnchor ) ,
covidStatisticsViewController . view . trailingAnchor . constraint ( equalTo : covidStatisticsContainer . trailingAnchor ) ,
covidStatisticsViewController . view . topAnchor . constraint ( equalTo : covidStatisticsContainer . topAnchor ) ,
covidStatisticsViewController . view . bottomAnchor . constraint ( equalTo : covidStatisticsContainer . bottomAnchor )
] )
covidStatisticsViewController . didMove ( toParent : self )
2020-07-21 15:42:48 +10:00
}
2020-06-19 17:43:33 +10:00
2020-05-08 17:49:14 +10:00
fileprivate func toggleViews ( ) {
DispatchQueue . main . async {
UNUserNotificationCenter . current ( ) . getNotificationSettings ( completionHandler : { [ weak self ] settings in
DispatchQueue . main . async {
self ? . readPermissions ( notificationSettings : settings )
self ? . toggleBluetoothStatusView ( )
self ? . toggleBluetoothPermissionStatusView ( )
self ? . toggleHeaderView ( )
self ? . toggleUploadView ( )
2020-05-26 17:13:26 +10:00
self ? . toggleUploadDateView ( )
2020-11-09 16:49:53 -08:00
self ? . toggleShareView ( )
self ? . toggleStatisticsView ( )
self ? . toggleRegistrationNeededView ( )
2020-05-08 17:49:14 +10:00
}
} )
}
}
2020-09-14 11:23:11 +10:00
fileprivate func performHealthCheck ( ) {
if UserDefaults . standard . bool ( forKey : " PerformHealthChecks " ) {
UserDefaults . standard . set ( false , forKey : " PerformHealthChecks " )
guard allPermissionOn else {
// i f a l l p e r m i s s i o n n o t O N , s t a y i n h o m e s c r e e n
return
}
getMessagesFromServer ( force : true ) {
if ( self . reachability . connection != . cellular && self . reachability . connection != . wifi ) ||
self . shouldShowUpdateApp {
DispatchQueue . main . async {
self . onSettingsTapped ( self )
}
} else if self . allPermissionOn &&
self . isInternetReachable ( ) &&
! self . shouldShowUpdateApp {
DispatchQueue . main . async {
self . covidActiveLabel . text = " home_header_active_title_thanks " . localizedString ( )
}
}
}
}
}
fileprivate func isInternetReachable ( ) -> Bool {
return reachability . connection = = . cellular || reachability . connection = = . wifi
}
2020-05-26 17:13:26 +10:00
fileprivate func toggleUploadDateView ( ) {
if shouldShowUploadDate , let lastUploadText = self . dataUploadedAttributedString {
uploadDateLabel . attributedText = lastUploadText
2020-09-14 11:23:11 +10:00
uploadDateView . isHidden = false
2020-05-26 17:13:26 +10:00
} else {
2020-09-14 11:23:11 +10:00
uploadDateView . isHidden = true
2020-05-26 17:13:26 +10:00
}
}
2020-05-08 17:49:14 +10:00
fileprivate func readPermissions ( notificationSettings : UNNotificationSettings ) {
self . bluetoothStatusOn = BluetraceManager . shared . isBluetoothOn ( )
self . bluetoothPermissionOn = BluetraceManager . shared . isBluetoothAuthorized ( )
self . pushNotificationOn = notificationSettings . authorizationStatus = = . authorized
2020-09-14 11:23:11 +10:00
let newAllPermissionsOn = self . bluetoothStatusOn && self . bluetoothPermissionOn
if newAllPermissionsOn != self . allPermissionOn {
self . allPermissionOn = newAllPermissionsOn
DispatchQueue . global ( qos : . background ) . async {
self . getMessagesFromServer ( force : true )
}
}
2020-05-08 17:49:14 +10:00
}
fileprivate func toggleViewVisibility ( view : UIView , isVisible : Bool ) {
view . isHidden = ! isVisible
}
func updateAnimationViewWithAnimationName ( name : String ) {
let bluetoothAnimation = AnimationView ( name : name )
bluetoothAnimation . loopMode = . loop
bluetoothAnimation . frame = CGRect ( origin : CGPoint ( x : 0 , y : 0 ) , size : self . animatedBluetoothHeader . frame . size )
if lottieBluetoothView != nil {
lottieBluetoothView . stop ( )
lottieBluetoothView . removeFromSuperview ( )
}
self . animatedBluetoothHeader . addSubview ( bluetoothAnimation )
lottieBluetoothView = bluetoothAnimation
lottieBluetoothView . play ( )
}
fileprivate func toggleUploadView ( ) {
2020-11-09 16:49:53 -08:00
toggleViewVisibility ( view : self . uploadView , isVisible : ! self . didUploadData && ! self . registrationNeeded )
}
fileprivate func toggleShareView ( ) {
toggleViewVisibility ( view : shareView , isVisible : ! registrationNeeded )
}
fileprivate func toggleStatisticsView ( ) {
toggleViewVisibility ( view : covidStatisticsSection , isVisible : ! registrationNeeded )
2020-05-08 17:49:14 +10:00
}
fileprivate func toggleHeaderView ( ) {
2020-11-09 16:49:53 -08:00
toggleViewVisibility ( view : inactiveAppSectionView , isVisible : ! self . allPermissionOn || registrationNeeded )
toggleViewVisibility ( view : inactiveSettingsContent , isVisible : ! self . allPermissionOn && ! registrationNeeded )
2020-09-14 11:23:11 +10:00
toggleViewVisibility ( view : activeAppSectionView , isVisible : self . allPermissionOn )
2020-05-08 17:49:14 +10:00
}
fileprivate func toggleBluetoothStatusView ( ) {
2020-11-09 16:49:53 -08:00
toggleViewVisibility ( view : bluetoothStatusOffView , isVisible : self . bluetoothPermissionOn && ! self . bluetoothStatusOn && ! registrationNeeded )
2020-05-08 17:49:14 +10:00
}
fileprivate func toggleBluetoothPermissionStatusView ( ) {
2020-11-09 16:49:53 -08:00
toggleViewVisibility ( view : bluetoothPermissionOffView , isVisible : ! self . allPermissionOn && ! self . bluetoothPermissionOn && ! registrationNeeded )
}
fileprivate func toggleRegistrationNeededView ( ) {
toggleViewVisibility ( view : inactiveTokenExpiredView , isVisible : registrationNeeded )
2020-05-08 17:49:14 +10:00
}
2020-09-14 11:23:11 +10:00
func attemptTurnOnBluetooth ( ) {
BluetraceManager . shared . toggleScanning ( false )
BluetraceManager . shared . turnOn ( )
2020-05-08 17:49:14 +10:00
}
2020-09-14 11:23:11 +10:00
fileprivate func updateAppActiveSubtitle ( ) {
let haveInternet = self . reachability . connection = = . cellular || self . reachability . connection = = . wifi
if shouldShowUpdateApp || ! haveInternet {
appActiveSubtitleLabel . font = UIFont . preferredFont ( for : . callout , weight : . bold )
appActiveSubtitleLabel . textColor = UIColor . covidSafeErrorColor
appActiveSubtitleLabel . text = " improve " . localizedString ( )
} else {
appActiveSubtitleLabel . font = UIFont . preferredFont ( for : . callout , weight : . regular )
appActiveSubtitleLabel . textColor = appActiveSubtitleLabelInitialColor
appActiveSubtitleLabel . text = " home_header_active_no_action_required " . localizedString ( )
}
2020-08-18 10:52:17 +10:00
}
2020-10-16 15:45:07 +11:00
// MARK: p o l i c y u p d a t e m e s s a g e
func shouldShowPolicyUpdateMessage ( ) -> Bool {
// t h i s i s t h e m i n v e r s i o n t h a t t h e d i s c l a m e r s h o u l d b e d i p l a y e d o n .
let minVersionShowPolicyUpdate = 77
let latestVersionShown = UserDefaults . standard . integer ( forKey : " latestPolicyUpdateVersionShown " )
guard let currentVersion = ( Bundle . main . version as NSString ? ) ? . integerValue else {
return false
}
if currentVersion >= minVersionShowPolicyUpdate && currentVersion > latestVersionShown {
UserDefaults . standard . set ( currentVersion , forKey : " latestPolicyUpdateVersionShown " )
return true
}
return false
}
func showPolicyUpdateMessage ( ) {
let privacyPolicyUrl = URLHelper . getPrivacyPolicyURL ( )
let disclaimerAlert = CSAlertViewController ( nibName : " CSAlertView " , bundle : nil )
let disclaimerMsg = NSMutableAttributedString ( string : " collection_message " . localizedString ( ) , attributes : [ . font : UIFont . preferredFont ( forTextStyle : . body ) ] )
disclaimerMsg . addLink ( enclosedIn : " * " , urlString : privacyPolicyUrl )
disclaimerAlert . set ( message : disclaimerMsg , buttonLabel : " dismiss " . localizedString ( ) )
disclaimerAlert . modalPresentationStyle = UIModalPresentationStyle . overCurrentContext
disclaimerAlert . modalTransitionStyle = UIModalTransitionStyle . crossDissolve
present ( disclaimerAlert , animated : true , completion : nil )
}
2020-11-09 16:49:53 -08:00
func showTokenExpiredMessage ( ) {
UserDefaults . standard . set ( true , forKey : reauthenticationNeededKey )
toggleViews ( )
}
2020-08-18 10:52:17 +10:00
2020-11-09 16:49:53 -08:00
// MARK: A P I c a l l s
2020-09-14 11:23:11 +10:00
func getMessagesFromServer ( force : Bool = false , completion : @ escaping ( ) -> Void = { } ) {
2020-11-09 16:49:53 -08:00
let onMessagesDone : ( MessageResponse ? , CovidSafeAPIError ? ) -> Void = { ( messageResponse , error ) in
2020-09-14 11:23:11 +10:00
if let error = error {
DLog ( " Get messages error: \( error . localizedDescription ) " )
2020-11-09 16:49:53 -08:00
if error = = . TokenExpiredError {
self . showTokenExpiredMessage ( )
}
2020-09-14 11:23:11 +10:00
completion ( )
return
}
// s h o w u p d a t e a v a i l a b l e s e c t i o n
guard let messages = messageResponse ? . messages else {
self . shouldShowUpdateApp = false
DispatchQueue . main . async {
self . updateAppActiveSubtitle ( )
}
completion ( )
return
}
self . shouldShowUpdateApp = messages . count > 0
DispatchQueue . main . async {
self . updateAppActiveSubtitle ( )
}
NotificationCenter . default . post ( name : . shouldUpdateAppFromMessages , object : nil )
completion ( )
}
if force {
MessageAPI . getMessages ( completion : onMessagesDone )
} else {
MessageAPI . getMessagesIfNeeded ( completion : onMessagesDone )
}
2020-08-18 10:52:17 +10:00
}
2020-09-14 11:23:11 +10:00
func getStatistics ( ) {
2020-10-16 15:45:07 +11:00
if covidStatisticsViewController . showStatistics {
self . covidStatisticsViewController . isLoading = true
StatisticsAPI . getStatistics { ( stats , error ) in
2020-11-09 16:49:53 -08:00
if error != nil {
switch error {
case . TokenExpiredError :
self . showTokenExpiredMessage ( )
default :
return
}
}
2020-10-16 15:45:07 +11:00
self . covidStatisticsViewController . isLoading = false
self . covidStatisticsViewController . setupData ( statistics : stats , errorType : error , hasInternet : self . isInternetReachable ( ) )
2020-11-09 16:49:53 -08:00
2020-10-16 15:45:07 +11:00
}
2020-09-14 11:23:11 +10:00
}
2020-06-19 17:43:33 +10:00
}
2020-08-18 10:52:17 +10:00
// MARK: R e a c h a b i l i t y
@objc func reachabilityChanged ( note : Notification ) {
let reachability = note . object as ! Reachability
switch reachability . connection {
case . wifi ,
. cellular :
2020-09-14 11:23:11 +10:00
updateAppActiveSubtitle ( )
2020-08-18 10:52:17 +10:00
case . unavailable ,
. none :
2020-09-14 11:23:11 +10:00
updateAppActiveSubtitle ( )
2020-08-18 10:52:17 +10:00
}
}
2020-06-19 17:43:33 +10:00
// MARK: I B A c t i o n s
@IBAction func onAppSettingsTapped ( _ sender : UITapGestureRecognizer ) {
2020-05-08 17:49:14 +10:00
guard let settingsURL = URL ( string : UIApplication . openSettingsURLString ) else {
return
}
UIApplication . shared . open ( settingsURL )
}
2020-06-19 17:43:33 +10:00
2020-08-18 10:52:17 +10:00
@IBAction func noInternetTapped ( _ sender : Any ) {
performSegue ( withIdentifier : " internetConnectionSegue " , sender : nil )
}
@IBAction func updateAvailableTapped ( _ sender : Any ) {
if let url = URL ( string : " itms-apps://itunes.apple.com/app/id1509242894 " ) ,
UIApplication . shared . canOpenURL ( url ) {
UIApplication . shared . open ( url , options : [ : ] , completionHandler : nil )
2020-08-03 16:01:39 +10:00
}
2020-08-18 10:52:17 +10:00
}
@IBAction func onChangeLanguageTapped ( _ sender : UITapGestureRecognizer ) {
let nav = HelpNavController ( )
nav . pageSectionId = " other-languages "
nav . modalTransitionStyle = . coverVertical
nav . modalPresentationStyle = . fullScreen
present ( nav , animated : true , completion : nil )
2020-08-03 16:01:39 +10:00
}
2020-06-19 17:43:33 +10:00
@IBAction func onBluetoothPhoneSettingsTapped ( _ sender : Any ) {
attemptTurnOnBluetooth ( )
}
2020-05-08 17:49:14 +10:00
@IBAction func onShareTapped ( _ sender : UITapGestureRecognizer ) {
let shareText = TracerRemoteConfig . defaultShareText
let activity = UIActivityViewController ( activityItems : [ shareText ] , applicationActivities : nil )
activity . popoverPresentationController ? . sourceView = shareView
present ( activity , animated : true , completion : nil )
}
2020-05-26 17:13:26 +10:00
@IBAction func bluetoothPairingTapped ( _ sender : Any ) {
2020-07-03 14:26:13 +10:00
guard let url = URL ( string : " \( URLHelper . getHelpURL ( ) ) #bluetooth-pairing-request " ) else {
2020-05-26 17:13:26 +10:00
return
}
let safariVC = SFSafariViewController ( url : url )
present ( safariVC , animated : true , completion : nil )
}
2020-05-08 17:49:14 +10:00
@IBAction func onPositiveButtonTapped ( _ sender : UITapGestureRecognizer ) {
2020-10-16 15:45:07 +11:00
guard let uploadVC = UIStoryboard ( name : " UploadData " , bundle : nil ) . instantiateInitialViewController ( ) else {
2020-05-08 17:49:14 +10:00
return
}
2020-10-16 15:45:07 +11:00
navigationController ? . pushViewController ( uploadVC , animated : true )
2020-05-08 17:49:14 +10:00
}
@IBAction func onHelpButtonTapped ( _ sender : Any ) {
let nav = HelpNavController ( )
nav . modalTransitionStyle = . coverVertical
nav . modalPresentationStyle = . fullScreen
present ( nav , animated : true , completion : nil )
}
2020-09-14 11:23:11 +10:00
@IBAction func improvementAvailableTapped ( _ sender : Any ) {
if shouldShowUpdateApp || ! isInternetReachable ( ) {
onSettingsTapped ( sender )
}
}
@IBAction func onSettingsTapped ( _ sender : Any ) {
let settingsVC = SettingsViewController ( nibName : " SettingsView " , bundle : nil )
settingsVC . showUpdateAvailable = shouldShowUpdateApp
2020-10-16 15:45:07 +11:00
navigationController ? . pushViewController ( settingsVC , animated : true )
2020-09-14 11:23:11 +10:00
}
2020-11-09 16:49:53 -08:00
@IBAction func registerAgainTapped ( _ sender : Any ) {
guard let regVC = UIStoryboard ( name : " Main " , bundle : nil ) . instantiateViewController ( withIdentifier : " personalDetails " ) as ? PersonalDetailsViewController else {
return
}
regVC . reauthenticating = true
let navigationController = UINavigationController ( rootViewController : regVC )
navigationController . setToolbarHidden ( true , animated : false )
navigationController . isNavigationBarHidden = true
navigationController . modalPresentationStyle = . fullScreen
navigationController . modalTransitionStyle = . coverVertical
present ( navigationController , animated : true , completion : nil )
}
2020-05-08 17:49:14 +10:00
@objc
func appWillResignActive ( _ notification : Notification ) {
self . lottieBluetoothView ? . stop ( )
}
@objc
func appWillEnterForeground ( _ notification : Notification ) {
self . lottieBluetoothView ? . play ( )
}
@objc
func enableUserInteraction ( _ notification : Notification ) {
self . view . isUserInteractionEnabled = true
lottieBluetoothView ? . play ( )
}
@objc
func disableUserInteraction ( _ notification : Notification ) {
self . view . isUserInteractionEnabled = false
lottieBluetoothView ? . stop ( )
}
@objc
func enableDeferringSystemGestures ( _ notification : Notification ) {
if #available ( iOS 11.0 , * ) {
_preferredScreenEdgesDeferringSystemGestures = . bottom
setNeedsUpdateOfScreenEdgesDeferringSystemGestures ( )
}
}
@objc
func disableDeferringSystemGestures ( _ notification : Notification ) {
if #available ( iOS 11.0 , * ) {
_preferredScreenEdgesDeferringSystemGestures = [ ]
setNeedsUpdateOfScreenEdgesDeferringSystemGestures ( )
}
}
2020-09-14 11:23:11 +10:00
@objc
func refreshControlEvent ( ) {
refreshView ( )
DispatchQueue . main . async {
self . scrollView . refreshControl ? . endRefreshing ( )
}
}
}
// MARK: S t a t i s t i c s d e l e g a t e
extension HomeViewController : StatisticsDelegate {
func refreshStatistics ( ) {
self . getStatistics ( )
}
func setStatisticsContainerHeight ( height : CGFloat ) {
self . statisticsSectionHeight . constant = height
UIView . animate ( withDuration : 0.3 ) {
self . view . layoutIfNeeded ( )
}
}
2020-05-08 17:49:14 +10:00
}
struct TracerRemoteConfig {
static let defaultShareText = " " "
2020-07-03 14:26:13 +10:00
\ ( " share_this_app_content " . localizedString ( comment : " Share app with friends text " ) )
2020-05-08 17:49:14 +10:00
" " "
}