NSNotification for TabBarControllers

645 Views Asked by At

I've a BaseController (BVC) where I've started to listen to notifications in viewWillAppear & stop listening to them at viewWillDisappear methods.

BVC
- CustomView (Notification received updates this VC) 

Now I've subclassed 4 different controllers from BVC.

BVC
|-- FirstVC (FVC)
|-- SecondVC (SVC)
|-- ThirdVC (TVC)

Now, I've one TarBarController which has these 3 VC's as its items via a NavigationViewController (NVC)

TabBarController
|- NVC->FVC
|- NVC->SVC
|- NVC->TVC

My problem is, I've sending across a playWillBegin notification from a Singleton object which is a AVPlayer instance. Notification is received from the VC which is active on top, but if I dont switch tabs quickly the notifications are not received in the other controllers.

I also read in SO, that other VC are not instantiated and that is reason the notifications are not received. But I can't use init method in VC because its asking me to use initWithaCoder

My project is all code and does not use storyboard etc. So my TabBarController is a swift class, has the items there and TabBarController instance in AppDelegate

EDIT 1: Relevant SO QA links Link1 Link2

EDIT 2: To clarify why I'm not being able to add anything in init() method of VC BaseViewController - Snippet

init() {
    super.init() //Please refer image for error message
}

required init?(coder aDecoder: NSCoder) {

}

TabBarController

class TabBarController: UITabBarController, UITabBarControllerDelegate {

    override func viewDidLoad() {

        tabBar.tintColor = UIColor.whiteColor()
        self.delegate = self

        let FNVC = UINavigationController(rootViewController: FVC())
        //Other initialization code 

        let SNVC = UINavigationController(rootViewController: SVC())
        //Other initialization code 

        let TNVC = UINavigationController(rootViewController: TVC())
        //Other initialization code 
        viewControllers = [FNVC, SNVC, TNVC]

     } 
}

enter image description here

1

There are 1 best solutions below

8
On BEST ANSWER

The first reason for the notification not being received by non-active tabs is that invisible controllers are unsubscribed. When you're switching, say, from tab #2 (SecondVC) to tab #1 (FirstVC), the -[viewWillDisappear:] is called for the SecondVC and it stops listening thus that controller won't receive notifications anymore.

I propose to start listening in -[viewDidLoad:] and stop in -[dealloc] (or deinit in Swift):

Obj-C:

- (void) viewDidLoad
{
    [super viewDidLoad];
    [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(someFunction:) name:@"notification_name" object:nil];
}

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

Swift:

override func viewDidLoad() {
    super.viewDidLoad()
    NotificationCenter.default.addObserver(self, selector: #selector(BaseController.someFunction), name:Notification.Name("foobar"), object: nil)
}

deinit {
    NotificationCenter.default.removeObserver(self)
}

The second reason is that view might not be loaded yet, therefore [-viewDidLoad:] is not called and the corresponding view controller is not subscribed. To solve this you may force load VC's view when you are creating you view controllers for tabs, e.g.:

Obj-C:

FirstVC *vc1 = [FirstVC new];
[vc1 view];    // loading view, subscribing

Swift:

let vc1 = FirstVC()
let _ = vc1.view

UPDATE

Assuming that you can use initializers (as explained by @danh in the comments below), it is better to subscribe in the init method instead of viewDidLoad. In this case there is no need to force view loading and you can omit [vc view] calls:

Obj-C:

- (instancetype) initWithNibName:(NSString *)nibNameOrNil bundle:(NSBundle *)nibBundleOrNil
{
    if ((self = [super initWithNibName:nibNameOrNil bundle:nibBundleOrNil])){
        [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(notified:) name:@"foobar" object:nil];
    }

    return self;
}

Swift:

override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: Bundle?) {
    super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
    NotificationCenter.default.addObserver(self, selector: #selector(BaseViewController.someFunc), name:Notification.Name("foobar"), object: nil)
}