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
|
2024-07-02 21:09:37 +02:00
|
|
|
import WidgetKit
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2025-04-05 22:54:01 +02:00
|
|
|
public struct CPU_Load: Codable, RemoteType {
|
2025-07-25 19:03:28 +02:00
|
|
|
public var totalUsage: Double = 0
|
2020-06-07 12:22:32 +02:00
|
|
|
var usagePerCore: [Double] = []
|
2022-08-13 17:48:54 +02:00
|
|
|
var usageECores: Double? = nil
|
|
|
|
|
var usagePCores: Double? = nil
|
2020-06-07 12:22:32 +02:00
|
|
|
|
|
|
|
|
var systemLoad: Double = 0
|
|
|
|
|
var userLoad: Double = 0
|
|
|
|
|
var idleLoad: Double = 0
|
2025-04-05 22:54:01 +02:00
|
|
|
|
|
|
|
|
public func remote() -> Data? {
|
|
|
|
|
var string = "1,1,\(self.totalUsage),\(self.usagePerCore.count),"
|
|
|
|
|
for c in self.usagePerCore {
|
|
|
|
|
string += "\(c),"
|
|
|
|
|
}
|
|
|
|
|
string += "$"
|
|
|
|
|
return string.data(using: .utf8)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public struct CPU_Frequency: Codable, RemoteType {
|
|
|
|
|
var value: Double = 0
|
|
|
|
|
var eCore: Double = 0
|
|
|
|
|
var pCore: Double = 0
|
|
|
|
|
|
|
|
|
|
public func remote() -> Data? {
|
|
|
|
|
let string = "1,1,\(self.value)$"
|
|
|
|
|
return string.data(using: .utf8)
|
|
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
}
|
|
|
|
|
|
2023-06-27 17:09:38 +02:00
|
|
|
public struct CPU_Limit: Codable {
|
2021-08-23 18:40:26 +02:00
|
|
|
var scheduler: Int = 0
|
|
|
|
|
var cpus: Int = 0
|
|
|
|
|
var speed: Int = 0
|
|
|
|
|
}
|
|
|
|
|
|
2025-09-27 19:28:42 +02:00
|
|
|
public struct CPU_AverageLoad: Codable, RemoteType {
|
|
|
|
|
var load1: Double = 0
|
|
|
|
|
var load5: Double = 0
|
|
|
|
|
var load15: Double = 0
|
|
|
|
|
|
|
|
|
|
public func remote() -> Data? {
|
|
|
|
|
let string = "1,1,\(self.load1),\(self.load5),\(self.load15)$"
|
|
|
|
|
return string.data(using: .utf8)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-07 12:22:32 +02:00
|
|
|
public class CPU: Module {
|
2023-02-24 22:43:29 +01:00
|
|
|
private let popupView: Popup
|
|
|
|
|
private let settingsView: Settings
|
|
|
|
|
private let portalView: Portal
|
2023-12-12 20:16:27 +01:00
|
|
|
private let notificationsView: Notifications
|
2020-06-07 12:22:32 +02:00
|
|
|
|
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
|
2025-09-27 19:28:42 +02:00
|
|
|
private var averageLoadReader: AverageLoadReader? = nil
|
2022-07-04 18:11:52 +02:00
|
|
|
|
2020-07-18 16:05:59 +02:00
|
|
|
private var usagePerCoreState: Bool {
|
2022-10-25 19:34:30 +02:00
|
|
|
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 {
|
2022-10-25 19:34:30 +02:00
|
|
|
Store.shared.bool(key: "\(self.config.name)_splitValue", defaultValue: false)
|
2021-08-03 20:11:18 +02:00
|
|
|
}
|
2022-10-14 17:24:35 +02:00
|
|
|
private var groupByClustersState: Bool {
|
2022-10-25 19:34:30 +02:00
|
|
|
Store.shared.bool(key: "\(self.config.name)_clustersGroup", defaultValue: false)
|
2022-10-14 17:24:35 +02:00
|
|
|
}
|
2022-10-25 19:34:30 +02:00
|
|
|
private var systemColor: NSColor {
|
2024-07-04 19:52:52 +02:00
|
|
|
let color = SColor.secondRed
|
2022-10-25 19:34:30 +02:00
|
|
|
let key = Store.shared.string(key: "\(self.config.name)_systemColor", defaultValue: color.key)
|
2024-07-04 19:52:52 +02:00
|
|
|
if let c = SColor.fromString(key).additional as? NSColor {
|
2022-10-25 19:34:30 +02:00
|
|
|
return c
|
|
|
|
|
}
|
|
|
|
|
return color.additional as! NSColor
|
|
|
|
|
}
|
|
|
|
|
private var userColor: NSColor {
|
2024-07-04 19:52:52 +02:00
|
|
|
let color = SColor.secondBlue
|
2023-04-22 15:46:41 +02:00
|
|
|
let key = Store.shared.string(key: "\(self.config.name)_userColor", defaultValue: color.key)
|
2024-07-04 19:52:52 +02:00
|
|
|
if let c = SColor.fromString(key).additional as? NSColor {
|
2022-10-25 19:34:30 +02:00
|
|
|
return c
|
2022-07-04 18:11:52 +02:00
|
|
|
}
|
2022-10-25 19:34:30 +02:00
|
|
|
return color.additional as! NSColor
|
2022-07-04 18:11:52 +02:00
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2023-11-25 22:31:50 +01:00
|
|
|
private var eCoreColor: NSColor {
|
2024-07-04 19:52:52 +02:00
|
|
|
let color = SColor.teal
|
2023-11-25 22:31:50 +01:00
|
|
|
let key = Store.shared.string(key: "\(self.config.name)_eCoresColor", defaultValue: color.key)
|
2024-07-04 19:52:52 +02:00
|
|
|
if let c = SColor.fromString(key).additional as? NSColor {
|
2023-11-25 22:31:50 +01:00
|
|
|
return c
|
|
|
|
|
}
|
|
|
|
|
return color.additional as! NSColor
|
|
|
|
|
}
|
|
|
|
|
private var pCoreColor: NSColor {
|
2024-12-20 17:07:27 +01:00
|
|
|
let color = SColor.indigo
|
2023-11-25 22:31:50 +01:00
|
|
|
let key = Store.shared.string(key: "\(self.config.name)_pCoresColor", defaultValue: color.key)
|
2024-07-04 19:52:52 +02:00
|
|
|
if let c = SColor.fromString(key).additional as? NSColor {
|
2023-11-25 22:31:50 +01:00
|
|
|
return c
|
|
|
|
|
}
|
|
|
|
|
return color.additional as! NSColor
|
|
|
|
|
}
|
|
|
|
|
|
2025-12-03 19:02:40 +01:00
|
|
|
private var systemWidgetsUpdatesState: Bool {
|
2026-02-14 18:39:49 +01:00
|
|
|
self.userDefaults?.bool(forKey: "systemWidgetsUpdates_state") ?? false
|
2025-12-03 19:02:40 +01:00
|
|
|
}
|
|
|
|
|
|
2021-03-25 22:14:20 +01:00
|
|
|
public init() {
|
2024-02-10 17:03:47 +01:00
|
|
|
self.settingsView = Settings(.CPU)
|
|
|
|
|
self.popupView = Popup(.CPU)
|
2024-01-20 19:05:12 +01:00
|
|
|
self.portalView = Portal(.CPU)
|
2023-12-12 20:16:27 +01:00
|
|
|
self.notificationsView = Notifications(.CPU)
|
2020-06-07 12:22:32 +02:00
|
|
|
|
|
|
|
|
super.init(
|
2024-12-13 12:49:30 +01:00
|
|
|
moduleType: .CPU,
|
2020-06-07 12:22:32 +02:00
|
|
|
popup: self.popupView,
|
2023-02-24 22:43:29 +01:00
|
|
|
settings: self.settingsView,
|
2023-12-12 20:16:27 +01:00
|
|
|
portal: self.portalView,
|
|
|
|
|
notifications: self.notificationsView
|
2020-06-07 12:22:32 +02:00
|
|
|
)
|
2020-07-06 19:25:41 +02:00
|
|
|
guard self.available else { return }
|
|
|
|
|
|
2024-02-10 17:03:47 +01:00
|
|
|
self.loadReader = LoadReader(.CPU) { [weak self] value in
|
|
|
|
|
self?.loadCallback(value)
|
|
|
|
|
}
|
|
|
|
|
self.processReader = ProcessReader(.CPU) { [weak self] value in
|
|
|
|
|
self?.popupView.processCallback(value)
|
|
|
|
|
}
|
2025-09-27 19:28:42 +02:00
|
|
|
self.averageLoadReader = AverageLoadReader(.CPU, popup: true) { [weak self] value in
|
2024-02-10 17:03:47 +01:00
|
|
|
self?.popupView.averageCallback(value)
|
|
|
|
|
}
|
|
|
|
|
self.temperatureReader = TemperatureReader(.CPU, popup: true) { [weak self] value in
|
|
|
|
|
self?.popupView.temperatureCallback(value)
|
|
|
|
|
}
|
2020-11-24 00:11:02 +01:00
|
|
|
|
|
|
|
|
#if arch(x86_64)
|
2024-02-10 17:03:47 +01:00
|
|
|
self.limitReader = LimitReader(.CPU, popup: true) { [weak self] value in
|
|
|
|
|
self?.popupView.limitCallback(value)
|
|
|
|
|
}
|
2024-12-20 17:07:27 +01:00
|
|
|
#else
|
|
|
|
|
self.frequencyReader = FrequencyReader(.CPU, popup: false) { [weak self] value in
|
2024-02-10 17:03:47 +01:00
|
|
|
self?.popupView.frequencyCallback(value)
|
|
|
|
|
}
|
2020-11-24 00:11:02 +01:00
|
|
|
#endif
|
2020-07-27 20:57:16 +02:00
|
|
|
|
2023-08-13 13:15:27 +02:00
|
|
|
self.settingsView.callback = { [weak self] in
|
|
|
|
|
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
|
|
|
}
|
2023-08-13 13:15:27 +02:00
|
|
|
self.settingsView.setInterval = { [weak self] value in
|
|
|
|
|
self?.loadReader?.setInterval(value)
|
2020-07-10 22:56:47 +02:00
|
|
|
}
|
2023-08-13 13:15:27 +02:00
|
|
|
self.settingsView.setTopInterval = { [weak self] value in
|
|
|
|
|
self?.processReader?.setInterval(value)
|
2021-07-26 20:32:24 +02:00
|
|
|
}
|
2020-06-24 00:23:06 +02:00
|
|
|
|
2024-02-10 17:03:47 +01:00
|
|
|
self.setReaders([
|
|
|
|
|
self.loadReader,
|
|
|
|
|
self.processReader,
|
|
|
|
|
self.temperatureReader,
|
|
|
|
|
self.frequencyReader,
|
|
|
|
|
self.limitReader,
|
2025-09-27 19:28:42 +02:00
|
|
|
self.averageLoadReader
|
2024-02-10 17:03:47 +01:00
|
|
|
])
|
2020-10-14 12:28:37 +02:00
|
|
|
}
|
|
|
|
|
|
2021-02-13 16:39:11 +01:00
|
|
|
private func loadCallback(_ raw: CPU_Load?) {
|
2023-08-13 13:15:27 +02:00
|
|
|
guard let value = raw, self.enabled else { return }
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2021-02-13 16:39:11 +01:00
|
|
|
self.popupView.loadCallback(value)
|
2024-01-20 19:05:12 +01:00
|
|
|
self.portalView.callback(value)
|
2023-12-12 20:16:27 +01:00
|
|
|
self.notificationsView.loadCallback(value)
|
2020-06-07 12:22:32 +02:00
|
|
|
|
2025-04-11 18:53:05 +02:00
|
|
|
self.menuBar.widgets.filter{ $0.isActive }.forEach { [self] (w: SWidget) 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)]]
|
2023-11-25 22:31:50 +01:00
|
|
|
let cores = SystemKit.shared.device.info.cpu?.cores ?? []
|
|
|
|
|
|
2021-08-02 20:11:48 +02:00
|
|
|
if self.usagePerCoreState {
|
2023-11-25 22:31:50 +01:00
|
|
|
if widget.colorState == .cluster {
|
|
|
|
|
val = []
|
|
|
|
|
for (i, v) in value.usagePerCore.enumerated() {
|
|
|
|
|
let core = cores.first(where: {$0.id == i })
|
|
|
|
|
val.append([ColorValue(v, color: core?.type == .efficiency ? self.eCoreColor : self.pCoreColor)])
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
val = value.usagePerCore.map({ [ColorValue($0)] })
|
|
|
|
|
}
|
2021-08-03 20:11:18 +02:00
|
|
|
} else if self.splitValueState {
|
|
|
|
|
val = [[
|
2022-10-25 19:34:30 +02:00
|
|
|
ColorValue(value.systemLoad, color: self.systemColor),
|
|
|
|
|
ColorValue(value.userLoad, color: self.userColor)
|
2021-08-03 20:11:18 +02:00
|
|
|
]]
|
2022-10-14 17:24:35 +02:00
|
|
|
} else if self.groupByClustersState, let e = value.usageECores, let p = value.usagePCores {
|
2023-11-25 22:31:50 +01:00
|
|
|
if widget.colorState == .cluster {
|
|
|
|
|
val = [
|
|
|
|
|
[ColorValue(e, color: self.eCoreColor)],
|
|
|
|
|
[ColorValue(p, color: self.pCoreColor)]
|
|
|
|
|
]
|
|
|
|
|
} else {
|
|
|
|
|
val = [[ColorValue(e)], [ColorValue(p)]]
|
|
|
|
|
}
|
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([
|
2022-10-25 19:34:30 +02:00
|
|
|
circle_segment(value: value.systemLoad, color: self.systemColor),
|
|
|
|
|
circle_segment(value: value.userLoad, color: self.userColor)
|
2021-02-13 16:39:11 +01:00
|
|
|
])
|
2021-10-13 18:27:31 +02:00
|
|
|
case let widget as Tachometer:
|
|
|
|
|
widget.setValue([
|
2022-10-25 19:34:30 +02:00
|
|
|
circle_segment(value: value.systemLoad, color: self.systemColor),
|
|
|
|
|
circle_segment(value: value.userLoad, color: self.userColor)
|
2021-10-13 18:27:31 +02:00
|
|
|
])
|
2021-02-13 16:39:11 +01:00
|
|
|
default: break
|
|
|
|
|
}
|
2020-12-01 17:19:56 +01:00
|
|
|
}
|
2024-07-08 13:14:50 +02:00
|
|
|
|
2025-12-03 19:02:40 +01:00
|
|
|
if self.systemWidgetsUpdatesState {
|
2026-02-22 15:52:36 +01:00
|
|
|
if isWidgetActive(self.userDefaults, [CPU_entry.kind, "UnitedWidget"]), let blobData = try? JSONEncoder().encode(value) {
|
|
|
|
|
self.userDefaults?.set(blobData, forKey: "CPU@LoadReader")
|
2025-11-14 11:12:06 +01:00
|
|
|
}
|
2026-02-22 15:52:36 +01:00
|
|
|
WidgetCenter.shared.reloadTimelines(ofKind: CPU_entry.kind)
|
|
|
|
|
WidgetCenter.shared.reloadTimelines(ofKind: "UnitedWidget")
|
2024-07-08 13:14:50 +02:00
|
|
|
}
|
2020-06-07 12:22:32 +02:00
|
|
|
}
|
|
|
|
|
}
|