思路:遍历UITabBar 判断所属类
import UIKit
//代理使用
protocol LYTabBarDelegate : NSObjectProtocol {
//方法
func didSelectedComposeButton()
}
class LYBaseView: UITabBar {
//代理对象使用 weak 修饰
weak var lyDelegate: LYTabBarDelegate?
//定义了闭包属性 使用闭包传递数据
//var call Back:()->() 延时初始化
var callBack:(()->())?
lazy var composeButton:UIButton = {
let button = UIButton()
button.addTarget(self, action: #selector(composeButtonAction), for: .touchUpInside)
button.setBackgroundImage(UIImage(named:"tabbar_compose_button"), for: .normal)
button.setBackgroundImage(UIImage(named:"tabbar_compose_button.highlighted"), for: .highlighted)
button.setImage(UIImage(named: "tabbar_compose_icon_add"), for: .normal)
button.setImage(UIImage(named: "tabbar_compose_icon_add_highlighted"), for: .highlighted)
button.sizeToFit()
return button
}()
//重写这个方法,表示使用手写的方式创建对象
override init(frame: CGRect) {
super.init(frame: frame)
setupUI()
}
//加载sb 和 xib 会调用
required init?(coder aDecoder: NSCoder) {
fatalError("init(coder:) has not been implemented")
//使用xib方式创建对象
}
}
extension LYBaseView {
func setupUI() {
self.addSubview(composeButton)
}
//调整子控件布局
override func layoutSubviews() {
super.layoutSubviews()
// composeButton.center.x = self.frame.size.width/2
// composeButton.center.y = self.frame.size.height/2
composeButton.center.x = width/2
composeButton.center.y = height/2
// let buttonWidth = self.frame.size.width/5
let buttonWidth = width/5
//记录当前遍历的是第几个系统的按钮
var index = 0
for childView in subviews {
// print(childView)
if childView.isKind(of: NSClassFromString("UITabBarButton")!) {
// childView.frame.size.width = buttonWidth
childView.width = buttonWidth
childView.frame.origin.x = CGFloat(index) * buttonWidth
index += 1
if index == 2 {
index += 1
}
}
}
}
//不想在外界访问该函数
//按钮的点击事件 由运行循环监听 以消息机制 传递 swift在编译时就决定了事件该怎么使用
//添加了 private修饰的事件函数在 swift 的运行循环里是找不到,编译型语言,编译的时候就需要知道执行的是哪个函数
//@objc 告诉编译器 使用oc的方式调用 这个方法:oc是基于运行时,使用kvc动态派发
@objc fileprivate func composeButtonAction() {
// func composeButtonAction() {
lyDelegate?.didSelectedComposeButton()
//闭包执行
callBack?()
}
//不想被其他文件访问,private 使得这个函数不能被『外部』调用,而 Selector 对它的调用,实际上就是一种『外部调用』
// KVO 是基于 KVC 和动态派发技术(Dynamic Dispatch)的,Swift 为了效率禁用了 Objective-C 中有的动态派发。那么你把这个动态特性再加回来就可以了~
// private 能够保证方法私有,仅在当前对象能够访问
// @objc 允许这个函数在运行时 通过OC的消息机制被调用
}
import UIKit
class LYMainViewController: UITabBarController {
override func viewDidLoad() {
super.viewDidLoad()
//tabBar只读属性
//self.tabBar = LYBaseView
let lyTabBar = LYBaseView()
setValue(lyTabBar, forKey: "tabBar")
//或者 weak var weakSelf = self
//闭包执行会返回到定义的地方
lyTabBar.callBack = { [weak self] in
print("发布的操作")
}
lyTabBar.lyDelegate = self
addChildViewController(childController: LYHomeTableViewController(), imageName: "tabbar_home", title: "首页")
addChildViewController(childController: LYMessageTableViewController(), imageName: "tabbar_message_center", title: "消息")
addChildViewController(childController: LYDiscoverTableViewController(), imageName: "tabbar_discover", title: "发现")
addChildViewController(childController: LYProfileTableViewController(), imageName: "tabbar_profile", title: "我的")
}
}
extension LYMainViewController : LYTabBarDelegate{
func addChildViewController(childController: UIViewController,imageName: String,title: String) {
childController.tabBarItem.image = UIImage(named: imageName)
childController.tabBarItem.selectedImage = UIImage(named: imageName + "_selected")?.withRenderingMode(.alwaysOriginal)
//颜色
childController.tabBarItem.setTitleTextAttributes([NSForegroundColorAttributeName: UIColor.orange], for: .selected)
//大小
childController.tabBarItem.setTitleTextAttributes([NSFontAttributeName: UIFont .systemFont(ofSize: 12)], for: .normal)
childController.title = title
let nav = UINavigationController(rootViewController: childController)
addChildViewController(nav)
}
func didSelectedComposeButton() {
print("代理对象调用方法")
}
}