Swift — перемещение GMSMarker по массиву CLCoordinates из GMSPath (Google Maps SDK для iOS)

Все попытки анимировать движение маркера в Картах Google между координатами указывают на использование следующего фрагмента кода в Swift:

CATransaction.begin()
CATransaction.setAnimationDuration(duration)
marker.position = coordindates
CATransaction.commit()

В качестве примера, вот сообщение SO с наибольшим количеством голосов: Как плавно перемещать GMSMarker по координатам в Objective c

Это работает нормально при анимации между парой координат источника и пункта назначения. Тем не менее, я хочу анимировать от начальной координаты в GMSPath до конечной координаты. При переборе точек пути единственная отображаемая анимация находится между двумя последними координатами. Маркер появляется только в предпоследней точке и анимируется до последней точки.

Вот мой код для ViewController. Он получает сегмент маршрута в виде закодированного пути (для тестирования первый закодированный путь: "ika~Exi|vN|AaDzAyCTc@N[lBeEvB_ExBkExBmEjBwDXo@").

Код перебирает все сохраненные координаты внутри объекта GMSPath и пытается анимировать с помощью приведенного выше фрагмента. Как уже говорилось, он показывает анимацию только между двумя последними точками.

Я попытался разместить все наборы кода в ViewDidLoad, ViewDidAppear и ViewWillAppear. ViewDidLoad сохранил уровень масштабирования на межконтинентальном уровне. ViewDidAppear и ViewWillAppear правильно увеличили масштаб, что привело к проблемам с анимацией, упомянутым в этом посте. В настоящее время код разделен между ViewDidAppear и ViewWillAppear, но будет действовать одинаково, если будет размещен только в любом методе.

import UIKit
import GoogleMaps
import CoreLocation

class MapVC:UIViewController {

    var mapView:GMSMapView?

    var polyline:GMSPolyline?

    var path:GMSPath?

    var encodedPath:String? = nil

    var marker:GMSMarker?

    override func viewDidLoad() {
        super.viewDidLoad()


        setupMap()

    }

    override func viewWillAppear(_ animated: Bool) {
        super.viewWillAppear(animated)

        if encodedPath != nil {

            self.path = GMSPath(fromEncodedPath: encodedPath!)

            self.polyline = GMSPolyline(path: path)
            self.polyline!.map = self.mapView!

            let bounds:GMSCoordinateBounds = GMSCoordinateBounds(path: path!)

            let update = GMSCameraUpdate.fit(bounds, withPadding: 10.0)
            self.mapView!.animate(with: update)



        } else {

            print("nil path")

        }
        let a=2

    }

    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)

        var index:UInt = 0

        let count:UInt = self.path!.count()

        if count > 0 {

            marker = GMSMarker(position: self.path!.coordinate(at:index))
            marker!.map = self.mapView

            index += 1

            while index < count {

                CATransaction.begin()
                CATransaction.setAnimationDuration(30)
                self.marker!.position = self.path!.coordinate(at:index)
                CATransaction.commit()
                index += 1

            }

        }

    }




    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }

    func setupMap() {

        let camera = GMSCameraPosition.camera(withLatitude: 36.5, longitude: -82.5, zoom: 16)
        mapView = GMSMapView.map(withFrame: CGRect.zero, camera: camera)

        self.view = mapView

    }

}

person Michael James    schedule 12.10.2018    source источник


Ответы (1)


Нашел решение с помощью таймера: https://github.com/antonyraphel/ARCarMovement/blob/master/ARCarMovementSwift/ARCarMovementSwift/ViewController.swift

Обновленный код контроллера представления, показанный выше:

    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)

        marker = GMSMarker(position: self.path!.coordinate(at:self.index))
        marker!.map = self.mapView

        timer = Timer.scheduledTimer(timeInterval: 2.0, target: self, selector: 
                #selector(MapVC.timerTriggered), userInfo: nil, repeats: true)


    }


    @objc func timerTriggered() {

        if self.index < self.path!.count() {

            CATransaction.begin()
            CATransaction.setAnimationDuration(1.9)
            self.marker!.position = self.path!.coordinate(at:index)
            CATransaction.commit()
            self.index += 1

        } else {

            timer.invalidate()
            timer = nil

        }

    }
person Michael James    schedule 13.10.2018