我正在使用mapkit和CLLocationManager开发一个iphone应用程序。
我在地图上放了很多MKPinAnnotationView(大约100个),我希望在收到它时更新所有标注的副标题用户距离。
怎么做?
由于
我尝试使用新位置更新字幕标注,但效果不佳。
在我的MyAppDelegate.h
中extern NSString * const GMAP_USERLOCATION_CHANGED;
@interface MyAppDelegate : NSObject <UIApplicationDelegate> {
CLLocationManager *locationManager;
CLLocation *userLocation;
}
@property (nonatomic, retain) CLLocationManager *locationManager;
@property (nonatomic, retain) CLLocation *userLocation;
@end
在我的MyAppDelegate.m
中@implementation MyAppDelegate
NSString * const GMAP_USERLOCATION_CHANGED = @"gMapUserLocationChanged";
@synthesize locationManager, userLocation;
- (void)applicationDidFinishLaunching:(UIApplication *)application
{
userLocation = nil;
[[self locationManager] startUpdatingLocation];
[window addSubview:tabBarController.view];
[window makeKeyAndVisible];
}
#pragma mark -
#pragma mark Core Location delegate
- (CLLocationManager *)locationManager
{
if (locationManager != nil)
{
return locationManager;
}
locationManager = [[CLLocationManager alloc] init];
locationManager.desiredAccuracy = kCLLocationAccuracyNearestTenMeters;
locationManager.delegate = self;
return locationManager;
}
- (void)locationManager:(CLLocationManager *)manager
didUpdateToLocation:(CLLocation *)newLocation
fromLocation:(CLLocation *)oldLocation
{
self.userLocation = newLocation;
// send notification to defaulcenter
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
[nc postNotificationName:GMAP_USERLOCATION_CHANGED object:self.userLocation];
}
- (void)dealloc {
[window release];
[locationManager release];
[userLocation release];
[super dealloc];
}
@end
我创建了一个名为MyAnnotation的customAnnotationView。
在MyAnnotation.h中:@interface MyAnnotation : MKPinAnnotationView<MKAnnotation>
{
double longitude;
double latitude;
NSString *title;
NSString *subtitle;
}
@property (nonatomic, retain) NSString *title;
@property (nonatomic, retain) NSString *subtitle;
@property double longitude;
@property double latitude;
@end
在MyAnnotation.m中:
#import "MyAnnotation.h"
#import "MyAppDelegate.h"
@implementation MyAnnotation
@synthesize title, subtitle;
@synthesize longitude;
@synthesize latitude;
-(id)initWithAnnotation:(id <MKAnnotation>)annotation reuseIdentifier:(NSString *)reuseIdentifier
{
self = [super initWithAnnotation:annotation reuseIdentifier:reuseIdentifier];
if (self != nil)
{
NSLog(@"add observer");
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
[nc addObserver:self selector:@selector(receivedNewUserLocation:) name:GMAP_USERLOCATION_CHANGED object:nil];
}
return self;
}
- (void)setTitleAndSubtitle
{
[self setTitleAndSubtitle:nil];
}
- (id)setTitleAndSubtitle:(CLLocation*)userLocation
{
CLLocationDistance dist = -1;
if(userLocation)
{
CLLocation *poiLoc = [[CLLocation alloc] initWithLatitude:self.latitude longitude:self.longitude];
dist = [userLocation distanceFromLocation:poiLoc] / 1000;
NSLog(@"distance is now %.f", dist);
}
title = @"the Title of the poi!";
subtitle = [NSString stringWithFormat:@"Distance: %@",
dist > -1 ? [NSString stringWithFormat:@"%.2f km", dist] : @"-"
];
return self;
}
- (void)receivedNewUserLocation:(NSNotification *)userLocationNotification
{
CLLocation *userlocation = (CLLocation*)[userLocationNotification object];
[self setTitleAndSubtitle:userlocation];
}
- (CLLocationCoordinate2D)coordinate;
{
CLLocationCoordinate2D theCoordinate;
theCoordinate.latitude = latitude;
theCoordinate.longitude = longitude;
return theCoordinate;
}
- (NSString *)title
{
return title;
}
- (NSString *)subtitle
{
return subtitle;
}
- (void)dealloc
{
[title release];
[subtitle release];
NSNotificationCenter *nc = [NSNotificationCenter defaultCenter];
[nc removeObserver:self];
[super dealloc];
}
@end
最后我在MapViewController中使用它(我在这里只放了viewForAnnotation委托方法):
- (MKAnnotationView *)mapView:(MKMapView *)theMapView viewForAnnotation:(id <MKAnnotation>)annotation
{
// if it's the user location, just return nil.
if ([annotation isKindOfClass:[MKUserLocation class]])
return nil;
MyAnnotation *annotationView = nil;
MyAnnotation* myAnnotation = (MyAnnotation *)annotation;
// try to dequeue an existing pin view first
NSString* identifier = @"CustomMapAnnotation";
MyAnnotation *customPinView = (MyAnnotation *)[self.mapView dequeueReusableAnnotationViewWithIdentifier:identifier];
if(nil == customPinView)
{
// if an existing pin view was not available, create one
customPinView = [[[MyAnnotation alloc]
initWithAnnotation:myAnnotation
reuseIdentifier:identifier]
autorelease];
customPinView.animatesDrop = YES;
customPinView.canShowCallout = YES;
}
annotationView = customPinView;
[annotationView setEnabled:YES];
[annotationView setCanShowCallout:YES];
return annotationView;
}
在mkannotation加载到地图后,所有这个字幕都没有更新.... 怎么了?
感谢您的帮助......
答案 0 :(得分:6)
更新标题时,您应该通过最适合您需要的KVO方式通知MKAnnotationView更新标注视图,例如:
为标题合成或实现setter并使用
self.title = @"new title";
使用显式KVO通知
[self willChangeValueForKey:@"title"];
[title release];
title = [newTitle copy];
[self didChangeValueForKey:@"title"];
同上字幕。
答案 1 :(得分:2)
这取决于您如何创建MKAnnotations。
你应该有一个像Place.h和Place.m所代表的“Place”这样的对象,它符合MKAnnotation协议......
地方将有一个属性
float distance;
然后你的字幕方法(MKAnnotation的一部分)会做这样的事情
- (NSString *)subtitle
{
return [NSString stringWithFormat:@"%0.2f Miles Away", distance];
}
这个字幕方法不断地被mapview调用(实际上它的调用频率几乎是荒谬的),所以一旦你操纵了距离的值,它就会反映在地图上(也许早在下一个地图上)你点击针脚的时间。)