从 `ViewWillAppear` 触发 `TableViewCell` 函数

Posted

技术标签:

【中文标题】从 `ViewWillAppear` 触发 `TableViewCell` 函数【英文标题】:Trigger `TableViewCell` function from `ViewWillAppear` 【发布时间】:2018-03-06 12:52:01 【问题描述】:

我想ViewWillAppear 触发func updateCellValueLabel(...),以便在我打开视图时我的数据是最新的。

我可能想多了,因为TableViewCell: 参数让我可以访问indexPath,但我不知道如何调用它。

视图控制器:

class WalletTableViewController: UIViewController, UITextFieldDelegate 

override func viewDidLoad() 
    super.viewDidLoad()

    registerForKeyboardNotifications()
    self.hideKeyboard()

    self.tableView.separatorStyle = UITableViewCellSeparatorStyle.none
    tableView.delegate = self
    tableView.dataSource = self

    if CDHandler.fetchObject() != nil 
        cryptos = CDHandler.fetchObject()!
        tableView.reloadData()
    

    erc20AddressTextField.delegate = self

    WalletTableViewCell.init().updateCellValue()


override func viewWillAppear(_ animated: Bool) 

    tableView.delegate = self
    tableView.dataSource = self


    if CDHandler.fetchObject() != nil 
        cryptos = CDHandler.fetchObject()!
        tableView.reloadData()
    

    WalletTableViewCell.init().updateCellValue()


我的表格视图扩展:

extension WalletTableViewController: UITableViewDelegate, UITableViewDataSource, CryptoCellDelegate 


func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
    return cryptos.count

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

    let cell = tableView.dequeueReusableCell(withIdentifier: "Cell", for: indexPath) as! WalletTableViewCell

    cell.cryptoNameLabel.text = cryptos[indexPath.row].name
    cell.cryptoCodeLabel.text = cryptos[indexPath.row].symbol
    cell.amountLabel.text = cryptos[indexPath.row].amount
    cell.amountTextField.placeholder = cryptos[indexPath.row].placeholder

    cell.delegate = self
    cell.amountTextField.delegate = self

    return cell

func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) 
    /*if editingStyle == .delete 
        cryptos.remove(at: indexPath.row)
        tableView.deleteRows(at: [indexPath], with: .fade)
        CDHandler.deleteObject(crypto: cryptos[indexPath.row])
    */
    if editingStyle == .delete 
        let selectedManagedObject = cryptos[indexPath.row]
        CDHandler.deleteObject(entity:"CryptosMO", deleteObject: selectedManagedObject)
        cryptos.remove(at: indexPath.row)
        tableView.deleteRows(at: [indexPath], with: .fade)
    


// Value calculation & label update
//----------------------------------
func updateCellValueLabel(_ walletTableViewCell: WalletTableViewCell) 

    print("CALLED") // <--- This is never printed

    if walletTableViewCell.amountLabel.text == "" 
        walletTableViewCell.amountLabel.text = "0.00000000"
    

    var newCryptos : [CryptosMO] = []
    var doubleAmount = 0.0
    var cryptoPrice = ""
    let indexPath = tableView.indexPath(for: walletTableViewCell)

    if CDHandler.fetchObject() != nil 
        newCryptos = CDHandler.fetchObject()!
        tableView.reloadData()
    

    cryptoPrice = cryptos[(indexPath?.row)!].code!
    guard let cryptoDoublePrice = CryptoInfo.cryptoPriceDic[cryptoPrice] else  return 

    let selectedAmount = newCryptos[(indexPath?.row)!]
    if let amount = selectedAmount.amount 
        doubleAmount = Double(amount)!
    

    amountValue = cryptoDoublePrice * doubleAmount

    if WalletViewController.currencyCode != "" 
        let formatter = NumberFormatter()
        formatter.numberStyle = .currency
        formatter.locale = Locale(identifier: "\(WalletViewController.currencyCode)")
        walletTableViewCell.cryptoValueLabel.text = formatter.string(from: NSNumber(value: amountValue))
    



表格视图单元格:

protocol CryptoCellDelegate 

func cellAmountEntered(_ walletTableViewCell: WalletTableViewCell)
func updateCellValueLabel(_ walletTableViewCell: WalletTableViewCell)

类 WalletTableViewCell: UITableViewCell

func updateCellValue() 
    delegate?.updateCellValueLabel(self)


@IBAction func amountTextFieldEntered(_ sender: Any) 
    delegate?.cellAmountEntered(self)
    delegate?.updateCellValueLabel(self)
  

我想刷新数据时如何调用updateCellValueLabel(...)

【问题讨论】:

你能详细解释一下你现在的场景吗?您要存档什么或您想要的输出是什么。? viewDidLoad 在 vi​​ewWillAppear 之前触发。所以最好在 viewDidLoad() 中更新数据 @ravi.p 这是一个钱包。你能在代码中找到updateCellValueLabel 吗?该函数获取用户存储在核心数据中的金额,并使用amount * coin value 更新钱包值。我希望钱包值保持更新为最新值(在别处获取)。 @JD。不幸的是,它也不是从viewDidLoad() 调用的:/ 当我看到这个WalletTableViewCell().updateCellValue()时,我记得我说过这个(***.com/questions/49047174/…) 【参考方案1】:

函数 updateCellValueLabel(_ walletTableViewCell: WalletTableViewCell) 将 WalletTableViewCell 作为参数,因此如果不将单元格作为参数传递,则无法调用它。因此,如果要在 viewDidAppear 上调用它,则必须循环 tableview并为每个单元格调用它,所以在 viewDidAppear 视图中你应该写类似..

for i in 0...self.tableview.numberOfSections - 1
   self.updateCellValueLabel(_ walletTableViewCell: self.tableview.cellForRow(at: i))

然后如果你想调用它更多次,你可以将它添加到一个函数中并调用该函数

【讨论】:

这看起来不错,但我在(at: i) 收到Cannot convert value of type 'Int' to expected argument type 'IndexPath' 试试for cell in tableView.visibleCells() as! [WalletTableViewCell] self.updateCellValueLabel(_ walletTableViewCell: cell) 现在我在 `updateCellValueLabel()` 中的 cryptoPrice = cryptos[(indexPath?.row)!].code! 得到 Thread 1: EXC_BAD_INSTRUCTIONindexPath又出问题了哈哈 好的,我认为错误是因为该视图尚未出现,并且该代码将在视图上工作确实出现了,但尝试此代码在视图上将出现for section in 0...self.tableView.numberOfSections-1 for row in 0...self.tableView.numberOfRows(inSection: section) - 1 var indexP:IndexPath = IndexPath(row: row, section: section) self.updateCellValueLabel(_ walletTableViewCell: self.tableview.cellForRow(at: indexP)) 如果没有行,我可以添加什么样的保护来避免Thread 1: Fatal error

以上是关于从 `ViewWillAppear` 触发 `TableViewCell` 函数的主要内容,如果未能解决你的问题,请参考以下文章

知道 -viewWillAppear 何时触发但 -viewWillDisappear 尚未触发

使用过渡委托时触发 viewWillAppear

返回导航控制器层次结构时如何触发viewWillAppear?

在我切换选项卡之前,viewWillAppear 不会为属于 UITabBarController 的 UIViewController 触发

Tab Bar 子 viewWillAppear 不会触发,添加 NavigationController 会给出两个导航栏

后台模式下的 viewWillAppear 等效项