2025-12-07 00:10:35 +01:00
|
|
|
|
import SwiftUI
|
|
|
|
|
|
|
|
|
|
|
|
struct AboutSettings: View {
|
2025-12-08 00:18:16 +01:00
|
|
|
|
weak var updater: UpdaterProviding?
|
2025-12-07 00:10:35 +01:00
|
|
|
|
@State private var iconHover = false
|
2025-12-08 00:18:16 +01:00
|
|
|
|
@AppStorage("autoUpdateEnabled") private var autoCheckEnabled = true
|
|
|
|
|
|
@State private var didLoadUpdaterState = false
|
2025-12-07 00:10:35 +01:00
|
|
|
|
|
|
|
|
|
|
var body: some View {
|
|
|
|
|
|
VStack(spacing: 8) {
|
|
|
|
|
|
let appIcon = NSApplication.shared.applicationIconImage ?? CritterIconRenderer.makeIcon(blink: 0)
|
|
|
|
|
|
Button {
|
2026-01-30 03:15:10 +01:00
|
|
|
|
if let url = URL(string: "https://github.com/openclaw/openclaw") {
|
2025-12-07 00:10:35 +01:00
|
|
|
|
NSWorkspace.shared.open(url)
|
|
|
|
|
|
}
|
|
|
|
|
|
} label: {
|
|
|
|
|
|
Image(nsImage: appIcon)
|
|
|
|
|
|
.resizable()
|
2025-12-07 15:34:44 +01:00
|
|
|
|
.frame(width: 160, height: 160)
|
|
|
|
|
|
.cornerRadius(24)
|
|
|
|
|
|
.shadow(color: self.iconHover ? .accentColor.opacity(0.25) : .clear, radius: 10)
|
|
|
|
|
|
.scaleEffect(self.iconHover ? 1.05 : 1.0)
|
2025-12-07 00:10:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
.buttonStyle(.plain)
|
2025-12-07 14:46:54 +01:00
|
|
|
|
.focusable(false)
|
2025-12-13 17:18:22 +00:00
|
|
|
|
.pointingHandCursor()
|
2025-12-07 00:10:35 +01:00
|
|
|
|
.onHover { hover in
|
|
|
|
|
|
withAnimation(.spring(response: 0.3, dampingFraction: 0.72)) { self.iconHover = hover }
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VStack(spacing: 3) {
|
2026-01-30 03:15:10 +01:00
|
|
|
|
Text("OpenClaw")
|
2025-12-07 00:10:35 +01:00
|
|
|
|
.font(.title3.bold())
|
|
|
|
|
|
Text("Version \(self.versionString)")
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
if let buildTimestamp {
|
2025-12-07 00:30:47 +01:00
|
|
|
|
Text("Built \(buildTimestamp)\(self.buildSuffix)")
|
2025-12-07 00:10:35 +01:00
|
|
|
|
.font(.footnote)
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
}
|
|
|
|
|
|
Text("Menu bar companion for notifications, screenshots, and privileged agent actions.")
|
|
|
|
|
|
.font(.footnote)
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
.multilineTextAlignment(.center)
|
|
|
|
|
|
.padding(.horizontal, 18)
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
VStack(alignment: .center, spacing: 6) {
|
|
|
|
|
|
AboutLinkRow(
|
|
|
|
|
|
icon: "chevron.left.slash.chevron.right",
|
|
|
|
|
|
title: "GitHub",
|
2026-01-30 03:15:10 +01:00
|
|
|
|
url: "https://github.com/openclaw/openclaw")
|
|
|
|
|
|
AboutLinkRow(icon: "globe", title: "Website", url: "https://openclaw.ai")
|
2025-12-07 00:10:35 +01:00
|
|
|
|
AboutLinkRow(icon: "bird", title: "Twitter", url: "https://twitter.com/steipete")
|
|
|
|
|
|
AboutLinkRow(icon: "envelope", title: "Email", url: "mailto:peter@steipete.me")
|
|
|
|
|
|
}
|
|
|
|
|
|
.frame(maxWidth: .infinity)
|
|
|
|
|
|
.multilineTextAlignment(.center)
|
|
|
|
|
|
.padding(.vertical, 10)
|
|
|
|
|
|
|
2025-12-08 00:18:16 +01:00
|
|
|
|
if let updater {
|
|
|
|
|
|
Divider()
|
|
|
|
|
|
.padding(.vertical, 8)
|
|
|
|
|
|
|
|
|
|
|
|
if updater.isAvailable {
|
|
|
|
|
|
VStack(spacing: 10) {
|
|
|
|
|
|
Toggle("Check for updates automatically", isOn: self.$autoCheckEnabled)
|
|
|
|
|
|
.toggleStyle(.checkbox)
|
|
|
|
|
|
.frame(maxWidth: .infinity, alignment: .center)
|
|
|
|
|
|
|
|
|
|
|
|
Button("Check for Updates…") { updater.checkForUpdates(nil) }
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
Text("Updates unavailable in this build.")
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
.padding(.top, 4)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-12-07 00:10:35 +01:00
|
|
|
|
Text("© 2025 Peter Steinberger — MIT License.")
|
|
|
|
|
|
.font(.footnote)
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
.padding(.top, 4)
|
|
|
|
|
|
|
|
|
|
|
|
Spacer()
|
|
|
|
|
|
}
|
|
|
|
|
|
.frame(maxWidth: .infinity, maxHeight: .infinity)
|
|
|
|
|
|
.padding(.top, 4)
|
|
|
|
|
|
.padding(.horizontal, 24)
|
|
|
|
|
|
.padding(.bottom, 24)
|
2025-12-08 00:18:16 +01:00
|
|
|
|
.onAppear {
|
|
|
|
|
|
guard let updater, !self.didLoadUpdaterState else { return }
|
|
|
|
|
|
// Keep Sparkle’s auto-check setting in sync with the persisted toggle.
|
|
|
|
|
|
updater.automaticallyChecksForUpdates = self.autoCheckEnabled
|
2025-12-23 01:42:33 +01:00
|
|
|
|
updater.automaticallyDownloadsUpdates = self.autoCheckEnabled
|
2025-12-08 00:18:16 +01:00
|
|
|
|
self.didLoadUpdaterState = true
|
|
|
|
|
|
}
|
|
|
|
|
|
.onChange(of: self.autoCheckEnabled) { _, newValue in
|
|
|
|
|
|
self.updater?.automaticallyChecksForUpdates = newValue
|
2025-12-23 01:42:33 +01:00
|
|
|
|
self.updater?.automaticallyDownloadsUpdates = newValue
|
2025-12-08 00:18:16 +01:00
|
|
|
|
}
|
2025-12-07 00:10:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private var versionString: String {
|
|
|
|
|
|
let version = Bundle.main.object(forInfoDictionaryKey: "CFBundleShortVersionString") as? String ?? "dev"
|
|
|
|
|
|
let build = Bundle.main.object(forInfoDictionaryKey: "CFBundleVersion") as? String
|
|
|
|
|
|
return build.map { "\(version) (\($0))" } ?? version
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private var buildTimestamp: String? {
|
2026-01-30 03:15:10 +01:00
|
|
|
|
guard
|
|
|
|
|
|
let raw =
|
2026-02-15 05:38:07 +01:00
|
|
|
|
(Bundle.main.object(forInfoDictionaryKey: "OpenClawBuildTimestamp") as? String) ??
|
|
|
|
|
|
(Bundle.main.object(forInfoDictionaryKey: "OpenClawBuildTimestamp") as? String)
|
2026-01-04 16:24:10 +01:00
|
|
|
|
else { return nil }
|
2025-12-07 00:10:35 +01:00
|
|
|
|
let parser = ISO8601DateFormatter()
|
|
|
|
|
|
parser.formatOptions = [.withInternetDateTime]
|
|
|
|
|
|
guard let date = parser.date(from: raw) else { return raw }
|
|
|
|
|
|
|
|
|
|
|
|
let formatter = DateFormatter()
|
|
|
|
|
|
formatter.dateStyle = .medium
|
|
|
|
|
|
formatter.timeStyle = .short
|
|
|
|
|
|
formatter.locale = .current
|
|
|
|
|
|
return formatter.string(from: date)
|
|
|
|
|
|
}
|
2025-12-07 00:30:47 +01:00
|
|
|
|
|
|
|
|
|
|
private var gitCommit: String {
|
2026-01-30 03:15:10 +01:00
|
|
|
|
(Bundle.main.object(forInfoDictionaryKey: "OpenClawGitCommit") as? String) ??
|
|
|
|
|
|
(Bundle.main.object(forInfoDictionaryKey: "OpenClawGitCommit") as? String) ??
|
|
|
|
|
|
"unknown"
|
2025-12-07 00:30:47 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private var bundleID: String {
|
|
|
|
|
|
Bundle.main.bundleIdentifier ?? "unknown"
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private var buildSuffix: String {
|
|
|
|
|
|
let git = self.gitCommit
|
|
|
|
|
|
guard !git.isEmpty, git != "unknown" else { return "" }
|
|
|
|
|
|
|
|
|
|
|
|
var suffix = " (\(git)"
|
|
|
|
|
|
#if DEBUG
|
|
|
|
|
|
suffix += " DEBUG"
|
|
|
|
|
|
#endif
|
|
|
|
|
|
suffix += ")"
|
|
|
|
|
|
return suffix
|
|
|
|
|
|
}
|
2025-12-07 00:10:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@MainActor
|
|
|
|
|
|
private struct AboutLinkRow: View {
|
|
|
|
|
|
let icon: String
|
|
|
|
|
|
let title: String
|
|
|
|
|
|
let url: String
|
|
|
|
|
|
|
|
|
|
|
|
@State private var hovering = false
|
|
|
|
|
|
|
|
|
|
|
|
var body: some View {
|
|
|
|
|
|
Button {
|
|
|
|
|
|
if let url = URL(string: url) { NSWorkspace.shared.open(url) }
|
|
|
|
|
|
} label: {
|
|
|
|
|
|
HStack(spacing: 6) {
|
|
|
|
|
|
Image(systemName: self.icon)
|
|
|
|
|
|
Text(self.title)
|
|
|
|
|
|
.underline(self.hovering, color: .accentColor)
|
|
|
|
|
|
}
|
|
|
|
|
|
.foregroundColor(.accentColor)
|
|
|
|
|
|
}
|
|
|
|
|
|
.buttonStyle(.plain)
|
|
|
|
|
|
.onHover { self.hovering = $0 }
|
2025-12-13 17:18:22 +00:00
|
|
|
|
.pointingHandCursor()
|
2025-12-07 00:10:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-12-07 00:30:47 +01:00
|
|
|
|
|
|
|
|
|
|
private struct AboutMetaRow: View {
|
|
|
|
|
|
let label: String
|
|
|
|
|
|
let value: String
|
|
|
|
|
|
|
|
|
|
|
|
var body: some View {
|
|
|
|
|
|
HStack {
|
|
|
|
|
|
Text(self.label)
|
|
|
|
|
|
.foregroundStyle(.secondary)
|
|
|
|
|
|
Spacer()
|
|
|
|
|
|
Text(self.value)
|
|
|
|
|
|
.font(.caption.monospaced())
|
|
|
|
|
|
.foregroundStyle(.primary)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-12-09 18:04:11 +01:00
|
|
|
|
|
|
|
|
|
|
#if DEBUG
|
|
|
|
|
|
struct AboutSettings_Previews: PreviewProvider {
|
|
|
|
|
|
private static let updater = DisabledUpdaterController()
|
|
|
|
|
|
static var previews: some View {
|
|
|
|
|
|
AboutSettings(updater: updater)
|
|
|
|
|
|
.frame(width: SettingsTab.windowWidth, height: SettingsTab.windowHeight)
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
#endif
|