mirror of
https://github.com/morgan9e/macos-stats
synced 2026-04-13 15:54:10 +09:00
326 lines
16 KiB
Swift
326 lines
16 KiB
Swift
//
|
|
// settings.swift
|
|
// Net
|
|
//
|
|
// Created by Serhiy Mytrovtsiy on 06/07/2020.
|
|
// Using Swift 5.0.
|
|
// Running on macOS 10.15.
|
|
//
|
|
// Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved.
|
|
//
|
|
|
|
import Cocoa
|
|
import Kit
|
|
import SystemConfiguration
|
|
|
|
var textWidgetHelp = """
|
|
<h2>Description</h2>
|
|
You can use a combination of any of the variables. There is only one limitation: there must be a space between each variable.
|
|
<h3>Examples:</h3>
|
|
<ul>
|
|
<li>$addr.public - $status</li>
|
|
<li>$addr.public - $wifi.ssid - $status</li>
|
|
</ul>
|
|
<h2>Available variables</h2>
|
|
<ul>
|
|
<li><b>$addr.public</b>: <small>Public IP address.</small></li>
|
|
<li><b>$addr.publicV4</b>: <small>Public IPv4 address.</small></li>
|
|
<li><b>$addr.publicV6</b>: <small>Public IPv6 address.</small></li>
|
|
<li><b>$addr.private</b>: <small>Private/local IP address.</small></li>
|
|
<li><b>$interface.displayName</b>: <small>Network interface name.</small></li>
|
|
<li><b>$interface.BSDName</b>: <small>BSD name of the network interface.</small></li>
|
|
<li><b>$interface.address</b>: <small>MAC address of the network interface.</small></li>
|
|
<li><b>$wifi.ssid</b>: <small>Wi-Fi network name.</small></li>
|
|
<li><b>$wifi.bssid</b>: <small>MAC address of the Wi-Fi access point (BSSID).</small></li>
|
|
<li><b>$wifi.RSSI</b>: <small>Signal strength of the Wi-Fi network (RSSI).</small></li>
|
|
<li><b>$wifi.noise</b>: <small>Noise level of the Wi-Fi network.</small></li>
|
|
<li><b>$wifi.transmitRate</b>: <small>Transmit rate (connection speed) of the Wi-Fi network.</small></li>
|
|
<li><b>$wifi.standard</b>: <small>Wi-Fi standard (e.g., 802.11a/b/g/n/ac).</small></li>
|
|
<li><b>$wifi.mode</b>: <small>Operating mode of the Wi-Fi (e.g., infrastructure, adhoc).</small></li>
|
|
<li><b>$wifi.security</b>: <small>Type of security used by the Wi-Fi network.</small></li>
|
|
<li><b>$wifi.channel</b>: <small>Wi-Fi channel being used.</small></li>
|
|
<li><b>$wifi.channelBand</b>: <small>Frequency band of the Wi-Fi channel (e.g., 2.4 GHz, 5 GHz).</small></li>
|
|
<li><b>$wifi.channelWidth</b>: <small>Channel width used in MHz.</small></li>
|
|
<li><b>$wifi.channelNumber</b>: <small>Channel number used by the Wi-Fi network.</small></li>
|
|
<li><b>$status</b>: <small>Status of the network connection. "UP" if active, "DOWN" if inactive.</small></li>
|
|
<li><b>$upload.total</b>: <small>Total amount of data uploaded over the connection.</small></li>
|
|
<li><b>$upload</b>: <small>Current upload bandwidth used.</small></li>
|
|
<li><b>$download.total</b>: <small>Total amount of data downloaded over the connection.</small></li>
|
|
<li><b>$download</b>: <small>Current download bandwidth used.</small></li>
|
|
<li><b>$type</b>: <small>Type of network connection (e.g., Ethernet, Wi-Fi, Cellular).</small></li>
|
|
</ul>
|
|
"""
|
|
|
|
internal class Settings: NSStackView, Settings_v, NSTextFieldDelegate {
|
|
private var numberOfProcesses: Int = 8
|
|
private var readerType: String = "interface"
|
|
private var usageReset: String = AppUpdateInterval.never.rawValue
|
|
private var VPNModeState: Bool = false
|
|
private var widgetActivationThresholdState: Bool = false
|
|
private var widgetActivationThreshold: Int = 0
|
|
private var widgetActivationThresholdSize: SizeUnit = .MB
|
|
private var ICMPHost: String = "1.1.1.1"
|
|
private var publicIPRefreshInterval: String = "never"
|
|
private var baseValue: String = "byte"
|
|
private var textValue: String = "$addr.public - $status"
|
|
|
|
public var callback: (() -> Void) = {}
|
|
public var callbackWhenUpdateNumberOfProcesses: (() -> Void) = {}
|
|
public var usageResetCallback: (() -> Void) = {}
|
|
public var ICMPHostCallback: ((_ newState: Bool) -> Void) = { _ in }
|
|
public var publicIPRefreshIntervalCallback: (() -> Void) = {}
|
|
|
|
private let title: String
|
|
private var sliderView: NSView? = nil
|
|
private var section: PreferencesSection? = nil
|
|
private var widgetThresholdSection: PreferencesSection? = nil
|
|
private let textWidgetHelpPanel: HelpHUD = HelpHUD(textWidgetHelp)
|
|
|
|
private var list: [Network_interface] = []
|
|
|
|
private var vpnConnection: Bool {
|
|
if let settings = CFNetworkCopySystemProxySettings()?.takeRetainedValue() as? [String: Any], let scopes = settings["__SCOPED__"] as? [String: Any] {
|
|
return !scopes.filter({ $0.key.contains("tap") || $0.key.contains("tun") || $0.key.contains("ppp") || $0.key.contains("ipsec") || $0.key.contains("ipsec0")}).isEmpty
|
|
}
|
|
return false
|
|
}
|
|
|
|
public init(_ module: ModuleType) {
|
|
self.title = module.stringValue
|
|
self.numberOfProcesses = Store.shared.int(key: "\(self.title)_processes", defaultValue: self.numberOfProcesses)
|
|
self.readerType = Store.shared.string(key: "\(self.title)_reader", defaultValue: self.readerType)
|
|
self.usageReset = Store.shared.string(key: "\(self.title)_usageReset", defaultValue: self.usageReset)
|
|
self.VPNModeState = Store.shared.bool(key: "\(self.title)_VPNMode", defaultValue: self.VPNModeState)
|
|
self.widgetActivationThresholdState = Store.shared.bool(key: "\(self.title)_widgetActivationThresholdState", defaultValue: self.widgetActivationThresholdState)
|
|
self.widgetActivationThreshold = Store.shared.int(key: "\(self.title)_widgetActivationThreshold", defaultValue: self.widgetActivationThreshold)
|
|
self.widgetActivationThresholdSize = SizeUnit.fromString(Store.shared.string(key: "\(self.title)_widgetActivationThresholdSize", defaultValue: self.widgetActivationThresholdSize.key))
|
|
self.ICMPHost = Store.shared.string(key: "\(self.title)_ICMPHost", defaultValue: self.ICMPHost)
|
|
self.publicIPRefreshInterval = Store.shared.string(key: "\(self.title)_publicIPRefreshInterval", defaultValue: self.publicIPRefreshInterval)
|
|
self.baseValue = Store.shared.string(key: "\(self.title)_base", defaultValue: self.baseValue)
|
|
self.textValue = Store.shared.string(key: "\(self.title)_textWidgetValue", defaultValue: self.textValue)
|
|
|
|
super.init(frame: NSRect.zero)
|
|
self.orientation = .vertical
|
|
self.spacing = Constants.Settings.margin
|
|
|
|
for interface in SCNetworkInterfaceCopyAll() as NSArray {
|
|
if let bsdName = SCNetworkInterfaceGetBSDName(interface as! SCNetworkInterface),
|
|
let displayName = SCNetworkInterfaceGetLocalizedDisplayName(interface as! SCNetworkInterface) {
|
|
self.list.append(Network_interface(displayName: displayName as String, BSDName: bsdName as String))
|
|
}
|
|
}
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
public func load(widgets: [widget_t]) {
|
|
self.subviews.forEach{ $0.removeFromSuperview() }
|
|
|
|
self.addArrangedSubview(PreferencesSection([
|
|
PreferencesRow(localizedString("Number of top processes"), component: selectView(
|
|
action: #selector(self.changeNumberOfProcesses),
|
|
items: NumbersOfProcesses.map{ KeyValue_t(key: "\($0)", value: "\($0)") },
|
|
selected: "\(self.numberOfProcesses)"
|
|
))
|
|
]))
|
|
|
|
let interfaces = selectView(
|
|
action: #selector(self.handleSelection),
|
|
items: [],
|
|
selected: ""
|
|
)
|
|
let selectedInterface = Store.shared.string(key: "\(self.title)_interface", defaultValue: "")
|
|
let menu = NSMenu()
|
|
let autodetection = NSMenuItem(title: localizedString("Autodetection"), action: nil, keyEquivalent: "")
|
|
autodetection.identifier = NSUserInterfaceItemIdentifier(rawValue: "autodetection")
|
|
autodetection.tag = 128
|
|
menu.addItem(autodetection)
|
|
menu.addItem(NSMenuItem.separator())
|
|
self.list.forEach { (interface: Network_interface) in
|
|
let interfaceMenu = NSMenuItem(title: "\(interface.displayName) (\(interface.BSDName))", action: nil, keyEquivalent: "")
|
|
interfaceMenu.identifier = NSUserInterfaceItemIdentifier(rawValue: interface.BSDName)
|
|
menu.addItem(interfaceMenu)
|
|
if selectedInterface != "" && selectedInterface == interface.BSDName {
|
|
interfaceMenu.state = .on
|
|
}
|
|
}
|
|
interfaces.menu = menu
|
|
interfaces.sizeToFit()
|
|
if selectedInterface == "" {
|
|
interfaces.selectItem(withTag: 128)
|
|
}
|
|
|
|
var prefs: [PreferencesRow] = [
|
|
PreferencesRow(localizedString("Reader type"), component: selectView(
|
|
action: #selector(self.changeReaderType),
|
|
items: NetworkReaders,
|
|
selected: self.readerType
|
|
)),
|
|
PreferencesRow(localizedString("Network interface"), component: interfaces),
|
|
PreferencesRow(localizedString("Base"), component: selectView(
|
|
action: #selector(self.toggleBase),
|
|
items: SpeedBase,
|
|
selected: self.baseValue
|
|
)),
|
|
PreferencesRow(localizedString("Reset data usage"), component: selectView(
|
|
action: #selector(self.toggleUsageReset),
|
|
items: AppUpdateIntervals.filter({ $0.key != "Silent" }),
|
|
selected: self.usageReset
|
|
)),
|
|
PreferencesRow(localizedString("Auto-refresh public IP address"), component: selectView(
|
|
action: #selector(self.toggleRefreshIPInterval),
|
|
items: PublicIPAddressRefreshIntervals,
|
|
selected: self.publicIPRefreshInterval
|
|
))
|
|
]
|
|
if self.vpnConnection {
|
|
prefs.append(PreferencesRow(localizedString("VPN mode"), component: switchView(
|
|
action: #selector(self.toggleVPNMode),
|
|
state: self.VPNModeState
|
|
)))
|
|
}
|
|
let section = PreferencesSection(prefs)
|
|
section.setRowVisibility(1, newState: self.readerType == "interface")
|
|
self.addArrangedSubview(section)
|
|
self.section = section
|
|
|
|
self.widgetThresholdSection = PreferencesSection([
|
|
PreferencesRow(localizedString("Widget activation threshold"), component: PreferencesSwitch(
|
|
action: self.toggleWidgetActivationThreshold, state: self.widgetActivationThresholdState, with: StepperInput(
|
|
self.widgetActivationThreshold, range: NSRange(location: 1, length: 1023),
|
|
unit: self.widgetActivationThresholdSize.key, units: SizeUnit.allCases,
|
|
callback: self.changeWidgetActivationThreshold, unitCallback: self.toggleWidgetActivationThresholdSize
|
|
)
|
|
))
|
|
])
|
|
self.addArrangedSubview(self.widgetThresholdSection!)
|
|
self.widgetThresholdSection?.setRowVisibility(1, newState: self.widgetActivationThresholdState)
|
|
|
|
let valueField: NSTextField = NSTextField()
|
|
valueField.widthAnchor.constraint(equalToConstant: 250).isActive = true
|
|
valueField.font = NSFont.systemFont(ofSize: 12, weight: .regular)
|
|
valueField.textColor = .textColor
|
|
valueField.isEditable = true
|
|
valueField.isSelectable = true
|
|
valueField.usesSingleLineMode = true
|
|
valueField.maximumNumberOfLines = 1
|
|
valueField.focusRingType = .none
|
|
valueField.stringValue = self.ICMPHost
|
|
valueField.delegate = self
|
|
valueField.placeholderString = localizedString("Leave empty to disable the check")
|
|
|
|
let ICMPField = self.inputField(id: "ICMP", value: self.ICMPHost, placeholder: localizedString("Leave empty to disable the check"))
|
|
self.addArrangedSubview(PreferencesSection([
|
|
PreferencesRow(localizedString("Connectivity host (ICMP)"), component: ICMPField) {
|
|
NSWorkspace.shared.open(URL(string: "https://en.wikipedia.org/wiki/Internet_Control_Message_Protocol")!)
|
|
}
|
|
]))
|
|
|
|
if widgets.contains(where: { $0 == .text }) {
|
|
let textField = self.inputField(id: "text", value: self.textValue, placeholder: localizedString("This will be visible in the text widget"))
|
|
self.addArrangedSubview(PreferencesSection([
|
|
PreferencesRow(localizedString("Text widget value"), component: textField) { [weak self] in
|
|
self?.textWidgetHelpPanel.show()
|
|
}
|
|
]))
|
|
}
|
|
}
|
|
|
|
private func inputField(id: String, value: String, placeholder: String) -> NSView {
|
|
let field: NSTextField = NSTextField()
|
|
field.identifier = NSUserInterfaceItemIdentifier(id)
|
|
field.widthAnchor.constraint(equalToConstant: 250).isActive = true
|
|
field.font = NSFont.systemFont(ofSize: 12, weight: .regular)
|
|
field.textColor = .textColor
|
|
field.isEditable = true
|
|
field.isSelectable = true
|
|
field.usesSingleLineMode = true
|
|
field.maximumNumberOfLines = 1
|
|
field.focusRingType = .none
|
|
field.stringValue = value
|
|
field.delegate = self
|
|
field.placeholderString = placeholder
|
|
return field
|
|
}
|
|
|
|
@objc private func handleSelection(_ sender: NSPopUpButton) {
|
|
guard let item = sender.selectedItem, let id = item.identifier?.rawValue else { return }
|
|
|
|
if id == "autodetection" {
|
|
Store.shared.remove("\(self.title)_interface")
|
|
} else {
|
|
if let bsdName = item.identifier?.rawValue {
|
|
Store.shared.set(key: "\(self.title)_interface", value: bsdName)
|
|
}
|
|
}
|
|
|
|
self.callback()
|
|
}
|
|
@objc private func changeNumberOfProcesses(_ sender: NSMenuItem) {
|
|
if let value = Int(sender.title) {
|
|
self.numberOfProcesses = value
|
|
Store.shared.set(key: "\(self.title)_processes", value: value)
|
|
self.callbackWhenUpdateNumberOfProcesses()
|
|
}
|
|
}
|
|
@objc private func changeReaderType(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.readerType = key
|
|
Store.shared.set(key: "\(self.title)_reader", value: key)
|
|
self.section?.setRowVisibility(1, newState: self.readerType == "interface")
|
|
NotificationCenter.default.post(name: .resetTotalNetworkUsage, object: nil, userInfo: nil)
|
|
}
|
|
@objc private func toggleUsageReset(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.usageReset = key
|
|
Store.shared.set(key: "\(self.title)_usageReset", value: key)
|
|
self.usageResetCallback()
|
|
}
|
|
@objc func toggleVPNMode(_ sender: NSControl) {
|
|
self.VPNModeState = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_VPNMode", value: self.VPNModeState)
|
|
}
|
|
@objc func toggleWidgetActivationThreshold(_ sender: NSControl) {
|
|
self.widgetActivationThresholdState = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_widgetActivationThresholdState", value: self.widgetActivationThresholdState)
|
|
self.widgetThresholdSection?.setRowVisibility(1, newState: self.widgetActivationThresholdState)
|
|
}
|
|
@objc private func changeWidgetActivationThreshold(_ newValue: Int) {
|
|
self.widgetActivationThreshold = newValue
|
|
Store.shared.set(key: "\(self.title)_widgetActivationThreshold", value: newValue)
|
|
}
|
|
private func toggleWidgetActivationThresholdSize(_ newValue: KeyValue_p) {
|
|
guard let newUnit = newValue as? SizeUnit else { return }
|
|
self.widgetActivationThresholdSize = newUnit
|
|
Store.shared.set(key: "\(self.title)_widgetActivationThresholdSize", value: self.widgetActivationThresholdSize.key)
|
|
self.display()
|
|
}
|
|
|
|
func controlTextDidChange(_ notification: Notification) {
|
|
if let field = notification.object as? NSTextField {
|
|
if field.identifier == NSUserInterfaceItemIdentifier("ICMP") {
|
|
self.ICMPHost = field.stringValue
|
|
Store.shared.set(key: "\(self.title)_ICMPHost", value: self.ICMPHost)
|
|
self.ICMPHostCallback(self.ICMPHost.isEmpty)
|
|
} else if field.identifier == NSUserInterfaceItemIdentifier("text") {
|
|
self.textValue = field.stringValue
|
|
Store.shared.set(key: "\(self.title)_textWidgetValue", value: self.textValue)
|
|
}
|
|
}
|
|
}
|
|
|
|
@objc private func toggleRefreshIPInterval(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.publicIPRefreshInterval = key
|
|
Store.shared.set(key: "\(self.title)_publicIPRefreshInterval", value: self.publicIPRefreshInterval)
|
|
self.publicIPRefreshIntervalCallback()
|
|
}
|
|
@objc private func toggleBase(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.baseValue = key
|
|
Store.shared.set(key: "\(self.title)_base", value: self.baseValue)
|
|
}
|
|
}
|