从自定义 UIViewController 导航到 TabBarController 没有 segue

Navigate from a Custom UIViewController to a TabBarController without segue

当我从自定义 UIViewController 导航到 TabBarController 的第二个选项卡时遇到问题,这当然是另一个 UIViewController。 如果我从 AdminPanel 导航到 Products 不会崩溃,因为我有一个从 Products 到 AdminPanel 的 push。但是从 AdminPanel 到 Cart 我没有任何 push

这是一张带有错误的图片:

这是错误:“无法将类型 'MyAppName.AdminPanelViewController' (0x10e878418) 的值转换为 'MyAppName.ProductsViewController' (0x10e877e00)。”

这是我的设计:

这是我的代码:

    // PRODUCTS VC
    class ProductsViewController: UIViewController{

var selectedProductsArray = [Product]()
    var priceForSelectedProductsArray = [Float]()

       // Func which is using GestureRecognition to access the Admin Panel when we press on User Avatar
        func accessToAdminPanel(){

            let tapGesture = UITapGestureRecognizer(target: self, action: #selector(ProductsViewController.adminImageTapped(gesture:)))

            userAvatarImageView.addGestureRecognizer(tapGesture)
        }

        // Function to open the AdminPanelViewController
        @objc func adminImageTapped(gesture: UIGestureRecognizer) {

            let storyBoard : UIStoryboard = UIStoryboard(name: Constants.nameOfMainSB, bundle:nil)
            let adminPanelVC = storyBoard.instantiateViewController(withIdentifier: Constants.adminPanelStoryboard) as! AdminPanelViewController
            self.navigationController?.pushViewController(adminPanelVC, animated: true)

        }


        // Func which will add the product into an array of products when the user press Add To Cart
        func didTapAddToCart(_ cell: ProductTableViewCell) {
            let indexPath = self.productsTableView.indexPath(for: cell)

            addProduct(at: indexPath!)
            selectedProductsArray.append(productsArray[(indexPath?.row)!]) // Append products for cart
            priceForSelectedProductsArray.append(productsArray[(indexPath?.row)!].price) // Append prices for selected products
        }

    }


    // CART VC
    class CartViewController: UIViewController {

 var productsInCartArray = [Product]()
    var productPricesArray = [Float]()

     // Append the selectedProducts into productsInCartArray using the TabBarController
        func fetchSelectedProducts() {

    // ------------------HERE IS CRASHING AT THIS LINE -----------------------
            productsInCartArray = ((self.tabBarController?.viewControllers![0] as! UINavigationController).topViewController as! ProductsViewController).selectedProductsArray
            productPricesArray = ((self.tabBarController?.viewControllers![0] as! UINavigationController).topViewController as! ProductsViewController).priceForSelectedProductsArray
            totalSum = productPricesArray.reduce(0, +)
        }

    }

如果您无法弄清楚这部分代码是哪里崩溃了,我可以给您 link 到 git 源代码,因为我没有什么可隐藏的。

如果您正在阅读本文,非常感谢您抽出宝贵的时间,我希望您能帮助我,因为我花了半天的时间试图找到一个解决方案来修复这个导致我的应用程序崩溃的错误。

topViewController 现在是管理员而不是产品,因此您需要在索引 0 处获取 VC,这是产品

productsInCartArray = ((self.tabBarController?.viewControllers![0] as! UINavigationController).viewControllers[0] as! ProductsViewController).selectedProductsArray

错误告诉您您正试图将 AdminPanelViewController 强制转换为 ProductsViewController(在该行的 topViewController as! ProductsViewController 部分。)这是因为当您按下到 AdminPanelViewController,它被移动到导航堆栈的顶部,替换您假设应该在那里的 ProductsViewController。

有很多方法可以在短期内处理这个错误,比如遍历导航堆栈而不是仅仅抓取 topViewController,就像这样:

guard let navController = self.tabBarController?.viewControllers![0] as? UINavigationController else {
    // handle error
    return
}
for controller in navController.viewControllers {
    if let productsController = controller as? ProductsViewController {
        // do work here
    }
}

但我认为更根本的是,您必须重新考虑访问内容的方式。通常,您不应该走遍导航堆栈来工作。您最好在导航堆栈时在控制器之间传递数据。

此外,您应该尽量避免使用 as! 进行强制转换,而应改用 as?,这样您就可以在应用崩溃的情况下发现转换不起作用的情况。我的个人目标是只在 IBOutlet 声明中使用 !,而不是其他任何地方,但你的里程可能会有所不同。