Gemini/Gemini/ContentView.swift

94 lines
2.6 KiB
Swift
Raw Normal View History

2020-07-13 01:35:00 +00:00
//
// ContentView.swift
// Gemini
//
// Created by Shadowfacts on 7/12/20.
//
import SwiftUI
2020-07-15 02:13:09 +00:00
import GeminiFormat
import GeminiRenderer
2020-07-15 02:53:47 +00:00
import GeminiProtocol
2020-07-13 01:35:00 +00:00
struct ContentView: View {
2020-07-15 03:15:56 +00:00
let navigator: NavigationManager
2020-07-15 02:53:47 +00:00
@State var task: GeminiDataTask?
2020-07-16 02:34:15 +00:00
@State var state: ViewState = .loading
2020-07-15 02:13:09 +00:00
2020-07-13 01:35:00 +00:00
var body: some View {
2020-07-15 22:30:12 +00:00
mainView
2020-07-15 02:13:09 +00:00
.frame(minWidth: 480, maxWidth: .infinity, minHeight: 300, maxHeight: .infinity)
2020-07-15 03:15:56 +00:00
.onReceive(navigator.$currentURL, perform: self.urlChanged)
2020-07-13 01:35:00 +00:00
}
2020-07-15 02:53:47 +00:00
@ViewBuilder
private var mainView: some View {
2020-07-16 02:34:15 +00:00
switch state {
case .loading:
loadingView
.onAppear(perform: self.loadDocument)
case let .error(message):
2020-07-15 02:53:47 +00:00
VStack {
Text("An error occurred")
.font(.headline)
2020-07-16 02:34:15 +00:00
Text(message)
2020-07-15 02:53:47 +00:00
}
2020-07-16 02:34:15 +00:00
case let .document(doc):
DocumentView(document: doc, changeURL: navigator.changeURL)
2020-07-15 02:53:47 +00:00
}
}
@ViewBuilder
private var loadingView: some View {
if #available(macOS 10.16, iOS 14.0, *) {
ProgressView("Loading...")
} else {
Text("Loading...")
}
}
private func loadDocument() {
2020-07-15 03:15:56 +00:00
let url = navigator.currentURL
2020-07-15 02:53:47 +00:00
task = try! GeminiDataTask(url: url, completion: { (response) in
2020-07-16 02:34:15 +00:00
self.task = nil
2020-07-15 02:53:47 +00:00
switch response {
case let .failure(error):
2020-07-16 02:34:15 +00:00
state = .error(error.localizedDescription)
2020-07-15 02:53:47 +00:00
case let .success(response):
2020-07-16 02:34:15 +00:00
if response.status.isRedirect {
if let redirect = URL(string: response.meta) {
navigator.changeURL(redirect)
loadDocument()
} else {
state = .error("Invalid redirect URL: '\(response.meta)'")
}
} else if response.status.isSuccess,
let text = response.bodyText {
state = .document(GeminiParser.parse(text: text, baseURL: url))
} else {
state = .error("Unknown error: \(response.header)")
2020-07-15 02:53:47 +00:00
}
}
})
task!.resume()
}
2020-07-15 03:15:56 +00:00
private func urlChanged(_ newValue: URL) {
2020-07-16 02:34:15 +00:00
state = .loading
}
}
extension ContentView {
enum ViewState {
case loading
case document(Document)
case error(String)
2020-07-15 03:15:56 +00:00
}
2020-07-13 01:35:00 +00:00
}
struct ContentView_Previews: PreviewProvider {
static var previews: some View {
2020-07-15 03:15:56 +00:00
ContentView(navigator: NavigationManager(url: URL(string: "gemini://localhost/overview.gmi")!))
2020-07-13 01:35:00 +00:00
}
}