Я хочу вернуться к третьему представлению в стеке навигации и вернуться к первому представлению.
Я знаю, как открыть одно представление сразу:
[self.navigationController popViewControllerAnimated:YES];
Но как мне сделать сразу два?
Я хочу вернуться к третьему представлению в стеке навигации и вернуться к первому представлению.
Я знаю, как открыть одно представление сразу:
[self.navigationController popViewControllerAnimated:YES];
Но как мне сделать сразу два?
Ответы:
Вы также можете попробовать это для перехода между стеком контроллера навигации
NSMutableArray *allViewControllers = [NSMutableArray arrayWithArray:[self.navigationController viewControllers]];
for (UIViewController *aViewController in allViewControllers) {
if ([aViewController isKindOfClass:[RequiredViewController class]]) {
[self.navigationController popToViewController:aViewController animated:NO];
}
}
Вот два UINavigationController
расширения, которые могут решить вашу проблему. Я бы рекомендовал использовать первый, который появляется в UIViewController
определенном классе:
extension UINavigationController {
func popToViewController(ofClass: AnyClass, animated: Bool = true) {
if let vc = viewControllers.filter({$0.isKind(of: ofClass)}).last {
popToViewController(vc, animated: animated)
}
}
func popViewControllers(viewsToPop: Int, animated: Bool = true) {
if viewControllers.count > viewsToPop {
let vc = viewControllers[viewControllers.count - viewsToPop - 1]
popToViewController(vc, animated: animated)
}
}
}
и используйте это так:
// pop to SomeViewController class
navigationController?.popToViewController(ofClass: SomeViewController.self)
// pop 2 view controllers
navigationController?.popViewControllers(viewsToPop: 2)
removeLast
на just removeLast(2)
.
let vc = viewControllers[viewControllers.count - viewsToPop + 1]
, для правильной работы вам необходимо заменить их на: let vc = viewControllers[viewControllers.count - (viewsToPop + 1)]
илиlet vc = viewControllers[viewControllers.count - viewsToPop - 1]
Вы можете перейти к «корневому» (первому) контроллеру представления с помощью popToRootViewControllerAnimated
:
[self.navigationController popToRootViewControllerAnimated:YES];
// If you want to know what view controllers were popd:
// NSArray *popdViewControllers = [self.navigationController popToRootViewControllerAnimated:YES];
UINavigationController
Ссылка :
Извлекает все контроллеры представления в стек, кроме контроллера корневого представления, и обновляет отображение.
Возвращаемое значение
Массив контроллеров представления, извлекаемых из стека.
Swift 2 - xCode 7.3
Это может быть очень полезным расширением для вывода UIViewControllers:
extension UINavigationController {
func popToViewControllerOfType(classForCoder: AnyClass) {
for controller in viewControllers {
if controller.classForCoder == classForCoder {
popToViewController(controller, animated: true)
break
}
}
}
func popViewControllers(controllersToPop: Int, animated: Bool) {
if viewControllers.count > controllersToPop {
popToViewController(viewControllers[viewControllers.count - (controllersToPop + 1)], animated: animated)
} else {
print("Trying to pop \(controllersToPop) view controllers but navigation controller contains only \(viewControllers.count) controllers in stack")
}
}
}
Если вы просто хотите вывести 2 сразу, потому что ваш rootViewController (способ) «глубже», то 2, вы можете подумать о добавлении категории в UIviewController, например:
#import <UIKit/UIKit.h>
@interface UINavigationController (popTwice)
- (void) popTwoViewControllersAnimated:(BOOL)animated;
@end
#import "UINavigationController+popTwice.h"
@implementation UINavigationController (popTwice)
- (void) popTwoViewControllersAnimated:(BOOL)animated{
[self popViewControllerAnimated:NO];
[self popViewControllerAnimated:animated];
}
@end
Импортируйте категорию #import "UINavigationController+popTwice.h"
где-нибудь в своей реализации и используйте эту строку кода для одновременного вывода двух контроллеров:
[self.navigationController popTwoViewControllersAnimated:YES];
разве это не здорово? :)
Swift 4:
func popViewControllerss(popViews: Int, animated: Bool = true) {
if self.navigationController!.viewControllers.count > popViews
{
let vc = self.navigationController!.viewControllers[self.navigationController!.viewControllers.count - popViews - 1]
self.navigationController?.popToViewController(vc, animated: animated)
}
}
Затем используйте этот метод
self.popViewControllerss(popViews: 2)
//viewIndex = 1;
//viewIndex = 2;
//viewIndex = 3;
// **[viewControllers objectAtIndex: *put here your viewindex number*]
NSArray *viewControllers = [self.navigationController viewControllers];
[self.navigationController popToViewController:[viewControllers objectAtIndex:viewIndex] animated:NO];
NSMutableArray *newStack = [NSMutableArray arrayWithArray:[self.navigationController viewControllers]];
[newStack removeLastObject];
[newStack removeLastObject];
[self.navigationController setViewControllers:newStack animated:YES];
В Swift 3 вы можете вытащить один, два или несколько контроллеров просмотра из контроллера навигации таким образом
let viewControllers = self.navigationController!.viewControllers as [UIViewController]
for aViewController:UIViewController in viewControllers {
if aViewController.isKind(of: FromWhereYouWantToGoController.self) {
_ = self.navigationController?.popToViewController(aViewController, animated: true)
}
}
Здесь FromWhereYouWantToGoController - это контроллер назначения. Надеюсь, это поможет.
Я не видел здесь этого ответа. Если вы хотите вставить только несколько (не до корня), вы можете просто выполнить итерацию через self.navigationController.viewControllers, проверяя типы классов, или, если у вас есть ссылка, используйте это:
for (UIViewController *aViewController in self.navigationController.viewControllers) {
if ([aViewController isKindOfClass:[SMThumbnailViewController class]]) {
[self.navigationController popToViewController:aViewController animated:YES];
}
}
вы можете вернуться к контроллеру корневого представления
[self.navigationController popToRootViewControllerAnimated:YES];
или, если представление, которое вы хотите открыть, не является первым, вам нужно будет снова открыть его в представлении предыдущего представленияWillAppear
Вот небольшая функция, которую я использую для возврата к X ViewControllers:
-(void)backMultiple:(id)data {
int back = 2; //Default to go back 2
int count = [self.navigationController.viewControllers count];
if(data[@"count"]) back = [data[@"count"] intValue];
//If we want to go back more than those that actually exist, just go to the root
if(back+1 > count) {
[self.navigationController popToRootViewControllerAnimated:YES];
}
//Otherwise go back X ViewControllers
else {
[self.navigationController popToViewController:[self.navigationController.viewControllers objectAtIndex:count-(back+1)] animated:YES];
}
}
Версия Swift от (Swift 1.2 / Xcode 6.3.1), появляющаяся дважды или более
var viewControllers = self.navigationController?.viewControllers
viewControllers?.removeLast()
viewControllers?.removeLast()
self.navigationController?.setViewControllers(viewControllers, animated: true)
или
var viewControllers = self.navigationController?.viewControllers
var viewsToPop = 2
var end = (viewControllers?.count)!
var start = end - viewsToPop
viewControllers?.removeRange(Range<Int>(start: start, end: end))
self.navigationController?.setViewControllers(viewControllers, animated: true)
Вы можете использовать стек UIViewControllers. 1. Получите массив всех viewControllers в стеке. 2. Просмотрите весь массив и найдите желаемый viewController
, сопоставив тип класса. 3. Откройте viewController.
func popToSpecificViewC
{
let viewControllers: [UIViewController] = self.navigationController!.viewControllers as [UIViewController];
for viewController:UIViewController in viewControllers
{
if viewController.isKind(of: WelcomeViewC.self)
{
_ = self.navigationController?.popToViewController(viewController, animated: true)
}
}
}
Используя простое расширение UINavigationController :
extension UINavigationController {
func popViewControllers(_ count: Int) {
guard viewControllers.count > count else { return }
popToViewController(viewControllers[viewControllers.count - count - 1], animated: true)
}
}