Network layer: - Replace static CookbookApi protocol with instance-based CookbookApiProtocol using async/throws instead of tuple returns - Refactor ApiRequest to use URLComponents for proper URL encoding, replace print statements with OSLog, and return typed NetworkError cases - Add structured NetworkError variants (httpError, connectionError, etc.) - Remove global cookbookApi constant in favor of injected dependency on AppState - Delete unused RecipeEditViewModel, RecipeScraper, and Scraper playground Data & model fixes: - Add custom Decodable for RecipeDetail with safe fallbacks for malformed JSON - Make Category Hashable/Equatable use only `name` so NavigationSplitView selection survives category refreshes with updated recipe_count - Return server-assigned ID from uploadRecipe so new recipes get their ID before the post-upload refresh block executes View updates: - Refresh both old and new category recipe lists after upload when category changes, mapping empty recipeCategory to "*" for uncategorized recipes - Raise deployment target to iOS 18, adopt new SwiftUI API conventions - Clean up alerts, onboarding views, and settings Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
33 lines
813 B
Swift
33 lines
813 B
Swift
//
|
|
// JSONCoderExtension.swift
|
|
// Nextcloud Cookbook iOS Client
|
|
//
|
|
// Created by Vincent Meilinger on 20.09.23.
|
|
//
|
|
|
|
import Foundation
|
|
import OSLog
|
|
|
|
extension JSONDecoder {
|
|
static func safeDecode<T: Decodable>(_ data: Data) -> T? {
|
|
let decoder = JSONDecoder()
|
|
do {
|
|
return try decoder.decode(T.self, from: data)
|
|
} catch {
|
|
Logger.data.error("JSONDecoder - safeDecode(): \(error.localizedDescription)")
|
|
return nil
|
|
}
|
|
}
|
|
}
|
|
|
|
extension JSONEncoder {
|
|
static func safeEncode<T: Encodable>(_ object: T) -> Data? {
|
|
do {
|
|
return try JSONEncoder().encode(object)
|
|
} catch {
|
|
Logger.data.error("JSONEncoder - safeEncode(): Could not encode \(String(describing: T.self))")
|
|
}
|
|
return nil
|
|
}
|
|
}
|