macOS - NSViewController

实例化

NSViewController-nib.png

建议创建NSViewController子类时同时生成xib文件,否则在创建对象时需要自行调用-[NSViewController setView:] 后才能使用,假如有nib文件,直接[NSViewController new]即可创建对象

显示方式

1.通过新窗口显示(相当于storyboard里的Show):

@interface ViewController ()
@property (nonatomic, strong) NSWindowController *childWindowController;
@end

@implementation ViewController

- (IBAction)presentViewController:(NSButton *)sender {

    CustomViewController *viewController    = [CustomViewController new];
    NSWindow *window                        = [NSWindow windowWithContentViewController:viewController];
    NSWindowController *windowController    = [[NSWindowController alloc] initWithWindow:window];
    // 需要强引用此对象,否则会被自动释放
    self.childWindowController              = windowController;
    // 相当于 [window makeKeyAndOrderFront:nil];
    [windowController showWindow:nil];
}

- (void)viewDidLoad {
    [super viewDidLoad];
    // 需要监听窗口关闭通知,用于清理强引用NSWindowController对象
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(windowWillClose:) name:NSWindowWillCloseNotification object:nil];
}

- (void)windowWillClose:(NSNotification *)notification {
    if (notification.object == self.childWindowController.window) {
        self.childWindowController = nil;
    }
}
NSViewController-Show.gif

2.通过Sheet弹窗显示(弹窗位置跟随父窗口,父窗口无法响应事件):

[self presentViewControllerAsSheet:viewController];

// 在父视图控制器里关闭弹窗
[self dismissViewController:viewController];

// 在子视图控制器里关闭自身
[self dismissController:nil];

macOS 10.15及之前版本效果(固定在顶部):

NSViewController-Sheet.gif

macOS 11及之后版本效果(居中):

NSViewController-macOS11 Sheet.gif

3.通过Modal窗口显示(其他窗口无法响应事件):

[self presentViewControllerAsModalWindow:viewController];

// 在父视图控制器里关闭弹窗
[self dismissViewController:viewController];

// 在子视图控制器里关闭自身
[self dismissController:nil];

*效果和Show类似,区别在于这种情况下其他窗口无法响应事件,且UI线程被阻塞


NSViewController-Show.gif

4.通过Popover弹窗显示(点击弹窗以外的地方时消失)

- (IBAction)presentViewController:(NSButton *)sender {
    CustomViewController *viewController = [CustomViewController new];
    [self presentViewController:viewController
        asPopoverRelativeToRect:sender.bounds
                         ofView:sender
                  preferredEdge:NSRectEdgeMinY
                       behavior:NSPopoverBehaviorTransient];
    // NSPopoverBehaviorApplicationDefined:点击弹窗以外的地方不会消失
    // NSPopoverBehaviorTransient/NSPopoverBehaviorSemitransient:点击弹窗以外的地方会消失
}

// 在父视图控制器里关闭弹窗
[self dismissViewController:viewController];

// 在子视图控制器里关闭自身
[self dismissController:nil];
NSViewController-Popover.gif

生命周期

-[NSViewController initWithCoder:]
-[NSViewController awakeFromNib]
-[NSViewController loadView]
-[NSViewController awakeFromNib]
-[NSViewController viewDidLoad]
-[NSViewController updateViewConstraints]
-[NSViewController viewWillLayout],-[NSViewController viewDidLayout]
-[NSViewController viewWillAppear]
-[NSViewController updateViewConstraints]
-[NSViewController viewWillLayout],-[NSViewController viewDidLayout]
-[NSViewController viewDidAppear]
-[NSViewController viewWillLayout],-[NSViewController viewDidLayout]
...
-[NSViewController viewWillDisappear]
-[NSViewController viewDidDisappear]
-[NSViewController dealloc]

方法

// 相当于调用了[[CustomViewController alloc] initWithNibName:nil bundle:[NSBundle mainBundle]]
+ (instancetype)new;

// 假如xib文件找不到或里面没有控件,view为nil且不执行-[NSViewController viewDidLoad],需要在-[NSViewController loadView]里设置view
// OSX 10.10 以后 “nibNameOrNil”如果为空,会尝试查找同类名的xib文件
- (instancetype)initWithNibName:(nullable NSNibName)nibNameOrNil bundle:(nullable NSBundle *)nibBundleOrNil NS_DESIGNATED_INITIALIZER;

// 通过xib、storyboard文件初始化的
- (nullable instancetype)initWithCoder:(NSCoder *)coder NS_DESIGNATED_INITIALIZER;

// 通过-[NSViewController initWithNibName:bundle:]创建的示例才有值
@property (nullable, copy, readonly) NSNibName nibName;
@property (nullable, strong, readonly) NSBundle *nibBundle;

// 可理解为默认的模型对象,比如-[NSCollectionView setContent:] 会通过此属性把数组里的对象依次传递给-[NSCollectionView itemPrototype] 里的NSCollectionViewItem实例
@property (nullable, strong) id representedObject;

// 标题,并不会显示出来,如果需要设置窗口标题,可使用self.view.window.title(self为NSViewController实例)
@property (nullable, copy) NSString *title;

// 假如未设置,会调用[self loadView]通过xib、storyboard文件生成view,假如不是通过xib、storyboard初始化的需要自己设置view属性,设置了之后会调用 -viewDidLoad 方法
@property (strong) IBOutlet NSView *view;

// 不建议直接调用此方法,可通过-[NSViewController view]间接调用
// 默认通过[self nibName]和[self nibBundle]生成view,假如没有xib文件,就需要自己设置view属性
// OSX 10.10之后[self nibName]假如为nil会尝试查找同类名的xib文件
- (void)loadView;


// 视图已加载,一般在此初始化属性,注意此时还没有窗口(self.view.window)
- (void)viewDidLoad API_AVAILABLE(macos(10.10));
// 视图已加载
@property (readonly, getter=isViewLoaded) BOOL viewLoaded API_AVAILABLE(macos(10.10));
// view添加到superview之前调用
- (void)viewWillAppear API_AVAILABLE(macos(10.10));
// view添加到superview之后调用,此时才有窗口(self.view.window)
- (void)viewWillAppear API_AVAILABLE(macos(10.10));

// 视图(self.view)被隐藏或移除、所在的窗口(self.view.window)被关闭或最小化时调用,注意窗口后置或被其他窗口覆盖时不会调用
- (void)viewWillDisappear API_AVAILABLE(macos(10.10));
- (void)viewDidDisappear API_AVAILABLE(macos(10.10));

// 预设的视图大小,有些控件会调用此方法设置view的frame
@property NSSize preferredContentSize API_AVAILABLE(macos(10.10));
// 更新视图约束
- (void)updateViewConstraints API_AVAILABLE(macos(10.10));
// 布局
- (void)viewWillLayout API_AVAILABLE(macos(10.10));
- (void)viewDidLayout API_AVAILABLE(macos(10.10));

...
// 父视图控制器
@property (nullable, readonly) NSViewController *parentViewController API_AVAILABLE(macos(10.10));
// 子视图控制器
@property (copy) NSArray<__kindof NSViewController *> *childViewControllers API_AVAILABLE(macos(10.10));
// 添加子视图控制器
- (void)addChildViewController:(NSViewController *)childViewController API_AVAILABLE(macos(10.10));
// 从父视图控制器移除自身
- (void)removeFromParentViewController API_AVAILABLE(macos(10.10));
// 插入子视图控制器
- (void)insertChildViewController:(NSViewController *)childViewController atIndex:(NSInteger)index API_AVAILABLE(macos(10.10));
// 移除子视图控制器
- (void)removeChildViewControllerAtIndex:(NSInteger)index API_AVAILABLE(macos(10.10));
// 子视图控制器preferredContentSize已更改
- (void)preferredContentSizeDidChangeForViewController:(NSViewController *)viewController API_AVAILABLE(macos(10.10));
- (void)viewWillTransitionToSize:(NSSize)newSize API_AVAILABLE(macos(10.10));

// 通过storyboard文件创建的才有
@property(nullable, readonly, strong) NSStoryboard *storyboard API_AVAILABLE(macos(10.10));
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353

推荐阅读更多精彩内容