Warning: file_get_contents(/data/phpspider/zhask/data//catemap/9/ios/100.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Ios 通知不适用于UISegmentController_Ios_Swift_Uisegmentedcontrol_Nsnotificationcenter - Fatal编程技术网

Ios 通知不适用于UISegmentController

Ios 通知不适用于UISegmentController,ios,swift,uisegmentedcontrol,nsnotificationcenter,Ios,Swift,Uisegmentedcontrol,Nsnotificationcenter,我有一个桌面视图。 我按如下方式注册该单元格: tableView.register(TVCellElementProperties.self, forCellReuseIdentifier: cellId1) let unitTypeSegmentedControl: UISegmentedControl = { let types = ["Blue", "White"] let sc = UISegmentedControl(items: types) sc

我有一个桌面视图。 我按如下方式注册该单元格:

tableView.register(TVCellElementProperties.self, forCellReuseIdentifier: cellId1)
    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)
}
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
    super.init(style: style, reuseIdentifier: reuseIdentifier)
    self.setupViews()
    print("test init")
}
在TVCellElementProperties中,我手动创建了一个段控制器,如下所示:

tableView.register(TVCellElementProperties.self, forCellReuseIdentifier: cellId1)
    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)
}
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
    super.init(style: style, reuseIdentifier: reuseIdentifier)
    self.setupViews()
    print("test init")
}
因此,我认为当我更改SegmentController内的值时,它应该将我重定向到函数handleUnitChange()

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

NotificationCenter.default.addObserver(self, selector: #selector(unitPicked), name: .unitPicked, object: nil)
运行应用程序时,tableviewCell内的函数**handleUnitChange**未被调用。 我做错了什么? 我怎么知道我点击了什么

编辑: 我正在调用一个setupView函数,该函数负责从UITableViewCell内部的init将段控制器插入单元内部,如下所示:

tableView.register(TVCellElementProperties.self, forCellReuseIdentifier: cellId1)
    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)
}
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
    super.init(style: style, reuseIdentifier: reuseIdentifier)
    self.setupViews()
    print("test init")
}
因此,当我运行它时,setupView只调用一次,因此**handleUnitChange**也只调用一次。 我的意思是,当应用程序启动并运行时,当我单击TableView中的段控制器时,不再调用函数handleUnitChange

我试图从CellForRow中的单元格调用该函数,但同上。我在段控制器内单击的函数**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
}

步骤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
发送用户操作时,您必须通过
UIViewController
使用
UITableView
cellForRowAt
委托方法接收用户操作。为了查看用户选择了哪个分段控件,可以设置一个标记(e)。在返回
单元格
之前设置闭包(f)。您可以在闭包内打印结果。如果要从委托方法中取出结果,请参见(g)和(h)

警告

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

测试

看看它是如何工作的。当用户使用第二行上的段控件时,您将通过
cellForRowAt
收到一个调用。结果为2和1,行中为2,所选分段索引值中为1。用户不必选择任何表行


@objc func handleUnitChange(发件人:UISegmentedControl){print(发件人.selectedSegmentIndex)}您不必在ViewDidLoad中放入任何内容。不过,您可能需要在tableViewCell中使用闭包触发该操作。是否可以发布您在单元格中执行的代码?检查此处给出的任何解决方案是否有效?没有。很抱歉,cell.handleUnitChange(发件人:cell.unitTypeSegmentedControl)对我来说没有意义。很好的解释。谢谢