通知不适用于 UISegmentController
Notification is not working for UISegmentController
我有一个tableView。
我按如下方式注册 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 将段控制器插入到 Cell 中,如下所示:
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
self.setupViews()
print("test init")
}
所以,当我 运行 它时,setupView 只被调用一次,因此 ** handleUnitChange** 也只被调用一次。
我的意思是说,当应用程序启动并 运行ning 时,当我单击 TableView 内的段控制器时,函数 handleUnitChange 不再被调用。
我尝试从 CellForRow 中的 Cell 调用该函数,但与上面相同。函数 ** handleUnitChange** 没有调用超时 我在 Segment Controller 里面点击了。
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 设置一个 select 或目标 (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
发送用户的操作时,您必须使用 UITableView
的 cellForRowAt
委托方法通过 UIViewController
接收它。为了查看用户使用了哪个 segmentControl selected,您设置了一个标记 (e)。在返回 cell
之前设置闭包 (f)。您可以在闭包内打印结果。如果要从委托方法中取出结果,请参见(g)和(h)。
警告
如果您仔细看一下 (e) 行,indexPath.row
用作标记。仅当您知道删除了 table 行中的 none 行时,此方法才有效。否则,您的 table 数据必须具有为每一行提供唯一编号的字典值。
测试
看看它是如何工作的。当用户使用第二行的段控件时,你会通过cellForRowAt
接到电话。结果是 2 和 1,行中的 2,selectedSegmentIndex
值中的 1。用户不必 select 任何 table 行。
我有一个tableView。 我按如下方式注册 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 将段控制器插入到 Cell 中,如下所示:
override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
super.init(style: style, reuseIdentifier: reuseIdentifier)
self.setupViews()
print("test init")
}
所以,当我 运行 它时,setupView 只被调用一次,因此 ** handleUnitChange** 也只被调用一次。 我的意思是说,当应用程序启动并 运行ning 时,当我单击 TableView 内的段控制器时,函数 handleUnitChange 不再被调用。
我尝试从 CellForRow 中的 Cell 调用该函数,但与上面相同。函数 ** handleUnitChange** 没有调用超时 我在 Segment Controller 里面点击了。
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 设置一个 select 或目标 (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
发送用户的操作时,您必须使用 UITableView
的 cellForRowAt
委托方法通过 UIViewController
接收它。为了查看用户使用了哪个 segmentControl selected,您设置了一个标记 (e)。在返回 cell
之前设置闭包 (f)。您可以在闭包内打印结果。如果要从委托方法中取出结果,请参见(g)和(h)。
警告
如果您仔细看一下 (e) 行,indexPath.row
用作标记。仅当您知道删除了 table 行中的 none 行时,此方法才有效。否则,您的 table 数据必须具有为每一行提供唯一编号的字典值。
测试
看看它是如何工作的。当用户使用第二行的段控件时,你会通过cellForRowAt
接到电话。结果是 2 和 1,行中的 2,selectedSegmentIndex
值中的 1。用户不必 select 任何 table 行。