做项目有个需求,UIAlertController
加的UIAlertAction
需要改变按钮的颜色,其实需求就是把删除按钮改成红色,这个需求完全不用自定义,下面一行代码就可以解决这个问题:
UIAlertAction *deleteAction = [UIAlertAction actionWithTitle:@"删除此文件夹" style:UIAlertActionStyleDestructive handler:^(UIAlertAction * _Nonnull action) {
}];
这个style
就是红色的,根据苹果官方的定义,“警示”样式的按钮是用在可能会改变或删除数据的操作上。因此用了红色的醒目标识来警示用户。
那么怎么自定义呢?网上查了下,用runtime
就可以解决这个问题,其实之前真的是没有用过runtime
,我把代码贴出来,然后大概解释下,以后就可以为所欲为的使用自定义UIAlertController
了,包括按钮的颜色,title的颜色等都可以自定义,再也不会被UI为难你了。
新建一个Controller
继承自UIAlertController
,然后.h
文件中是这样子的:
#import <UIKit/UIKit.h>
@interface Y2WAlertAction : UIAlertAction
@property (nonatomic,strong) UIColor *textColor; /**< 按钮title字体颜色 */
@end
@interface Y2WAlertController : UIAlertController
@property (nonatomic,strong) UIColor *tintColor; /**< 统一按钮样式 不写系统默认的蓝色 */
@property (nonatomic,strong) UIColor *titleColor; /**< 标题的颜色 */
@property (nonatomic,strong) UIColor *messageColor; /**< 信息的颜色 */
@end
接下来.m
是这样的:
#import "Y2WAlertController.h"
#import <objc/runtime.h>
@interface Y2WAlertController ()
@end
@implementation Y2WAlertAction
//按钮标题的字体颜色
-(void)setTextColor:(UIColor *)textColor
{
_textColor = textColor;
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([UIAlertAction class], &count);
for(int i =0;i < count;i ++){
Ivar ivar = ivars[i];
NSString *ivarName = [NSString stringWithCString:ivar_getName(ivar) encoding:NSUTF8StringEncoding];
if ([ivarName isEqualToString:@"_titleTextColor"]) {
[self setValue:textColor forKey:@"titleTextColor"];
}
}
}
@end
@implementation Y2WAlertController
- (void)viewDidLoad {
[super viewDidLoad];
unsigned int count = 0;
Ivar *ivars = class_copyIvarList([UIAlertController class], &count);
for(int i = 0;i < count;i ++){
Ivar ivar = ivars[i];
NSString *ivarName = [NSString stringWithCString:ivar_getName(ivar) encoding:NSUTF8StringEncoding];
//标题颜色
if ([ivarName isEqualToString:@"_attributedTitle"] && self.title && self.titleColor) {
NSMutableAttributedString *attr = [[NSMutableAttributedString alloc]initWithString:self.title attributes:@{NSForegroundColorAttributeName:self.titleColor,NSFontAttributeName:[UIFont boldSystemFontOfSize:14.0]}];
[self setValue:attr forKey:@"attributedTitle"];
}
//描述颜色
if ([ivarName isEqualToString:@"_attributedMessage"] && self.message && self.messageColor) {
NSMutableAttributedString *attr = [[NSMutableAttributedString alloc] initWithString:self.message attributes:@{NSForegroundColorAttributeName:self.messageColor,NSFontAttributeName:[UIFont systemFontOfSize:14.0]}];
[self setValue:attr forKey:@"attributedMessage"];
}
}
//按钮统一颜色
if (self.tintColor) {
for (Y2WAlertAction *action in self.actions) {
if (!action.textColor) {
action.textColor = self.tintColor;
}
}
}
}
思路就是通过runtime
来遍历UIAlertAction
和UIAlertController
的属性,然后设置对应按钮的颜色,如下面这句:
NSMutableAttributedString *attr = [[NSMutableAttributedString alloc]initWithString:self.title attributes:@{NSForegroundColorAttributeName:self.titleColor,NSFontAttributeName:[UIFont boldSystemFontOfSize:14.0]}];
[self setValue:attr forKey:@"attributedTitle"];
为attributedTitle
设置颜色,调用的话是这么用的:
导入头文件:#import "Y2WAlertController.h"
然后在需要使用的地方把系统的改成你自定义的就OK了:
Y2WAlertController *alertController = [Y2WAlertController alertControllerWithTitle:@"你确定要退出吗?" message:nil preferredStyle:UIAlertControllerStyleActionSheet];
alertController.tintColor = [UIColor blueColor]; //这里统一设置各个按钮的颜色都为红色.
alertController.titleColor = [UIColor grayColor];
//取消
Y2WAlertAction *cancelAction = [Y2WAlertAction actionWithTitle:@"我不想退出" style:UIAlertActionStyleCancel handler:nil];
//单独修改一个按钮的颜色
cancelAction.textColor = [UIColor greenColor];
[alertController addAction:cancelAction];
//退出
Y2WAlertAction *exitAction = [Y2WAlertAction actionWithTitle:@"退出" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
}];
Y2WAlertAction *deleAction = [Y2WAlertAction actionWithTitle:@"删除" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
}];
deleAction.textColor = [UIColor redColor];
[alertController addAction:deleAction];
[alertController addAction:exitAction];
[self presentViewController:alertController animated:YES completion:nil];
效果图是这样子的:
大概就这么解决这个问题,以后就可以用自定义的
UIAlertController
。