使用泛型加载带有nib的视图控制器

3

目前我使用以下代码加载相同nibname的视图控制器

let recommendationVC : RecommendationVC = RecommendationVC(nibName: "RecommendationVC", bundle: nil)

我认为指定nibname是不必要的,因为它与控制器名称相同。因此,我决定使用泛型,并使用泛型推断类型和nib名称。

protocol NibIdentifiable {
    static var nibNameIdentifier: String { get }
}

// MARK: - Indentifies each storyboard from its classname.
extension NibIdentifiable where Self: UIViewController {
    static var nibNameIdentifier: String {
        return String(describing: self)
    }
}
extension UIViewController :NibIdentifiable
{

}

extension UIViewController {
    func instantiate<Controller: UIViewController>(_: Controller.Type) -> Controller where Controller: NibIdentifiable {

        guard let controller = Self(nibName:Controller.nibNameIdentifier,bundle:nil) as? Controller else {
            fatalError("Could not dequeue cell with identifier: \(Controller.nibNameIdentifier)")
        }

        return controller
    }
}

但是在尝试创建VC实例时,

 let recommendationVC :RecommendationVC = UIViewController.instantiate()

收到错误信息“无法推断出通用参数'Controller'”

这种方法有什么问题吗?


让我们尝试像这样编写代码:let recommendationVC: RecommendationVC = RecommendationVC.instantiate() - Konstantin Kryzhanovsky
@KonstantinKryzhanovsky 我尝试过了,遇到了同样的问题。 - Deekshith Bellare
1
当nib名称与类名相同时,甚至无需指定它。只需执行let recommendationVC = RecommendationVC()即可正常工作。 - dan
你可以在这里查看我的答案:https://dev59.com/ul4d5IYBdhLWcg3wLf0g#44215068 - Ankit Srivastava
2个回答

9

添加UIViewController的扩展

extension UIViewController {
    static func instantiateFromNib() -> Self {
        func instantiateFromNib<T: UIViewController>(_ viewType: T.Type) -> T {
            return T.init(nibName: String(describing: T.self), bundle: nil)
        }

        return instantiateFromNib(self)
    }
}

然后就像这样使用

让myViewController等于从nib文件中实例化的MyViewController


2
谢谢!这对我有用,尽管我没有使用内部函数;我直接调用了 return self.init(nibName: String(describing: self), bundle: nil); - hasen

2
class Rec : UIViewController  {
    let r : String = "1231231"
}

protocol NibIdentifiable {
    static var nibNameIdentifier: String { get }
}

// MARK: - Indentifies each storyboard from its classname.
extension NibIdentifiable where Self: UIViewController {
    static var nibNameIdentifier: String {
        return String(describing: self)
    }

    static func instantiate() -> Self {

        guard let controller = Self(nibName:Self.nibNameIdentifier,bundle:nil) as? Self else {
            fatalError("Could not dequeue cell with identifier: \(Self.nibNameIdentifier)")
        }

        return controller
    }

}

extension UIViewController : NibIdentifiable {
}

let x : Rec = Rec.instantiate()

这一定能够正常工作。

在我的情况下,我使用了一些Storyboardable协议,并从特定的故事板中初始化控制器。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接