2020-07-15 23:45:37 -04:00
|
|
|
//
|
|
|
|
// ContentView.swift
|
|
|
|
// Gemini
|
|
|
|
//
|
|
|
|
// Created by Shadowfacts on 7/12/20.
|
|
|
|
//
|
|
|
|
|
|
|
|
import SwiftUI
|
|
|
|
import GeminiFormat
|
|
|
|
import GeminiRenderer
|
|
|
|
import GeminiProtocol
|
|
|
|
|
|
|
|
public struct BrowserView: View {
|
|
|
|
let navigator: NavigationManager
|
2020-09-28 15:20:06 -04:00
|
|
|
let scrollingEnabled: Bool
|
2020-07-15 23:45:37 -04:00
|
|
|
@State var task: GeminiDataTask?
|
|
|
|
@State var state: ViewState = .loading
|
|
|
|
|
2020-09-28 15:20:06 -04:00
|
|
|
public init(navigator: NavigationManager, scrollingEnabled: Bool = true) {
|
2020-07-15 23:45:37 -04:00
|
|
|
self.navigator = navigator
|
2020-09-28 15:20:06 -04:00
|
|
|
self.scrollingEnabled = scrollingEnabled
|
2020-07-15 23:45:37 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
public var body: some View {
|
|
|
|
mainView
|
|
|
|
.onReceive(navigator.$currentURL, perform: self.urlChanged)
|
|
|
|
}
|
|
|
|
|
|
|
|
@ViewBuilder
|
|
|
|
private var mainView: some View {
|
2020-09-28 15:49:02 -04:00
|
|
|
VStack {
|
|
|
|
switch state {
|
|
|
|
case .loading:
|
|
|
|
Spacer()
|
|
|
|
loadingView
|
|
|
|
.onAppear(perform: self.loadDocument)
|
|
|
|
Spacer()
|
|
|
|
case let .error(message):
|
2020-07-15 23:45:37 -04:00
|
|
|
Text("An error occurred")
|
|
|
|
.font(.headline)
|
|
|
|
Text(message)
|
2020-09-29 16:28:37 -04:00
|
|
|
.lineLimit(nil)
|
2020-09-28 15:49:02 -04:00
|
|
|
case let .document(doc):
|
|
|
|
DocumentView(document: doc, scrollingEnabled: scrollingEnabled, changeURL: navigator.changeURL)
|
|
|
|
Spacer()
|
2020-07-15 23:45:37 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@ViewBuilder
|
|
|
|
private var loadingView: some View {
|
|
|
|
if #available(macOS 10.16, iOS 14.0, *) {
|
|
|
|
ProgressView("Loading...")
|
|
|
|
} else {
|
|
|
|
Text("Loading...")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private func loadDocument() {
|
|
|
|
let url = navigator.currentURL
|
|
|
|
task = try! GeminiDataTask(url: url, completion: { (response) in
|
|
|
|
self.task = nil
|
|
|
|
switch response {
|
|
|
|
case let .failure(error):
|
|
|
|
state = .error(error.localizedDescription)
|
|
|
|
case let .success(response):
|
|
|
|
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)")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
task!.resume()
|
|
|
|
}
|
|
|
|
|
|
|
|
private func urlChanged(_ newValue: URL) {
|
2020-09-28 15:49:02 -04:00
|
|
|
if case .loading = state {
|
|
|
|
task?.cancel()
|
|
|
|
loadDocument()
|
|
|
|
} else {
|
|
|
|
state = .loading
|
|
|
|
}
|
2020-07-15 23:45:37 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extension BrowserView {
|
|
|
|
enum ViewState {
|
|
|
|
case loading
|
|
|
|
case document(Document)
|
|
|
|
case error(String)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct ContentView_Previews: PreviewProvider {
|
|
|
|
static var previews: some View {
|
|
|
|
BrowserView(navigator: NavigationManager(url: URL(string: "gemini://localhost/overview.gmi")!))
|
|
|
|
}
|
|
|
|
}
|