164 lines
4.8 KiB
Swift
164 lines
4.8 KiB
Swift
//
|
|
// Swiftfin is subject to the terms of the Mozilla Public
|
|
// License, v2.0. If a copy of the MPL was not distributed with this
|
|
// file, you can obtain one at https://mozilla.org/MPL/2.0/.
|
|
//
|
|
// Copyright (c) 2025 Jellyfin & Jellyfin Contributors
|
|
//
|
|
|
|
import Combine
|
|
import Defaults
|
|
import SwiftUI
|
|
|
|
struct ConnectToServerView: View {
|
|
|
|
@Default(.accentColor)
|
|
private var accentColor
|
|
|
|
@FocusState
|
|
private var isURLFocused: Bool
|
|
|
|
@Router
|
|
private var router
|
|
|
|
@StateObject
|
|
private var viewModel = ConnectToServerViewModel()
|
|
|
|
@State
|
|
private var duplicateServer: ServerState? = nil
|
|
@State
|
|
private var isPresentingDuplicateServer: Bool = false
|
|
@State
|
|
private var url: String = ""
|
|
|
|
private let timer = Timer.publish(every: 12, on: .main, in: .common).autoconnect()
|
|
|
|
private func onEvent(_ event: ConnectToServerViewModel._Event) {
|
|
switch event {
|
|
case let .connected(server):
|
|
UIDevice.feedback(.success)
|
|
Notifications[.didConnectToServer].post(server)
|
|
router.dismiss()
|
|
case let .duplicateServer(server):
|
|
UIDevice.feedback(.warning)
|
|
duplicateServer = server
|
|
isPresentingDuplicateServer = true
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var connectSection: some View {
|
|
Section(L10n.connectToServer) {
|
|
TextField(L10n.serverURL, text: $url)
|
|
.disableAutocorrection(true)
|
|
.textInputAutocapitalization(.never)
|
|
.keyboardType(.URL)
|
|
.focused($isURLFocused)
|
|
}
|
|
|
|
if viewModel.state == .connecting {
|
|
ListRowButton(L10n.cancel) {
|
|
viewModel.cancel()
|
|
}
|
|
.foregroundStyle(.red, .red.opacity(0.2))
|
|
} else {
|
|
ListRowButton(L10n.connect) {
|
|
isURLFocused = false
|
|
viewModel.connect(url: url)
|
|
}
|
|
.disabled(url.isEmpty)
|
|
.foregroundStyle(
|
|
accentColor.overlayColor,
|
|
accentColor
|
|
)
|
|
.opacity(url.isEmpty ? 0.5 : 1)
|
|
}
|
|
}
|
|
|
|
// MARK: - Local Servers Section
|
|
|
|
@ViewBuilder
|
|
private var localServersSection: some View {
|
|
Section(L10n.localServers) {
|
|
if viewModel.localServers.isEmpty {
|
|
L10n.noLocalServersFound.text
|
|
.font(.callout)
|
|
.foregroundColor(.secondary)
|
|
.frame(maxWidth: .infinity)
|
|
} else {
|
|
ForEach(viewModel.localServers) { server in
|
|
LocalServerButton(server: server) {
|
|
url = server.currentURL.absoluteString
|
|
viewModel.connect(url: server.currentURL.absoluteString)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
@ViewBuilder
|
|
private var contentView: some View {
|
|
#if os(iOS)
|
|
List {
|
|
connectSection
|
|
|
|
localServersSection
|
|
}
|
|
.navigationBarTitleDisplayMode(.inline)
|
|
.navigationBarCloseButton(disabled: viewModel.state == .connecting) {
|
|
router.dismiss()
|
|
}
|
|
#else
|
|
SplitLoginWindowView(
|
|
isLoading: viewModel.state == .connecting
|
|
) {
|
|
connectSection
|
|
} trailingContentView: {
|
|
localServersSection
|
|
}
|
|
#endif
|
|
}
|
|
|
|
// MARK: - Body
|
|
|
|
var body: some View {
|
|
contentView
|
|
.navigationTitle(L10n.connect)
|
|
.interactiveDismissDisabled(viewModel.state == .connecting)
|
|
.onFirstAppear {
|
|
isURLFocused = true
|
|
viewModel.searchForServers()
|
|
}
|
|
.onReceive(timer) { _ in
|
|
guard viewModel.state != .connecting else { return }
|
|
viewModel.searchForServers()
|
|
}
|
|
.onReceive(viewModel.events, perform: onEvent)
|
|
.onReceive(viewModel.$error) { error in
|
|
guard error != nil else { return }
|
|
UIDevice.feedback(.error)
|
|
isURLFocused = true
|
|
}
|
|
.topBarTrailing {
|
|
if viewModel.state == .connecting {
|
|
ProgressView()
|
|
}
|
|
}
|
|
.alert(
|
|
L10n.server.text,
|
|
isPresented: $isPresentingDuplicateServer,
|
|
presenting: duplicateServer
|
|
) { server in
|
|
Button(L10n.dismiss, role: .destructive)
|
|
|
|
Button(L10n.addURL) {
|
|
viewModel.addNewURL(serverState: server)
|
|
router.dismiss()
|
|
}
|
|
} message: { server in
|
|
L10n.serverAlreadyExistsPrompt(server.name).text
|
|
}
|
|
.errorMessage($viewModel.error)
|
|
}
|
|
}
|