如何跳转到 UIPageViewController 中的特定页面,setViewControllers 不起作用

Posted

技术标签:

【中文标题】如何跳转到 UIPageViewController 中的特定页面,setViewControllers 不起作用【英文标题】:How to jump to a specific page in UIPageViewController, setViewControllers not working 【发布时间】:2018-08-05 17:22:37 【问题描述】:

我试图在按下按钮时跳转到特定页面,我制作了一个 ViewController,其中包含 PageViewController 的 ContainerView,这样我就可以在顶部拥有永久按钮。从那里我做:

@IBAction func jump(_ sender: Any) 
    PageViewController().jump()

这是PageViewController中的跳转功能

func jump() 

    var lastCoinNumber:Int = Manager.shared.coins.index(of: last!)!
    lastNumber = lastCoinNumber

    if let jumpView = viewControllerAtIndex(lastNumber, storyboard: self.storyboard!) 
        self.setViewControllers([jumpView], direction: .forward, animated: false, completion: nil)
    

由于某种原因,我总是在跳转函数的 ViewControllerAtIndex 部分出错:

线程 1:致命错误:在展开可选值时意外发现 nil .即使我手动选择一个索引,它也会这样做。

if let firstViewController = viewControllerAtIndex(0, storyboard: self.storyboard!)     
    self.setViewControllers([firstViewController], direction: .forward, animated: false, completion: nil)

以上是我在viewDidLoad() 中初始设置 ViewControllers 的方式,效果很好,如果我基本上在做同样的事情,为什么它不起作用?这是我索引页面和实例化视图控制器的方法,它们使用相同的单个视图控制器,因此为每个页面创建它的新实例,否则我会创建一个视图控制器数组。

func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool)  if !completed  return 
    DispatchQueue.main.async() 
        self.dataSource = nil
        self.dataSource = self
        self.pageControl.numberOfPages = (Manager.shared.coins.count)
    


func viewControllerAtIndex(_ index: Int, storyboard: UIStoryboard) -> TemplateViewController? 

    if Manager.shared.coins.count == 0 || index >= Manager.shared.coins.count 
        return nil
    

    let templateViewController = storyboard.instantiateViewController(withIdentifier: "templateController") as! TemplateViewController

    templateViewController.dataObject = Manager.shared.coins[index]
    return templateViewController


func indexOfViewController(_ viewController: TemplateViewController) -> Int 
    self.pageControl.currentPage = Manager.shared.coins.index(of: viewController.dataObject)!
    return Manager.shared.coins.index(of: viewController.dataObject)!


func pageViewController(_ pageViewController: UIPageViewController, viewControllerBefore viewController: UIViewController) -> UIViewController? 

    var index = self.indexOfViewController(viewController as! TemplateViewController)
    if (index == 0) 
        return nil
    
    index -= 1
    return self.viewControllerAtIndex(index, storyboard: viewController.storyboard!)


func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? 

    var index = self.indexOfViewController(viewController as! TemplateViewController)
    if index == -1 
        return nil
    
    index += 1
    if index == Manager.shared.coins.count 
        return nil
    
    return self.viewControllerAtIndex(index, storyboard: viewController.storyboard!)

我希望这是足够的信息,谢谢。

编辑:

Coin 是一个字符串数组,我声明:

var last = Manager.shared.coins.last
var lastNumber: Int = 0

然后 jump() 负责其余的工作。所以我得到最后一个对象,找到它的索引,然后用最后一个数字的索引执行 setViewControllers,它应该与 viewControllerAtIndex 对应,因为它使用相同的数组来索引和加载页面。

出于某种原因,当我尝试这样做时: var lastCoinNumber:Int = Manager.shared.coins.index(of: last) 它迫使我使用!因为: “可选类型'Array.Index的值?' (又名“可选”)未展开;您的意思是使用“!”还是‘?’?”

我不确定我在这方面做错了什么,这是硬币的声明方式:

var coins = [""]

编辑:完整代码,PageViewController:

import UIKit

var pageView = PageViewController()

class PageViewController: UIPageViewController, UIPageViewControllerDataSource, UIPageViewControllerDelegate 

    //page Control dots
    var pageControl = UIPageControl()
    var last = Manager.shared.coins.last!
    var lastNumber: Int = 0


    var viewControllerArray = [UIViewController()]


    func configurePageControl() 
    // The total number of pages that are available is based on ManagerCoins names array
        pageControl = UIPageControl(frame: CGRect(x: 0,y: 
    UIScreen.main.bounds.maxY - 50,width: UIScreen.main.bounds.width,height: 
    50))
        self.pageControl.numberOfPages = (Manager.shared.coins.count)
        self.pageControl.currentPage = 0
        self.pageControl.tintColor = UIColor.black
        self.pageControl.pageIndicatorTintColor = UIColor.gray
        self.pageControl.currentPageIndicatorTintColor = UIColor.white
        self.view.addSubview(pageControl)
    

    // MARK: Delegate functions
    func pageViewController(_ pageViewController: UIPageViewController, didFinishAnimating finished: Bool, previousViewControllers: [UIViewController], transitionCompleted completed: Bool) 

    

    func viewControllerAtIndex(_ index: Int, storyboard: UIStoryboard) -> TemplateViewController? 

        if Manager.shared.coins.count == 0 || index >= Manager.shared.coins.count 
            return nil
        

        let templateViewController = storyboard.instantiateViewController(withIdentifier: "templateController") as! TemplateViewController

        templateViewController.dataObject = Manager.shared.coins[index]
        return templateViewController
    

    func indexOfViewController(_ viewController: TemplateViewController) -> Int 
        self.pageControl.currentPage = Manager.shared.coins.index(of: viewController.dataObject)!
        return Manager.shared.coins.index(of: viewController.dataObject)!
    

    func pageViewController(_ pageViewController: UIPageViewController, viewControllerBefore viewController: UIViewController) -> UIViewController? 

        var index = self.indexOfViewController(viewController as! TemplateViewController)
        if (index == 0) 
            return nil
        
        index -= 1
        return self.viewControllerAtIndex(index, storyboard: viewController.storyboard!)
    

    func pageViewController(_ pageViewController: UIPageViewController, viewControllerAfter viewController: UIViewController) -> UIViewController? 

        var index = self.indexOfViewController(viewController as! TemplateViewController)
        if index == -1 
            return nil
        
        index += 1
        if index == Manager.shared.coins.count 
            return nil
        
        return self.viewControllerAtIndex(index, storyboard: viewController.storyboard!)
    

    func jump() 

        var lastCoinNumber:Int = Manager.shared.coins.index(of: last!)!
        lastNumber = lastCoinNumber

        if let jumpView = viewControllerAtIndex(lastCoinNumber, storyboard: self.storyboard!)
            self.setViewControllers([jumpView], direction: .forward, animated: false, completion: nil)
        
    

    override func viewDidLoad() 
        super.viewDidLoad()

        self.delegate = self
        configurePageControl()

        self.dataSource = self

        if let firstViewController = viewControllerAtIndex(0, storyboard: self.storyboard!)

            self.setViewControllers([firstViewController], direction: .forward, animated: false, completion: nil)
        
    

编辑:

经过大量研究,这似乎是跳转页面的正确方法,通过调用 setViewControllers,对此答案的第一条评论似乎有同样的问题,它适用于在 viewDidLoad() 上设置视图但试图以编程方式导航到基于索引的页面出现错误。 https://***.com/a/24335008/10058239

【问题讨论】:

【参考方案1】:

我认为它因为强制展开而崩溃:

var lastCoinNumber:Int = Manager.shared.coins.index(of: last!)!

如果硬币是一个集合,您可以使用 .last 来获取最后一个元素,例如:

let numbers = [10, 20, 30, 40, 50]
if let lastNumber = numbers.last 
    print(lastNumber)

// Prints "50" 

【讨论】:

已编辑问题,硬币是一个字符串数组,我只是使用该数组的 .last 来获取正确的数字,以正确调用 viewControllerAtIndex 中的最后一页,如果我更改 setViewControllers 中的数字以使用例如 0 而不是 lastCoinNumber,它仍然不起作用,即使它的写法与 viewDidLoad() 中的完全相同,有什么想法吗? @PeterRuppert 你的代码没有意义。你说的是var lastCoinNumber:Int = Manager.shared.coins.index(of: last!)!,但你没有什么叫last。请向我们展示足够多的代码,以便我们了解您在做什么——尤其是因为您说那行 is 是有问题的行! @Matt 我编辑了问题以显示 var last = Manager.shared.coins.last 所以它正在获取数组中最后一项的索引,并且它确实有效,因为我可以在崩溃后识别数字出于某种原因,即如果数组中有 5 个项目,当我点击它时 lastNumber = 4 @PeterRuppert 好的,但现在以新的方式没有意义。你在哪里这么说?如果在视图控制器的顶部,那么你会得到last once。它不会神奇地自我更新以不断引用最后一个元素。所以当你定义时它是nil,永远是nil。我再次请求您,希望是最后一次,显示您的代码。不是sn-ps。不是你认为相关的。不是单行。您的实际代码,批发粘贴。 @Matt 添加了 pageViewController 的完整代码,因此您现在可以看到所有这些是如何工作的。我考虑到您的担忧并尝试将 last 声明为空字符串,然后在 jump() 函数中执行 last = Manager.shared.coins.last ,以确保在按下按钮后 last 包含正确的项目,但是仍然是完全相同的问题。

以上是关于如何跳转到 UIPageViewController 中的特定页面,setViewControllers 不起作用的主要内容,如果未能解决你的问题,请参考以下文章

如何杀掉 android 如何跳转到 main activity

如何让微信小程序直接跳转到某个网页

Android应用如何跳转到应用市场详情页面

Android应用如何跳转到应用市场详情页面

jsp中如何点击按钮跳转到另一个页面?

下拉后点击信息直接跳转到如何设置