iosswiftuitableviewuitabbarcontrollerswrevealviewcontroller

UINavigation controller to UITabbarcontroller issue with SWRevealViewController in iOS Swift


In my project i have three tabBar item home, notification and profile. And side menu controller has home, bookings, profile and logout. Side bar menu controller was done by using SWRevealViewController cocopods.

When i navigating side bar menu to home tab bar index was selected correctly and navigating properly. While navigating from bookings it navigates properly but again navigating home app gets crashed. And console output says Could not cast value of type 'UINavigationController' (0x10ef79420) to 'UITabBarController' (0x10ef79970).

Since bookings controller is custom view controller and remaining are tab bar controller. And when navigating to booking screen view controller tab bar should be hide and user tap again menu button and navigating to home or any other controller.

And crashed due to booking controller does not has tab bar index. So how can navigate without crash to custom controller and tabbar controller with selected index item.

Here is my screenshot:

bookings screen controller menu controller where swrevealviewcontroller used

My storyboard screenshot:

enter image description here

Here is the code which i have tried:

func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {

    //        tableView.deselectRow(at: indexPath, animated: true)

    let row = indexPath.row

    if row == 0{


        let tabBarController = revealViewController().frontViewController as! UITabBarController


        let storyboard = UIStoryboard(name: "Home", bundle: nil)

        let obj = storyboard.instantiateViewController(withIdentifier: "HomeViewController") as! HomeViewController

        let navController = UINavigationController.init(rootViewController: obj)
        tabBarController.selectedIndex = (indexPath as NSIndexPath).row
        tabBarController.tabBar.isHidden = false
        self.revealViewController().pushFrontViewController(tabBarController, animated: true)



    } else if row == 1{

        let tabBarController = revealViewController().frontViewController as! UITabBarController

        let storyboard = UIStoryboard(name: "Bookings", bundle: nil)
        let obj = storyboard.instantiateViewController(withIdentifier: "BookingsViewController") as! BookingsViewController
        let navController = UINavigationController.init(rootViewController: obj)
    //            tabBarController.selectedIndex = 1
    //            tabBarController.tabBar.isHidden = false
        self.revealViewController().pushFrontViewController(navController, animated: true)



    } else if row == 2 {

        let tabBarController = revealViewController().frontViewController as! UITabBarController

        let storyboard = UIStoryboard(name: "Profile", bundle: nil)
        let obj = storyboard.instantiateViewController(withIdentifier: "profileViewController") as! profileViewController
        let navController = UINavigationController.init(rootViewController: obj)
        tabBarController.selectedIndex = (indexPath as NSIndexPath).row
        tabBarController.tabBar.isHidden = false
        self.revealViewController().pushFrontViewController(tabBarController, animated: true)


    } else if row == 3 {
        print(indexPath)
        // Log out user from Firebase
        AuthService.signOut(onSuccess: {
            // Present the Sign In VC
     //                PrefsManager.sharedinstance.logoutprefences()
            let storyboard = UIStoryboard(name: "Main", bundle: nil)
            let signInVC = storyboard.instantiateViewController(withIdentifier: "signInViewController")
                        self.present(signInVC, animated: true)

      //                self.navigationController?.pushViewController(signInVC, animated: true)

        }) { (errorMessage) in

            ProgressHUD.showError(errorMessage)

        }



    }


}

Solution

  • Here is Working Code of SWRevealViewController with UINavigationController and UITabBarController (Swift 4)

    Your Storyborad it will be like this you not directly assign UITabBarController to revealViewController().frontViewController need to Use Like this.

    enter image description here

    Code of MenuViewController Like this

    import UIKit
    
    class menuViewController: UIViewController,UITableViewDelegate,UITableViewDataSource {
    
    @IBOutlet weak var tblTableView: UITableView!
    @IBOutlet weak var imgProfile: UIImageView!
    
    var ManuNameArray:Array = [String]()
    var iconArray:Array = [UIImage]()
    
    override func viewDidLoad() {
        super.viewDidLoad()
        ManuNameArray = ["Home","Booking","Profile","Logout"]
        iconArray = [UIImage(named:"home")!,UIImage(named:"message")!,UIImage(named:"map")!,UIImage(named:"setting")!]
    
        imgProfile.layer.borderWidth = 2
        imgProfile.layer.borderColor = UIColor.green.cgColor
        imgProfile.layer.cornerRadius = 50
    
        imgProfile.layer.masksToBounds = false
        imgProfile.clipsToBounds = true 
        // Do any additional setup after loading the view.
    }
    
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return ManuNameArray.count
    
    }
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCell(withIdentifier: "MenuCell", for: indexPath) as! MenuCell
    
        cell.lblMenuname.text! = ManuNameArray[indexPath.row]
        cell.imgIcon.image = iconArray[indexPath.row]
    
        return cell
    }
    
    func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
    
        let revealviewcontroller:SWRevealViewController = self.revealViewController()
    
        let cell:MenuCell = tableView.cellForRow(at: indexPath) as! MenuCell
        print(cell.lblMenuname.text!)
    
        if cell.lblMenuname.text! == "Home"
        {
            print("Home Tapped")
    
            let mainstoryboard:UIStoryboard = UIStoryboard(name: "Main", bundle: nil)
    
            // Here TabbarController is StoryboardID of UITabBarController
            if let tabBarController = mainstoryboard.instantiateViewController(withIdentifier: "TabbarController") as? UITabBarController{
    
                tabBarController.selectedIndex = 0
                revealviewcontroller.pushFrontViewController(tabBarController, animated: true)
            }
        }
        if cell.lblMenuname.text! == "Booking"
        {
            print("message Tapped")
    
            let mainstoryboard:UIStoryboard = UIStoryboard(name: "Main", bundle: nil)
            let newViewcontroller = mainstoryboard.instantiateViewController(withIdentifier: "BookingViewController") as! BookingViewController
            let newFrontController = UINavigationController.init(rootViewController: newViewcontroller)
    
            revealviewcontroller.pushFrontViewController(newFrontController, animated: true)
        }
        if cell.lblMenuname.text! == "Profile"
        {
            let mainstoryboard:UIStoryboard = UIStoryboard(name: "Main", bundle: nil)
    
            if let tabBarController = mainstoryboard.instantiateViewController(withIdentifier: "TabbarController") as? UITabBarController{
    
                tabBarController.selectedIndex = 2
                revealviewcontroller.pushFrontViewController(tabBarController, animated: true)
            }
        }
        if cell.lblMenuname.text! == "Logout"
        {
           print("Logout Tapped")
        }
    }
    }
    

    Code for Other Remaining ViewControllers Same as Following for All Home, Notification, Profile and Booking

    import UIKit
    
    class ProfileViewController: UIViewController {
    
    @IBOutlet weak var btnMenuButton: UIBarButtonItem!
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        if revealViewController() != nil {
    
            btnMenuButton.target = revealViewController()
            btnMenuButton.action = #selector(SWRevealViewController.revealToggle(_:))
    
            view.addGestureRecognizer(self.revealViewController().panGestureRecognizer())
            view.addGestureRecognizer(self.revealViewController().tapGestureRecognizer())
        }
    
      }
    }
    

    Your Output :

    enter image description here enter image description here