diff --git a/iphone/Maps/Classes/MapViewController.mm b/iphone/Maps/Classes/MapViewController.mm index 5743e89b33..a35306c76a 100644 --- a/iphone/Maps/Classes/MapViewController.mm +++ b/iphone/Maps/Classes/MapViewController.mm @@ -105,6 +105,20 @@ NSString *const kPP2BookmarkEditingSegue = @"PP2BookmarkEditing"; #pragma mark - Map Navigation +- (void)showOrUpdatePlacePage { + if (!PlacePageData.hasData) { + return; + } + + self.controlsManager.trafficButtonHidden = YES; + + if (self.placePageVC != nil) { + [PlacePageBuilder update:(PlacePageViewController *)self.placePageVC]; + return; + } + [self showRegularPlacePage]; +} + - (void)showRegularPlacePage { self.placePageVC = [PlacePageBuilder build]; self.placePageContainer.hidden = NO; @@ -121,15 +135,6 @@ NSString *const kPP2BookmarkEditingSegue = @"PP2BookmarkEditing"; [self.placePageVC didMoveToParentViewController:self]; } -- (void)showPlacePage { - if (!PlacePageData.hasData) { - return; - } - - self.controlsManager.trafficButtonHidden = YES; - [self showRegularPlacePage]; -} - - (void)dismissPlacePage { GetFramework().DeactivateMapSelection(true); } @@ -179,8 +184,7 @@ NSString *const kPP2BookmarkEditingSegue = @"PP2BookmarkEditing"; } - (void)onMapObjectSelected { - [self hidePlacePage]; - [self showPlacePage]; + [self showOrUpdatePlacePage]; } - (void)onMapObjectUpdated { diff --git a/iphone/Maps/UI/PlacePage/Components/PlacePagePreviewViewController.swift b/iphone/Maps/UI/PlacePage/Components/PlacePagePreviewViewController.swift index eb4dc75cf0..473ba0c9f2 100644 --- a/iphone/Maps/UI/PlacePage/Components/PlacePagePreviewViewController.swift +++ b/iphone/Maps/UI/PlacePage/Components/PlacePagePreviewViewController.swift @@ -109,15 +109,16 @@ final class PlacePagePreviewViewController: UIViewController { } func updateHeading(_ angle: CGFloat) { - heading = angle placePageDirectionView?.imageView.isHidden = false - UIView.animate(withDuration: kDefaultAnimationDuration, + let duration = heading == nil ? .zero : kDefaultAnimationDuration // skip the initial setup animation + UIView.animate(withDuration: duration, delay: 0, options: [.beginFromCurrentState, .curveEaseInOut], animations: { [unowned self] in self.placePageDirectionView?.imageView.transform = CGAffineTransform(rotationAngle: CGFloat.pi / 2 - angle) }) fullScreenDirectionView.updateHeading(angle) + heading = angle } func updateSpeedAndAltitude(_ speedAndAltitude: String) { diff --git a/iphone/Maps/UI/PlacePage/PlacePageBuilder.swift b/iphone/Maps/UI/PlacePage/PlacePageBuilder.swift index 49f003fd6d..7a6c36ab38 100644 --- a/iphone/Maps/UI/PlacePage/PlacePageBuilder.swift +++ b/iphone/Maps/UI/PlacePage/PlacePageBuilder.swift @@ -15,14 +15,33 @@ } else { layout = PlacePageCommonLayout(interactor: interactor, storyboard: storyboard, data: data) } - let presenter = PlacePagePresenter(view: viewController, - interactor: interactor, - layout: layout) + let presenter = PlacePagePresenter(view: viewController, interactor: interactor) - interactor.presenter = presenter + viewController.setLayout(layout) viewController.presenter = presenter + interactor.presenter = presenter layout.presenter = presenter - return viewController } + + @objc static func update(_ viewController: PlacePageViewController) { + let data = PlacePageData(localizationProvider: OpeinigHoursLocalization()) + viewController.isPreviewPlus = data.isPreviewPlus + let interactor = PlacePageInteractor(viewController: viewController, + data: data, + mapViewController: MapViewController.shared()!) + let layout:IPlacePageLayout + if data.elevationProfileData != nil { + layout = PlacePageElevationLayout(interactor: interactor, storyboard: viewController.storyboard!, data: data) + } else { + layout = PlacePageCommonLayout(interactor: interactor, storyboard: viewController.storyboard!, data: data) + } + let presenter = PlacePagePresenter(view: viewController, interactor: interactor) + + viewController.presenter = presenter + interactor.presenter = presenter + layout.presenter = presenter + viewController.updateWithLayout(layout) + viewController.updatePreviewOffset() + } } diff --git a/iphone/Maps/UI/PlacePage/PlacePageLayout/Layouts/PlacePageCommonLayout.swift b/iphone/Maps/UI/PlacePage/PlacePageLayout/Layouts/PlacePageCommonLayout.swift index 86a6794616..9f230375bd 100644 --- a/iphone/Maps/UI/PlacePage/PlacePageLayout/Layouts/PlacePageCommonLayout.swift +++ b/iphone/Maps/UI/PlacePage/PlacePageLayout/Layouts/PlacePageCommonLayout.swift @@ -190,10 +190,9 @@ extension PlacePageCommonLayout { headerViewController.setTitle(title, secondaryTitle: secondaryTitle) placePageNavigationViewController.setTitle(title, secondaryTitle: secondaryTitle) } - self.presenter?.layoutIfNeeded() + presenter?.layoutIfNeeded() UIView.animate(withDuration: kDefaultAnimationDuration) { [unowned self] in self.bookmarkViewController.view.isHidden = !isBookmark - self.presenter?.layoutIfNeeded() } } } diff --git a/iphone/Maps/UI/PlacePage/PlacePagePresenter.swift b/iphone/Maps/UI/PlacePage/PlacePagePresenter.swift index e0b1e73e8f..f517e9ab99 100644 --- a/iphone/Maps/UI/PlacePage/PlacePagePresenter.swift +++ b/iphone/Maps/UI/PlacePage/PlacePagePresenter.swift @@ -9,15 +9,11 @@ protocol PlacePagePresenterProtocol: AnyObject { class PlacePagePresenter: NSObject { private weak var view: PlacePageViewProtocol! private let interactor: PlacePageInteractorProtocol - private let layout: IPlacePageLayout init(view: PlacePageViewProtocol, - interactor: PlacePageInteractorProtocol, - layout: IPlacePageLayout) { + interactor: PlacePageInteractorProtocol) { self.view = view self.interactor = interactor - self.layout = layout - view.setLayout(layout) } } diff --git a/iphone/Maps/UI/PlacePage/PlacePageViewController.swift b/iphone/Maps/UI/PlacePage/PlacePageViewController.swift index 7b6e2470c2..dad112eb85 100644 --- a/iphone/Maps/UI/PlacePage/PlacePageViewController.swift +++ b/iphone/Maps/UI/PlacePage/PlacePageViewController.swift @@ -1,10 +1,12 @@ protocol PlacePageViewProtocol: AnyObject { var presenter: PlacePagePresenterProtocol! { get set } + func setLayout(_ layout: IPlacePageLayout) - func layoutIfNeeded() func closeAnimated() func updatePreviewOffset() func showNextStop() + func layoutIfNeeded() + func updateWithLayout(_ layout: IPlacePageLayout) } final class PlacePageScrollView: UIScrollView { @@ -49,36 +51,8 @@ final class PlacePageScrollView: UIScrollView { override func viewDidLoad() { super.viewDidLoad() - layout.headerViewControllers.forEach({ addToHeader($0) }) - layout.bodyViewControllers.forEach({ addToBody($0) }) - - if let actionBar = layout.actionBar { - hideActionBar(false) - addActionBar(actionBar) - } else { - hideActionBar(true) - } - - let bgView = UIView() - bgView.styleName = "PPBackgroundView" - stackView.insertSubview(bgView, at: 0) - bgView.alignToSuperview() - - scrollView.decelerationRate = .fast - scrollView.backgroundColor = .clear - - stackView.backgroundColor = .clear - - let cornersToMask: CACornerMask = alternativeSizeClass(iPhone: [], iPad: [.layerMinXMaxYCorner, .layerMaxXMaxYCorner]) - actionBarContainerView.layer.setCorner(radius: 16, corners: cornersToMask) - actionBarContainerView.layer.masksToBounds = true - - // See https://github.com/organicmaps/organicmaps/issues/6917 for the details. - if #available(iOS 13.0, *), previousTraitCollection == nil { - scrollView.contentInset = alternativeSizeClass(iPhone: UIEdgeInsets(top: view.height, left: 0, bottom: 0, right: 0), - iPad: UIEdgeInsets.zero) - scrollView.layoutIfNeeded() - } + setupView() + setupLayout(layout) } override func viewDidLayoutSubviews() { @@ -112,21 +86,39 @@ final class PlacePageScrollView: UIScrollView { } } - func updateSteps() { + + // MARK: - Actions + + @IBAction func onPan(gesture: UIPanGestureRecognizer) { + let xOffset = gesture.translation(in: view.superview).x + gesture.setTranslation(CGPoint.zero, in: view.superview) + view.minX += xOffset + view.minX = min(view.minX, 0) + let alpha = view.maxX / view.width + view.alpha = alpha + + let state = gesture.state + if state == .ended || state == .cancelled { + if alpha < 0.8 { + closeAnimated() + } else { + UIView.animate(withDuration: kDefaultAnimationDuration) { + self.view.minX = 0 + self.view.alpha = 1 + } + } + } + } + + // MARK: - Private methods + + private func updateSteps() { layoutIfNeeded() scrollSteps = layout.calculateSteps(inScrollView: scrollView, compact: traitCollection.verticalSizeClass == .compact) } - func updatePreviewOffset() { - updateSteps() - if !beginDragging { - let stateOffset = isPreviewPlus ? scrollSteps[2].offset : scrollSteps[1].offset + Constants.additionalPreviewOffset - scrollTo(CGPoint(x: 0, y: stateOffset)) - } - } - - func findNextStop(_ offset: CGFloat, velocity: CGFloat) -> PlacePageState { + private func findNextStop(_ offset: CGFloat, velocity: CGFloat) -> PlacePageState { if velocity == 0 { return findNearestStop(offset) } @@ -153,6 +145,51 @@ final class PlacePageScrollView: UIScrollView { return result } + private func setupView() { + let bgView = UIView() + bgView.styleName = "PPBackgroundView" + stackView.insertSubview(bgView, at: 0) + bgView.alignToSuperview() + + scrollView.decelerationRate = .fast + scrollView.backgroundColor = .clear + + stackView.backgroundColor = .clear + + let cornersToMask: CACornerMask = alternativeSizeClass(iPhone: [], iPad: [.layerMinXMaxYCorner, .layerMaxXMaxYCorner]) + actionBarContainerView.layer.setCorner(radius: 16, corners: cornersToMask) + actionBarContainerView.layer.masksToBounds = true + + // See https://github.com/organicmaps/organicmaps/issues/6917 for the details. + if #available(iOS 13.0, *), previousTraitCollection == nil { + scrollView.contentInset = alternativeSizeClass(iPhone: UIEdgeInsets(top: view.height, left: 0, bottom: 0, right: 0), + iPad: UIEdgeInsets.zero) + scrollView.layoutIfNeeded() + } + } + + private func setupLayout(_ layout: IPlacePageLayout) { + setLayout(layout) + + layout.headerViewControllers.forEach({ addToHeader($0) }) + layout.bodyViewControllers.forEach({ addToBody($0) }) + + beginDragging = false + if let actionBar = layout.actionBar { + hideActionBar(false) + addActionBar(actionBar) + } else { + hideActionBar(true) + } + } + + private func cleanupLayout() { + layout?.actionBar?.view.removeFromSuperview() + layout?.navigationBar?.view.removeFromSuperview() + headerStackView.arrangedSubviews.forEach { $0.removeFromSuperview() } + stackView.arrangedSubviews.forEach { $0.removeFromSuperview() } + } + private func findNearestStop(_ offset: CGFloat) -> PlacePageState { var result = scrollSteps[0] scrollSteps.suffix(from: 1).forEach { ppState in @@ -163,42 +200,35 @@ final class PlacePageScrollView: UIScrollView { return result } - func showLastStop() { + private func showLastStop() { if let lastStop = scrollSteps.last { scrollTo(CGPoint(x: 0, y: lastStop.offset), forced: true) } } - @IBAction func onPan(gesture: UIPanGestureRecognizer) { - let xOffset = gesture.translation(in: view.superview).x - gesture.setTranslation(CGPoint.zero, in: view.superview) - view.minX += xOffset - view.minX = min(view.minX, 0) - let alpha = view.maxX / view.width - view.alpha = alpha - - let state = gesture.state - if state == .ended || state == .cancelled { - if alpha < 0.8 { - closeAnimated() - } else { - UIView.animate(withDuration: kDefaultAnimationDuration) { - self.view.minX = 0 - self.view.alpha = 1 - } - } - } - } - - func updateTopBound(_ bound: CGFloat, duration: TimeInterval) { + private func updateTopBound(_ bound: CGFloat, duration: TimeInterval) { alternativeSizeClass(iPhone: { presenter.updateTopBound(bound, duration: duration) }, iPad: {}) } } +// MARK: - PlacePageViewProtocol + extension PlacePageViewController: PlacePageViewProtocol { + func layoutIfNeeded() { + guard layout != nil else { return } + view.layoutIfNeeded() + } + + func updateWithLayout(_ layout: IPlacePageLayout) { + setupLayout(layout) + } + func setLayout(_ layout: IPlacePageLayout) { + if self.layout != nil { + cleanupLayout() + } self.layout = layout } @@ -213,6 +243,14 @@ extension PlacePageViewController: PlacePageViewProtocol { headerStackView.addArrangedSubview(headerViewController.view) } + func updatePreviewOffset() { + updateSteps() + if !beginDragging { + let stateOffset = isPreviewPlus ? scrollSteps[2].offset : scrollSteps[1].offset + Constants.additionalPreviewOffset + scrollTo(CGPoint(x: 0, y: stateOffset)) + } + } + func addToBody(_ viewController: UIViewController) { addChild(viewController) stackView.addArrangedSubview(viewController.view) @@ -274,14 +312,9 @@ extension PlacePageViewController: PlacePageViewProtocol { } } - func layoutIfNeeded() { - view.layoutIfNeeded() - } - func closeAnimated() { alternativeSizeClass(iPhone: { self.scrollTo(CGPoint(x: 0, y: -self.scrollView.height + 1), - animated: true, forced: true) { self.rootViewController.dismissPlacePage() } @@ -342,7 +375,7 @@ extension PlacePageViewController: UIScrollViewDelegate { } private func setNavigationBarVisible(_ visible: Bool) { - guard visible != isNavigationBarVisible, let navigationBar = layout.navigationBar else { return } + guard visible != isNavigationBarVisible, let navigationBar = layout?.navigationBar else { return } isNavigationBarVisible = visible if isNavigationBarVisible { addNavigationBar(navigationBar)