最近一个一直在迭代的老项目收到一份新的开发需求,项目需要做国际化适配,简体中文+英文。由于项目中采用了storyboard和纯代码两种布局方式,所以国际化也要同时实现。上网查了些资料,实现了更改系统语言后,修改app内语言的问题。具体国际化方式可以参考下文:
这篇文章讲的比较详细,很容易实现。
这个需求实现后不久,产品又给我提了一个需求,让我要在app内实现语言切换。还好之前的国际化也做了些准备,不慌不慌。
接下来就是方案的选定,通过广泛查阅资料,得出两个备选方案:
方案一:在原国际化版本的基础上做修改,在info.plist文件中新增key="appLanguage"的键值对,保存用户设定的语言类别。通过切换语言类别来改变语言。(例子:微信)
优点:之前有国际化操作的基础,执行起来并不复杂。
缺点:切换完语言后,需要重新创建app keywindow的跟控制器,会有个跳转的过程,用户体验不好。
方案二:切换语言后,发送通知,每个控制器收到通知后,更改语言。(例子:新浪微博)
优点:很自然的切换语言,选择语言后即可切换,不需要重置根控制器,用户体验好。
缺点:每个控制器都得注册接收通知,工作量太大,而且storyboard也得单独处理。
综合两个方案的优缺点,我们选择方案一。
中英切换,就是让App根据自身设置的语言去读取对应的国际化文件。在NSUserDefault中有一个字段:"AppleLanguages",这个字段就是负责存储App语言的字段,默认这个字段会根据系统语言去变动,中文系统他就存储中文,英文系统就存储英文。
废话少说,切换语言的过程上代:
// NTVLocalized.h
#import <Foundation/Foundation.h>
static NSString * const AppLanguage = @"appLanguage";
@interface NTVLocalized : NSObject
+ (NTVLocalized *)sharedInstance;
//初始化多语言功能
- (void)initLanguage;
//当前语言
- (NSString *)currentLanguage;
//设置要转换的语言
- (void)setLanguage:(NSString *)language;
//设置为系统语言
- (void)systemLanguage;
@end
// NTVLocalized.m
#import "NTVLocalized.h"
@implementation NTVLocalized
+ (NTVLocalized *)sharedInstance {
static NTVLocalized *instance = nil;
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
instance = [[NTVLocalized alloc] init];
});
return instance;
}
- (void)initLanguage{
NSString *language=[self currentLanguage];
if (language.length>0) {
NSLog(@"自设置语言:%@",language);
}else{
[self systemLanguage];
}
}
- (NSString *)currentLanguage{
NSString *language=[[NSUserDefaults standardUserDefaults]objectForKey:AppLanguage];
return language;
}
- (void)setLanguage:(NSString *)language{
[[NSUserDefaults standardUserDefaults] setObject:language forKey:AppLanguage];
}
- (void)systemLanguage{
NSString *languageCode = [[NSUserDefaults standardUserDefaults] objectForKey:@"AppleLanguages"][0];
NSLog(@"系统语言:%@",languageCode);
if([languageCode hasPrefix:@"zh-Hans"]){
languageCode = @"zh-Hans";//简体中文
}else if([languageCode hasPrefix:@"en"]){
languageCode = @"en";//英语
}
[self setLanguage:languageCode];
}
@end
当语言设置完成后,需要重新设置keywindow的rootViewController才可以实现语言的切换。
然而这样设置后,我们发现只有NSLocalizedString(key, comment)设置的语言才能正常显示我们需要的语言,storyBoard和xib配置的页面语言不跟着切换。
设置AppleLanguages字段的话,只会在下次启动App才会生效,在App启动后就已经生成了一个Bundle,里面识别好了对应着AppleLanguages的国际化文件,在App运行期间设置这个字段,是不生效的,所以我们去修改这个Bundle,写一个NSBundle的扩展。
// NSBundle+language.h
#import <Foundation/Foundation.h>
@interface NSBundle (language)
// 设置语言
+ (void)setLanguage:(NSString *)language;
@end
// NSBundle+language.m
#import "NSBundle+language.h"
#import <objc/runtime.h>
static const char _bundle = 0;
@interface BundleEx : NSBundle
@end
@implementation BundleEx
- (NSString *)localizedStringForKey:(NSString *)key value:(NSString *)value table:(NSString *)tableName {
NSBundle *bundle = objc_getAssociatedObject(self, &_bundle);
return bundle ? [bundle localizedStringForKey:key value:value table:tableName] : [super localizedStringForKey:key value:value table:tableName];
}
@end
@implementation NSBundle (Language)
+ (void)setLanguage:(NSString *)language {
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
object_setClass([NSBundle mainBundle], [BundleEx class]);
});
objc_setAssociatedObject([NSBundle mainBundle], &_bundle, language ? [NSBundle bundleWithPath:[[NSBundle mainBundle] pathForResource:language ofType:@"lproj"]] : nil, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end
重新写一下设置语言的方法:
- (void)setLanguage:(NSString *)language{
[NSBundle setLanguage:language];
[[NSUserDefaults standardUserDefaults] setObject:language forKey:AppLanguage];
[[NSUserDefaults standardUserDefaults] synchronize];
}
综上所述,只是修改appleLanguage,在不重启应用的情况下,不能修改语言。所以我们选择修改bundle的方法。
代码在github上可以下载到:
https://github.com/FrankiezZZ/NTVLocalized
欢迎各位小伙伴加入iOS交流群:140147825