2020-06-07 12:22:32 +02:00
|
|
|
//
|
|
|
|
|
// main.swift
|
|
|
|
|
// CPU
|
|
|
|
|
//
|
|
|
|
|
// Created by Serhiy Mytrovtsiy on 09/04/2020.
|
|
|
|
|
// Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved.
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
import Cocoa
|
2021-06-04 19:37:29 +02:00
|
|
|
import Kit
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2020-06-28 22:31:40 +02:00
|
|
|
public struct CPU_Load: value_t {
|
2020-06-07 12:22:32 +02:00
|
|
|
var totalUsage: Double = 0
|
|
|
|
|
var usagePerCore: [Double] = []
|
|
|
|
|
|
|
|
|
|
var systemLoad: Double = 0
|
|
|
|
|
var userLoad: Double = 0
|
|
|
|
|
var idleLoad: Double = 0
|
|
|
|
|
|
2021-05-22 14:58:20 +02:00
|
|
|
public var widgetValue: Double {
|
2020-06-07 12:22:32 +02:00
|
|
|
get {
|
|
|
|
|
return self.totalUsage
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-23 18:40:26 +02:00
|
|
|
public struct CPU_Limit {
|
|
|
|
|
var scheduler: Int = 0
|
|
|
|
|
var cpus: Int = 0
|
|
|
|
|
var speed: Int = 0
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-07 12:22:32 +02:00
|
|
|
public class CPU: Module {
|
2020-07-27 20:57:16 +02:00
|
|
|
private var popupView: Popup
|
2020-06-07 12:22:32 +02:00
|
|
|
private var settingsView: Settings
|
|
|
|
|
|
2020-07-06 19:25:41 +02:00
|
|
|
private var loadReader: LoadReader? = nil
|
2020-07-27 20:57:16 +02:00
|
|
|
private var processReader: ProcessReader? = nil
|
2020-11-24 00:11:02 +01:00
|
|
|
private var temperatureReader: TemperatureReader? = nil
|
|
|
|
|
private var frequencyReader: FrequencyReader? = nil
|
2021-08-23 18:40:26 +02:00
|
|
|
private var limitReader: LimitReader? = nil
|
2021-09-07 19:53:13 +02:00
|
|
|
private var averageReader: AverageReader? = nil
|
2020-07-18 16:05:59 +02:00
|
|
|
|
2022-07-04 18:11:52 +02:00
|
|
|
private var notificationLevelState: Bool = false
|
|
|
|
|
private var notificationID: String? = nil
|
|
|
|
|
|
2020-07-18 16:05:59 +02:00
|
|
|
private var usagePerCoreState: Bool {
|
|
|
|
|
get {
|
2021-03-20 16:33:14 +01:00
|
|
|
return Store.shared.bool(key: "\(self.config.name)_usagePerCore", defaultValue: false)
|
2020-07-18 16:05:59 +02:00
|
|
|
}
|
|
|
|
|
}
|
2021-08-03 20:11:18 +02:00
|
|
|
private var splitValueState: Bool {
|
|
|
|
|
get {
|
|
|
|
|
return Store.shared.bool(key: "\(self.config.name)_splitValue", defaultValue: false)
|
|
|
|
|
}
|
|
|
|
|
}
|
2022-07-04 18:11:52 +02:00
|
|
|
private var notificationLevel: String {
|
|
|
|
|
get {
|
|
|
|
|
return Store.shared.string(key: "\(self.config.name)_notificationLevel", defaultValue: "Disabled")
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2021-03-25 22:14:20 +01:00
|
|
|
public init() {
|
2021-03-20 16:33:14 +01:00
|
|
|
self.settingsView = Settings("CPU")
|
|
|
|
|
self.popupView = Popup("CPU")
|
2020-06-07 12:22:32 +02:00
|
|
|
|
|
|
|
|
super.init(
|
|
|
|
|
popup: self.popupView,
|
|
|
|
|
settings: self.settingsView
|
|
|
|
|
)
|
2020-07-06 19:25:41 +02:00
|
|
|
guard self.available else { return }
|
|
|
|
|
|
|
|
|
|
self.loadReader = LoadReader()
|
2021-03-20 16:33:14 +01:00
|
|
|
self.processReader = ProcessReader()
|
2021-09-07 19:53:13 +02:00
|
|
|
self.averageReader = AverageReader(popup: true)
|
2022-01-14 20:30:05 +01:00
|
|
|
self.temperatureReader = TemperatureReader(popup: true)
|
2020-11-24 00:11:02 +01:00
|
|
|
|
|
|
|
|
#if arch(x86_64)
|
2022-01-14 20:31:16 +01:00
|
|
|
self.limitReader = LimitReader(popup: true)
|
2021-03-25 22:14:20 +01:00
|
|
|
self.frequencyReader = FrequencyReader(popup: true)
|
2020-11-24 00:11:02 +01:00
|
|
|
#endif
|
2020-07-27 20:57:16 +02:00
|
|
|
|
2020-06-24 00:23:06 +02:00
|
|
|
self.settingsView.callback = { [unowned self] in
|
2020-06-28 22:31:40 +02:00
|
|
|
self.loadReader?.read()
|
2020-06-24 00:23:06 +02:00
|
|
|
}
|
2020-10-24 17:30:46 +02:00
|
|
|
self.settingsView.callbackWhenUpdateNumberOfProcesses = {
|
|
|
|
|
self.popupView.numberOfProcessesUpdated()
|
2020-10-29 16:46:22 +01:00
|
|
|
DispatchQueue.global(qos: .background).async {
|
|
|
|
|
self.processReader?.read()
|
|
|
|
|
}
|
2020-10-24 17:30:46 +02:00
|
|
|
}
|
2020-07-10 22:56:47 +02:00
|
|
|
self.settingsView.setInterval = { [unowned self] value in
|
|
|
|
|
self.loadReader?.setInterval(value)
|
|
|
|
|
}
|
2021-07-26 20:32:24 +02:00
|
|
|
self.settingsView.setTopInterval = { [unowned self] value in
|
|
|
|
|
self.processReader?.setInterval(value)
|
|
|
|
|
}
|
2021-03-12 17:51:29 +01:00
|
|
|
self.settingsView.IPGCallback = { [unowned self] value in
|
2021-03-12 17:56:43 +01:00
|
|
|
if value {
|
|
|
|
|
self.frequencyReader?.setup()
|
|
|
|
|
}
|
2021-03-12 17:51:29 +01:00
|
|
|
self.popupView.toggleFrequency(state: value)
|
|
|
|
|
}
|
2020-06-24 00:23:06 +02:00
|
|
|
|
2020-06-28 22:31:40 +02:00
|
|
|
self.loadReader?.callbackHandler = { [unowned self] value in
|
2020-06-07 12:22:32 +02:00
|
|
|
self.loadCallback(value)
|
|
|
|
|
}
|
2021-02-16 20:01:46 +01:00
|
|
|
self.loadReader?.readyCallback = { [unowned self] in
|
|
|
|
|
self.readyHandler()
|
|
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2020-07-27 20:57:16 +02:00
|
|
|
self.processReader?.callbackHandler = { [unowned self] value in
|
|
|
|
|
if let list = value {
|
|
|
|
|
self.popupView.processCallback(list)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-24 00:11:02 +01:00
|
|
|
self.temperatureReader?.callbackHandler = { [unowned self] value in
|
2021-08-24 21:38:14 +02:00
|
|
|
if let v = value {
|
|
|
|
|
self.popupView.temperatureCallback(v)
|
2020-11-24 00:11:02 +01:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self.frequencyReader?.callbackHandler = { [unowned self] value in
|
2021-08-24 21:38:14 +02:00
|
|
|
if let v = value {
|
|
|
|
|
self.popupView.frequencyCallback(v)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self.limitReader?.callbackHandler = { [unowned self] value in
|
|
|
|
|
if let v = value {
|
|
|
|
|
self.popupView.limitCallback(v)
|
2020-10-22 20:50:34 +02:00
|
|
|
}
|
|
|
|
|
}
|
2021-09-07 19:53:13 +02:00
|
|
|
self.averageReader?.callbackHandler = { [unowned self] value in
|
|
|
|
|
if let v = value {
|
|
|
|
|
self.popupView.averageCallback(v)
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-10-22 20:50:34 +02:00
|
|
|
|
2020-07-06 19:25:41 +02:00
|
|
|
if let reader = self.loadReader {
|
|
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
2020-07-27 20:57:16 +02:00
|
|
|
if let reader = self.processReader {
|
|
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
2020-11-24 00:11:02 +01:00
|
|
|
if let reader = self.temperatureReader {
|
|
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
|
|
|
|
if let reader = self.frequencyReader {
|
2020-10-22 20:50:34 +02:00
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
2021-08-23 18:40:26 +02:00
|
|
|
if let reader = self.limitReader {
|
|
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
2021-09-07 19:53:13 +02:00
|
|
|
if let reader = self.averageReader {
|
|
|
|
|
self.addReader(reader)
|
|
|
|
|
}
|
2020-10-14 12:28:37 +02:00
|
|
|
}
|
|
|
|
|
|
2021-02-13 16:39:11 +01:00
|
|
|
private func loadCallback(_ raw: CPU_Load?) {
|
2021-08-15 13:03:28 +02:00
|
|
|
guard let value = raw, self.enabled else {
|
2020-06-07 12:22:32 +02:00
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-13 16:39:11 +01:00
|
|
|
self.popupView.loadCallback(value)
|
2022-07-04 18:11:52 +02:00
|
|
|
self.checkNotificationLevel(value.totalUsage)
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2022-07-09 11:37:00 +02:00
|
|
|
self.menuBar.widgets.filter{ $0.isActive }.forEach { (w: Widget) in
|
2021-02-13 16:39:11 +01:00
|
|
|
switch w.item {
|
|
|
|
|
case let widget as Mini: widget.setValue(value.totalUsage)
|
|
|
|
|
case let widget as LineChart: widget.setValue(value.totalUsage)
|
2021-08-02 20:11:48 +02:00
|
|
|
case let widget as BarChart:
|
|
|
|
|
var val: [[ColorValue]] = [[ColorValue(value.totalUsage)]]
|
|
|
|
|
if self.usagePerCoreState {
|
|
|
|
|
val = value.usagePerCore.map({ [ColorValue($0)] })
|
2021-08-03 20:11:18 +02:00
|
|
|
} else if self.splitValueState {
|
|
|
|
|
val = [[
|
|
|
|
|
ColorValue(value.systemLoad, color: NSColor.systemRed),
|
|
|
|
|
ColorValue(value.userLoad, color: NSColor.systemBlue)
|
|
|
|
|
]]
|
2021-08-02 20:11:48 +02:00
|
|
|
}
|
|
|
|
|
widget.setValue(val)
|
2021-02-13 16:39:11 +01:00
|
|
|
case let widget as PieChart:
|
|
|
|
|
widget.setValue([
|
|
|
|
|
circle_segment(value: value.systemLoad, color: NSColor.systemRed),
|
|
|
|
|
circle_segment(value: value.userLoad, color: NSColor.systemBlue)
|
|
|
|
|
])
|
2021-10-13 18:27:31 +02:00
|
|
|
case let widget as Tachometer:
|
|
|
|
|
widget.setValue([
|
|
|
|
|
circle_segment(value: value.systemLoad, color: NSColor.systemRed),
|
|
|
|
|
circle_segment(value: value.userLoad, color: NSColor.systemBlue)
|
|
|
|
|
])
|
2021-02-13 16:39:11 +01:00
|
|
|
default: break
|
|
|
|
|
}
|
2020-12-01 17:19:56 +01:00
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
}
|
2022-07-04 18:11:52 +02:00
|
|
|
|
|
|
|
|
private func checkNotificationLevel(_ value: Double) {
|
|
|
|
|
guard self.notificationLevel != "Disabled", let level = Double(self.notificationLevel) else { return }
|
|
|
|
|
|
|
|
|
|
if let id = self.notificationID, value < level && self.notificationLevelState {
|
|
|
|
|
if #available(macOS 10.14, *) {
|
|
|
|
|
removeNotification(id)
|
|
|
|
|
} else {
|
|
|
|
|
removeNSNotification(id)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.notificationID = nil
|
|
|
|
|
self.notificationLevelState = false
|
|
|
|
|
} else if value >= level && !self.notificationLevelState {
|
|
|
|
|
let title = localizedString("CPU usage threshold")
|
|
|
|
|
let subtitle = localizedString("CPU usage is", "\(Int((value)*100))%")
|
|
|
|
|
|
|
|
|
|
if #available(macOS 10.14, *) {
|
|
|
|
|
self.notificationID = showNotification(title: title, subtitle: subtitle)
|
|
|
|
|
} else {
|
|
|
|
|
self.notificationID = showNSNotification(title: title, subtitle: subtitle)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
self.notificationLevelState = true
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
}
|