it-swarm.com.ru

Отклонение Представленного Контроллера Представления

У меня есть теоретический вопрос. Сейчас я читаю Apple ViewController руководство.

Они написали: 

Когда приходит время отклонить представленный контроллер представления, предпочтительный подход заключается в том, чтобы позволить представителю представления отклонить Это. Другими словами, когда это возможно, тот же контроллер представления, что Представленный контроллер представления также должен взять на себя ответственность за отклоняя это. Хотя есть несколько способов уведомления Представление контроллера представления, что его представленный контроллер представления должен быть отклоненным, предпочтительный метод - делегирование.

Но я не могу объяснить, почему я должен создать протокол в представленном VC и добавить переменную делегата, создать метод делегата в представлении VC для отклонения представленного VC вместо простого вызова в present просмотр метода контроллера 

[self dismissViewControllerAnimated:NO completion:nil]

Почему первый выбор лучше? Почему Apple рекомендует это?

94
nikitahils

Я думаю, что Apple немного прикрывает свои спины ради потенциально грязного API. 

  [self dismissViewControllerAnimated:NO completion:nil]

Это на самом деле немного скрипки. Хотя вы можете - законно - вызвать это на представленном контроллере представления, все, что он делает, это передает сообщение на представляющий контроллер представления. Если вы хотите сделать что-то сверх того, что просто отмените VC, вам нужно это знать, и вам нужно относиться к этому почти так же, как к методу делегата - так как это почти то же, что и запутанный, несколько негибкий метод делегата. 

Возможно, они сталкивались с множеством плохого кода людьми, которые не совсем понимают, как все это сложить, и поэтому их осторожность. 

Но, конечно, если все, что вам нужно сделать, это отказаться от этой вещи, продолжайте.

Мой собственный подход - это компромисс, по крайней мере, он напоминает мне, что происходит:

  [[self presentingViewController] dismissViewControllerAnimated:NO completion:nil]

[Свифт]

  self.presentingViewController?.dismiss(animated: false, completion:nil)
102
foundry

Это для многократного использования контроллера представления.

Ваш контроллер представления не должен заботиться о том, будет ли он представлен как модальный, помещен в контроллер навигации или как-то еще. Если ваш контроллер представления отклоняет сам себя, то вы предполагаете, что он представлен модально. Вы не сможете перенести этот контроллер вида на контроллер навигации.

Реализуя протокол, вы позволяете родительскому контроллеру представления решить, как он должен быть представлен/выдвинут и отклонен/вытолкнут.

46
Michael Enriquez

Обновлено для Swift 3

Я пришел сюда просто чтобы уволить текущий (представленный) View Controller. Я делаю этот ответ для тех, кто приезжает сюда с той же целью.

Навигационный контроллер

Если вы используете навигационный контроллер, то это довольно просто.

Вернитесь к предыдущему контроллеру представления:

// Swift
self.navigationController?.popViewController(animated: true)

// Objective-C
[self.navigationController popViewControllerAnimated:YES];

Вернитесь к контроллеру корневого представления:

// Swift
self.navigationController?.popToRootViewController(animated: true)

// Objective-C
[self.navigationController popToRootViewControllerAnimated:YES];

(Спасибо этот ответ за Objective-C.)

Контроллер модального вида

Когда View Controller представлен модально, вы можете удалить его (из второго контроллера view), вызвав

// Swift
self.dismiss(animated: true, completion: nil)

// Objective-C
[self dismissViewControllerAnimated:YES completion:nil];

документация говорит,

Контроллер представления представления ответственен за отклонение представления Контроллер это представил. Если вы вызываете этот метод в представленном представлении Сам контроллер, UIKit просит текущий контроллер представления обработать увольнение.

Таким образом, это работает для представленного контроллера представления, чтобы вызвать его на себя. Здесь является полным примером.

Делегаты

Вопрос ОП касался сложности использования делегатов для отклонения представления. 

До этого момента мне не нужно было использовать делегатов, так как у меня обычно есть контроллер навигации или контроллеры модального представления, но если мне действительно понадобится шаблон делегата в будущем, я добавлю обновление.

41
Suragch

По моему опыту, это удобно, когда вам нужно удалить его излюбогоViewController, который вы хотите, и выполнить различные задачи для каждого viewcontroller, который его отклоняет. Любой viewController, который принимает протокол, может отклонить представление по-своему. (ipad против iphone или передача разных данных при отклонении из разных представлений, вызов разных методов при отклонении и т. д.)

Правка:

Итак, чтобы уточнить, если все, что вы когда-либо захотите сделать, это отклонить представление, я не вижу необходимости настраивать протокол делегирования. Если вам нужно делать разные вещипослевы исключаете это из разных контроллеров представления представления, это был бы ваш лучший способ использовать делегат.

6
jhilgert00

попробуй это:

[self dismissViewControllerAnimated:true completion:nil];
6
Oshitha Wimalasuriya

Один момент заключается в том, что это хороший подход к кодированию. Он удовлетворяет многим принципам OOP, например, SRP, разделение интересов и т.д. 

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

Например, риэлторская компания, которая сдает дом в аренду, должна иметь право забрать его обратно.

2
Abdurrahman Mubeen Ali

Swift 3.0 // Отключить контроллер представления в Swift 

self.navigationController?.popViewController(animated: true)
dismiss(animated: true, completion: nil)
2
Pranit

Цитата из View Руководство по программированию контроллеров , "Как контроллеры представления представляют другие контроллеры представления".

Каждый контроллер представления в цепочке представленных контроллеров представления имеет указатели на другие объекты, окружающие его в цепочке. В других слова, представленный контроллер представления, который представляет другой вид Контроллер имеет действительные объекты как в представлении ViewController, так и в представленные свойства ViewController. Вы можете использовать эти отношения для проследить через цепочку контроллеров представления по мере необходимости. Например, если пользователь отменяет текущую операцию, вы можете удалить все объекты в цепочка путем отклонения первого представленного контроллера представления . Отклонение контроллера представления отклоняет не только тот контроллер представления но также любые контроллеры представления это представило.

Так что, с одной стороны, это обеспечивает хороший сбалансированный дизайн, хорошее расцепление и т.д. Но с другой стороны, это очень практично, потому что вы можете быстро вернуться к определенной точке навигации. 

Хотя я лично предпочел бы использовать разматывание сегментов, чем пытаться пройти назад по дереву представление контроллеров представления, о чем Apple говорит в этой главе, откуда взята цитата. 

2
svena

В дополнение к ответу Майкла Энрикеса, я могу вспомнить еще одну причину, по которой это может быть хорошим способом защитить себя от неопределенного состояния:

Скажем, ViewControllerA представляет ViewControllerB модально. Но, поскольку вы, возможно, не написали код для ViewControllerA, вы не знаете о жизненном цикле ViewControllerA. Это может отклонить 5 секунд (скажем) после представления вашего контроллера представления, ViewControllerB.

В этом случае, если бы вы просто использовали dismissViewController из ViewControllerB, чтобы закрыть себя, вы бы оказались в неопределенном состоянии - возможно, не сбоем или черным экраном, а неопределенным состоянием с вашей точки зрения.

Если бы вместо этого вы использовали шаблон делегата, вы были бы осведомлены о состоянии ViewControllerB, и вы можете программировать для случая, подобного описанному мной.

1
Mayur

Стриж

let rootViewController:UIViewController = (UIApplication.shared.keyWindow?.rootViewController)!

        if (rootViewController.presentedViewController != nil) {
            rootViewController.dismiss(animated: true, completion: {
                //completion block.
            })
        }
0
Rishi Chaurasia

Если вы используете модальное использование, используйте view dismiss.

[self dismissViewControllerAnimated:NO completion:nil];
0
ErasmoOliveira

Вы можете закрыть окно супер просмотра

self.view.superview?.window?.close()

0
MR.Code

Это много глупостей. Делегирование хорошо, когда это необходимо, но если это делает код более сложным - и это делает - тогда должна быть причина для этого.

Я уверен, что у Apple есть свои причины. Но более ясно и более кратко просто заставить представленного VC выполнить отставку, если только нет веской причины поступить иначе, и никто здесь на сегодняшний день не представил такую, которую я вижу. 

Протоколы превосходны, когда они необходимы, но объектно-ориентированное проектирование никогда не сводилось к тому, чтобы модули без необходимости общались друг с другом. 

Том Лав (соавтор Objective C) однажды заметил, что Objective C был «элегантным», «маленьким», «четким» и «четким» (при сравнении с C++). Легко ему сказать. Делегирование - это полезная функция, которая, кажется, использовалась слишком часто «просто потому, что», и хотя мне нравится работать на языке, я боюсь, что вынужден использовать ненужный синтаксис, чтобы сделать вещи более сложными, чем они должны быть.

0
RegularExpression