punchnet-macos/punchnet/Views/Settings/SettingsView.swift
2026-03-19 22:44:50 +08:00

146 lines
4.9 KiB
Swift
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

//
// SettingsView.swift
// punchnet
//
// Created by on 2026/1/16.
//
import SwiftUI
struct SettingsView: View {
@State private var columnVisibility: NavigationSplitViewVisibility = .all
@State private var state = SettingsState()
@State private var selectedMenu: MenuItem = .accout
enum MenuItem: String, CaseIterable {
case accout = "账号"
case network = "网络"
case device = "设备"
case system = "软件"
case about = "关于"
//
var icon: String {
switch self {
case .accout:
return "person.crop.circle.fill"
case .network:
return "network"
case .device:
return "laptopcomputer"
case .system:
return "gearshape.fill"
case .about:
return "info.circle.fill"
}
}
}
var body: some View {
NavigationSplitView(columnVisibility: $columnVisibility) {
// MARK: -
VStack(alignment: .leading, spacing: 20) {
Text("设置")
.font(.system(size: 24, weight: .bold))
.padding(.horizontal, 16)
.padding(.top, 24)
VStack(spacing: 4) {
ForEach(MenuItem.allCases, id: \.self) { menu in
SidebarItem(
icon: menu.icon,
title: menu.rawValue,
isSelected: selectedMenu == menu
)
.onTapGesture {
// 使 spring
withAnimation(.spring(response: 0.35, dampingFraction: 0.8)) {
self.selectedMenu = menu
}
}
}
Spacer()
}
.padding(.horizontal, 12)
}
.background(VisualEffectView(material: .sidebar, blendingMode: .behindWindow))
.navigationSplitViewColumnWidth(min: 180, ideal: 200, max: 250)
} detail: {
// MARK: -
ZStack(alignment: .topLeading) {
//
VisualEffectView(material: .hudWindow, blendingMode: .behindWindow)
.ignoresSafeArea()
VStack(alignment: .leading, spacing: 0) {
Group {
// 使 ID SwiftUI transition
switch self.selectedMenu {
case .accout:
SettingsAccountView(state: self.state)
case .network:
SettingsNetworkView(state: self.state)
case .device:
SettingsDeviceView()
case .system:
SettingsSystemView()
case .about:
SettingsAboutView()
}
}
.id(selectedMenu) //
.transition(.asymmetric(
insertion: .move(edge: .bottom).combined(with: .opacity),
removal: .opacity
))
Spacer()
}
.padding(32) //
.frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .topLeading)
}
}
}
}
// MARK: -
struct SidebarItem: View {
let icon: String
let title: String
let isSelected: Bool
@State private var isHovering = false
var body: some View {
HStack(spacing: 12) {
Image(systemName: icon)
.font(.system(size: 14, weight: .medium))
.frame(width: 20)
Text(title)
.font(.system(size: 14, weight: .medium))
Spacer()
}
.padding(.horizontal, 12)
.padding(.vertical, 8)
.foregroundColor(isSelected ? .white : .primary.opacity(0.8))
.background {
if isSelected {
RoundedRectangle(cornerRadius: 8, style: .continuous)
.fill(Color.blue.gradient) //
} else if isHovering {
RoundedRectangle(cornerRadius: 8, style: .continuous)
.fill(Color.primary.opacity(0.05))
}
}
.onHover {
isHovering = $0
}
.contentShape(Rectangle())
}
}
#Preview {
SettingsView()
}