2009-09-02 16 views
2

3タブ付きのiPhoneアプリで作業しています。ここで、各タブの表示が同じマップを共有するようにしたいと考えています。だから、現時点では、新しいタブをクリックしたときに各ビューのMKMapViewの領域をリセットする方法を理解しようとしています。 [言い換えれば、タブ1を使用していて、ある場所でズームした場合は、タブ2をクリックして同じマップを表示したい場合は、ズームインします。MKMapView setRegion:奇妙な動作ですか?

MKCoordinateRegionをApp Delegateに持っていて、各ビューはそれがなくなる前に保存され、新しいビューはそれぞれ表示される前に読み込まれます。

[言い換えれば、私のアプリケーションデリゲートは私のTabBarControllerDelegateで、私はtabBarController:shouldSelectViewController:をオーバーライドしました。この関数は、タブボタンをクリックした後、新しいタブのビューのviewWillAppear:関数が呼び出される前に呼び出されます。したがって、この関数では、現在のビューの領域をAppDelegate :: MKCoordinateRegion変数に保存し、新しいビューのviewWillAppear:(またはviewDidLoad :)が呼び出されるようにします。次に、新しいビューのviewWillAppear:では、アプリケーションデリゲートからMKCoordinateRegionを取得し、それを新しいビューの領域に割り当てます。]

ただし、MKMapKit :: setRegionの値は一貫していないようです。私はちょうど何かが間違っているか、完全に何かを逃しているのか、何か他のことが起こっているのかどうかは分かりません。

私はiPhone devを行っています。 (Obj-cなど)を2週間ほど使っていますので、これはnewbyの誤りかもしれません。もしそうなら、私は先に進み、あなたの時間を無駄にして今や謝罪します。それにもかかわらず、ここでいくつかのコードは(と私はあなたがそれを自分で実行してみてください念のために...私が使っていたコメントに残してきた)です:

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(@"*********************"); 
} 
... 

、I RU私が最初に気づきますこれは、初期ビューのviewDidLoad:が呼び出されると、デリゲートのリージョンと新しいビューのリージョンは同じです(setRegionをコールした後:)。しかし、setRegion:がviewWillAppear:で呼び出された後、私のビューのMKMapViewの領域はもはやデリゲートと等しくなりません。何とかして、viewDidLoadの全く同じコードは、viewWillAppearで実行したときに別の結果を生成します。そうではないはずですか? setRegion:をviewWillAppearから呼び出すのはなぜですか?私のビューのMKMapViewの領域の値を変更しますか?これはregionThatFitsと何か関係がありますか:内部的に呼び出されていますか?理解できません!

しかし、私がその部分を理解することができれば、私は他の問題を理解することができるかもしれません。だから、ここでは詳しくは触れません。しかし、もしあなたが簡単なアプリを作れば。これらの機能を使用して実行すると、問題が表示されます。主に、ビューのマップは時折同期されますが、ズームして移動すると(タブ間で前後に切り替えながら)、スパンが縮小し始め、新しいタブを押すたびにズームが遠くになります。

注:これはシミュレータ(3.0)でのみ実行しています。[私はまだハードウェアを持っていません]ので、それが何か関係があるかどうかはわかりません。しかし、私は疑いがあります。

とにかく、ありがとうございました。ほんとうにありがとう。

答えて

3

ポイントの周りにズームインフィット機能を実装するのと同様の問題に取り組んでいます。現時点で私があなたに与えることができる唯一の助けは、はい、regionThatFits:は実際に内部的に呼ばれています。私はすべての点に正確にフィットする領域を持つsetRegionを呼び出しています。そのため、可視領域内の点とともに、各辺の端に正確に位置する点があるはずです。 setRegion呼び出し後にmapviewが実際に持っている領域は異なりますが、regionThatFits:を使用したい領域で呼び出した場合に得られる領域とまったく同じです。この地域は通常、私が望むものよりやや大きめです。

私はregionThatFitsが単にアスペクト比をキャプチャしようとしているという理論の下で、アスペクト比補正をバングラしています。しかし、これが唯一のことではないことを示すものが2つあります。まず、regionThatFitsによって返される領域のアスペクト比は、ではなく、で、マップビューのアスペクト比と同じです。次に、自分のアスペクト修正コードを使って、regionThatFitsが出力しているのと同じアスペクト比を得ると、regionThatFits(およびmapView)から誤って大きな領域が得られます。

これまでのところ、MapKitはかなり怒っています。