Files
lockdown-iOS-mirror/LockdowniOS/ConnectivityService.swift
2023-05-11 10:54:40 +03:00

87 lines
3.1 KiB
Swift

//
// ConnectivityService.swift
// Lockdown
//
// Created by Aliaksandr Dvoineu on 11.05.23.
// Copyright © 2023 Confirmed Inc. All rights reserved.
//
import CoreTelephony
import SwiftMessages
import Network
import NetworkExtension
final class ConnectivityService {
let noInternetMessageView = MessageView.viewFromNib(layout: .statusLine)
private var connectionState: ConnectionState = .unknown {
didSet { showConnectionErrorIfNeeded() }
}
private var cellularDataRestrictedState = CTCellularDataRestrictedState.restrictedStateUnknown {
didSet { showConnectionErrorIfNeeded() }
}
private var isCellularDataRestricted: Bool { cellularDataRestrictedState == .restricted }
init(connectionState: ConnectionState = .unknown) {
self.connectionState = connectionState
}
func startObservingConnectivity() {
startMonitoringCellularRestriction()
startMonitoringConnection()
}
func showConnectionErrorIfNeeded() {
// When VPN is in a transitioning state (.disconnecting, .connecting), internet is temporarily cut off.
// We should not show any warning in this case.
guard ![.disconnecting, .connecting].contains(NEVPNManager.shared().connection.status) else { return }
DispatchQueue.main.async {
if let errorMessage = self.connectionState.errorMessage {
self.showBanner(text: errorMessage, color: self.connectionState.color)
} else {
SwiftMessages.hideAll()
}
}
}
// MARK: - Private Methods
private func startMonitoringCellularRestriction() {
let cellularState = CTCellularData.init()
cellularState.cellularDataRestrictionDidUpdateNotifier = { (dataRestrictedState) in
self.cellularDataRestrictedState = dataRestrictedState
}
}
private func startMonitoringConnection() {
let monitor = NWPathMonitor()
monitor.pathUpdateHandler = { path in
if path.status == .satisfied {
self.connectionState = .satisfied
} else if path.usesInterfaceType(.cellular), self.isCellularDataRestricted {
self.connectionState = .restrictedCellular
} else {
self.connectionState = .noConnection
}
}
let queue = DispatchQueue(label: "Monitor")
monitor.start(queue: queue)
}
private func showBanner(text: String, color: UIColor) {
noInternetMessageView.backgroundView.backgroundColor = color
noInternetMessageView.bodyLabel?.textColor = .white
noInternetMessageView.configureContent(body: text)
var noInternetMessageViewConfig = SwiftMessages.defaultConfig
noInternetMessageViewConfig.presentationContext = .window(windowLevel: .init(rawValue: 0))
noInternetMessageViewConfig.preferredStatusBarStyle = .lightContent
noInternetMessageViewConfig.duration = .forever
SwiftMessages.show(config: noInternetMessageViewConfig, view: noInternetMessageView)
}
}