mastodon-ios/Mastodon/Scene/SuggestionAccount/SuggestionAccountViewModel....

141 lines
4.9 KiB
Swift
Raw Normal View History

2021-04-21 08:46:31 +02:00
//
// SuggestionAccountViewModel.swift
// Mastodon
//
// Created by sxiaojian on 2021/4/21.
//
import Combine
import CoreData
import CoreDataStack
import GameplayKit
import MastodonSDK
2022-10-08 07:43:06 +02:00
import MastodonCore
2021-04-21 08:46:31 +02:00
import UIKit
protocol SuggestionAccountViewModelDelegate: AnyObject {
2021-04-22 04:11:19 +02:00
var homeTimelineNeedRefresh: PassthroughSubject<Void, Never> { get }
}
2021-04-22 04:29:53 +02:00
2021-04-21 08:46:31 +02:00
final class SuggestionAccountViewModel: NSObject {
var disposeBag = Set<AnyCancellable>()
weak var delegate: SuggestionAccountViewModelDelegate?
2021-04-21 08:46:31 +02:00
// input
let context: AppContext
let authContext: AuthContext
let userFetchedResultsController: UserFetchedResultsController
var viewWillAppear = PassthroughSubject<Void, Never>()
// output
var tableViewDiffableDataSource: UITableViewDiffableDataSource<RecommendAccountSection, RecommendAccountItem>?
init(
context: AppContext,
authContext: AuthContext
) {
2021-04-21 08:46:31 +02:00
self.context = context
self.authContext = authContext
self.userFetchedResultsController = UserFetchedResultsController(
managedObjectContext: context.managedObjectContext,
domain: nil,
2022-02-16 10:25:55 +01:00
additionalPredicate: nil
)
super.init()
2022-02-16 10:25:55 +01:00
userFetchedResultsController.domain = authContext.mastodonAuthenticationBox.domain
// fetch recommended users
Task {
var userIDs: [MastodonUser.ID] = []
2021-04-21 11:58:56 +02:00
do {
let response = try await context.apiService.suggestionAccountV2(
2023-05-12 22:07:37 +02:00
query: .init(limit: 5),
authenticationBox: authContext.mastodonAuthenticationBox
)
userIDs = response.value.map { $0.account.id }
} catch let error as Mastodon.API.Error where error.httpResponseStatus == .notFound {
let response = try await context.apiService.suggestionAccount(
query: nil,
authenticationBox: authContext.mastodonAuthenticationBox
)
userIDs = response.value.map { $0.id }
2021-04-21 11:58:56 +02:00
} catch {
2023-05-12 22:07:37 +02:00
2021-04-21 11:58:56 +02:00
}
guard userIDs.isNotEmpty else { return }
userFetchedResultsController.userIDs = userIDs
2021-04-21 11:58:56 +02:00
}
2022-02-16 10:25:55 +01:00
// fetch relationship
userFetchedResultsController.$records
.removeDuplicates()
.sink { [weak self] records in
guard let _ = self else { return }
Task {
_ = try await context.apiService.relationship(
records: records,
authenticationBox: authContext.mastodonAuthenticationBox
2022-02-16 10:25:55 +01:00
)
}
}
.store(in: &disposeBag)
2021-04-21 11:58:56 +02:00
}
func setupDiffableDataSource(
tableView: UITableView,
suggestionAccountTableViewCellDelegate: SuggestionAccountTableViewCellDelegate
) {
tableViewDiffableDataSource = RecommendAccountSection.tableViewDiffableDataSource(
tableView: tableView,
context: context,
configuration: RecommendAccountSection.Configuration(
authContext: authContext,
suggestionAccountTableViewCellDelegate: suggestionAccountTableViewCellDelegate
)
)
userFetchedResultsController.$records
.receive(on: DispatchQueue.main)
.sink { [weak self] records in
guard let self = self else { return }
guard let tableViewDiffableDataSource = self.tableViewDiffableDataSource else { return }
var snapshot = NSDiffableDataSourceSnapshot<RecommendAccountSection, RecommendAccountItem>()
snapshot.appendSections([.main])
let items: [RecommendAccountItem] = records.map { RecommendAccountItem.account($0) }
snapshot.appendItems(items, toSection: .main)
tableViewDiffableDataSource.applySnapshotUsingReloadData(snapshot)
}
.store(in: &disposeBag)
}
2023-05-23 12:55:24 +02:00
func followAllSuggestedAccounts(_ dependency: NeedsDependency & AuthContextProvider, completion: (() -> Void)? = nil) {
let userRecords = userFetchedResultsController.records.compactMap {
$0.object(in: dependency.context.managedObjectContext)?.asRecord
}
2023-05-23 12:55:24 +02:00
Task {
await withTaskGroup(of: Void.self, body: { taskGroup in
for user in userRecords {
taskGroup.addTask {
try? await DataSourceFacade.responseToUserViewButtonAction(
dependency: dependency,
user: user,
buttonState: .follow
)
}
}
})
2023-05-23 13:13:34 +02:00
delegate?.homeTimelineNeedRefresh.send()
2023-05-23 12:55:24 +02:00
completion?()
}
}
2021-04-21 08:46:31 +02:00
}