通知不适用于 UISegment 控制器

Posted

技术标签:

【中文标题】通知不适用于 UISegment 控制器【英文标题】:Notification is not working for UISegmentController 【发布时间】:2018-02-20 13:52:47 【问题描述】:

我有一个表格视图。 我将 Cell 注册如下:

tableView.register(TVCellElementProperties.self, forCellReuseIdentifier: cellId1)

TVCellElementProperties 中,我手动创建了一个分段控制器,如下所示:

    let unitTypeSegmentedControl: UISegmentedControl = 
    let types = ["Blue", "White"]
    let sc = UISegmentedControl(items: types)
    sc.selectedSegmentIndex = 0
    sc.translatesAutoresizingMaskIntoConstraints = false
    sc.tintColor = UIColor.darkBlue
    sc.addTarget(self, action: #selector(handleUnitChange), for: .valueChanged)

    return sc
()

@objc func handleUnitChange() 
    NotificationCenter.default.post(name: .unitPicked, object: self)

所以,我认为当我更改 SegmentController 中的值时,它应该将我重定向到函数 handleUnitChange()

在tableView中,我将这一行插入到ViewDidLoad中:

NotificationCenter.default.addObserver(self, selector: #selector(unitPicked), name: .unitPicked, object: nil)

当我运行应用程序时,不会调用 tableviewCell 中的函数 **handleUnitChange**。 我做错了什么? 我怎么知道我点击了什么?

编辑: 我正在调用一个 setupView 函数,该函数负责从 UITableViewCell 内部的 init 将 Segment Controller 插入到 Cell 中,如下所示:

override init(style: UITableViewCellStyle, reuseIdentifier: String?) 
    super.init(style: style, reuseIdentifier: reuseIdentifier)
    self.setupViews()
    print("test init")

所以,当我运行它时,setupView 只被调用一次,因此 ** handleUnitChange** 也只被调用一次。 我的意思是说,当应用程序启动并运行时,当我点击 TableView 中的段控制器时,函数 handleUnitChange 不再被调用。

我尝试从 CellForRow 中的 Cell 调用该函数,但与上面相同。我在Segment Controller里面点击的函数**handleUnitChange**没有被调用超时。

if let cell = tableView.dequeueReusableCell(withIdentifier: cellId1, for: indexPath) as? TVCellElementProperties 

    //cell.backgroundColor = UIColor.rgb(red: 12, green: 122, blue: 12)
    //cell.contentView.isUserInteractionEnabled = false
    cell.setupViews()
    //print("\(cell.handleUnitChange(sender: u))")
    cell.handleUnitChange(sender: cell.unitTypeSegmentedControl)
    return cell

【问题讨论】:

@objc func handleUnitChange(sender: UISegmentedControl) print(sender.selectedSegmentIndex) 您不必在 ViewDidLoad 中添加任何内容。不过,您可能必须使用 tableViewCell 内的闭包来触发操作。 你能把你在你的单元格中做的代码贴出来吗? 检查这里给出的任何解决方案是否有效? ***.com/questions/18848689/… 没有。很抱歉,但 cell.handleUnitChange(sender: cell.unitTypeSegmentedControl) 对我来说没有意义。 【参考方案1】:

第 1 步

我们首先处理UITableViewCell

class ProfileTableViewCell: UITableViewCell 
    // MARK: - IBOutlet
    @IBOutlet var firstLabel: UILabel!
    @IBOutlet var lastLabel: UILabel!
    @IBOutlet var ageLabel: UILabel!
    @IBOutlet var pictureImageView: UIImageView!
    @IBOutlet weak var segmentControl: UISegmentedControl! // (a)

    // MARK: - awakeFromNib
    override func awakeFromNib() 
        super.awakeFromNib()
        firstLabel.backgroundColor = UIColor.clear
        lastLabel.backgroundColor = UIColor.clear
        ageLabel.backgroundColor = UIColor.clear
        segmentControl.addTarget(self, action: #selector(valueChanged), for: .valueChanged) // (b)
    

    var onSegmentChanged: ((Int, Int) -> Void)? // (c)

    // (d)
    @objc func valueChanged(sender: UISegmentedControl) 
        onSegmentChanged!(sender.tag, sender.selectedSegmentIndex)
    

首先,通过 Interface Builder 添加一个IBOutlet 连接 (a)。通过 awakFromNib 设置选择器目标 (b)。为了通过UITableView 接收用户的操作,您必须发送一个带有闭包的信号 (c)。最后,添加一个目标接收器(c)。

第 2 步

我们现在使用UIViewController

class HomeViewController: UIViewController, UITableViewDataSource, UITableViewDelegate 
    func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int 
        guard let people = fetchedResultsController.fetchedObjects else  return 0 
        return people.count
    

    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell 
        let cell = tableView.dequeueReusableCell(withIdentifier: "cell", for: indexPath) as! ProfileTableViewCell
        let person = fetchedResultsController.object(at: indexPath)
        cell.segmentControl.tag = indexPath.row // (e)
        // action //
        (f)
        cell.onSegmentChanged =  tag, selectedSegment in
            // (g) self.segmentTapped(tag: tag, index: selectedSegment)
            // print(tag, selectedSegment)
        
        return cell
    

    func tableView(_ tableView: UITableView, commit editingStyle: UITableViewCellEditingStyle, forRowAt indexPath: IndexPath) 
        if editingStyle == .delete 
            // fetching data //
            let person = fetchedResultsController.object(at: indexPath)

            // deleting data //
            person.managedObjectContext?.delete(person)

            // saving the change //
            let context = persistentContainer.viewContext
            do 
                try context.save()
                print("saved...")
             catch 
                print("failed saving")
            
        
    

    (h)
    func segmentTapped(tag: Int, index: Int) 
        print(tag, index)
    

当您通过UITableViewCell 发送用户的操作时,您必须使用UITableViewcellForRowAt 委托方法通过UIViewController 接收它。为了查看用户选择了哪个 segmentControl,您设置了一个标签 (e)。闭包 (f) 在返回 cell 之前设置。您可以在闭包内打印结果。如果要从委托方法中取出结果,请参见 (g) 和 (h)。

警告

如果您仔细查看 (e) 行,indexPath.row 被用作标记。仅当您知道没有删除任何表行时,此方法才有效。否则,您的表数据必须有一个字典值,为每一行提供一个唯一编号。

测试

看看它是如何工作的。当用户使用第二行的段控件时,您将通过cellForRowAt 接到电话。结果是 2 和 1,行中为 2,selectedSegmentIndex 值中为 1。用户不必选择任何表格行。

【讨论】:

很好的解释。非常感谢

以上是关于通知不适用于 UISegment 控制器的主要内容,如果未能解决你的问题,请参考以下文章

iOS:Firebase 通知不适用于用户段

“此游戏不适用于您的手机” Facebook 通知错误

Firebase 云消息传递 - PHP Rest API 不适用于 iOS

Riverpod StateNotifier 不适用于悬停逻辑

预授权不适用于控制器

UINavigationController 不适用于 SearchDisplay 控制器