添加多语言

This commit is contained in:
2025-11-03 16:37:28 +08:00
parent e4cebeac85
commit 1673a2f4be
11 changed files with 449 additions and 13 deletions

View File

@@ -0,0 +1,180 @@
//
// KBLocalizationManager.m
//
//
#import "KBLocalizationManager.h"
#import <Security/Security.h>
#import "KBConfig.h"
///
NSString * const KBLocalizationDidChangeNotification = @"KBLocalizationDidChangeNotification";
// Target
static NSString * const kKBLocService = @"com.keyBoardst.loc";
static NSString * const kKBLocAccount = @"lang"; // UTF8
@interface KBLocalizationManager ()
@property (nonatomic, copy, readwrite) NSString *currentLanguageCode; //
@property (nonatomic, strong) NSBundle *langBundle; // .lproj
@end
// +shared
// C +shared
static inline NSMutableDictionary *KBLocBaseKCQuery(void) {
NSMutableDictionary *q = [@{ (__bridge id)kSecClass: (__bridge id)kSecClassGenericPassword,
(__bridge id)kSecAttrService: kKBLocService,
(__bridge id)kSecAttrAccount: kKBLocAccount } mutableCopy];
if (KB_KEYCHAIN_ACCESS_GROUP.length > 0) {
q[(__bridge id)kSecAttrAccessGroup] = KB_KEYCHAIN_ACCESS_GROUP;
}
return q;
}
@implementation KBLocalizationManager
+ (instancetype)shared {
static KBLocalizationManager *m; static dispatch_once_t onceToken; dispatch_once(&onceToken, ^{
m = [KBLocalizationManager new];
//
m.supportedLanguageCodes = @[ @"en", @"zh-Hans" ];
// 退
NSString *saved = [[self class] kc_read];
if (saved.length == 0) {
saved = [m bestSupportedLanguageForPreferred:[NSLocale preferredLanguages]] ?: @"en";
}
[m applyLanguage:saved];
});
return m;
}
#pragma mark - API
- (void)setSupportedLanguageCodes:(NSArray<NSString *> *)supportedLanguageCodes {
//
NSMutableOrderedSet *set = [NSMutableOrderedSet orderedSet];
for (NSString *c in supportedLanguageCodes) {
if (c.length) { [set addObject:c]; }
}
_supportedLanguageCodes = set.array.count ? set.array : @[ @"en" ];
// 广
if (self.currentLanguageCode.length && ![set containsObject:self.currentLanguageCode]) {
NSString *best = [self bestSupportedLanguageForPreferred:@[self.currentLanguageCode]];
[self applyLanguage:best ?: _supportedLanguageCodes.firstObject];
[[NSNotificationCenter defaultCenter] postNotificationName:KBLocalizationDidChangeNotification object:nil];
}
}
- (void)setCurrentLanguageCode:(NSString *)code persist:(BOOL)persist {
if (code.length == 0) return; //
if ([code isEqualToString:self.currentLanguageCode]) return; //
[self applyLanguage:code];
if (persist) { [[self class] kc_write:code]; } // App/
[[NSNotificationCenter defaultCenter] postNotificationName:KBLocalizationDidChangeNotification object:nil];
}
- (void)resetToSystemLanguage {
NSString *best = [self bestSupportedLanguageForPreferred:[NSLocale preferredLanguages]] ?: @"en";
[self setCurrentLanguageCode:best persist:NO];
}
- (NSString *)localizedStringForKey:(NSString *)key {
return [self localizedStringForKey:key table:nil value:key];
}
- (NSString *)localizedStringForKey:(NSString *)key table:(NSString *)table value:(NSString *)value {
if (key.length == 0) return @"";
NSBundle *bundle = self.langBundle ?: NSBundle.mainBundle;
NSString *tbl = table ?: @"Localizable";
// 使 bundle API NSLocalizedString
NSString *str = [bundle localizedStringForKey:key value:value table:tbl];
return str ?: (value ?: key);
}
- (NSString *)bestSupportedLanguageForPreferred:(NSArray<NSString *> *)preferred {
if (self.supportedLanguageCodes.count == 0) return @"en";
// 1)
for (NSString *p in preferred) {
NSString *pLC = p.lowercaseString;
for (NSString *s in self.supportedLanguageCodes) {
if ([pLC isEqualToString:s.lowercaseString]) { return s; }
}
}
// 2) zh-Hans-CN -> zh-Hans, en-GB -> en
for (NSString *p in preferred) {
NSString *pLC = p.lowercaseString;
for (NSString *s in self.supportedLanguageCodes) {
NSString *sLC = s.lowercaseString;
if ([pLC hasPrefix:[sLC stringByAppendingString:@"-"]] || [pLC hasPrefix:[sLC stringByAppendingString:@"_"]]) {
return s;
}
// also allow reverse: when supported is regional (rare)
if ([sLC hasPrefix:[pLC stringByAppendingString:@"-"]] || [sLC hasPrefix:[pLC stringByAppendingString:@"_"]]) {
return s;
}
}
}
// 3) zh-Hant/zh-TW/zh-HK zh-Hant
for (NSString *p in preferred) {
NSString *pLC = p.lowercaseString;
if ([pLC hasPrefix:@"zh-hant"] || [pLC hasPrefix:@"zh-tw"] || [pLC hasPrefix:@"zh-hk"]) {
for (NSString *s in self.supportedLanguageCodes) {
if ([s.lowercaseString isEqualToString:@"zh-hant"]) { return s; }
}
}
if ([pLC hasPrefix:@"zh-hans"] || [pLC hasPrefix:@"zh-cn"]) {
for (NSString *s in self.supportedLanguageCodes) {
if ([s.lowercaseString isEqualToString:@"zh-hans"]) { return s; }
}
}
}
// 4)
return self.supportedLanguageCodes.firstObject ?: @"en";
}
#pragma mark -
- (void)applyLanguage:(NSString *)code {
_currentLanguageCode = [code copy];
// TargetApp bundle .lproj
NSString *path = [NSBundle.mainBundle pathForResource:code ofType:@"lproj"];
if (!path) {
// en-GB -> en
NSString *shortCode = [[code componentsSeparatedByCharactersInSet:[NSCharacterSet characterSetWithCharactersInString:@"-_"]] firstObject];
if (shortCode.length > 0) {
path = [NSBundle.mainBundle pathForResource:shortCode ofType:@"lproj"];
}
}
if (path) {
self.langBundle = [NSBundle bundleWithPath:path];
} else {
self.langBundle = NSBundle.mainBundle; //
}
}
#pragma mark - App/
+ (BOOL)kc_write:(NSString *)lang {
NSMutableDictionary *q = KBLocBaseKCQuery();
SecItemDelete((__bridge CFDictionaryRef)q);
if (lang.length == 0) return YES; //
NSData *data = [lang dataUsingEncoding:NSUTF8StringEncoding];
q[(__bridge id)kSecValueData] = data ?: [NSData data];
q[(__bridge id)kSecAttrAccessible] = (__bridge id)kSecAttrAccessibleAfterFirstUnlockThisDeviceOnly;
OSStatus st = SecItemAdd((__bridge CFDictionaryRef)q, NULL);
return (st == errSecSuccess);
}
+ (NSString *)kc_read {
NSMutableDictionary *q = KBLocBaseKCQuery();
q[(__bridge id)kSecReturnData] = @YES;
q[(__bridge id)kSecMatchLimit] = (__bridge id)kSecMatchLimitOne;
CFTypeRef dataRef = NULL; OSStatus st = SecItemCopyMatching((__bridge CFDictionaryRef)q, &dataRef);
if (st != errSecSuccess || !dataRef) return nil;
NSData *data = (__bridge_transfer NSData *)dataRef;
if (data.length == 0) return nil;
NSString *lang = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
return lang;
}
@end