From 74d1fbe89aef9519e1c147820c46200b59291e30 Mon Sep 17 00:00:00 2001 From: Nathan Mattes Date: Tue, 21 Nov 2023 14:40:31 +0100 Subject: [PATCH 1/5] Migrate FamiliarFollowers over to Entity (IOS-195) --- Mastodon.xcodeproj/project.pbxproj | 4 -- Mastodon/Coordinator/SceneCoordinator.swift | 4 +- .../DiscoveryForYouViewController.swift | 4 +- .../ForYou/DiscoveryForYouViewModel.swift | 4 +- .../FamiliarFollowersViewController.swift | 39 ++++++------ .../FamiliarFollowersViewModel+Diffable.swift | 39 ------------ .../FamiliarFollowersViewModel.swift | 61 +++++++++++-------- 7 files changed, 62 insertions(+), 93 deletions(-) delete mode 100644 Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel+Diffable.swift diff --git a/Mastodon.xcodeproj/project.pbxproj b/Mastodon.xcodeproj/project.pbxproj index 58de81668..924225de8 100644 --- a/Mastodon.xcodeproj/project.pbxproj +++ b/Mastodon.xcodeproj/project.pbxproj @@ -293,7 +293,6 @@ DB4FFC2C269EC39600D62E92 /* SearchTransitionController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB4FFC2A269EC39600D62E92 /* SearchTransitionController.swift */; }; DB5B549A2833A60400DEF8B2 /* FamiliarFollowersViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */; }; DB5B549D2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */; }; - DB5B549F2833A72500DEF8B2 /* FamiliarFollowersViewModel+Diffable.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B549E2833A72500DEF8B2 /* FamiliarFollowersViewModel+Diffable.swift */; }; DB5B54A12833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */; }; DB5B54A32833BD1A00DEF8B2 /* UserListViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A22833BD1A00DEF8B2 /* UserListViewModel.swift */; }; DB5B54A62833BE0000DEF8B2 /* UserListViewModel+State.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A52833BE0000DEF8B2 /* UserListViewModel+State.swift */; }; @@ -1005,7 +1004,6 @@ DB519B17281BCC2F00F0C99D /* tr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.stringsdict; name = tr; path = tr.lproj/Intents.stringsdict; sourceTree = ""; }; DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FamiliarFollowersViewController.swift; sourceTree = ""; }; DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FamiliarFollowersViewModel.swift; sourceTree = ""; }; - DB5B549E2833A72500DEF8B2 /* FamiliarFollowersViewModel+Diffable.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "FamiliarFollowersViewModel+Diffable.swift"; sourceTree = ""; }; DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "FamiliarFollowersViewController+DataSourceProvider.swift"; sourceTree = ""; }; DB5B54A22833BD1A00DEF8B2 /* UserListViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserListViewModel.swift; sourceTree = ""; }; DB5B54A52833BE0000DEF8B2 /* UserListViewModel+State.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UserListViewModel+State.swift"; sourceTree = ""; }; @@ -2288,7 +2286,6 @@ DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */, DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */, DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */, - DB5B549E2833A72500DEF8B2 /* FamiliarFollowersViewModel+Diffable.swift */, ); path = FamiliarFollowers; sourceTree = ""; @@ -3718,7 +3715,6 @@ 0FAA101C25E10E760017CCDE /* UIFont.swift in Sources */, 2D38F1D525CD465300561493 /* HomeTimelineViewController.swift in Sources */, DB6180E926391BDF0018D199 /* MediaHostToMediaPreviewViewControllerAnimatedTransitioning.swift in Sources */, - DB5B549F2833A72500DEF8B2 /* FamiliarFollowersViewModel+Diffable.swift in Sources */, DB6B351E2601FAEE00DC1E11 /* ComposeStatusAttachmentCollectionViewCell.swift in Sources */, DB8F7076279E954700E1225B /* DataSourceFacade+Follow.swift in Sources */, DB63F7542799491600455B82 /* DataSourceFacade+SearchHistory.swift in Sources */, diff --git a/Mastodon/Coordinator/SceneCoordinator.swift b/Mastodon/Coordinator/SceneCoordinator.swift index ef15ce6e4..4a01c4ea5 100644 --- a/Mastodon/Coordinator/SceneCoordinator.swift +++ b/Mastodon/Coordinator/SceneCoordinator.swift @@ -462,9 +462,7 @@ private extension SceneCoordinator { let followingListViewController = FollowingListViewController(viewModel: viewModel, coordinator: self, context: appContext) viewController = followingListViewController case .familiarFollowers(let viewModel): - let _viewController = FamiliarFollowersViewController() - _viewController.viewModel = viewModel - viewController = _viewController + viewController = FamiliarFollowersViewController(viewModel: viewModel, context: appContext, coordinator: self) case .rebloggedBy(let viewModel): let _viewController = RebloggedByViewController() _viewController.viewModel = viewModel diff --git a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift index 4d43e319a..a6b6720af 100644 --- a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift +++ b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift @@ -143,8 +143,8 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { return } - let familiarFollowersViewModel = FamiliarFollowersViewModel(context: context, authContext: authContext) - familiarFollowersViewModel.familiarFollowers = familiarFollowers + let familiarFollowersViewModel = FamiliarFollowersViewModel(context: context, authContext: authContext, accounts: viewModel.accounts, relationships: viewModel.relationships) + _ = coordinator.present( scene: .familiarFollowers(viewModel: familiarFollowersViewModel), from: self, diff --git a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewModel.swift b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewModel.swift index 4f0f730f4..0f118d0cf 100644 --- a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewModel.swift +++ b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewModel.swift @@ -23,7 +23,7 @@ final class DiscoveryForYouViewModel { @Published var familiarFollowers: [Mastodon.Entity.FamiliarFollowers] = [] @Published var isFetching = false @Published var accounts: [Mastodon.Entity.Account] - var relationships: [Mastodon.Entity.Relationship?] + var relationships: [Mastodon.Entity.Relationship] // output var diffableDataSource: UITableViewDiffableDataSource? @@ -72,7 +72,7 @@ extension DiscoveryForYouViewModel { snapshot.appendSections([.forYou]) let items = self.accounts.map { account in - let relationship = relationships.first { $0?.id == account.id } ?? nil + let relationship = relationships.first { $0.id == account.id } ?? nil return DiscoveryItem.account(account, relationship: relationship) } diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift index 59de6d880..cae1cc741 100644 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift +++ b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift @@ -14,36 +14,34 @@ import CoreDataStack final class FamiliarFollowersViewController: UIViewController, NeedsDependency { - weak var context: AppContext! { willSet { precondition(!isViewLoaded) } } - weak var coordinator: SceneCoordinator! { willSet { precondition(!isViewLoaded) } } - + weak var context: AppContext! + weak var coordinator: SceneCoordinator! + var disposeBag = Set() - var viewModel: FamiliarFollowersViewModel! + var viewModel: FamiliarFollowersViewModel - lazy var tableView: UITableView = { - let tableView = UITableView() + let tableView: UITableView + + init(viewModel: FamiliarFollowersViewModel, context: AppContext, coordinator: SceneCoordinator) { + self.viewModel = viewModel + self.context = context + self.coordinator = coordinator + + tableView = UITableView() tableView.rowHeight = UITableView.automaticDimension tableView.separatorStyle = .none tableView.backgroundColor = .clear - return tableView - }() - - -} -extension FamiliarFollowersViewController { - - override func viewDidLoad() { - super.viewDidLoad() - + super.init(nibName: nil, bundle: nil) + title = L10n.Scene.Familiarfollowers.title - + view.backgroundColor = .secondarySystemBackground - + tableView.translatesAutoresizingMaskIntoConstraints = false view.addSubview(tableView) tableView.pinToParent() - + tableView.delegate = self viewModel.setupDiffableDataSource( tableView: tableView, @@ -51,10 +49,13 @@ extension FamiliarFollowersViewController { ) } + required init?(coder: NSCoder) { fatalError("init(coder:) has not been implemented") } + override func viewWillAppear(_ animated: Bool) { super.viewWillAppear(animated) tableView.deselectRow(with: transitionCoordinator, animated: animated) + viewModel.viewWillAppear() } } diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel+Diffable.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel+Diffable.swift deleted file mode 100644 index 291a6b3ff..000000000 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel+Diffable.swift +++ /dev/null @@ -1,39 +0,0 @@ -// -// FamiliarFollowersViewModel+Diffable.swift -// Mastodon -// -// Created by MainasuK on 2022-5-17. -// - -import UIKit -import Combine - -extension FamiliarFollowersViewModel { - func setupDiffableDataSource( - tableView: UITableView, - userTableViewCellDelegate: UserTableViewCellDelegate? - ) { - diffableDataSource = UserSection.diffableDataSource( - tableView: tableView, - context: context, - authContext: authContext, - userTableViewCellDelegate: userTableViewCellDelegate - ) - - userFetchedResultsController.$records - .receive(on: DispatchQueue.main) - .sink { [weak self] records in - guard let self = self else { return } - guard let diffableDataSource = self.diffableDataSource else { return } - - var snapshot = NSDiffableDataSourceSnapshot() - snapshot.appendSections([.main]) - let items = records.map { UserItem.user(record: $0) } - snapshot.appendItems(items, toSection: .main) - - diffableDataSource.apply(snapshot, animatingDifferences: false) - } - .store(in: &disposeBag) - } - -} diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel.swift index 3e4a17e5e..3b936deef 100644 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel.swift +++ b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewModel.swift @@ -6,41 +6,54 @@ // import UIKit -import Combine import MastodonCore import MastodonSDK -import CoreDataStack final class FamiliarFollowersViewModel { - var disposeBag = Set() - - // input let context: AppContext let authContext: AuthContext - let userFetchedResultsController: UserFetchedResultsController - @Published var familiarFollowers: Mastodon.Entity.FamiliarFollowers? - + var accounts: [Mastodon.Entity.Account] + var relationships: [Mastodon.Entity.Relationship] + // output var diffableDataSource: UITableViewDiffableDataSource? - init(context: AppContext, authContext: AuthContext) { + init(context: AppContext, authContext: AuthContext, accounts: [Mastodon.Entity.Account], relationships: [Mastodon.Entity.Relationship]) { self.context = context self.authContext = authContext - self.userFetchedResultsController = UserFetchedResultsController( - managedObjectContext: context.managedObjectContext, - domain: authContext.mastodonAuthenticationBox.domain, - additionalPredicate: nil - ) - // end init - - $familiarFollowers - .map { familiarFollowers -> [MastodonUser.ID] in - guard let familiarFollowers = familiarFollowers else { return [] } - return familiarFollowers.accounts.map { $0.id } - } - .assign(to: \.userIDs, on: userFetchedResultsController) - .store(in: &disposeBag) + self.accounts = accounts + self.relationships = relationships + } + + func setupDiffableDataSource( + tableView: UITableView, + userTableViewCellDelegate: UserTableViewCellDelegate? + ) { + diffableDataSource = UserSection.diffableDataSource( + tableView: tableView, + context: context, + authContext: authContext, + userTableViewCellDelegate: userTableViewCellDelegate + ) + } + + func viewWillAppear() { + guard let diffableDataSource else { return } + + var snapshot = NSDiffableDataSourceSnapshot() + snapshot.appendSections([.main]) + let accountsWithRelationship: [(account: Mastodon.Entity.Account, relationship: Mastodon.Entity.Relationship?)] = accounts.compactMap { account in + guard let relationship = self.relationships.first(where: {$0.id == account.id }) else { return (account: account, relationship: nil)} + + return (account: account, relationship: relationship) + } + + let items = accountsWithRelationship.map { UserItem.account(account: $0.account, relationship: $0.relationship) } + + snapshot.appendItems(items, toSection: .main) + + diffableDataSource.apply(snapshot, animatingDifferences: false) + } - } From 5a3a10071bb256fc972d258aec138c5739123f73 Mon Sep 17 00:00:00 2001 From: Nathan Mattes Date: Tue, 21 Nov 2023 15:26:10 +0100 Subject: [PATCH 2/5] Show the correct familiar followers and their relationships (IOS-195) --- Mastodon.xcodeproj/project.pbxproj | 4 - .../DiscoveryForYouViewController.swift | 52 +++++++++---- ...ersViewController+DataSourceProvider.swift | 34 -------- .../FamiliarFollowersViewController.swift | 77 ++++++++++++++++++- 4 files changed, 112 insertions(+), 55 deletions(-) delete mode 100644 Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController+DataSourceProvider.swift diff --git a/Mastodon.xcodeproj/project.pbxproj b/Mastodon.xcodeproj/project.pbxproj index 924225de8..2b9d89174 100644 --- a/Mastodon.xcodeproj/project.pbxproj +++ b/Mastodon.xcodeproj/project.pbxproj @@ -293,7 +293,6 @@ DB4FFC2C269EC39600D62E92 /* SearchTransitionController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB4FFC2A269EC39600D62E92 /* SearchTransitionController.swift */; }; DB5B549A2833A60400DEF8B2 /* FamiliarFollowersViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */; }; DB5B549D2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */; }; - DB5B54A12833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */; }; DB5B54A32833BD1A00DEF8B2 /* UserListViewModel.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A22833BD1A00DEF8B2 /* UserListViewModel.swift */; }; DB5B54A62833BE0000DEF8B2 /* UserListViewModel+State.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A52833BE0000DEF8B2 /* UserListViewModel+State.swift */; }; DB5B54A82833BFA500DEF8B2 /* FavoritedByViewController.swift in Sources */ = {isa = PBXBuildFile; fileRef = DB5B54A72833BFA500DEF8B2 /* FavoritedByViewController.swift */; }; @@ -1004,7 +1003,6 @@ DB519B17281BCC2F00F0C99D /* tr */ = {isa = PBXFileReference; lastKnownFileType = text.plist.stringsdict; name = tr; path = tr.lproj/Intents.stringsdict; sourceTree = ""; }; DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FamiliarFollowersViewController.swift; sourceTree = ""; }; DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FamiliarFollowersViewModel.swift; sourceTree = ""; }; - DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "FamiliarFollowersViewController+DataSourceProvider.swift"; sourceTree = ""; }; DB5B54A22833BD1A00DEF8B2 /* UserListViewModel.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = UserListViewModel.swift; sourceTree = ""; }; DB5B54A52833BE0000DEF8B2 /* UserListViewModel+State.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = "UserListViewModel+State.swift"; sourceTree = ""; }; DB5B54A72833BFA500DEF8B2 /* FavoritedByViewController.swift */ = {isa = PBXFileReference; lastKnownFileType = sourcecode.swift; path = FavoritedByViewController.swift; sourceTree = ""; }; @@ -2284,7 +2282,6 @@ isa = PBXGroup; children = ( DB5B54992833A60400DEF8B2 /* FamiliarFollowersViewController.swift */, - DB5B54A02833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift */, DB5B549C2833A67400DEF8B2 /* FamiliarFollowersViewModel.swift */, ); path = FamiliarFollowers; @@ -3751,7 +3748,6 @@ D8E5C349296DB8A3007E76A7 /* StatusEditHistoryViewController.swift in Sources */, DBE0822425CD3F1E00FD6BBD /* MastodonRegisterViewModel.swift in Sources */, D8318A882A4468D300C0FB73 /* NotificationSettingsViewController.swift in Sources */, - DB5B54A12833A89600DEF8B2 /* FamiliarFollowersViewController+DataSourceProvider.swift in Sources */, 2D82B9FF25E7863200E36F0F /* OnboardingViewControllerAppearance.swift in Sources */, DB025B78278D606A002F581E /* StatusItem.swift in Sources */, D82BD7532ABC44C2009A374A /* Coordinator.swift in Sources */, diff --git a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift index a6b6720af..e0b78d39b 100644 --- a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift +++ b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift @@ -9,6 +9,7 @@ import UIKit import Combine import MastodonUI import MastodonCore +import MastodonSDK final class DiscoveryForYouViewController: UIViewController, NeedsDependency, MediaPreviewableViewController { @@ -136,20 +137,38 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { guard let indexPath = tableView.indexPath(for: cell) else { return } guard case let .account(account, _) = viewModel.diffableDataSource?.itemIdentifier(for: indexPath) else { return } - let userID = account.id - let _familiarFollowers = viewModel.familiarFollowers.first(where: { $0.id == userID }) - guard let familiarFollowers = _familiarFollowers else { - assertionFailure() - return - } - - let familiarFollowersViewModel = FamiliarFollowersViewModel(context: context, authContext: authContext, accounts: viewModel.accounts, relationships: viewModel.relationships) + coordinator.showLoading() - _ = coordinator.present( - scene: .familiarFollowers(viewModel: familiarFollowersViewModel), - from: self, - transition: .show - ) + Task { [weak self] in + + guard let self else { return } + do { + let userID = account.id + let familiarFollowers = viewModel.familiarFollowers.first(where: { $0.id == userID })?.accounts ?? [] + let relationships = try await context.apiService.relationship(forAccounts: familiarFollowers, authenticationBox: authContext.mastodonAuthenticationBox).value + + let familiarFollowersViewModel = FamiliarFollowersViewModel( + context: context, + authContext: authContext, + accounts: familiarFollowers, + relationships: relationships + ) + + coordinator.hideLoading() + + let viewController = coordinator.present( + scene: .familiarFollowers(viewModel: familiarFollowersViewModel), + from: self, + transition: .show + ) + + if let familiarFollowersViewController = viewController as? FamiliarFollowersViewController { + familiarFollowersViewController.delegate = self + } + } catch { + + } + } } } @@ -158,3 +177,10 @@ extension DiscoveryForYouViewController: ScrollViewContainer { var scrollView: UIScrollView { tableView } } +extension DiscoveryForYouViewController: FamiliarFollowersViewControllerDelegate { + func relationshipChanged(_ viewController: UIViewController, account: Mastodon.Entity.Account) { + Task { + try? await viewModel.fetch() + } + } +} diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController+DataSourceProvider.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController+DataSourceProvider.swift deleted file mode 100644 index c014a7900..000000000 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController+DataSourceProvider.swift +++ /dev/null @@ -1,34 +0,0 @@ -// -// FamiliarFollowersViewController+DataSourceProvider.swift -// Mastodon -// -// Created by MainasuK on 2022-5-17. -// - -import UIKit - -extension FamiliarFollowersViewController: DataSourceProvider { - func item(from source: DataSourceItem.Source) async -> DataSourceItem? { - var _indexPath = source.indexPath - if _indexPath == nil, let cell = source.tableViewCell { - _indexPath = await self.indexPath(for: cell) - } - guard let indexPath = _indexPath else { return nil } - - guard let item = viewModel.diffableDataSource?.itemIdentifier(for: indexPath) else { - return nil - } - - switch item { - case .user(let record): - return .user(record: record) - default: - return nil - } - } - - @MainActor - private func indexPath(for cell: UITableViewCell) async -> IndexPath? { - return tableView.indexPath(for: cell) - } -} diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift index cae1cc741..5ae64d00c 100644 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift +++ b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift @@ -10,16 +10,20 @@ import Combine import MastodonCore import MastodonLocalization import MastodonUI +import MastodonSDK import CoreDataStack +protocol FamiliarFollowersViewControllerDelegate: AnyObject { + func relationshipChanged(_ viewController: UIViewController, account: Mastodon.Entity.Account) +} + final class FamiliarFollowersViewController: UIViewController, NeedsDependency { weak var context: AppContext! weak var coordinator: SceneCoordinator! + let viewModel: FamiliarFollowersViewModel + weak var delegate: FamiliarFollowersViewControllerDelegate? - var disposeBag = Set() - var viewModel: FamiliarFollowersViewModel - let tableView: UITableView init(viewModel: FamiliarFollowersViewModel, context: AppContext, coordinator: SceneCoordinator) { @@ -81,4 +85,69 @@ extension FamiliarFollowersViewController: UITableViewDelegate, AutoGenerateTabl } // MARK: - UserTableViewCellDelegate -extension FamiliarFollowersViewController: UserTableViewCellDelegate {} +extension FamiliarFollowersViewController: UserTableViewCellDelegate { + func userView(_ view: UserView, didTapButtonWith state: UserView.ButtonState, for user: MastodonUser) { } + func userView(_ view: UserView, didTapButtonWith state: UserView.ButtonState, for account: Mastodon.Entity.Account, me: MastodonUser?) { + + // Can we call the default implementation somehow? Maybe add a FollowAction-class with a completion-block? + Task { + await MainActor.run { view.setButtonState(.loading) } + + try await DataSourceFacade.responseToUserViewButtonAction( + dependency: self, + user: account, + buttonState: state + ) + + // this is a dirty hack to give the backend enough time to process the relationship-change + // Otherwise the relationship might still be `pending` + try await Task.sleep(for: .seconds(1)) + + let relationship = try await self.context.apiService.relationship(forAccounts: [account], authenticationBox: authContext.mastodonAuthenticationBox).value.first + + let isMe: Bool + if let me { + isMe = account.id == me.id + } else { + isMe = false + } + + await MainActor.run { + view.viewModel.relationship = relationship + view.updateButtonState(with: relationship, isMe: isMe) + + delegate?.relationshipChanged(self, account: account) + } + } + } + +} + +//MARK: - DataSourceProvider +extension FamiliarFollowersViewController: DataSourceProvider { + func item(from source: DataSourceItem.Source) async -> DataSourceItem? { + var _indexPath = source.indexPath + if _indexPath == nil, let cell = source.tableViewCell { + _indexPath = await self.indexPath(for: cell) + } + guard let indexPath = _indexPath else { return nil } + + guard let item = viewModel.diffableDataSource?.itemIdentifier(for: indexPath) else { + return nil + } + + switch item { + case .account(let account, relationship: let relationship): + return .account(account: account, relationship: relationship) + + default: + return nil + } + } + + @MainActor + private func indexPath(for cell: UITableViewCell) async -> IndexPath? { + return tableView.indexPath(for: cell) + } +} + From 1fbc2a8d1c3ba95fe03afb58e7d2f6e650bdf11f Mon Sep 17 00:00:00 2001 From: Nathan Mattes Date: Tue, 21 Nov 2023 16:51:46 +0100 Subject: [PATCH 3/5] Remove delegate again (IOS-195) --- .../DiscoveryForYouViewController.swift | 16 +------ .../FamiliarFollowersViewController.swift | 43 +------------------ 2 files changed, 3 insertions(+), 56 deletions(-) diff --git a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift index e0b78d39b..ec4e25a10 100644 --- a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift +++ b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift @@ -147,6 +147,8 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { let familiarFollowers = viewModel.familiarFollowers.first(where: { $0.id == userID })?.accounts ?? [] let relationships = try await context.apiService.relationship(forAccounts: familiarFollowers, authenticationBox: authContext.mastodonAuthenticationBox).value + coordinator.hideLoading() + let familiarFollowersViewModel = FamiliarFollowersViewModel( context: context, authContext: authContext, @@ -154,17 +156,11 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { relationships: relationships ) - coordinator.hideLoading() - let viewController = coordinator.present( scene: .familiarFollowers(viewModel: familiarFollowersViewModel), from: self, transition: .show ) - - if let familiarFollowersViewController = viewController as? FamiliarFollowersViewController { - familiarFollowersViewController.delegate = self - } } catch { } @@ -176,11 +172,3 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { extension DiscoveryForYouViewController: ScrollViewContainer { var scrollView: UIScrollView { tableView } } - -extension DiscoveryForYouViewController: FamiliarFollowersViewControllerDelegate { - func relationshipChanged(_ viewController: UIViewController, account: Mastodon.Entity.Account) { - Task { - try? await viewModel.fetch() - } - } -} diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift index 5ae64d00c..d750bb43a 100644 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift +++ b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift @@ -13,16 +13,11 @@ import MastodonUI import MastodonSDK import CoreDataStack -protocol FamiliarFollowersViewControllerDelegate: AnyObject { - func relationshipChanged(_ viewController: UIViewController, account: Mastodon.Entity.Account) -} - final class FamiliarFollowersViewController: UIViewController, NeedsDependency { weak var context: AppContext! weak var coordinator: SceneCoordinator! let viewModel: FamiliarFollowersViewModel - weak var delegate: FamiliarFollowersViewControllerDelegate? let tableView: UITableView @@ -85,43 +80,7 @@ extension FamiliarFollowersViewController: UITableViewDelegate, AutoGenerateTabl } // MARK: - UserTableViewCellDelegate -extension FamiliarFollowersViewController: UserTableViewCellDelegate { - func userView(_ view: UserView, didTapButtonWith state: UserView.ButtonState, for user: MastodonUser) { } - func userView(_ view: UserView, didTapButtonWith state: UserView.ButtonState, for account: Mastodon.Entity.Account, me: MastodonUser?) { - - // Can we call the default implementation somehow? Maybe add a FollowAction-class with a completion-block? - Task { - await MainActor.run { view.setButtonState(.loading) } - - try await DataSourceFacade.responseToUserViewButtonAction( - dependency: self, - user: account, - buttonState: state - ) - - // this is a dirty hack to give the backend enough time to process the relationship-change - // Otherwise the relationship might still be `pending` - try await Task.sleep(for: .seconds(1)) - - let relationship = try await self.context.apiService.relationship(forAccounts: [account], authenticationBox: authContext.mastodonAuthenticationBox).value.first - - let isMe: Bool - if let me { - isMe = account.id == me.id - } else { - isMe = false - } - - await MainActor.run { - view.viewModel.relationship = relationship - view.updateButtonState(with: relationship, isMe: isMe) - - delegate?.relationshipChanged(self, account: account) - } - } - } - -} +extension FamiliarFollowersViewController: UserTableViewCellDelegate {} //MARK: - DataSourceProvider extension FamiliarFollowersViewController: DataSourceProvider { From e894596108efad7c03dd6b57bec4523663a87374 Mon Sep 17 00:00:00 2001 From: Nathan Mattes Date: Tue, 21 Nov 2023 16:55:39 +0100 Subject: [PATCH 4/5] Remove imports --- .../FamiliarFollowers/FamiliarFollowersViewController.swift | 2 -- 1 file changed, 2 deletions(-) diff --git a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift index d750bb43a..57b2ac897 100644 --- a/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift +++ b/Mastodon/Scene/Profile/FamiliarFollowers/FamiliarFollowersViewController.swift @@ -10,8 +10,6 @@ import Combine import MastodonCore import MastodonLocalization import MastodonUI -import MastodonSDK -import CoreDataStack final class FamiliarFollowersViewController: UIViewController, NeedsDependency { From 1637f0e1466e60a1be937f062a3eadf6b4a44d51 Mon Sep 17 00:00:00 2001 From: Nathan Mattes Date: Tue, 21 Nov 2023 17:12:17 +0100 Subject: [PATCH 5/5] Fix warning :facepalm: --- .../Scene/Discovery/ForYou/DiscoveryForYouViewController.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift index ec4e25a10..90a136611 100644 --- a/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift +++ b/Mastodon/Scene/Discovery/ForYou/DiscoveryForYouViewController.swift @@ -156,7 +156,7 @@ extension DiscoveryForYouViewController: ProfileCardTableViewCellDelegate { relationships: relationships ) - let viewController = coordinator.present( + _ = coordinator.present( scene: .familiarFollowers(viewModel: familiarFollowersViewModel), from: self, transition: .show