一、需求背景介紹
在項目比較成熟的基礎上,遇到了這樣一個需求,應用中需要引入新的字體,需要更換所有Label的默認字體,但是同時,對於一些特殊設置了字體的label又不需要更換。乍看起來,這個問題確實十分棘手,首先項目比較大,一個一個設置所有使用到的label的font工作量是巨大的,並且在許多動態展示的界面中,可能會漏掉一些label,產生bug。其次,項目中的label來源並不唯一,有用代碼創建的,有xib和storyBoard中的,這也將浪費很大的精力。這種情況下,我們可能會有下面兩種處理方式。
二、處理方式
1、使用框架
創建我們自己的BaseLabel類,在其中進行默認字體的設置,並且並不影響在使用過程中特殊設置字體的label,這種方式可以滿足我們的需求,但是並不適於我們的場景,項目已經成熟,重建一個label基類,來讓所有的UILabel都換成它的工作量不會比重新設置所有label字體的工作量小太多。但這也是有優勢的,至少如果下次再換字體,我們就不用麻煩了。
2、使用runtime替換UILabel初始化方法
這是最簡單方便的方法,我們可以使用runtime機制替換掉UILabel的初始化方法,在其中對label的字體進行默認設置。因為Label可以從initWithFrame、init和nib文件三個來源初始化,所以我們需要將這三個初始化的方法都替換掉。
首先,我們創建一個UILabel的類別:
#import <UIKit/UIKit.h>
@interface UILabel (YHBaseChangeDefaultFont)
@end
在其中加入如下代碼:
#import "UILabel+YHBaseChangeDefaultFont.h"
#import <objc/runtime.h>
@implementation UILabel (YHBaseChangeDefaultFont)
/**
*每個NSObject的子類都會調用下面這個方法 在這裡將init方法進行替換,使用我們的新字體
*如果在程序中又特殊設置了字體 則特殊設置的字體不會受影響 但是不要在Label的init方法中設置字體
*從init和initWithFrame和nib文件的加載方法 都支持更換默認字體
*/
+(void)load{
//只執行一次這個方法
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
Class class = [self class];
// When swizzling a class method, use the following:
// Class class = object_getClass((id)self);
//替換三個方法
SEL originalSelector = @selector(init);
SEL originalSelector2 = @selector(initWithFrame:);
SEL originalSelector3 = @selector(awakeFromNib);
SEL swizzledSelector = @selector(YHBaseInit);
SEL swizzledSelector2 = @selector(YHBaseInitWithFrame:);
SEL swizzledSelector3 = @selector(YHBaseAwakeFromNib);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method originalMethod2 = class_getInstanceMethod(class, originalSelector2);
Method originalMethod3 = class_getInstanceMethod(class, originalSelector3);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
Method swizzledMethod2 = class_getInstanceMethod(class, swizzledSelector2);
Method swizzledMethod3 = class_getInstanceMethod(class, swizzledSelector3);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
BOOL didAddMethod2 =
class_addMethod(class,
originalSelector2,
method_getImplementation(swizzledMethod2),
method_getTypeEncoding(swizzledMethod2));
BOOL didAddMethod3 =
class_addMethod(class,
originalSelector3,
method_getImplementation(swizzledMethod3),
method_getTypeEncoding(swizzledMethod3));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
if (didAddMethod2) {
class_replaceMethod(class,
swizzledSelector2,
method_getImplementation(originalMethod2),
method_getTypeEncoding(originalMethod2));
}else {
method_exchangeImplementations(originalMethod2, swizzledMethod2);
}
if (didAddMethod3) {
class_replaceMethod(class,
swizzledSelector3,
method_getImplementation(originalMethod3),
method_getTypeEncoding(originalMethod3));
}else {
method_exchangeImplementations(originalMethod3, swizzledMethod3);
}
});
}
/**
*在這些方法中將你的字體名字換進去
*/
- (instancetype)YHBaseInit
{
id __self = [self YHBaseInit];
UIFont * font = [UIFont fontWithName:@"這裡輸入你的字體名字" size:self.font.pointSize];
if (font) {
self.font=font;
}
return __self;
}
-(instancetype)YHBaseInitWithFrame:(CGRect)rect{
id __self = [self YHBaseInitWithFrame:rect];
UIFont * font = [UIFont fontWithName:@"這裡輸入你的字體名字" size:self.font.pointSize];
if (font) {
self.font=font;
}
return __self;
}
-(void)YHBaseAwakeFromNib{
[self YHBaseAwakeFromNib];
UIFont * font = [UIFont fontWithName:@"這裡輸入你的字體名字" size:self.font.pointSize];
if (font) {
self.font=font;
}
}
@end
在上面的方法中寫入我們想要UILabel默認顯示的字體,我們分別從init,initWithFrame和nib文件創建一個UILabel添加到視圖上,不做任何其他的操作:
UILabel * label = [[UILabel alloc]initWithFrame:CGRectMake(20, 100, 280, 30)];
label.text = @"你是從initWithFrame來的label";
UILabel * label2 = [[UILabel alloc]init];
label2.frame= CGRectMake(20, 200, 280, 30);
label2.text = @"你是從init來的label";
[self.view addSubview:label];
[self.view addSubview:label2];
運行效果如下,可以看出,字體全部換掉了: