2020-05-08 17:49:14 +10:00
import UIKit
import Lottie
2020-06-18 17:48:18 +10:00
import KeychainSwift
2020-05-08 17:49:14 +10:00
import SafariServices
class HomeViewController : UIViewController {
private var observer : NSObjectProtocol ?
@IBOutlet weak var screenStack : UIStackView !
@IBOutlet weak var bluetoothStatusOffView : UIView !
@IBOutlet weak var bluetoothStatusOnView : UIView !
@IBOutlet weak var bluetoothPermissionOffView : UIView !
@IBOutlet weak var bluetoothPermissionOnView : UIView !
@IBOutlet weak var homeHeaderView : UIView !
@IBOutlet weak var homeHeaderInfoText : UILabel !
@IBOutlet weak var homeHeaderPermissionsOffImage : UIImageView !
@IBOutlet weak var shareView : UIView !
@IBOutlet weak var appPermissionsLabel : UIView !
@IBOutlet weak var animatedBluetoothHeader : UIView !
@IBOutlet weak var versionNumberLabel : UILabel !
@IBOutlet weak var versionView : UIView !
@IBOutlet weak var uploadView : UIView !
@IBOutlet weak var helpButton : UIButton !
@IBOutlet weak var seeOurFAQ : UIButton !
2020-05-15 00:47:40 -07:00
@IBOutlet weak var pushNotificationStatusTitle : UILabel !
@IBOutlet weak var pushNotificationStatusIcon : UIImageView !
@IBOutlet weak var pushNotificationStatusLabel : UILabel !
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-06-18 17:48:18 +10:00
2020-05-08 17:49:14 +10:00
var lottieBluetoothView : AnimationView !
var allPermissionOn = true
var bluetoothStatusOn = true
var bluetoothPermissionOn = true
var pushNotificationOn = true
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-18 17:48:18 +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 ,
value : UIFont . boldSystemFont ( ofSize : 18 ) ,
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
}
override func viewDidLoad ( ) {
super . viewDidLoad ( )
2020-06-18 17:48:18 +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
self . toggleViews ( )
}
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-06-18 17:48:18 +10:00
" VersionNumber " . 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-06-05 10:26:40 +10:00
let pairingRequestString = NSLocalizedString ( " PairingRequestsInfo " , comment : " Text explaining COVIDSafe does not send pairing requests " )
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 " )
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-05-08 17:49:14 +10:00
}
deinit {
if let observer = observer {
NotificationCenter . default . removeObserver ( observer )
}
}
override func viewWillAppear ( _ animated : Bool ) {
super . viewWillAppear ( animated )
self . toggleViews ( )
}
override func viewDidAppear ( _ animated : Bool ) {
super . viewDidAppear ( animated )
self . lottieBluetoothView ? . play ( )
self . becomeFirstResponder ( )
2020-06-18 17:48:18 +10:00
self . updateJWTKeychainAccess ( )
2020-05-08 17:49:14 +10:00
}
override func viewWillDisappear ( _ animated : Bool ) {
super . viewWillDisappear ( animated )
self . lottieBluetoothView ? . stop ( )
}
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-18 17:48:18 +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-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 ? . togglePushNotificationsStatusView ( )
self ? . toggleBluetoothStatusView ( )
self ? . toggleBluetoothPermissionStatusView ( )
self ? . toggleHeaderView ( )
self ? . toggleUploadView ( )
2020-05-26 17:13:26 +10:00
self ? . toggleUploadDateView ( )
2020-05-08 17:49:14 +10:00
}
} )
}
}
2020-05-26 17:13:26 +10:00
fileprivate func toggleUploadDateView ( ) {
if shouldShowUploadDate , let lastUploadText = self . dataUploadedAttributedString {
uploadDateLabel . attributedText = lastUploadText
uploadDateLabel . isHidden = false
} else {
uploadDateLabel . isHidden = true
}
}
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-05-15 00:47:40 -07:00
self . allPermissionOn = self . bluetoothStatusOn && self . bluetoothPermissionOn
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 ( ) {
toggleViewVisibility ( view : self . uploadView , isVisible : ! self . didUploadData )
}
fileprivate func toggleHeaderView ( ) {
self . allPermissionOn ? self . lottieBluetoothView ? . play ( ) : self . lottieBluetoothView ? . stop ( )
toggleViewVisibility ( view : appPermissionsLabel , isVisible : ! self . allPermissionOn )
toggleViewVisibility ( view : homeHeaderPermissionsOffImage , isVisible : ! self . allPermissionOn )
toggleViewVisibility ( view : lottieBluetoothView , isVisible : self . allPermissionOn )
self . helpButton . setImage ( UIImage ( named : " ic-help-selected " ) , for : . normal )
self . helpButton . setTitleColor ( UIColor . black , for : . normal )
2020-05-15 00:47:40 -07:00
2020-06-18 17:48:18 +10:00
self . homeHeaderInfoText . text = " HomeHeaderNoAction " . localizedString ( comment : " Header with no action req " )
2020-05-15 00:47:40 -07:00
2020-05-08 17:49:14 +10:00
if ( ! self . allPermissionOn ) {
2020-06-18 17:48:18 +10:00
self . homeHeaderInfoText . text = " HomeHeaderPermissions " . localizedString ( comment : " Header with check permisisons text " )
2020-05-08 17:49:14 +10:00
self . homeHeaderView . backgroundColor = UIColor . covidHomePermissionErrorColor
} else {
self . homeHeaderView . backgroundColor = UIColor . covidHomeActiveColor
updateAnimationViewWithAnimationName ( name : " Spinner_home " )
}
}
fileprivate func toggleBluetoothStatusView ( ) {
2020-06-18 17:48:18 +10:00
toggleViewVisibility ( view : bluetoothStatusOnView , isVisible : ! self . bluetoothPermissionOn && self . bluetoothStatusOn )
toggleViewVisibility ( view : bluetoothStatusOffView , isVisible : self . bluetoothPermissionOn && ! self . bluetoothStatusOn )
2020-05-08 17:49:14 +10:00
}
fileprivate func toggleBluetoothPermissionStatusView ( ) {
toggleViewVisibility ( view : bluetoothPermissionOnView , isVisible : ! self . allPermissionOn && self . bluetoothPermissionOn )
toggleViewVisibility ( view : bluetoothPermissionOffView , isVisible : ! self . allPermissionOn && ! self . bluetoothPermissionOn )
}
fileprivate func togglePushNotificationsStatusView ( ) {
2020-05-15 00:47:40 -07:00
let paragraphStyle = NSMutableParagraphStyle ( )
paragraphStyle . lineSpacing = 4
switch self . pushNotificationOn {
case true :
pushNotificationStatusIcon . isHighlighted = false
2020-06-05 10:26:40 +10:00
pushNotificationStatusTitle . text = NSLocalizedString ( " NotificationsEnabled " , comment : " Notifications Enabled " )
2020-06-18 17:48:18 +10:00
let newAttributedLabel = NSMutableAttributedString ( string : NSLocalizedString ( " NotificationsEnabledBlurb " , comment : " Notifications Enabled content blurb " ) , attributes : [ . font : UIFont . preferredFont ( forTextStyle : . callout ) ] )
2020-05-15 00:47:40 -07:00
// s e t s o m e a t t r i b u t e s
2020-06-18 17:48:18 +10:00
guard let linkRange = newAttributedLabel . string . range ( of : " NotificationsBlurbLink " . localizedString ( comment : " Notifications blurb link " ) ) else { return }
2020-05-15 00:47:40 -07:00
let nsRange = NSRange ( linkRange , in : newAttributedLabel . string )
newAttributedLabel . addAttribute ( . foregroundColor ,
value : UIColor . covidSafeColor ,
range : nsRange )
newAttributedLabel . addAttribute ( . paragraphStyle , value : paragraphStyle , range : NSMakeRange ( 0 , newAttributedLabel . length ) )
pushNotificationStatusLabel . attributedText = newAttributedLabel
default :
pushNotificationStatusIcon . isHighlighted = true
2020-06-18 17:48:18 +10:00
pushNotificationStatusTitle . text = " NotificationsDisabled " . localizedString ( comment : " Notifications Enabled " )
2020-06-05 10:26:40 +10:00
let newAttributedLabel = NSMutableAttributedString ( string :
2020-06-18 17:48:18 +10:00
NSLocalizedString ( " NotificationsDisabledBlurb " , comment : " Notifications Enabled content blurb " ) , attributes : [ . font : UIFont . preferredFont ( forTextStyle : . callout ) ] )
2020-05-15 00:47:40 -07:00
// s e t s o m e a t t r i b u t e s
2020-06-18 17:48:18 +10:00
guard let linkRange = newAttributedLabel . string . range ( of : " NotificationsBlurbLink " . localizedString ( ) ) else { return }
2020-05-15 00:47:40 -07:00
let nsRange = NSRange ( linkRange , in : newAttributedLabel . string )
newAttributedLabel . addAttribute ( . foregroundColor ,
value : UIColor . covidSafeColor ,
range : nsRange )
newAttributedLabel . addAttribute ( . paragraphStyle , value : paragraphStyle , range : NSMakeRange ( 0 , newAttributedLabel . length ) )
pushNotificationStatusLabel . attributedText = newAttributedLabel
}
2020-05-08 17:49:14 +10:00
}
2020-06-18 17:48:18 +10:00
func attemptTurnOnBluetooth ( ) {
BluetraceManager . shared . toggleScanning ( false )
BluetraceManager . shared . turnOn ( )
}
// 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-18 17:48:18 +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 )
}
@IBAction func onLatestNewsTapped ( _ sender : UITapGestureRecognizer ) {
// o p e n i n s a f a r i h t t p s : / / w w w . a u s t r a l i a . g o v . a u
guard let url = URL ( string : " https://www.australia.gov.au " ) else {
return
}
let safariVC = SFSafariViewController ( url : url )
present ( safariVC , animated : true , completion : nil )
}
@IBAction func getCoronaVirusApp ( _ sender : UITapGestureRecognizer ) {
guard let url = URL ( string : " https://www.health.gov.au/resources/apps-and-tools/coronavirus-australia-app " ) else {
return
}
let safariVC = SFSafariViewController ( url : url )
present ( safariVC , animated : true , completion : nil )
}
2020-05-26 17:13:26 +10:00
@IBAction func bluetoothPairingTapped ( _ sender : Any ) {
guard let url = URL ( string : " https://www.covidsafe.gov.au/help-topics.html#bluetooth-pairing-request " ) else {
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 ) {
guard let helpVC = UIStoryboard ( name : " UploadData " , bundle : nil ) . instantiateInitialViewController ( ) else {
return
}
let nav = UploadDataNavigationController ( rootViewController : helpVC )
nav . modalTransitionStyle = . coverVertical
nav . modalPresentationStyle = . fullScreen
present ( nav , animated : true , completion : nil )
}
@IBAction func onHelpButtonTapped ( _ sender : Any ) {
let nav = HelpNavController ( )
nav . modalTransitionStyle = . coverVertical
nav . modalPresentationStyle = . fullScreen
present ( nav , animated : true , completion : nil )
}
@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 ( )
}
}
}
struct TracerRemoteConfig {
static let defaultShareText = " " "
2020-06-18 17:48:18 +10:00
\ ( " ShareText " . localizedString ( comment : " Share app with friends text " ) ) # COVID19
2020-05-08 17:49:14 +10:00
# coronavirusaustralia # stayhomesavelives https : // c o v i d s a f e . g o v . a u
" " "
}