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 @@