前言
iOS 开发中, 随着 App 越来越完善, 很多 App 都做了国际化, 也实现了 App 内部切换语言.
原理
国际化都会走到 NSBundle 的 - (NSString *)localizedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName 方法, 使用自定义 Bundle 替换 NSBundle, 通过切换根控制器来完成 App 内部切换语言.
1. 拦截国际化方法
通过分类和 runtime 来实现拦截, 将 Bundle 替换为自定义 CLBundle.
- + (void)load {
- static dispatch_once_t onceToken;
- dispatch_once(&onceToken, ^{
- // 动态继承, 交换, 方法类似 KVO, 通过修改 [NSBundle mainBundle] 对象的 isa 指针, 使其指向它的子类 CLBundle, 这样便可以调用子类的方法; 其实这里也可以使用 method_swizzling 来交换 mainBundle 的实现, 来动态判断, 可以同样实现.
- object_setClass([NSBundle mainBundle], [CLBundle class]);
- });
- }
2. 重写方法判断加载哪一种语言
重写 mainBundle 方法, 通过语言管理类获取当前应该加载哪一种语言.
- - (NSString *)localizedStringForKey:(NSString *)key value:(NSString *)value table:(NSString *)tableName {
- if ([CLBundle cl_mainBundle]) {
- return [[CLBundle cl_mainBundle] localizedStringForKey:key value:value table:tableName];
- } else {
- return [super localizedStringForKey:key value:value table:tableName];
- }
- }
- + (NSBundle *)cl_mainBundle {
- if ([NSBundle currentLanguage].length) {
- NSString *path = [[NSBundle mainBundle] pathForResource:[NSBundle currentLanguage] ofType:@"lproj"];
- if (path.length) {
- return [NSBundle bundleWithPath:path];
- }
- }
- return nil;
- }
- + (NSString *)currentLanguage {
- return [CLLanguageManager userLanguage] ? : [NSLocale preferredLanguages].firstObject;
- }
3. 语言管理类
将语言切换调用设置简单封装, 方便调用.
- + (void)setUserLanguage:(NSString *)userLanguage {
- // 跟随手机系统
- if (!userLanguage.length) {
- [self resetSystemLanguage];
- return;
- }
- // 用户自定义
- [[NSUserDefaults standardUserDefaults] setValue:userLanguage forKey:CLUserLanguageKey];
- [[NSUserDefaults standardUserDefaults] setValue:@[userLanguage] forKey:@"AppleLanguages"];
- [[NSUserDefaults standardUserDefaults] synchronize];
- }
- + (NSString *)userLanguage {
- return [[NSUserDefaults standardUserDefaults] valueForKey:CLUserLanguageKey];
- }
- /**
- 重置系统语言
- */
- + (void)resetSystemLanguage {
- [[NSUserDefaults standardUserDefaults] removeObjectForKey:CLUserLanguageKey];
- [[NSUserDefaults standardUserDefaults] setValue:nil forKey:@"AppleLanguages"];
- [[NSUserDefaults standardUserDefaults] synchronize];
- }
4. 切换根控制器来切换语言
为了减少工作量, 直接切换根控制器来达到语言切换.
- [CLLanguageManager setUserLanguage:@"en"];
- CLTabbarController *tabbarController = [[CLTabbarController alloc] init];
- tabbarController.selectedIndex = 3;
- UINavigationController *navigationController = tabbarController.selectedViewController;
- NSMutableArray *viewControllers = navigationController.viewControllers.mutableCopy;
- // 取出我的页面, 提前加载, 解决返回按钮不变化
- CLMyController *me = (CLMyController *)[viewControllers firstObject];
- [me loadViewIfNeeded];
- // 新建设置语言页面
- CLChangeLanguageController *languageController = [[CLChangeLanguageController alloc] init];
- languageController.hidesBottomBarWhenPushed = YES;
- [viewControllers addObject:languageController];
- // 解决奇怪的动画 bug.
- dispatch_async(dispatch_get_main_queue(), ^{
- [UIApplication sharedApplication].keyWindow.rootViewController = tabbarController;
- navigationController.viewControllers = viewControllers;
- CLLog(@"已切换到语言 %@", [NSBundle currentLanguage]);
- });
效果图
总结
以上是使用国际化来实现 App 内部切换语言, 希望能够给大家帮助, demo 地址 --->> https://github.com/JmoVxia/CLDemo
来源: https://juejin.im/post/5c130156e51d451d9a74887b