从分离的 UITableView 推送/呈现 ViewController

Posted

技术标签:

【中文标题】从分离的 UITableView 推送/呈现 ViewController【英文标题】:Pushing/Presenting ViewController from Separated UITableView 【发布时间】:2017-10-22 14:51:50 【问题描述】:

我有一个 ticketVC,它有一个 segmentedControl,它可以切换两个 tableView(一个即将到来的 TableView 和一个 pastTransactionsTableView)。这两个 tableView 是在不同的文件上创建的。我想从 tableViews 推送/呈现一个新的 VC,但不幸的是我无法从这些 tableViews 访问 navigationController 或 presentVC。

我的做法是这样的:

class TicketsViewController: UIViewController 

   let segmentedControllerView: SegmentedController = 
        let sc = SegmentedController()
        sc.translatesAutoresizingMaskIntoConstraints = false
        sc.segmentedController.addTarget(self, action: #selector(segmentedControlValueChanged), for: .valueChanged)
        sc.segmentedController.selectedSegmentIndex = 0
        return sc
    ()

    let upcomingTableView: UpcomingTableView = 
        let tv = UpcomingTableView(frame: .zero, style: .grouped)
        tv.translatesAutoresizingMaskIntoConstraints = false
        return tv
    ()

    let pastTransactionsTableView: PastTransactionsTableView = 
        let tv = PastTransactionsTableView(frame: .zero, style: .grouped)
        tv.translatesAutoresizingMaskIntoConstraints = false
        return tv
    ()

    override func viewDidLoad() 
        super.viewDidLoad()

        //setupNavigationBar()
        //setupViews()
    

    @objc func segmentedControlValueChanged(_ sender: UISegmentedControl) 
        let segmentedControl = sender

        if segmentedControl.selectedSegmentIndex == 0 
            upcomingTableView.isHidden = false
            pastTransactionsTableView.isHidden = true
         else 
            upcomingTableView.isHidden = true
            pastTransactionsTableView.isHidden = false
        
    

//Both UpcomingTableView and PastTransactionsTableView
//are created similar as below
class UpcomingTableView: UITableView 

    override init(frame: CGRect, style: UITableViewStyle) 
        super.init(frame: frame, style: style)

        delegate = self
        dataSource = self

        register(CustomCell.self, forCellReuseIdentifier: "cell")

    

    required init?(coder aDecoder: NSCoder) 
        fatalError("init(coder:) has not been implemented")
    


extension UpcomingTableView: UITableViewDataSource, UITableViewDelegate 
    //all the neccessary dataSource and delegate functions

     func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) 
        let vc = TicketsViewController()
        vc.present(UIViewController(), animated: true, completion: nil)
        //This method cannot access TicketsVC
    

我采用这种方法的原因是,即将到来的表视图和过去的事务表视图都显示了非常不同的单元格并提取了单独的模型。有人对我如何从这些 tableViews 本身推送/呈现新的 VC 有任何建议吗?

【问题讨论】:

只需使用 didselect 方法并在其中添加代码以呈现新的 VC @HarshalBhavsar 我无法通过didSelect 方法访问navigationController?.pushViewController()present 检查 self.presenting 视图 @HarshalBhavsar 我已经查过了,方法不可访问。 你应该可以使用它你是如何使用它的请粘贴代码 【参考方案1】:

如果我理解正确,我会这样做: 在 ticketsVC 中创建一个分段控件并将其放置在顶部。然后在下面添加一个简单的 UIView,我们将使用它作为表格的容器。它在 Storyboard 中应该是这样的:

我更喜欢将 UITableView 保留在 UIViewController 中,因为我里面总是有其他东西。 现在创建两个 UIViewController 并在每个中添加一个 UITableView 并执行与这两个 UIViewController 中的表相关的所有操作。

    class SampleOneVC: UIViewController, UITableViewDelegate, UITableViewDataSource 
// Your code for table 1


    class SampleTwoVC: UIViewController, UITableViewDelegate, UITableViewDataSource 
// Your code for table 2

为 UIViewController 创建一个扩展:

extension UIViewController 
    func configureChildViewController(childController: UIViewController, onView: UIView?) 
        var holderView = self.view
        if let onView = onView 
            holderView = onView
        
        addChildViewController(childController)
        holderView!.addSubview(childController.view)
        constrainViewEqual(holderView: holderView!, view: childController.view)
        childController.didMove(toParentViewController: self)
    


    func constrainViewEqual(holderView: UIView, view: UIView) 
        view.translatesAutoresizingMaskIntoConstraints = false

        let pinTop = NSLayoutConstraint(item: view, attribute: .top, relatedBy: .equal, toItem: holderView, attribute: .top, multiplier: 1.0, constant: 0)
        let pinBottom = NSLayoutConstraint(item: view, attribute: .bottom, relatedBy: .equal, toItem: holderView, attribute: .bottom, multiplier: 1.0, constant: 0)
        let pinLeft = NSLayoutConstraint(item: view, attribute: .left, relatedBy: .equal, toItem: holderView, attribute: .left, multiplier: 1.0, constant: 0)
        let pinRight = NSLayoutConstraint(item: view, attribute: .right, relatedBy: .equal, toItem: holderView, attribute: .right, multiplier: 1.0, constant: 0)

        holderView.addConstraints([pinTop, pinBottom, pinLeft, pinRight])
    

现在回到 ticketVC 并声明变量

var sampleOneVC: UIViewController!
var sampleTwoVC: UIViewController!

在 ViewDidLoad 中初始化它们

// Don't forget to set the IDs in storyboard (Identity Inspector tab) for view controllers 
// field: 'Storyboard ID' 
// values: SampleOneVC & SampleTwoVC
sampleOneVC = mainStoryboard.instantiateViewController(withIdentifier: "SampleOneVC") as! SampleOneVC
sampleTwoVC = mainStoryboard.instantiateViewController(withIdentifier: "SampleTwoVC") as! SampleTwoVC

现在在segmentedControlValueChanged 更新显示的视图控制器

self.configureChildViewController(childController: sampleOneVC, onView: myContainerView)

现在您应该可以在 SampleOneVC 和 SampleTwoVC 的 didSelectRowAt 中使用 segues。我没有编写整个代码,但我希望您知道如何以另一种方式完成它。

【讨论】:

谢谢!我创建了一个单独的项目来测试您的代码,它完全按照我的意愿工作。 只有一个问题,请问有没有必要做childController.didMove(toParentViewController: self)。从功能上讲,即使没有这条线,它也能正常工作。我只是想了解目的。 我在将 viewController 添加到容器时总是使用它。老实说,没有那个就没有尝试过。 :) Here 是文档,因此您可以更好地理解它。

以上是关于从分离的 UITableView 推送/呈现 ViewController的主要内容,如果未能解决你的问题,请参考以下文章

有啥方法可以从 UITableView 索引中呈现 UIPopover 吗?

从 UIPopoverController 内的 UITableView 中的一行推送 viewController

如何从 UITableView 推送 UIViewController

初学者:如何以编程方式从另一个视图呈现 UITableView

从 iOS 中呈现的视图控制器推送视图

Objective-C:从AppsDelegate呈现/推送ViewController后后退按钮消失