// // 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) 2022 Jellyfin & Jellyfin Contributors // import Combine import Foundation import JellyfinAPI import UIKit class ItemViewModel: ViewModel { @Published var item: BaseItemDto @Published var playButtonItem: BaseItemDto? { didSet { if let playButtonItem = playButtonItem { refreshItemVideoPlayerViewModel(for: playButtonItem) } } } @Published var similarItems: [BaseItemDto] = [] @Published var isWatched = false @Published var isFavorited = false @Published var selectedVideoPlayerViewModel: VideoPlayerViewModel? @Published var videoPlayerViewModels: [VideoPlayerViewModel] = [] init(item: BaseItemDto) { self.item = item super.init() switch item.type { case .episode, .movie: if !item.missing && !item.unaired { self.playButtonItem = item } default: () } isFavorited = item.userData?.isFavorite ?? false isWatched = item.userData?.played ?? false getSimilarItems() refreshItemVideoPlayerViewModel(for: item) Notifications[.didSendStopReport].subscribe(self, selector: #selector(receivedStopReport(_:))) } @objc private func receivedStopReport(_ notification: NSNotification) { guard let itemID = notification.object as? String else { return } if itemID == item.id { updateItem() } else { // Remove if necessary. Note that this cannot be in deinit as // holding as an observer won't allow the object to be deinit-ed Notifications.unsubscribe(self) } } func refreshItemVideoPlayerViewModel(for item: BaseItemDto) { guard item.type == .episode || item.type == .movie else { return } guard !item.missing, !item.unaired else { return } item.createVideoPlayerViewModel() .sink { completion in self.handleAPIRequestError(completion: completion) } receiveValue: { viewModels in self.videoPlayerViewModels = viewModels self.selectedVideoPlayerViewModel = viewModels.first } .store(in: &cancellables) } func playButtonText() -> String { if item.unaired { return L10n.unaired } if item.missing { return L10n.missing } if let itemProgressString = item.getItemProgressString() { return itemProgressString } return L10n.play } func getSimilarItems() { LibraryAPI.getSimilarItems( itemId: item.id!, userId: SessionManager.main.currentLogin.user.id, limit: 20, fields: ItemFields.allCases ) .trackActivity(loading) .sink(receiveCompletion: { [weak self] completion in self?.handleAPIRequestError(completion: completion) }, receiveValue: { [weak self] response in self?.similarItems = response.items ?? [] }) .store(in: &cancellables) } func toggleWatchState() { let current = isWatched isWatched.toggle() let request: AnyPublisher if current { request = PlaystateAPI.markUnplayedItem(userId: SessionManager.main.currentLogin.user.id, itemId: item.id!) } else { request = PlaystateAPI.markPlayedItem(userId: SessionManager.main.currentLogin.user.id, itemId: item.id!) } request .trackActivity(loading) .sink(receiveCompletion: { [weak self] completion in switch completion { case .failure: self?.isWatched = !current case .finished: () } self?.handleAPIRequestError(completion: completion) }, receiveValue: { _ in }) .store(in: &cancellables) } func toggleFavoriteState() { let current = isFavorited isFavorited.toggle() let request: AnyPublisher if current { request = UserLibraryAPI.unmarkFavoriteItem(userId: SessionManager.main.currentLogin.user.id, itemId: item.id!) } else { request = UserLibraryAPI.markFavoriteItem(userId: SessionManager.main.currentLogin.user.id, itemId: item.id!) } request .trackActivity(loading) .sink(receiveCompletion: { [weak self] completion in switch completion { case .failure: self?.isFavorited = !current case .finished: () } self?.handleAPIRequestError(completion: completion) }, receiveValue: { _ in }) .store(in: &cancellables) } // Overridden by subclasses func updateItem() {} }