Как показать оповещение из другого класса в Swift?

#swift #uiviewcontroller #uialertcontroller #swift2.3

#swift #uiviewcontroller #uialertcontroller #swift2

Вопрос:

У меня есть основной класс, AddFriendsController , который запускает следующую строку кода:

 ErrorReporting.showMessage("Error", msg: "Could not add student to storage.")
  

Затем у меня есть этот ErrorReporting.swift файл:

import Foundation

 class ErrorReporting {
    func showMessage(title: String, msg: String) {
        let alert = UIAlertController(title: title, message: msg, preferredStyle: UIAlertControllerStyle.Alert)
        alert.addAction(UIAlertAction(title: "Ok", style: UIAlertActionStyle.Default, handler: nil))
        self.presentViewController(alert, animated: true, completion: nil)
    }
}
  

Очевидно, self что здесь это не сработает и выдает ошибку. Как я могу обратиться к текущему открытому контроллеру просмотра (т. Е. AddFriendsController В этом случае), поскольку я хочу использовать этот же метод во многих разных файлах swift?

Спасибо.

Комментарии:

1. Создайте расширение ShowMessage . Как сообщение об ошибке расширения. а затем ваша функция.

2. @sourav Не могли бы вы объяснить более подробно? Все еще изучаю swift…

3. сообщение об ошибке расширения { функция ShowMessage(заголовок: строка, сообщение: строка) { пусть alert = UIAlertController(заголовок: заголовок, сообщение: сообщение, предпочтительный стиль: UIAlertControllerStyle. Предупреждение) alert.addAction(UIAlertAction(заголовок: «Ok», стиль: UIAlertActionStyle. По умолчанию, обработчик: ноль)) self.presentViewController(оповещение, анимация: истина, завершение: ноль) } }

Ответ №1:

Вы можете создать метод расширения для UIApplication (например), который вернет ваш topViewController:

 extension UIApplication {

    static func topViewController(base: UIViewController? = UIApplication.sharedApplication().delegate?.window??.rootViewController) -> UIViewController? {
        if let nav = base as? UINavigationController {
            return topViewController(nav.visibleViewController)
        }
        if let tab = base as? UITabBarController, selected = tab.selectedViewController {
            return topViewController(selected)
        }
        if let presented = base?.presentedViewController {
            return topViewController(presented)
        }

        return base
    }
}
  

И тогда ваш класс будет выглядеть так:

 class ErrorReporting {

    static func showMessage(title: String, msg: String) {
        let alert = UIAlertController(title: title, message: msg, preferredStyle: UIAlertControllerStyle.Alert)
        alert.addAction(UIAlertAction(title: "Ok", style: UIAlertActionStyle.Default, handler: nil))
        UIApplication.topViewController()?.presentViewController(alert, animated: true, completion: nil)
    }
}
  

Метод должен быть static , чтобы иметь возможность вызывать его как ErrorReporting.showMessage .

Комментарии:

1. Черт возьми, это расширение так чертовски хорошо XD

2. Это спасает жизнь! Так элегантно! Большое спасибо!

Ответ №2:

На самом деле, на мой взгляд, операция представления контроллера представления должна выполняться на UIViewController экземпляре, а не в классе модели.

Простой обходной путь для этого — передать UIViewController экземпляр в качестве параметра

 class ErrorReporting {
    func showMessage(title: String, msg: String, `on` controller: UIViewController) {
        let alert = UIAlertController(title: title, message: msg, preferredStyle: UIAlertControllerStyle.Alert)
        alert.addAction(UIAlertAction(title: "Ok", style: UIAlertActionStyle.Default, handler: nil))
        controller.presentViewController(alert, animated: true, completion: nil)
    }
}
  

И назовите его, как показано ниже

 ErrorReporting.showMessage("Error", msg: "Could not add student to storage.", on: self)
  

Ответ №3:

Swift 3 версия ответа Максима Мусиенко будет следующей:

 extension UIApplication {

    static func topViewController(base: UIViewController? = UIApplication.shared.delegate?.window??.rootViewController) -> UIViewController? {

        if let nav = base as? UINavigationController {
            return topViewController(base: nav.visibleViewController)
        }

        if let tab = base as? UITabBarController, let selected = tab.selectedViewController {
            return topViewController(base: selected)
        }

        if let presented = base?.presentedViewController {
            return topViewController(base: presented)
        }

        return base
    }
}