iOS 平滑移动大头针视图(类似滴滴和UBER)

原文发布于简书: http://www.jianshu.com/p/9a51f7a4bfa3
作者: zerojian

大家都知道,地图上的大头针视图如果想平滑的从一个坐标点移动到另一个坐标点,重新添加一个大头针是无法实现这种效果的,那样不会有一个移动效果,如果想达到类似滴滴车辆平滑移动效果,就必须在同一个大头针对象中改变它的坐标,大头针的视图方向也要根据新的角度改变他的视图角度.

我们这里新建一个 LocationManager 类用来获取用户的坐标,当然如果业务需求是服务器获取,实现思路类似,只是改变一下坐标点的获取方式.具体 LocationManager 单例的创建细节就不多说了,我们在获取到用户位置的回调里面添加一个通知

- (void)locationManager:(CLLocationManager *)manager didUpdateLocations:(NSArray<CLLocation *> *)locations
{
    CLLocation *location = locations.firstObject;
    [[NSNotificationCenter defaultCenter] postNotificationName:kdidUpdateLocation object:location];
}

主要说说自定义大头针,大头针分 MKAnnotationView 用来显示大头针视图, 和 MKAnnotation 用来设置大头针的一些属性,可以简单的把它看成一个 view - model 的关系

如果需要自定义大头针, 需要遵守 BKAnnotation 协议,其中 coordinate 属性是必须实现的属性

@protocol MKAnnotation <NSObject>

// Center latitude and longitude of the annotation view.
// The implementation of this property must be KVO compliant.
@property (nonatomic, readonly) CLLocationCoordinate2D coordinate;

@optional

// Title and subtitle for use by selection UI.
@property (nonatomic, readonly, copy, nullable) NSString *title;
@property (nonatomic, readonly, copy, nullable) NSString *subtitle;

// Called as a result of dragging an annotation view.
- (void)setCoordinate:(CLLocationCoordinate2D)newCoordinate NS_AVAILABLE(10_9, 4_0);

@end

在自定义的大头针里面我们添加几个自定义属性

@property (nonatomic, assign) MyAnnotationType type;

@property (nonatomic, assign) double angle;

@property (nonatomic, strong) MKAnnotationView *annotationView;

其中 type 是一个枚举类型,用来自定义大头针的类型, angle 表示方向,用来显示大头针的方向

在 .m 文件中自定义几个内部属性

    NSMutableArray *locations;
    NSArray *moveArray;
    BOOL movingOver;
    NSInteger currentIndex;
    UIImage *annnationImage;

在大头针的初始化方法中,监听定位通知,初始化保存定位的数组

- (instancetype)initWithType:(MyAnnotationType)type
                  coordinate:(CLLocationCoordinate2D)coordinate
                       angle:(double)angle {

    if (self == [super init]) {

        _type = type;
        _coordinate = coordinate;
        _angle = angle;
        if (type == MyAnnotationTypeCar) {
            [[NSNotificationCenter defaultCenter] addObserver:self                   selector:@selector(onNotifiedCurrentLocationUpdated:)                             name:kdidUpdateLocation
                         object:nil];
            locations = [[NSMutableArray alloc] initWithCapacity:20];
            movingOver = YES;
        }
    }
    return self;
}

在监听的通知中, 我们保存获取到的定位对象,为了保证大头针的移动效果,我们需要在建一个移动的数组,这个移动数组保证有一定量的定位数据才开始移动动画效果,而定位数组只负责把所有获取到的定位对象保存下来

- (void)onNotifiedCurrentLocationUpdated:(NSNotification *)notification
{
    if (notification.object) {
            CLLocation *location = notification.object;
            [locations addObject:location];
            
            if (locations.count >= 5 && movingOver == YES) {
                    moveArray = [NSArray arrayWithArray:locations];
                    [locations removeAllObjects];
                    NSLog(@"----- moveArrayCount: %ld",moveArray.count);
                    NSLog(@"------beging moving -----");
                    currentIndex = 0;
                    movingOver = NO;
                    [self startMoving];
            }
    }
}   

核心的移动大头针方法,使用递归的方式一次次的移动大头针,在移动数组全部使用后结束递归,等待新获取到的数组

- (void)startMoving
{
    NSInteger index = currentIndex % moveArray.count;
    CLLocation *newLocation = moveArray[index];
    
    self.annotationView.image = [annnationImage zj_imageRotatedByAngle:newLocation.course];
    
    CLLocation *currentLocation = [[CLLocation alloc] initWithLatitude:self.coordinate.latitude longitude:self.coordinate.longitude];
    double distance = [newLocation distanceFromLocation:currentLocation];
    
    double speed = newLocation.speed;
    
    CLLocationCoordinate2D newCoordinate = newLocation.coordinate;
    
    [UIView animateWithDuration:distance/speed animations:^{
            self.coordinate = newCoordinate;
            currentIndex ++;
    } completion:^(BOOL finished) {
            if (currentIndex == moveArray.count) {
                    movingOver = YES;
                    moveArray = nil;
            } else {
                    [self startMoving];
            }
    }];
}

只能需要注意的几个细节,一个是大头针角度,在 CLLocation 对象里面是有一个角度属性和速度属性的,当然如果只有两个坐标点也是可以通过算法获取到的.

这里要用到的速度和两个坐标点的距离用来设定车辆移动动画的时间,这样可以保证车辆的移动速度是均匀的平滑的.

我们在每次动画后把创建的 index + 1 ,然后和数组的 count 通过取余运算获取到下次递归数组使用的 index

大头针的平滑移动,只需使用 UIView 动画简单的把新的坐标赋值给大头针即可实现.

项目 DEMO : https://github.com/ZeroJian/SmoothMovingAnnotation

有帮助给个 star 吧!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,047评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,807评论 3 386
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,501评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,839评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,951评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,117评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,188评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,929评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,372评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,679评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,837评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,536评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,168评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,886评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,129评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,665评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,739评论 2 351

推荐阅读更多精彩内容