mirror of
https://github.com/vector-im/element-ios.git
synced 2024-09-28 23:32:41 +00:00
79b8f80ce9
* Update voice broadcast tiles UI (#6965) * Translated using Weblate (German) Currently translated at 100.0% (2307 of 2307 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * speeding the animation a bit * tests and identifier improvements * fix * changelog * removed unused code * Avoid unnecessary send state request (#6970) * comment * Curate MXCrypto protocol methods * Add voice broadcast initial state in bubble data (#6972) - Add voice broadcast initial state in bubble data - Remove the local record after sending * Voice Broadcast: log and block unexpected state change * Sing out bottom bar * new line * Enable WYSIWYG plain text support * Remove change on Apple swift-collections revision * removed RiotSettings a non RiotSwiftUI reference from the ViewState code * fixed a test * Complete MXCryptoV2 implementation * Multi session logut * Switch the CI to code 14 and the iOS 14 simulator, fix UI tests * Fixes #6987 - Prevent ZXing from unnecessarily requesting camera access * Fixes #6988 - Prevent actor switching when tearing down the rendezvous * Separator fix * Removed warnings * add Z-Labs tag or rich text editor and update to the new label naming * changelog * Hide old sessions list when the new dm is enabled * Add changelog.d file * Sing out filtering * Avoid simultaneous state changes (#6986) * Improve kebab menu in UserSessionOverview * Add UI tests * Add changelog.d file * No customization for emptycell (#7000) * PSG-976 Exclude current session from security recommendations and other sessions * Padding fix * Fixed unit tests * Add empty onLearnMoreAction closure * Add InfoView skeleton * Add UserSessionOverviewViewBindings * Style info view * Add bottom sheet modifier * Localise content * Add inactive sessions copy * Fix bug in InlineTextButton * Improve UserSessionCardView * Add “learn more” button in UserOtherSessions * Show bottom sheet in user other sessions * Show rename info alert * Refine UX * Add iOS 15- fallback * Refine InfoView * Add UI tests * Improve UserOtherSessionsUITests * Improve InlineTextButton API * Add changelod.d file * Fix failing UTs * Hide keyboard in UserSessionName * Add .viewSessionInfo view action * Voice Broadcast - BugFix - send the last chunk (#7002) * Voice Broadcast - BugFix - send the last chunk with the right sequence number - we reset now and teardown the service only after the last chunk is sent * updated package + tests * change log * Bug Fix : Crash if the room has avatar and voice broadcast tiles * Add MVVM-C for InfoSheet * improving UI tests for slow CI * removing comment * test improvements for slow ci * Show bottom sheet in other sessions screen * Show bottom sheet in rename session screen * Delete bottom sheet modifier * Show rename sheet * UI and unit tests * Refresh fix * Changelog * Add InfoSheet SwiftUI preview * simplify the test to make it pass on the CI * Fix memory leak * Cleanup UI tests * improving tests for the CI * Fixed IRC-style message and commands support in Rich text editor * tests updated for the CI * test improvements * removing a test that can't pass on the CI due to its speed * Changelog * CryptoV2 changes * Display crypto version * Voice broadcast - Disable the sleep mode during the recording until we are able to handle it Currently go to "sleep mode" pauses the voice broadcast recording * Add issue automation for the VoIP team * Renamed sign out to logout * Renamed sign out to logout * Renamed sign out to logout * Sign out of all other sessions * Fix typo in issue automation * Fixed unit tests * Translations update from Weblate (#7017) * Translated using Weblate (Portuguese (Brazil)) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/pt_BR/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (German) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (German) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Dutch) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/nl/ Co-authored-by: lvre <7uu3qrbvm@relay.firefox.com> Co-authored-by: Priit Jõerüüt <riot@joeruut.com> Co-authored-by: Vri <element@vrifox.cc> Co-authored-by: Johan Smits <johan@smitsmail.net> * Prepare for new sprint * Prepare for new sprint * Threads: added support to read receipts (MSC3771) - Update after review * Threads: added support to notifications count (MSC3773) * Update RiotSwiftUI/Modules/UserSessions/UserOtherSessions/Test/UI/UserOtherSessionsUITests.swift Co-authored-by: aringenbach <80891108+aringenbach@users.noreply.github.com> * Update RiotSwiftUI/Modules/UserSessions/UserOtherSessions/Test/UI/UserOtherSessionsUITests.swift Co-authored-by: aringenbach <80891108+aringenbach@users.noreply.github.com> * Comment fix * the test may fail on CI without blocking the task/check * tests may fail on CI * test improvement * test may fail on CI * Hide push toggles for http pushers when there is no server support * changelog * Code review fixes * Threads: added support to read receipts (MSC3771) - Update after review * Synchronise composer and toolbar resizing animation duration * Add kResizeComposerAnimationDuration constant description * fix for 6946 * Threads: add support to labs flag for read receipts * Cleanup * Code review fixes, created DestructiveButton * Update issue automation Stop using deprecated ProjectNext API in favour of the new ProjectV2 one * Update PR automation Stop using deprecated ProjectNext API in favour of the new ProjectV2 one * Code review fixes * Map location info * Map location info * Add location feature in UserSessionsOverview * Add “show location” feature in other sessions list * Add “show location“ feature in session overview * Fix Package.resolved * Cleanup merge leftovers * Cleanup code * Cleanup * Add show/hide ip persistency * Add location info in UserOtherSessions * Refine settings logic * Mock settings in UserSessionsOverviewViewModel * Add settings service in UserOtherSessionsViewModel * Inject setting service in UserSessionOverviewViewModel * Add changelog.d file * Fix UTs * Cleanup merge leftovers * Add animations * Fix failing test * Amend title font * Amend copies * Device Manager: Session list item is not tappable everywhere * changelog * Threads notification count in main timeline including un participated threads * Changed title and body * Removed "Do not ask again" button * Remove indication about plain text mode coming soon * Prevent `Unable to activate constraint with anchors .. because they have no common ancestor.` crashes. Only link toasts to the top safe area instead of the navigation controller * Revert "Replace attributed string height calculation with a more reliable implementation" This reverts commit81773cd1e5
. * Revert "Fix timeline items text height calculation" This reverts commit8f9eddee50
. * Revert "Fixes vector-im/element-ios/issues/6441 - Incorrect timeline item text height calculation (#6679)" This reverts commit405c2d8e32
. * Fixes vector-im/element-ios/issues/6441 - Incorrect timeline item text height calculation * Prepare for new sprint * Refine bottom sheet layout * updated pod * changelog * Switch to using an API key for interactions with AppStoreConnect while on CI; update fastlane and dependencies * Rich-text editor: Fix text formatting enabled inconsistent state * Labs: Rich-text editor - Fix text formatting switch losing the current content of the composer * Re-order View computed properties and move to private mark * Add intrinsic sized bottom sheet * Snooze controller * Changelog * Fix composer view model tests * Rich-text editor: enable translations between Markdown and HTML when toggling text formatting * Force a layout on the room bubble cell messageTextView to get a correct frame * Move Move UserAgentParserTests * Add UserSessionDetailsUITests * Improve UserSessionNameUITests * Cleanup tests * Improve UserSessionNameViewModelTests * Test empty state for UserOtherSessions * Fix typo * Cleanup unused code * Add changelog.d file * Threads: removed "unread_thread_notifications" from sync filters for server that doesn't support MSC3773 * Remove 10s wait on failed initial sync * Threads: removed "unread_thread_notifications" from sync filters for server that doesn't support MSC3773 - Update after review * Revert "Device Manager: Session list item is not tappable everywhere" This reverts commite6367cba4c
. * Fixup session list item is not tappable everywhere * Fix accessibility id in UserOtherSessions * Poll not usable after logging out and back in * Changelog * Removed init * voice dictation now works * plain text * Add voice broadcast slider (#7010) * Display number of unread messages above threads button * Translations update from Weblate (#7080) * Translated using Weblate (Dutch) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/nl/ * Translated using Weblate (German) Currently translated at 100.0% (2311 of 2311 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Dutch) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/nl/ * Translated using Weblate (German) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Albanian) Currently translated at 99.6% (2303 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sq/ * Translated using Weblate (Portuguese (Brazil)) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/pt_BR/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (Indonesian) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/id/ * Translated using Weblate (Slovak) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sk/ * Translated using Weblate (Italian) Currently translated at 100.0% (2312 of 2312 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/it/ * Translated using Weblate (German) Currently translated at 100.0% (2313 of 2313 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2313 of 2313 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Indonesian) Currently translated at 100.0% (2313 of 2313 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/id/ * Translated using Weblate (Dutch) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/nl/ * Translated using Weblate (German) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Hungarian) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/hu/ * Translated using Weblate (Italian) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/it/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (Indonesian) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/id/ * Translated using Weblate (Slovak) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sk/ * Translated using Weblate (Portuguese (Brazil)) Currently translated at 100.0% (2315 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/pt_BR/ * Translated using Weblate (Albanian) Currently translated at 99.5% (2305 of 2315 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sq/ * Translated using Weblate (German) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Japanese) Currently translated at 66.2% (1534 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/ja/ * Translated using Weblate (Portuguese (Brazil)) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/pt_BR/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (Indonesian) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/id/ * Translated using Weblate (Slovak) Currently translated at 100.0% (2317 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sk/ * Translated using Weblate (Japanese) Currently translated at 66.2% (1534 of 2317 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/ja/ * Translated using Weblate (German) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (German) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/de/ * Translated using Weblate (Ukrainian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/uk/ * Translated using Weblate (Estonian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/et/ * Translated using Weblate (Slovak) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/sk/ * Translated using Weblate (Italian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/it/ * Translated using Weblate (Hungarian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/hu/ * Translated using Weblate (Indonesian) Currently translated at 100.0% (2326 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/id/ * Translated using Weblate (French) Currently translated at 97.6% (2272 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/fr/ * Translated using Weblate (Russian) Currently translated at 80.5% (1873 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/ru/ * Translated using Weblate (Russian) Currently translated at 80.7% (1879 of 2326 strings) Translation: Element iOS/Element iOS Translate-URL: https://translate.element.io/projects/riot-ios/riot-ios/ru/ Co-authored-by: Roel ter Maat <roel.termaat@nedap.com> Co-authored-by: Vri <element@vrifox.cc> Co-authored-by: Besnik Bleta <besnik@programeshqip.org> Co-authored-by: lvre <7uu3qrbvm@relay.firefox.com> Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com> Co-authored-by: Priit Jõerüüt <riot@joeruut.com> Co-authored-by: Linerly <linerly@protonmail.com> Co-authored-by: Jozef Gaal <preklady@mayday.sk> Co-authored-by: random <dictionary@tutamail.com> Co-authored-by: Szimszon <github@oregpreshaz.eu> Co-authored-by: Suguru Hirahara <ovestekona@protonmail.com> Co-authored-by: Thibault Martin <mail@thibaultmart.in> Co-authored-by: Platon Terekhov <ockenfels_vevent@aleeas.com> * changelog.d: Upgrade MatrixSDK version ([v0.24.3](https://github.com/matrix-org/matrix-ios-sdk/releases/tag/v0.24.3)). * version++ Co-authored-by: giomfo <gforet@matrix.org> Co-authored-by: Yoan Pintas <y.pintas@gmail.com> Co-authored-by: Vri <element@vrifox.cc> Co-authored-by: Anderas <andyuhnak@gmail.com> Co-authored-by: Mauro Romito <mauro.romito@element.io> Co-authored-by: Velin92 <34335419+Velin92@users.noreply.github.com> Co-authored-by: Giom Foret <giom@matrix.org> Co-authored-by: Aleksandrs Proskurins <paleksandrs@gmail.com> Co-authored-by: aringenbach <arnaudr@element.io> Co-authored-by: manuroe <manuroe@users.noreply.github.com> Co-authored-by: David Langley <langley.dave@gmail.com> Co-authored-by: Stefan Ceriu <stefanc@matrix.org> Co-authored-by: Alfonso Grillo <alfogrillo@gmail.com> Co-authored-by: Alfonso Grillo <alfogrillo@element.io> Co-authored-by: Kat Gerasimova <ekaterinag@element.io> Co-authored-by: Element Translate Bot <admin@riot.im> Co-authored-by: lvre <7uu3qrbvm@relay.firefox.com> Co-authored-by: Priit Jõerüüt <riot@joeruut.com> Co-authored-by: Johan Smits <johan@smitsmail.net> Co-authored-by: gulekismail <ismailgulek0@gmail.com> Co-authored-by: Gil Eluard <gile@element.io> Co-authored-by: Aleksandrs Proskurins <aleksandrsp@element.io> Co-authored-by: aringenbach <80891108+aringenbach@users.noreply.github.com> Co-authored-by: Stefan Ceriu <stefan.ceriu@gmail.com> Co-authored-by: Roel ter Maat <roel.termaat@nedap.com> Co-authored-by: Besnik Bleta <besnik@programeshqip.org> Co-authored-by: Ihor Hordiichuk <igor_ck@outlook.com> Co-authored-by: Linerly <linerly@protonmail.com> Co-authored-by: Jozef Gaal <preklady@mayday.sk> Co-authored-by: random <dictionary@tutamail.com> Co-authored-by: Szimszon <github@oregpreshaz.eu> Co-authored-by: Suguru Hirahara <ovestekona@protonmail.com> Co-authored-by: Thibault Martin <mail@thibaultmart.in> Co-authored-by: Platon Terekhov <ockenfels_vevent@aleeas.com>
1003 lines
35 KiB
Objective-C
1003 lines
35 KiB
Objective-C
/*
|
||
Copyright 2017 Vector Creations Ltd
|
||
Copyright 2018 New Vector Ltd
|
||
|
||
Licensed under the Apache License, Version 2.0 (the "License");
|
||
you may not use this file except in compliance with the License.
|
||
You may obtain a copy of the License at
|
||
|
||
http://www.apache.org/licenses/LICENSE-2.0
|
||
|
||
Unless required by applicable law or agreed to in writing, software
|
||
distributed under the License is distributed on an "AS IS" BASIS,
|
||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
See the License for the specific language governing permissions and
|
||
limitations under the License.
|
||
*/
|
||
|
||
#import "MasterTabBarController.h"
|
||
|
||
#import "RecentsDataSource.h"
|
||
|
||
|
||
#import "MXRoom+Riot.h"
|
||
#import "MXSession+Riot.h"
|
||
|
||
#import "SettingsViewController.h"
|
||
#import "SecurityViewController.h"
|
||
|
||
#import "GeneratedInterface-Swift.h"
|
||
|
||
@interface MasterTabBarController () <UITabBarControllerDelegate>
|
||
{
|
||
// Array of `MXSession` instances.
|
||
NSMutableArray<MXSession*> *mxSessionArray;
|
||
|
||
// The recents data source shared between all the view controllers of the tab bar.
|
||
RecentsDataSource *recentsDataSource;
|
||
|
||
// Current alert (if any).
|
||
UIAlertController *currentAlert;
|
||
|
||
// Keep reference on the pushed view controllers to release them correctly
|
||
NSMutableArray *childViewControllers;
|
||
|
||
// Observe kThemeServiceDidChangeThemeNotification to handle user interface theme change.
|
||
id kThemeServiceDidChangeThemeNotificationObserver;
|
||
|
||
// Custom title view of the navigation bar
|
||
MainTitleView *titleView;
|
||
|
||
id spaceNotificationCounterDidUpdateNotificationCountObserver;
|
||
}
|
||
|
||
@property(nonatomic,getter=isHidden) BOOL hidden;
|
||
|
||
@property (nonatomic, readwrite) OnboardingCoordinatorBridgePresenter *onboardingCoordinatorBridgePresenter;
|
||
|
||
// Tell whether the onboarding screen is preparing.
|
||
@property (nonatomic, readwrite) BOOL isOnboardingCoordinatorPreparing;
|
||
@property (nonatomic, readwrite) BOOL isOnboardingInProgress;
|
||
|
||
@property (nonatomic) BOOL reviewSessionAlertHasBeenDisplayed;
|
||
|
||
@end
|
||
|
||
@implementation MasterTabBarController
|
||
@synthesize onboardingCoordinatorBridgePresenter, selectedRoomId, selectedEventId, selectedRoomSession, selectedRoomPreviewData, selectedContact, isOnboardingInProgress;
|
||
|
||
#pragma mark - Properties override
|
||
|
||
- (HomeViewController *)homeViewController
|
||
{
|
||
UIViewController *wrapperVC = [self viewControllerForClass:HomeViewControllerWithBannerWrapperViewController.class];
|
||
return [(HomeViewControllerWithBannerWrapperViewController *)wrapperVC homeViewController];
|
||
}
|
||
|
||
- (FavouritesViewController *)favouritesViewController
|
||
{
|
||
return (FavouritesViewController*)[self viewControllerForClass:FavouritesViewController.class];
|
||
}
|
||
|
||
- (PeopleViewController *)peopleViewController
|
||
{
|
||
return (PeopleViewController*)[self viewControllerForClass:PeopleViewController.class];
|
||
}
|
||
|
||
- (RoomsViewController *)roomsViewController
|
||
{
|
||
return (RoomsViewController*)[self viewControllerForClass:RoomsViewController.class];
|
||
}
|
||
|
||
#pragma mark - Life cycle
|
||
|
||
- (void)viewDidLoad
|
||
{
|
||
[super viewDidLoad];
|
||
// Do any additional setup after loading the view, typically from a nib.
|
||
|
||
self.delegate = self;
|
||
|
||
self.isOnboardingInProgress = NO;
|
||
|
||
// Note: UITabBarViewController should not be embed in a UINavigationController (https://github.com/vector-im/riot-ios/issues/3086)
|
||
[self vc_removeBackTitle];
|
||
|
||
[self setupTitleView];
|
||
titleView.titleLabel.text = [VectorL10n allChatsTitle];
|
||
|
||
childViewControllers = [NSMutableArray array];
|
||
|
||
MXWeakify(self);
|
||
spaceNotificationCounterDidUpdateNotificationCountObserver = [[NSNotificationCenter defaultCenter] addObserverForName:MXSpaceNotificationCounter.didUpdateNotificationCount object:nil queue:nil usingBlock:^(NSNotification * _Nonnull note) {
|
||
MXStrongifyAndReturnIfNil(self);
|
||
[self updateSideMenuNotifcationIcon];
|
||
}];
|
||
}
|
||
|
||
- (void)userInterfaceThemeDidChange
|
||
{
|
||
id<Theme> theme = ThemeService.shared.theme;
|
||
[theme applyStyleOnNavigationBar:self.navigationController.navigationBar];
|
||
|
||
[theme applyStyleOnTabBar:self.tabBar];
|
||
|
||
self.view.backgroundColor = theme.backgroundColor;
|
||
[titleView updateWithTheme:theme];
|
||
|
||
[self setNeedsStatusBarAppearanceUpdate];
|
||
}
|
||
|
||
- (UIViewController *)childViewControllerForStatusBarStyle
|
||
{
|
||
return self.selectedViewController;
|
||
}
|
||
|
||
- (UIViewController *)childViewControllerForStatusBarHidden
|
||
{
|
||
return self.selectedViewController;
|
||
}
|
||
|
||
- (void)viewWillAppear:(BOOL)animated
|
||
{
|
||
[super viewWillAppear:animated];
|
||
|
||
// Show the tab bar view controller content only when a user is logged in.
|
||
self.hidden = ([MXKAccountManager sharedManager].accounts.count == 0);
|
||
|
||
if (!kThemeServiceDidChangeThemeNotificationObserver)
|
||
{
|
||
// Observe user interface theme change.
|
||
kThemeServiceDidChangeThemeNotificationObserver = [[NSNotificationCenter defaultCenter] addObserverForName:kThemeServiceDidChangeThemeNotification object:nil queue:[NSOperationQueue mainQueue] usingBlock:^(NSNotification *notif) {
|
||
|
||
[self userInterfaceThemeDidChange];
|
||
|
||
}];
|
||
[self userInterfaceThemeDidChange];
|
||
}
|
||
}
|
||
|
||
- (void)viewDidAppear:(BOOL)animated
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] viewDidAppear");
|
||
[super viewDidAppear:animated];
|
||
|
||
// Check whether we're not logged in
|
||
BOOL authIsShown = NO;
|
||
if (![MXKAccountManager sharedManager].accounts.count)
|
||
{
|
||
[self showOnboardingFlow];
|
||
authIsShown = YES;
|
||
}
|
||
else if (![MXKAccountManager sharedManager].activeAccounts.count)
|
||
{
|
||
// Display a login screen if the account is soft logout
|
||
// Note: We support only one account
|
||
MXKAccount *account = [MXKAccountManager sharedManager].accounts.firstObject;
|
||
if (account.isSoftLogout)
|
||
{
|
||
[self showSoftLogoutOnboardingFlowWithCredentials:account.mxCredentials];
|
||
authIsShown = YES;
|
||
}
|
||
}
|
||
|
||
if (!authIsShown)
|
||
{
|
||
[self refreshTabBarBadges];
|
||
|
||
// Release properly pushed and/or presented view controller
|
||
if (childViewControllers.count)
|
||
{
|
||
for (id viewController in childViewControllers)
|
||
{
|
||
if ([viewController isKindOfClass:[UINavigationController class]])
|
||
{
|
||
UINavigationController *navigationController = (UINavigationController*)viewController;
|
||
for (id subViewController in navigationController.viewControllers)
|
||
{
|
||
if ([subViewController respondsToSelector:@selector(destroy)])
|
||
{
|
||
[subViewController destroy];
|
||
}
|
||
}
|
||
}
|
||
else if ([viewController respondsToSelector:@selector(destroy)])
|
||
{
|
||
[viewController destroy];
|
||
}
|
||
}
|
||
|
||
[childViewControllers removeAllObjects];
|
||
}
|
||
|
||
[[AppDelegate theDelegate] checkAppVersion];
|
||
}
|
||
}
|
||
|
||
- (void)viewDidDisappear:(BOOL)animated
|
||
{
|
||
[super viewDidDisappear:animated];
|
||
}
|
||
|
||
- (void)dealloc
|
||
{
|
||
mxSessionArray = nil;
|
||
|
||
if (currentAlert)
|
||
{
|
||
[currentAlert dismissViewControllerAnimated:NO completion:nil];
|
||
currentAlert = nil;
|
||
}
|
||
|
||
if (kThemeServiceDidChangeThemeNotificationObserver)
|
||
{
|
||
[[NSNotificationCenter defaultCenter] removeObserver:kThemeServiceDidChangeThemeNotificationObserver];
|
||
kThemeServiceDidChangeThemeNotificationObserver = nil;
|
||
}
|
||
|
||
if (spaceNotificationCounterDidUpdateNotificationCountObserver)
|
||
{
|
||
[[NSNotificationCenter defaultCenter] removeObserver:spaceNotificationCounterDidUpdateNotificationCountObserver];
|
||
spaceNotificationCounterDidUpdateNotificationCountObserver = nil;
|
||
}
|
||
|
||
childViewControllers = nil;
|
||
}
|
||
|
||
#pragma mark - Public
|
||
|
||
- (void)updateViewControllers:(NSArray<UIViewController*>*)viewControllers
|
||
{
|
||
self.viewControllers = viewControllers;
|
||
|
||
[self initializeDataSources];
|
||
|
||
// Need to be called in case of the controllers have been replaced
|
||
[self.selectedViewController viewWillAppear:NO];
|
||
|
||
// Adjust the display of the icons in the tabbar.
|
||
for (UITabBarItem *tabBarItem in self.tabBar.items)
|
||
{
|
||
if (@available(iOS 13.0, *))
|
||
{
|
||
// Fix iOS 13 misalignment tab bar images. Some titles are nil and other empty strings. Nil title behaves as if a non-empty title was set.
|
||
// Note: However no need to modify imageInsets property on iOS 13.
|
||
tabBarItem.title = @"";
|
||
}
|
||
else
|
||
{
|
||
tabBarItem.imageInsets = UIEdgeInsetsMake(5, 0, -5, 0);
|
||
}
|
||
}
|
||
|
||
self.titleLabelText = [self getTitleForItemViewController:self.selectedViewController];
|
||
|
||
// Need to be called in case of the controllers have been replaced
|
||
[self.selectedViewController viewDidAppear:NO];
|
||
}
|
||
|
||
- (void)removeTabAt:(MasterTabBarIndex)tag
|
||
{
|
||
NSInteger index = [self indexOfTabItemWithTag:tag];
|
||
if (index != NSNotFound) {
|
||
NSMutableArray<UIViewController*> *viewControllers = [NSMutableArray arrayWithArray:self.viewControllers];
|
||
[viewControllers removeObjectAtIndex:index];
|
||
self.viewControllers = viewControllers;
|
||
}
|
||
}
|
||
|
||
- (void)selectTabAtIndex:(MasterTabBarIndex)tabBarIndex
|
||
{
|
||
NSInteger index = [self indexOfTabItemWithTag:tabBarIndex];
|
||
self.selectedIndex = index;
|
||
|
||
self.titleLabelText = [self getTitleForItemViewController:self.selectedViewController];
|
||
}
|
||
|
||
#pragma mark -
|
||
|
||
- (NSArray<MXSession*>*)mxSessions
|
||
{
|
||
return [NSArray arrayWithArray:mxSessionArray];
|
||
}
|
||
|
||
- (void)initializeDataSources
|
||
{
|
||
MXSession *mainSession = mxSessionArray.firstObject;
|
||
|
||
if (mainSession)
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] initializeDataSources");
|
||
|
||
// Init the recents data source
|
||
RecentsListService *recentsListService = [[RecentsListService alloc] initWithSession:mainSession];
|
||
recentsDataSource = [[RecentsDataSource alloc] initWithMatrixSession:mainSession
|
||
recentsListService:recentsListService];
|
||
|
||
[self.homeViewController displayList:recentsDataSource];
|
||
[self.favouritesViewController displayList:recentsDataSource];
|
||
[self.peopleViewController displayList:recentsDataSource];
|
||
[self.roomsViewController displayList:recentsDataSource];
|
||
|
||
// Restore the right delegate of the shared recent data source.
|
||
id<MXKDataSourceDelegate> recentsDataSourceDelegate = self.homeViewController;
|
||
RecentsDataSourceMode recentsDataSourceMode = self.homeViewController.recentsDataSourceMode;
|
||
|
||
NSInteger tabItemTag = self.tabBar.items[self.selectedIndex].tag;
|
||
|
||
switch (tabItemTag)
|
||
{
|
||
case TABBAR_HOME_INDEX:
|
||
break;
|
||
case TABBAR_FAVOURITES_INDEX:
|
||
recentsDataSourceDelegate = self.favouritesViewController;
|
||
recentsDataSourceMode = RecentsDataSourceModeFavourites;
|
||
break;
|
||
case TABBAR_PEOPLE_INDEX:
|
||
recentsDataSourceDelegate = self.peopleViewController;
|
||
recentsDataSourceMode = RecentsDataSourceModePeople;
|
||
break;
|
||
case TABBAR_ROOMS_INDEX:
|
||
recentsDataSourceDelegate = self.roomsViewController;
|
||
recentsDataSourceMode = RecentsDataSourceModeRooms;
|
||
break;
|
||
|
||
default:
|
||
break;
|
||
}
|
||
[recentsDataSource setDelegate:recentsDataSourceDelegate andRecentsDataSourceMode:recentsDataSourceMode];
|
||
|
||
// Check whether there are others sessions
|
||
NSArray<MXSession*>* mxSessions = self.mxSessions;
|
||
if (mxSessions.count > 1)
|
||
{
|
||
for (MXSession *mxSession in mxSessions)
|
||
{
|
||
if (mxSession != mainSession)
|
||
{
|
||
// Add the session to the recents data source
|
||
[recentsDataSource addMatrixSession:mxSession];
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
- (void)addMatrixSession:(MXSession *)mxSession
|
||
{
|
||
if ([mxSessionArray containsObject:mxSession])
|
||
{
|
||
MXLogDebug(@"MasterTabBarController already has %@ in mxSessionArray", mxSession)
|
||
return;
|
||
}
|
||
|
||
// Check whether the controller's view is loaded into memory.
|
||
if (self.homeViewController)
|
||
{
|
||
// Check whether the data sources have been initialized.
|
||
if (!recentsDataSource)
|
||
{
|
||
// Add first the session. The updated sessions list will be used during data sources initialization.
|
||
mxSessionArray = [NSMutableArray array];
|
||
[mxSessionArray addObject:mxSession];
|
||
|
||
// Prepare data sources and return
|
||
[self initializeDataSources];
|
||
|
||
// Add matrix sessions observer on first added session
|
||
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(onMatrixSessionStateDidChange:) name:kMXSessionStateDidChangeNotification object:nil];
|
||
return;
|
||
}
|
||
else
|
||
{
|
||
// Add the session to the existing data sources
|
||
[recentsDataSource addMatrixSession:mxSession];
|
||
}
|
||
}
|
||
|
||
if (!mxSessionArray)
|
||
{
|
||
mxSessionArray = [NSMutableArray array];
|
||
|
||
// Add matrix sessions observer on first added session
|
||
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(onMatrixSessionStateDidChange:) name:kMXSessionStateDidChangeNotification object:nil];
|
||
}
|
||
[mxSessionArray addObject:mxSession];
|
||
}
|
||
|
||
- (void)removeMatrixSession:(MXSession *)mxSession
|
||
{
|
||
if (![mxSessionArray containsObject:mxSession])
|
||
{
|
||
MXLogDebug(@"MasterTabBarController does not contain %@ in mxSessionArray", mxSession)
|
||
return;
|
||
}
|
||
|
||
[recentsDataSource removeMatrixSession:mxSession];
|
||
|
||
// Check whether there are others sessions
|
||
if (!recentsDataSource.mxSessions.count)
|
||
{
|
||
// Remove matrix sessions observer
|
||
[[NSNotificationCenter defaultCenter] removeObserver:self name:kMXSessionStateDidChangeNotification object:nil];
|
||
|
||
[self.homeViewController displayList:nil];
|
||
[self.favouritesViewController displayList:nil];
|
||
[self.peopleViewController displayList:nil];
|
||
[self.roomsViewController displayList:nil];
|
||
|
||
[recentsDataSource destroy];
|
||
recentsDataSource = nil;
|
||
}
|
||
|
||
[mxSessionArray removeObject:mxSession];
|
||
}
|
||
|
||
- (void)onMatrixSessionStateDidChange:(NSNotification *)notif
|
||
{
|
||
[self refreshTabBarBadges];
|
||
}
|
||
|
||
// TODO: Manage the onboarding coordinator at the AppCoordinator level
|
||
- (void)presentOnboardingFlow
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] presentOnboardingFlow");
|
||
|
||
MXWeakify(self);
|
||
OnboardingCoordinatorBridgePresenter *onboardingCoordinatorBridgePresenter = [[OnboardingCoordinatorBridgePresenter alloc] init];
|
||
onboardingCoordinatorBridgePresenter.completion = ^{
|
||
MXStrongifyAndReturnIfNil(self);
|
||
[self.onboardingCoordinatorBridgePresenter dismissWithAnimated:YES completion:nil];
|
||
self.onboardingCoordinatorBridgePresenter = nil;
|
||
|
||
self.isOnboardingInProgress = NO; // Must be set before calling didCompleteAuthentication
|
||
[self.masterTabBarDelegate masterTabBarControllerDidCompleteAuthentication:self];
|
||
};
|
||
|
||
[onboardingCoordinatorBridgePresenter presentFrom:self animated:NO];
|
||
|
||
self.onboardingCoordinatorBridgePresenter = onboardingCoordinatorBridgePresenter;
|
||
self.isOnboardingCoordinatorPreparing = NO;
|
||
}
|
||
|
||
- (void)showOnboardingFlow
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] showOnboardingFlow");
|
||
[self showOnboardingFlowAndResetSessionFlags:YES];
|
||
}
|
||
|
||
- (void)showSoftLogoutOnboardingFlowWithCredentials:(MXCredentials*)credentials;
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] showAuthenticationScreenAfterSoftLogout");
|
||
|
||
// This method can be called after the user chooses to clear their data as the MXSession
|
||
// is opened to call logout from. So we only set the credentials when authentication isn't
|
||
// in progress to prevent a second soft logout screen being shown.
|
||
if (!self.onboardingCoordinatorBridgePresenter && !self.isOnboardingCoordinatorPreparing)
|
||
{
|
||
AuthenticationService.shared.softLogoutCredentials = credentials;
|
||
|
||
[self showOnboardingFlowAndResetSessionFlags:NO];
|
||
}
|
||
}
|
||
|
||
- (void)showOnboardingFlowAndResetSessionFlags:(BOOL)resetSessionFlags
|
||
{
|
||
// Check whether an authentication screen is not already shown or preparing
|
||
if (!self.onboardingCoordinatorBridgePresenter && !self.isOnboardingCoordinatorPreparing)
|
||
{
|
||
self.isOnboardingCoordinatorPreparing = YES;
|
||
self.isOnboardingInProgress = YES;
|
||
|
||
if (resetSessionFlags)
|
||
{
|
||
[self resetReviewSessionsFlags];
|
||
}
|
||
|
||
[[AppDelegate theDelegate] restoreInitialDisplay:^{
|
||
|
||
[self presentOnboardingFlow];
|
||
}];
|
||
}
|
||
}
|
||
|
||
- (void)selectRoomWithParameters:(RoomNavigationParameters*)paramaters completion:(void (^)(void))completion
|
||
{
|
||
[self releaseSelectedItem];
|
||
|
||
selectedRoomId = paramaters.roomId;
|
||
selectedEventId = paramaters.eventId;
|
||
selectedRoomSession = paramaters.mxSession;
|
||
|
||
[self.masterTabBarDelegate masterTabBarController:self didSelectRoomWithParameters:paramaters completion:completion];
|
||
|
||
[self refreshSelectedControllerSelectedCellIfNeeded];
|
||
}
|
||
|
||
- (void)selectRoomPreviewWithParameters:(RoomPreviewNavigationParameters*)parameters completion:(void (^)(void))completion
|
||
{
|
||
[self releaseSelectedItem];
|
||
|
||
RoomPreviewData *roomPreviewData = parameters.previewData;
|
||
|
||
selectedRoomPreviewData = roomPreviewData;
|
||
selectedRoomId = roomPreviewData.roomId;
|
||
selectedRoomSession = roomPreviewData.mxSession;
|
||
|
||
[self.masterTabBarDelegate masterTabBarController:self didSelectRoomPreviewWithParameters:parameters completion:completion];
|
||
|
||
[self refreshSelectedControllerSelectedCellIfNeeded];
|
||
}
|
||
|
||
- (void)selectContact:(MXKContact*)contact
|
||
{
|
||
ScreenPresentationParameters *presentationParameters = [[ScreenPresentationParameters alloc] initWithRestoreInitialDisplay:YES stackAboveVisibleViews:NO];
|
||
|
||
[self selectContact:contact withPresentationParameters:presentationParameters];
|
||
}
|
||
|
||
- (void)selectContact:(MXKContact*)contact withPresentationParameters:(ScreenPresentationParameters*)presentationParameters
|
||
{
|
||
[self releaseSelectedItem];
|
||
|
||
selectedContact = contact;
|
||
|
||
[self.masterTabBarDelegate masterTabBarController:self didSelectContact:contact withPresentationParameters:presentationParameters];
|
||
|
||
[self refreshSelectedControllerSelectedCellIfNeeded];
|
||
}
|
||
|
||
- (void)releaseSelectedItem
|
||
{
|
||
selectedRoomId = nil;
|
||
selectedEventId = nil;
|
||
selectedRoomSession = nil;
|
||
selectedRoomPreviewData = nil;
|
||
|
||
selectedContact = nil;
|
||
}
|
||
|
||
- (NSUInteger)missedDiscussionsCount
|
||
{
|
||
NSUInteger roomCount = 0;
|
||
|
||
// Considering all the current sessions.
|
||
for (MXSession *session in mxSessionArray)
|
||
{
|
||
roomCount += [session vc_missedDiscussionsCount];
|
||
}
|
||
|
||
return roomCount;
|
||
}
|
||
|
||
- (NSUInteger)missedHighlightDiscussionsCount
|
||
{
|
||
NSUInteger roomCount = 0;
|
||
|
||
for (MXSession *session in mxSessionArray)
|
||
{
|
||
roomCount += [session missedHighlightDiscussionsCount];
|
||
}
|
||
|
||
return roomCount;
|
||
}
|
||
|
||
- (UIViewController*)viewControllerForClass:(Class)klass
|
||
{
|
||
UIViewController *foundViewController;
|
||
|
||
NSInteger viewControllerIndex = [self.viewControllers indexOfObjectPassingTest:^BOOL(__kindof UIViewController * _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
|
||
if ([obj isKindOfClass:klass])
|
||
{
|
||
*stop = YES;
|
||
return YES;
|
||
}
|
||
return NO;
|
||
}];
|
||
|
||
if (viewControllerIndex != NSNotFound)
|
||
{
|
||
foundViewController = self.viewControllers[viewControllerIndex];
|
||
}
|
||
|
||
return foundViewController;
|
||
}
|
||
|
||
- (void)filterRoomsWithParentId:(NSString*)roomParentId
|
||
inMatrixSession:(MXSession*)mxSession
|
||
{
|
||
if (roomParentId) {
|
||
NSString *parentName = [mxSession roomSummaryWithRoomId:roomParentId].displayname;
|
||
NSMutableArray<NSString *> *breadcrumbs = [[NSMutableArray alloc] initWithObjects:parentName, nil];
|
||
|
||
MXSpace *firstRootAncestor = roomParentId ? [mxSession.spaceService firstRootAncestorForRoomWithId:roomParentId] : nil;
|
||
NSString *rootName = nil;
|
||
if (firstRootAncestor)
|
||
{
|
||
rootName = [mxSession roomSummaryWithRoomId:firstRootAncestor.spaceId].displayname;
|
||
[breadcrumbs insertObject:rootName atIndex:0];
|
||
}
|
||
titleView.breadcrumbView.breadcrumbs = breadcrumbs;
|
||
}
|
||
else
|
||
{
|
||
titleView.breadcrumbView.breadcrumbs = @[];
|
||
}
|
||
|
||
recentsDataSource.currentSpace = [mxSession.spaceService getSpaceWithId:roomParentId];
|
||
[self updateSideMenuNotifcationIcon];
|
||
}
|
||
|
||
- (void)updateSideMenuNotifcationIcon
|
||
{
|
||
BOOL displayNotification = NO;
|
||
|
||
for (MXRoomSummary *summary in recentsDataSource.mxSession.spaceService.rootSpaceSummaries) {
|
||
if (summary.membership == MXMembershipInvite) {
|
||
displayNotification = YES;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!displayNotification) {
|
||
MXSpaceNotificationState *notificationState = [recentsDataSource.mxSession.spaceService.notificationCounter notificationStateForAllSpacesExcept: recentsDataSource.currentSpace.spaceId];
|
||
|
||
if (recentsDataSource.currentSpace)
|
||
{
|
||
MXSpaceNotificationState *homeNotificationState = recentsDataSource.mxSession.spaceService.notificationCounter.homeNotificationState;
|
||
displayNotification = notificationState.groupMissedDiscussionsCount > 0 || notificationState.groupMissedDiscussionsHighlightedCount > 0 || homeNotificationState.allCount > 0 || homeNotificationState.allHighlightCount > 0;
|
||
}
|
||
else
|
||
{
|
||
displayNotification = notificationState.groupMissedDiscussionsCount > 0 || notificationState.groupMissedDiscussionsHighlightedCount > 0;
|
||
}
|
||
}
|
||
|
||
[self.masterTabBarDelegate masterTabBarController:self needsSideMenuIconWithNotification:displayNotification];
|
||
}
|
||
|
||
#pragma mark -
|
||
|
||
-(void)setupTitleView
|
||
{
|
||
titleView = [MainTitleView new];
|
||
self.navigationItem.titleView = titleView;
|
||
}
|
||
|
||
-(void)setTitleLabelText:(NSString *)text
|
||
{
|
||
titleView.titleLabel.text = text;
|
||
self.navigationItem.backButtonTitle = text;
|
||
}
|
||
|
||
- (void)presentViewController:(UIViewController *)viewControllerToPresent animated:(BOOL)flag completion:(void (^)(void))completion
|
||
{
|
||
// Keep ref on presented view controller
|
||
[childViewControllers addObject:viewControllerToPresent];
|
||
|
||
[super presentViewController:viewControllerToPresent animated:flag completion:completion];
|
||
}
|
||
|
||
- (void)refreshSelectedControllerSelectedCellIfNeeded
|
||
{
|
||
if (self.splitViewController)
|
||
{
|
||
// Refresh selected cell without scrolling the selected cell (We suppose it's visible here)
|
||
[self refreshCurrentSelectedCell:NO];
|
||
}
|
||
}
|
||
|
||
// Made the actual selected view controller update its selected cell.
|
||
- (void)refreshCurrentSelectedCell:(BOOL)forceVisible
|
||
{
|
||
UIViewController *selectedViewController = self.selectedViewController;
|
||
|
||
if ([selectedViewController respondsToSelector:@selector(refreshCurrentSelectedCell:)])
|
||
{
|
||
[(id)selectedViewController refreshCurrentSelectedCell:forceVisible];
|
||
}
|
||
}
|
||
|
||
- (void)setHidden:(BOOL)hidden
|
||
{
|
||
_hidden = hidden;
|
||
|
||
[self.view superview].backgroundColor = ThemeService.shared.theme.backgroundColor;
|
||
self.view.hidden = hidden;
|
||
self.navigationController.navigationBar.hidden = hidden;
|
||
}
|
||
|
||
- (NSString*)getTitleForItemViewController:(UIViewController*)itemViewController
|
||
{
|
||
if ([itemViewController conformsToProtocol:@protocol(MasterTabBarItemDisplayProtocol)])
|
||
{
|
||
UIViewController<MasterTabBarItemDisplayProtocol> *masterTabBarItem = (UIViewController<MasterTabBarItemDisplayProtocol>*)itemViewController;
|
||
return masterTabBarItem.masterTabBarItemTitle;
|
||
}
|
||
|
||
return nil;
|
||
}
|
||
|
||
#pragma mark -
|
||
|
||
- (void)refreshTabBarBadges
|
||
{
|
||
// Use a middle dot to signal missed notif in favourites
|
||
if (RiotSettings.shared.homeScreenShowFavouritesTab)
|
||
{
|
||
[self setMissedDiscussionsMark:(recentsDataSource.favoriteMissedDiscussionsCount.numberOfNotified ? @"\u00B7": nil)
|
||
onTabBarItem:TABBAR_FAVOURITES_INDEX
|
||
withBadgeColor:(recentsDataSource.favoriteMissedDiscussionsCount.hasHighlight ? ThemeService.shared.theme.noticeColor : ThemeService.shared.theme.noticeSecondaryColor)];
|
||
}
|
||
|
||
// Update the badge on People and Rooms tabs
|
||
if (RiotSettings.shared.homeScreenShowPeopleTab)
|
||
{
|
||
if (recentsDataSource.directMissedDiscussionsCount.hasUnsent)
|
||
{
|
||
[self setBadgeValue:@"!"
|
||
onTabBarItem:TABBAR_PEOPLE_INDEX
|
||
withBadgeColor:ThemeService.shared.theme.noticeColor];
|
||
}
|
||
else
|
||
{
|
||
[self setMissedDiscussionsCount:recentsDataSource.directMissedDiscussionsCount.numberOfNotified
|
||
onTabBarItem:TABBAR_PEOPLE_INDEX
|
||
withBadgeColor:(recentsDataSource.directMissedDiscussionsCount.hasHighlight ? ThemeService.shared.theme.noticeColor : ThemeService.shared.theme.noticeSecondaryColor)];
|
||
}
|
||
}
|
||
|
||
if (RiotSettings.shared.homeScreenShowRoomsTab)
|
||
{
|
||
if (recentsDataSource.groupMissedDiscussionsCount.hasUnsent)
|
||
{
|
||
[self setMissedDiscussionsCount:recentsDataSource.groupMissedDiscussionsCount.numberOfUnsent
|
||
onTabBarItem:TABBAR_ROOMS_INDEX
|
||
withBadgeColor:ThemeService.shared.theme.noticeColor];
|
||
}
|
||
else
|
||
{
|
||
[self setMissedDiscussionsCount:recentsDataSource.groupMissedDiscussionsCount.numberOfNotified
|
||
onTabBarItem:TABBAR_ROOMS_INDEX
|
||
withBadgeColor:(recentsDataSource.groupMissedDiscussionsCount.hasHighlight ? ThemeService.shared.theme.noticeColor : ThemeService.shared.theme.noticeSecondaryColor)];
|
||
}
|
||
}
|
||
}
|
||
|
||
- (void)setMissedDiscussionsCount:(NSUInteger)count onTabBarItem:(NSUInteger)index withBadgeColor:(UIColor*)badgeColor
|
||
{
|
||
[self setBadgeValue:count ? [self tabBarBadgeStringValue:count] : nil onTabBarItem:index withBadgeColor:badgeColor];
|
||
}
|
||
|
||
- (void)setBadgeValue:(NSString *)value onTabBarItem:(NSUInteger)index withBadgeColor:(UIColor*)badgeColor
|
||
{
|
||
NSInteger itemIndex = [self indexOfTabItemWithTag:index];
|
||
if (itemIndex != NSNotFound)
|
||
{
|
||
if (value)
|
||
{
|
||
self.tabBar.items[itemIndex].badgeValue = value;
|
||
|
||
self.tabBar.items[itemIndex].badgeColor = badgeColor;
|
||
|
||
[self.tabBar.items[itemIndex] setBadgeTextAttributes:@{
|
||
NSForegroundColorAttributeName: ThemeService.shared.theme.baseTextPrimaryColor
|
||
}
|
||
forState:UIControlStateNormal];
|
||
}
|
||
else
|
||
{
|
||
self.tabBar.items[itemIndex].badgeValue = nil;
|
||
}
|
||
}
|
||
}
|
||
|
||
- (void)setMissedDiscussionsMark:(NSString*)mark onTabBarItem:(NSUInteger)index withBadgeColor:(UIColor*)badgeColor
|
||
{
|
||
NSInteger itemIndex = [self indexOfTabItemWithTag:index];
|
||
if (itemIndex != NSNotFound)
|
||
{
|
||
if (mark)
|
||
{
|
||
self.tabBar.items[itemIndex].badgeValue = mark;
|
||
|
||
self.tabBar.items[itemIndex].badgeColor = badgeColor;
|
||
|
||
[self.tabBar.items[itemIndex] setBadgeTextAttributes:@{
|
||
NSForegroundColorAttributeName: ThemeService.shared.theme.baseTextPrimaryColor
|
||
}
|
||
forState:UIControlStateNormal];
|
||
}
|
||
else
|
||
{
|
||
self.tabBar.items[itemIndex].badgeValue = nil;
|
||
}
|
||
}
|
||
}
|
||
|
||
- (NSString*)tabBarBadgeStringValue:(NSUInteger)count
|
||
{
|
||
NSString *badgeValue;
|
||
|
||
if (count > 1000)
|
||
{
|
||
CGFloat value = count / 1000.0;
|
||
badgeValue = [VectorL10n largeBadgeValueKFormat:value];
|
||
}
|
||
else
|
||
{
|
||
badgeValue = [NSString stringWithFormat:@"%tu", count];
|
||
}
|
||
|
||
return badgeValue;
|
||
}
|
||
|
||
- (NSInteger)indexOfTabItemWithTag:(NSUInteger)tag
|
||
{
|
||
for (int i = 0 ; i < self.tabBar.items.count ; i++)
|
||
{
|
||
if (self.tabBar.items[i].tag == tag)
|
||
{
|
||
return i;
|
||
}
|
||
}
|
||
|
||
return NSNotFound;
|
||
}
|
||
|
||
#pragma mark - Review session
|
||
|
||
- (void)presentVerifyCurrentSessionAlertIfNeededWithSession:(MXSession*)session
|
||
{
|
||
if (RiotSettings.shared.hideVerifyThisSessionAlert
|
||
|| self.reviewSessionAlertHasBeenDisplayed
|
||
|| self.isOnboardingInProgress)
|
||
{
|
||
return;
|
||
}
|
||
|
||
self.reviewSessionAlertHasBeenDisplayed = YES;
|
||
|
||
// Force verification if required by the HS configuration
|
||
if (session.vc_homeserverConfiguration.encryption.isSecureBackupRequired)
|
||
{
|
||
NSLog(@"[MasterTabBarController] presentVerifyCurrentSessionAlertIfNeededWithSession: Force verification of the device");
|
||
[[AppDelegate theDelegate] presentCompleteSecurityForSession:session];
|
||
return;
|
||
}
|
||
|
||
[self presentVerifyCurrentSessionAlertWithSession:session];
|
||
}
|
||
|
||
- (void)presentVerifyCurrentSessionAlertWithSession:(MXSession*)session
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] presentVerifyCurrentSessionAlertWithSession");
|
||
|
||
[currentAlert dismissViewControllerAnimated:NO completion:nil];
|
||
|
||
UIAlertController *alert = [UIAlertController alertControllerWithTitle:[VectorL10n keyVerificationSelfVerifyCurrentSessionAlertTitle]
|
||
message:[VectorL10n keyVerificationSelfVerifyCurrentSessionAlertMessage]
|
||
preferredStyle:UIAlertControllerStyleAlert];
|
||
|
||
[alert addAction:[UIAlertAction actionWithTitle:[VectorL10n keyVerificationSelfVerifyCurrentSessionAlertValidateAction]
|
||
style:UIAlertActionStyleDefault
|
||
handler:^(UIAlertAction * action) {
|
||
[[AppDelegate theDelegate] presentCompleteSecurityForSession:session];
|
||
}]];
|
||
|
||
[alert addAction:[UIAlertAction actionWithTitle:[VectorL10n later]
|
||
style:UIAlertActionStyleCancel
|
||
handler:nil]];
|
||
|
||
[alert addAction:[UIAlertAction actionWithTitle:[VectorL10n doNotAskAgain]
|
||
style:UIAlertActionStyleDestructive
|
||
handler:^(UIAlertAction * action) {
|
||
RiotSettings.shared.hideVerifyThisSessionAlert = YES;
|
||
}]];
|
||
|
||
|
||
[self presentViewController:alert animated:YES completion:nil];
|
||
|
||
currentAlert = alert;
|
||
}
|
||
|
||
- (void)presentReviewUnverifiedSessionsAlertIfNeededWithSession:(MXSession*)session
|
||
{
|
||
if (self.reviewSessionAlertHasBeenDisplayed)
|
||
{
|
||
return;
|
||
}
|
||
|
||
NSArray<MXDeviceInfo*> *devices = [session.crypto devicesForUser:session.myUserId].allValues;
|
||
|
||
BOOL isUserHasOneUnverifiedDevice = NO;
|
||
|
||
for (MXDeviceInfo *device in devices)
|
||
{
|
||
if (!device.trustLevel.isCrossSigningVerified)
|
||
{
|
||
isUserHasOneUnverifiedDevice = YES;
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (isUserHasOneUnverifiedDevice)
|
||
{
|
||
self.reviewSessionAlertHasBeenDisplayed = YES;
|
||
[self presentReviewUnverifiedSessionsAlertWithSession:session];
|
||
}
|
||
}
|
||
|
||
- (void)presentReviewUnverifiedSessionsAlertWithSession:(MXSession*)session
|
||
{
|
||
MXLogDebug(@"[MasterTabBarController] presentReviewUnverifiedSessionsAlertWithSession");
|
||
|
||
[currentAlert dismissViewControllerAnimated:NO completion:nil];
|
||
|
||
UIAlertController *alert = [UIAlertController alertControllerWithTitle:[VectorL10n keyVerificationAlertTitle]
|
||
message:[VectorL10n keyVerificationAlertBody]
|
||
preferredStyle:UIAlertControllerStyleAlert];
|
||
|
||
[alert addAction:[UIAlertAction actionWithTitle:[VectorL10n keyVerificationSelfVerifyUnverifiedSessionsAlertValidateAction]
|
||
style:UIAlertActionStyleDefault
|
||
handler:^(UIAlertAction * action) {
|
||
[self showSettingsSecurityScreenForSession:session];
|
||
}]];
|
||
|
||
[alert addAction:[UIAlertAction actionWithTitle:[VectorL10n later]
|
||
style:UIAlertActionStyleCancel
|
||
handler:nil]];
|
||
|
||
[self presentViewController:alert animated:YES completion:nil];
|
||
|
||
currentAlert = alert;
|
||
}
|
||
|
||
- (void)showSettingsSecurityScreenForSession:(MXSession*)session
|
||
{
|
||
SettingsViewController *settingsViewController = [SettingsViewController instantiate];
|
||
[settingsViewController loadViewIfNeeded];
|
||
SecurityViewController *securityViewController = [SecurityViewController instantiateWithMatrixSession:session];
|
||
|
||
[[AppDelegate theDelegate] restoreInitialDisplay:^{
|
||
self.navigationController.viewControllers = @[self, settingsViewController, securityViewController];
|
||
}];
|
||
}
|
||
|
||
- (void)resetReviewSessionsFlags
|
||
{
|
||
self.reviewSessionAlertHasBeenDisplayed = NO;
|
||
RiotSettings.shared.hideVerifyThisSessionAlert = NO;
|
||
}
|
||
|
||
#pragma mark - UITabBarDelegate
|
||
|
||
- (void)tabBar:(UITabBar *)tabBar didSelectItem:(UITabBarItem *)item
|
||
{
|
||
// Detect multi-tap on the current selected tab.
|
||
if (item.tag == self.selectedIndex)
|
||
{
|
||
// Scroll to the next room with missed notifications.
|
||
if (item.tag == TABBAR_ROOMS_INDEX)
|
||
{
|
||
[self.roomsViewController scrollToNextRoomWithMissedNotifications];
|
||
}
|
||
else if (item.tag == TABBAR_PEOPLE_INDEX)
|
||
{
|
||
[self.peopleViewController scrollToNextRoomWithMissedNotifications];
|
||
}
|
||
else if (item.tag == TABBAR_FAVOURITES_INDEX)
|
||
{
|
||
[self.favouritesViewController scrollToNextRoomWithMissedNotifications];
|
||
}
|
||
}
|
||
}
|
||
|
||
#pragma mark - UITabBarControllerDelegate
|
||
|
||
- (void)tabBarController:(UITabBarController *)tabBarController didSelectViewController:(UIViewController *)viewController
|
||
{
|
||
self.titleLabelText = [self getTitleForItemViewController:viewController];
|
||
}
|
||
|
||
@end
|