200 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			200 lines
		
	
	
		
			6.0 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) 2024 Jellyfin & Jellyfin Contributors
 | |
| //
 | |
| 
 | |
| import Defaults
 | |
| import Stinsen
 | |
| import SwiftUI
 | |
| 
 | |
| struct ConnectToServerView: View {
 | |
| 
 | |
|     @EnvironmentObject
 | |
|     private var router: ConnectToServerCoodinator.Router
 | |
| 
 | |
|     @ObservedObject
 | |
|     var viewModel: ConnectToServerViewModel
 | |
| 
 | |
|     @State
 | |
|     private var connectionError: Error?
 | |
|     @State
 | |
|     private var connectionTask: Task<Void, Never>?
 | |
|     @State
 | |
|     private var duplicateServer: (server: ServerState, url: URL)?
 | |
|     @State
 | |
|     private var isConnecting: Bool = false
 | |
|     @State
 | |
|     private var isPresentingConnectionError: Bool = false
 | |
|     @State
 | |
|     private var isPresentingDuplicateServerAlert: Bool = false
 | |
|     @State
 | |
|     private var isPresentingError: Bool = false
 | |
|     @State
 | |
|     private var url = ""
 | |
| 
 | |
|     private func connectToServer(at url: String) {
 | |
|         let task = Task {
 | |
|             isConnecting = true
 | |
|             connectionError = nil
 | |
| 
 | |
|             do {
 | |
|                 let serverConnection = try await viewModel.connectToServer(url: url)
 | |
| 
 | |
|                 if viewModel.isDuplicate(server: serverConnection.server) {
 | |
|                     duplicateServer = serverConnection
 | |
|                     isPresentingDuplicateServerAlert = true
 | |
|                 } else {
 | |
|                     try viewModel.save(server: serverConnection.server)
 | |
|                     router.route(to: \.userSignIn, serverConnection.server)
 | |
|                 }
 | |
|             } catch {
 | |
|                 connectionError = error
 | |
|                 isPresentingConnectionError = true
 | |
|             }
 | |
| 
 | |
|             isConnecting = false
 | |
|         }
 | |
| 
 | |
|         connectionTask = task
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var connectForm: some View {
 | |
|         VStack(alignment: .leading) {
 | |
| 
 | |
|             L10n.connectToJellyfinServer.text
 | |
| 
 | |
|             TextField(L10n.serverURL, text: $url)
 | |
|                 .disableAutocorrection(true)
 | |
|                 .autocapitalization(.none)
 | |
|                 .keyboardType(.URL)
 | |
| 
 | |
|             if isConnecting {
 | |
|                 Button {
 | |
|                     connectionTask?.cancel()
 | |
|                     isConnecting = false
 | |
|                 } label: {
 | |
|                     L10n.cancel.text
 | |
|                         .foregroundColor(.red)
 | |
|                         .bold()
 | |
|                         .font(.callout)
 | |
|                         .frame(height: 75)
 | |
|                         .frame(maxWidth: .infinity)
 | |
|                 }
 | |
|                 .buttonStyle(.card)
 | |
|             } else {
 | |
|                 Button {
 | |
|                     if !url.contains("://") {
 | |
|                         url = "http://" + url
 | |
|                     }
 | |
|                     connectToServer(at: url)
 | |
|                 } label: {
 | |
|                     L10n.connect.text
 | |
|                         .bold()
 | |
|                         .font(.callout)
 | |
|                         .frame(height: 75)
 | |
|                         .frame(maxWidth: .infinity)
 | |
|                         .background {
 | |
|                             if isConnecting || url.isEmpty {
 | |
|                                 Color.secondary
 | |
|                             } else {
 | |
|                                 Color.jellyfinPurple
 | |
|                             }
 | |
|                         }
 | |
|                 }
 | |
|                 .disabled(isConnecting || url.isEmpty)
 | |
|                 .buttonStyle(.card)
 | |
|             }
 | |
| 
 | |
|             Spacer()
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var searchingDiscoverServers: some View {
 | |
|         HStack(spacing: 5) {
 | |
|             ProgressView()
 | |
| 
 | |
|             L10n.searchingDots.text
 | |
|                 .foregroundColor(.secondary)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var noLocalServersFound: some View {
 | |
|         L10n.noLocalServersFound.text
 | |
|             .font(.callout)
 | |
|             .foregroundColor(.secondary)
 | |
|     }
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var publicServers: some View {
 | |
|         VStack(alignment: .center) {
 | |
| 
 | |
|             HStack {
 | |
|                 L10n.localServers.text
 | |
|                     .font(.title3)
 | |
|                     .fontWeight(.semibold)
 | |
| 
 | |
|                 SFSymbolButton(systemName: "arrow.clockwise")
 | |
|                     .onSelect {
 | |
|                         viewModel.discoverServers()
 | |
|                     }
 | |
|                     .frame(width: 30, height: 30)
 | |
|                     .disabled(viewModel.isSearching || viewModel.isLoading)
 | |
|             }
 | |
| 
 | |
|             if viewModel.isSearching {
 | |
|                 searchingDiscoverServers
 | |
|                     .frame(maxHeight: .infinity)
 | |
|             } else if viewModel.discoveredServers.isEmpty {
 | |
|                 noLocalServersFound
 | |
|                     .frame(maxHeight: .infinity)
 | |
|             } else {
 | |
|                 ScrollView {
 | |
|                     VStack {
 | |
|                         ForEach(viewModel.discoveredServers, id: \.id) { server in
 | |
|                             ServerButton(server: server)
 | |
|                                 .onSelect {
 | |
|                                     connectToServer(at: server.currentURL.absoluteString)
 | |
|                                 }
 | |
|                         }
 | |
|                     }
 | |
|                     .padding()
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     var body: some View {
 | |
|         HStack(alignment: .top) {
 | |
|             connectForm
 | |
|                 .frame(maxWidth: .infinity)
 | |
| 
 | |
|             publicServers
 | |
|                 .frame(maxWidth: .infinity)
 | |
|         }
 | |
|         .navigationTitle(L10n.connect.text)
 | |
|         .onAppear {
 | |
|             viewModel.discoverServers()
 | |
|         }
 | |
| //        .alert(item: $viewModel.errorMessage) { _ in
 | |
| //            Alert(
 | |
| //                title: Text(viewModel.alertTitle),
 | |
| //                message: Text(viewModel.errorMessage?.message ?? L10n.unknownError),
 | |
| //                dismissButton: .cancel()
 | |
| //            )
 | |
| //        }
 | |
| //        .alert(item: $viewModel.addServerURIPayload) { _ in
 | |
| //            Alert(
 | |
| //                title: L10n.existingServer.text,
 | |
| //                message: L10n.serverAlreadyExistsPrompt(viewModel.addServerURIPayload?.server.name ?? .emptyDash).text,
 | |
| //                dismissButton: .cancel()
 | |
| //            )
 | |
| //        }
 | |
|     }
 | |
| }
 |