Implement comprehensive release notes feature: - RELEASE_NOTES.md with version 1.0 and 1.1 content in English - VersionManager to track app versions and detect updates - ReleaseNotesView with native markdown rendering - Auto-popup sheet on first launch after version update - Manual access via "What's New" button in General Settings Features: - Markdown-based release notes stored in app bundle - Automatic version detection using CFBundleShortVersionString - UserDefaults tracking of last seen version - Dismissable sheet with "Done" button - Settings button shows current version number Technical implementation: - VersionManager singleton for version tracking - Sheet presentation in MainTabView on new version - Settings integration with sparkles icon - Native SwiftUI Text markdown rendering - Bundle resource loading for RELEASE_NOTES.md Release notes content: - Version 1.1: iOS 26 features, floating buttons, progress tracking - Version 1.0: Initial release features and capabilities
134 lines
4.6 KiB
Swift
134 lines
4.6 KiB
Swift
//
|
|
// SettingsGeneralView.swift
|
|
// readeck
|
|
//
|
|
// Created by Ilyas Hallak on 29.06.25.
|
|
//
|
|
|
|
import SwiftUI
|
|
|
|
struct SettingsGeneralView: View {
|
|
@State private var viewModel: SettingsGeneralViewModel
|
|
@State private var showReleaseNotes = false
|
|
|
|
init(viewModel: SettingsGeneralViewModel = SettingsGeneralViewModel()) {
|
|
self.viewModel = viewModel
|
|
}
|
|
|
|
var body: some View {
|
|
VStack(spacing: 20) {
|
|
SectionHeader(title: "General Settings".localized, icon: "gear")
|
|
.padding(.bottom, 4)
|
|
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("General")
|
|
.font(.headline)
|
|
|
|
// What's New Button
|
|
Button(action: {
|
|
showReleaseNotes = true
|
|
}) {
|
|
HStack {
|
|
Label("What's New", systemImage: "sparkles")
|
|
Spacer()
|
|
Text("Version \(VersionManager.shared.currentVersion)")
|
|
.font(.caption)
|
|
.foregroundColor(.secondary)
|
|
Image(systemName: "chevron.right")
|
|
.font(.caption)
|
|
.foregroundColor(.secondary)
|
|
}
|
|
}
|
|
.buttonStyle(.plain)
|
|
|
|
Toggle("Read Aloud Feature", isOn: $viewModel.enableTTS)
|
|
.toggleStyle(.switch)
|
|
.onChange(of: viewModel.enableTTS) {
|
|
Task {
|
|
await viewModel.saveGeneralSettings()
|
|
}
|
|
}
|
|
Text("Activate the Read Aloud Feature to read aloud your articles. This is a really early preview and might not work perfectly.")
|
|
.font(.footnote)
|
|
}
|
|
|
|
// Reading Settings
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Open external links in".localized)
|
|
.font(.headline)
|
|
Picker("urlOpener", selection: $viewModel.urlOpener) {
|
|
ForEach(UrlOpener.allCases, id: \.self) { urlOpener in
|
|
Text(urlOpener.displayName.localized).tag(urlOpener)
|
|
}
|
|
}
|
|
.pickerStyle(.segmented)
|
|
.onChange(of: viewModel.urlOpener) {
|
|
Task {
|
|
await viewModel.saveGeneralSettings()
|
|
}
|
|
}
|
|
}
|
|
|
|
#if DEBUG
|
|
// Sync Settings
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Sync Settings")
|
|
.font(.headline)
|
|
Toggle("Automatic sync", isOn: $viewModel.autoSyncEnabled)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
if viewModel.autoSyncEnabled {
|
|
HStack {
|
|
Text("Sync interval")
|
|
Spacer()
|
|
Stepper("\(viewModel.syncInterval) minutes", value: $viewModel.syncInterval, in: 1...60)
|
|
}
|
|
}
|
|
}
|
|
|
|
// Reading Settings
|
|
VStack(alignment: .leading, spacing: 12) {
|
|
Text("Reading Settings")
|
|
.font(.headline)
|
|
Toggle("Safari Reader Mode", isOn: $viewModel.enableReaderMode)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
Toggle("Automatically mark articles as read", isOn: $viewModel.autoMarkAsRead)
|
|
.toggleStyle(SwitchToggleStyle())
|
|
}
|
|
|
|
// 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)
|
|
}
|
|
}
|
|
#endif
|
|
|
|
}
|
|
.sheet(isPresented: $showReleaseNotes) {
|
|
ReleaseNotesView()
|
|
}
|
|
.task {
|
|
await viewModel.loadGeneralSettings()
|
|
}
|
|
}
|
|
}
|
|
|
|
#Preview {
|
|
SettingsGeneralView(viewModel: .init(
|
|
MockUseCaseFactory()
|
|
))
|
|
}
|