mirror of
https://github.com/morgan9e/macos-stats
synced 2026-04-14 00:04:15 +09:00
657 lines
26 KiB
Swift
657 lines
26 KiB
Swift
//
|
|
// Battery.swift
|
|
// Kit
|
|
//
|
|
// Created by Serhiy Mytrovtsiy on 06/06/2020.
|
|
// Using Swift 5.0.
|
|
// Running on macOS 10.15.
|
|
//
|
|
// Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved.
|
|
//
|
|
|
|
import Cocoa
|
|
|
|
public class BatteryWidget: WidgetWrapper {
|
|
private var additional: String = "none"
|
|
private var timeFormat: String = "short"
|
|
private var iconState: Bool = true
|
|
private var colorState: Bool = false
|
|
private var hideAdditionalWhenFull: Bool = true
|
|
private var xlSizeState: Bool = false
|
|
private var chargerIconInside: Bool = true
|
|
|
|
private var _percentage: Double? = nil
|
|
private var _time: Int = 0
|
|
private var _charging: Bool = false
|
|
private var _ACStatus: Bool = false
|
|
private var _optimizedCharging: Bool = false
|
|
|
|
public init(title: String, preview: Bool = false) {
|
|
let widgetTitle: String = title
|
|
|
|
super.init(.battery, title: widgetTitle, frame: CGRect(
|
|
x: Constants.Widget.margin.x,
|
|
y: Constants.Widget.margin.y,
|
|
width: 30 + (2*Constants.Widget.margin.x),
|
|
height: Constants.Widget.height - (2*Constants.Widget.margin.y)
|
|
))
|
|
|
|
self.canDrawConcurrently = true
|
|
|
|
if !preview {
|
|
self.additional = Store.shared.string(key: "\(self.title)_\(self.type.rawValue)_additional", defaultValue: self.additional)
|
|
self.timeFormat = Store.shared.string(key: "\(self.title)_timeFormat", defaultValue: self.timeFormat)
|
|
self.iconState = Store.shared.bool(key: "\(self.title)_\(self.type.rawValue)_icon", defaultValue: self.iconState)
|
|
self.colorState = Store.shared.bool(key: "\(self.title)_\(self.type.rawValue)_color", defaultValue: self.colorState)
|
|
self.hideAdditionalWhenFull = Store.shared.bool(key: "\(self.title)_\(self.type.rawValue)_hideAdditionalWhenFull", defaultValue: self.hideAdditionalWhenFull)
|
|
self.xlSizeState = Store.shared.bool(key: "\(self.title)_\(self.type.rawValue)_xlSize", defaultValue: self.xlSizeState)
|
|
self.chargerIconInside = Store.shared.bool(key: "\(self.title)_\(self.type.rawValue)_chargerInside", defaultValue: self.chargerIconInside)
|
|
}
|
|
|
|
if preview {
|
|
self._percentage = 0.72
|
|
self.additional = "none"
|
|
self.iconState = true
|
|
self.colorState = false
|
|
}
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
public override func draw(_ dirtyRect: NSRect) {
|
|
super.draw(dirtyRect)
|
|
|
|
guard let ctx = NSGraphicsContext.current?.cgContext else { return }
|
|
|
|
var percentage: Double? = nil
|
|
var time: Int = 0
|
|
var charging: Bool = false
|
|
var ACStatus: Bool = false
|
|
var optimizedCharging: Bool = false
|
|
self.queue.sync {
|
|
percentage = self._percentage
|
|
time = self._time
|
|
charging = self._charging
|
|
ACStatus = self._ACStatus
|
|
optimizedCharging = self._optimizedCharging
|
|
}
|
|
|
|
var width: CGFloat = 0
|
|
var x: CGFloat = 0
|
|
let isShortTimeFormat: Bool = self.timeFormat == "short"
|
|
|
|
if !self.hideAdditionalWhenFull || (self.hideAdditionalWhenFull && percentage != 1 && !optimizedCharging) {
|
|
switch self.additional {
|
|
case "percentage":
|
|
var value = "n/a"
|
|
if let percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
let rowWidth = self.drawOneRow(value: value, x: x).rounded(.up)
|
|
width += rowWidth
|
|
x += rowWidth + Constants.Widget.spacing
|
|
case "time":
|
|
let rowWidth = self.drawOneRow(
|
|
value: Double(time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
x: x
|
|
).rounded(.up)
|
|
width += rowWidth
|
|
x += rowWidth + Constants.Widget.spacing
|
|
case "percentageAndTime":
|
|
var value = "n/a"
|
|
if let percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
let rowWidth = self.drawTwoRows(
|
|
first: value,
|
|
second: Double(time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
x: x
|
|
).rounded(.up)
|
|
width += rowWidth
|
|
x += rowWidth + Constants.Widget.spacing
|
|
case "timeAndPercentage":
|
|
var value = "n/a"
|
|
if let percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
let rowWidth = self.drawTwoRows(
|
|
first: Double(time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
second: value,
|
|
x: x
|
|
).rounded(.up)
|
|
width += rowWidth
|
|
x += rowWidth + Constants.Widget.spacing
|
|
default: break
|
|
}
|
|
}
|
|
|
|
let batterySize: CGSize = self.xlSizeState ? CGSize(width: 26, height: 14) : CGSize(width: 22, height: 12)
|
|
|
|
if ACStatus && !self.chargerIconInside {
|
|
if x != 0 {
|
|
width += Constants.Widget.spacing
|
|
x += Constants.Widget.spacing
|
|
}
|
|
self.drawACIcon(
|
|
ctx: ctx,
|
|
center: CGPoint(x: x+3, y: self.frame.size.height/2),
|
|
height: 12,
|
|
charging: charging
|
|
)
|
|
width += 6
|
|
x += 6 + Constants.Widget.spacing
|
|
}
|
|
|
|
let borderWidth: CGFloat = 1
|
|
let batteryRadius: CGFloat = self.xlSizeState ? 3 : 2
|
|
let offset: CGFloat = 0.5 // contant!
|
|
width += batterySize.width + borderWidth*2 // add battery width
|
|
if x != 0 {
|
|
width += Constants.Widget.spacing
|
|
x += Constants.Widget.spacing
|
|
}
|
|
|
|
let batteryFrame = NSBezierPath(roundedRect: NSRect(
|
|
x: x + borderWidth + offset,
|
|
y: ((self.frame.size.height - batterySize.height)/2) + offset,
|
|
width: batterySize.width - borderWidth,
|
|
height: batterySize.height - borderWidth
|
|
), xRadius: batteryRadius, yRadius: batteryRadius)
|
|
|
|
NSColor.textColor.withAlphaComponent(0.5).set()
|
|
batteryFrame.lineWidth = borderWidth
|
|
batteryFrame.stroke()
|
|
|
|
let bPX: CGFloat = batteryFrame.bounds.origin.x + batteryFrame.bounds.width + 1
|
|
let bPY: CGFloat = batteryFrame.bounds.origin.y + batteryFrame.bounds.height/2 - 2
|
|
let batteryPoint = NSBezierPath(roundedRect: NSRect(x: bPX - 1, y: bPY, width: 3, height: 4), xRadius: 2, yRadius: 2)
|
|
batteryPoint.fill()
|
|
|
|
let batteryPointSeparator = NSBezierPath()
|
|
batteryPointSeparator.move(to: CGPoint(x: bPX, y: batteryFrame.bounds.origin.y))
|
|
batteryPointSeparator.line(to: CGPoint(x: bPX, y: batteryFrame.bounds.origin.y + batteryFrame.bounds.height))
|
|
ctx.saveGState()
|
|
ctx.setBlendMode(.destinationOut)
|
|
NSColor.white.set()
|
|
batteryPointSeparator.lineWidth = borderWidth
|
|
batteryPointSeparator.stroke()
|
|
ctx.restoreGState()
|
|
width += 2 // add battery point width
|
|
|
|
if let percentage {
|
|
let maxWidth = batterySize.width - offset*2 - borderWidth*2 - 1
|
|
let innerWidth: CGFloat = max(1, maxWidth * CGFloat(percentage))
|
|
let innerOffset: CGFloat = -offset + borderWidth + 1
|
|
let innerRadius: CGFloat = self.xlSizeState ? 2 : 1
|
|
var colorState = self.colorState
|
|
let color = percentage.batteryColor(color: colorState)
|
|
let innerPercentage = self.additional == "innerPercentage" && (!ACStatus || !self.chargerIconInside)
|
|
|
|
if innerPercentage {
|
|
colorState = false
|
|
let innerUnderground = NSBezierPath(roundedRect: NSRect(
|
|
x: batteryFrame.bounds.origin.x + innerOffset,
|
|
y: batteryFrame.bounds.origin.y + innerOffset,
|
|
width: maxWidth,
|
|
height: batterySize.height - offset*2 - borderWidth*2 - 1
|
|
), xRadius: innerRadius, yRadius: innerRadius)
|
|
(self.colorState ? color : NSColor.textColor).withAlphaComponent(0.5).set()
|
|
innerUnderground.fill()
|
|
}
|
|
|
|
let inner = NSBezierPath(roundedRect: NSRect(
|
|
x: batteryFrame.bounds.origin.x + innerOffset,
|
|
y: batteryFrame.bounds.origin.y + innerOffset,
|
|
width: innerWidth,
|
|
height: batterySize.height - offset*2 - borderWidth*2 - 1
|
|
), xRadius: innerRadius, yRadius: innerRadius)
|
|
|
|
color.set()
|
|
inner.fill()
|
|
|
|
if innerPercentage {
|
|
let fontSize: CGFloat = self.xlSizeState ? 9 : 8
|
|
let style = NSMutableParagraphStyle()
|
|
style.alignment = .center
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: fontSize, weight: .bold),
|
|
NSAttributedString.Key.foregroundColor: NSColor.clear,
|
|
NSAttributedString.Key.paragraphStyle: style
|
|
]
|
|
|
|
let value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))"
|
|
let rect = CGRect(x: inner.bounds.origin.x, y: (Constants.Widget.height-(fontSize+2))/2, width: maxWidth, height: fontSize)
|
|
let str = NSAttributedString.init(string: value, attributes: attributes)
|
|
|
|
ctx.saveGState()
|
|
ctx.setBlendMode(.destinationIn)
|
|
str.draw(with: rect)
|
|
ctx.restoreGState()
|
|
}
|
|
} else {
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: 11, weight: .regular),
|
|
NSAttributedString.Key.foregroundColor: isDarkMode ? NSColor.white : NSColor.textColor,
|
|
NSAttributedString.Key.paragraphStyle: NSMutableParagraphStyle()
|
|
]
|
|
|
|
let batteryCenter: CGPoint = CGPoint(
|
|
x: batteryFrame.bounds.origin.x + (batteryFrame.bounds.width/2),
|
|
y: batteryFrame.bounds.origin.y + (batteryFrame.bounds.height/2)
|
|
)
|
|
|
|
let rect = CGRect(x: batteryCenter.x-3, y: batteryCenter.y-4, width: 8, height: 12)
|
|
NSAttributedString.init(string: "?", attributes: attributes).draw(with: rect)
|
|
}
|
|
|
|
if ACStatus && self.chargerIconInside {
|
|
let batteryCenter: CGPoint = CGPoint(
|
|
x: batteryFrame.bounds.origin.x + (batteryFrame.bounds.width/2),
|
|
y: batteryFrame.bounds.origin.y + (batteryFrame.bounds.height/2)
|
|
)
|
|
self.drawACIcon(
|
|
ctx: ctx,
|
|
center: batteryCenter,
|
|
height: 12,
|
|
charging: charging
|
|
)
|
|
}
|
|
|
|
self.setWidth(width)
|
|
}
|
|
|
|
private func drawOneRow(value: String, x: CGFloat) -> CGFloat {
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: 12, weight: .regular),
|
|
NSAttributedString.Key.foregroundColor: isDarkMode ? NSColor.white : NSColor.textColor,
|
|
NSAttributedString.Key.paragraphStyle: NSMutableParagraphStyle()
|
|
]
|
|
|
|
let rowWidth = value.widthOfString(usingFont: .systemFont(ofSize: 12, weight: .regular))
|
|
let rect = CGRect(x: x, y: (Constants.Widget.height-13)/2, width: rowWidth, height: 12)
|
|
let str = NSAttributedString.init(string: value, attributes: attributes)
|
|
str.draw(with: rect)
|
|
|
|
return rowWidth
|
|
}
|
|
|
|
private func drawTwoRows(first: String, second: String, x: CGFloat) -> CGFloat {
|
|
let style = NSMutableParagraphStyle()
|
|
style.alignment = .center
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: 9, weight: .regular),
|
|
NSAttributedString.Key.foregroundColor: NSColor.textColor,
|
|
NSAttributedString.Key.paragraphStyle: style
|
|
]
|
|
let rowHeight: CGFloat = self.frame.height / 2
|
|
|
|
let rowWidth = max(
|
|
first.widthOfString(usingFont: .systemFont(ofSize: 9, weight: .regular)),
|
|
second.widthOfString(usingFont: .systemFont(ofSize: 9, weight: .regular))
|
|
)
|
|
|
|
var str = NSAttributedString.init(string: first, attributes: attributes)
|
|
str.draw(with: CGRect(x: x, y: rowHeight+1, width: rowWidth, height: rowHeight))
|
|
|
|
str = NSAttributedString.init(string: second, attributes: attributes)
|
|
str.draw(with: CGRect(x: x, y: 1, width: rowWidth, height: rowHeight))
|
|
|
|
return rowWidth
|
|
}
|
|
|
|
private func drawACIcon(ctx: CGContext, center batteryCenter: CGPoint, height: CGFloat, charging: Bool) {
|
|
var points: [CGPoint] = []
|
|
|
|
if charging {
|
|
let iconSize: CGSize = CGSize(width: 9, height: height + 6)
|
|
let min = CGPoint(
|
|
x: batteryCenter.x - (iconSize.width/2),
|
|
y: batteryCenter.y - (iconSize.height/2)
|
|
)
|
|
let max = CGPoint(
|
|
x: batteryCenter.x + (iconSize.width/2),
|
|
y: batteryCenter.y + (iconSize.height/2)
|
|
)
|
|
|
|
points = [
|
|
CGPoint(x: batteryCenter.x-3, y: min.y), // bottom
|
|
CGPoint(x: max.x, y: batteryCenter.y+1.5),
|
|
CGPoint(x: batteryCenter.x+1, y: batteryCenter.y+1.5),
|
|
CGPoint(x: batteryCenter.x+3, y: max.y), // top
|
|
CGPoint(x: min.x, y: batteryCenter.y-1.5),
|
|
CGPoint(x: batteryCenter.x-1, y: batteryCenter.y-1.5)
|
|
]
|
|
} else {
|
|
let iconSize: CGSize = CGSize(width: 9, height: height + 2)
|
|
let minY = batteryCenter.y - (iconSize.height/2)
|
|
let maxY = batteryCenter.y + (iconSize.height/2)
|
|
|
|
points = [
|
|
CGPoint(x: batteryCenter.x-1.5, y: minY+0.5),
|
|
|
|
CGPoint(x: batteryCenter.x+1.5, y: minY+0.5),
|
|
CGPoint(x: batteryCenter.x+1.5, y: batteryCenter.y - 2.5),
|
|
|
|
CGPoint(x: batteryCenter.x+4, y: batteryCenter.y + 0.5),
|
|
CGPoint(x: batteryCenter.x+4, y: batteryCenter.y + 4.25),
|
|
|
|
// right
|
|
CGPoint(x: batteryCenter.x+2.75, y: batteryCenter.y + 4.25),
|
|
CGPoint(x: batteryCenter.x+2.75, y: maxY-0.25),
|
|
CGPoint(x: batteryCenter.x+0.25, y: maxY-0.25),
|
|
CGPoint(x: batteryCenter.x+0.25, y: batteryCenter.y + 4.25),
|
|
|
|
// left
|
|
CGPoint(x: batteryCenter.x-0.25, y: batteryCenter.y + 4.25),
|
|
CGPoint(x: batteryCenter.x-0.25, y: maxY-0.25),
|
|
CGPoint(x: batteryCenter.x-2.75, y: maxY-0.25),
|
|
CGPoint(x: batteryCenter.x-2.75, y: batteryCenter.y + 4.25),
|
|
|
|
CGPoint(x: batteryCenter.x-4, y: batteryCenter.y + 4.25),
|
|
CGPoint(x: batteryCenter.x-4, y: batteryCenter.y + 0.5),
|
|
|
|
CGPoint(x: batteryCenter.x-1.5, y: batteryCenter.y - 2.5),
|
|
CGPoint(x: batteryCenter.x-1.5, y: minY+0.5)
|
|
]
|
|
}
|
|
|
|
let linePath = NSBezierPath()
|
|
linePath.move(to: CGPoint(x: points[0].x, y: points[0].y))
|
|
for i in 1..<points.count {
|
|
linePath.line(to: CGPoint(x: points[i].x, y: points[i].y))
|
|
}
|
|
linePath.line(to: CGPoint(x: points[0].x, y: points[0].y))
|
|
|
|
NSColor.textColor.set()
|
|
linePath.fill()
|
|
|
|
ctx.saveGState()
|
|
ctx.setBlendMode(.destinationOut)
|
|
|
|
NSColor.textColor.set()
|
|
linePath.lineWidth = 1
|
|
linePath.stroke()
|
|
|
|
ctx.restoreGState()
|
|
}
|
|
|
|
public func setValue(percentage: Double? = nil, ACStatus: Bool? = nil, isCharging: Bool? = nil, optimizedCharging: Bool? = nil, time: Int? = nil) {
|
|
var updated: Bool = false
|
|
let timeFormat: String = Store.shared.string(key: "\(self.title)_timeFormat", defaultValue: self.timeFormat)
|
|
|
|
if self._percentage != percentage {
|
|
self._percentage = percentage
|
|
updated = true
|
|
}
|
|
if let status = ACStatus, self._ACStatus != status {
|
|
self._ACStatus = status
|
|
updated = true
|
|
}
|
|
if let charging = isCharging, self._charging != charging {
|
|
self._charging = charging
|
|
updated = true
|
|
}
|
|
if let time = time, self._time != time {
|
|
self._time = time
|
|
updated = true
|
|
}
|
|
if self.timeFormat != timeFormat {
|
|
self.timeFormat = timeFormat
|
|
updated = true
|
|
}
|
|
if let state = optimizedCharging, self._optimizedCharging != state {
|
|
self._optimizedCharging = state
|
|
updated = true
|
|
}
|
|
|
|
if updated {
|
|
self.needsDisplay = true
|
|
DispatchQueue.main.async(execute: {
|
|
self.display()
|
|
})
|
|
}
|
|
}
|
|
|
|
// MARK: - Settings
|
|
|
|
public override func settings() -> NSView {
|
|
let view = SettingsContainerView()
|
|
|
|
var additionalOptions = BatteryAdditionals
|
|
if self.title == "Bluetooth" {
|
|
additionalOptions = additionalOptions.filter({ $0.key == "none" || $0.key == "percentage" })
|
|
}
|
|
|
|
view.addArrangedSubview(PreferencesSection([
|
|
PreferencesRow(localizedString("Additional information"), component: selectView(
|
|
action: #selector(self.toggleAdditional),
|
|
items: additionalOptions,
|
|
selected: self.additional
|
|
)),
|
|
PreferencesRow(localizedString("Hide additional information when full"), component: switchView(
|
|
action: #selector(self.toggleHideAdditionalWhenFull),
|
|
state: self.hideAdditionalWhenFull
|
|
)),
|
|
PreferencesRow(localizedString("Colorize"), component: switchView(
|
|
action: #selector(self.toggleColor),
|
|
state: self.colorState
|
|
)),
|
|
PreferencesRow(localizedString("XL size"), component: switchView(
|
|
action: #selector(self.toggleXLSize),
|
|
state: self.xlSizeState
|
|
)),
|
|
PreferencesRow(localizedString("Charger state inside the battery"), component: switchView(
|
|
action: #selector(self.toggleChargerIconInside),
|
|
state: self.chargerIconInside
|
|
))
|
|
]))
|
|
|
|
return view
|
|
}
|
|
|
|
@objc private func toggleAdditional(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.additional = key
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_additional", value: key)
|
|
self.display()
|
|
}
|
|
|
|
@objc private func toggleHideAdditionalWhenFull(_ sender: NSControl) {
|
|
self.hideAdditionalWhenFull = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_hideAdditionalWhenFull", value: self.hideAdditionalWhenFull)
|
|
self.display()
|
|
}
|
|
|
|
@objc private func toggleColor(_ sender: NSControl) {
|
|
self.colorState = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_color", value: self.colorState)
|
|
self.display()
|
|
}
|
|
|
|
@objc private func toggleXLSize(_ sender: NSControl) {
|
|
self.xlSizeState = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_xlSize", value: self.xlSizeState)
|
|
self.display()
|
|
}
|
|
|
|
@objc private func toggleChargerIconInside(_ sender: NSControl) {
|
|
self.chargerIconInside = controlState(sender)
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_chargerInside", value: self.chargerIconInside)
|
|
self.display()
|
|
}
|
|
}
|
|
|
|
public class BatteryDetailsWidget: WidgetWrapper {
|
|
private var mode: String = "percentage"
|
|
private var timeFormat: String = "short"
|
|
|
|
private var percentage: Double? = nil
|
|
private var time: Int = 0
|
|
|
|
public init(title: String, preview: Bool = false) {
|
|
super.init(.batteryDetails, title: title, frame: CGRect(
|
|
x: Constants.Widget.margin.x,
|
|
y: Constants.Widget.margin.y,
|
|
width: 20 + (2*Constants.Widget.margin.x),
|
|
height: Constants.Widget.height - (2*Constants.Widget.margin.y)
|
|
))
|
|
|
|
self.canDrawConcurrently = true
|
|
|
|
if preview {
|
|
self.percentage = 0.72
|
|
self.time = 415
|
|
self.mode = "percentageAndTime"
|
|
} else {
|
|
self.mode = Store.shared.string(key: "\(self.title)_\(self.type.rawValue)_mode", defaultValue: self.mode)
|
|
self.timeFormat = Store.shared.string(key: "\(self.title)_timeFormat", defaultValue: self.timeFormat)
|
|
}
|
|
}
|
|
|
|
required init?(coder: NSCoder) {
|
|
fatalError("init(coder:) has not been implemented")
|
|
}
|
|
|
|
public override func draw(_ dirtyRect: NSRect) {
|
|
super.draw(dirtyRect)
|
|
|
|
var width: CGFloat = Constants.Widget.margin.x*2
|
|
let x: CGFloat = Constants.Widget.margin.x
|
|
let isShortTimeFormat: Bool = self.timeFormat == "short"
|
|
|
|
switch self.mode {
|
|
case "percentage":
|
|
var value = "n/a"
|
|
if let percentage = self.percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
width = self.drawOneRow(value: value, x: x).rounded(.up)
|
|
case "time":
|
|
width = self.drawOneRow(
|
|
value: Double(self.time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
x: x
|
|
).rounded(.up)
|
|
case "percentageAndTime":
|
|
var value = "n/a"
|
|
if let percentage = self.percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
if self.time > 0 {
|
|
width = self.drawTwoRows(
|
|
first: value,
|
|
second: Double(self.time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
x: x
|
|
).rounded(.up)
|
|
} else {
|
|
width = self.drawOneRow(value: value, x: x).rounded(.up)
|
|
}
|
|
case "timeAndPercentage":
|
|
var value = "n/a"
|
|
if let percentage = self.percentage {
|
|
value = "\(Int((percentage.rounded(toPlaces: 2)) * 100))%"
|
|
}
|
|
if self.time > 0 {
|
|
width = self.drawTwoRows(
|
|
first: Double(self.time*60).printSecondsToHoursMinutesSeconds(short: isShortTimeFormat),
|
|
second: value,
|
|
x: x
|
|
).rounded(.up)
|
|
} else {
|
|
width = self.drawOneRow(value: value, x: x).rounded(.up)
|
|
}
|
|
default: break
|
|
}
|
|
|
|
self.setWidth(width)
|
|
}
|
|
|
|
private func drawOneRow(value: String, x: CGFloat) -> CGFloat {
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: 12, weight: .regular),
|
|
NSAttributedString.Key.foregroundColor: isDarkMode ? NSColor.white : NSColor.textColor,
|
|
NSAttributedString.Key.paragraphStyle: NSMutableParagraphStyle()
|
|
]
|
|
|
|
let rowWidth = value.widthOfString(usingFont: .systemFont(ofSize: 12, weight: .regular))
|
|
let rect = CGRect(x: x, y: (Constants.Widget.height-12)/2, width: rowWidth, height: 12)
|
|
let str = NSAttributedString.init(string: value, attributes: attributes)
|
|
str.draw(with: rect)
|
|
|
|
return rowWidth
|
|
}
|
|
|
|
private func drawTwoRows(first: String, second: String, x: CGFloat) -> CGFloat {
|
|
let style = NSMutableParagraphStyle()
|
|
style.alignment = .center
|
|
let attributes = [
|
|
NSAttributedString.Key.font: NSFont.systemFont(ofSize: 9, weight: .regular),
|
|
NSAttributedString.Key.foregroundColor: NSColor.textColor,
|
|
NSAttributedString.Key.paragraphStyle: style
|
|
]
|
|
let rowHeight: CGFloat = self.frame.height / 2
|
|
|
|
let rowWidth = max(
|
|
first.widthOfString(usingFont: .systemFont(ofSize: 9, weight: .regular)),
|
|
second.widthOfString(usingFont: .systemFont(ofSize: 9, weight: .regular))
|
|
)
|
|
|
|
var str = NSAttributedString.init(string: first, attributes: attributes)
|
|
str.draw(with: CGRect(x: x, y: rowHeight+1, width: rowWidth, height: rowHeight))
|
|
|
|
str = NSAttributedString.init(string: second, attributes: attributes)
|
|
str.draw(with: CGRect(x: x, y: 1, width: rowWidth, height: rowHeight))
|
|
|
|
return rowWidth
|
|
}
|
|
|
|
public func setValue(percentage: Double? = nil, time: Int? = nil) {
|
|
var updated: Bool = false
|
|
let timeFormat: String = Store.shared.string(key: "\(self.title)_timeFormat", defaultValue: self.timeFormat)
|
|
|
|
if self.percentage != percentage {
|
|
self.percentage = percentage
|
|
updated = true
|
|
}
|
|
if let time = time, self.time != time {
|
|
self.time = time
|
|
updated = true
|
|
}
|
|
if self.timeFormat != timeFormat {
|
|
self.timeFormat = timeFormat
|
|
updated = true
|
|
}
|
|
|
|
if updated {
|
|
self.needsDisplay = true
|
|
DispatchQueue.main.async(execute: {
|
|
self.display()
|
|
})
|
|
}
|
|
}
|
|
|
|
// MARK: - Settings
|
|
|
|
public override func settings() -> NSView {
|
|
let view = SettingsContainerView()
|
|
|
|
view.addArrangedSubview(PreferencesSection([
|
|
PreferencesRow(localizedString("Details"), component: selectView(
|
|
action: #selector(self.toggleMode),
|
|
items: BatteryInfo,
|
|
selected: self.mode
|
|
))
|
|
]))
|
|
|
|
return view
|
|
}
|
|
|
|
@objc private func toggleMode(_ sender: NSMenuItem) {
|
|
guard let key = sender.representedObject as? String else { return }
|
|
self.mode = key
|
|
Store.shared.set(key: "\(self.title)_\(self.type.rawValue)_mode", value: key)
|
|
self.display()
|
|
}
|
|
}
|