为什么我不能在 UIPageViewController 中设置视图控制器?

Why can't I set view controllers in UIPageViewController?

我正在尝试使用从 API 中提取的 5 个事件来填充页面视图控制器。这个想法是用户可以翻阅前 5 个最新事件并单击它们以查看有关该事件的更多详细信息。我 运行 遇到的问题是,一旦我获得事件,我想为页面视图控制器重新加载视图控制器,但是当我尝试设置视图控制器时,我收到错误:fatal error: unexpectedly found nil while unwrapping an Optional value。我为我的事件数组设置了一个 属性 观察器,以便在设置后重新加载视图控制器。这就是我的 class 的样子:

class EventPageViewController: UIPageViewController {

var eventViewControllers = [EventViewController]()

var events: [Event] = []{
    didSet {
        self.reloadViewControllers()
    }
}


func reloadViewControllers(){

    self.dataSource = nil
    self.dataSource = self
    // Get top 5 events
    let topFiveEvents = Array(events.prefix(5))
    print(self.events.count)
    self.eventViewControllers.removeAll()
    print(eventViewControllers.count)

    // Set up view controllers
    for event in topFiveEvents{
        if let controller = self.storyboard?.instantiateViewController(withIdentifier: "event") as? EventViewController{
            print(event.title)
            // Just this line alone is what crashes the app
            controller.eventTitle.text = event.title
            self.eventViewControllers.append(controller)
        }
    }

    if self.eventViewControllers.count != 0 {
        let first = [self.eventViewControllers[0]]
        self.setViewControllers(first, direction: .forward, animated: false, completion: nil)

    }


}

override func viewDidLoad() {
    super.viewDidLoad()
    let timeMin = GTLRDateTime(date: Date()).rfc3339String
    let params = ["maxResults": "250",
                  "singleEvents": "true",
                  "timeMin": timeMin]
    NetworkManager.events(forPark: .all, withParameters: params, query: nil) { (events, error) in
        if error == nil {
            DispatchQueue.main.async {
                self.events = events
            }
        }
    }
}

override func viewDidLayoutSubviews() {
    for subView in self.view.subviews {
        if subView is UIScrollView {
            subView.frame = self.view.bounds
        } else if subView is UIPageControl {
            let pageControl = subView as! UIPageControl
            pageControl.currentPageIndicatorTintColor = UIColor(red:0.00, green:0.15, blue:0.29, alpha:1.0)
            pageControl.pageIndicatorTintColor = UIColor(red:0.00, green:0.15, blue:0.29, alpha:0.30)
            self.view.bringSubview(toFront: subView)


        }
    }
    super.viewDidLayoutSubviews()
}



}

如果我没有在视图控制器中设置任何内容,那么我将让页面视图控制器显示带有静态信息的视图控制器。我不确定我做错了什么。

在我看来,您的页面视图控制器正试图直接操作其子视图控制器的视图。不要那样做。

这一行:

controller.eventTitle.text = event.title

可能是原因,假设 eventTitle 是一个插座。

不要这样做,而是向您的子视图控制器添加一个字符串 属性:

public var eventTitleString: String

然后设置为:

controller.eventTitleString = event.title

然后在您的子视图控制器中,添加代码以将该值复制到您的字段中:

func viewWillAppear(_: animated: Bool) {
  super.viewWillAppear(animated)
  eventTitle.text = eventTitleString
}

@MrSaturn 建议调用 controller.loadViewIfNeeded(),这可能会解决您的崩溃问题,但这是不好的做法。您应该将视图控制器的视图视为私有视图,并且永远不要尝试从外部操作它们。而是添加 public 属性 或直接更新视图控制器视图的函数。