前言
在應(yīng)用開發(fā)時腻惠,某一天我們的產(chǎn)品經(jīng)理興高采烈的和我們說:“我們的產(chǎn)品即將走向國際化,我們要做美國糜芳,英國飒货,德國...進(jìn)行推廣∏涂ⅲ” 這意味著我們需要做國際化版本了塘辅。
怎么會這樣。皆撩。
我們的代碼經(jīng)常會有下面這樣的代碼
cell.textLabel.text = @"我的文本";
或者是在直接在 xib扣墩、StoryBoard 直接設(shè)置屬性了。
國際化
在iOS開發(fā)扛吞,我們是如何實(shí)現(xiàn)國際化的呢呻惕?對這一塊不了解的同學(xué)可以看這篇文章。
雖然 Xib滥比、StoryBoard 都可以設(shè)置國際化亚脆。但我們
還是習(xí)慣全部寫在一個 strings 中,這樣方便做翻譯的同學(xué)進(jìn)行翻譯盲泛。
那這樣我們要進(jìn)行國際化的流程是
這樣實(shí)在太煩了濒持。。這么多控件寺滚。柑营。
使用runtime解決問題
setText 國際化
國際化主要的工作就是在 setText
之前需要調(diào)用 NSLocalizedString
生成國際化后的字符串。
目前代碼使我們糾結(jié)的地方是我們就直接使用 setText
了村视。我們希望在setText
時插入一段國際化的代碼官套。
我們希望在執(zhí)行某個函數(shù)之前插入一段代碼,Runtime的 Method Swizzling
可以實(shí)現(xiàn)這樣的功能。
@implementation UILabel(NewLabel)
+ (void)load {
[UILabel configSwizzled];
}
+ (void)configSwizzled {
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(setText:);
SEL swizzledSelector = @selector(setNewText:);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
});
}
- (void)setNewText:(NSString *)text {
[self setNewText:NSLocalizedString(text, nil)];
}
@end
- 我們使用分類擴(kuò)展
UILabel
奶赔。 - 然后重寫
load
這個函數(shù)惋嚎,在里面進(jìn)行Swizzle的初始化。 - 在這里我們把
setText
SwizzlesetNewText
. - 在
setNewText
中我們我們調(diào)用NSLocalizedString
進(jìn)行國際化處理纺阔。
好了瘸彤,這樣我們解決了在代碼中 setText
的國際化問題。
Xib StoryBoard 國際化
這里我們發(fā)現(xiàn)笛钝,Xib StoryBoard 中設(shè)置屬性的控件不會調(diào)用 setText
质况。
那這我們怎么解決呢? 讓他們調(diào)用一下 setText
吧玻靡。那我們需要怎么做结榄? Xib StoryBoard 的控件,必然會走 initWithCoder
這個初始化函數(shù)囤捻。我們在再次使用 Runtime 的黑魔法臼朗,讓 initWithCoder
執(zhí)行完后,我們在調(diào)用一下 setText
蝎土。
直接看代碼吧:
+ (void)configSwizzled {
...
dispatch_once(&onceToken2, ^{
Class class = [self class];
// When swizzling a class method, use the following:
// Class class = object_getClass((id)self);
SEL originalSelector = @selector(initWithCoder:);
SEL swizzledSelector = @selector(initNewWithCoder:);
Method originalMethod = class_getInstanceMethod(class, originalSelector);
Method swizzledMethod = class_getInstanceMethod(class, swizzledSelector);
BOOL didAddMethod =
class_addMethod(class,
originalSelector,
method_getImplementation(swizzledMethod),
method_getTypeEncoding(swizzledMethod));
if (didAddMethod) {
class_replaceMethod(class,
swizzledSelector,
method_getImplementation(originalMethod),
method_getTypeEncoding(originalMethod));
} else {
method_exchangeImplementations(originalMethod, swizzledMethod);
}
});
}
- (instancetype)initNewWithCoder:(NSCoder *)aDecoder {
id result = [self initNewWithCoder:aDecoder];
[self setText:self.text];
return result;
}
So easy J友啤!
不用進(jìn)行國際化的控件怎么辦
我們可以添加一個變量來控制代碼是否進(jìn)行國際化誊涯。那就使用關(guān)聯(lián)對象(Associated Object)
吧挡毅。
@interface UILabel (NewLabel)
@property (nonatomic, assign)IBInspectable BOOL localizedEnlabe;
@end
@implementation UILabel(NewLabel)
static char *localizedEnlabeChar = "LocalizedEnlabe";
- (void)setLocalizedEnlabe:(BOOL)localizedEnlabe {
objc_setAssociatedObject(self, &localizedEnlabeChar, [NSNumber numberWithBool:localizedEnlabe], OBJC_ASSOCIATION_ASSIGN);
}
- (BOOL)localizedEnlabe {
NSNumber *value = objc_getAssociatedObject(self, &localizedEnlabeChar);
if (value) {
return [value boolValue];
}
return YES;
}
@end
- 這里我使用
IBInspectable
屬性方便 Xib StoryBoard 設(shè)置屬性.
總結(jié)
這只是個 Demo, 需要國際化的控件還有 UITextField
暴构, UIButton
等控件跪呈。其實(shí)我們這些代碼可以直接在 UIView
的分類中實(shí)現(xiàn)。
然后把我們要處理的屬性方法以同樣的方式 Swizzle 取逾。
雖然這種方法不見得能解決所有問題耗绿,但應(yīng)該是可以解決 80% 的問題的。