[ios] 푸시 컨트롤러에서 RootViewController를 어떻게 얻습니까?

그래서 RootViewController에서 다음과 같이 뷰 컨트롤러를 푸시합니다.

[self.navigationController pushViewController : anotherViewController animated : YES];

그러나 anotherViewController지금부터는 RootViewController에 다시 액세스하고 싶습니다.

노력하고있어

// (지금 anotherViewController 내부)
/// RootViewController * root = (RootViewController *) self.parentViewController; // 아니.
// 오류
RootViewController * root = (RootViewController *) [self.navigationController.viewControllers objectAtIndex : 0]; // 예!! 효과가있다

왜 이것이 효과가 있는지 잘 모르겠으며 최선의 방법인지 확실하지 않습니다. 누군가 당신이 RootViewController의 navigationController로 푸시 한 컨트롤러에서 RootViewController를 얻는 더 좋은 방법과 내가 한 방법이 신뢰할 수 있는지 여부에 대해 언급 할 수 있습니까?



답변

UINavigationController 의 viewControllers 특성을 사용하십시오 . 예제 코드 :

// Inside another ViewController
NSArray *viewControllers = self.navigationController.viewControllers;
UIViewController *rootViewController = [viewControllers objectAtIndex:viewControllers.count - 2];

이것이 “백”뷰 컨트롤러를 얻는 표준 방법입니다. objectAtIndex:0작동하려는 이유 는 액세스하려는 뷰 컨트롤러가 루트 컨트롤러이기도하기 때문입니다. 탐색이 더 깊다면 후면보기가 루트보기와 같지 않을 것입니다.


답변

스위프트 버전 :

var rootViewController = self.navigationController?.viewControllers.first

ObjectiveC 버전 :

UIViewController *rootViewController = [self.navigationController.viewControllers firstObject];

여기서 self는 UINavigationController에 포함 된 UIViewController의 인스턴스입니다.


답변

거의 모든 대답에서 언급 된 것과 동일한 것의 약간 덜 추한 버전 :

UIViewController *rootViewController = [[self.navigationController viewControllers] firstObject];

귀하의 경우, 아마도 다음과 같은 일을 할 것입니다 :

UINavigationController 서브 클래스 내부 :

- (UIViewController *)rootViewController
{
    return [[self viewControllers] firstObject];
}

다음을 사용할 수 있습니다.

UIViewController *rootViewController = [self.navigationController rootViewController];

편집하다

OP는 의견에 재산을 물었다.

원하는 경우 self.navigationController.rootViewController헤더에 읽기 전용 속성을 추가하여 다음과 같은 방법 으로 액세스 할 수 있습니다 .

@property (nonatomic, readonly, weak) UIViewController *rootViewController;


답변

신속한 확장에 관심이있는 모든 사람들에게 이것이 현재 사용중인 것입니다.

extension UINavigationController {
    var rootViewController : UIViewController? {
        return self.viewControllers.first
    }
}


답변

@dulgan의 답변 외에도 항상 firstObjectover 를 사용하는 것이 좋습니다. objectAtIndex:0처음에는 배열에 객체가 없으면 nil을 반환하지만 후자는 예외를 throw하기 때문입니다.

UIViewController *rootViewController = self.navigationController.rootViewController;

또는 이름이 지정된 카테고리를 작성하고 해당 UINavigationController+Additions메소드를 정의 하는 것이 큰 장점 입니다.

@interface UINavigationController (Additions)

- (UIViewController *)rootViewController;

@end

@implementation UINavigationController (Additions)

- (UIViewController *)rootViewController
{
    return self.viewControllers.firstObject;
}

@end


답변

keyWindow에 대한 UIApplication 싱글 톤 을 요청 하고 UIWindow 에서 루트보기 컨트롤러 ( rootViewController 속성) 를 요청하는 방법은 무엇 입니까 ?

UIViewController root = [[[UIApplication sharedApplication] keyWindow] rootViewController];


답변

여기서 나는 어디에서나 루트로 이동하는 보편적 인 방법을 생각해 냈습니다.

  1. 이 클래스를 사용하여 새 클래스 파일을 작성하면 프로젝트의 어느 곳에서나 액세스 할 수 있습니다.

    import UIKit
    
    class SharedControllers
    {
        static func navigateToRoot(viewController: UIViewController)
        {
            var nc = viewController.navigationController
    
            // If this is a normal view with NavigationController, then we just pop to root.
            if nc != nil
            {
                nc?.popToRootViewControllerAnimated(true)
                return
            }
    
            // Most likely we are in Modal view, so we will need to search for a view with NavigationController.
            let vc = viewController.presentingViewController
    
            if nc == nil
            {
                nc = viewController.presentingViewController?.navigationController
            }
    
            if nc == nil
            {
                nc = viewController.parentViewController?.navigationController
            }
    
            if vc is UINavigationController && nc == nil
            {
                nc = vc as? UINavigationController
            }
    
            if nc != nil
            {
                viewController.dismissViewControllerAnimated(false, completion:
                    {
                        nc?.popToRootViewControllerAnimated(true)
                })
            }
        }
    }
  2. 프로젝트의 어느 곳에서나 사용 :

    {
        ...
        SharedControllers.navigateToRoot(self)
        ...
    }