Files
macos-stats/Modules/RAM/widget.swift

132 lines
5.3 KiB
Swift

//
// widget.swift
// RAM
//
// Created by Serhiy Mytrovtsiy on 03/07/2024
// Using Swift 5.0
// Running on macOS 14.5
//
// Copyright © 2024 Serhiy Mytrovtsiy. All rights reserved.
//
import SwiftUI
import WidgetKit
import Charts
import Kit
public struct RAM_entry: TimelineEntry {
public static let kind = "RAMWidget"
public static var snapshot: RAM_entry = RAM_entry(value:
RAM_Usage(
total: 34359738368.0,
used: 18993741824.0,
free: 15365996544.0,
active: 14518042624.0,
inactive: 13899530240.0,
wired: 2209333248.0,
compressed: 414629888.0,
app: 16369778688.0,
cache: 12575948800.0,
swap: Swap(total: 0, used: 0, free: 0),
pressure: Pressure(level: 1, value: .normal),
swapins: 14,
swapouts: 16
)
)
public var date: Date {
Calendar.current.date(byAdding: .second, value: 5, to: Date())!
}
public var value: RAM_Usage? = nil
}
@available(macOS 11.0, *)
public struct Provider: TimelineProvider {
public typealias Entry = RAM_entry
private let userDefaults: UserDefaults? = UserDefaults(suiteName: "\(Bundle.main.object(forInfoDictionaryKey: "TeamId") as! String).eu.exelban.Stats.widgets")
public func placeholder(in context: Context) -> RAM_entry {
RAM_entry()
}
public func getSnapshot(in context: Context, completion: @escaping (RAM_entry) -> Void) {
completion(RAM_entry.snapshot)
}
public func getTimeline(in context: Context, completion: @escaping (Timeline<RAM_entry>) -> Void) {
self.userDefaults?.set(Date().timeIntervalSince1970, forKey: RAM_entry.kind)
var entry = RAM_entry()
if let raw = userDefaults?.data(forKey: "RAM@UsageReader"), let load = try? JSONDecoder().decode(RAM_Usage.self, from: raw) {
entry.value = load
}
let entries: [RAM_entry] = [entry]
completion(Timeline(entries: entries, policy: .atEnd))
}
}
@available(macOS 14.0, *)
public struct RAMWidget: Widget {
var usedColor: Color = Color(nsColor: NSColor.systemBlue)
var freeColor: Color = Color(nsColor: NSColor.lightGray)
public init() {}
public var body: some WidgetConfiguration {
StaticConfiguration(kind: RAM_entry.kind, provider: Provider()) { entry in
VStack(spacing: 10) {
if let value = entry.value {
HStack {
Chart {
SectorMark(angle: .value(localizedString("Used"), value.used/value.total), innerRadius: .ratio(0.8)).foregroundStyle(self.usedColor)
SectorMark(angle: .value(localizedString("Free"), 1-(value.used/value.total)), innerRadius: .ratio(0.8)).foregroundStyle(self.freeColor)
}
.frame(maxWidth: .infinity, maxHeight: 84)
.chartLegend(.hidden)
.chartBackground { chartProxy in
GeometryReader { geometry in
if let anchor = chartProxy.plotFrame {
let frame = geometry[anchor]
Text("\(Int((value.used/value.total)*100))%")
.font(.system(size: 14, weight: .regular))
.position(x: frame.midX, y: frame.midY-5)
Text("RAM")
.font(.system(size: 8, weight: .semibold))
.position(x: frame.midX, y: frame.midY+8)
}
}
}
}
VStack(spacing: 3) {
HStack {
Rectangle().fill(self.usedColor).frame(width: 12, height: 12).cornerRadius(2)
Text(localizedString("Used")).font(.system(size: 12, weight: .regular)).foregroundColor(.secondary)
Spacer()
Text(Units(bytes: Int64(value.used)).getReadableMemory(style: .memory))
}
HStack {
Rectangle().fill(self.freeColor).frame(width: 12, height: 12).cornerRadius(2)
Text(localizedString("Free")).font(.system(size: 12, weight: .regular)).foregroundColor(.secondary)
Spacer()
Text(Units(bytes: Int64(value.free)).getReadableMemory(style: .memory))
}
HStack {
Text(localizedString("Pressure level")).font(.system(size: 12, weight: .regular)).foregroundColor(.secondary)
Spacer()
Text("\(value.pressure.level)")
}
}
} else {
Text("No data")
}
}
.containerBackground(for: .widget) {
Color.clear
}
}
.configurationDisplayName("RAM widget")
.description("Displays RAM stats")
.supportedFamilies([.systemSmall])
}
}