Android实现应用内置语言切换功能
作者:cekiasoo 发布时间:2021-11-14 13:19:50
一、需求
有时候应用需要在内部切换语言但又不影响系统的语言,比如是应用现在是中文的,系统语言也是中文的,我把应用的切换成英文显示后系统语言还是中文的,系统语言切换后也不会被改变,还有就是有些机的系统是被改造精简过的,比如有些国产机的系统的语言就被精简剩中文和英文。支付宝、微信、Top Story都有在应用内部设置语言切换这样的功能。
二、实现效果
先看看实现效果吧。
三、实现
(一)添加多种语言的资源文件夹及文件
我这默认是英语再添加了个俄文(Google翻译的)和中文。
values/strings.xml
<resources>
<string name="app_name">SwitchLanguage</string>
<string name="helloworld">Hello World!</string>
<string name="language">Eng</string>
<string name="english">English</string>
<string name="chinese">中文</string>
<string name="russian">русский</string>
<string name="secondact">Second Activity</string>
</resources>
values-ru/strings.xml
<resources>
<string name="app_name">Переключение язык</string>
<string name="helloworld">привет мир!</string>
<string name="language">русский</string>
<string name="secondact">второй активность</string>
</resources>
values-zh/strings.xml
<resources>
<string name="app_name">切换语言</string>
<string name="helloworld">你好 世界!</string>
<string name="language">中文</string>
<string name="secondact">第二屏</string>
</resources>
(二)布局文件
activity_main.xml
默认标题栏的文字是切换语言后是不会被改变的,使用Toobar替换掉就可以了。
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
android:id="@+id/activity_main"
xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
tools:context="com.ce.switchlanguage.MainActivity"
xmlns:app="http://schemas.android.com/apk/res-auto">
<android.support.design.widget.AppBarLayout
android:layout_height="wrap_content"
android:layout_width="match_parent"
android:theme="@style/AppTheme.AppBarOverlay">
<android.support.v7.widget.Toolbar
android:id="@+id/toolbar"
android:layout_width="match_parent"
android:layout_height="?attr/actionBarSize"
android:background="?attr/colorPrimary"
app:popupTheme="@style/AppTheme.PopupOverlay"
app:title="@string/app_name"/>
</android.support.design.widget.AppBarLayout>
<TextView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:text="@string/helloworld"/>
<Button
android:id="@+id/change"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:text="@string/language"/>
</LinearLayout>
styles.xml
设置语言后需要重启下activity,启动会有个效果,使用windowDisablePreview屏蔽掉它。
<resources>
<style name="AppTheme" parent="Theme.AppCompat.Light.DarkActionBar">
<item name="colorPrimary">@color/colorPrimary</item>
<item name="colorPrimaryDark">@color/colorPrimaryDark</item>
<item name="colorAccent">@color/colorAccent</item>
<item name="windowActionBar">false</item>
<item name="windowNoTitle">true</item>
<item name="android:windowDisablePreview">true</item>
</style>
<style name="AppTheme.AppBarOverlay" parent="ThemeOverlay.AppCompat.Dark.ActionBar"/>
<style name="AppTheme.PopupOverlay" parent="ThemeOverlay.AppCompat.Dark"/>
</resources>
main_menu.xml
<?xml version="1.0" encoding="utf-8"?>
<menu xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto">
<item android:id="@+id/chinese" android:title="@string/chinese" app:showAsAction="never" />
<item android:id="@+id/english" android:title="@string/english" app:showAsAction="never" />
<item android:id="@+id/russian" android:title="@string/russian" app:showAsAction="never" />
</menu>
(三)LocaleUtils
package com.ce.switchlanguage;
import android.content.Context;
import android.content.SharedPreferences;
import android.content.res.Configuration;
import android.os.Build;
import android.util.DisplayMetrics;
import com.google.gson.Gson;
import java.util.Locale;
public class LocaleUtils {
/**
* 中文
*/
public static final Locale LOCALE_CHINESE = Locale.CHINESE;
/**
* 英文
*/
public static final Locale LOCALE_ENGLISH = Locale.ENGLISH;
/**
* 俄文
*/
public static final Locale LOCALE_RUSSIAN = new Locale("ru");
/**
* 保存SharedPreferences的文件名
*/
private static final String LOCALE_FILE = "LOCALE_FILE";
/**
* 保存Locale的key
*/
private static final String LOCALE_KEY = "LOCALE_KEY";
/**
* 获取用户设置的Locale
* @param pContext Context
* @return Locale
*/
public static Locale getUserLocale(Context pContext) {
SharedPreferences _SpLocale = pContext.getSharedPreferences(LOCALE_FILE, Context.MODE_PRIVATE);
String _LocaleJson = _SpLocale.getString(LOCALE_KEY, "");
return jsonToLocale(_LocaleJson);
}
/**
* 获取当前的Locale
* @param pContext Context
* @return Locale
*/
public static Locale getCurrentLocale(Context pContext) {
Locale _Locale;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) { //7.0有多语言设置获取顶部的语言
_Locale = pContext.getResources().getConfiguration().getLocales().get(0);
} else {
_Locale = pContext.getResources().getConfiguration().locale;
}
return _Locale;
}
/**
* 保存用户设置的Locale
* @param pContext Context
* @param pUserLocale Locale
*/
public static void saveUserLocale(Context pContext, Locale pUserLocale) {
SharedPreferences _SpLocal=pContext.getSharedPreferences(LOCALE_FILE, Context.MODE_PRIVATE);
SharedPreferences.Editor _Edit=_SpLocal.edit();
String _LocaleJson = localeToJson(pUserLocale);
_Edit.putString(LOCALE_KEY, _LocaleJson);
_Edit.apply();
}
/**
* Locale转成json
* @param pUserLocale UserLocale
* @return json String
*/
private static String localeToJson(Locale pUserLocale) {
Gson _Gson = new Gson();
return _Gson.toJson(pUserLocale);
}
/**
* json转成Locale
* @param pLocaleJson LocaleJson
* @return Locale
*/
private static Locale jsonToLocale(String pLocaleJson) {
Gson _Gson = new Gson();
return _Gson.fromJson(pLocaleJson, Locale.class);
}
/**
* 更新Locale
* @param pContext Context
* @param pNewUserLocale New User Locale
*/
public static void updateLocale(Context pContext, Locale pNewUserLocale) {
if (needUpdateLocale(pContext, pNewUserLocale)) {
Configuration _Configuration = pContext.getResources().getConfiguration();
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
_Configuration.setLocale(pNewUserLocale);
} else {
_Configuration.locale =pNewUserLocale;
}
DisplayMetrics _DisplayMetrics = pContext.getResources().getDisplayMetrics();
pContext.getResources().updateConfiguration(_Configuration, _DisplayMetrics);
saveUserLocale(pContext, pNewUserLocale);
}
}
/**
* 判断需不需要更新
* @param pContext Context
* @param pNewUserLocale New User Locale
* @return true / false
*/
public static boolean needUpdateLocale(Context pContext, Locale pNewUserLocale) {
return pNewUserLocale != null && !getCurrentLocale(pContext).equals(pNewUserLocale);
}
}
Locale工具类,这里我用SharedPreferences来保存所设置的Locale,Locale是实现了Serializable的。
(四)Activity
package com.ce.switchlanguage;
import android.content.Intent;
import android.os.Bundle;
import android.support.v7.app.AppCompatActivity;
import android.support.v7.widget.Toolbar;
import android.view.Menu;
import android.view.MenuItem;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar _Toolbar =(Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(_Toolbar);
}
@Override
public boolean onCreateOptionsMenu(Menu menu) {
getMenuInflater().inflate(R.menu.main_menu,menu);
return true;
}
@Override
public boolean onOptionsItemSelected(MenuItem item) {
int _ItemId=item.getItemId();
switch (_ItemId) {
case R.id.chinese:
if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_CHINESE)) {
LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_CHINESE);
restartAct();
}
break;
case R.id.english:
if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_ENGLISH)) {
LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_ENGLISH);
restartAct();
}
break;
case R.id.russian:
if (LocaleUtils.needUpdateLocale(this, LocaleUtils.LOCALE_RUSSIAN)) {
LocaleUtils.updateLocale(this, LocaleUtils.LOCALE_RUSSIAN);
restartAct();
}
}
return true;
}
/**
* 重启当前Activity
*/
private void restartAct() {
finish();
Intent _Intent = new Intent(this, MainActivity.class);
startActivity(_Intent);
//清除Activity退出和进入的动画
overridePendingTransition(0, 0);
}
}
这里只有一个Activity所以切换的时候重启下当前Activity就好了,栈里还有其他Activity的自己再处理吧。
(五)Application
package com.ce.switchlanguage;
import android.app.Application;
import android.content.res.Configuration;
import android.os.Build;
import java.util.Locale;
public class MyApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
Locale _UserLocale=LocaleUtils.getUserLocale(this);
LocaleUtils.updateLocale(this, _UserLocale);
}
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
Locale _UserLocale=LocaleUtils.getUserLocale(this);
//系统语言改变了应用保持之前设置的语言
if (_UserLocale != null) {
Locale.setDefault(_UserLocale);
Configuration _Configuration = new Configuration(newConfig);
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) {
_Configuration.setLocale(_UserLocale);
} else {
_Configuration.locale =_UserLocale;
}
getResources().updateConfiguration(_Configuration, getResources().getDisplayMetrics());
}
}
}
在Application onCreate的时候更新下,在系统语言改变的时候也要保持之前设置的语言不变。
源码地址:Android应用内置语言切换
来源:http://blog.csdn.net/cekiasoo/article/details/54933135


猜你喜欢
- 对于触摸屏,其原生的消息无非按下、抬起、移动这几种,我们只需要简单重载onTouch或者设置触摸 * setOnTouchListener即
- 本来是想写一篇《委托与lambda表达式的前世今生》,但仅委托部分已经写了很多内容,于是就此分开关于Lambda表达是的内容后续再写吧。不知
- 引言C#应用通过 Microsoft.Toolkit.Uwp.Notifications NuGet包可以很方便的发送本地通知(Window
- 首先我们常用的注解包括@Entity、@Table、@Id、@IdClass、@GeneratedValue、@Basic、@Transie
- C# WinForm控件的拖动和缩放是个很有用的功能。实现起来其实很简单的,主要是设计控件的MouseDown、MouseLeave、Mou
- IDEA 在接入外接屏且扩展的情况下,如果突然拔掉外接屏,就可能会产生IDEA 整个窗口只在屏幕的右侧显示一点点边框且无法拖拽到当前屏幕的情
- 接口的灵活性就在于“规定一个类必须做什么,而不管你如何做”。我们可以定义一个接口类型的引用变量来引用实现接口的类的实例,当这个引用调用方法时
- AssertJ是我目前见过的最强大的断言api,没有之一。官网传送门为什么使用assertJ?1、流式断言,代码即用例,直观易懂。举个例子:
- 一个界面,实现在向页面添加图片时,在标题上显示一个水平进度条,当图片载入完毕后,隐藏进度条并显示图片具体实现方法:res/layout/ma
- 微服务之间token传递问题假设现在有A服务,B服务,外部使用RESTApi请求调用A服务,在请求头上有token字段,A服务使用完后,B服
- 一. 概念简介在开始学习今天的知识之前,有必要先给大家讲解一下与今天内容相关的一些概念,否则可能会让一些小白产生迷惑。1. 日期和时间的区别
- 作为Android应用开发者,不得不面对一个尴尬的局面,就是自己辛辛苦苦开发的应用可以被别人很轻易的就反编译出来。Google似乎也发现了这
- 本文实例讲述了C++实现的O(n)复杂度内查找第K大数算法。分享给大家供大家参考,具体如下:题目:是在一组数组(数组元素为整数,可正可负可为
- 页面拖动到最后一页 再向下滑动回复到 第一页,第一页向前滑动回到 最后一页同时,底部红色小圆点随着页面的滑动距离比例随时改变位置布局:<
- 监听通知Android 中的 AccessibilityService 可以监听通知信息的变化,首先需要创建一个无障碍服务,这个教程可以自行
- 废话不多说了,给大家贴关键代码了,具体代码如下所示:import java.io.File;import java.io.FileOutpu
- 最近为公司做的一个Demo里面用到了ScrollView嵌套了GridView和ListView,然而在嵌套的时候我发现GridView和L
- Android中的ListView应该算是布局中几种最常用的组件之一了,使用也十分方便,下面将介绍ListView几种比较常见的优化方法:首
- 案例需求:访问带有验证码的登录页面login.jsp用户输入用户名,密码以及验证码。如果用户名和密码输入有误,跳转登录页面,提示:用户名或密
- 本文实例为大家分享了使用C#写出一个简单的记事本程序,供大家参考,具体内容如下编程语言: C# 编程环境: Visual Studio 20