* Buildable! * Update file names. * Default sort to sort name NOT name. * SessionInfoDto vs SessionInfo * Targetting * Fix many invalid `ItemSortBy` existing. Will need to revisit later to see which can still be used! * ExtraTypes Patch. * Move from Binding to OnChange. Tested and Working. * Update README.md Update README to use 10.10.6. Bumped up from 10.8.13 * Update to Main on https://github.com/jellyfin/jellyfin-sdk-swift.git * Now using https://github.com/jellyfin/jellyfin-sdk-swift.git again! * Paths.getUserViews() userId moved to parameters * Fix ViewModels where -Dto suffixes were removed by https://github.com/jellyfin/Swiftfin/pull/1465 auto-merge. * SupportedCaseIterable * tvOS supportedCases fixes for build issue. * cleanup * update API to 0.5.1 and correct VideoRangeTypes. * Remove deviceProfile.responseProfiles = videoPlayer.responseProfiles * Second to last adjustment: Resolved: // TODO: 10.10 - Filter to only valid SortBy's for each BaseItemKind. Last outstanding item: // TODO: 10.10 - What should authenticationProviderID & passwordResetProviderID be? * Trailers itemID must precede userID * Force User Policy to exist. --------- Co-authored-by: Ethan Pippin <ethanpippin2343@gmail.com>
		
			
				
	
	
		
			219 lines
		
	
	
		
			7.0 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			7.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) 2025 Jellyfin & Jellyfin Contributors
 | |
| //
 | |
| 
 | |
| import Defaults
 | |
| import JellyfinAPI
 | |
| import OrderedCollections
 | |
| import SwiftUI
 | |
| 
 | |
| struct DevicesView: View {
 | |
| 
 | |
|     @EnvironmentObject
 | |
|     private var router: AdminDashboardCoordinator.Router
 | |
| 
 | |
|     @State
 | |
|     private var isPresentingDeleteSelectionConfirmation = false
 | |
|     @State
 | |
|     private var isPresentingDeleteConfirmation = false
 | |
|     @State
 | |
|     private var isPresentingSelfDeleteError = false
 | |
|     @State
 | |
|     private var selectedDevices: Set<String> = []
 | |
|     @State
 | |
|     private var isEditing: Bool = false
 | |
| 
 | |
|     @StateObject
 | |
|     private var viewModel = DevicesViewModel()
 | |
| 
 | |
|     // MARK: - Body
 | |
| 
 | |
|     var body: some View {
 | |
|         ZStack {
 | |
|             switch viewModel.state {
 | |
|             case .content:
 | |
|                 deviceListView
 | |
|             case let .error(error):
 | |
|                 ErrorView(error: error)
 | |
|                     .onRetry {
 | |
|                         viewModel.send(.refresh)
 | |
|                     }
 | |
|             case .initial:
 | |
|                 DelayedProgressView()
 | |
|             }
 | |
|         }
 | |
|         .animation(.linear(duration: 0.2), value: viewModel.state)
 | |
|         .navigationTitle(L10n.devices)
 | |
|         .navigationBarTitleDisplayMode(.inline)
 | |
|         .navigationBarBackButtonHidden(isEditing)
 | |
|         .toolbar {
 | |
|             ToolbarItem(placement: .navigationBarLeading) {
 | |
|                 if isEditing {
 | |
|                     navigationBarSelectView
 | |
|                 }
 | |
|             }
 | |
|             ToolbarItem(placement: .navigationBarTrailing) {
 | |
|                 if viewModel.devices.isNotEmpty {
 | |
|                     navigationBarEditView
 | |
|                 }
 | |
|             }
 | |
|             ToolbarItem(placement: .bottomBar) {
 | |
|                 if isEditing {
 | |
|                     Button(L10n.delete) {
 | |
|                         isPresentingDeleteSelectionConfirmation = true
 | |
|                     }
 | |
|                     .buttonStyle(.toolbarPill(.red))
 | |
|                     .disabled(selectedDevices.isEmpty)
 | |
|                     .frame(maxWidth: .infinity, alignment: .trailing)
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         .onFirstAppear {
 | |
|             viewModel.send(.refresh)
 | |
|         }
 | |
|         .confirmationDialog(
 | |
|             L10n.deleteSelectedDevices,
 | |
|             isPresented: $isPresentingDeleteSelectionConfirmation,
 | |
|             titleVisibility: .visible
 | |
|         ) {
 | |
|             deleteSelectedDevicesConfirmationActions
 | |
|         } message: {
 | |
|             Text(L10n.deleteSelectionDevicesWarning)
 | |
|         }
 | |
|         .confirmationDialog(
 | |
|             L10n.deleteDevice,
 | |
|             isPresented: $isPresentingDeleteConfirmation,
 | |
|             titleVisibility: .visible
 | |
|         ) {
 | |
|             deleteDeviceConfirmationActions
 | |
|         } message: {
 | |
|             Text(L10n.deleteDeviceWarning)
 | |
|         }
 | |
|         .alert(L10n.deleteDeviceFailed, isPresented: $isPresentingSelfDeleteError) {
 | |
|             Button(L10n.ok, role: .cancel) {}
 | |
|         } message: {
 | |
|             Text(L10n.deleteDeviceSelfDeletion(viewModel.userSession.client.configuration.deviceName))
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // MARK: - Device List View
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var deviceListView: some View {
 | |
|         List {
 | |
|             InsetGroupedListHeader(
 | |
|                 L10n.devices,
 | |
|                 description: L10n.allDevicesDescription
 | |
|             ) {
 | |
|                 UIApplication.shared.open(.jellyfinDocsDevices)
 | |
|             }
 | |
|             .listRowBackground(Color.clear)
 | |
|             .listRowSeparator(.hidden)
 | |
|             .padding(.vertical, 24)
 | |
| 
 | |
|             if viewModel.devices.isEmpty {
 | |
|                 Text(L10n.none)
 | |
|                     .frame(maxWidth: .infinity, maxHeight: .infinity, alignment: .center)
 | |
|                     .listRowSeparator(.hidden)
 | |
|                     .listRowInsets(.zero)
 | |
|             } else {
 | |
|                 ForEach(viewModel.devices, id: \.self) { device in
 | |
|                     DeviceRow(device: device) {
 | |
|                         guard let id = device.id else { return }
 | |
| 
 | |
|                         if isEditing {
 | |
|                             if selectedDevices.contains(id) {
 | |
|                                 selectedDevices.remove(id)
 | |
|                             } else {
 | |
|                                 selectedDevices.insert(id)
 | |
|                             }
 | |
|                         } else {
 | |
|                             router.route(to: \.deviceDetails, device)
 | |
|                         }
 | |
|                     } onDelete: {
 | |
|                         guard let id = device.id else { return }
 | |
| 
 | |
|                         selectedDevices.removeAll()
 | |
|                         selectedDevices.insert(id)
 | |
|                         isPresentingDeleteConfirmation = true
 | |
|                     }
 | |
|                     .environment(\.isEditing, isEditing)
 | |
|                     .environment(\.isSelected, selectedDevices.contains(device.id ?? ""))
 | |
|                     .listRowInsets(.edgeInsets)
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         .listStyle(.plain)
 | |
|     }
 | |
| 
 | |
|     // MARK: - Navigation Bar Edit Content
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var navigationBarEditView: some View {
 | |
|         if viewModel.backgroundStates.contains(.refreshing) {
 | |
|             ProgressView()
 | |
|         } else {
 | |
|             Button(isEditing ? L10n.cancel : L10n.edit) {
 | |
|                 isEditing.toggle()
 | |
|                 UIDevice.impact(.light)
 | |
|                 if !isEditing {
 | |
|                     selectedDevices.removeAll()
 | |
|                 }
 | |
|             }
 | |
|             .buttonStyle(.toolbarPill)
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // MARK: - Navigation Bar Select/Remove All Content
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var navigationBarSelectView: some View {
 | |
|         let isAllSelected: Bool = selectedDevices.count == viewModel.devices.count
 | |
| 
 | |
|         Button(isAllSelected ? L10n.removeAll : L10n.selectAll) {
 | |
|             if isAllSelected {
 | |
|                 selectedDevices = []
 | |
|             } else {
 | |
|                 selectedDevices = Set(viewModel.devices.compactMap(\.id))
 | |
|             }
 | |
|         }
 | |
|         .buttonStyle(.toolbarPill)
 | |
|         .disabled(!isEditing)
 | |
|     }
 | |
| 
 | |
|     // MARK: - Delete Selected Devices Confirmation Actions
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var deleteSelectedDevicesConfirmationActions: some View {
 | |
|         Button(L10n.cancel, role: .cancel) {}
 | |
| 
 | |
|         Button(L10n.confirm, role: .destructive) {
 | |
|             viewModel.send(.delete(ids: Array(selectedDevices)))
 | |
|             isEditing = false
 | |
|             selectedDevices.removeAll()
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     // MARK: - Delete Device Confirmation Actions
 | |
| 
 | |
|     @ViewBuilder
 | |
|     private var deleteDeviceConfirmationActions: some View {
 | |
|         Button(L10n.cancel, role: .cancel) {}
 | |
| 
 | |
|         Button(L10n.delete, role: .destructive) {
 | |
|             if let deviceToDelete = selectedDevices.first, selectedDevices.count == 1 {
 | |
|                 if deviceToDelete == viewModel.userSession.client.configuration.deviceID {
 | |
|                     isPresentingSelfDeleteError = true
 | |
|                 } else {
 | |
|                     viewModel.send(.delete(ids: [deviceToDelete]))
 | |
|                     selectedDevices.removeAll()
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |