协议方法没有传递正确的参数
Protocol method does not pass the correct parameter
考虑这种情况,假设您有一个 UIView 的子类和一个协议,即
protocol MyViewProtocol
{
func didTapSelf(thisView: MyView)
}
class MyView: UIView
{
lazy var tf: UITextField =
{
let obj = UITextField()
obj.translatesAutoresizingMaskIntoConstraints = false
obj.placeholder = "Enter text"
obj.backgroundColor = .white
return obj
}()
var btn: UIButton =
{
let obj = UIButton()
obj.translatesAutoresizingMaskIntoConstraints = false
obj.addTarget(self, action: #selector(didTapBtn), for: .touchUpInside)
obj.backgroundColor = .orange
return obj
}()
var delegate: MyViewProtocol?
init(backgroundColor: UIColor, placeholder: String, delegate: MyViewProtocol?)
{
super.init(frame: .zero)
self.backgroundColor = backgroundColor
self.delegate = delegate
tf.placeholder = placeholder
configureUI()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
func configureUI()
{
addAllSubviews(tf,
btn)
NSLayoutConstraint.activate([
btn.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 12),
btn.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -12),
btn.topAnchor.constraint(equalTo: topAnchor, constant: 32),
tf.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 12),
tf.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -12),
tf.topAnchor.constraint(equalTo: btn.bottomAnchor, constant: 12),
tf.bottomAnchor.constraint(equalTo: bottomAnchor, constant: -12),
tf.heightAnchor.constraint(equalToConstant: 44)
])
// let gesture = UITapGestureRecognizer(target: self, action: #selector(didTapBtn))
// gesture.cancelsTouchesInView = false
// addGestureRecognizer(gesture)
}
@objc func didTapBtn()
{
delegate?.didTapSelf(thisView: self)
}
}
现在,在您的视图控制器中,创建 2 个 MyView 实例,即
lazy var myView1: MyView =
{
let obj = MyView(backgroundColor: .yellow, placeholder: "view 1", delegate: self)
obj.translatesAutoresizingMaskIntoConstraints = false
return obj
}()
lazy var myView2: MyView =
{
let obj = MyView(backgroundColor: .green, placeholder: "view 2", delegate: self)
obj.translatesAutoresizingMaskIntoConstraints = false
return obj
}()
并使用如下协议方法实现您的视图控制器:
extension ViewController: MyViewProtocol
{
func didTapSelf(thisView: MyView)
{
if thisView == myView1
{
print("tapped view 1")
}
else if thisView == myView2
{
print("tapped view 2")
}
else
{
print("i am ambigious")
}
}
}
现在,当您点击 myView1
或 myView2
中的按钮时,它会正确打印,即点击视图 1 或 2,无论点击哪个按钮,但是当您开始编辑 myView1
textField 和处于编辑状态的时间,无论你点击 myView1
或 myView2
中的哪个按钮,它只会打印点击的视图 1,或任何 textField 处于编辑模式的视图。
据我所知,无论 textField 的编辑状态如何,它都必须传递正确的 self 对象。为什么会出现这种行为?
p.s当您在上述任何视图上使用点击手势时,不会发生此行为。
这实际上与协议或故事中的任何其他转移话题无关。稍微调试一下就会发现实际上点击了正确的按钮,但运行时确实将目标-操作消息发送到错误的 MyView。根本原因是在这一行中:
obj.addTarget(self, action: #selector(didTapBtn), for: .touchUpInside)
...self
一词并不代表您认为的那样。
要解决此问题,只需更改
var btn: UIButton =
到
lazy var btn: UIButton =
考虑这种情况,假设您有一个 UIView 的子类和一个协议,即
protocol MyViewProtocol
{
func didTapSelf(thisView: MyView)
}
class MyView: UIView
{
lazy var tf: UITextField =
{
let obj = UITextField()
obj.translatesAutoresizingMaskIntoConstraints = false
obj.placeholder = "Enter text"
obj.backgroundColor = .white
return obj
}()
var btn: UIButton =
{
let obj = UIButton()
obj.translatesAutoresizingMaskIntoConstraints = false
obj.addTarget(self, action: #selector(didTapBtn), for: .touchUpInside)
obj.backgroundColor = .orange
return obj
}()
var delegate: MyViewProtocol?
init(backgroundColor: UIColor, placeholder: String, delegate: MyViewProtocol?)
{
super.init(frame: .zero)
self.backgroundColor = backgroundColor
self.delegate = delegate
tf.placeholder = placeholder
configureUI()
}
required init?(coder: NSCoder) {
fatalError("init(coder:) has not been implemented")
}
func configureUI()
{
addAllSubviews(tf,
btn)
NSLayoutConstraint.activate([
btn.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 12),
btn.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -12),
btn.topAnchor.constraint(equalTo: topAnchor, constant: 32),
tf.leadingAnchor.constraint(equalTo: leadingAnchor, constant: 12),
tf.trailingAnchor.constraint(equalTo: trailingAnchor, constant: -12),
tf.topAnchor.constraint(equalTo: btn.bottomAnchor, constant: 12),
tf.bottomAnchor.constraint(equalTo: bottomAnchor, constant: -12),
tf.heightAnchor.constraint(equalToConstant: 44)
])
// let gesture = UITapGestureRecognizer(target: self, action: #selector(didTapBtn))
// gesture.cancelsTouchesInView = false
// addGestureRecognizer(gesture)
}
@objc func didTapBtn()
{
delegate?.didTapSelf(thisView: self)
}
}
现在,在您的视图控制器中,创建 2 个 MyView 实例,即
lazy var myView1: MyView =
{
let obj = MyView(backgroundColor: .yellow, placeholder: "view 1", delegate: self)
obj.translatesAutoresizingMaskIntoConstraints = false
return obj
}()
lazy var myView2: MyView =
{
let obj = MyView(backgroundColor: .green, placeholder: "view 2", delegate: self)
obj.translatesAutoresizingMaskIntoConstraints = false
return obj
}()
并使用如下协议方法实现您的视图控制器:
extension ViewController: MyViewProtocol
{
func didTapSelf(thisView: MyView)
{
if thisView == myView1
{
print("tapped view 1")
}
else if thisView == myView2
{
print("tapped view 2")
}
else
{
print("i am ambigious")
}
}
}
现在,当您点击 myView1
或 myView2
中的按钮时,它会正确打印,即点击视图 1 或 2,无论点击哪个按钮,但是当您开始编辑 myView1
textField 和处于编辑状态的时间,无论你点击 myView1
或 myView2
中的哪个按钮,它只会打印点击的视图 1,或任何 textField 处于编辑模式的视图。
据我所知,无论 textField 的编辑状态如何,它都必须传递正确的 self 对象。为什么会出现这种行为?
p.s当您在上述任何视图上使用点击手势时,不会发生此行为。
这实际上与协议或故事中的任何其他转移话题无关。稍微调试一下就会发现实际上点击了正确的按钮,但运行时确实将目标-操作消息发送到错误的 MyView。根本原因是在这一行中:
obj.addTarget(self, action: #selector(didTapBtn), for: .touchUpInside)
...self
一词并不代表您认为的那样。
要解决此问题,只需更改
var btn: UIButton =
到
lazy var btn: UIButton =