* 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>
		
			
				
	
	
		
			173 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			173 lines
		
	
	
		
			4.9 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 Foundation
 | |
| import JellyfinAPI
 | |
| import OrderedCollections
 | |
| import SwiftUI
 | |
| 
 | |
| final class ActiveSessionsViewModel: ViewModel, Stateful {
 | |
| 
 | |
|     // MARK: - Action
 | |
| 
 | |
|     enum Action: Equatable {
 | |
|         case getSessions
 | |
|         case refreshSessions
 | |
|     }
 | |
| 
 | |
|     // MARK: - BackgroundState
 | |
| 
 | |
|     enum BackgroundState: Hashable {
 | |
|         case gettingSessions
 | |
|     }
 | |
| 
 | |
|     // MARK: - State
 | |
| 
 | |
|     enum State: Hashable {
 | |
|         case content
 | |
|         case error(JellyfinAPIError)
 | |
|         case initial
 | |
|     }
 | |
| 
 | |
|     @Published
 | |
|     var backgroundStates: Set<BackgroundState> = []
 | |
|     @Published
 | |
|     var sessions: OrderedDictionary<String, BindingBox<SessionInfoDto?>> = [:]
 | |
|     @Published
 | |
|     var state: State = .initial
 | |
| 
 | |
|     private let activeWithinSeconds: Int = 960
 | |
|     private var sessionTask: AnyCancellable?
 | |
| 
 | |
|     func respond(to action: Action) -> State {
 | |
|         switch action {
 | |
|         case .getSessions:
 | |
|             sessionTask?.cancel()
 | |
| 
 | |
|             sessionTask = Task { [weak self] in
 | |
|                 await MainActor.run {
 | |
|                     let _ = self?.backgroundStates.insert(.gettingSessions)
 | |
|                 }
 | |
| 
 | |
|                 do {
 | |
|                     try await self?.updateSessions()
 | |
|                 } catch {
 | |
|                     guard let self else { return }
 | |
|                     await MainActor.run {
 | |
|                         self.state = .error(.init(error.localizedDescription))
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 await MainActor.run {
 | |
|                     let _ = self?.backgroundStates.remove(.gettingSessions)
 | |
|                 }
 | |
|             }
 | |
|             .asAnyCancellable()
 | |
| 
 | |
|             return state
 | |
|         case .refreshSessions:
 | |
|             sessionTask?.cancel()
 | |
| 
 | |
|             sessionTask = Task { [weak self] in
 | |
|                 await MainActor.run {
 | |
|                     self?.state = .initial
 | |
|                 }
 | |
| 
 | |
|                 do {
 | |
|                     try await self?.updateSessions()
 | |
| 
 | |
|                     guard let self else { return }
 | |
| 
 | |
|                     await MainActor.run {
 | |
|                         self.state = .content
 | |
|                     }
 | |
|                 } catch {
 | |
|                     guard let self else { return }
 | |
|                     await MainActor.run {
 | |
|                         self.state = .error(.init(error.localizedDescription))
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             .asAnyCancellable()
 | |
| 
 | |
|             return .initial
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     private func updateSessions() async throws {
 | |
|         var parameters = Paths.GetSessionsParameters()
 | |
|         parameters.activeWithinSeconds = activeWithinSeconds
 | |
| 
 | |
|         let request = Paths.getSessions(parameters: parameters)
 | |
|         let response = try await userSession.client.send(request)
 | |
| 
 | |
|         let removedSessionIDs = sessions.keys.filter { !response.value.map(\.id).contains($0) }
 | |
| 
 | |
|         let existingIDs = sessions.keys
 | |
|             .filter {
 | |
|                 response.value.map(\.id).contains($0)
 | |
|             }
 | |
|         let newSessions = response.value
 | |
|             .filter {
 | |
|                 guard let id = $0.id else { return false }
 | |
|                 return !sessions.keys.contains(id)
 | |
|             }
 | |
|             .map { s in
 | |
|                 BindingBox<SessionInfoDto?>(
 | |
|                     source: .init(
 | |
|                         get: { s },
 | |
|                         set: { _ in }
 | |
|                     )
 | |
|                 )
 | |
|             }
 | |
| 
 | |
|         await MainActor.run {
 | |
|             for id in removedSessionIDs {
 | |
|                 let t = sessions[id]
 | |
|                 sessions[id] = nil
 | |
|                 t?.value = nil
 | |
|             }
 | |
| 
 | |
|             for id in existingIDs {
 | |
|                 sessions[id]?.value = response.value.first(where: { $0.id == id })
 | |
|             }
 | |
| 
 | |
|             for session in newSessions {
 | |
|                 guard let id = session.value?.id else { continue }
 | |
| 
 | |
|                 sessions[id] = session
 | |
|             }
 | |
| 
 | |
|             sessions.sort { x, y in
 | |
|                 let xs = x.value.value
 | |
|                 let ys = y.value.value
 | |
| 
 | |
|                 let isPlaying0 = xs?.nowPlayingItem != nil
 | |
|                 let isPlaying1 = ys?.nowPlayingItem != nil
 | |
| 
 | |
|                 if isPlaying0 && !isPlaying1 {
 | |
|                     return true
 | |
|                 } else if !isPlaying0 && isPlaying1 {
 | |
|                     return false
 | |
|                 }
 | |
| 
 | |
|                 if xs?.userName != ys?.userName {
 | |
|                     return (xs?.userName ?? "") < (ys?.userName ?? "")
 | |
|                 }
 | |
| 
 | |
|                 if isPlaying0 && isPlaying1 {
 | |
|                     return (xs?.nowPlayingItem?.name ?? "") < (ys?.nowPlayingItem?.name ?? "")
 | |
|                 } else {
 | |
|                     return (xs?.lastActivityDate ?? Date.now) > (ys?.lastActivityDate ?? Date.now)
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |