This commit introduces a comprehensive refactoring of the tag management system, replacing the previous API-based approach with a Core Data-first strategy for improved performance and offline support. Major Changes: Tag Management Architecture: - Add CoreDataTagManagementView using @FetchRequest for reactive updates - Implement cache-first sync strategy in LabelsRepository - Create SyncTagsUseCase following Clean Architecture principles - Add TagSortOrder enum for configurable tag sorting (by count/alphabetically) - Mark LegacyTagManagementView as deprecated Share Extension Improvements: - Replace API-based tag loading with Core Data queries - Display top 150 tags sorted by usage count - Remove unnecessary label fetching logic - Add "Most used tags" localized title - Improve offline bookmark tag management Main App Enhancements: - Add tag sync triggers in AddBookmarkView and BookmarkLabelsView - Implement user-configurable tag sorting in settings - Add sort order indicator labels with localization - Automatic UI updates via SwiftUI @FetchRequest reactivity Settings & Configuration: - Add TagSortOrder setting with persistence - Refactor Settings model structure - Add FontFamily and FontSize domain models - Improve settings repository with tag sort order support Use Case Layer: - Add SyncTagsUseCase for background tag synchronization - Update UseCaseFactory with tag sync support - Add mock implementations for testing Localization: - Add German and English translations for: - "Most used tags" - "Sorted by usage count" - "Sorted alphabetically" Technical Improvements: - Batch tag updates with conflict detection - Background sync with silent failure handling - Reduced server load through local caching - Better separation of concerns following Clean Architecture
116 lines
3.3 KiB
Swift
116 lines
3.3 KiB
Swift
//
|
|
// readeckApp.swift
|
|
// readeck
|
|
//
|
|
// Created by Ilyas Hallak on 10.06.25.
|
|
//
|
|
|
|
import SwiftUI
|
|
import netfox
|
|
|
|
@main
|
|
struct readeckApp: App {
|
|
@State private var appViewModel = AppViewModel()
|
|
@StateObject private var appSettings = AppSettings()
|
|
@Environment(\.scenePhase) private var scenePhase
|
|
|
|
var body: some Scene {
|
|
WindowGroup {
|
|
Group {
|
|
if appViewModel.hasFinishedSetup {
|
|
MainTabView()
|
|
} else {
|
|
OnboardingServerView()
|
|
.padding()
|
|
}
|
|
}
|
|
.environmentObject(appSettings)
|
|
.environment(\.managedObjectContext, CoreDataManager.shared.context)
|
|
.preferredColorScheme(appSettings.theme.colorScheme)
|
|
.onAppear {
|
|
#if DEBUG
|
|
NFX.sharedInstance().start()
|
|
#endif
|
|
Task {
|
|
await loadAppSettings()
|
|
}
|
|
}
|
|
.onReceive(NotificationCenter.default.publisher(for: .settingsChanged)) { _ in
|
|
Task {
|
|
await loadAppSettings()
|
|
}
|
|
}
|
|
.onChange(of: scenePhase) { oldPhase, newPhase in
|
|
if newPhase == .active {
|
|
Task {
|
|
await appViewModel.onAppResume()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private func loadAppSettings() async {
|
|
let settingsRepository = SettingsRepository()
|
|
let settings = try? await settingsRepository.loadSettings()
|
|
await MainActor.run {
|
|
appSettings.settings = settings
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
struct TestView: View {
|
|
var body: some View {
|
|
if #available(iOS 26.0, *) {
|
|
Text("hello")
|
|
.toolbar {
|
|
ToolbarSpacer(.flexible)
|
|
|
|
ToolbarItem {
|
|
Button {
|
|
|
|
} label: {
|
|
Label("Favorite", systemImage: "share")
|
|
.symbolVariant(.none)
|
|
}
|
|
}
|
|
|
|
ToolbarSpacer(.fixed)
|
|
|
|
ToolbarItemGroup {
|
|
Button {
|
|
|
|
} label: {
|
|
Label("Favorite", systemImage: "heart")
|
|
.symbolVariant(.none)
|
|
}
|
|
|
|
Button("Info", systemImage: "info") {
|
|
|
|
}
|
|
}
|
|
|
|
ToolbarItemGroup(placement: .bottomBar) {
|
|
Spacer()
|
|
Button {
|
|
|
|
} label: {
|
|
Label("Favorite", systemImage: "heart")
|
|
.symbolVariant(.none)
|
|
}
|
|
|
|
Button("Info", systemImage: "info") {
|
|
|
|
}
|
|
}
|
|
|
|
}
|
|
.toolbar(removing: .title)
|
|
.ignoresSafeArea(edges: .top)
|
|
} else {
|
|
Text("hello1")
|
|
}
|
|
}
|
|
}
|