Search code examples
uitableviewswiftuisearchbarcustom-celluisearchbardisplaycontrol

SearchBar, Custom Cell and setCell method: fatal error: unexpectedly found nil while unwrapping an Optional value


I implemented a searchBar and searchDisplayController on my tableView (in a MasterViewController). I have a CustomCell which works fine for displaying my tableView's cells but it doesn't work for my search tableView's cells and I dont' understand why.

Here's my CustomCell class:

import UIKit

class CustomCell: UITableViewCell {

@IBOutlet weak var pgrmLabel: UILabel!
@IBOutlet weak var noteLabel: UILabel!
@IBOutlet weak var logoView: UIImageView!


override func awakeFromNib() {
    super.awakeFromNib()
    // Initialization code
}

override func setSelected(selected: Bool, animated: Bool) {
    super.setSelected(selected, animated: animated)
    // Configure the view for the selected state
}

func setCell(pgrmName:String, noteInt:Int, logoName:String){
    println("setCell")
    if let itemName=self.pgrmLabel{
        itemName.text=pgrmName
        println("pgrmName: \(pgrmName)")
    }
    if let itemNote=self.noteLabel{
        itemNote.text=String(noteInt)
    }
    if let itemLogo=self.logoView{
        itemLogo.image=UIImage(named: logoName)
    }
    else{
        println("setCell ELSE")
        self.pgrmLabel.text=pgrmName as String
        self.noteLabel.text=String(noteInt)
        self.logoView.image=UIImage(named: logoName as String)
    }   
}
}

And the cellForRowAtIndexPath method in the masterViewController:

override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {

var program:Program
var cell: CustomCell!=tableView.dequeueReusableCellWithIdentifier("Cell") as CustomCell!

    if(cell==nil){
        println("cell nil")
        tableView.registerClass(CustomCell.classForCoder(), forCellReuseIdentifier: "Cell")
        cell=CustomCell(style: UITableViewCellStyle.Default, reuseIdentifier: "Cell")
    }
    else{
    if tableView == searchDisplayController?.searchResultsTableView{
        println("SearchCell")
        program=filteredPrograms[indexPath.row]
    }
    else{
        println("MasterCell")
        program=arrayOfPrograms[indexPath.row]
    }
    cell.setCell(program.name as String, noteInt: program.note, logoName: program.imageName as String)
    }
    return cell
}

Also, here's the Program class:

import Foundation

class Program{
    var name:String
    var note:Int
    var imageName:String
    var channel:String

init(name:String, note:Int, imageName:String, channel:String){
    self.name=name
    self.note=note
    self.imageName=imageName
    self.channel=channel
}
}

The app crashes when I type a letter in the searchBar in the setCell Method. It prints:

cell nil
SearchCell
setCell
setCell ELSE
fatal error: unexpectedly found nil while unwrapping an Optional value

If someone knows what's happening, that would be very helpful.

Thanks a lot.

EDIT: It doesn't crash but I have an empty search tableView if I change the setCell method into this:

self.pgrmLabel?.text=pgrmName
self.noteLabel?.text=String(noteInt)
self.logoView?.image=UIImage(named: logoName)

The three parameters pgrmName, noteInt and logoName are not nil but pgrmLabel, noteLabel and logoView are. I guess (because these are optional values) I don't assign values the right way.


Solution

  • I finally did another way: I deleted the cell from the tableview in the mainstoryboard and I created a new file (with xib file) to create my custom cell.

    Here's the code of my custom class cell:

    class TableCell: UITableViewCell {
    
    @IBOutlet weak var pgrmLabel: UILabel!
    @IBOutlet weak var noteLabel: UILabel!
    @IBOutlet weak var logoView: UIImageView!
    
    override func awakeFromNib() {
        super.awakeFromNib()
        // Initialization code
    }
    
    override func setSelected(selected: Bool, animated: Bool) {
        super.setSelected(selected, animated: animated)
        // Configure the view for the selected state
    }
    
    func setCell(cell: TableCell, pgrmName:String, noteInt:Int, logoName:String){
        if let itemName=self.pgrmLabel{
            itemName.text=pgrmName
        }
        if let itemNote=self.noteLabel{
            itemNote.text="\(noteInt)"
        }
        if let itemLogo=self.logoView{
            itemLogo.image=UIImage(named: logoName)
            if ((itemLogo.image) == nil){
                itemLogo.image=UIImage(named: "placeholder.jpg")
            }
        }
      }
    }
    

    I also created a BaseTableViewController class, and my MasterViewController (main tableview) and my ResultsTableViewController (which I also created) will inherit from it.

    BaseTableViewController:

    class BaseTableViewController: UITableViewController {
    // MARK: Types
    
    struct Constants {
        struct Nib {
            static let name = "TableCell"
        }
    
        struct TableViewCell {
            static let identifier = "cellID"
        }
    }
    
    // MARK: View Life Cycle
    
    override func viewDidLoad() {
        super.viewDidLoad()
    
        let nib = UINib(nibName: Constants.Nib.name, bundle: nil)
    
        // Required if our subclasses are to use: dequeueReusableCellWithIdentifier:forIndexPath:
        tableView.registerNib(nib, forCellReuseIdentifier: Constants.TableViewCell.identifier)
    }
    
    
    // MARK:
    
    func configureCell(cell: TableCell, forProgram program: Program) {
        cell.setCell(cell, pgrmName: program.name, noteInt: program.note, logoName: program.imageName)
    }
    
    override func tableView(tableView: UITableView, estimatedHeightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat {
        return 100
    }
    
    override func tableView(tableView: UITableView, heightForRowAtIndexPath indexPath: NSIndexPath) -> CGFloat {
        return 100
    }
    }
    

    MasterViewController:

    class MasterViewController: BaseTableViewController, UITableViewDataSource, UITableViewDelegate, UISearchBarDelegate, UISearchDisplayDelegate, UISearchResultsUpdating {
    
    struct RestorationKeys {
        static let viewControllerTitle = "ViewControllerTitleKey"
        static let searchControllerIsActive = "SearchControllerIsActiveKey"
        static let searchBarText = "SearchBarTextKey"
        static let searchBarIsFirstResponder = "SearchBarIsFirstResponderKey"
    }
    
    //controllers
    var detailViewController: DetailViewController? = nil
    var searchController: UISearchController!
    var resultsTableController: ResultsTableViewController!
    var restoredState=SearchControllerRestorableState()
    
    //array of programs to display in the tableview
    var arrayOfPrograms=[Program]()
    
    struct SearchControllerRestorableState {
        var wasActive=false
        var wasFirstResponder=false
    }
    
    override func awakeFromNib() {
        super.awakeFromNib()
        if UIDevice.currentDevice().userInterfaceIdiom == .Pad {
            self.clearsSelectionOnViewWillAppear = false
            self.preferredContentSize = CGSize(width: 320.0, height: 600.0)
        }
    }
    
    override func viewDidLoad() {
        super.viewDidLoad()
        //self.navigationItem.leftBarButtonItem = self.editButtonItem()
        if let split = self.splitViewController {
            let controllers = split.viewControllers
            self.detailViewController = controllers[controllers.count-1].topViewController as? DetailViewController
        }
        self.setUpPrograms()
    
        let tableViewController=navigationController?.viewControllers[0] as MasterViewController
    
        resultsTableController=ResultsTableViewController()
        resultsTableController.tableView.delegate=self
    
        searchController=UISearchController(searchResultsController: resultsTableController)
        searchController.searchResultsUpdater=self
        searchController.searchBar.sizeToFit()
        tableView.tableHeaderView=searchController.searchBar
        searchController.delegate=self
        searchController.dimsBackgroundDuringPresentation=false
        searchController.searchBar.delegate=self
        definesPresentationContext=true
    
    override func viewDidAppear(animated: Bool) {
        super.viewDidAppear(animated)
    
        // Restore the searchController's active state.
        if restoredState.wasActive {
            searchController.active = restoredState.wasActive
            restoredState.wasActive = false
    
            if restoredState.wasFirstResponder {
                searchController.searchBar.becomeFirstResponder()
                restoredState.wasFirstResponder = false
            }
        }
    }
    
    func searchBarSearchButtonClicked(searchBar: UISearchBar) {
        searchBar.resignFirstResponder()
    }
    
    override func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath) {
        var selectedProgram:Program
    
        if tableView == self.tableView {
            selectedProgram=arrayOfPrograms[indexPath.row]
        }
        else{
            selectedProgram=resultsTableController.filteredPrograms[indexPath.row]
        }
    
        let detailViewController=DetailViewController.forProgram(selectedProgram)
        navigationController!.pushViewController(detailViewController, animated: true)
        tableView.deselectRowAtIndexPath(indexPath, animated: true)
    }
    
    // MARK: - Table View
    
    override func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return 1
    }
    
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
            return arrayOfPrograms.count
    }
    
    
    
    func updateSearchResultsForSearchController(searchController: UISearchController) {
        let searchResults=arrayOfPrograms
    
        // Strip out all the leading and trailing spaces.
        let whitespaceCharacterSet = NSCharacterSet.whitespaceCharacterSet()
        let strippedString = searchController.searchBar.text.stringByTrimmingCharactersInSet(whitespaceCharacterSet)
        let searchItems = strippedString.componentsSeparatedByString(" ") as [String]
    
        // Build all the "AND" expressions for each value in the searchString.
        var andMatchPredicates = [NSPredicate]()
    
        for searchString in searchItems {
            // Each searchString creates an OR predicate for: name, yearIntroduced, introPrice.
            //
            // Example if searchItems contains "iphone 599 2007":
            //      name CONTAINS[c] "iphone"
            //      name CONTAINS[c] "599", yearIntroduced ==[c] 599, introPrice ==[c] 599
            //      name CONTAINS[c] "2007", yearIntroduced ==[c] 2007, introPrice ==[c] 2007
            //
            var searchItemsPredicate = [NSPredicate]()
    
            // Name field matching.
            var lhs = NSExpression(forKeyPath: "name")
            var rhs = NSExpression(forConstantValue: searchString)
            var finalPredicate = NSComparisonPredicate(leftExpression: lhs, rightExpression: rhs, modifier: .DirectPredicateModifier, type: .ContainsPredicateOperatorType, options: .CaseInsensitivePredicateOption)
            searchItemsPredicate.append(finalPredicate)
    
            // Channel field matching.
            lhs = NSExpression(forKeyPath: "channel")
            rhs = NSExpression(forConstantValue: searchString)
            finalPredicate = NSComparisonPredicate(leftExpression: lhs, rightExpression: rhs, modifier: .DirectPredicateModifier, type: .ContainsPredicateOperatorType, options: .CaseInsensitivePredicateOption)
            searchItemsPredicate.append(finalPredicate)
    
            // Add this OR predicate to our master AND predicate.
            let orMatchPredicates = NSCompoundPredicate.orPredicateWithSubpredicates(searchItemsPredicate)
            andMatchPredicates.append(orMatchPredicates)
        }
    
        // Match up the fields of the Product object.
        let finalCompoundPredicate = NSCompoundPredicate.andPredicateWithSubpredicates(andMatchPredicates)
    
        let filteredResults = searchResults.filter { finalCompoundPredicate.evaluateWithObject($0) }
    
        // Hand over the filtered results to our search results table.
        let resultsController = searchController.searchResultsController as ResultsTableViewController
        resultsController.filteredPrograms = filteredResults
        resultsController.tableView.reloadData()
    }
    
    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier(Constants.TableViewCell.identifier, forIndexPath: indexPath) as TableCell
        let program=arrayOfPrograms[indexPath.row]
        configureCell(cell, forProgram: program)
        return cell
    }
    
    override func tableView(tableView: UITableView, canEditRowAtIndexPath indexPath: NSIndexPath) -> Bool {
        // Return false if you do not want the specified item to be editable.
        return false 
    }
    
    override func encodeRestorableStateWithCoder(coder: NSCoder) {
        super.encodeRestorableStateWithCoder(coder)
    
        // Encode the title.
        coder.encodeObject(navigationItem.title!, forKey:RestorationKeys.viewControllerTitle)
    
        // Encode the search controller's active state.
        coder.encodeBool(searchController.active, forKey:RestorationKeys.searchControllerIsActive)
    
        // Encode the first responser status.
        coder.encodeBool(searchController.searchBar.isFirstResponder(), forKey:RestorationKeys.searchBarIsFirstResponder)
    
        // Encode the search bar text.
        coder.encodeObject(searchController.searchBar.text, forKey:RestorationKeys.searchBarText)
    }
    
    override func decodeRestorableStateWithCoder(coder: NSCoder) {
        super.decodeRestorableStateWithCoder(coder)
    
        // Restore the title.
        if let decodedTitle = coder.decodeObjectForKey(RestorationKeys.viewControllerTitle) as? String {
            title = decodedTitle
        }
        else {
            fatalError("A title did not exist. In your app, handle this gracefully.")
        }
    
        // Restore the active state:
        // We can't make the searchController active here since it's not part of the view
        // hierarchy yet, instead we do it in viewWillAppear.
        //
        restoredState.wasActive = coder.decodeBoolForKey(RestorationKeys.searchControllerIsActive)
    
        // Restore the first responder status:
        // Like above, we can't make the searchController first responder here since it's not part of the view
        // hierarchy yet, instead we do it in viewWillAppear.
        //
        restoredState.wasFirstResponder = coder.decodeBoolForKey(RestorationKeys.searchBarIsFirstResponder)
    
        // Restore the text in the search field.
        searchController.searchBar.text = coder.decodeObjectForKey(RestorationKeys.searchBarText) as String
    }
    
    }
    

    ResultsTableViewController

    class ResultsTableViewController: BaseTableViewController {
    
    var filteredPrograms=[Program]()
    
    override func viewDidLoad() {
        super.viewDidLoad()
        println("results table view")
    }
    
    override func didReceiveMemoryWarning() {
        super.didReceiveMemoryWarning()
        // Dispose of any resources that can be recreated.
    }
    
    override func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        println("results table view nb of raws in section")
        return filteredPrograms.count
    }
    
    override func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        //let cell=tableView.dequeueReusableCellWithIdentifier(Constants.TableViewCell.identifier) as TableCell!
        let cell = tableView.dequeueReusableCellWithIdentifier(Constants.TableViewCell.identifier, forIndexPath: indexPath) as TableCell
        let program = filteredPrograms[indexPath.row]
        configureCell(cell, forProgram: program)
        return cell
    }
    }