material-components_materia.../components/Tabs/examples/MDCTabBarViewTypicalExampleViewController.m
Robert Moore 4cb495c435
[Tabs] Add item style options for TabBarView example. (#7808)
Adds the option to display text-only, image-only, or text-and-image Tabs in the TabBarView example view controller.

Part of #7748
2019-07-08 15:29:43 -04:00

275 lines
11 KiB
Objective-C

// Copyright 2019-present the Material Components for iOS authors. All Rights Reserved.
//
// 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 <UIKit/UIKit.h>
#import <MaterialComponents/MaterialContainerScheme.h>
#import "MaterialTabs+TabBarView.h"
static NSString *const kExampleTitle = @"TabBarView";
/** A custom view to place in an MDCTabBarView. */
@interface MDCTabBarViewTypicalExampleViewControllerCustomView
: UIView <MDCTabBarViewIndicatorSupporting>
/** A switch shown in the view. */
@property(nonatomic, strong) UISwitch *aSwitch;
@end
@implementation MDCTabBarViewTypicalExampleViewControllerCustomView
- (CGRect)contentFrame {
return CGRectStandardize(self.aSwitch.frame);
}
- (UISwitch *)aSwitch {
if (!_aSwitch) {
_aSwitch = [[UISwitch alloc] init];
[self addSubview:_aSwitch];
}
return _aSwitch;
}
- (void)layoutSubviews {
[super layoutSubviews];
self.aSwitch.center = CGPointMake(CGRectGetMidX(self.bounds), CGRectGetMidY(self.bounds));
}
- (CGSize)intrinsicContentSize {
return self.aSwitch.intrinsicContentSize;
}
- (CGSize)sizeThatFits:(CGSize)size {
return [self.aSwitch sizeThatFits:size];
}
@end
/**
Typical use example showing how to place an @c MDCTabBarView within another view.
*/
@interface MDCTabBarViewTypicalExampleViewController : UIViewController <MDCTabBarViewDelegate>
/** The tab bar for this example. */
@property(nonatomic, strong) MDCTabBarView *tabBar;
/** The container scheme injected into this example. */
@property(nonatomic, strong) id<MDCContainerScheming> containerScheme;
/** Titles for the items. */
@property(nonatomic, copy) NSArray<NSString *> *tabBarItemTitles;
/** Images for the items. */
@property(nonatomic, copy) NSArray<UIImage *> *tabBarItemIcons;
@end
@implementation MDCTabBarViewTypicalExampleViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.title = kExampleTitle;
[self applyFixForInjectedAppBar];
if (!self.containerScheme) {
self.containerScheme = [[MDCContainerScheme alloc] init];
}
self.view.backgroundColor = self.containerScheme.colorScheme.backgroundColor;
NSMutableArray<UIImage *> *itemIcons = [NSMutableArray array];
[itemIcons addObject:[[UIImage imageNamed:@"Home"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"Favorite"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"Cake"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"Email"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"Search"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
self.tabBarItemIcons = itemIcons;
self.tabBarItemTitles = @[ @"Home", @"Unselectable", @"Cake", @"Email", @"Search" ];
UITabBarItem *item1 = [[UITabBarItem alloc] initWithTitle:self.tabBarItemTitles[0]
image:itemIcons[0]
tag:0];
UITabBarItem *item2 = [[UITabBarItem alloc] initWithTitle:self.tabBarItemTitles[1]
image:itemIcons[1]
tag:1];
item2.accessibilityTraits = UIAccessibilityTraitStaticText;
UITabBarItem *item3 = [[UITabBarItem alloc] initWithTitle:self.tabBarItemTitles[2]
image:itemIcons[2]
tag:2];
UITabBarItem *item4 = [[UITabBarItem alloc] initWithTitle:self.tabBarItemTitles[3]
image:itemIcons[3]
tag:3];
UITabBarItem *item5 = [[UITabBarItem alloc] initWithTitle:self.tabBarItemTitles[4]
image:itemIcons[4]
tag:4];
MDCTabBarItem *item6 = [[MDCTabBarItem alloc] initWithTitle:@"A switch" image:nil tag:5];
MDCTabBarViewTypicalExampleViewControllerCustomView *switchView =
[[MDCTabBarViewTypicalExampleViewControllerCustomView alloc] init];
item6.mdc_customView = switchView;
switchView.aSwitch.onTintColor = self.containerScheme.colorScheme.primaryColor;
self.tabBar = [[MDCTabBarView alloc] init];
self.tabBar.tabBarDelegate = self;
self.tabBar.items = @[ item1, item2, item3, item4, item5, item6 ];
self.tabBar.selectedItem = item4;
self.tabBar.translatesAutoresizingMaskIntoConstraints = NO;
[self.view addSubview:self.tabBar];
if (@available(iOS 11.0, *)) {
[self.view.layoutMarginsGuide.topAnchor constraintEqualToAnchor:self.tabBar.topAnchor].active =
YES;
} else {
[self.topLayoutGuide.bottomAnchor constraintEqualToAnchor:self.tabBar.topAnchor].active = YES;
}
[self.view.leftAnchor constraintEqualToAnchor:self.tabBar.leftAnchor].active = YES;
[self.view.rightAnchor constraintEqualToAnchor:self.tabBar.rightAnchor].active = YES;
[self applyThemingToTabBarView];
[self addSegmentedControl];
}
- (void)applyThemingToTabBarView {
self.tabBar.barTintColor = self.containerScheme.colorScheme.secondaryColor;
[self.tabBar setTitleColor:self.containerScheme.colorScheme.onSecondaryColor
forState:UIControlStateNormal];
[self.tabBar setTitleColor:self.containerScheme.colorScheme.primaryColor
forState:UIControlStateSelected];
[self.tabBar setImageTintColor:self.containerScheme.colorScheme.onSecondaryColor
forState:UIControlStateNormal];
[self.tabBar setImageTintColor:self.containerScheme.colorScheme.primaryColor
forState:UIControlStateSelected];
[self.tabBar setTitleFont:self.containerScheme.typographyScheme.button
forState:UIControlStateNormal];
[self.tabBar setTitleFont:[UIFont systemFontOfSize:16] forState:UIControlStateSelected];
self.tabBar.selectionIndicatorStrokeColor = self.containerScheme.colorScheme.onSecondaryColor;
self.tabBar.rippleColor =
[self.containerScheme.colorScheme.primaryColor colorWithAlphaComponent:(CGFloat)0.1];
}
- (void)addSegmentedControl {
UISegmentedControl *segmentedControl =
[[UISegmentedControl alloc] initWithItems:@[ @"Titles", @"Icons", @"Titles and Icons" ]];
segmentedControl.selectedSegmentIndex = 2;
[segmentedControl addTarget:self
action:@selector(segmentedControlChangedValue:)
forControlEvents:UIControlEventValueChanged];
[self.view addSubview:segmentedControl];
segmentedControl.translatesAutoresizingMaskIntoConstraints = NO;
if (@available(iOS 11.0, *)) {
[self.view.layoutMarginsGuide.centerXAnchor
constraintEqualToAnchor:segmentedControl.centerXAnchor]
.active = YES;
[self.view.layoutMarginsGuide.centerYAnchor
constraintEqualToAnchor:segmentedControl.centerYAnchor]
.active = YES;
[self.view.layoutMarginsGuide.leadingAnchor
constraintLessThanOrEqualToAnchor:segmentedControl.leadingAnchor]
.active = YES;
[self.view.layoutMarginsGuide.trailingAnchor
constraintGreaterThanOrEqualToAnchor:segmentedControl.trailingAnchor]
.active = YES;
} else {
[self.view.centerXAnchor constraintEqualToAnchor:segmentedControl.centerXAnchor].active = YES;
[self.view.centerYAnchor constraintEqualToAnchor:segmentedControl.centerYAnchor].active = YES;
[self.view.leadingAnchor constraintLessThanOrEqualToAnchor:segmentedControl.leadingAnchor]
.active = YES;
[self.view.trailingAnchor constraintGreaterThanOrEqualToAnchor:segmentedControl.trailingAnchor]
.active = YES;
}
}
#pragma mark - MDCTabBarViewDelegate
- (BOOL)tabBarView:(MDCTabBarView *)tabBarView shouldSelectItem:(nonnull UITabBarItem *)item {
// Just to demonstrate preventing selection of an item.
return [self.tabBar.items indexOfObject:item] != 1;
}
- (void)tabBarView:(MDCTabBarView *)tabBarView didSelectItem:(nonnull UITabBarItem *)item {
NSLog(@"Item (%@) was selected.", item.title);
}
#pragma mark - Errata
- (void)applyFixForInjectedAppBar {
// The injected AppBar has a bug where it will attempt to manipulate the Tab bar. To prevent
// that bug, we need to inject a scroll view into the view hierarchy before the tab bar. The App
// Bar will manipulate with that one instead.
UIScrollView *bugFixScrollView = [[UIScrollView alloc] init];
bugFixScrollView.userInteractionEnabled = NO;
bugFixScrollView.hidden = YES;
[self.view addSubview:bugFixScrollView];
}
#pragma mark - Item style variations
- (void)segmentedControlChangedValue:(id)sender {
if ([sender isKindOfClass:[UISegmentedControl class]]) {
UISegmentedControl *segmentedControl = (UISegmentedControl *)sender;
if (segmentedControl.selectedSegmentIndex == 0) {
[self changeItemsToTextOnly];
} else if (segmentedControl.selectedSegmentIndex == 1) {
[self changeItemsToImageOnly];
} else {
[self changeItemsToTextAndImage];
}
}
}
- (void)changeItemsToTextOnly {
for (NSUInteger index = 0; index < self.tabBar.items.count; ++index) {
UITabBarItem *item = self.tabBar.items[index];
item.image = nil;
item.selectedImage = nil;
item.title = self.tabBarItemTitles[index % self.tabBarItemTitles.count];
}
}
- (void)changeItemsToImageOnly {
for (NSUInteger index = 0; index < self.tabBar.items.count; ++index) {
UITabBarItem *item = self.tabBar.items[index];
item.image = self.tabBarItemIcons[index % self.tabBarItemIcons.count];
item.title = nil;
}
}
- (void)changeItemsToTextAndImage {
for (NSUInteger index = 0; index < self.tabBar.items.count; ++index) {
UITabBarItem *item = self.tabBar.items[index];
item.image = self.tabBarItemIcons[index % self.tabBarItemIcons.count];
item.title = self.tabBarItemTitles[index % self.tabBarItemTitles.count];
}
}
@end
#pragma mark - CatalogByConvention
@implementation MDCTabBarViewTypicalExampleViewController (CatalogByConvention)
+ (NSDictionary *)catalogMetadata {
return @{
@"breadcrumbs" : @[ @"Tab Bar", kExampleTitle ],
@"primaryDemo" : @NO,
@"presentable" : @NO,
};
}
@end