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
|
|
|
|
|
2021-04-21 12:52:09 +02:00
|
|
|
protocol SuggestionAccountViewModelDelegate: AnyObject {
|
2021-04-22 04:11:19 +02:00
|
|
|
var homeTimelineNeedRefresh: PassthroughSubject<Void, Never> { get }
|
2021-04-21 12:52:09 +02:00
|
|
|
}
|
2021-04-22 04:29:53 +02:00
|
|
|
|
2021-04-21 08:46:31 +02:00
|
|
|
final class SuggestionAccountViewModel: NSObject {
|
|
|
|
var disposeBag = Set<AnyCancellable>()
|
|
|
|
|
2022-02-10 12:30:41 +01:00
|
|
|
weak var delegate: SuggestionAccountViewModelDelegate?
|
|
|
|
|
2021-04-21 08:46:31 +02:00
|
|
|
// input
|
|
|
|
let context: AppContext
|
2022-10-09 14:07:57 +02:00
|
|
|
let authContext: AuthContext
|
2022-02-10 12:30:41 +01:00
|
|
|
let userFetchedResultsController: UserFetchedResultsController
|
2023-05-12 21:04:47 +02:00
|
|
|
|
2022-02-10 12:30:41 +01:00
|
|
|
var viewWillAppear = PassthroughSubject<Void, Never>()
|
2021-04-23 04:25:08 +02:00
|
|
|
|
2022-02-10 12:30:41 +01:00
|
|
|
// output
|
|
|
|
var tableViewDiffableDataSource: UITableViewDiffableDataSource<RecommendAccountSection, RecommendAccountItem>?
|
|
|
|
|
|
|
|
init(
|
2022-10-09 14:07:57 +02:00
|
|
|
context: AppContext,
|
|
|
|
authContext: AuthContext
|
2022-02-10 12:30:41 +01:00
|
|
|
) {
|
2021-04-21 08:46:31 +02:00
|
|
|
self.context = context
|
2022-10-09 14:07:57 +02:00
|
|
|
self.authContext = authContext
|
2022-02-10 12:30:41 +01:00
|
|
|
self.userFetchedResultsController = UserFetchedResultsController(
|
|
|
|
managedObjectContext: context.managedObjectContext,
|
|
|
|
domain: nil,
|
2022-02-16 10:25:55 +01:00
|
|
|
additionalPredicate: nil
|
|
|
|
)
|
2022-02-10 12:30:41 +01:00
|
|
|
super.init()
|
2022-02-16 10:25:55 +01:00
|
|
|
|
2022-10-09 14:07:57 +02:00
|
|
|
userFetchedResultsController.domain = authContext.mastodonAuthenticationBox.domain
|
2023-05-12 21:04:47 +02:00
|
|
|
|
2023-05-22 14:45:52 +02:00
|
|
|
// fetch recommended users
|
2022-02-10 12:30:41 +01:00
|
|
|
Task {
|
|
|
|
var userIDs: [MastodonUser.ID] = []
|
2021-04-21 11:58:56 +02:00
|
|
|
do {
|
2022-02-10 12:30:41 +01:00
|
|
|
let response = try await context.apiService.suggestionAccountV2(
|
2023-05-12 22:07:37 +02:00
|
|
|
query: .init(limit: 5),
|
2022-10-09 14:07:57 +02:00
|
|
|
authenticationBox: authContext.mastodonAuthenticationBox
|
2022-02-10 12:30:41 +01:00
|
|
|
)
|
|
|
|
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,
|
2022-10-09 14:07:57 +02:00
|
|
|
authenticationBox: authContext.mastodonAuthenticationBox
|
2022-02-10 12:30:41 +01:00
|
|
|
)
|
|
|
|
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
|
|
|
}
|
2022-02-10 12:30:41 +01:00
|
|
|
|
2023-09-26 14:26:16 +02:00
|
|
|
guard userIDs.isNotEmpty else { return }
|
2022-02-10 12:30:41 +01:00
|
|
|
userFetchedResultsController.userIDs = userIDs
|
2021-04-21 11:58:56 +02:00
|
|
|
}
|
2022-02-10 12:30:41 +01: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,
|
2022-10-09 14:07:57 +02:00
|
|
|
authenticationBox: authContext.mastodonAuthenticationBox
|
2022-02-16 10:25:55 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
.store(in: &disposeBag)
|
2021-04-21 11:58:56 +02:00
|
|
|
}
|
2022-02-10 12:30:41 +01:00
|
|
|
|
2023-05-22 16:14:06 +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) {
|
2023-05-22 11:41:42 +02:00
|
|
|
|
|
|
|
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?()
|
2023-05-22 11:41:42 +02:00
|
|
|
}
|
|
|
|
}
|
2021-04-21 08:46:31 +02:00
|
|
|
}
|