Swift - делегирование через Nav Controller

Я собираюсь включить мой полный код в это, но я постараюсь дать указатели, где находятся соответствующие биты. В основном я возвращаюсь к контроллеру представления с Unwind Segue с некоторыми новыми данными. Я успешно использую эти данные в VC 'NBARotoHome', но мне дополнительно необходимо передать некоторые из этих данных через встроенный контроллер Nav в V 'NBARotoTabPager'.

Я пытаюсь сделать это, используя делегат UpdateChildView (в верхней части первого блока кода) и вызывая его метод в loadViewData() (в операторе if в нижней части первого блока).

protocol UpdateChildView : class {
 func updateView()
 func test()
 var playerSelected: Player? { get set }
}



class RotoViewRoundCell: UITableViewCell{

@IBOutlet weak var categoryLabel: UILabel!

}

class RotoViewRoundHeader: UITableViewCell{

}


class NBARotoHome: UIViewController{


@IBOutlet weak var posPaidLabel: UILabel!
@IBOutlet weak var progressIndicator: UIProgressView!
@IBOutlet weak var vsLabel: UILabel!
@IBOutlet weak var fantasyFundsAmountLabel: UILabel!
@IBOutlet weak var fantasyFundsLabel: UILabel!
@IBOutlet weak var playerName: UILabel!
@IBOutlet weak var roundIndicator: UILabel!


var selectedPlayer: Player!

var firstNavController: UINavigationController!

weak var updateChildView : UpdateChildView?

override func viewDidLoad() {
    super.viewDidLoad()

    loadViewData()

    firstNavController = self.navigationController



    let rightBarButton = UIBarButtonItem(title: "Select", style: UIBarButtonItemStyle.plain, target: self, action: #selector(myRightSideBarButtonItemTapped(_:)))
    self.navigationItem.rightBarButtonItem = rightBarButton
    self.title = "NBA Roto"


}


func myRightSideBarButtonItemTapped(_ sender:UIBarButtonItem!){

    performSegue(withIdentifier: "ShowDraft", sender: nil)
}


override func prepare(for segue: UIStoryboardSegue, sender: Any?) {

    if segue.identifier == "ShowDraft" {
        let navVC = segue.destination as? UINavigationController
        let nbaDraftList = navVC?.viewControllers.first as! NBADraftList
        nbaDraftList.mainNavController = firstNavController
    }


    if (segue.identifier == "buyNavControllerChild"){

//            let navVC = segue.destination as? UINavigationController
//            let buyVC = navVC?.viewControllers.first as! NBARotoTabPager
//            buyVC.delegate = self

    }


}


override func viewWillAppear(_ animated: Bool) {
    super.viewWillAppear(animated)


}

@IBAction func prepareForUnwind(segue: UIStoryboardSegue) {



}

func loadViewData(){

    if((selectedPlayer) != nil){
        roundIndicator.text = "Upcoming: " + selectedPlayer.game_time
        playerName.text = selectedPlayer.Name
        vsLabel.text = selectedPlayer.visiting + " @ " + selectedPlayer.home
        fantasyFundsLabel.text = ""
        fantasyFundsAmountLabel.text = ""

          updateChildView?.test()
     //   updateChildView?.playerSelected = selectedPlayer
    //    updateChildView?.updateView()

    }else{
        roundIndicator.text = "Select a Player"
        playerName.text = "No Player Selected"
        vsLabel.text = "--"
        fantasyFundsLabel.text = "Fantasy Funds"
        fantasyFundsAmountLabel.text = "$10,000"

    }

}

}

Поскольку я не смог заставить делегата работать, я поиграл с настройкой его свойства делегата в вышеупомянутом методе "prepare" - "buyVC.delegate = self" - но я получаю "buyVC не имеет члена" делегат, так что это был тупик.

Следующий бит кода - это NBARotoTabPager vc, который встроен в контроллер навигации. По причинам, в которых я больше не уверен, я решил сделать его подклассом NBARotoHome, но это в основном пользовательский пейджер с вкладками, который использует сегментированный элемент управления для переключения между двумя дополнительными vcs.

Самый важный шаг на этом этапе - просто заставить работать функцию "test" (которая просто печатает "test". Она реализована в блоке кода ниже, снизу вверху updateView).

class NBARotoTabPager: NBARotoHome, UpdateChildView{

@IBOutlet weak var segmentedControl: UISegmentedControl!
@IBOutlet weak var scoreKey: UIBarButtonItem!
@IBOutlet weak var standings: UIBarButtonItem!

 var playerSelected: Player?


override func viewDidLoad() {



    navigationController?.navigationBar.barTintColor = UIColor(red: 27/255, green: 27/255, blue: 27/255, alpha: 1)


    scoreKey.setTitleTextAttributes([NSFontAttributeName: UIFont(name: "Helvetica", size: 13.0)!], for: UIControlState.normal)
    scoreKey.tintColor = UIColor.blue

    standings.setTitleTextAttributes([NSFontAttributeName: UIFont(name: "Helvetica", size: 13.0)!], for: UIControlState.normal)
    standings.tintColor = UIColor.blue

    setupView()
}

private func setupView() {
    setupSegmentedControl()
    updateView()
}

private func setupSegmentedControl() {
    // Configure Segmented Control
    segmentedControl.removeAllSegments()
    segmentedControl.insertSegment(withTitle: "Live", at: 0, animated: false)
    segmentedControl.insertSegment(withTitle: "Avg / +", at: 1, animated: false)
    segmentedControl.addTarget(self, action: #selector(selectionDidChange(_:)), for: .valueChanged)
    segmentedControl.selectedSegmentIndex = 0
}

func selectionDidChange(_ sender: UISegmentedControl) {
    updateView()
}


private lazy var viewLiveTab: NBARotoLive = {
    // Load Storyboard
    let storyboard = UIStoryboard(name: "Main", bundle: Bundle.main)

    // Instantiate View Controller
    var viewController = storyboard.instantiateViewController(withIdentifier: "NBARotoLive") as! NBARotoLive


    if((self.playerSelected) != nil){
        viewController.selectedPlayer = self.playerSelected
    }


    // Add View Controller as Child View Controller
    self.add(asChildViewController: viewController)

    return viewController
}()

private lazy var viewAvgsTab: NBARotoAvgs = {
    // Load Storyboard
    let storyboard = UIStoryboard(name: "Main", bundle: Bundle.main)

    // Instantiate View Controller
    var viewController = storyboard.instantiateViewController(withIdentifier: "NBARotoAvgs") as! NBARotoAvgs

    if((self.playerSelected) != nil){
        viewController.selectedPlayer = self.playerSelected
    }


    // Add View Controller as Child View Controller
    self.add(asChildViewController: viewController)

    return viewController
}()

private func add(asChildViewController viewController: UIViewController) {
    // Add Child View Controller
    addChildViewController(viewController)

    // Add Child View as Subview
    view.addSubview(viewController.view)

    // Configure Child View
    viewController.view.frame = view.bounds
    viewController.view.autoresizingMask = [.flexibleWidth, .flexibleHeight]

    // Notify Child View Controller
    viewController.didMove(toParentViewController: self)
}

private func remove(asChildViewController viewController: UIViewController) {
    // Notify Child View Controller
    viewController.willMove(toParentViewController: nil)

    // Remove Child View From Superview
    viewController.view.removeFromSuperview()

    // Notify Child View Controller
    viewController.removeFromParentViewController()
}

internal func test(){
    print("test")
}

internal func updateView() {

    if segmentedControl.selectedSegmentIndex == 0 {



        let position = viewAvgsTab.tableView.contentOffset.y;
        viewLiveTab.tableView.contentOffset = CGPoint(x:0, y:position);


        remove(asChildViewController: viewAvgsTab)
        add(asChildViewController: viewLiveTab)

    } else {

        let position = viewLiveTab.tableView.contentOffset.y;
        viewAvgsTab.tableView.contentOffset = CGPoint(x:0, y:position);

        remove(asChildViewController: viewLiveTab)
        add(asChildViewController: viewAvgsTab)
    }
}

}

Я рассмотрел много примеров, но я не понимаю всего, что связано с "установкой делегата", т.е. theSecondViewController.delegate = self. Иногда я вижу примеры, когда вам не нужно это делать. И в других случаях кажется, что мои венчурные капиталисты даже не имеют делегированного свойства. Так что я не уверен, является ли это моей конкретной проблемой или нет, но любое направление будет с благодарностью. Спасибо

1 ответ

Решение

Есть три шага для реализации делегата.

  1. создать протокол.. (вы уже сделали это, создав протокол updateChildView)
  2. вам нужно реализовать этот протокол в классе, который вы хотите получать и обрабатывать эти данные.. (вы не сделали этого шага, и поэтому вы не можете установить buyVC.delegate = self)
  3. вам нужно добавить свойство в ViewController2, называемое "делегат", и сделать его типом протокола на шаге 1 (вы не сделали этого шага, и в vc2 нет свойства с именем "делегат"). Вот почему вы получаете это ошибка 'у buyVC нет участника-участника')

Вот быстрый пример:

Протокол:

protocol UpdateChildView{ //removed :class
  func updateView()
  func test()
  var playerSelected: Player? { get set }
}

Viewcontroller A:

class NBARotoHome: UIViewController, UpdateChildView { //added conformance to the protocol

  //add the methods for conforming to protocol and add your implementation

   func updateView() {
        //add your implementation
   }
   func test(){
        //add your implementation
   }
   var playerSelected: Player? {
        //add your implementation
   }

  prepare(for: Segue) {
    /** code for passing data **/
    let navVC = segue.destination as? UINavigationController
    let buyVC = navVC?.viewControllers.first as! NBARotoTabPager
    buyVC.delegate = self
            //sets the delegate in the new viewcontroller 
            //remember.. delegate is a property in the next vc
           // and the property only accepts any viewcontroller that is implements updatechildview protocol
    present(vc2)
    }
  } 

viewcontroller2:

class viewControllerB: UIViewController {

  var delegate: UpdateChildView? //add this

  viewdidload {
      delegate?.test() //call the func in the previous vc
  }
}
Другие вопросы по тегам