- Split TabView and Sidebar logic into PhoneTabView, PadSidebarView, SidebarTab, and BookmarkState for better device adaptation - Remove old SettingsViewModel, introduce SettingsGeneralViewModel and SettingsServerViewModel for modular settings - Update BookmarksView and BookmarksViewModel for new paginated and filtered data model - Clean up and modularize settings UI (SettingsGeneralView, SettingsServerView, FontSettingsView) - Remove obsolete files (old TabView, File.swift, SettingsViewModel, etc.) - Add BookmarksPageDto and update related data flow - Various UI/UX improvements and code cleanup BREAKING: Settings and navigation structure refactored, old settings logic removed
174 lines
5.8 KiB
Swift
174 lines
5.8 KiB
Swift
//
|
|
// SettingsGeneralView.swift
|
|
// readeck
|
|
//
|
|
// Created by Ilyas Hallak on 29.06.25.
|
|
//
|
|
|
|
import SwiftUI
|
|
// SectionHeader wird jetzt zentral importiert
|
|
|
|
struct SettingsGeneralView: View {
|
|
@State private var viewModel = SettingsGeneralViewModel()
|
|
|
|
var body: some View {
|
|
VStack(spacing: 20) {
|
|
SectionHeader(title: "Allgemeine Einstellungen", icon: "gear")
|
|
.padding(.bottom, 4)
|
|
|
|
// Theme
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Theme")
|
|
.font(.headline)
|
|
Picker("Theme", selection: $viewModel.selectedTheme) {
|
|
ForEach(Theme.allCases, id: \.self) { theme in
|
|
Text(theme.displayName).tag(theme)
|
|
}
|
|
}
|
|
.pickerStyle(.segmented)
|
|
}
|
|
|
|
// Sync Settings
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Sync-Einstellungen")
|
|
.font(.headline)
|
|
Toggle("Automatischer Sync", isOn: $viewModel.autoSyncEnabled)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
if viewModel.autoSyncEnabled {
|
|
HStack {
|
|
Text("Sync-Intervall")
|
|
Spacer()
|
|
Stepper("\(viewModel.syncInterval) Minuten", value: $viewModel.syncInterval, in: 1...60)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Reading Settings
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Leseeinstellungen")
|
|
.font(.headline)
|
|
Toggle("Safari Reader Modus", isOn: $viewModel.enableReaderMode)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
Toggle("Externe Links in In-App Safari öffnen", isOn: $viewModel.openExternalLinksInApp)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
Toggle("Artikel automatisch als gelesen markieren", isOn: $viewModel.autoMarkAsRead)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
}
|
|
|
|
// Data Management
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Datenmanagement")
|
|
.font(.headline)
|
|
Button(role: .destructive) {
|
|
Task {
|
|
// await viewModel.clearCache()
|
|
}
|
|
} label: {
|
|
HStack {
|
|
Image(systemName: "trash")
|
|
.foregroundColor(.red)
|
|
Text("Cache leeren")
|
|
.foregroundColor(.red)
|
|
Spacer()
|
|
}
|
|
}
|
|
Button(role: .destructive) {
|
|
Task {
|
|
// await viewModel.resetSettings()
|
|
}
|
|
} label: {
|
|
HStack {
|
|
Image(systemName: "arrow.clockwise")
|
|
.foregroundColor(.red)
|
|
Text("Einstellungen zurücksetzen")
|
|
.foregroundColor(.red)
|
|
Spacer()
|
|
}
|
|
}
|
|
}
|
|
|
|
// App Info
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Über die App")
|
|
.font(.headline)
|
|
HStack {
|
|
Image(systemName: "info.circle")
|
|
.foregroundColor(.secondary)
|
|
Text("Version \(viewModel.appVersion)")
|
|
Spacer()
|
|
}
|
|
HStack {
|
|
Image(systemName: "person.crop.circle")
|
|
.foregroundColor(.secondary)
|
|
Text("Entwickler: \(viewModel.developerName)")
|
|
Spacer()
|
|
}
|
|
HStack {
|
|
Image(systemName: "globe")
|
|
.foregroundColor(.secondary)
|
|
Link("Website", destination: URL(string: "https://example.com")!)
|
|
Spacer()
|
|
}
|
|
}
|
|
|
|
// Save Button
|
|
Button(action: {
|
|
Task {
|
|
await viewModel.saveGeneralSettings()
|
|
}
|
|
}) {
|
|
HStack {
|
|
Text("Einstellungen speichern")
|
|
.fontWeight(.semibold)
|
|
}
|
|
.frame(maxWidth: .infinity)
|
|
.padding()
|
|
.background(Color.accentColor)
|
|
.foregroundColor(.white)
|
|
.cornerRadius(10)
|
|
}
|
|
// Messages
|
|
if let successMessage = viewModel.successMessage {
|
|
HStack {
|
|
Image(systemName: "checkmark.circle.fill")
|
|
.foregroundColor(.green)
|
|
Text(successMessage)
|
|
.foregroundColor(.green)
|
|
.font(.caption)
|
|
}
|
|
}
|
|
if let errorMessage = viewModel.errorMessage {
|
|
HStack {
|
|
Image(systemName: "exclamationmark.triangle.fill")
|
|
.foregroundColor(.red)
|
|
Text(errorMessage)
|
|
.foregroundColor(.red)
|
|
.font(.caption)
|
|
}
|
|
}
|
|
}
|
|
.task {
|
|
await viewModel.loadGeneralSettings()
|
|
}
|
|
}
|
|
}
|
|
|
|
enum Theme: String, CaseIterable {
|
|
case system = "system"
|
|
case light = "light"
|
|
case dark = "dark"
|
|
|
|
var displayName: String {
|
|
switch self {
|
|
case .system: return "System"
|
|
case .light: return "Hell"
|
|
case .dark: return "Dunkel"
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#Preview {
|
|
SettingsGeneralView()
|
|
}
|