// // Chart.swift // ModuleKit // // Created by Serhiy Mytrovtsiy on 18/04/2020. // Using Swift 5.0. // Running on macOS 10.15. // // Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved. // import Cocoa import StatsKit public class LineChart: Widget { private var labelState: Bool = true private var boxState: Bool = true private var frameState: Bool = false private var valueState: Bool = false private var valueColorState: Bool = false private var colorState: widget_c = .systemAccent private let store: UnsafePointer? private var chart: LineChartView private var colors: [widget_c] = widget_c.allCases private var value: Double = 0 private var pressureLevel: Int = 0 private var boxSettingsView: NSView? = nil private var frameSettingsView: NSView? = nil public init(preview: Bool, title: String, config: NSDictionary?, store: UnsafePointer?) { var widgetTitle: String = title self.store = store if config != nil { if let titleFromConfig = config!["Title"] as? String { widgetTitle = titleFromConfig } if let label = config!["Label"] as? Bool { self.labelState = label } if let box = config!["Box"] as? Bool { self.boxState = box } if let value = config!["Value"] as? Bool { self.valueState = value } if let colorsToDisable = config!["Unsupported colors"] as? [String] { self.colors = self.colors.filter { (color: widget_c) -> Bool in return !colorsToDisable.contains("\(color.self)") } } if let color = config!["Color"] as? String { if let defaultColor = colors.first(where: { "\($0.self)" == color }) { self.colorState = defaultColor } } } self.chart = LineChartView(frame: NSRect(x: 0, y: 0, width: Constants.Widget.width, height: Constants.Widget.height - (2*Constants.Widget.margin)), num: 60) super.init(frame: CGRect(x: 0, y: Constants.Widget.margin, width: Constants.Widget.width, height: Constants.Widget.height - (2*Constants.Widget.margin))) self.preview = preview self.title = widgetTitle self.type = .lineChart self.canDrawConcurrently = true if self.store != nil && !preview { self.boxState = store!.pointee.bool(key: "\(self.title)_\(self.type.rawValue)_box", defaultValue: self.boxState) self.frameState = store!.pointee.bool(key: "\(self.title)_\(self.type.rawValue)_frame", defaultValue: self.frameState) self.valueState = store!.pointee.bool(key: "\(self.title)_\(self.type.rawValue)_value", defaultValue: self.valueState) self.labelState = store!.pointee.bool(key: "\(self.title)_\(self.type.rawValue)_label", defaultValue: self.labelState) self.valueColorState = store!.pointee.bool(key: "\(self.title)_\(self.type.rawValue)_valueColor", defaultValue: self.valueColorState) self.colorState = widget_c(rawValue: store!.pointee.string(key: "\(self.title)_\(self.type.rawValue)_color", defaultValue: self.colorState.rawValue)) ?? self.colorState } if self.labelState { self.setFrameSize(NSSize(width: Constants.Widget.width + 6 + (Constants.Widget.margin*2), height: self.frame.size.height)) } if preview { var list: [Double] = [] for _ in 0..<16 { list.append(Double(CGFloat(Float(arc4random()) / Float(UINT32_MAX)))) } self.chart.points = list self.value = 0.38 } } public override func draw(_ dirtyRect: NSRect) { super.draw(dirtyRect) let ctx = NSGraphicsContext.current!.cgContext ctx.saveGState() var width = Constants.Widget.width var x: CGFloat = Constants.Widget.margin var chartPadding: CGFloat = 0 if self.labelState { let style = NSMutableParagraphStyle() style.alignment = .center let stringAttributes = [ NSAttributedString.Key.font: NSFont.systemFont(ofSize: 7, weight: .regular), NSAttributedString.Key.foregroundColor: NSColor.textColor, NSAttributedString.Key.paragraphStyle: style ] let letterHeight = self.frame.height / 3 let letterWidth: CGFloat = 6.0 var yMargin: CGFloat = 0 for char in String(self.title.prefix(3)).uppercased().reversed() { let rect = CGRect(x: x, y: yMargin, width: letterWidth, height: letterHeight) let str = NSAttributedString.init(string: "\(char)", attributes: stringAttributes) str.draw(with: rect) yMargin += letterHeight } width = width + letterWidth + (Constants.Widget.margin*2) x = letterWidth + (Constants.Widget.margin*3) } var boxHeight: CGFloat = self.frame.size.height var boxRadius: CGFloat = 2 let boxWidth: CGFloat = Constants.Widget.width - (Constants.Widget.margin*2) var color: NSColor = NSColor.controlAccentColor switch self.colorState { case .systemAccent: color = NSColor.controlAccentColor case .utilization: color = value.usageColor() case .pressure: color = self.pressureLevel.pressureColor() case .monochrome: if self.boxState { color = (isDarkMode ? NSColor.black : NSColor.white) } else { color = (isDarkMode ? NSColor.white : NSColor.black) } default: color = colorFromString("\(self.colorState.self)") } if self.valueState { let style = NSMutableParagraphStyle() style.alignment = .right var valueColor = isDarkMode ? NSColor.white : NSColor.black if self.valueColorState { valueColor = color } let stringAttributes = [ NSAttributedString.Key.font: NSFont.systemFont(ofSize: 8, weight: .regular), NSAttributedString.Key.foregroundColor: valueColor, NSAttributedString.Key.paragraphStyle: style ] let rect = CGRect(x: x, y: boxHeight-7, width: boxWidth - chartPadding, height: 7) let str = NSAttributedString.init(string: "\(Int((value.rounded(toPlaces: 2)) * 100))%", attributes: stringAttributes) str.draw(with: rect) boxHeight = 9 boxRadius = 1 } let box = NSBezierPath(roundedRect: NSRect(x: x, y: 0, width: boxWidth, height: boxHeight), xRadius: boxRadius, yRadius: boxRadius) if self.boxState { (isDarkMode ? NSColor.white : NSColor.black).set() box.stroke() box.fill() self.chart.transparent = false chartPadding = 1 } else if self.frameState { chartPadding = 1 self.chart.transparent = true } else { self.chart.transparent = true } chart.setFrameSize(NSSize(width: box.bounds.width - chartPadding, height: box.bounds.height - (chartPadding*2))) self.chart.color = color chart.draw(NSRect(x: box.bounds.origin.x + 1, y: chartPadding, width: chart.frame.width, height: chart.frame.height)) if self.boxState || self.frameState { (isDarkMode ? NSColor.white : NSColor.black).set() box.lineWidth = 1 box.stroke() } ctx.restoreGState() self.setWidth(width) } required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } public override func settings(superview: NSView) { let rowHeight: CGFloat = 30 let settingsNumber: CGFloat = 6 let height: CGFloat = ((rowHeight + Constants.Settings.margin) * settingsNumber) + Constants.Settings.margin superview.setFrameSize(NSSize(width: superview.frame.width, height: height)) let view: NSView = NSView(frame: NSRect(x: Constants.Settings.margin, y: Constants.Settings.margin, width: superview.frame.width - (Constants.Settings.margin*2), height: superview.frame.height - (Constants.Settings.margin*2))) view.addSubview(ToggleTitleRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 5, width: view.frame.width, height: rowHeight), title: LocalizedString("Label"), action: #selector(toggleLabel), state: self.labelState )) view.addSubview(ToggleTitleRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 4, width: view.frame.width, height: rowHeight), title: LocalizedString("Value"), action: #selector(toggleValue), state: self.valueState )) self.boxSettingsView = ToggleTitleRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 3, width: view.frame.width, height: rowHeight), title: LocalizedString("Box"), action: #selector(toggleBox), state: self.boxState ) view.addSubview(self.boxSettingsView!) self.frameSettingsView = ToggleTitleRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 2, width: view.frame.width, height: rowHeight), title: LocalizedString("Frame"), action: #selector(toggleFrame), state: self.frameState ) view.addSubview(self.frameSettingsView!) view.addSubview(SelectColorRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 1, width: view.frame.width, height: rowHeight), title: LocalizedString("Color"), action: #selector(toggleColor), items: self.colors.map{ $0.rawValue }, selected: self.colorState.rawValue )) view.addSubview(ToggleTitleRow( frame: NSRect(x: 0, y: (rowHeight + Constants.Settings.margin) * 0, width: view.frame.width, height: rowHeight), title: LocalizedString("Colorize value"), action: #selector(toggleValueColor), state: self.valueColorState )) superview.addSubview(view) } public override func setValues(_ values: [value_t]) { let historyValues = values.map{ $0.widget_value }.suffix(60) let end = self.chart.points!.count self.chart.points!.replaceSubrange(end-historyValues.count...end-1, with: historyValues) self.display() } public func setValue(_ value: Double) { self.value = value DispatchQueue.main.async(execute: { self.chart.addValue(value) self.display() }) } public func setPressure(_ level: Int) { guard self.pressureLevel != level else { return } self.pressureLevel = level DispatchQueue.main.async(execute: { self.display() }) } @objc private func toggleLabel(_ sender: NSControl) { var state: NSControl.StateValue? = nil if #available(OSX 10.15, *) { state = sender is NSSwitch ? (sender as! NSSwitch).state: nil } else { state = sender is NSButton ? (sender as! NSButton).state: nil } self.labelState = state! == .on ? true : false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_label", value: self.labelState) self.display() } @objc private func toggleBox(_ sender: NSControl) { var state: NSControl.StateValue? = nil if #available(OSX 10.15, *) { state = sender is NSSwitch ? (sender as! NSSwitch).state: nil } else { state = sender is NSButton ? (sender as! NSButton).state: nil } self.boxState = state! == .on ? true : false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_box", value: self.boxState) if self.frameState { FindAndToggleNSControlState(self.frameSettingsView, state: .off) self.frameState = false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_frame", value: self.frameState) } self.display() } @objc private func toggleFrame(_ sender: NSControl) { var state: NSControl.StateValue? = nil if #available(OSX 10.15, *) { state = sender is NSSwitch ? (sender as! NSSwitch).state: nil } else { state = sender is NSButton ? (sender as! NSButton).state: nil } self.frameState = state! == .on ? true : false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_frame", value: self.frameState) if self.boxState { FindAndToggleNSControlState(self.boxSettingsView, state: .off) self.boxState = false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_box", value: self.boxState) } self.display() } @objc private func toggleValue(_ sender: NSControl) { var state: NSControl.StateValue? = nil if #available(OSX 10.15, *) { state = sender is NSSwitch ? (sender as! NSSwitch).state: nil } else { state = sender is NSButton ? (sender as! NSButton).state: nil } self.valueState = state! == .on ? true : false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_value", value: self.valueState) self.display() } @objc private func toggleColor(_ sender: NSMenuItem) { if let newColor = widget_c.allCases.first(where: { $0.rawValue == sender.title }) { self.colorState = newColor self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_color", value: self.colorState.rawValue) self.display() } } @objc private func toggleValueColor(_ sender: NSControl) { var state: NSControl.StateValue? = nil if #available(OSX 10.15, *) { state = sender is NSSwitch ? (sender as! NSSwitch).state: nil } else { state = sender is NSButton ? (sender as! NSButton).state: nil } self.valueColorState = state! == .on ? true : false self.store?.pointee.set(key: "\(self.title)_\(self.type.rawValue)_valueColor", value: self.valueColorState) self.display() } }