微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

Swift 3 – 支持“刷卡删除”的iOS 10 UITableView

关于如何为UITableView启用滑动删除有很多问题,他们都说同样的事情:

覆盖tableView(_:commit editingStyle:forRowAt indexPath :).

除此之外,我已经完成了这项工作,而且我仍然没有刷卡删除功能.我试过的事情:

>在代码和IB中将tableView.allowsMultipleSelectionDuringEditing设置为true和false.
>重写tableView(_:canEditRowAt indexPath :)并返回true.
>重写tableView(_:editingStyleForRowAt indexPath :)并返回.delete.
>以及上面的每一个组合.

我正在使用firebaseui自定义UITableViewCell来填充表格.这是我的表视图控制器:

import UIKit
import FirebaseDatabaseUI

class ScheduleViewController: UITableViewController {

    private let TAG = String(describing: ScheduleViewController.self)

    private var dataSource: FUITableViewDataSource!

    override func viewDidLoad() {
        super.viewDidLoad()

        dataSource = self.tableView.bind(to: DataManager.instance.habitsQuery(),populateCell: populateCell())

        self.tableView.dataSource = dataSource

        // Automatically resize table cells to fit its content.
        self.tableView.estimatedRowHeight = ScheduleTableViewCell.HEIGHT
        self.tableView.rowHeight = UITableViewAutomaticDimension

        // I have also
        self.tableView.allowsMultipleSelectionDuringEditing = false
    }

    func populateCell() -> (UITableView,IndexPath,FIRDataSnapshot) -> UITableViewCell {
        return { tableView,indexPath,snapshot in
            let cell =
                tableView.dequeueReusableCell(withIdentifier: ScheduleTableViewCell.IDENTIFIER,for: indexPath) as! ScheduleTableViewCell

            if let dict = snapshot.value as? Dictionary<String,Any?> {
                cell.set(habit: Habit(withKey: snapshot.key,from: dict))
            } else {
                Log.e(self.TAG,"Invalid data returned from Firebase.")
            }

            return cell
        }
    }

    // MARK: TableView Delegate

    override func tableView(_ tableView: UITableView,canEditRowAt indexPath: IndexPath) -> Bool {
        return true
    }

    override func tableView(_ tableView: UITableView,editingStyleForRowAt indexPath: IndexPath) -> UITableViewCellEditingStyle
    {
        return .delete
    }

    override func tableView(_ tableView: UITableView,commit editingStyle: UITableViewCellEditingStyle,forRowAt indexPath: IndexPath)
    {

    }

    // MARK: - Navigation

    override func prepare(for segue: UIStoryboardSegue,sender: Any?) {
    }

}

解决方法

最近的firebaseui更新打破了原来的答案.

更新的答案:

只需将FUITableViewDataSource子类化为实现自定义UITableViewDataSource功能,然后将子类绑定到UITableView.

FUITableViewDataSource子类:

import UIKit
import FirebaseDatabaseUI

class EditableTableDataSource: FUITableViewDataSource {

    /// Called to populate each cell in the UITableView.
    typealias PopulateCellBlock = (UITableView,FIRDataSnapshot) -> UITableViewCell

    /// Called to commit an edit to the UITableView.
    typealias CommitEditBlock = (UITableView,UITableViewCellEditingStyle,IndexPath) -> Void

    private let commitEditBlock: CommitEditBlock?

    /// A wrapper around FUITableViewDataSource.init(query:view tableView:populateCell:),with the
    /// addition of a CommitEditBlock.
    public init(query: FIRDatabaseQuery,populateCell: @escaping PopulateCellBlock,commitEdit: @escaping CommitEditBlock)
    {
        commitEditBlock = commitEdit
        super.init(collection: FUIArray.init(query: query),populateCell: populateCell)
    }

    override func tableView(_ tableView: UITableView,forRowAt indexPath: IndexPath)
    {
        if (commitEditBlock != nil) {
            commitEditBlock!(tableView,editingStyle,indexPath)
        }
    }

}

extension UITableView {

    /// Creates a data source,binds it to the table view,and returns it. Note that this is the
    /// `EditableTableViewDataSource` equivalent of the 
    /// `FUITableViewDataSource.bind(to:populateCell:)` method.
    ///
    /// - parameters:
    ///   - to:             The Firebase query to bind to.
    ///   - populateCell:   A closure that's called to populate each cell.
    ///   - commitEdit:     A closure that's called when the user commits some kind of edit. Maps to
    ///                     `tableView(:commit:forRowAt:)`.
    func bind(to query: FIRDatabaseQuery,populateCell: @escaping EditableTableDataSource.PopulateCellBlock,commitEdit: @escaping EditableTableDataSource.CommitEditBlock)
        -> EditableTableDataSource
    {
        let dataSource = EditableTableDataSource(query: query,populateCell: populateCell,commitEdit: commitEdit)
        dataSource.bind(to: self)
        return dataSource
    }

}

用法

import UIKit
import FirebaseDatabaseUI

class ScheduleViewController: UITableViewController {

    private let TAG = String(describing: ScheduleViewController.self)

    private var dataSource: FUITableViewDataSource!
    private var dataManager: DataManager!

    override func viewDidLoad() {
        super.viewDidLoad()

        dataManager = AppManager.defaultInstance.dataManager()

        dataSource = tableView.bind(
            to: dataManager.scheduledHabitsQuery(),populateCell: populateCellBlock(),commitEdit: commitEditBlock())
    }


    // MARK: TableView Data Source

    func populateCellBlock() -> EditableTableDataSource.PopulateCellBlock {
        return { tableView,snapshot in
            let cell = ScheduledHabitTableViewCell.from(tableView: tableView,at: indexPath)
            cell.set(habit: ScheduledHabit(fromSnapshot: snapshot))
            return cell
        }
    }

    func commitEditBlock() -> EditableTableDataSource.CommitEditBlock {
        return { tableView,indexPath in
            if (editingStyle != .delete) {
                return
            }

            // Delete the data from Firebase.
            let snapshot = self.dataSource.snapshot(at: indexPath.row)
            self.dataManager.movetoTrash(ScheduledHabit(fromSnapshot: snapshot))

            // Deleting the table view row is done automatically by the firebaseui data source.
        }
    }


    // MARK: - Navigation

    override func prepare(for segue: UIStoryboardSegue,sender: Any?) {
    }

}

原始答案:

解决方案是子类化FUITableViewDataSource并覆盖您想要的UITableViewDataSource方法.之后一切都很完美.

import UIKit
import FirebaseDatabaseUI

class FUIEditableTableViewDataSource: FUITableViewDataSource {

    /// Called to populate each cell in the UITableView.
    typealias PopulateCellBlock = (UITableView,tableView: UITableView,commitEdit: @escaping CommitEditBlock)
    {
        commitEditBlock = commitEdit
        super.init(query: query,view: tableView,indexPath)
        }
    }

}

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。

相关推荐