не удается получить навигацию ControlController? .pushViewController работает, но присутствует ли? - PullRequest
2 голосов
/ 08 октября 2019

У меня есть не Storyboard, 100% кодированный UIViewController, UICollectionView и UICollectionViewCell - работает отлично.

вот код, о котором идет речь:

SceneDelegate не уверен, насколько это актуально, хотя

class SceneDelegate: UIResponder, UIWindowSceneDelegate {

    var window: UIWindow?

    func scene(_ scene: UIScene, willConnectTo session: UISceneSession, options connectionOptions: UIScene.ConnectionOptions) {

        guard let windowScene = (scene as? UIWindowScene) else { return }
        window = UIWindow(windowScene: windowScene)

        let layout = UICollectionViewFlowLayout()
        layout.scrollDirection = .horizontal
        let myController = MyViewController(collectionViewLayout: layout)
        window?.rootViewController = myController
        window?.makeKeyAndVisible()
    }
.
.

ViewController очень просто и просто ...

class MyViewController: UICollectionViewController, UICollectionViewDelegateFlowLayout {
    let data = loadOnboardingData()
.
.
override func viewDidLoad() {
        super.viewDidLoad()

        collectionView?.backgroundColor = .white
        collectionView?.register(MyPageCell.self, forCellWithReuseIdentifier: "cellId")
        collectionView?.isPagingEnabled = true
        collectionView.showsHorizontalScrollIndicator = false
        collectionView?.tag = myPageControl.currentPage

        setupMyPageControl()

    }

ViewControllerExtention вот в чем проблема: метод pushViewController просто ничего не делает, но модальное настоящее работает как шарм, а я нетпонимая, что не так и почему:

extension MyViewController: MyPageCellDelegate {
.
.
func didTabOnActionButton(title: String) {


        let storyboard: UIStoryboard = UIStoryboard(name: "Main", bundle: nil)
        guard let homeViewController = storyboard.instantiateViewController(withIdentifier: "HomeViewController") as? HomeViewController else {
            print("Coun't find controller")
            return
        }
        navigationController?.pushViewController(homeViewController, animated: true) <- NO EFFECT
        //present(homeViewController, animated: true, completion: nil) <- WORKS PERFECT!

    }

MyPageCell Я настроил делегата по протоколу, и кажется, что это тоже хорошо

protocol MyPageCellDelegate {
    func didTabOnActionButton(title: String)
}

class MyPageCell: UICollectionViewCell {

    var delegate: MyPageCellDelegate?

let myActionButton: UIButton = {
        let button = UIButton(type: .system)
        return button
    }()

myActionButton.addTarget(self, action: #selector(self.doAction), for: .touchUpInside)
.
.
@objc private func doAction(_ sende: Any) {
        delegate?.didTabEndOnboardingActionButton(title: "end Onboarding")
    }

так что, любая идея, что не такс:

navigationController?.pushViewController(homeViewController, animated: true)

РЕДАКТИРОВАТЬ --------------------------------------------------------------------

Как указано @Michcio здесь: window?.rootViewController = UINavigationController(rootViewController: myController) работает на полпути, и, насколько я понимаю, я встраиваю myController в UINavigationController, который добавляет панель навигации к текущим и следующим контроллерам.

Но это не то, что мне нужно!

Мне нужен чистый и простойодин для включения, т.е. MyViewController и HomeViewController должны быть с вкладкой и панелью навигации

В основном, начиная с нуля после включения.

Я использовал для решения этой проблемы в предыдущей версии, редактируя AppDelegateпервый метод, подобный этому (в этом примере я использовал раскадровки):

extension AppDelegate {

    func showOnboarding() {
        if let window = UIApplication.shared.keyWindow, let onboardingViewController = UIStoryboard(name: "Onboarding", bundle: nil).instantiateInitialViewController() as? OnboardingViewController {
            onboardingViewController.delegate = self
            window.rootViewController = onboardingViewController
        }
    }

    func hideOnboarding() {
        if let window = UIApplication.shared.keyWindow, let mainViewController = UIStoryboard(name: "Main", bundle: nil).instantiateInitialViewController() {
            mainViewController.view.frame = window.bounds
            UIView.transition(with: window, duration: 0.5, options: .transitionCrossDissolve, animations: {
                window.rootViewController = mainViewController
            }, completion: nil)
        }
    }
}

и в самом делегате вот так:

func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
    let isFirstRun = true // logic to determine goes here
    if isFirstRun {
        showOnboarding()
    }
    return true
}

, но я серьезно не получаю новый SceneDelegate илипросто не понимаю

Действительно был бы признателен, если бы кто-нибудь мог вставить сюда какой-то код для повторного использования.

1 Ответ

1 голос
/ 08 октября 2019

Это не сработало, потому что вы установили MyViewController как window.rootViewController. Просто измените строку в SceneDelegate на:

window?.rootViewController = UINavigationController(rootViewController: myController)

...