element-ios/Riot/Categories/UIViewController+RiotSearch.m

189 lines
6.0 KiB
Objective-C

/*
Copyright 2024 New Vector Ltd.
Copyright 2017 Vector Creations Ltd
Copyright 2015 OpenMarket Ltd
SPDX-License-Identifier: AGPL-3.0-only
Please see LICENSE in the repository root for full details.
*/
#import "UIViewController+RiotSearch.h"
#import <objc/runtime.h>
#import "ThemeService.h"
#import "GeneratedInterface-Swift.h"
/**
`UIViewControllerRiotSearchInternals` is the internal single point storage for the search feature.
It hosts all required data so that only one associated object can be used in the category.
*/
@interface UIViewControllerRiotSearchInternals : NSObject
// The search bar
@property (nonatomic) UISearchBar *searchBar;
@property (nonatomic) BOOL searchBarHidden;
// Backup of view when displaying search
@property (nonatomic) UIView *backupTitleView;
@property (nonatomic) UIBarButtonItem *backupLeftBarButtonItem;
@property (nonatomic) UIBarButtonItem *backupRightBarButtonItem;
@end
@implementation UIViewControllerRiotSearchInternals
@end
#pragma mark - UIViewController+RiotSearch
#pragma mark -
@interface UIViewController ()
// The single associated object hosting all data.
@property(nonatomic) UIViewControllerRiotSearchInternals *searchInternals;
@end
@implementation UIViewController (RiotSearch)
- (UISearchBar *)searchBar
{
return self.searchInternals.searchBar;
}
- (BOOL)searchBarHidden
{
return self.searchInternals.searchBarHidden;
}
- (void)showSearch:(BOOL)animated
{
if (self.searchInternals.searchBarHidden)
{
// Backup screen header before displaying the search bar in it
self.searchInternals.backupTitleView = self.navigationItem.titleView;
self.searchInternals.backupLeftBarButtonItem = self.navigationItem.leftBarButtonItem;
self.searchInternals.backupRightBarButtonItem = self.navigationItem.rightBarButtonItem;
self.searchInternals.searchBarHidden = NO;
// Reset searches
self.searchBar.text = @"";
// Customize search bar
[ThemeService.shared.theme applyStyleOnSearchBar:self.searchBar];
// Remove navigation buttons
self.navigationItem.hidesBackButton = YES;
self.navigationItem.rightBarButtonItem = nil;
self.navigationItem.leftBarButtonItem = nil;
// Add the search bar
UIView *searchBarContainer = [[UIView alloc] initWithFrame:CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, 44)];
searchBarContainer.backgroundColor = [UIColor clearColor];
searchBarContainer.autoresizingMask = UIViewAutoresizingFlexibleWidth;
self.navigationItem.titleView = searchBarContainer;
[searchBarContainer addSubview:self.searchBar];
self.extendedLayoutIncludesOpaqueBars = YES;
// On iPad, there is no cancel button inside the UISearchBar
// So, add a classic cancel right bar button
if ([[UIDevice currentDevice] userInterfaceIdiom] == UIUserInterfaceIdiomPad)
{
UIBarButtonItem *cancelButton = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemCancel target:self action:@selector(onIPadCancelPressed:)];
[self.navigationItem setRightBarButtonItem: cancelButton animated:YES];
}
}
// And display the keyboard
[self.searchBar becomeFirstResponder];
}
- (void)hideSearch:(BOOL)animated
{
if (!self.searchInternals.searchBarHidden)
{
// Restore the screen header
self.navigationItem.hidesBackButton = NO;
self.navigationItem.titleView = self.searchInternals.backupTitleView;
self.navigationItem.leftBarButtonItem = self.searchInternals.backupLeftBarButtonItem;
self.navigationItem.rightBarButtonItem = self.searchInternals.backupRightBarButtonItem;
self.searchInternals.searchBarHidden = YES;
}
}
#pragma mark - UISearchBarDelegate
- (void)searchBarSearchButtonClicked:(UISearchBar *)searchBar2
{
// "Search" key has been pressed
[self.searchBar resignFirstResponder];
}
- (void)searchBarCancelButtonClicked:(UISearchBar *)searchBar2
{
[self hideSearch:YES];
}
- (BOOL)searchBarShouldEndEditing:(UISearchBar *)searchBar2
{
// Keep the search bar cancel button enabled even if the keyboard is not displayed
dispatch_async(dispatch_get_main_queue(), ^{
for (UIView *subView in self.searchBar.subviews)
{
for (UIView *view in subView.subviews)
{
if ([view isKindOfClass:[UIButton class]])
{
[(UIButton *)view setEnabled:YES];
}
}
}
});
return YES;
}
#pragma mark - Private methods
- (void)onIPadCancelPressed:(id)sender
{
// Mimic the same behavior as on iPhones and call the UISearchBar cancel delegate method
if ([self.searchBar.delegate respondsToSelector:@selector(searchBarCancelButtonClicked:)])
{
[self.searchBar.delegate searchBarCancelButtonClicked:self.searchBar];
}
}
#pragma mark - Internal associated object
- (void)setSearchInternals:(UIViewControllerRiotSearchInternals *)searchInternals
{
objc_setAssociatedObject(self, @selector(searchInternals), searchInternals, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
- (UIViewControllerRiotSearchInternals *)searchInternals
{
UIViewControllerRiotSearchInternals *searchInternals = objc_getAssociatedObject(self, @selector(searchInternals));
if (!searchInternals)
{
// Initialise internal data at the first call
searchInternals = [[UIViewControllerRiotSearchInternals alloc] init];
UISearchBar *searchBar = [[UISearchBar alloc] initWithFrame:CGRectMake(0, 0, [UIScreen mainScreen].bounds.size.width, 44)];
searchBar.showsCancelButton = YES;
searchBar.autoresizingMask = UIViewAutoresizingFlexibleWidth;
searchBar.delegate = (id<UISearchBarDelegate>)self;
searchInternals.searchBar = searchBar;
self.searchInternals = searchInternals;
self.searchInternals.searchBarHidden = YES;
}
return searchInternals;
}
@end