cpu chart mvp

This commit is contained in:
Serhiy Mytrovtsiy
2019-06-14 22:53:05 +02:00
parent 98d90f58a7
commit 102ab549f3
6 changed files with 136 additions and 129 deletions

View File

@@ -8,144 +8,30 @@
import Cocoa
class ChartView: NSView {
var valueLabel: NSTextField = NSTextField()
var label: Bool = false
var points: [Double] {
didSet {
setNeedsDisplay(self.frame)
}
}
override init(frame: NSRect) {
self.points = Array(repeating: 0.0, count: 50)
super.init(frame: frame)
self.wantsLayer = true
if self.label {
let valueLabel = NSTextField(frame: NSMakeRect(2, MODULE_HEIGHT - 11, self.frame.size.width, 10))
valueLabel.textColor = NSColor.red
valueLabel.isEditable = false
valueLabel.isSelectable = false
valueLabel.isBezeled = false
valueLabel.wantsLayer = true
valueLabel.textColor = .labelColor
valueLabel.backgroundColor = .controlColor
valueLabel.canDrawSubviewsIntoLayer = true
valueLabel.alignment = .natural
valueLabel.font = NSFont.systemFont(ofSize: 8, weight: .ultraLight)
valueLabel.stringValue = ""
valueLabel.addSubview(NSView())
self.valueLabel = valueLabel
self.addSubview(self.valueLabel)
} else {
self.addSubview(NSView())
}
}
required init?(coder decoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func draw(_ dirtyRect: NSRect) {
super.draw(dirtyRect)
if (self.points.count < 2) {
return
}
let xOffset: CGFloat = 4.0
let yOffset: CGFloat = 3.0
var height: Double = Double(self.frame.size.height) - Double((yOffset * 2))
if self.label {
height = 7.0
}
let xRatio = Double(self.frame.size.width - (xOffset * 2)) / (Double(self.points.count) - 1)
let chartLine = NSBezierPath()
chartLine.lineWidth = 0.5
for i in 0..<self.points.count {
let x: CGFloat = CGFloat((Double(i) * xRatio)) + xOffset
let y: CGFloat = CGFloat((Double(truncating: points[i] as NSNumber) * height)) + yOffset
let point = CGPoint(x: x, y: y)
if i == 0 {
chartLine.move(to: point)
} else {
chartLine.line(to: point)
}
}
// chartLine.close()
NSColor.blue.setStroke()
chartLine.stroke()
// let gradient: NSGradient = NSGradient(starting: NSColor(red: 0.0, green: 0.0, blue: 0.0, alpha: 1.0), ending: NSColor(red: 1.0, green: 1.0, blue: 1.0, alpha: 1.0))!
// gradient.draw(in: chartLine, angle: 0.0)
}
func addValue(value: Double) {
if self.label {
self.valueLabel.stringValue = "\(Int(Float(Float(value).roundTo(decimalPlaces: 2))! * 100))%"
self.valueLabel.textColor = Float(value).usageColor()
}
if self.points.count < 50 {
self.points.append(value)
return
}
for (i, _) in self.points.enumerated() {
if i+1 < self.points.count {
self.points[i] = self.points[i+1]
} else {
self.points[i] = value
}
}
}
}
class CPU: Module {
let name: String = "CPU"
var view: NSView = NSView()
var chart: ChartView = ChartView()
let defaults = UserDefaults.standard
var active: Observable<Bool>
var reader: Reader = CPUReader()
@IBOutlet weak var value: NSTextField!
init() {
self.active = Observable(defaults.object(forKey: name) != nil ? defaults.bool(forKey: name) : true)
self.chart = ChartView(frame: NSMakeRect(0, 0, MODULE_WIDTH + 7, MODULE_HEIGHT))
self.view.wantsLayer = true
self.view = self.chart
self.view = ChartView(frame: NSMakeRect(0, 0, MODULE_WIDTH + 7, MODULE_HEIGHT))
}
func start() {
if !self.reader.usage.value.isNaN {
self.chart.addValue(value: Double(self.reader.usage!.value))
// self.value.stringValue = "\(Int(Float(self.reader.usage.value.roundTo(decimalPlaces: 2))! * 100))%"
// self.value.textColor = self.reader.usage.value.usageColor()
(self.view as! ChartView).value(value: self.reader.usage!.value)
}
//
self.reader.start()
self.reader.usage.subscribe(observer: self) { (value, _) in
if !value.isNaN {
self.chart.addValue(value: Double(self.reader.usage!.value))
// self.value.stringValue = "\(Int(Float(value.roundTo(decimalPlaces: 2))! * 100))%"
// self.value.textColor = value.usageColor()
(self.view as! ChartView).value(value: value)
}
}
//
// colors.subscribe(observer: self) { (value, _) in
// self.value.textColor = self.reader.usage.value.usageColor()
// }
}
func menu() -> NSMenuItem {
@@ -156,13 +42,11 @@ class CPU: Module {
menu.state = NSControl.StateValue.on
}
menu.target = self
menu.isEnabled = true
return menu
}
@objc func toggle(_ sender: NSMenuItem) {
let state = sender.state != NSControl.StateValue.on
sender.state = sender.state == NSControl.StateValue.on ? NSControl.StateValue.off : NSControl.StateValue.on
self.defaults.set(state, forKey: name)
self.active << state

View File

@@ -0,0 +1,125 @@
//
// CPUView.swift
// Stats
//
// Created by Serhiy Mytrovtsiy on 14.06.2019.
// Copyright © 2019 Serhiy Mytrovtsiy. All rights reserved.
//
import Cocoa
class ChartView: NSView {
var valueLabel: NSTextField = NSTextField()
var label: Bool = true
var points: [Float] {
didSet {
self.needsDisplay = true
setNeedsDisplay(self.frame)
}
}
override init(frame: NSRect) {
self.points = Array(repeating: 0.0, count: 50)
super.init(frame: frame)
self.wantsLayer = true
if self.label {
let valueLabel = NSTextField(frame: NSMakeRect(2, MODULE_HEIGHT - 11, self.frame.size.width, 10))
valueLabel.textColor = NSColor.red
valueLabel.isEditable = false
valueLabel.isSelectable = false
valueLabel.isBezeled = false
valueLabel.wantsLayer = true
valueLabel.textColor = .labelColor
valueLabel.backgroundColor = .controlColor
valueLabel.canDrawSubviewsIntoLayer = true
valueLabel.alignment = .natural
valueLabel.font = NSFont.systemFont(ofSize: 8, weight: .ultraLight)
valueLabel.stringValue = ""
valueLabel.addSubview(NSView())
self.valueLabel = valueLabel
self.addSubview(self.valueLabel)
} else {
self.addSubview(NSView())
}
}
required init?(coder decoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func draw(_ dirtyRect: NSRect) {
super.draw(dirtyRect)
let lineColor: NSColor = NSColor.selectedMenuItemColor
let gradientColor: NSColor = NSColor(red: (26/255.0), green: (126/255.0), blue: (252/255.0), alpha: 0.5)
let context = NSGraphicsContext.current!.cgContext
let xOffset: CGFloat = 4.0
let yOffset: CGFloat = 3.0
var height: CGFloat = self.frame.size.height - CGFloat((yOffset * 2))
if self.label {
height = 7.0
}
let xRatio = Double(self.frame.size.width - (xOffset * 2)) / (Double(self.points.count) - 1)
let columnXPoint = { (point: Int) -> CGFloat in
return CGFloat((Double(point) * xRatio)) + xOffset
}
let columnYPoint = { (point: Int) -> CGFloat in
return CGFloat((CGFloat(truncating: self.points[point] as NSNumber) * height)) + yOffset
}
let graphPath = NSBezierPath()
let x: CGFloat = columnXPoint(0)
let y: CGFloat = columnYPoint(0)
graphPath.move(to: CGPoint(x: x, y: y))
for i in 1..<self.points.count {
graphPath.line(to: CGPoint(x: columnXPoint(i), y: columnYPoint(i)))
}
lineColor.setStroke()
graphPath.stroke()
context.saveGState()
let clippingPath = graphPath.copy() as! NSBezierPath
clippingPath.line(to: CGPoint(x: columnXPoint(self.points.count - 1), y: yOffset - 0.5))
clippingPath.line(to: CGPoint(x: columnXPoint(0), y: yOffset - 0.5))
clippingPath.close()
clippingPath.addClip()
gradientColor.setFill()
let rectPath = NSBezierPath(rect: dirtyRect)
rectPath.fill()
context.restoreGState()
graphPath.lineWidth = 1.0
graphPath.stroke()
}
func value(value: Float) {
if self.label {
self.valueLabel.stringValue = "\(Int(Float(Float(value).roundTo(decimalPlaces: 2))! * 100))%"
self.valueLabel.textColor = Float(value).usageColor()
}
if self.points.count < 50 {
self.points.append(value)
return
}
for (i, _) in self.points.enumerated() {
if i+1 < self.points.count {
self.points[i] = self.points[i+1]
} else {
self.points[i] = value
}
}
}
}

View File

@@ -10,6 +10,7 @@ import Cocoa
class Disk: Module {
let name: String = "Disk"
var colors: Bool = false
var view: NSView = NSView()
let defaults = UserDefaults.standard
@@ -36,10 +37,6 @@ class Disk: Module {
self.value.textColor = value.usageColor()
}
}
colors.subscribe(observer: self) { (value, _) in
self.value.textColor = self.reader.usage.value.usageColor()
}
}
func menu() -> NSMenuItem {

View File

@@ -10,6 +10,7 @@ import Cocoa
class Memory: Module {
let name: String = "Memory"
var colors: Bool = false
var view: NSView = NSView()
let defaults = UserDefaults.standard
@@ -36,10 +37,6 @@ class Memory: Module {
self.value.textColor = value.usageColor()
}
}
colors.subscribe(observer: self) { (value, _) in
self.value.textColor = self.reader.usage.value.usageColor()
}
}
func menu() -> NSMenuItem {

View File

@@ -10,9 +10,9 @@ import Cocoa
protocol Module {
var name: String { get }
var view: NSView { get }
var active: Observable<Bool> { get }
var reader: Reader { get }
var view: NSView { get }
func menu() -> NSMenuItem
func start()