如何覆盖init方法并在swift中返回一个storyboard实例

时间:2016-01-02 16:26:27

标签: ios objective-c swift

在Objective-C中,我可以轻松地使用以下代码:

- (instancetype)init {
    UIStoryboard *sb = [UIStoryboard storyboardWithName:@"Main" bundle:nil];
    self = [sb instantiateViewControllerWithIdentifier:@"WMViewController"];
    return self;
}

如何使用Swift实现此功能? 我知道这样做很奇怪,但出于某些原因,我必须初始化一个viewController,如下所示:

let vcClass = WMTableViewController.self
// ...
let vc = vcClass.init()

因此,为了支持storybord / xib,如果可以覆盖init方法并返回另一个实例,那将很容易。

这是我正在做的工作:

我正在尝试将我的小lib(WMPageController)转换为Swift(WMPageController-Swift),然后我被困在这里。如果您有其他建议或解决方案来处理它,我会很高兴。

非常感谢,如果您愿意帮助我。

1 个答案:

答案 0 :(得分:7)

好问题。我过去试图解决同样的问题。

具体解决方案

首先,您可以使用类方法

class WMViewController: UIViewController {
    class func loadFromStoryboard() -> WMViewController? {
        return UIStoryboard(name: "main", bundle: nil).instantiateViewControllerWithIdentifier("WMViewController") as? WMViewController
    }
}

并像这样使用

let controller = WMViewController.loadFromStoryboard()

一般解决方案

然而,将loadFromStoryboard方法添加到您想要“加载”的每个ViewController是一种不好的做法。所以我们可以在协议扩展中移动该代码。

protocol StoryboardLoadable { }
extension StoryboardLoadable where Self:UIViewController {
    private static var identifier: String { return "\(Self.self)" }
    static func loadFromStoryboard() -> Self? {
        return UIStoryboard(name: "main", bundle: nil).instantiateViewControllerWithIdentifier(identifier) as? Self
    }
}

现在,您需要做的就是将loadFromStoryboard方法添加到自定义视图控制器中就是这一行

extension WMViewController: StoryboardLoadable {}

就是这样。现在您可以加载视图控制器编写

let controller = WMViewController.loadFromStoryboard()