Files
openclaw/apps/macos/Sources/OpenClaw/SettingsComponents.swift
2026-05-18 15:37:36 +01:00

180 lines
4.8 KiB
Swift

import SwiftUI
enum SettingsLayout {
static let sidebarWidth: CGFloat = 250
static let detailHorizontalPadding: CGFloat = 22
static let detailVerticalPadding: CGFloat = 18
static let nestedSidebarWidth: CGFloat = 260
static let detailBottomPadding: CGFloat = 16
}
extension View {
func settingsDetailContent() -> some View {
self
.frame(maxWidth: .infinity, alignment: .leading)
.padding(.vertical, 4)
.padding(.bottom, SettingsLayout.detailBottomPadding)
}
}
struct SettingsPageHeader: View {
let title: String
let subtitle: String?
init(title: String, subtitle: String? = nil) {
self.title = title
self.subtitle = subtitle
}
var body: some View {
VStack(alignment: .leading, spacing: 5) {
Text(self.title)
.font(.title3.weight(.semibold))
if let subtitle, !subtitle.isEmpty {
Text(subtitle)
.font(.callout)
.foregroundStyle(.secondary)
.fixedSize(horizontal: false, vertical: true)
}
}
}
}
struct SettingsSection<Content: View>: View {
let title: String
let content: Content
init(_ title: String, @ViewBuilder content: () -> Content) {
self.title = title
self.content = content()
}
var body: some View {
VStack(alignment: .leading, spacing: 12) {
Text(self.title)
.font(.headline)
VStack(alignment: .leading, spacing: 12) {
self.content
}
.frame(maxWidth: .infinity, alignment: .leading)
}
}
}
struct SettingsCardGroup<Content: View>: View {
let title: String
let content: Content
init(_ title: String, @ViewBuilder content: () -> Content) {
self.title = title
self.content = content()
}
var body: some View {
VStack(alignment: .leading, spacing: 8) {
Text(self.title)
.font(.subheadline.weight(.semibold))
.foregroundStyle(.secondary)
VStack(alignment: .leading, spacing: 0) {
self.content
}
.frame(maxWidth: .infinity, alignment: .leading)
.background(.quaternary.opacity(0.38), in: RoundedRectangle(cornerRadius: 12, style: .continuous))
.overlay {
RoundedRectangle(cornerRadius: 12, style: .continuous)
.strokeBorder(.white.opacity(0.055))
}
}
}
}
struct SettingsCardRow<Content: View>: View {
let title: String
let subtitle: String?
var showsDivider = true
let content: Content
init(
title: String,
subtitle: String? = nil,
showsDivider: Bool = true,
@ViewBuilder content: () -> Content)
{
self.title = title
self.subtitle = subtitle
self.showsDivider = showsDivider
self.content = content()
}
var body: some View {
HStack(alignment: .center, spacing: 18) {
VStack(alignment: .leading, spacing: 3) {
Text(self.title)
.font(.callout.weight(.medium))
if let subtitle, !subtitle.isEmpty {
Text(subtitle)
.font(.footnote)
.foregroundStyle(.secondary)
.fixedSize(horizontal: false, vertical: true)
}
}
Spacer(minLength: 18)
self.content
}
.frame(maxWidth: .infinity, alignment: .leading)
.padding(.horizontal, 14)
.padding(.vertical, 11)
.overlay(alignment: .bottom) {
if self.showsDivider {
Divider()
.padding(.leading, 14)
}
}
}
}
struct SettingsCardToggleRow: View {
let title: String
let subtitle: String?
@Binding var binding: Bool
var showsDivider = true
var body: some View {
SettingsCardRow(
title: self.title,
subtitle: self.subtitle,
showsDivider: self.showsDivider)
{
Toggle(self.title, isOn: self.$binding)
.labelsHidden()
.toggleStyle(.switch)
}
}
}
struct SettingsToggleRow: View {
let title: String
let subtitle: String?
@Binding var binding: Bool
var body: some View {
VStack(alignment: .leading, spacing: 6) {
Toggle(isOn: self.$binding) {
Text(self.title)
.font(.body)
}
.toggleStyle(.checkbox)
if let subtitle, !subtitle.isEmpty {
Text(subtitle)
.font(.footnote)
.foregroundStyle(.tertiary)
.fixedSize(horizontal: false, vertical: true)
}
}
}
}