为什么我不能在 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 属性 或直接更新视图控制器视图的函数。
我正在尝试使用从 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 属性 或直接更新视图控制器视图的函数。