目录
- 一、项目中配置多语言
- 二、具体实现
- 三、AndroidX和多进程存在的问题
- 四、WebView导致的语言重置的问题
- 五、枚举类的多语言实现
一、项目中配置多语言
多语言的实现是通过AndroidUtilCode实现的,表示感谢!
项目里面有4种语言:中文,英文,德文,俄文。文件夹如下:
配置多语言的思路是:
1、判断是否为国内版本,如果为国内版本则设置为简体中文
2、 如果为国外版本,获取用户之前设置的App语言,如果用户之前有设置App语言,则设置为之前用户设置的语言;如果用户之前没有设置App语言则获取手机系统的语言。
3、判断当前手机系统的语言是否App有做语言适配,如果有适配则设置成跟手机系统一样的语言,如果没有适配则设置为英文。
二、具体实现
1、初始化PropertiesUtil和MMKV,具体代码请参考上篇博客
2、在BaseApplication中设置语言
abstract class BaseApplication : Application() { | |
abstract fun init() | |
override fun onCreate() { | |
super.onCreate() | |
init() | |
PropertiesUtil.init(this) | |
MMKV.initialize(this) | |
MMKVUtil.setUserId(L) | |
//设置App语言 | |
setAppLanguage() | |
} | |
/** | |
* 判断是否为国内版本,如果为国内版本则设置为简体中文 | |
* 如果为国外版本,获取用户之前设置的App语言, | |
* 如果用户之前有设置App语言,则设置为之前用户设置的语言 | |
* 如果用户之前没有设置App语言则获取手机系统的语言 | |
* 判断手机系统的语言是否App有做语言适配,如果有适配则设置成跟手机系统一样的语言 | |
* 如果App没有对当前系统语言做适配则设置为英文 | |
*/ | |
private fun setAppLanguage() { | |
if (PropertiesUtil.isCN()) { //国内版本 | |
LanguageUtils.applyLanguage(Locale.SIMPLIFIED_CHINESE, false) | |
} else { | |
MMKVUtil.getLanguage().also { | |
if (it.isNotEmpty()) { | |
setLanguageAndBackCountry(it) | |
} else { | |
//获取系统语言 | |
LanguageUtils.getSystemLanguage().country.also { country -> | |
setLanguageAndBackCountry(country).also { value -> | |
//保存设置的语言 | |
MMKVUtil.setLanguage(value) | |
} | |
} | |
} | |
} | |
} | |
} | |
private fun setLanguageAndBackCountry(it: String): String { | |
return when (it) { | |
LanguageType.CN.name -> { | |
LanguageUtils.applyLanguage(Locale.SIMPLIFIED_CHINESE, false) | |
it | |
} | |
LanguageType.US.name -> { | |
LanguageUtils.applyLanguage(Locale.ENGLISH, false) | |
it | |
} | |
LanguageType.DE.name -> { | |
LanguageUtils.applyLanguage(Locale.GERMANY, false) | |
it | |
} | |
LanguageType.RU.name -> { | |
LanguageUtils.applyLanguage(Locale("ru"), false) | |
it | |
} | |
else -> { | |
LanguageUtils.applyLanguage(Locale.ENGLISH, false) | |
LanguageType.US.name | |
} | |
} | |
} | |
} |
3、切换语言
比如设置为德文,按钮触发:
MMKVUtil.setLanguage(LanguageType.DE.name) | |
LanguageUtils.applyLanguage(Locale.GERMANY, false) //true:重启App false:不重启App |
4、注意gradle配置resConfigs不要限制为只有中文,比如:resConfigs "zh-rCN", "en"
三、AndroidX和多进程存在的问题
1、多进程读取Configuration时发现其他进程与主进程获取的Configuration值不一致,导致主进程切换语言后其他语言并没有切换成功。
2、AndroidX切换失败的问题,具体可以看下这篇博客【踩坑记录】多语言切换在Androidx失效
解决办法:重写Activity的attachBaseContext方法,修改Context
/** | |
* 多语言的切换类, 解决多进程切换语言失败的问题以及AndroidX多语言切换失效的问题 | |
* 解决由于 WebView 初始化会修改 Activity 语种配置,间接导致 Activity 语种会被还原,所以需要你手动重写 WebView 对这个问题进行修复 | |
*/ | |
object MultiLanguageUtil { | |
fun getAttachBaseContext(context: Context): Context { | |
return if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { | |
configAppcompatLanguage(setAppLanguageApi(context)) | |
} else { | |
setAppLanguage(context) | |
configAppcompatLanguage(context) | |
} | |
} | |
/** | |
* 设置应用语言 | |
*/ | |
private fun setAppLanguage(context: Context) { | |
val resources = context.resources | |
val displayMetrics = resources.displayMetrics | |
val configuration = resources.configuration | |
// 获取当前系统语言,默认设置跟随系统 | |
val locale = getLocale() | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR) { | |
configuration.setLocale(locale) | |
} else { | |
configuration.locale = locale | |
} | |
resources.updateConfiguration(configuration, displayMetrics) | |
} | |
/** | |
* 兼容.0 及以上 | |
*/ | |
fun setAppLanguageApi(context: Context): Context { | |
val locale = getLocale() | |
val resource = context.resources | |
val configuration = resource.configuration | |
configuration.setLocale(locale) | |
configuration.setLocales(LocaleList(locale)) | |
return context.createConfigurationContext(configuration) | |
} | |
private fun configAppcompatLanguage(context: Context): Context { | |
val configuration = context.resources.configuration | |
//兼容appcompat.2.0后切换语言失效问题 | |
return object : ContextThemeWrapper(context, R.style.Base_Theme_AppCompat_Empty) { | |
override fun applyOverrideConfiguration(overrideConfiguration: Configuration?) { | |
overrideConfiguration?.setTo(configuration) | |
super.applyOverrideConfiguration(overrideConfiguration) | |
} | |
} | |
} | |
private fun getLocale(): Locale { | |
return when (CacheUtil.getInt(GlobalConstants.LANGUAGE_KEY, true)) { -> { | |
Locale.SIMPLIFIED_CHINESE | |
} -> { | |
Locale.ENGLISH | |
} -> { | |
Locale.GERMANY | |
} -> { | |
Locale("ru") | |
} | |
else -> Locale.ENGLISH | |
} | |
} | |
/** | |
* 解决WebView多语言失效的问题 | |
*/ | |
fun updateLanguage(context: Context) { | |
val resources = context.resources | |
val config = resources.configuration | |
val settingLanguage = getLocale().language | |
val systemLanguage = config.locales[].language | |
if (settingLanguage != systemLanguage) { | |
setLocale(config, Locale(settingLanguage)) | |
resources.updateConfiguration(config, resources.displayMetrics) | |
} | |
} | |
private fun setLocale(config: Configuration, locale: Locale?) { | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR) { | |
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { | |
val localeList = LocaleList(locale) | |
config.setLocales(localeList) | |
} else { | |
config.setLocale(locale) | |
} | |
} else { | |
config.locale = locale | |
} | |
} | |
} |
获取appContext
lateinit var appContext: Application | |
//BaseApplication中调用方法获取Application的上下文 | |
fun BaseApplication.getContext(application: BaseApplication) { | |
appContext = application | |
} |
四、WebView导致的语言重置的问题
由于 WebView 初始化会修改 Activity 语种配置,间接导致 Activity 语种会被还原,所以需要你手动重写 WebView 对这个问题进行修复,如下:
/** | |
* 由于 WebView 初始化会修改 Activity 语种配置,间接导致 Activity 语种会被还原回去,所以需要你手动重写 WebView 对这个问题进行修复 | |
*/ | |
class LanguagesWebView( | |
context: Context, | |
@Nullable attrs: AttributeSet?, | |
defStyleAttr: Int | |
) : WebView(context, attrs, defStyleAttr) { | |
constructor(context: Context) : this(context, null) {} | |
constructor(context: Context, @Nullable attrs: AttributeSet?) : this( | |
context, | |
attrs, | |
) | |
init { | |
//修复 WebView 初始化时会修改Activity 语种配置的问题 | |
MultiLanguageUtil.updateLanguage(context) | |
} | |
} |
项目中用这个WebView即可。这个问题在华为手机鸿蒙系统上会出现。
五、枚举类的多语言实现
枚举类型是线程安全的,并且只会装载一次,这就导致下面的写法导致枚举的err值在切换语言后不会发生变化。
enum class Error( var code: Int, var err: String) { | |
/** | |
* 未知错误 | |
*/ | |
UNKNOWN(,appContext.getString(R.string.error_1000)), | |
/** | |
* 解析错误 | |
*/ | |
PARSE_ERROR(, appContext.getString(R.string.error_1001)), | |
/** | |
* 网络错误 | |
*/ | |
NETWORK_ERROR(, appContext.getString(R.string.error_1002)), | |
/** | |
* 证书出错 | |
*/ | |
SSL_ERROR(, appContext.getString(R.string.error_1004)), | |
/** | |
* 连接超时 | |
*/ | |
TIMEOUT_ERROR(, appContext.getString(R.string.error_1002)); | |
fun getValue(): String { | |
return err | |
} | |
fun getKey(): Int { | |
return code | |
} | |
} |
那么如果做枚举类的多语言适配呢? 代码如下:
enum class Error(private val code: Int, private val err: Int) { | |
/** | |
* 未知错误 | |
*/ | |
UNKNOWN(, R.string.error_1000), | |
/** | |
* 解析错误 | |
*/ | |
PARSE_ERROR(, R.string.error_1001), | |
/** | |
* 网络错误 | |
*/ | |
NETWORK_ERROR(, R.string.error_1002), | |
/** | |
* 证书出错 | |
*/ | |
SSL_ERROR(, R.string.error_1004), | |
/** | |
* 连接超时 | |
*/ | |
TIMEOUT_ERROR(, R.string.error_1002); | |
fun getValue(): String { | |
return appContext.getString(err) | |
} | |
fun getKey(): Int { | |
return code | |
} | |
} |
因为字符串的id是固定的不会发生变化,所以即使枚举类只会装载一次也不会有影响,通过getValue就能取到正确语言的字符串。
参考
【踩坑记录】多语言切换在Androidx失效