blob: 2adb164ce3b49dd7783cf7fd4b105876b7304334 [file] [log] [blame]
// 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 <QuartzCore/QuartzCore.h>
#import <UIKit/UIKit.h>
#import "MDCActionSheetAction.h"
#import "MDCActionSheetController.h"
#import "MDCActionSheetController+MaterialTheming.h"
#import "MDCButton.h"
#import "MDCButton+MaterialTheming.h"
#import "MDCTabBarItem.h"
#import "MDCTabBarView.h"
#import "MDCTabBarViewCustomViewable.h"
#import "MDCTabBarViewDelegate.h"
#import "MaterialIcons+ic_check.h" // PrivateSubtargetImport
#import "MaterialIcons+ic_settings.h" // PrivateSubtargetImport
#import "MDCMath.h"
#import "MDCSemanticColorScheme.h"
#import "MDCContainerScheme.h"
#import "MDCTypographyScheming.h"
static NSString *const kExampleTitle = @"TabBarView";
/** Accessibility label for the content insets toggle button. */
static NSString *const kToggleContentInsetsAccessibilityLabel = @"Toggle content insets";
/** Accessibility label for the preferred layout menu button. */
static NSString *const kPreferredLayoutMenuAccessibilityLabel = @"Change preferred alignment";
/** A custom view to place in an MDCTabBarView. */
@interface MDCTabBarViewTypicalExampleViewControllerCustomView
: UIView <MDCTabBarViewCustomViewable>
/** A switch shown in the view. */
@property(nonatomic, strong) UISwitch *aSwitch;
/** Duration for animating changes to the tab bar view. */
@property(nonatomic, assign) CFTimeInterval animationDuration;
/** The timing function for animating changes to the tab bar view. */
@property(nonatomic, strong) CAMediaTimingFunction *animationTimingFunction;
@end
@implementation MDCTabBarViewTypicalExampleViewControllerCustomView
- (instancetype)init {
self = [super init];
if (self) {
_aSwitch = [[UISwitch alloc] init];
_animationTimingFunction =
[CAMediaTimingFunction functionWithName:kCAMediaTimingFunctionEaseInEaseOut];
}
return self;
}
- (CGRect)contentFrame {
return CGRectStandardize(self.aSwitch.frame);
}
- (void)setSelected:(BOOL)selected animated:(BOOL)animated {
// This is where a real custom view would handle its selection state change.
}
- (void)layoutSubviews {
[super layoutSubviews];
if (self.aSwitch.superview != self) {
[self addSubview:_aSwitch];
[self.aSwitch addTarget:self
action:@selector(switchTapped:)
forControlEvents:UIControlEventValueChanged];
}
self.aSwitch.center = CGPointMake(CGRectGetMidX(self.bounds), CGRectGetMidY(self.bounds));
}
- (void)switchTapped:(id)sender {
[self invalidateIntrinsicContentSize];
[self setNeedsLayout];
[CATransaction begin];
[CATransaction setAnimationTimingFunction:self.animationTimingFunction];
[UIView animateWithDuration:self.animationDuration
delay:0
options:0
animations:^{
[self.superview setNeedsLayout];
[self.superview layoutIfNeeded];
}
completion:nil];
[CATransaction commit];
}
- (CGSize)intrinsicContentSize {
if (self.aSwitch.isOn) {
return CGSizeMake(self.aSwitch.intrinsicContentSize.width * 2,
self.aSwitch.intrinsicContentSize.height * 2);
}
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, UIScrollViewDelegate>
/** 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;
/** Tracks the UITabBarItem views that are currently on-screen. */
@property(nonatomic, copy) NSSet<UITabBarItem *> *visibleItems;
/** Image for toggle button when contentInset is non-zero. */
@property(nonatomic, strong) UIImage *contentInsetToggleEnabledImage;
/** Image for toggle button when contentInset is zero. */
@property(nonatomic, strong) UIImage *contentInsetToggleDisabledImage;
/** Tapping this button goes to the next tab. */
@property(nonatomic, strong) MDCButton *forwardButton;
/** Tapping this button goes to the previous tab. */
@property(nonatomic, strong) MDCButton *backwardButton;
/** Segmented control. */
@property(nonatomic, strong) UISegmentedControl *segmentedControl;
@end
@implementation MDCTabBarViewTypicalExampleViewController
- (void)viewDidLoad {
[super viewDidLoad];
self.title = kExampleTitle;
NSBundle *selfBundle = [NSBundle bundleForClass:[self class]];
self.contentInsetToggleEnabledImage = [[UIImage imageNamed:@"contentInset_enabled"
inBundle:selfBundle
compatibleWithTraitCollection:nil]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
self.contentInsetToggleDisabledImage = [[UIImage imageNamed:@"contentInset_disabled"
inBundle:selfBundle
compatibleWithTraitCollection:nil]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
[self applyFixForInjectedAppBar];
if (!self.containerScheme) {
self.containerScheme = [[MDCContainerScheme alloc] init];
}
self.view.backgroundColor = self.containerScheme.colorScheme.backgroundColor;
self.tabBar = [[MDCTabBarView alloc] init];
self.tabBar.tabBarDelegate = self;
self.tabBar.delegate = self;
[self.view addSubview:self.tabBar];
NSMutableArray<UIImage *> *itemIcons = [NSMutableArray array];
[itemIcons addObject:[[UIImage imageNamed:@"system_icons/home"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"system_icons/favorite"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"system_icons/cake"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"system_icons/email"]
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]];
[itemIcons addObject:[[UIImage imageNamed:@"system_icons/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;
switchView.animationDuration = self.tabBar.selectionChangeAnimationDuration;
switchView.animationTimingFunction = self.tabBar.selectionChangeAnimationTimingFunction;
self.tabBar.items = @[ item1, item2, item6, item4, item5, item3 ];
self.tabBar.selectedItem = item4;
self.tabBar.translatesAutoresizingMaskIntoConstraints = NO;
[self.view.layoutMarginsGuide.topAnchor 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];
[self addButtons];
UIBarButtonItem *alignmentButton = [[UIBarButtonItem alloc]
initWithImage:[MDCIcons.imageFor_ic_settings
imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate]
style:UIBarButtonItemStylePlain
target:self
action:@selector(didTapAlignmentButton)];
alignmentButton.accessibilityLabel = kPreferredLayoutMenuAccessibilityLabel;
UIBarButtonItem *insetsButton =
[[UIBarButtonItem alloc] initWithImage:self.contentInsetToggleDisabledImage
style:UIBarButtonItemStylePlain
target:self
action:@selector(didToggleInsets:)];
insetsButton.accessibilityLabel = kToggleContentInsetsAccessibilityLabel;
self.navigationItem.rightBarButtonItems = @[ alignmentButton, insetsButton ];
}
- (void)applyThemingToTabBarView {
self.tabBar.barTintColor = self.containerScheme.colorScheme.surfaceColor;
[self.tabBar setTitleColor:[self.containerScheme.colorScheme.onSurfaceColor
colorWithAlphaComponent:(CGFloat)0.6]
forState:UIControlStateNormal];
[self.tabBar setTitleColor:self.containerScheme.colorScheme.primaryColor
forState:UIControlStateSelected];
[self.tabBar setImageTintColor:[self.containerScheme.colorScheme.onSurfaceColor
colorWithAlphaComponent:(CGFloat)0.6]
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.primaryColor;
self.tabBar.rippleColor =
[self.containerScheme.colorScheme.primaryColor colorWithAlphaComponent:(CGFloat)0.1];
self.tabBar.bottomDividerColor =
[self.containerScheme.colorScheme.onSurfaceColor colorWithAlphaComponent:(CGFloat)0.12];
}
- (void)addButtons {
self.forwardButton = [[MDCButton alloc] init];
[self.forwardButton setTitle:@"Next tab" forState:UIControlStateNormal];
[self.forwardButton addTarget:self
action:@selector(forwardButtonTapped:)
forControlEvents:UIControlEventTouchUpInside];
[self.forwardButton applyTextThemeWithScheme:self.containerScheme];
[self.forwardButton sizeToFit];
[self.view addSubview:self.forwardButton];
self.backwardButton = [[MDCButton alloc] init];
[self.backwardButton setTitle:@"Previous tab" forState:UIControlStateNormal];
[self.backwardButton addTarget:self
action:@selector(backwardButtonTapped:)
forControlEvents:UIControlEventTouchUpInside];
[self.backwardButton applyTextThemeWithScheme:self.containerScheme];
[self.backwardButton sizeToFit];
[self.view addSubview:self.backwardButton];
}
- (void)backwardButtonTapped:(id)sender {
NSInteger index = [self.tabBar.items indexOfObject:self.tabBar.selectedItem];
if (index == NSNotFound) {
return;
}
index--;
if (index < 0) {
return;
}
self.tabBar.selectedItem = self.tabBar.items[index];
}
- (void)forwardButtonTapped:(id)sender {
NSInteger index = [self.tabBar.items indexOfObject:self.tabBar.selectedItem];
if (index == NSNotFound) {
return;
}
index++;
if (index >= (NSInteger)self.tabBar.items.count) {
return;
}
self.tabBar.selectedItem = self.tabBar.items[index];
}
- (void)addSegmentedControl {
self.segmentedControl =
[[UISegmentedControl alloc] initWithItems:@[ @"Titles", @"Icons", @"Titles and Icons" ]];
self.segmentedControl.selectedSegmentIndex = 2;
[self.segmentedControl addTarget:self
action:@selector(segmentedControlChangedValue:)
forControlEvents:UIControlEventValueChanged];
[self.view addSubview:self.segmentedControl];
self.segmentedControl.tintColor = self.containerScheme.colorScheme.primaryColor;
self.segmentedControl.translatesAutoresizingMaskIntoConstraints = NO;
[self.view.layoutMarginsGuide.centerXAnchor
constraintEqualToAnchor:self.segmentedControl.centerXAnchor]
.active = YES;
[self.view.layoutMarginsGuide.centerYAnchor
constraintEqualToAnchor:self.segmentedControl.centerYAnchor]
.active = YES;
[self.view.layoutMarginsGuide.leadingAnchor
constraintLessThanOrEqualToAnchor:self.segmentedControl.leadingAnchor]
.active = YES;
[self.view.layoutMarginsGuide.trailingAnchor
constraintGreaterThanOrEqualToAnchor:self.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)didTapAlignmentButton {
MDCTabBarViewLayoutStyle currentStyle = self.tabBar.preferredLayoutStyle;
UIImage *checkIcon =
[MDCIcons.imageFor_ic_check imageWithRenderingMode:UIImageRenderingModeAlwaysTemplate];
MDCActionSheetController *actionSheet =
[MDCActionSheetController actionSheetControllerWithTitle:@"Preferred Layout Style"];
MDCActionSheetAction *fixedJustifiedAction = [MDCActionSheetAction
actionWithTitle:@"Fixed"
image:((currentStyle == MDCTabBarViewLayoutStyleFixed) ? checkIcon : nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleFixed;
}];
MDCActionSheetAction *fixedClusteredLeadingAction = [MDCActionSheetAction
actionWithTitle:@"Fixed Clustered Leading"
image:((currentStyle == MDCTabBarViewLayoutStyleFixedClusteredLeading) ? checkIcon
: nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleFixedClusteredLeading;
}];
MDCActionSheetAction *fixedClusteredTrailingAction = [MDCActionSheetAction
actionWithTitle:@"Fixed Clustered Trailing"
image:((currentStyle == MDCTabBarViewLayoutStyleFixedClusteredTrailing) ? checkIcon
: nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleFixedClusteredTrailing;
}];
MDCActionSheetAction *fixedClusteredCenteredAction = [MDCActionSheetAction
actionWithTitle:@"Fixed Clustered Centered"
image:((currentStyle == MDCTabBarViewLayoutStyleFixedClusteredCentered) ? checkIcon
: nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleFixedClusteredCentered;
}];
MDCActionSheetAction *scrollableAction = [MDCActionSheetAction
actionWithTitle:@"Scrollable"
image:((currentStyle == MDCTabBarViewLayoutStyleScrollable) ? checkIcon : nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleScrollable;
}];
MDCActionSheetAction *scrollableCenteredAction = [MDCActionSheetAction
actionWithTitle:@"Scrollable Centered"
image:((currentStyle == MDCTabBarViewLayoutStyleScrollableCentered) ? checkIcon
: nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle = MDCTabBarViewLayoutStyleScrollableCentered;
}];
MDCActionSheetAction *nonFixedClusteredCenteredAction = [MDCActionSheetAction
actionWithTitle:@"Non-Fixed Clustered Centered"
image:((currentStyle == MDCTabBarViewLayoutStyleNonFixedClusteredCentered)
? checkIcon
: nil)
handler:^(MDCActionSheetAction *_Nonnull action) {
self.tabBar.preferredLayoutStyle =
MDCTabBarViewLayoutStyleNonFixedClusteredCentered;
}];
[actionSheet addAction:fixedJustifiedAction];
[actionSheet addAction:fixedClusteredLeadingAction];
[actionSheet addAction:fixedClusteredTrailingAction];
[actionSheet addAction:fixedClusteredCenteredAction];
[actionSheet addAction:scrollableAction];
[actionSheet addAction:scrollableCenteredAction];
[actionSheet addAction:nonFixedClusteredCenteredAction];
[actionSheet applyThemeWithScheme:self.containerScheme];
actionSheet.alwaysAlignTitleLeadingEdges = YES;
[self presentViewController:actionSheet animated:YES completion:nil];
}
- (void)didToggleInsets:(UIBarButtonItem *)sender {
if (UIEdgeInsetsEqualToEdgeInsets(self.tabBar.contentInset, UIEdgeInsetsZero)) {
self.tabBar.contentInset = UIEdgeInsetsMake(0, 10, 0, 30);
sender.image = self.contentInsetToggleEnabledImage;
} else {
self.tabBar.contentInset = UIEdgeInsetsZero;
sender.image = self.contentInsetToggleDisabledImage;
}
}
- (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 {
NSMutableArray<UITabBarItem *> *newItems = [NSMutableArray array];
NSUInteger selectedIndex = self.tabBar.selectedItem
? [self.tabBar.items indexOfObject:self.tabBar.selectedItem]
: NSNotFound;
for (NSUInteger index = 0; index < self.tabBar.items.count; ++index) {
UITabBarItem *originalItem = self.tabBar.items[index];
if ([originalItem isKindOfClass:[MDCTabBarItem class]]) {
MDCTabBarItem *originalCustomItem = (MDCTabBarItem *)originalItem;
MDCTabBarItem *newCustomItem = [[MDCTabBarItem alloc] initWithTitle:nil
image:nil
tag:originalItem.tag];
newCustomItem.mdc_customView = originalCustomItem.mdc_customView;
[newItems addObject:newCustomItem];
continue;
}
UITabBarItem *newItem = [[UITabBarItem alloc] initWithTitle:nil image:nil tag:originalItem.tag];
newItem.title = self.tabBarItemTitles[index % self.tabBarItemTitles.count];
[newItems addObject:newItem];
}
self.tabBar.items = newItems;
if (selectedIndex != NSNotFound) {
self.tabBar.selectedItem = self.tabBar.items[selectedIndex];
}
}
- (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];
}
}
#pragma mark - UIScrollViewDelegate
- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
[self logItemVisibilityChanges];
}
#pragma mark - UIViewController
- (void)viewWillTransitionToSize:(CGSize)size
withTransitionCoordinator:(id<UIViewControllerTransitionCoordinator>)coordinator {
[super viewWillTransitionToSize:size withTransitionCoordinator:coordinator];
[coordinator
animateAlongsideTransition:nil
completion:^(
id<UIViewControllerTransitionCoordinatorContext> _Nonnull context) {
[self logItemVisibilityChanges];
}];
}
- (void)viewDidLayoutSubviews {
[super viewDidLayoutSubviews];
[self logItemVisibilityChanges];
CGFloat centerX = self.segmentedControl.center.x;
CGFloat centerY = CGRectGetMaxY(self.segmentedControl.frame) + 50;
self.forwardButton.center = CGPointMake(centerX, centerY);
centerY = centerY + 50;
self.backwardButton.center = CGPointMake(centerX, centerY);
}
- (void)logItemVisibilityChanges {
NSMutableSet<UITabBarItem *> *allVisibleItems = [NSMutableSet set];
NSMutableSet<UITabBarItem *> *itemsThatEnteredTheWindowBounds = [NSMutableSet set];
NSMutableSet<UITabBarItem *> *itemsThatLeftTheWindowBounds = [NSMutableSet set];
for (UITabBarItem *item in self.tabBar.items) {
CGRect itemViewInWindow = [self.tabBar rectForItem:item inCoordinateSpace:self.view.window];
CGRect overlapRect = CGRectIntersection(self.view.window.bounds, itemViewInWindow);
// Views that don't intersect (or only at the very edge) the window's bounds
if (CGRectIsNull(overlapRect) || MDCCGFloatEqual(CGRectGetWidth(itemViewInWindow), 0)) {
if ([self.visibleItems containsObject:item]) {
[itemsThatLeftTheWindowBounds addObject:item];
}
continue;
}
[allVisibleItems addObject:item];
if (![self.visibleItems containsObject:item]) {
[itemsThatEnteredTheWindowBounds addObject:item];
}
}
self.visibleItems = allVisibleItems;
if (itemsThatEnteredTheWindowBounds.count) {
for (UITabBarItem *item in itemsThatEnteredTheWindowBounds) {
NSLog(@"(%@) became visible.", item.title ?: @(item.tag));
}
}
if (itemsThatLeftTheWindowBounds.count) {
for (UITabBarItem *item in itemsThatLeftTheWindowBounds) {
NSLog(@"(%@) is no longer visible.", item.title ?: @(item.tag));
}
}
}
@end
#pragma mark - CatalogByConvention
@implementation MDCTabBarViewTypicalExampleViewController (CatalogByConvention)
+ (NSDictionary *)catalogMetadata {
return @{
@"breadcrumbs" : @[ @"Tab Bar", kExampleTitle ],
@"primaryDemo" : @NO,
@"presentable" : @NO,
};
}
@end