diff --git a/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuView.mm b/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuView.mm index 0c3c76c309..77f7a415b9 100644 --- a/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuView.mm +++ b/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuView.mm @@ -277,9 +277,13 @@ CGFloat constexpr kTimeWidthRegular = 128; { case MWMBottomMenuStateHidden: self.minY = self.superview.height; return; case MWMBottomMenuStateInactive: - case MWMBottomMenuStateCompact: case MWMBottomMenuStatePlanning: case MWMBottomMenuStateGo: break; + case MWMBottomMenuStateCompact: + if (self.restoreState == MWMBottomMenuStateRouting && !IPAD && + UIDeviceOrientationIsLandscape([UIDevice currentDevice].orientation)) + self.mainButtonsHeight.constant = kRoutingMainButtonsHeightLandscape; + break; case MWMBottomMenuStateActive: { self.separatorHeight.constant = 1.0; diff --git a/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuViewController.xib b/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuViewController.xib index 3af84d382e..fa023f6497 100644 --- a/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuViewController.xib +++ b/iphone/Maps/Classes/CustomViews/MapViewControls/BottomMenu/MWMBottomMenuViewController.xib @@ -392,7 +392,7 @@ - + diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.h b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.h index 5f089a448e..ea549bde14 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.h +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.h @@ -35,6 +35,7 @@ typedef NS_ENUM(NSUInteger, MWMNavigationDashboardState) { @property(weak, nonatomic, readonly) id delegate; @property(nonatomic) CGFloat topBound; @property(nonatomic) CGFloat leftBound; +@property(nonatomic, readonly) CGFloat extraCompassBottomOffset; @property(nonatomic, readonly) CGFloat height; - (instancetype)init __attribute__((unavailable("init is not available"))); diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.mm b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.mm index dc8b7d8677..3cdf1653bb 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.mm +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationDashboardManager.mm @@ -238,4 +238,11 @@ using TInfoDisplays = NSHashTable<__kindof TInfoDisplay>; return _entity; } +- (CGFloat)extraCompassBottomOffset +{ + if (!_navigationInfoView) + return 0; + return self.navigationInfoView.extraCompassBottomOffset; +} + @end diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.h b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.h index e71b4c7734..1eeda5ea64 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.h +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.h @@ -3,6 +3,7 @@ @interface MWMNavigationInfoView : UIView @property(nonatomic) CGFloat leftBound; +@property(nonatomic, readonly) CGFloat extraCompassBottomOffset; @property(nonatomic, readonly) CGFloat visibleHeight; - (void)addToView:(UIView *)superview; diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.mm b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.mm index ae8c5b8d0b..8b5f766242 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.mm +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.mm @@ -1,5 +1,6 @@ #import "MWMNavigationInfoView.h" #import "Common.h" +#import "MWMButton.h" #import "MWMLocationHelpers.h" #import "MWMLocationManager.h" #import "MWMRouter.h" @@ -12,6 +13,44 @@ namespace { CGFloat constexpr kTurnsiPhoneWidth = 96; CGFloat constexpr kTurnsiPadWidth = 140; + +CGFloat constexpr kSearchMainButtonBottomOffsetPortrait = 174; +CGFloat constexpr kSearchMainButtonBottomOffsetLandscape = 48; + +CGFloat constexpr kSearchButtonsViewHeightPortrait = 200; +CGFloat constexpr kSearchButtonsViewWidthPortrait = 200; +CGFloat constexpr kSearchButtonsViewHeightLandscape = 56; +CGFloat constexpr kSearchButtonsViewWidthLandscape = 286; +CGFloat constexpr kSearchButtonsSideSize = 44; + +NSTimeInterval constexpr kCollapseSearchTimeout = 5.0; + +enum class SearchState +{ + Maximized, + MinimizedNormal, + MinimizedSearch, + MinimizedGas, + MinimizedParking, + MinimizedFood, + MinimizedShop, + MinimizedATM +}; + +map const kSearchStateButtonImageNames{ + {SearchState::Maximized, @"ic_routing_search"}, + {SearchState::MinimizedNormal, @"ic_routing_search"}, + {SearchState::MinimizedSearch, @"ic_routing_search_off"}, + {SearchState::MinimizedGas, @"ic_routing_fuel_off"}, + {SearchState::MinimizedParking, @"ic_routing_parking_off"}, + {SearchState::MinimizedFood, @"ic_routing_food_off"}, + {SearchState::MinimizedShop, @"ic_routing_shop_off"}, + {SearchState::MinimizedATM, @"ic_routing_atm_off"}}; + +BOOL defaultOrientation() +{ + return IPAD || UIDeviceOrientationIsPortrait([UIDevice currentDevice].orientation); +} } // namespace @interface MWMNavigationInfoView () @@ -25,6 +64,22 @@ CGFloat constexpr kTurnsiPadWidth = 140; @property(weak, nonatomic) IBOutlet UIImageView * secondTurnImageView; @property(weak, nonatomic) IBOutlet NSLayoutConstraint * turnsWidth; +@property(weak, nonatomic) IBOutlet UIView * searchView; +@property(weak, nonatomic) IBOutlet UIView * searchButtonsView; +@property(weak, nonatomic) IBOutlet MWMButton * searchMainButton; +@property(weak, nonatomic) IBOutlet NSLayoutConstraint * searchButtonsViewHeight; +@property(weak, nonatomic) IBOutlet NSLayoutConstraint * searchButtonsViewWidth; +@property(weak, nonatomic) IBOutlet NSLayoutConstraint * searchMainButtonBottomOffset; +@property(nonatomic) IBOutletCollection(NSLayoutConstraint) NSArray * searchLandscapeConstraints; +@property(nonatomic) IBOutletCollection(UIButton) NSArray * searchButtons; +@property(nonatomic) IBOutletCollection(NSLayoutConstraint) NSArray * searchButtonsSideSize; +@property(weak, nonatomic) IBOutlet MWMButton * searchButtonGas; +@property(weak, nonatomic) IBOutlet MWMButton * searchButtonParking; +@property(weak, nonatomic) IBOutlet MWMButton * searchButtonFood; +@property(weak, nonatomic) IBOutlet MWMButton * searchButtonShop; +@property(weak, nonatomic) IBOutlet MWMButton * searchButtonATM; + +@property(nonatomic) SearchState searchState; @property(nonatomic) BOOL isVisible; @property(weak, nonatomic) MWMNavigationDashboardEntity * navigationInfo; @@ -36,6 +91,7 @@ CGFloat constexpr kTurnsiPadWidth = 140; - (void)addToView:(UIView *)superview { self.isVisible = YES; + [self setSearchState:SearchState::MinimizedNormal animated:NO]; if (IPAD) { self.turnsWidth.constant = kTurnsiPadWidth; @@ -60,22 +116,51 @@ CGFloat constexpr kTurnsiPadWidth = 140; self.frame = self.defaultFrame; [self setNeedsLayout]; } + if (!self.isVisible) [self removeFromSuperview]; [super layoutSubviews]; } -- (void)setIsVisible:(BOOL)isVisible +- (CGFloat)visibleHeight { return self.streetNameView.maxY; } +#pragma mark - Search + +- (IBAction)searchMainButtonTouchUpInside { - _isVisible = isVisible; - [self setNeedsLayout]; - if (isVisible && [MWMRouter router].type == routing::RouterType::Pedestrian) - [MWMLocationManager addObserver:self]; - else - [MWMLocationManager removeObserver:self]; + switch (self.searchState) + { + case SearchState::Maximized: + [self setSearchState:SearchState::MinimizedSearch animated:YES]; + break; + case SearchState::MinimizedNormal: + [self setSearchState:SearchState::Maximized animated:YES]; + break; + case SearchState::MinimizedSearch: + case SearchState::MinimizedGas: + case SearchState::MinimizedParking: + case SearchState::MinimizedFood: + case SearchState::MinimizedShop: + case SearchState::MinimizedATM: + [self setSearchState:SearchState::MinimizedNormal animated:YES]; + break; + } } -- (CGFloat)visibleHeight { return self.streetNameView.maxY; } +- (IBAction)searchButtonTouchUpInside:(MWMButton *)sender +{ + if (sender == self.searchButtonGas) + [self setSearchState:SearchState::MinimizedGas animated:YES]; + else if (sender == self.searchButtonParking) + [self setSearchState:SearchState::MinimizedParking animated:YES]; + else if (sender == self.searchButtonFood) + [self setSearchState:SearchState::MinimizedFood animated:YES]; + else if (sender == self.searchButtonShop) + [self setSearchState:SearchState::MinimizedShop animated:YES]; + else if (sender == self.searchButtonATM) + [self setSearchState:SearchState::MinimizedATM animated:YES]; +} + +- (void)collapseSearchOnTimer { [self setSearchState:SearchState::MinimizedNormal animated:YES]; } #pragma mark - MWMNavigationDashboardInfoProtocol - (void)updateNavigationInfo:(MWMNavigationDashboardEntity *)info @@ -119,6 +204,37 @@ CGFloat constexpr kTurnsiPadWidth = 140; self.hidden = self.streetNameView.hidden && self.turnsView.hidden; } +- (void)layoutSearch +{ + BOOL const defaultView = defaultOrientation(); + self.searchMainButtonBottomOffset.constant = + defaultView ? kSearchMainButtonBottomOffsetPortrait : kSearchMainButtonBottomOffsetLandscape; + CGFloat alpha = 1; + CGFloat searchButtonsSideSize = kSearchButtonsSideSize; + if (self.searchState == SearchState::Maximized) + { + self.searchButtonsViewWidth.constant = + defaultView ? kSearchButtonsViewWidthPortrait : kSearchButtonsViewWidthLandscape; + self.searchButtonsViewHeight.constant = + defaultView ? kSearchButtonsViewHeightPortrait : kSearchButtonsViewHeightLandscape; + } + else + { + self.searchButtonsViewWidth.constant = 0; + self.searchButtonsViewHeight.constant = 0; + alpha = 0; + searchButtonsSideSize = 0; + } + for (UIButton * searchButton in self.searchButtons) + searchButton.alpha = alpha; + UILayoutPriority const priority = + defaultView ? UILayoutPriorityDefaultLow : UILayoutPriorityDefaultHigh; + for (NSLayoutConstraint * constraint in self.searchLandscapeConstraints) + constraint.priority = priority; + for (NSLayoutConstraint * constraint in self.searchButtonsSideSize) + constraint.constant = searchButtonsSideSize; +} + #pragma mark - MWMLocationObserver - (void)onHeadingUpdate:(location::CompassInfo const &)info @@ -134,8 +250,102 @@ CGFloat constexpr kTurnsiPadWidth = 140; self.nextTurnImageView.transform = CGAffineTransformMakeRotation(M_PI_2 - angle); } +#pragma mark - SolidTouchView + +- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event +{ + if (self.searchState == SearchState::Maximized) + return; + [super touchesBegan:touches withEvent:event]; +} + +- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event +{ + if (self.searchState == SearchState::Maximized) + return; + [super touchesMoved:touches withEvent:event]; +} + +- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event +{ + if (self.searchState == SearchState::Maximized) + [self setSearchState:SearchState::MinimizedNormal animated:YES]; + else + [super touchesEnded:touches withEvent:event]; +} + +- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event +{ + if (self.searchState == SearchState::Maximized) + [self setSearchState:SearchState::MinimizedNormal animated:YES]; + else + [super touchesCancelled:touches withEvent:event]; +} + #pragma mark - Properties +- (void)setFrame:(CGRect)frame +{ + if (CGRectEqualToRect(self.frame, frame)) + return; + super.frame = frame; + [self layoutIfNeeded]; + [self layoutSearch]; + [UIView animateWithDuration:kDefaultAnimationDuration + animations:^{ + self.searchButtonsView.layer.cornerRadius = + (defaultOrientation() ? kSearchButtonsViewHeightPortrait + : kSearchButtonsViewHeightLandscape) / + 2; + [self layoutIfNeeded]; + }]; +} + +- (void)setSearchState:(SearchState)searchState animated:(BOOL)animated +{ + self.searchState = searchState; + auto block = ^{ + [self layoutSearch]; + [self layoutIfNeeded]; + }; + if (animated) + { + [self layoutIfNeeded]; + [UIView animateWithDuration:kDefaultAnimationDuration animations:block]; + } + else + { + block(); + } + SEL const collapseSelector = @selector(collapseSearchOnTimer); + [NSObject cancelPreviousPerformRequestsWithTarget:self selector:collapseSelector object:self]; + if (self.searchState == SearchState::Maximized) + { + [self.superview bringSubviewToFront:self]; + [self performSelector:collapseSelector withObject:self afterDelay:kCollapseSearchTimeout]; + } + else + { + [self.superview sendSubviewToBack:self]; + } +} + +- (void)setSearchState:(SearchState)searchState +{ + _searchState = searchState; + self.searchMainButton.imageName = kSearchStateButtonImageNames.at(searchState); +} + +- (void)setIsVisible:(BOOL)isVisible +{ + _isVisible = isVisible; + [self setNeedsLayout]; + if (isVisible && [MWMRouter router].type == routing::RouterType::Pedestrian) + [MWMLocationManager addObserver:self]; + else + [MWMLocationManager removeObserver:self]; +} + - (CGRect)defaultFrame { return CGRectMake(self.leftBound, 0.0, self.superview.width - self.leftBound, @@ -148,4 +358,9 @@ CGFloat constexpr kTurnsiPadWidth = 140; [self setNeedsLayout]; } +- (CGFloat)extraCompassBottomOffset +{ + return defaultOrientation() || self.searchView.hidden ? 0 : kSearchButtonsViewHeightLandscape; +} + @end diff --git a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.xib b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.xib index db4564e436..3d60143905 100644 --- a/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.xib +++ b/iphone/Maps/Classes/CustomViews/NavigationDashboard/MWMNavigationInfoView.xib @@ -22,7 +22,7 @@