处理容器控制器中子视图控制器的UINavigationItem更改的最佳实践?

Rei*_*ain 23 cocoa-touch uinavigationcontroller uinavigationitem ios

假设我有一个容器控制器,它接受一个UIViewControllers数组并将它们放置出去,这样用户就可以左右滑动以在它们之间进行转换.此容器控制器包含在导航控制器内,并成为应用程序主窗口的根视图控制器.

每个子控制器向API发出请求并加载在表视图中显示的项列表.根据显示的项目,可以在导航栏中添加一个按钮,允许用户对表格视图中的所有项目进行操作.

因为UINavigationController仅使用其子视图控制器的UINavigationItems,所以容器控制器需要更新其UINavigationItem以与其子视图的UINavigationItem同步.

容器控制器需要处理两种情况:

  1. 容器控制器的选定视图控制器发生更改,因此容器控制器的UINavigationItem应更新自身以模仿所选视图控制器的UINavigationItem.
  2. 子控制器更新其UINavigationItem,并且必须使容器控制器知道更改并更新其UINavigationItem以匹配.

我提出的最佳解决方案是:

  1. 在setSelectedViewController:方法中查询所选视图控制器的导航项,并将容器控制器的UINavigationItem的leftBarButtonItems,rightBarButtonItems和title属性更新为与所选视图控制器的UINavigationItem相同.
  2. 在setSelectedViewController方法中,KVO到leftBarButtonItems,rightBarButtonItems和所选视图控制器的UINavigationItem的title属性,每当其中一个属性改变容器控制器的UINavigationItem时.

这是我编写的许多容器控制器的一个反复出现的问题,我似乎找不到任何有关这些问题的文档解决方案.

人们对这个问题有什么解决方案?

Rei*_*ain 6

所以我目前实现的解决方案是在UIViewController上创建一个类别,其中的方法允许您设置该控制器导航项的右侧栏按钮,然后该控制器发布通知,让任何关心的人都知道正确的栏按钮项有被改变了.

在我的容器控制器中,我从当前选择的视图控制器中侦听此通知,并相应地更新容器控制器的导航项.

在我的场景中,容器控制器会覆盖类别中的方法,以便它可以保留已分配给它的右侧栏按钮项的本地副本,如果有任何通知,它会将其右侧栏按钮项与其子项连接起来,然后发送通知只是因为它也在容器控制器内.

这是我正在使用的代码.

的UIViewController + ContainerNavigationItem.h

#import <UIKit/UIKit.h>

extern NSString *const UIViewControllerRightBarButtonItemsChangedNotification;

@interface UIViewController (ContainerNavigationItem)

- (void)setRightBarButtonItems:(NSArray *)rightBarButtonItems;
- (void)setRightBarButtonItem:(UIBarButtonItem *)rightBarButtonItem;

@end
Run Code Online (Sandbox Code Playgroud)

的UIViewController + ContainerNavigationItem.m

#import "UIViewController+ContainerNavigationItem.h"

NSString *const UIViewControllerRightBarButtonItemsChangedNotification = @"UIViewControllerRightBarButtonItemsChangedNotification";

@implementation UIViewController (ContainerNavigationItem)

- (void)setRightBarButtonItems:(NSArray *)rightBarButtonItems
{
    [[self navigationItem] setRightBarButtonItems:rightBarButtonItems];

    NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
    [notificationCenter postNotificationName:UIViewControllerRightBarButtonItemsChangedNotification object:self];
}

- (void)setRightBarButtonItem:(UIBarButtonItem *)rightBarButtonItem
{
    if(rightBarButtonItem != nil)
        [self setRightBarButtonItems:@[ rightBarButtonItem ]];
    else
        [self setRightBarButtonItems:nil];
}

@end
Run Code Online (Sandbox Code Playgroud)

ContainerController.m

- (void)setRightBarButtonItems:(NSArray *)rightBarButtonItems
{
    _rightBarButtonItems = rightBarButtonItems;

    [super setRightBarButtonItems:_rightBarButtonItems];
}

- (void)setSelectedViewController:(UIViewController *)selectedViewController
{
    if(_selectedViewController != selectedViewController)
    {
        if(_selectedViewController != nil)
        {
            // Stop listening for right bar button item changed notification on the view controller.
            NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
            [notificationCenter removeObserver:self name:UIViewControllerRightBarButtonItemsChangedNotification object:_selectedViewController];
        }

        _selectedViewController = selectedViewController;

        if(_selectedViewController != nil)
        {
            // Listen for right bar button item changed notification on the view controller.
            NSNotificationCenter *notificationCenter = [NSNotificationCenter defaultCenter];
            [notificationCenter addObserver:self selector:@selector(_childRightBarButtonItemsChanged) name:UIViewControllerRightBarButtonItemsChangedNotification object:_selectedViewController];
        }
    }
}

- (void)_childRightBarButtonItemsChanged
{
    NSArray *childRightBarButtonItems = [[_selectedViewController navigationItem] rightBarButtonItems];

    NSMutableArray *rightBarButtonItems = [NSMutableArray arrayWithArray:_rightBarButtonItems];
    [rightBarButtonItems addObjectsFromArray:childRightBarButtonItems];

    [super setRightBarButtonItems:rightBarButtonItems];
}
Run Code Online (Sandbox Code Playgroud)


tom*_*cca 5

我知道这个问题很旧,但我认为我找到了这个问题的解决方案!

navigationItema 的属性在头文件UIViewController的类别/扩展中定义UINavigationController

该属性定义为:

open var navigationItem: UINavigationItem { get } 
Run Code Online (Sandbox Code Playgroud)

因此,正如我刚刚发现的,您可以覆盖容器视图控制器中的属性,在我的例子中:

public override var navigationItem: UINavigationItem {
    return child?.navigationItem ?? super.navigationItem
}
Run Code Online (Sandbox Code Playgroud)

我尝试过这种方法,它对我有用。所有按钮、标题和视图都会在所包含的视图控制器上发生变化时显示和更新。