Files
macos-stats/Stats/Views/AppSettings.swift
2024-04-11 19:05:50 +02:00

706 lines
27 KiB
Swift

//
// AppSettings.swift
// Stats
//
// Created by Serhiy Mytrovtsiy on 15/04/2020.
// Using Swift 5.0.
// Running on macOS 10.15.
//
// Copyright © 2020 Serhiy Mytrovtsiy. All rights reserved.
//
import Cocoa
import Kit
class ApplicationSettings: NSStackView {
private var updateIntervalValue: String {
Store.shared.string(key: "update-interval", defaultValue: AppUpdateInterval.silent.rawValue)
}
private var temperatureUnitsValue: String {
get { Store.shared.string(key: "temperature_units", defaultValue: "system") }
set { Store.shared.set(key: "temperature_units", value: newValue) }
}
private var combinedModulesState: Bool {
get { Store.shared.bool(key: "CombinedModules", defaultValue: false) }
set { Store.shared.set(key: "CombinedModules", value: newValue) }
}
private var combinedModulesSpacing: String {
get { Store.shared.string(key: "CombinedModules_spacing", defaultValue: "none") }
set { Store.shared.set(key: "CombinedModules_spacing", value: newValue) }
}
private var combinedModulesPopup: Bool {
get { Store.shared.bool(key: "CombinedModules_popup", defaultValue: true) }
set { Store.shared.set(key: "CombinedModules_popup", value: newValue) }
}
private let updateWindow: UpdateWindow = UpdateWindow()
private let moduleSelector: ModuleSelectorView = ModuleSelectorView()
private var updateSelector: NSPopUpButton?
private var startAtLoginBtn: NSButton?
private var uninstallHelperButton: NSButton?
private var buttonsContainer: NSStackView?
private var telemetryBtn: NSButton?
private var combinedModules: NSView?
private var combinedModulesSeparator: NSView?
private var buttons: NSView?
private var buttonsSeparator: NSView?
init() {
super.init(frame: NSRect(x: 0, y: 0, width: Constants.Settings.width, height: Constants.Settings.height))
self.translatesAutoresizingMaskIntoConstraints = false
let scrollView = ScrollableStackView()
scrollView.stackView.spacing = 0
let settings = self.settingsView()
let appSettings = self.appSettingsView()
scrollView.stackView.addArrangedSubview(self.informationView())
scrollView.stackView.addArrangedSubview(self.separatorView())
scrollView.stackView.addArrangedSubview(settings)
scrollView.stackView.addArrangedSubview(self.separatorView())
scrollView.stackView.addArrangedSubview(self.combinedModulesView())
var separator = self.separatorView()
self.combinedModulesSeparator = separator
scrollView.stackView.addArrangedSubview(separator)
scrollView.stackView.addArrangedSubview(appSettings)
separator = self.separatorView()
self.buttonsSeparator = separator
scrollView.stackView.addArrangedSubview(separator)
scrollView.stackView.addArrangedSubview(self.buttonsView())
self.toggleCombinedModulesView()
self.toggleButtonsView()
self.addArrangedSubview(scrollView)
if let settingsGrid = settings.subviews.first {
appSettings.subviews.first?.widthAnchor.constraint(equalTo: settingsGrid.widthAnchor).isActive = true
}
NotificationCenter.default.addObserver(self, selector: #selector(toggleUninstallHelperButton), name: .fanHelperState, object: nil)
}
required public init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
deinit {
NotificationCenter.default.removeObserver(self, name: .fanHelperState, object: nil)
}
public func viewWillAppear() {
self.startAtLoginBtn?.state = LaunchAtLogin.isEnabled ? .on : .off
self.telemetryBtn?.state = telemetry.isEnabled ? .on : .off
var idx = self.updateSelector?.indexOfSelectedItem ?? 0
if let items = self.updateSelector?.menu?.items {
for (i, item) in items.enumerated() {
if let obj = item.representedObject as? String, obj == self.updateIntervalValue {
idx = i
}
}
}
self.updateSelector?.selectItem(at: idx)
}
private func informationView() -> NSView {
let view = NSStackView()
view.heightAnchor.constraint(equalToConstant: 240).isActive = true
view.orientation = .vertical
view.distribution = .fill
view.alignment = .centerY
view.spacing = 0
let container: NSGridView = NSGridView()
container.heightAnchor.constraint(equalToConstant: 180).isActive = true
container.rowSpacing = 0
container.yPlacement = .center
container.xPlacement = .center
let iconView: NSImageView = NSImageView(image: NSImage(named: NSImage.Name("AppIcon"))!)
let statsName: NSTextField = TextView(frame: NSRect(x: 0, y: 0, width: view.frame.width, height: 22))
statsName.alignment = .center
statsName.font = NSFont.systemFont(ofSize: 20, weight: .regular)
statsName.stringValue = "Stats"
statsName.isSelectable = true
let versionNumber = Bundle.main.object(forInfoDictionaryKey: "CFBundleShortVersionString") as! String
let buildNumber = Bundle.main.object(forInfoDictionaryKey: "CFBundleVersion") as! String
let statsVersion: NSTextField = TextView(frame: NSRect(x: 0, y: 0, width: view.frame.width, height: 16))
statsVersion.alignment = .center
statsVersion.font = NSFont.systemFont(ofSize: 12, weight: .regular)
statsVersion.stringValue = "\(localizedString("Version")) \(versionNumber)"
statsVersion.isSelectable = true
statsVersion.toolTip = "\(localizedString("Build number")) \(buildNumber)"
let updateButton: NSButton = NSButton()
updateButton.title = localizedString("Check for update")
updateButton.bezelStyle = .rounded
updateButton.target = self
updateButton.action = #selector(self.updateAction)
container.addRow(with: [iconView])
container.addRow(with: [statsName])
container.addRow(with: [statsVersion])
container.addRow(with: [updateButton])
container.row(at: 1).height = 22
container.row(at: 2).height = 20
container.row(at: 3).height = 30
view.addArrangedSubview(container)
return view
}
private func settingsView() -> NSView {
let view: NSStackView = NSStackView()
view.orientation = .vertical
view.edgeInsets = NSEdgeInsets(
top: Constants.Settings.margin,
left: Constants.Settings.margin,
bottom: Constants.Settings.margin,
right: Constants.Settings.margin
)
view.spacing = 10
view.translatesAutoresizingMaskIntoConstraints = false
view.widthAnchor.constraint(equalToConstant: self.frame.width - 15).isActive = true
let grid: NSGridView = NSGridView(frame: NSRect(x: 0, y: 0, width: view.frame.width, height: 0))
grid.rowSpacing = 10
grid.columnSpacing = 20
grid.xPlacement = .trailing
grid.rowAlignment = .firstBaseline
grid.translatesAutoresizingMaskIntoConstraints = false
grid.setContentHuggingPriority(.defaultHigh, for: .horizontal)
grid.setContentHuggingPriority(.defaultHigh, for: .vertical)
self.updateSelector = selectView(
action: #selector(self.toggleUpdateInterval),
items: AppUpdateIntervals,
selected: self.updateIntervalValue
)
grid.addRow(with: [
self.titleView(localizedString("Check for updates")),
self.updateSelector!
])
grid.addRow(with: [
self.titleView(localizedString("Temperature")),
selectView(
action: #selector(self.toggleTemperatureUnits),
items: TemperatureUnits,
selected: self.temperatureUnitsValue
)
])
grid.addRow(with: [NSGridCell.emptyContentView, self.toggleView(
action: #selector(self.toggleDock),
state: Store.shared.bool(key: "dockIcon", defaultValue: false),
text: localizedString("Show icon in dock")
)])
self.startAtLoginBtn = self.toggleView(
action: #selector(self.toggleLaunchAtLogin),
state: LaunchAtLogin.isEnabled,
text: localizedString("Start at login")
)
grid.addRow(with: [NSGridCell.emptyContentView, self.startAtLoginBtn!])
self.telemetryBtn = self.toggleView(
action: #selector(self.toggleTelemetry),
state: telemetry.isEnabled,
text: localizedString("Share anonymous telemetry")
)
grid.addRow(with: [NSGridCell.emptyContentView, self.telemetryBtn!])
grid.addRow(with: [NSGridCell.emptyContentView, self.toggleView(
action: #selector(self.toggleCombinedModules),
state: self.combinedModulesState,
text: localizedString("Combined modules")
)])
view.addArrangedSubview(grid)
return view
}
private func combinedModulesView() -> NSView {
let view: NSStackView = NSStackView()
view.orientation = .vertical
view.edgeInsets = NSEdgeInsets(
top: Constants.Settings.margin,
left: Constants.Settings.margin,
bottom: Constants.Settings.margin,
right: Constants.Settings.margin
)
view.spacing = 10
view.translatesAutoresizingMaskIntoConstraints = false
view.widthAnchor.constraint(equalToConstant: self.frame.width - 15).isActive = true
let grid: NSGridView = NSGridView(frame: NSRect(x: 0, y: 0, width: view.frame.width, height: 0))
grid.rowSpacing = 10
grid.columnSpacing = 20
grid.xPlacement = .trailing
grid.rowAlignment = .firstBaseline
grid.translatesAutoresizingMaskIntoConstraints = false
grid.setContentHuggingPriority(.defaultHigh, for: .horizontal)
grid.setContentHuggingPriority(.defaultHigh, for: .vertical)
grid.addRow(with: [
self.titleView(localizedString("Spacing")),
selectView(
action: #selector(self.toggleCombinedModulesSpacing),
items: CombinedModulesSpacings,
selected: self.combinedModulesSpacing
)
])
grid.addRow(with: [NSGridCell.emptyContentView, self.toggleView(
action: #selector(self.toggleCombinedModulesPopup),
state: self.combinedModulesPopup,
text: localizedString("Combined details")
)])
view.addArrangedSubview(self.moduleSelector)
view.addArrangedSubview(grid)
self.combinedModules = view
return view
}
private func appSettingsView() -> NSView {
let view: NSStackView = NSStackView()
view.orientation = .vertical
view.edgeInsets = NSEdgeInsets(
top: Constants.Settings.margin,
left: Constants.Settings.margin,
bottom: Constants.Settings.margin,
right: Constants.Settings.margin
)
view.spacing = 10
view.translatesAutoresizingMaskIntoConstraints = false
view.widthAnchor.constraint(equalToConstant: self.frame.width - 15).isActive = true
let grid: NSGridView = NSGridView(frame: NSRect(x: 0, y: 0, width: view.frame.width, height: 0))
grid.rowSpacing = 10
grid.columnSpacing = 20
grid.xPlacement = .trailing
grid.rowAlignment = .firstBaseline
grid.translatesAutoresizingMaskIntoConstraints = false
grid.setContentHuggingPriority(.defaultHigh, for: .horizontal)
grid.setContentHuggingPriority(.defaultHigh, for: .vertical)
let importBtn: NSButton = NSButton()
importBtn.title = localizedString("Import")
importBtn.bezelStyle = .rounded
importBtn.target = self
importBtn.action = #selector(self.importSettings)
let exportBtn: NSButton = NSButton()
exportBtn.title = localizedString("Export")
exportBtn.bezelStyle = .rounded
exportBtn.target = self
exportBtn.action = #selector(self.exportSettings)
let resetBtn: NSButton = NSButton()
resetBtn.title = localizedString("Reset")
resetBtn.bezelStyle = .rounded
resetBtn.target = self
resetBtn.action = #selector(self.resetSettings)
resetBtn.widthAnchor.constraint(equalToConstant: 225).isActive = true
grid.addRow(with: [
self.titleView(localizedString("Settings")),
importBtn
])
grid.addRow(with: [NSGridCell.emptyContentView, exportBtn])
grid.addRow(with: [NSGridCell.emptyContentView, resetBtn])
view.addArrangedSubview(grid)
return view
}
private func buttonsView() -> NSView {
let view: NSView = NSView(frame: NSRect(x: 0, y: 0, width: self.frame.width, height: 60))
view.heightAnchor.constraint(equalToConstant: 60).isActive = true
let row = NSStackView()
row.translatesAutoresizingMaskIntoConstraints = false
row.orientation = .vertical
row.alignment = .centerY
row.distribution = .fill
self.buttonsContainer = row
let uninstall: NSButton = NSButton()
uninstall.title = localizedString("Uninstall fan helper")
uninstall.bezelStyle = .rounded
uninstall.target = self
uninstall.action = #selector(self.uninstallHelper)
self.uninstallHelperButton = uninstall
row.addArrangedSubview(uninstall)
view.addSubview(row)
NSLayoutConstraint.activate([
row.centerXAnchor.constraint(equalTo: view.centerXAnchor),
row.centerYAnchor.constraint(equalTo: view.centerYAnchor)
])
self.buttons = view
return view
}
// MARK: - helpers
private func separatorView() -> NSBox {
let view = NSBox()
view.boxType = .separator
return view
}
private func titleView(_ value: String) -> NSTextField {
let field: NSTextField = TextView(frame: NSRect(x: 0, y: 0, width: 120, height: 17))
field.font = NSFont.systemFont(ofSize: 13, weight: .regular)
field.textColor = .secondaryLabelColor
field.stringValue = value
return field
}
private func toggleView(action: Selector, state: Bool, text: String) -> NSButton {
let button: NSButton = NSButton(frame: NSRect(x: 0, y: 0, width: 30, height: 20))
button.setButtonType(.switch)
button.state = state ? .on : .off
button.title = text
button.action = action
button.isBordered = false
button.isTransparent = false
button.target = self
return button
}
private func toggleCombinedModulesView() {
self.combinedModules?.isHidden = !self.combinedModulesState
self.combinedModulesSeparator?.isHidden = !self.combinedModulesState
}
private func toggleButtonsView() {
self.buttons?.isHidden = !SMCHelper.shared.isInstalled
self.buttonsSeparator?.isHidden = !SMCHelper.shared.isInstalled
}
// MARK: - actions
@objc private func updateAction(_ sender: NSObject) {
updater.check(force: true, completion: { result, error in
if error != nil {
debug("error updater.check(): \(error!.localizedDescription)")
return
}
guard error == nil, let version: version_s = result else {
debug("download error(): \(error!.localizedDescription)")
return
}
DispatchQueue.main.async(execute: {
self.updateWindow.open(version, settingButton: true)
return
})
})
}
@objc private func toggleUpdateInterval(_ sender: NSMenuItem) {
guard let key = sender.representedObject as? String else { return }
Store.shared.set(key: "update-interval", value: key)
}
@objc private func toggleTemperatureUnits(_ sender: NSMenuItem) {
guard let key = sender.representedObject as? String else { return }
self.temperatureUnitsValue = key
}
@objc private func toggleDock(_ sender: NSButton) {
let state = sender.state
Store.shared.set(key: "dockIcon", value: state == NSControl.StateValue.on)
let dockIconStatus = state == NSControl.StateValue.on ? NSApplication.ActivationPolicy.regular : NSApplication.ActivationPolicy.accessory
NSApp.setActivationPolicy(dockIconStatus)
if state == .off {
NSApplication.shared.activate(ignoringOtherApps: true)
}
}
@objc private func toggleLaunchAtLogin(_ sender: NSButton) {
LaunchAtLogin.isEnabled = sender.state == NSControl.StateValue.on
if !Store.shared.exist(key: "runAtLoginInitialized") {
Store.shared.set(key: "runAtLoginInitialized", value: true)
}
}
@objc private func importSettings(_ sender: NSObject) {
let panel = NSOpenPanel()
panel.level = NSWindow.Level(rawValue: Int(CGWindowLevelForKey(.modalPanelWindow)))
panel.begin { (result) in
guard result.rawValue == NSApplication.ModalResponse.OK.rawValue else { return }
if let url = panel.url {
Store.shared.import(from: url)
}
}
}
@objc private func exportSettings(_ sender: NSObject) {
let panel = NSSavePanel()
panel.nameFieldStringValue = "Stats.plist"
panel.showsTagField = false
panel.level = NSWindow.Level(rawValue: Int(CGWindowLevelForKey(.modalPanelWindow)))
panel.begin { (result) in
guard result.rawValue == NSApplication.ModalResponse.OK.rawValue else { return }
if let url = panel.url {
Store.shared.export(to: url)
}
}
}
@objc private func resetSettings(_ sender: NSObject) {
let alert = NSAlert()
alert.messageText = localizedString("Reset settings")
alert.informativeText = localizedString("Reset settings text")
alert.alertStyle = .warning
alert.addButton(withTitle: localizedString("Yes"))
alert.addButton(withTitle: localizedString("No"))
if alert.runModal() == .alertFirstButtonReturn {
Store.shared.reset()
if let path = Bundle.main.resourceURL?.deletingLastPathComponent().deletingLastPathComponent().absoluteString {
asyncShell("/usr/bin/open \(path)")
NSApp.terminate(self)
}
}
}
@objc private func toggleUninstallHelperButton(_ notification: Notification) {
guard let state = notification.userInfo?["state"] as? Bool, let v = self.uninstallHelperButton else {
return
}
if state && v.superview == nil {
self.buttonsContainer?.addArrangedSubview(v)
} else if !state && v.superview != nil {
v.removeFromSuperview()
}
}
@objc private func uninstallHelper() {
SMCHelper.shared.uninstall()
}
@objc private func toggleCombinedModules(_ sender: NSButton) {
self.combinedModulesState = sender.state == NSControl.StateValue.on
self.toggleCombinedModulesView()
NotificationCenter.default.post(name: .toggleOneView, object: nil, userInfo: nil)
}
@objc private func toggleCombinedModulesSpacing(_ sender: NSMenuItem) {
guard let key = sender.representedObject as? String else { return }
self.combinedModulesSpacing = key
NotificationCenter.default.post(name: .moduleRearrange, object: nil, userInfo: nil)
}
@objc private func toggleTelemetry(_ sender: NSButton) {
telemetry.isEnabled = sender.state == NSControl.StateValue.on
}
@objc private func toggleCombinedModulesPopup(_ sender: NSButton) {
self.combinedModulesPopup = sender.state == NSControl.StateValue.on
NotificationCenter.default.post(name: .combinedModulesPopup, object: nil, userInfo: nil)
}
}
private class ModuleSelectorView: NSStackView {
init() {
super.init(frame: NSRect(x: 0, y: 0, width: 0, height: Constants.Widget.height + (Constants.Settings.margin*2)))
self.translatesAutoresizingMaskIntoConstraints = false
self.edgeInsets = NSEdgeInsets(
top: Constants.Settings.margin,
left: Constants.Settings.margin,
bottom: Constants.Settings.margin,
right: Constants.Settings.margin
)
self.spacing = Constants.Settings.margin
let background: NSVisualEffectView = {
let view = NSVisualEffectView(frame: NSRect.zero)
view.blendingMode = .withinWindow
view.material = .contentBackground
view.state = .active
view.wantsLayer = true
view.layer?.cornerRadius = 5
return view
}()
var w = self.spacing
modules.filter({ $0.available }).sorted(by: { $0.combinedPosition < $1.combinedPosition }).forEach { (m: Module) in
let v = ModulePreview(id: m.name, icon: m.config.icon)
self.addArrangedSubview(v)
w += v.frame.width + self.spacing
}
if w < 20 {
w = 20
}
self.addSubview(background, positioned: .below, relativeTo: .none)
self.setFrameSize(NSSize(width: w, height: self.frame.height))
background.setFrameSize(NSSize(width: w, height: self.frame.height))
self.widthAnchor.constraint(equalToConstant: w).isActive = true
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func mouseDown(with event: NSEvent) {
let location = convert(event.locationInWindow, from: nil)
guard let targetIdx = self.views.firstIndex(where: { $0.hitTest(location) != nil }),
let window = self.window, self.views[targetIdx].identifier != nil else {
super.mouseDragged(with: event)
return
}
let view = self.views[targetIdx]
let copy = ViewCopy(view)
copy.zPosition = 2
copy.transform = CATransform3DMakeScale(0.9, 0.9, 1)
// hide the original view, show the copy
view.subviews.forEach({ $0.isHidden = true })
self.layer?.addSublayer(copy)
// hide the copy view, show the original
defer {
copy.removeFromSuperlayer()
view.subviews.forEach({ $0.isHidden = false })
}
var newIdx = -1
let originCenter = view.frame.midX
let originX = view.frame.origin.x
let p0 = convert(event.locationInWindow, from: nil).x
window.trackEvents(matching: [.leftMouseDragged, .leftMouseUp], timeout: 1e6, mode: .eventTracking) { event, stop in
guard let event = event else {
stop.pointee = true
return
}
if event.type == .leftMouseDragged {
let p1 = self.convert(event.locationInWindow, from: nil).x
let diff = p1 - p0
CATransaction.begin()
CATransaction.setDisableActions(true)
copy.frame.origin.x = originX + diff
CATransaction.commit()
let reordered = self.views.map{
(view: $0, x: $0 !== view ? $0.frame.midX : originCenter + diff)
}.sorted{ $0.x < $1.x }.map { $0.view }
guard let nextIndex = reordered.firstIndex(of: view),
let prevIndex = self.views.firstIndex(of: view) else {
stop.pointee = true
return
}
if nextIndex != prevIndex {
newIdx = nextIndex
view.removeFromSuperviewWithoutNeedingDisplay()
self.insertArrangedSubview(view, at: newIdx)
self.layoutSubtreeIfNeeded()
for (i, v) in self.views(in: .leading).compactMap({$0 as? ModulePreview}).enumerated() {
if let m = modules.first(where: { $0.name == v.identifier?.rawValue }) {
m.combinedPosition = i
}
}
}
} else {
if newIdx != -1, let view = self.views[newIdx] as? ModulePreview, let id = view.identifier?.rawValue {
NotificationCenter.default.post(name: .moduleRearrange, object: nil, userInfo: ["id": id])
}
view.mouseUp(with: event)
stop.pointee = true
}
}
}
}
internal class ModulePreview: NSStackView {
private let id: String
private let imageView: NSImageView
public init(id: String, icon: NSImage?) {
self.id = id
self.imageView = NSImageView(frame: NSRect(origin: .zero, size: NSSize(width: Constants.Widget.height, height: Constants.Widget.height)))
let size: CGSize = CGSize(width: Constants.Widget.height + (Constants.Widget.spacing * 2), height: Constants.Widget.height)
super.init(frame: NSRect(x: 0, y: 0, width: size.width, height: size.height))
self.wantsLayer = true
self.layer?.cornerRadius = 2
self.layer?.borderColor = NSColor(red: 221/255, green: 221/255, blue: 221/255, alpha: 1).cgColor
self.layer?.borderWidth = 1
self.layer?.backgroundColor = NSColor.white.cgColor
self.identifier = NSUserInterfaceItemIdentifier(rawValue: id)
self.toolTip = localizedString("Move module", id)
self.orientation = .vertical
self.distribution = .fill
self.alignment = .centerY
self.spacing = 0
self.imageView.image = icon
self.imageView.contentTintColor = self.isDarkMode ? .textBackgroundColor : .textColor
self.addArrangedSubview(self.imageView)
self.addTrackingArea(NSTrackingArea(
rect: NSRect(x: 0, y: 0, width: size.width, height: size.height),
options: [NSTrackingArea.Options.activeAlways, NSTrackingArea.Options.mouseEnteredAndExited, NSTrackingArea.Options.activeInActiveApp],
owner: self,
userInfo: nil
))
NSLayoutConstraint.activate([
self.widthAnchor.constraint(equalToConstant: size.width),
self.heightAnchor.constraint(equalToConstant: size.height)
])
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
override func updateLayer() {
self.imageView.contentTintColor = self.isDarkMode ? .textBackgroundColor : .textColor
}
override func mouseEntered(with: NSEvent) {
NSCursor.pointingHand.set()
}
override func mouseExited(with: NSEvent) {
NSCursor.arrow.set()
}
}