2023-01-17 11:36:01 +01:00
|
|
|
import Combine
|
2023-01-06 12:14:05 +01:00
|
|
|
import DesignSystem
|
2023-01-17 11:36:01 +01:00
|
|
|
import Env
|
|
|
|
import Models
|
|
|
|
import Network
|
2023-01-06 12:14:05 +01:00
|
|
|
import NukeUI
|
|
|
|
import Shimmer
|
2023-01-17 11:36:01 +01:00
|
|
|
import SwiftUI
|
2023-01-06 12:14:05 +01:00
|
|
|
|
2023-01-06 17:14:34 +01:00
|
|
|
struct AddRemoteTimelineView: View {
|
2023-01-06 12:14:05 +01:00
|
|
|
@Environment(\.dismiss) private var dismiss
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 17:14:34 +01:00
|
|
|
@EnvironmentObject private var preferences: UserPreferences
|
2023-01-06 12:14:05 +01:00
|
|
|
@EnvironmentObject private var theme: Theme
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 12:14:05 +01:00
|
|
|
@State private var instanceName: String = ""
|
|
|
|
@State private var instance: Instance?
|
|
|
|
@State private var instances: [InstanceSocial] = []
|
2023-01-16 06:43:53 +01:00
|
|
|
|
|
|
|
private let instanceNamePublisher = PassthroughSubject<String, Never>()
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 12:14:05 +01:00
|
|
|
@FocusState private var isInstanceURLFieldFocused: Bool
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 12:14:05 +01:00
|
|
|
var body: some View {
|
|
|
|
NavigationStack {
|
|
|
|
Form {
|
|
|
|
TextField("Instance URL", text: $instanceName)
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
.keyboardType(.URL)
|
|
|
|
.textContentType(.URL)
|
|
|
|
.textInputAutocapitalization(.never)
|
|
|
|
.autocorrectionDisabled()
|
|
|
|
.focused($isInstanceURLFieldFocused)
|
|
|
|
if let instance {
|
|
|
|
Label("\(instance.title) is a valid instance", systemImage: "checkmark.seal.fill")
|
|
|
|
.foregroundColor(.green)
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
|
|
|
Button {
|
|
|
|
guard instance != nil else { return }
|
2023-01-06 17:14:34 +01:00
|
|
|
preferences.remoteLocalTimelines.append(instanceName)
|
2023-01-06 12:14:05 +01:00
|
|
|
dismiss()
|
|
|
|
} label: {
|
|
|
|
Text("Add")
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 12:14:05 +01:00
|
|
|
instancesListView
|
|
|
|
}
|
|
|
|
.formStyle(.grouped)
|
|
|
|
.navigationTitle("Add remote local timeline")
|
|
|
|
.navigationBarTitleDisplayMode(.inline)
|
|
|
|
.scrollContentBackground(.hidden)
|
|
|
|
.background(theme.secondaryBackgroundColor)
|
|
|
|
.scrollDismissesKeyboard(.immediately)
|
|
|
|
.toolbar {
|
|
|
|
ToolbarItem(placement: .navigationBarLeading) {
|
|
|
|
Button("Cancel", action: { dismiss() })
|
|
|
|
}
|
|
|
|
}
|
2023-01-16 06:43:53 +01:00
|
|
|
.onChange(of: instanceName) { newValue in
|
|
|
|
instanceNamePublisher.send(newValue)
|
|
|
|
}
|
|
|
|
.onReceive(instanceNamePublisher.debounce(for: .milliseconds(500), scheduler: DispatchQueue.main)) { newValue in
|
2023-01-06 12:14:05 +01:00
|
|
|
Task {
|
|
|
|
let client = Client(server: newValue)
|
|
|
|
instance = try? await client.get(endpoint: Instances.instance)
|
|
|
|
}
|
2023-01-16 06:43:53 +01:00
|
|
|
}
|
2023-01-06 12:14:05 +01:00
|
|
|
.onAppear {
|
|
|
|
isInstanceURLFieldFocused = true
|
|
|
|
let client = InstanceSocialClient()
|
|
|
|
Task {
|
|
|
|
self.instances = await client.fetchInstances()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-01-17 11:36:01 +01:00
|
|
|
|
2023-01-06 12:14:05 +01:00
|
|
|
private var instancesListView: some View {
|
|
|
|
Section("Suggestions") {
|
|
|
|
if instances.isEmpty {
|
|
|
|
ProgressView()
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
} else {
|
2023-01-17 11:36:01 +01:00
|
|
|
ForEach(instanceName.isEmpty ? instances : instances.filter { $0.name.contains(instanceName.lowercased()) }) { instance in
|
2023-01-06 12:14:05 +01:00
|
|
|
Button {
|
|
|
|
self.instanceName = instance.name
|
|
|
|
} label: {
|
|
|
|
VStack(alignment: .leading, spacing: 4) {
|
|
|
|
Text(instance.name)
|
|
|
|
.font(.headline)
|
|
|
|
.foregroundColor(.primary)
|
|
|
|
Text(instance.info?.shortDescription ?? "")
|
|
|
|
.font(.body)
|
|
|
|
.foregroundColor(.gray)
|
|
|
|
Text("\(instance.users) users ⸱ \(instance.statuses) posts")
|
|
|
|
.font(.footnote)
|
|
|
|
.foregroundColor(.gray)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.listRowBackground(theme.primaryBackgroundColor)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|