UISearchController в iOS 9 — текстовое поле из UISearchBar вырезается из элемента навигации

#ios #swift #ios9 #uisearchcontroller

#iOS #swift #ios9 #uisearchcontroller

Вопрос:

Я пытаюсь получить плавную панель поиска для элемента навигации в iOS 9, что означает, что я не могу использовать navigationItem.searchController свойство, поскольку оно доступно только для iOS 11.

 class SearchContainerViewController: UITableViewController {
    let dataSource = ["1", "2", "3", "4", "5"]

    override public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return dataSource.count
    }

    override public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let cell = UITableViewCell(style: .default, reuseIdentifier: nil)
        cell.textLabel?.text = dataSource[indexPath.row]

        return cell
    }

    override func tableView(_ tableView: UITableView, didSelectRowAt indexPath: IndexPath) {
        dismiss(animated: true, completion: nil)
    }
}

class SearchViewController: UISearchController {
    override func viewDidLoad() {
        super.viewDidLoad()
    }
}

class MyViewController : UIViewController, UISearchResultsUpdating, UISearchBarDelegate {
    lazy var searchButton = UIBarButtonItem(title: "Search", style: UIBarButtonItem.Style.plain, target: self, action: #selector(showSearchBar))

    var searchViewController: SearchViewController = {
        let container = SearchContainerViewController()
        let searchController = SearchViewController(searchResultsController: container)

        return searchController
    }()

    override func viewDidLoad() {
        super.viewDidLoad()

        setupSearchController()
        setupSearchButton()
    }

    func setupSearchController() {
        searchViewController.searchResultsUpdater = self
        searchViewController.searchBar.delegate = self

        searchViewController.dimsBackgroundDuringPresentation = false
        searchViewController.hidesNavigationBarDuringPresentation = false
        searchViewController.searchBar.searchBarStyle = .minimal
        searchViewController.searchBar.showsCancelButton = true

        definesPresentationContext = true
    }

    @objc func showSearchBar() {
        UIView.animate(withDuration: 0.75) {
            self.navigationItem.titleView = self.searchViewController.searchBar
            self.navigationItem.rightBarButtonItem = nil
            self.searchViewController.searchBar.becomeFirstResponder()
        }
    }

    func setupSearchButton() {
        UIView.animate(withDuration: 0.75) {
            self.navigationItem.titleView = nil
            self.navigationItem.rightBarButtonItem = self.searchButton
        }
    }

    //  MARK: Conforms to UISearchResultUpdating

    public func updateSearchResults(for searchController: UISearchController) { }

    func searchBarCancelButtonClicked(_ searchBar: UISearchBar) {
        setupSearchButton()
    }

    override func viewDidAppear(_ animated: Bool) {
        super.viewDidAppear(animated)
        view.layoutSubviews()
    }
}


@UIApplicationMain
class AppDelegate: UIResponder, UIApplicationDelegate {

    var window: UIWindow?

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplication.LaunchOptionsKey: Any]?) -> Bool {
        // Override point for customization after application launch.
        let newWindow = UIWindow(frame: UIScreen.main.bounds)

        let mainViewController = MyViewController()
        let navigationController = UINavigationController(rootViewController: mainViewController)

        newWindow.backgroundColor = .white
        newWindow.rootViewController = navigationController
        newWindow.makeKeyAndVisible()

        window = newWindow

        return true
    }
}
  

Хотя результат немного разочаровывает, поскольку textview со строкой состояния вырезается из контекста элемента навигации, есть что-то, что я делаю неправильно и мог бы сделать лучше?

Ценю вашу поддержку.

Ответ №1:

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

Обрезка произошла потому, что в обеих ситуациях высота navigationItem была разной из-за того, что они несколько растягиваются, если поместить большой контент в titleView (например, панель поиска).

Я установил панель поиска с самого начала в navigationItem и просто настраиваю их свойство isHidden, когда это должно быть сделано.

     @objc private func activateSearch() {
        UIView.animate(withDuration: 0.75) {
            self.navigationItem.titleView?.isHidden = false
            self.navigationItem.rightBarButtonItem = nil
            self.searchController.isActive = true
        }
    }

    private func deactivateSearch() {
        UIView.animate(withDuration: 0.75) {
            self.navigationItem.titleView?.isHidden = true
            self.navigationItem.rightBarButtonItem = self.searchButton
            self.searchController.isActive = false
        }
    }