mirror of
https://github.com/SideStore/SideStore.git
synced 2026-02-09 06:43:25 +01:00
Added Latin American Spanish (probably not the best translation) Made everything reference the swiftgen stuff rather than having strings
269 lines
8.9 KiB
Swift
269 lines
8.9 KiB
Swift
//
|
|
// SourcesView.swift
|
|
// SideStore
|
|
//
|
|
// Created by Fabian Thies on 20.11.22.
|
|
// Copyright © 2022 Fabian Thies. All rights reserved.
|
|
//
|
|
|
|
import SwiftUI
|
|
import SFSafeSymbols
|
|
import AltStoreCore
|
|
import CoreData
|
|
|
|
struct SourcesView: View {
|
|
|
|
@Environment(\.dismiss)
|
|
private var dismiss
|
|
|
|
@Environment(\.managedObjectContext)
|
|
var managedObjectContext
|
|
|
|
@SwiftUI.FetchRequest(sortDescriptors: [
|
|
NSSortDescriptor(keyPath: \Source.name, ascending: true),
|
|
NSSortDescriptor(keyPath: \Source.sourceURL, ascending: true),
|
|
NSSortDescriptor(keyPath: \Source.identifier, ascending: true)
|
|
])
|
|
var installedSources: FetchedResults<Source>
|
|
|
|
|
|
@State var isShowingAddSourceAlert = false
|
|
@State var sourceToConfirm: FetchedSource?
|
|
|
|
var body: some View {
|
|
ScrollView {
|
|
LazyVStack(alignment: .leading, spacing: 24) {
|
|
// Installed Sources
|
|
LazyVStack(alignment: .leading, spacing: 12) {
|
|
Text(L10n.SourcesView.sourcesDescription)
|
|
.font(.callout)
|
|
.foregroundColor(.secondary)
|
|
|
|
ForEach(installedSources, id: \.identifier) { source in
|
|
|
|
VStack(alignment: .leading) {
|
|
Text(source.name)
|
|
.bold()
|
|
|
|
Text(source.sourceURL.absoluteString)
|
|
.font(.callout)
|
|
.foregroundColor(.secondary)
|
|
}
|
|
.frame(maxWidth: .infinity, alignment: .leading)
|
|
.padding()
|
|
.tintedBackground(.accentColor)
|
|
.clipShape(RoundedRectangle(cornerRadius: 30, style: .circular))
|
|
.if(source.identifier != Source.altStoreIdentifier) { view in
|
|
view.contextMenu(ContextMenu(menuItems: {
|
|
SwiftUI.Button {
|
|
self.removeSource(source)
|
|
} label: {
|
|
Label(L10n.SourcesView.remove, systemSymbol: .trash)
|
|
}
|
|
}))
|
|
}
|
|
}
|
|
}
|
|
|
|
// Trusted Sources
|
|
LazyVStack(alignment: .leading) {
|
|
Text(L10n.SourcesView.trustedSources)
|
|
.font(.title3)
|
|
.bold()
|
|
|
|
Text(L10n.SourcesView.reviewedText)
|
|
.font(.callout)
|
|
.foregroundColor(.secondary)
|
|
|
|
|
|
}
|
|
}
|
|
.padding()
|
|
}
|
|
.navigationTitle(L10n.SourcesView.sources)
|
|
.toolbar {
|
|
ToolbarItem(placement: .navigationBarLeading) {
|
|
SwiftUI.Button {
|
|
self.isShowingAddSourceAlert = true
|
|
} label: {
|
|
Image(systemSymbol: .plus)
|
|
}
|
|
.sheet(isPresented: self.$isShowingAddSourceAlert) {
|
|
NavigationView {
|
|
AddSourceView(continueHandler: fetchSource(with:))
|
|
}
|
|
}
|
|
.sheet(item: self.$sourceToConfirm) { source in
|
|
if #available(iOS 16.0, *) {
|
|
NavigationView {
|
|
ConfirmAddSourceView(fetchedSource: source, confirmationHandler: addSource(_:)) {
|
|
self.sourceToConfirm = nil
|
|
}
|
|
}
|
|
.presentationDetents([.medium])
|
|
}
|
|
}
|
|
}
|
|
|
|
ToolbarItem(placement: .navigationBarTrailing) {
|
|
SwiftUI.Button(action: self.dismiss) {
|
|
Text(L10n.SourcesView.done).bold()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
func fetchSource(with urlText: String) {
|
|
self.isShowingAddSourceAlert = false
|
|
|
|
guard let url = URL(string: urlText) else {
|
|
return
|
|
}
|
|
|
|
AppManager.shared.fetchSource(sourceURL: url) { result in
|
|
|
|
switch result {
|
|
case let .success(source):
|
|
self.sourceToConfirm = FetchedSource(source: source)
|
|
case let .failure(error):
|
|
print(error)
|
|
}
|
|
}
|
|
}
|
|
|
|
func addSource(_ source: FetchedSource) {
|
|
source.context?.perform {
|
|
do {
|
|
try source.context?.save()
|
|
} catch {
|
|
print(error)
|
|
}
|
|
}
|
|
|
|
self.sourceToConfirm = nil
|
|
}
|
|
|
|
func removeSource(_ source: Source) {
|
|
DatabaseManager.shared.persistentContainer.performBackgroundTask { (context) in
|
|
let source = context.object(with: source.objectID) as! Source
|
|
context.delete(source)
|
|
|
|
do {
|
|
try context.save()
|
|
} catch {
|
|
print(error)
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
// func fetchTrustedSources() {
|
|
// func finish(_ result: Result<[Source], Error>)
|
|
// {
|
|
// self.fetchTrustedSourcesResult = result.map { _ in () }
|
|
//
|
|
// DispatchQueue.main.async {
|
|
// do
|
|
// {
|
|
// let sources = try result.get()
|
|
// print("Fetched trusted sources:", sources.map { $0.identifier })
|
|
//
|
|
// let sectionUpdate = RSTCellContentChange(type: .update, sectionIndex: 0)
|
|
// self.trustedSourcesDataSource.setItems(sources, with: [sectionUpdate])
|
|
// }
|
|
// catch
|
|
// {
|
|
// print("Error fetching trusted sources:", error)
|
|
//
|
|
// let sectionUpdate = RSTCellContentChange(type: .update, sectionIndex: 0)
|
|
// self.trustedSourcesDataSource.setItems([], with: [sectionUpdate])
|
|
// }
|
|
// }
|
|
// }
|
|
//
|
|
// self.fetchTrustedSourcesOperation = AppManager.shared.fetchTrustedSources { result in
|
|
// switch result
|
|
// {
|
|
// case .failure(let error): finish(.failure(error))
|
|
// case .success(let trustedSources):
|
|
// // Cache trusted source IDs.
|
|
// UserDefaults.shared.trustedSourceIDs = trustedSources.map { $0.identifier }
|
|
//
|
|
// // Don't show sources without a sourceURL.
|
|
// let featuredSourceURLs = trustedSources.compactMap { $0.sourceURL }
|
|
//
|
|
// // This context is never saved, but keeps the managed sources alive.
|
|
// let context = DatabaseManager.shared.persistentContainer.newBackgroundSavingViewContext()
|
|
// self._fetchTrustedSourcesContext = context
|
|
//
|
|
// let dispatchGroup = DispatchGroup()
|
|
//
|
|
// var sourcesByURL = [URL: Source]()
|
|
// var fetchError: Error?
|
|
//
|
|
// for sourceURL in featuredSourceURLs
|
|
// {
|
|
// dispatchGroup.enter()
|
|
//
|
|
// AppManager.shared.fetchSource(sourceURL: sourceURL, managedObjectContext: context) { result in
|
|
// // Serialize access to sourcesByURL.
|
|
// context.performAndWait {
|
|
// switch result
|
|
// {
|
|
// case .failure(let error): fetchError = error
|
|
// case .success(let source): sourcesByURL[source.sourceURL] = source
|
|
// }
|
|
//
|
|
// dispatchGroup.leave()
|
|
// }
|
|
// }
|
|
// }
|
|
//
|
|
// dispatchGroup.notify(queue: .main) {
|
|
// if let error = fetchError
|
|
// {
|
|
// finish(.failure(error))
|
|
// }
|
|
// else
|
|
// {
|
|
// let sources = featuredSourceURLs.compactMap { sourcesByURL[$0] }
|
|
// finish(.success(sources))
|
|
// }
|
|
// }
|
|
// }
|
|
// }
|
|
// }
|
|
}
|
|
|
|
struct SourcesListView_Previews: PreviewProvider {
|
|
static var previews: some View {
|
|
SourcesView()
|
|
}
|
|
}
|
|
|
|
|
|
extension Source: Identifiable {
|
|
public var id: String {
|
|
self.identifier
|
|
}
|
|
}
|
|
|
|
|
|
struct FetchedSource: Identifiable {
|
|
let source: Source
|
|
let context: NSManagedObjectContext?
|
|
|
|
var id: String {
|
|
source.identifier
|
|
}
|
|
|
|
init?(source: Source) {
|
|
guard let context = source.managedObjectContext else{
|
|
return nil
|
|
}
|
|
self.source = source
|
|
self.context = context
|
|
}
|
|
}
|