mastodon-app-ufficiale-ipho.../Mastodon/Scene/HomeTimeline/HomeTimelineViewModel+LoadL...

155 lines
5.7 KiB
Swift
Raw Normal View History

2021-02-07 07:42:50 +01:00
//
// HomeTimelineViewModel+LoadLatestState.swift
// Mastodon
//
// Created by sxiaojian on 2021/2/5.
//
import func QuartzCore.CACurrentMediaTime
import Foundation
import CoreData
import CoreDataStack
import GameplayKit
2022-10-08 07:43:06 +02:00
import MastodonCore
2024-01-08 11:17:40 +01:00
import MastodonSDK
2021-02-07 07:42:50 +01:00
extension HomeTimelineViewModel {
class LoadLatestState: GKState {
let id = UUID()
var name: String {
String(describing: Self.self)
}
2021-02-07 07:42:50 +01:00
weak var viewModel: HomeTimelineViewModel?
init(viewModel: HomeTimelineViewModel) {
self.viewModel = viewModel
}
override func didEnter(from previousState: GKState?) {
super.didEnter(from: previousState)
2021-02-07 07:42:50 +01:00
viewModel?.loadLatestStateMachinePublisher.send(self)
}
@MainActor
func enter(state: LoadLatestState.Type) {
stateMachine?.enter(state)
}
2021-02-07 07:42:50 +01:00
}
}
extension HomeTimelineViewModel.LoadLatestState {
class Initial: HomeTimelineViewModel.LoadLatestState {
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
return stateClass == Loading.self || stateClass == LoadingManually.self
2021-02-07 07:42:50 +01:00
}
}
class Loading: HomeTimelineViewModel.LoadLatestState {
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
return stateClass == Fail.self || stateClass == Idle.self
}
override func didEnter(from previousState: GKState?) {
didEnter(from: previousState, viewModel: viewModel, isUserInitiated: false)
}
}
class LoadingManually: HomeTimelineViewModel.LoadLatestState {
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
return stateClass == Fail.self || stateClass == Idle.self
}
override func didEnter(from previousState: GKState?) {
didEnter(from: previousState, viewModel: viewModel, isUserInitiated: true)
2021-02-07 07:42:50 +01:00
}
}
class Fail: HomeTimelineViewModel.LoadLatestState {
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
return stateClass == Loading.self || stateClass == Idle.self
}
}
class Idle: HomeTimelineViewModel.LoadLatestState {
override func isValidNextState(_ stateClass: AnyClass) -> Bool {
return stateClass == Loading.self || stateClass == LoadingManually.self
2021-02-07 07:42:50 +01:00
}
}
private func didEnter(from previousState: GKState?, viewModel: HomeTimelineViewModel?, isUserInitiated: Bool) {
super.didEnter(from: previousState)
guard let viewModel else { return }
2024-01-08 11:17:40 +01:00
let latestFeedRecords = viewModel.dataController.records.prefix(APIService.onceRequestStatusMaxCount)
Task {
let latestStatusIDs: [Status.ID] = latestFeedRecords.compactMap { record in
2024-01-08 11:17:40 +01:00
return record.status?.id
}
do {
let response = try await viewModel.context.apiService.homeTimeline(
authenticationBox: viewModel.authContext.mastodonAuthenticationBox
)
await enter(state: Idle.self)
viewModel.homeTimelineNavigationBarTitleViewModel.receiveLoadingStateCompletion(.finished)
viewModel.context.instanceService.updateMutesAndBlocks()
// stop refresher if no new statuses
let statuses = response.value
let newStatuses = statuses.filter { !latestStatusIDs.contains($0.id) }
2023-09-21 12:52:05 +02:00
if newStatuses.isEmpty {
viewModel.didLoadLatest.send()
} else {
2024-01-15 15:03:49 +01:00
var oldRecords = viewModel.dataController.records
if !latestStatusIDs.isEmpty {
viewModel.homeTimelineNavigationBarTitleViewModel.newPostsIncoming()
}
2024-01-08 11:17:40 +01:00
var newRecords: [MastodonFeed] = newStatuses.map {
MastodonFeed.fromStatus(.fromEntity($0), kind: .home)
}
2024-01-15 15:03:49 +01:00
if let lastNewId = newRecords.last?.status?.id, let firstOld = oldRecords.first?.status, lastNewId != firstOld.id {
// Add "Load more"
newRecords.append(
MastodonFeed(hasMore: true, isLoadingMore: false, status: firstOld, notification: nil, kind: .none)
)
}
2024-01-08 11:17:40 +01:00
viewModel.dataController.records = {
for (i, record) in newRecords.enumerated() {
if let index = oldRecords.firstIndex(where: { $0.status?.reblog?.id == record.id || $0.status?.id == record.id }) {
oldRecords[index] = record
if newRecords.count > index {
2024-01-15 15:03:49 +01:00
newRecords.remove(at: index)
2024-01-08 11:17:40 +01:00
}
}
}
return (newRecords + oldRecords).removingDuplicates()
}()
}
viewModel.timelineIsEmpty.value = latestStatusIDs.isEmpty && statuses.isEmpty
if !isUserInitiated {
await UIImpactFeedbackGenerator(style: .light)
.impactOccurred()
}
} catch {
await enter(state: Idle.self)
viewModel.didLoadLatest.send()
viewModel.homeTimelineNavigationBarTitleViewModel.receiveLoadingStateCompletion(.failure(error))
}
} // end Task
}
2021-02-07 07:42:50 +01:00
}