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

197 lines
6.4 KiB
Objective-C

/*
Copyright 2015 OpenMarket Ltd
Copyright 2017 Vector Creations 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 "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