241 lines
7.4 KiB
Swift
241 lines
7.4 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) 2022 Jellyfin & Jellyfin Contributors
|
|
//
|
|
|
|
import ActivityIndicator
|
|
import Combine
|
|
import Foundation
|
|
import JellyfinAPI
|
|
|
|
final class HomeViewModel: ViewModel {
|
|
|
|
@Published
|
|
var latestAddedItems: [BaseItemDto] = []
|
|
@Published
|
|
var resumeItems: [BaseItemDto] = []
|
|
@Published
|
|
var nextUpItems: [BaseItemDto] = []
|
|
@Published
|
|
var librariesShowRecentlyAddedIDs: [String] = []
|
|
@Published
|
|
var libraries: [BaseItemDto] = []
|
|
|
|
// temp
|
|
var recentFilterSet = LibraryFilters(filters: [], sortOrder: [.descending], sortBy: [.dateAdded])
|
|
|
|
override init() {
|
|
super.init()
|
|
refresh()
|
|
|
|
// Nov. 6, 2021
|
|
// This is a workaround since Stinsen doesn't have the ability to rebuild a root at the time of writing.
|
|
// See ServerDetailViewModel.swift for feature request issue
|
|
Notifications[.didSignIn].subscribe(self, selector: #selector(didSignIn))
|
|
Notifications[.didSignOut].subscribe(self, selector: #selector(didSignOut))
|
|
}
|
|
|
|
@objc
|
|
private func didSignIn() {
|
|
for cancellable in cancellables {
|
|
cancellable.cancel()
|
|
}
|
|
|
|
librariesShowRecentlyAddedIDs = []
|
|
libraries = []
|
|
resumeItems = []
|
|
nextUpItems = []
|
|
|
|
refresh()
|
|
}
|
|
|
|
@objc
|
|
private func didSignOut() {
|
|
for cancellable in cancellables {
|
|
cancellable.cancel()
|
|
}
|
|
|
|
cancellables.removeAll()
|
|
}
|
|
|
|
@objc
|
|
func refresh() {
|
|
LogManager.log.debug("Refresh called.")
|
|
|
|
refreshLibrariesLatest()
|
|
refreshLatestAddedItems()
|
|
refreshResumeItems()
|
|
refreshNextUpItems()
|
|
}
|
|
|
|
// MARK: Libraries Latest Items
|
|
|
|
private func refreshLibrariesLatest() {
|
|
UserViewsAPI.getUserViews(userId: SessionManager.main.currentLogin.user.id)
|
|
.trackActivity(loading)
|
|
.sink(receiveCompletion: { completion in
|
|
switch completion {
|
|
case .finished: ()
|
|
case .failure:
|
|
self.libraries = []
|
|
}
|
|
|
|
self.handleAPIRequestError(completion: completion)
|
|
}, receiveValue: { response in
|
|
|
|
var newLibraries: [BaseItemDto] = []
|
|
|
|
response.items!.forEach { item in
|
|
LogManager.log
|
|
.debug("Retrieved user view: \(item.id!) (\(item.name ?? "nil")) with type \(item.collectionType ?? "nil")")
|
|
if item.collectionType == "movies" || item.collectionType == "tvshows" {
|
|
newLibraries.append(item)
|
|
}
|
|
}
|
|
|
|
UserAPI.getCurrentUser()
|
|
.trackActivity(self.loading)
|
|
.sink(receiveCompletion: { completion in
|
|
switch completion {
|
|
case .finished: ()
|
|
case .failure:
|
|
self.libraries = []
|
|
self.handleAPIRequestError(completion: completion)
|
|
}
|
|
}, receiveValue: { response in
|
|
let excludeIDs = response.configuration?.latestItemsExcludes != nil ? response.configuration!
|
|
.latestItemsExcludes! : []
|
|
|
|
for excludeID in excludeIDs {
|
|
newLibraries.removeAll { library in
|
|
library.id == excludeID
|
|
}
|
|
}
|
|
|
|
self.libraries = newLibraries
|
|
})
|
|
.store(in: &self.cancellables)
|
|
})
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
// MARK: Latest Added Items
|
|
|
|
private func refreshLatestAddedItems() {
|
|
UserLibraryAPI.getLatestMedia(
|
|
userId: SessionManager.main.currentLogin.user.id,
|
|
fields: [
|
|
.primaryImageAspectRatio,
|
|
.seriesPrimaryImage,
|
|
.seasonUserData,
|
|
.overview,
|
|
.genres,
|
|
.people,
|
|
.chapters,
|
|
],
|
|
includeItemTypes: [.movie, .series],
|
|
enableImageTypes: [.primary, .backdrop, .thumb],
|
|
enableUserData: true,
|
|
limit: 8
|
|
)
|
|
.sink { completion in
|
|
switch completion {
|
|
case .finished: ()
|
|
case .failure:
|
|
self.nextUpItems = []
|
|
self.handleAPIRequestError(completion: completion)
|
|
}
|
|
} receiveValue: { items in
|
|
LogManager.log.debug("Retrieved \(String(items.count)) resume items")
|
|
|
|
self.latestAddedItems = items
|
|
}
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
// MARK: Resume Items
|
|
|
|
private func refreshResumeItems() {
|
|
ItemsAPI.getResumeItems(
|
|
userId: SessionManager.main.currentLogin.user.id,
|
|
limit: 6,
|
|
fields: [
|
|
.primaryImageAspectRatio,
|
|
.seriesPrimaryImage,
|
|
.seasonUserData,
|
|
.overview,
|
|
.genres,
|
|
.people,
|
|
.chapters,
|
|
],
|
|
enableUserData: true
|
|
)
|
|
.trackActivity(loading)
|
|
.sink(receiveCompletion: { completion in
|
|
switch completion {
|
|
case .finished: ()
|
|
case .failure:
|
|
self.resumeItems = []
|
|
self.handleAPIRequestError(completion: completion)
|
|
}
|
|
}, receiveValue: { response in
|
|
LogManager.log.debug("Retrieved \(String(response.items!.count)) resume items")
|
|
|
|
self.resumeItems = response.items ?? []
|
|
})
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
func removeItemFromResume(_ item: BaseItemDto) {
|
|
guard let itemID = item.id, resumeItems.contains(where: { $0.id == itemID }) else { return }
|
|
|
|
PlaystateAPI.markUnplayedItem(
|
|
userId: SessionManager.main.currentLogin.user.id,
|
|
itemId: item.id!
|
|
)
|
|
.sink(receiveCompletion: { [weak self] completion in
|
|
self?.handleAPIRequestError(completion: completion)
|
|
}, receiveValue: { _ in
|
|
self.refreshResumeItems()
|
|
self.refreshNextUpItems()
|
|
})
|
|
.store(in: &cancellables)
|
|
}
|
|
|
|
// MARK: Next Up Items
|
|
|
|
private func refreshNextUpItems() {
|
|
TvShowsAPI.getNextUp(
|
|
userId: SessionManager.main.currentLogin.user.id,
|
|
limit: 6,
|
|
fields: [
|
|
.primaryImageAspectRatio,
|
|
.seriesPrimaryImage,
|
|
.seasonUserData,
|
|
.overview,
|
|
.genres,
|
|
.people,
|
|
.chapters,
|
|
],
|
|
enableUserData: true
|
|
)
|
|
.trackActivity(loading)
|
|
.sink(receiveCompletion: { completion in
|
|
switch completion {
|
|
case .finished: ()
|
|
case .failure:
|
|
self.nextUpItems = []
|
|
self.handleAPIRequestError(completion: completion)
|
|
}
|
|
}, receiveValue: { response in
|
|
LogManager.log.debug("Retrieved \(String(response.items!.count)) nextup items")
|
|
|
|
self.nextUpItems = response.items ?? []
|
|
})
|
|
.store(in: &cancellables)
|
|
}
|
|
}
|