每个 View Controller 中应用的自定义导航栏图像,不符合 NSUserDefaults 同步

Posted

技术标签:

【中文标题】每个 View Controller 中应用的自定义导航栏图像,不符合 NSUserDefaults 同步【英文标题】:Custom Navigation Bar Image applied in each View Controller, not conforming to NSUserDefaults Synchronisation 【发布时间】:2014-03-19 11:14:24 【问题描述】:

我有一个选项卡栏控制器,由 3 个选项卡式表视图控制器(时间轴、事件和设置)组成,我正在使用主题。

如果用户点击设置,他们可以点击“主题”单元格并从各种主题中进行选择。

通过使用 NSUserDefaults,我将主题保存在 SelectThemesTableViewController 的 didSelectCell 中:

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath

    UITableViewCell *selectedCell = [tableView cellForRowAtIndexPath:indexPath];
    self.selectedTheme = selectedCell.textLabel.text;
    [[NSUserDefaults standardUserDefaults] setObject:self.selectedTheme forKey:@"Theme"];
    [[NSUserDefaults standardUserDefaults] synchronize];

在时间轴和事件视图控制器中,我在 viewWillAppear 中设置背景、导航栏图像和标签栏图像,具体取决于用户选择的单元格,如下所示:

- (void)viewWillAppear:(BOOL)animated

    [super viewWillAppear:animated];
    self.selectedTheme = [[NSUserDefaults standardUserDefaults] objectForKey:@"Theme"];

    if ([self.selectedTheme isEqualToString:@"Blur"])
    
        // View Background
        UIImageView *backgroundImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"Purplepink.png"]];
        self.tableView.backgroundView = backgroundImageView;

        // Navigation Bar
        UIImage *navBackgroundImage = [UIImage imageNamed:@"Purplepinknav.png"];
        [[UINavigationBar appearance] setBackgroundImage:navBackgroundImage forBarMetrics:UIBarMetricsDefault];

        // Tab Bar
        UIImage *tabBackground = [[UIImage imageNamed:@"SolidPurple.png"]
                                  resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 0)];
        [self.tabBarController.tabBar setBackgroundImage:tabBackground];

    
    else if ([self.selectedTheme isEqualToString:@"Red Slanted"])
    
        // View Background
        UIImageView *backgroundImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"RedSlanted.png"]];
        self.tableView.backgroundView = backgroundImageView;

        // Navigation Bar
        UIImage *navBackgroundImage = [UIImage imageNamed:@"RedSlantedNav.png"];
        [[UINavigationBar appearance] setBackgroundImage:navBackgroundImage forBarMetrics:UIBarMetricsDefault];

        // Tab Bar
        UIImage *tabBackground = [[UIImage imageNamed:@"RedSlantedTab.png"]
                                  resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 0)];
        [self.tabBarController.tabBar setBackgroundImage:tabBackground];

    

    else if ([self.selectedTheme isEqualToString:@"Twirl"])
    
        // View Background
        UIImageView *backgroundImageView = [[UIImageView alloc] initWithImage:[UIImage imageNamed:@"Twirl.png"]];
        self.tableView.backgroundView = backgroundImageView;

        // Navigation Bar
        UIImage *navBackgroundImage = [UIImage imageNamed:@"ReddishBlackNav.png"];
        [[UINavigationBar appearance] setBackgroundImage:navBackgroundImage forBarMetrics:UIBarMetricsDefault];

        // Tab Bar
        UIImage *tabBackground = [[UIImage imageNamed:@"ReddishBlackTab.png"]
                                  resizableImageWithCapInsets:UIEdgeInsetsMake(0, 0, 0, 0)];
        [self.tabBarController.tabBar setBackgroundImage:tabBackground];
    

当我在 SelectTheme 中设置主题时,它可以工作,当我进入时间轴表视图时,主题完全应用于背景、导航栏和标签栏图像。如果我退出应用程序并重新进入,一切都还在那里。

但是,问题是:如果我从模拟器或设备上的多任务处理中终止应用程序并重新打开我的应用程序,背景和标签栏是我上次设置的,但导航栏会返回应用程序委托中应用的默认值。如果我更改为另一个选项卡并返回时间轴,导航栏会更改为之前选择的导航,但问题是加载默认导航栏而不是选定的初始视图,如果我更改选项卡并返回时间轴选项卡,然后再次显示。

我在 viewDidDisappear 中有:

- (void)viewWillDisappear:(BOOL)animated

    UIImage *navBackgroundImage = [UIImage imageNamed:@"purplynav.png"];
    [[UINavigationBar appearance] setBackgroundImage:navBackgroundImage forBarMetrics:UIBarMetricsDefault];
    UIImage *tabBackgroundImage = [UIImage imageNamed:@"purplytab.png"];
    [[UITabBar appearance] setBackgroundImage:tabBackgroundImage];


如果我删除这个 viewDidAppear 代码,同样的事情也会发生。 我错过了什么吗?为什么导航栏不符合 NSUserDefaults?

AppDelegate 如下:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions

    NSLog(@"Is this getting called?");
    [UIApplication sharedApplication].statusBarStyle = UIStatusBarStyleLightContent;
    UIImage *navBackgroundImage = [UIImage imageNamed:@"purplynav.png"];
    [[UINavigationBar appearance] setBackgroundImage:navBackgroundImage forBarMetrics:UIBarMetricsDefault];
    UIImage *tabBackgroundImage = [UIImage imageNamed:@"purplytab.png"];
    [[UITabBar appearance] setBackgroundImage:tabBackgroundImage];
    ...

编辑:我根据评论添加了一些日志记录。 AppDelegate 中的 didFinishLaunchingWithOptions 中的 NSLog 总是首先运行。在此之后,时间轴表视图(初始视图控制器)的 viewWillAppear 被调用,其中包含 NSLog。

这解释了为什么导航栏在应用程序从多任务处理中终止(或几个小时未打开)后第一次启动时会返回默认值。但是,我不一定需要在 AppDelegate 中设置代码。起初,该应用程序没有主题,因此设置 AppDelegate 的导航栏很重要。 However now, I want to start the app off with a theme, but when a new theme is selected, to NOT push forward the App Delegate's Nav Bar the very first time the app is run after being killed.

对此的任何帮助将不胜感激。

【问题讨论】:

应用委托中调用的默认设置在哪里?你在用故事板吗?添加日志并检查您的代码运行的顺序... 感谢@Wain - 默认设置在 App Delegate 的 didFinishLaunchingWithOptions 中被调用。我在 didFinishLaunchingWithOptions 方法中也有默认的标签栏图像,当从多任务处理中终止应用程序时不会调用该方法。我在这里为整个应用程序使用所有故事板。登录 AppDelegate - 当我启动应用程序时会调用它,但在我关闭应用程序时不会被击中,但是当我从多任务处理中终止应用程序时会被调用。谢谢韦恩 抱歉@Wain,您对这个问题还有什么想法吗? 您是否添加了日志记录并检查调用了哪些代码以及何时调用? 我已将 Logging 中的 cmets 添加到最后的问题中。 AppDelegate 每次都会在 viewWillAppear 之前被调用..这可以解释为什么它会以这种方式工作,但我不知道如何解决这个问题.. 【参考方案1】:

精简并简化您的解决方案,以便您在 2 个地方使用 UIAppearance 代理配置所有基于主题的 UI:

    应用委托 - 使用存储的设置(最好是调用主题管理器) 主题管理器 - 使用新存储的设置

通过这种方式,您的主题代码被包含和组织,并且您始终应用适当的主题(而不是某些默认主题)。

【讨论】:

非常感谢@Wain - 选项 1 在这里绝对是一种享受。

以上是关于每个 View Controller 中应用的自定义导航栏图像,不符合 NSUserDefaults 同步的主要内容,如果未能解决你的问题,请参考以下文章

我可以在 SwiftUI 应用程序中使用 View Controller (CalendarKit) 吗?

MVC模式中M,V,C每个代表意义,并简述在Struts中MVC的表现方式。

iOS - 内存警告会卸载 View Controller 并使应用程序无响应

Model-View-Controller MVC 设计模式

Xcode View Controller - 水平方向问题

iPad View Controller 内存管理