需求一:tableViewCell上添加一个详情按钮,点击实现效果和点击cell一致,需要获取数据源然后跳转。
解决方法:
1.由cell处理点击,添加一个label代替button或者将button的点击用户交互取消(.isUserInteractionEnabled),点击事件交给cell处理。
2.调用代理实现 :
tableView.delegate?.tableView(tableView, didSelectRowAt: IndexPath(row: 0, section: 0))//首行响应点击
需求二:tableViewCell上添加多个按钮,比如购买、收藏、查看详情
此时点击不同的按钮就会实现不同的功能或者页面跳转。这时候就需要知道具体的事件源。
当然“查看详情”依然可以像需求一一样简单处理。
方案1:事件处理放在自定义Cell中
将页面跳转在cell的按钮的selector中执行
自定义cell的时候,你需要做四件事
a.将数据模型传入cell,
b.在cell内查找到依赖控制器(或者通过weak 属性传入),比如push的NavigationController等
获取cell当前控制器的方法【Swfit】
///获取某个view的Controller /// - parameter className : 控制器的AnyClass格式 /// - parameter subview : 需要获取的视图 fileprivate func getController(className:AnyClass , from subview:UIView) -> UIViewController? { var nextView = subview.superview repeat { print("uiview:\(nextView)") if let nextResponder = nextView?.next, nextResponder.isKind(of: className){ return nextResponder as! UIViewController } nextView = nextView?.superview }while nextView != nil print("_____!!!!!_____:没有找到响应控制器") return nil }
【Objective-C】一般是通过类扩展添加属性的。- (UIViewController *)viewController { for (UIView *view = self; view; view = view.superview) { UIResponder *nextResponder = [view nextResponder]; if ([nextResponder isKindOfClass:[UIViewController class]]) { return (UIViewController *)nextResponder; } } return nil; }
b.将按钮的事件响应放在cell类中,添加target的Selector方法或者处理IBAction,
完整流程代码示例:
import UIKit
class TableViewCell: UITableViewCell {
var orderDetail:Order! //要传递的数据模型
@IBOutlet weak var lookDetailBt:UIButton!
override func awakeFromNib() {
super.awakeFromNib()
self.selectionStyle = .none
lookDetailBt.addTarget(self, action: #selector(buttonTap(sender:)), for: .touchUpInside)//绑定按钮target
}
///获取某个view的Controller
/// - parameter className : 控制器的AnyClass格式
/// - parameter subview : 需要获取的视图
fileprivate func getController(className:AnyClass , from subview:UIView) -> UIViewController? {
var nextView = subview.superview
repeat {
print("uiview:\(nextView)")
if let nextResponder = nextView?.next, nextResponder.isKind(of: className){
return nextResponder as! UIViewController
}
nextView = nextView?.superview
}while nextView != nil
print("_____!!!!!_____:没有找到响应控制器")
return nil
}
//按钮的selector方法
@objc func buttonTap(sender : UIButton) -> Void {
//做业务处理和逻辑(如果有多按钮,可以通过tag区分)
//准备下一个页面的VC
let storyboard = UIStoryboard(name: "Main", bundle: nil)
let vc1 = storyboard.instantiateViewController(withIdentifier: "OrderDetailController") as! OrderDetailController
vc1.orderId = 715
vc1.stateStr = "测试"
//获取当前cell的控制器和navigationController
let vc = getController(className: UIViewController.classForCoder(), from:sender )
if let navigationController = vc?.navigationController{
//跳转
navigationController.pushViewController(vc1, animated: true)
}
}
}
缺点:获取数据比较单一,无法获取更多控件值和状态。
方案2:将按钮的事件处理放在tableView的控制器中。
可以获得的东西较多。不用传递数据模型到cell中
a..将tableview的代理的cellforItem设置时绑定target和tag值
cell.lookDetailBt.addTarget(self, action: #selector(buttonTap(sender:)), for: .touchUpInside) cell.lookDetailBt.tag = 100 + indexPath.section
b.将按钮的selector方法写在控制器中,
@objc func buttonTap(sender:UIButton) -> Void { //获取到按钮对应的cell及index let cell = getView(className: TableViewCell.classForCoder(), from: sender) as! TableViewCell //自定义搜索cell对象 let index = tableView.indexPath(for: cell) print("____获取到的index:\(index?.section)") //筛选数据源、做页面跳转、做业务逻辑等操作,还可以绑定不同的tag值区分按钮做不同处理 }
通过子View搜索cell对象的代码段【Swift】///获取某个view的目的父View /// - parameter className : UIVeiw的详细AnyClass格式,比如 UITableViewCell.classForCoder() /// - parameter subview : 入口View,从改View一层一层往上找 fileprivate func getView(className:AnyClass , from subview:UIView) -> UIView? { var nextView = subview repeat { if nextView.isKind(of: className){ print("uiview:\(String(describing: nextView))") return nextView as? UIView } nextView = nextView.superview! }while nextView != nil print("_____!!!!!_____:没有找到目的View") return nil }
方案3:自定义cell的时候写一个按钮的代理,将事件通过代理传递到控制器来,实现按钮的响应。
需求三:tableView 默认选中某行。
参考代码(OC):
[self.popTableView selectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0] animated:YES scrollPosition:UITableViewScrollPositionTop] ; //默认选中首行,只是样式上的
[self.popTableView.delegate tableView:self.popTableView didSelectRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:0]]; //可以响应点击代理。