【发布时间】:2009-09-02 01:06:24
【问题描述】:
我正在开发一个 3 选项卡的 iPhone 应用程序。我希望每个选项卡的视图看起来共享同一张地图。所以目前,我只是想弄清楚当单击新选项卡时如何重置每个视图的 MKMapView 区域。 [换句话说,如果您在选项卡 1 上并在某个位置一直放大,然后单击选项卡 2,我希望它显示相同的放大地图。]
为此,我在应用程序委托中有一个 MKCoordinateRegion,每个视图在它消失之前保存到它,并且每个新视图在它被查看之前读取。
[换句话说,我的应用程序委托是我的 TabBarControllerDelegate,并且我已经覆盖了 tabBarController: shouldSelectViewController:。此函数在有人单击选项卡按钮之后调用,并且在新选项卡视图的 viewWillAppear: 函数被调用之前调用。因此,在这个函数中,我将当前视图的区域保存到我的 AppDelegate::MKCoordinateRegion 变量中,然后调用新视图的 viewWillAppear:(或 viewDidLoad:)。然后,在新视图的 viewWillAppear: 中,我从应用委托中获取 MKCoordinateRegion,并将其分配给我的新视图的区域。]
但是,来自 MKMapKit::setRegion: 的值似乎并不一致,我不知道我是否只是做错了什么,完全错过了某些东西,或者是否发生了其他事情。
我只在做 iPhone 开发。 (和 Obj-c 等)现在大约 2 周,所以这很好可能是一个新错误。如果是这样,我将继续为浪费您的时间道歉。尽管如此,这里有一些代码(我已经留在了我正在使用的 cmets 中......以防万一你自己尝试运行它):
iPhoneTestAppDelegate.h
...
@interface iPhoneTestAppDelegate : NSObject
{
UIWindow* _window;
UITabBarController* _tabBarController;
SubViewController* _subViewController;
MKCoordinateRegion _mapRegion;
}
@property (nonatomic, retain) IBOutlet UIWindow* window;
@property (nonatomic, retain) IBOutlet UITabBarController* tabBarController;
@property (nonatomic, retain) IBOutlet SubViewController* subViewController;
@property (nonatomic, assign) MKCoordinateRegion mapRegion;
@end
iPhoneAppDelegate.m
...
static CLLocationDegrees INITIAL_LATITUDE = 40.754019;
static CLLocationDegrees INITIAL_LONGITUDE = -73.973351;
static CLLocationDegrees INITIAL_SPAN_LAT_DEG = .10767;
static CLLocationDegrees INITIAL_SPAN_LONG_DEG = .109863;
...
- (void)applicationDidFinishLaunching:(UIApplication *)application
{
NSLog(@"Begin");
// initialize the delegate's region
MKCoordinateRegion region = {{0.0f, 0.0f}, {0.0f, 0.0f}};
region.center.latitude = INITIAL_LATITUDE;
region.center.longitude = INITIAL_LONGITUDE;
region.span.latitudeDelta = INITIAL_SPAN_LAT_DEG;
region.span.longitudeDelta = INITIAL_SPAN_LONG_DEG;
self.mapRegion = region;
self.tabBarController.delegate = self;
[self.window addSubview:self.tabBarController.view];
}
- (BOOL)tabBarController:(UITabBarController *)tabBarController
shouldSelectViewController:(UIViewController *)viewController
{
NSLog(@"in UITabBarControllerDelegate: shouldSelectViewController:");
if([tabBarController.viewControllers indexOfObject:viewController] !=
[tabBarController selectedIndex])
{
NSLog(@"different view selected");
BaseViewController* currView = (BaseViewController *)[tabBarController selectedViewController];
NSLog(@"%f %f %f %f",
self.mapRegion.center.latitude, self.mapRegion.center.longitude,
self.mapRegion.span.latitudeDelta, self.mapRegion.span.longitudeDelta);
NSLog(@"%f %f %f %f",
currView.mapView.region.center.latitude, currView.mapView.region.center.longitude,
currView.mapView.region.span.latitudeDelta, currView.mapView.region.span.longitudeDelta);
self.mapRegion = currView.mapView.region;
NSLog(@"%f %f %f %f",
self.mapRegion.center.latitude, self.mapRegion.center.longitude,
self.mapRegion.span.latitudeDelta, self.mapRegion.span.longitudeDelta);
}
else
NSLog(@"same view selected");
return YES;
}
BaseViewController.m
...
- (void)viewDidLoad
{
[super viewDidLoad];
NSLog(@"Base: viewDidLoad");
iPhoneTestAppDelegate* appDelegate = (iPhoneTestAppDelegate *)
[[UIApplication sharedApplication] delegate];
[self.mapView setDelegate:self];
//[self.mapView setRegion:appDelegate.mapRegion animated:YES];
[self.mapView setRegion:[self.mapView regionThatFits:appDelegate.mapRegion]];
NSLog(@"---------------------");
NSLog(@"%f %f %f %f",
appDelegate.mapRegion.center.latitude, appDelegate.mapRegion.center.longitude,
appDelegate.mapRegion.span.latitudeDelta, appDelegate.mapRegion.span.longitudeDelta);
NSLog(@"---------------------");
NSLog(@"%f %f %f %f",
self.mapView.region.center.latitude, self.mapView.region.center.longitude,
self.mapView.region.span.latitudeDelta, self.mapView.region.span.longitudeDelta);
NSLog(@"---------------------");
}
- (void)viewWillAppear:(BOOL)animated
{
[super viewWillAppear:animated];
NSLog(@"Base: viewWillAppear");
iPhoneTestAppDelegate* appDelegate = (iPhoneTestAppDelegate *)
[[UIApplication sharedApplication] delegate];
//[self.mapView setRegion:appDelegate.mapRegion animated:YES];
[self.mapView setRegion:[self.mapView regionThatFits:appDelegate.mapRegion]];
NSLog(@"*********************");
NSLog(@"%f %f %f %f",
appDelegate.mapRegion.center.latitude, appDelegate.mapRegion.center.longitude,
appDelegate.mapRegion.span.latitudeDelta, appDelegate.mapRegion.span.longitudeDelta);
NSLog(@"*********************");
NSLog(@"%f %f %f %f",
self.mapView.region.center.latitude, self.mapView.region.center.longitude,
self.mapView.region.span.latitudeDelta, self.mapView.region.span.longitudeDelta);
NSLog(@"*********************");
}
...
现在,当我运行它时,我首先注意到的是,当调用初始视图的 viewDidLoad: 时,委托的区域和我的新视图的区域是相同的(在我调用 setRegion: 之后)。但是,在 viewWillAppear: 中调用 setRegion: 之后,我的视图的 MKMapView 的区域不再等于代表。不知何故,viewDidLoad: 中完全相同的代码在 viewWillAppear: 中运行时会产生不同的结果:!不应该是这样吗?为什么调用 setRegion: from viewWillAppear: 会改变我视图的 MKMapView 区域的值?这是否与 regionThatFits: 内部调用有关?我不明白!
但是,我有点认为,如果我能弄清楚那部分,我也许能够弄清楚其他问题;所以,我不会在这里详细介绍。但是,如果您制作一个快速应用程序。使用这些功能并运行它,您会看到问题。主要是,视图的地图偶尔会同步,但是当您缩放和移动(并不断在选项卡之间来回切换)时,跨度开始缩小,每次新的选项卡按下只会让我越来越远。
注意:我只是在模拟器(3.0)上运行这个[我还没有硬件],所以我不知道这是否与它有关。但是,我有点怀疑。
无论如何,感谢您提供的任何帮助。我真的很感激。
【问题讨论】:
标签: iphone