0

アプリは体の測定を行うためのものです。ユーザーは、脚、腕、首を測定したいと言うことができます。設定タブとメインタブには、ループして各測定を行うビューがあります。
これは次のように達成されます。

私はタブコントローラーを持っています 最初のタブにはナビゲーションコントローラーがあります ストーリーボードの最初のビューコントローラーとそれ自体へのセグエが1つあります ボードはすべての測定値が得られるまでループし、その後別のコントローラーにセグエします

問題は、ユーザーが設定タブで行っている測定値を変更した場合、最初のタブを完全にリロードする必要があることです。これは、アプリがちょうど起動したかのように、ナビゲーション スタック全体をクリアするなどです。

現時点では、タブ コントローラーが測定タブのナビゲーション コントローラーで popToRootViewControllerAnimated を呼び出していますが、これによりクラッシュが発生しています。各画面にはスライダー コントロールがあり、titleForRow:forComponent: への呼び出しが削除されたビューで呼び出され、クラッシュが発生します。

私は何を間違っていますか?!

タブバーコントローラーのコードは次のとおりです

//  TabBarController.m
//

#import "TabBarController.h"
#import "TodaysMeasurementObject.h"
#import "AppDelegateProtocol.h"
#import "AddMeasurementViewController.h"
#import "ReadPerson.h"
#import "AppDelegate.h"

@interface TabBarController () <UITabBarControllerDelegate>



@end

@implementation TabBarController

bool resetWizardView = false;

- (id)initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil];
    if (self) {
    }
    return self;
}

- (void)viewDidLoad
{
    [super viewDidLoad];
    // Do any additional setup after loading the view.
    self.delegate = self;

    [[UIDevice currentDevice] beginGeneratingDeviceOrientationNotifications];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(orientationChanged:)
                                                 name:UIDeviceOrientationDidChangeNotification object:nil];

    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(measurementsSettingsUpdated:) name:@"MeasurementsSettingsUpdated" object:nil];

}

- (void) measurementsSettingsUpdated:(NSNotification *) notification
{
//    UINavigationController *navigationController = [self.viewControllers objectAtIndex:0];
//    AddMeasurementViewController *addMeasurement = [[AddMeasurementViewController alloc] init];
//    [navigationController setViewControllers: [[NSArray alloc] initWithObjects:addMeasurement, nil]];
    resetWizardView = YES;
}

- (void) viewDidAppear:(BOOL)animated
{
    if (![ReadPerson userHasRecords]) {
        [self setSelectedIndex:3];
    }
}

- (void)orientationChanged:(NSNotification *)notification
{
    // We must add a delay here, otherwise we'll swap in the new view
    // too quickly and we'll get an animation glitch
    [self performSelector:@selector(showGraphs) withObject:nil afterDelay:0];
}

- (void)showGraphs
{
    UIDeviceOrientation deviceOrientation = [UIDevice currentDevice].orientation;
    if (deviceOrientation == UIDeviceOrientationLandscapeLeft && !isShowingLandscapeView)
    {
        [self performSegueWithIdentifier: @"toGraph" sender: self];
        isShowingLandscapeView = YES;
    }

    else if (deviceOrientation != UIDeviceOrientationLandscapeLeft && isShowingLandscapeView)
    {
        [self dismissModalViewControllerAnimated:YES];
        isShowingLandscapeView = NO;
    }
}

- (void)viewDidUnload
{
    [super viewDidUnload];
    // Release any retained subviews of the main view.
}

- (void)dealloc
{
    [[NSNotificationCenter defaultCenter] removeObserver:self];
    [[UIDevice currentDevice] endGeneratingDeviceOrientationNotifications];
}

- (BOOL)shouldAutorotateToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation
{
    if(interfaceOrientation == UIInterfaceOrientationLandscapeRight)
    {
        [self performSegueWithIdentifier: @"toGraph" sender: self];
    }

    return false;
}

- (void)tabBarController:(UITabBarController *)tabBarController didSelectViewController:(UIViewController *)viewController {
    int tbi = tabBarController.selectedIndex;
    if (tbi == 0) {
        [[viewController view] setNeedsDisplay];
        if (resetWizardView) {
            [(UINavigationController*)[self.viewControllers objectAtIndex:0] popToRootViewControllerAnimated: NO]; // ******* POP CALLED HERE ******
            resetWizardView = false;
        }
    }
}

- (TodaysMeasurementObject*) theAppDataObject
{
    id<AppDelegateProtocol> theDelegate = (id<AppDelegateProtocol>) [UIApplication sharedApplication].delegate;
    TodaysMeasurementObject* theDataObject;
    theDataObject = (TodaysMeasurementObject*) theDelegate.theAppDataObject;
    return theDataObject;
}

- (BOOL)shouldAutorotate {
    return NO;
}

- (NSUInteger)supportedInterfaceOrientations {
    return UIInterfaceOrientationMaskPortrait;
}


@end

更新しました

- (void) measurementsSettingsUpdated:(NSNotification *) notification
{
    NSMutableArray *viewControllers = [[NSMutableArray alloc] initWithArray: self.viewControllers];
    UINavigationController *navigationController = [viewControllers objectAtIndex:0];
    AddMeasurementViewController *addMeasurement = [[AddMeasurementViewController alloc] init];
    [navigationController setViewControllers: [[NSArray alloc] initWithObjects:addMeasurement, nil]];

    [viewControllers setObject:navigationController atIndexedSubscript:0];

    self.viewControllers = viewControllers;
}

- tabBarController:didSelectViewController: からコードを削除しました。

それでも同じエラー。問題は、ビューが削除された後にスライド コントロールの値を取得しようとしていることだと思います。しかし、ビューの一部はまだ生きている必要があります...? とにかくそれを殺すために?それとも生のまま??

4

2 に答える 2

1

あなたのコードで:

    [[viewController view] setNeedsDisplay];
    if (resetWizardView) {
        [(UINavigationController*)[self.viewControllers objectAtIndex:0] popToRootViewControllerAnimated: NO]; // ******* POP CALLED HERE ******

いくつかのポイント:

  1. おそらく必要なのはsetNeedsDisplaywhenだけですresetWizardView == NO。本当のことを言う必要はまったくないと思います。

  2. 最後の行ではviewControllerなく、ターゲティングしたいと思います。[self.viewControllers objectAtIndex:0]

[(UINavigationController*)viewController popToRootViewControllerAnimated:NO]

于 2012-11-13T15:39:53.480 に答える
0

結局、これを直接修正することをあきらめました。代わりに、View Controller で通知をキャプチャし、1 つのボタンで UIAlertView をポップアップします。ユーザーがこれを押すと、 popToRootViewControllerAnimated が呼び出されます。ユーザーがタブの内容が変更された理由についてのメッセージを受け取るので、これは実際にはより良いユーザーエクスペリエンスでもあると思います.

これにより、将来誰かの時間/ストレスが節約されることを願っています。

于 2012-11-14T13:13:24.877 に答える