2016-11-20 5 views
0

Я пытаюсь создать чат-приложения с tableview для отображения сообщений. Все работает отлично, за исключением того, что некоторые ячейки просто не будут отображаться. Это не всегда одни и те же ячейки. Я получаю сообщения из базы данных Firebase.Некоторые ячейки не будут отображаться

Мой код:

func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 

    if let cell = cellCache.object(forKey: indexPath as AnyObject) as? UITableViewCell{ 
     return cell 
    } 

    let cell = messagesTableView.dequeueReusableCell(withIdentifier: "otherMessageCell") as! OtherMessageTableViewCell 

    DispatchQueue.global(qos: .userInteractive).async { 

     let message = self.messages[indexPath.row] 
     let ref = FIRDatabase.database().reference() 

     let uid = FIRAuth.auth()?.currentUser?.uid 

     if(uid == message.sender) { 
      cell.sender.textAlignment = .right 
      cell.message.textAlignment = .right 
     }else{ 
      cell.sender.textAlignment = .left 
      cell.message.textAlignment = .left 
     } 

     let uidReference = ref.child("Users").child(message.sender!) 

     uidReference.observeSingleEvent(of: .value, with: { (snapshot) in 

      if let dictionary = snapshot.value as? [String: AnyObject] { 
       let username = dictionary["Username"] as! String 
       let imageLink = dictionary["Image Link"] as! String 

       cell.sender.text = username 
       cell.message.text = message.message 
       cell.profileImage.image = nil 
       cell.profileImage.loadImageWithURLString(urlString: imageLink) 
      } 

     }, withCancel: nil) 
    } 

    DispatchQueue.main.async { 
     self.cellCache.setObject(cell, forKey: indexPath as AnyObject) 
    } 

    return cell 
} 

Пример: enter image description here

Я надеюсь, что кто-то сможет мне помочь. Благодаря

+0

Ваш код активно сражается (и, возможно, разбивает) рамки. И вызовы DispatchQueue, и этот «cellCache» thingamabob - это ненужные трещины, которые мешают правильному повторному использованию ячеек. – vzsg

ответ

0

Я нашел решение:

var savedSenders = [Int: String]() 
var savedMessages = [Int: String]() 
var savedImages = [Int: UIImage]() 
var savedAlignments = [Int: NSTextAlignment]() 

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell { 

    let cell = messagesTableView.dequeueReusableCell(withIdentifier: "otherMessageCell") as! OtherMessageTableViewCell 

    if(savedSenders[indexPath.row] != nil && savedMessages[indexPath.row] != nil && savedImages[indexPath.row] != nil && savedAlignments[indexPath.row] != nil) { 
     cell.sender.textAlignment = savedAlignments[indexPath.row]! 
     cell.message.textAlignment = savedAlignments[indexPath.row]! 

     cell.sender.text = savedSenders[indexPath.row] 
     cell.message.text = savedMessages[indexPath.row] 
     cell.profileImage.image = savedImages[indexPath.row] 

     return cell 
    } 

    cell.sender.text = "" 
    cell.message.text = "" 
    cell.profileImage.image = nil 

    DispatchQueue.global(qos: .userInteractive).async { 

     let message = self.messages[indexPath.row] 
     let ref = FIRDatabase.database().reference() 

     let uid = FIRAuth.auth()?.currentUser?.uid 

     if(uid == message.sender) { 
      cell.sender.textAlignment = .right 
      cell.message.textAlignment = .right 
      self.savedAlignments.updateValue(.right, forKey: indexPath.row) 
     }else{ 
      cell.sender.textAlignment = .left 
      cell.message.textAlignment = .left 
      self.savedAlignments.updateValue(.left, forKey: indexPath.row) 
     } 

     let uidReference = ref.child("Users").child(message.sender!) 

     uidReference.observeSingleEvent(of: .value, with: { (snapshot) in 

      if let dictionary = snapshot.value as? [String: AnyObject] { 
       let username = dictionary["Username"] as! String 
       let imageLink = dictionary["Image Link"] as! String 

       cell.sender.text = username 
       cell.message.text = message.message 
       cell.profileImage.image = nil 
       cell.profileImage.loadImageWithURLString(urlString: imageLink) 

        let url = URL(string: imageLink) 
        URLSession.shared.dataTask(with: url!) { (data, response, error) in 

         if(error != nil){ 
          print(error as Any) 
          return 
         } 

         DispatchQueue.main.async { 
          if let downloadedImage = UIImage(data: data!) { 
           let image = downloadedImage 

           self.savedSenders.updateValue(username, forKey: indexPath.row) 
           self.savedMessages.updateValue(message.message!, forKey: indexPath.row) 
           self.savedImages.updateValue(image, forKey: indexPath.row) 
          } 
         } 
        }.resume() 
      } 

     }, withCancel: nil) 
    } 

    return cell 
} 

Данные для каждой ячейки сохраняется в массивы (savedSenders, savedMessages, savedImages и savedAlignments). Если я не сохраню его в массивах, ячеям придется загружать все данные из Firebase, и это займет больше времени. Если это займет больше времени, это не будет выглядеть хорошо. Я тестировал все, и он работает.

Смежные вопросы