之前一直很好奇UITableView有一个delegate属性,遵守 <UITableViewDelegate>协议的 ,而它的父类UIScrollView也有一个同名的delegate,但是遵守的是<UIScrollViewDelegate>协议。这是怎么实现的呢?
先上代码
.h文件
#import <UIKit/UIKit.h>
@class CustomTextView;
// 创建自己的协议,同时遵守于父类协议(也就相当于继承子父类协议,实际上协议本身是不能被继承的,遵守就相当于抄了一份过来吧)
@protocol CustomTextViewDelegate <UITextViewDelegate>
// 这里添加自己在原协议方法基础上,新增的一些协议方法
// 示例:
- (void)customTextView:(CustomTextView *)customTextView someNewAction:(BOOL)action;
@end
@interface CustomTextView : UITextView
// 因为delegate此时要遵守我们新创建的协议,而不是原本的协议,所以要重写父类中的属性(这里会有警告,稍后会在.m文件中消除)
@property (nonatomic, weak) id<CustomTextViewDelegate> delegate;
@end
.m文件
#import "CustomTextView.h"
@implementation CustomTextView
@dynamic delegate;// .h中警告说delegate在父类已经声明过了,子类再声明也不会重新生成新的方法了。我们就在这里使用@dynamic告诉系统delegate的setter与getter方法由用户自己实现,不由系统自动生成
#pragma Set & Get
- (void)setDelegate:(id<CustomTextViewDelegate>)delegate
{
super.delegate = delegate;
}
- (id<CustomTextViewDelegate>)delegate
{
id curDelegate = super.delegate;
return curDelegate;
}
// 找个地方触发下新加的协议方法,看是否正常工作
- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event
{
if ([self.delegate respondsToSelector:@selector(customTextView:someNewAction:)]) {
[self.delegate customTextView:self someNewAction:YES];
}
}
上面就是最简单的:继承父类协议,然后在其基础上增加新的协议方法
然而我们更常见的需求是,需要对父类中的协议方法做修改,比如在父类的某协议调用前插入一些操作,然后再调用代理(或者在调用父类某协议后,再插入一些操作)
思路是:将父类的delegate指向自己,自己实现父类中的协议方法,并在其中做修改,然后在适当的时间调用自己的delegate,将事件调用传递出去
通俗些说:有个作者告诉编辑说:“我的文章写好了”(通过作者的delegate告诉编辑),然后编辑说:“我要校审下这个文章,改一改”(编辑实现了作者的delegate中的方法,在其内做修改),最后编辑告诉一个读者:“文章改好了,你可以去读了”(最后通过编辑的delegate将事件传递出去)
需要修改父类协议的做法
.h文件的代码上面一样
#import <UIKit/UIKit.h>
@class CustomTextView;
// 创建自己的协议,同时遵守于父类协议(也就相当于继承子父类协议,实际上协议本身是不能被继承的,遵守就相当于抄了一份过来吧)
@protocol CustomTextViewDelegate <UITextViewDelegate>
// 这里添加自己在原协议方法基础上,新增的一些协议方法
// 示例:
- (void)customTextView:(CustomTextView *)customTextView someNewAction:(BOOL)action;
@end
@interface CustomTextView : UITextView
// 因为delegate此时要遵守我们新创建的协议,而不是原本的协议,所以要重写父类中的属性(这里会有警告,稍后会在.m文件中消除)
@property (nonatomic, weak) id<CustomTextViewDelegate> delegate;
@end
.m文件
#import "CustomTextView.h"
@interface CustomTextView () <UITextViewDelegate> // 这里要遵守父类原本的协议,因为继承类时,协议是不会被继承的,所以需要重新声明
@property (nonatomic, weak) id<CustomTextViewDelegate> myDelegate;// 因为父类的delegate对象是self,self的delegate是外部类,所以需要新增一个myDelegate来保存外部类
@end
@implementation CustomTextView
@dynamic delegate;// .h中警告说delegate在父类已经声明过了,子类再声明也不会重新生成新的方法了。我们就在这里使用@dynamic告诉系统delegate的setter与getter方法由用户自己实现,不由系统自动生成
#pragma mark - Set & Get
- (void)setDelegate:(id<CustomTextViewDelegate>)delegate
{
super.delegate = self;// 这句也可以放在初始化方法中
_myDelegate = delegate;
}
- (id<CustomTextViewDelegate>)delegate
{
return _myDelegate;
}
#pragma mark - UITextViewDelegate
- (BOOL)textViewShouldBeginEditing:(UITextView *)textView
{
BOOL result = NO;
/*
调用前可以做一些操作
。。。
*/
// 将协议事件传递出去
if ([_myDelegate respondsToSelector:@selector(textViewShouldBeginEditing:)]) {
result = [_myDelegate textViewShouldBeginEditing:self];
}
/*
调用后也可以做一些操作
。。。
*/
// 可以在达到某个条件的情况下,调用新增的协议方法
if ([_myDelegate respondsToSelector:@selector(customTextView:someNewAction:)]) {
[_myDelegate customTextView:self someNewAction:YES];
}
return result;
}
/*
建议将原协议的所有方法都实现,不需要插入操作的就直接用_myDelegate传递出去
*/
- (BOOL)textViewShouldEndEditing:(UITextView *)textView
{
BOOL result = NO;
if ([_myDelegate respondsToSelector:@selector(textViewShouldEndEditing:)]) {
result = [_myDelegate textViewShouldEndEditing:self];
}
return result;
}
// 剩余方法类似
- (void)textViewDidBeginEditing:(UITextView *)textView;
- (void)textViewDidEndEditing:(UITextView *)textView;
。。。