app内切换语言 swift
2018-04-17 22:44:00 weixin_33690963 阅读数 34

写在前面

本文同步 个人博客 简书 掘金 慕课
使用Xcode 9.3 Swift4.1

前言

语言本地化 大家肯定都多少都听过,今天我要分享的是快速实现语言本地化,与App内语言切换

核心内容主要是三个部分

  • storyboard/xib本地化
  • 纯代码本地化
  • 语言切换

准备工作

项目中添加语言


4009159-1814d0334c28c200.png

storyboard/xib本地化

storyboard/xib做本地化Xcode基本上是一键搞定了。
很简单
只要勾勾选选就可以了
这边只涉及到一个更新的问题
通过 ibtools命令 可以使storyboard/xib生成新的代码
首先cd 到stroyboard/xib 目录
执行ibtool xxx.storyboard --generate-strings-file new.strings
打开new.strings 将新内容手动复制到原来的string上。

纯代码本地化

创建string文件


4009159-2620622828bbe7b7.png
4009159-412b8488339a3fe7.png

勾选语言,把几种全部勾上,包括Base (为下文使用脚本生成代码做准备)

参考此篇文章进行脚本添加 iOS中多语言本地化流程的优化

4009159-5776e8194e680872.png
添加脚本

将脚本执行移动到编译上方


4009159-b2e6fe7e8945978f.png
移动位置

添加脚本

# Localizable.strings文件路径
localizableFile="${SRCROOT}/${PROJECT_NAME}/Support/en.lproj/Localizable.strings"
# 生成的swift文件路径(根据个人习惯修改)
localizedFile="${SRCROOT}/${PROJECT_NAME}/Source/Utils/LocalizedUtils.swift"
# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件
sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" > "${localizedFile}.tmp"
# 先将localized作为计算属性输出到目标文件
echo -e "import Foundation\n\nextension String {\n  var localized: String { return NSLocalizedString(self, comment: self) }" > "${localizedFile}"
# 再将临时文件中的常量增量输出到目标文件
cat "${localizedFile}.tmp" >> "${localizedFile}"
# 最后增量输出一个"}"到目标文件,完成输出
echo -e "\n}" >> "${localizedFile}"
# 删除临时文件
rm "${localizedFile}.tmp"

这里需要注意的是几个目录需要对应好,否则会报错

build一下就能自动生成相关代码 就可以直接用了,具体用法可以参考上面提到的那篇文章 iOS中多语言本地化流程的优化

语言切换

语言切换的基本原理是使用Userdefault存储当前选择的语言,在设置的时候改变其内容即可

主要涉及到两个问题

  • storyboard/xib如何切换语言
  • 如何刷新界面

对于上面都算是正常的本地化的内容,基本上介绍本地化的教程都会有。
对于自动化脚本这块算是比较新颖。

但是,脚本对于带空格的字符串生成的内容还是有问题,由于是使用sed命令,本人还不是很熟,只能想其他办法,这时候Base.lproj就派上用场了
我们将空格都替换成下划线,或者驼峰命名,在Base中一一对应,
在具体的en和zh中写具体内容,这时Base的作用就是为了方便自动生成代码而已了。(如果不想搞乱Base,新建一个即可)

关于storyboard/xib切换语言

替换Bundle即可
自定义一个Bundle,重写localizedString方法,每次都从Userdefault中获取当前选择语言,再使用方法替换将Bundle.main替换成自定义的Bundle

enum Language : String {
    case english = "en"
    case chinese = "zh-Hans"
}

/**
 *  当调用onLanguage后替换掉mainBundle为当前语言的bundle
 */

class BundleEx: Bundle {
    
    override func localizedString(forKey key: String, value: String?, table tableName: String?) -> String {
        if let bundle = Bundle.getLanguageBundel() {
            return bundle.localizedString(forKey: key, value: value, table: tableName)
        }else {
            return super.localizedString(forKey: key, value: value, table: tableName)
        }
    }
}


extension Bundle {
    
    private static var onLanguageDispatchOnce: ()->Void = {
        //替换Bundle.main为自定义的BundleEx
        object_setClass(Bundle.main, BundleEx.self)
    }
    
    func onLanguage(){
        Bundle.onLanguageDispatchOnce()
    }
    
    class func getLanguageBundel() -> Bundle? {
        let languageBundlePath = Bundle.main.path(forResource: UserDefaults.standard[AppStatic.kCurrentLanguage] as? String, ofType: "lproj")
//        print("path = \(languageBundlePath)")
        guard languageBundlePath != nil else {
            return nil
        }
        let languageBundle = Bundle.init(path: languageBundlePath!)
        guard languageBundle != nil else {
            return nil
        }
        return languageBundle!
        
    }
}

其中为Userdefault自定义了下标

    public subscript(key: String) -> Any? {
        get {
            return object(forKey: key)
        }
        set {
            set(newValue, forKey: key)
        }
    }

在读取字符串时执行一次
Bundle.main.onLanguage()

我就直接写到了脚本里,将脚本修改如下(几个文件的路径自己注意一下)

# Localizable.strings文件路径
localizableFile="${SRCROOT}/Base.lproj/Localizable.strings"
# 生成的swift文件路径(根据个人习惯修改)
localizedFile="${SRCROOT}/Public/LocalizedUtils.swift"
# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件
sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" > "${localizedFile}.tmp"
# 先将localized作为计算属性输出到目标文件
echo -e "import Foundation\n\nextension String {\n  var localized: String { Bundle.main.onLanguage() \n return NSLocalizedString(self, comment: self) }" > "${localizedFile}"
# 再将临时文件中的常量增量输出到目标文件
cat "${localizedFile}.tmp" >> "${localizedFile}"
# 最后增量输出一个"}"到目标文件,完成输出
echo -e "\n}" >> "${localizedFile}"
# 删除临时文件
rm "${localizedFile}.tmp"

关于刷新界面

对于所有界面的刷新最方便的就是重新设置rootViewController
将keyWindow先变黑,假装loading个几秒,再变回来即可。
如果需要再次回到之前所在页面,再添加相应的跳转VC的方法

    func chooseLanguage() {
        DispatchQueue.global().async {
            let sheet = UIAlertController.init(title: String.localized_Choose_Language, message: nil, preferredStyle: .actionSheet)
            
            sheet.addAction(UIAlertAction.init(title: String.localized_English, style: .default, handler: { (action) in
                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.english.rawValue
                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()
                UIApplication.shared.keyWindow?.alpha = 0
                AlertHelper.showHudWithMessage(message: "Setting Language...")
                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {
                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})
                    AlertHelper.hideHudMessage()
                })
                
            }))
            sheet.addAction(UIAlertAction.init(title: String.localized_Chinese, style: .default, handler: { (action) in
                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.chinese.rawValue
                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()
                UIApplication.shared.keyWindow?.alpha = 0
                AlertHelper.showHudWithMessage(message: "Setting Language...")
                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {
                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})
                    AlertHelper.hideHudMessage()
                })
            }))
            sheet.addAction(UIAlertAction.init(title: String.localized_Cancel, style: .cancel, handler: nil))
            self.present(sheet, animated: true, completion: nil)
        }
    }

至此App的语言切换与本地化就都讲完了,是不是很简单呢~~

后记

对于普通的小项目本地化的内容其实远没有那么复杂,需要替换的内容也很少,只要添加过一次语言,再添加新语言就非常简单了。

参考文章:
http://www.cocoachina.com/ios/20170809/20190.html

2018-04-27 09:09:44 qq_40201300 阅读数 463

前言

语言本地化 大家肯定都多少都听过,今天我要分享的是快速实现语言本地化,与App内语言切换

核心内容主要是三个部分

  • storyboard/xib本地化

  • 纯代码本地化

  • 语言切换

准备工作

项目中添加语言

storyboard/xib本地化

storyboard/xib做本地化Xcode基本上是一键搞定了。

很简单

只要勾勾选选就可以了

这边只涉及到一个更新的问题

通过 ibtools命令 可以使storyboard/xib生成新的代码

首先cd 到stroyboard/xib 目录

执行ibtool xxx.storyboard --generate-strings-file new.strings

打开new.strings 将新内容手动复制到原来的string上。

纯代码本地化

创建string文件

勾选语言,把几种全部勾上,包括Base (为下文使用脚本生成代码做准备)


添加脚本

将脚本执行移动到编译上方


移动位置

添加脚本

1
2
3
4
5
6
7
8
9
10
11
12
13
14
# Localizable.strings文件路径
localizableFile="${SRCROOT}/${PROJECT_NAME}/Support/en.lproj/Localizable.strings"
# 生成的swift文件路径(根据个人习惯修改)
localizedFile="${SRCROOT}/${PROJECT_NAME}/Source/Utils/LocalizedUtils.swift"
# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件
sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" "${localizedFile}.tmp"
# 先将localized作为计算属性输出到目标文件
echo -e "import Foundation\n\nextension String {\n  var localized: String { return NSLocalizedString(self, comment: self) }" "${localizedFile}"
# 再将临时文件中的常量增量输出到目标文件
cat "${localizedFile}.tmp" >> "${localizedFile}"
# 最后增量输出一个"}"到目标文件,完成输出
echo -e "\n}" >> "${localizedFile}"
# 删除临时文件
rm "${localizedFile}.tmp"

这里需要注意的是几个目录需要对应好,否则会报错

build一下就能自动生成相关代码 就可以直接用了

语言切换

语言切换的基本原理是使用Userdefault存储当前选择的语言,在设置的时候改变其内容即可

主要涉及到两个问题

  • storyboard/xib如何切换语言

  • 如何刷新界面

对于上面都算是正常的本地化的内容,基本上介绍本地化的教程都会有。

对于自动化脚本这块算是比较新颖。

但是,脚本对于带空格的字符串生成的内容还是有问题,由于是使用sed命令,本人还不是很熟,只能想其他办法,这时候Base.lproj就派上用场了

我们将空格都替换成下划线,或者驼峰命名,在Base中一一对应,

在具体的en和zh中写具体内容,这时Base的作用就是为了方便自动生成代码而已了。(如果不想搞乱Base,新建一个即可)

关于storyboard/xib切换语言

替换Bundle即可

自定义一个Bundle,重写localizedString方法,每次都从Userdefault中获取当前选择语言,再使用方法替换将Bundle.main替换成自定义的Bundle

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
enum Language : String {
    case english = "en"
    case chinese = "zh-Hans"
}
 
/**
 *  当调用onLanguage后替换掉mainBundle为当前语言的bundle
 */
 
class BundleEx: Bundle {
     
    override func localizedString(forKey key: String, value: String?, table tableName: String?) -> String {
        if let bundle = Bundle.getLanguageBundel() {
            return bundle.localizedString(forKey: key, value: value, table: tableName)
        }else {
            return super.localizedString(forKey: key, value: value, table: tableName)
        }
    }
}
 
 
extension Bundle {
     
    private static var onLanguageDispatchOnce: ()->Void = {
        //替换Bundle.main为自定义的BundleEx
        object_setClass(Bundle.main, BundleEx.self)
    }
     
    func onLanguage(){
        Bundle.onLanguageDispatchOnce()
    }
     
    class func getLanguageBundel() -> Bundle? {
        let languageBundlePath = Bundle.main.path(forResource: UserDefaults.standard[AppStatic.kCurrentLanguage] asString, ofType: "lproj")
//        print("path = \(languageBundlePath)")
        guard languageBundlePath != nil else {
            return nil
        }
        let languageBundle = Bundle.init(path: languageBundlePath!)
        guard languageBundle != nil else {
            return nil
        }
        return languageBundle!
         
    }
}

其中为Userdefault自定义了下标

1
2
3
4
5
6
7
8
    public subscript(key: String) -> Any? {
        get {
            return object(forKey: key)
        }
        set {
            set(newValue, forKey: key)
        }
    }

在读取字符串时执行一次

Bundle.main.onLanguage()

我就直接写到了脚本里,将脚本修改如下(几个文件的路径自己注意一下)

1
2
3
4
5
6
7
8
9
10
11
12
13
14
# Localizable.strings文件路径
localizableFile="${SRCROOT}/Base.lproj/Localizable.strings"
# 生成的swift文件路径(根据个人习惯修改)
localizedFile="${SRCROOT}/Public/LocalizedUtils.swift"
# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件
sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" "${localizedFile}.tmp"
# 先将localized作为计算属性输出到目标文件
echo -e "import Foundation\n\nextension String {\n  var localized: String { Bundle.main.onLanguage() \n return NSLocalizedString(self, comment: self) }" "${localizedFile}"
# 再将临时文件中的常量增量输出到目标文件
cat "${localizedFile}.tmp" >> "${localizedFile}"
# 最后增量输出一个"}"到目标文件,完成输出
echo -e "\n}" >> "${localizedFile}"
# 删除临时文件
rm "${localizedFile}.tmp"

关于刷新界面

对于所有界面的刷新最方便的就是重新设置rootViewController

将keyWindow先变黑,假装loading个几秒,再变回来即可。

如果需要再次回到之前所在页面,再添加相应的跳转VC的方法

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
    func chooseLanguage() {
        DispatchQueue.global().async {
            let sheet = UIAlertController.init(title: String.localized_Choose_Language, message: nil, preferredStyle: .actionSheet)
             
            sheet.addAction(UIAlertAction.init(title: String.localized_English, style: .default, handler: { (action) in
                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.english.rawValue
                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()
                UIApplication.shared.keyWindow?.alpha = 0
                AlertHelper.showHudWithMessage(message: "Setting Language...")
                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {
                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})
                    AlertHelper.hideHudMessage()
                })
                 
            }))
            sheet.addAction(UIAlertAction.init(title: String.localized_Chinese, style: .default, handler: { (action) in
                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.chinese.rawValue
                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()
                UIApplication.shared.keyWindow?.alpha = 0
                AlertHelper.showHudWithMessage(message: "Setting Language...")
                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {
                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})
                    AlertHelper.hideHudMessage()
                })
            }))
            sheet.addAction(UIAlertAction.init(title: String.localized_Cancel, style: .cancel, handler: nil))
            self.present(sheet, animated: true, completion: nil)
        }
    }

至此App的语言切换与本地化就都讲完了,是不是很简单呢~~

后记

对于普通的小项目本地化的内容其实远没有那么复杂,需要替换的内容也很少,只要添加过一次语言,再添加新语言就非常简单了。


2018-11-26 13:03:12 ZY_FlyWay 阅读数 132

前言

语言本地化 大家肯定都多少都听过,今天我要分享的是快速实现语言本地化,与App内语言切换

核心内容主要是三个部分

  • storyboard/xib本地化

  • 纯代码本地化

  • 语言切换

准备工作

项目中添加语言

storyboard/xib本地化

storyboard/xib做本地化Xcode基本上是一键搞定了。

很简单

只要勾勾选选就可以了

这边只涉及到一个更新的问题

通过 ibtools命令 可以使storyboard/xib生成新的代码

首先cd 到stroyboard/xib 目录

执行ibtool xxx.storyboard --generate-strings-file new.strings

打开new.strings 将新内容手动复制到原来的string上。

纯代码本地化

创建string文件

 

 

 

勾选语言,把几种全部勾上,包括Base (为下文使用脚本生成代码做准备)

参考此篇文章进行脚本添加iOS中多语言本地化流程的优化

 

添加脚本

将脚本执行移动到编译上方

 

移动位置

添加脚本

1

2

3

4

5

6

7

8

9

10

11

12

13

14

# Localizable.strings文件路径

localizableFile="${SRCROOT}/${PROJECT_NAME}/Support/en.lproj/Localizable.strings"

# 生成的swift文件路径(根据个人习惯修改)

localizedFile="${SRCROOT}/${PROJECT_NAME}/Source/Utils/LocalizedUtils.swift"

# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件

sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" "${localizedFile}.tmp"

# 先将localized作为计算属性输出到目标文件

echo -e "import Foundation\n\nextension String {\n  var localized: String { return NSLocalizedString(self, comment: self) }" "${localizedFile}"

# 再将临时文件中的常量增量输出到目标文件

cat "${localizedFile}.tmp" >> "${localizedFile}"

# 最后增量输出一个"}"到目标文件,完成输出

echo -e "\n}" >> "${localizedFile}"

# 删除临时文件

rm "${localizedFile}.tmp"

这里需要注意的是几个目录需要对应好,否则会报错

build一下就能自动生成相关代码 就可以直接用了,具体用法可以参考上面提到的那篇文章iOS中多语言本地化流程的优化

语言切换

语言切换的基本原理是使用Userdefault存储当前选择的语言,在设置的时候改变其内容即可

主要涉及到两个问题

  • storyboard/xib如何切换语言

  • 如何刷新界面

对于上面都算是正常的本地化的内容,基本上介绍本地化的教程都会有。

对于自动化脚本这块算是比较新颖。

但是,脚本对于带空格的字符串生成的内容还是有问题,由于是使用sed命令,本人还不是很熟,只能想其他办法,这时候Base.lproj就派上用场了

我们将空格都替换成下划线,或者驼峰命名,在Base中一一对应,

在具体的en和zh中写具体内容,这时Base的作用就是为了方便自动生成代码而已了。(如果不想搞乱Base,新建一个即可)

关于storyboard/xib切换语言

替换Bundle即可

自定义一个Bundle,重写localizedString方法,每次都从Userdefault中获取当前选择语言,再使用方法替换将Bundle.main替换成自定义的Bundle

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

30

31

32

33

34

35

36

37

38

39

40

41

42

43

44

45

46

enum Language : String {

    case english = "en"

    case chinese = "zh-Hans"

}

 

/**

 *  当调用onLanguage后替换掉mainBundle为当前语言的bundle

 */

 

class BundleEx: Bundle {

     

    override func localizedString(forKey key: String, value: String?, table tableName: String?) -> String {

        if let bundle = Bundle.getLanguageBundel() {

            return bundle.localizedString(forKey: key, value: value, table: tableName)

        }else {

            return super.localizedString(forKey: key, value: value, table: tableName)

        }

    }

}

 

 

extension Bundle {

     

    private static var onLanguageDispatchOnce: ()->Void = {

        //替换Bundle.main为自定义的BundleEx

        object_setClass(Bundle.main, BundleEx.self)

    }

     

    func onLanguage(){

        Bundle.onLanguageDispatchOnce()

    }

     

    class func getLanguageBundel() -> Bundle? {

        let languageBundlePath = Bundle.main.path(forResource: UserDefaults.standard[AppStatic.kCurrentLanguage] asString, ofType: "lproj")

//        print("path = \(languageBundlePath)")

        guard languageBundlePath != nil else {

            return nil

        }

        let languageBundle = Bundle.init(path: languageBundlePath!)

        guard languageBundle != nil else {

            return nil

        }

        return languageBundle!

         

    }

}

其中为Userdefault自定义了下标

1

2

3

4

5

6

7

8

    public subscript(key: String) -> Any? {

        get {

            return object(forKey: key)

        }

        set {

            set(newValue, forKey: key)

        }

    }

在读取字符串时执行一次

Bundle.main.onLanguage()

我就直接写到了脚本里,将脚本修改如下(几个文件的路径自己注意一下)

1

2

3

4

5

6

7

8

9

10

11

12

13

14

# Localizable.strings文件路径

localizableFile="${SRCROOT}/Base.lproj/Localizable.strings"

# 生成的swift文件路径(根据个人习惯修改)

localizedFile="${SRCROOT}/Public/LocalizedUtils.swift"

# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件

sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" "${localizedFile}.tmp"

# 先将localized作为计算属性输出到目标文件

echo -e "import Foundation\n\nextension String {\n  var localized: String { Bundle.main.onLanguage() \n return NSLocalizedString(self, comment: self) }" "${localizedFile}"

# 再将临时文件中的常量增量输出到目标文件

cat "${localizedFile}.tmp" >> "${localizedFile}"

# 最后增量输出一个"}"到目标文件,完成输出

echo -e "\n}" >> "${localizedFile}"

# 删除临时文件

rm "${localizedFile}.tmp"

关于刷新界面

对于所有界面的刷新最方便的就是重新设置rootViewController

将keyWindow先变黑,假装loading个几秒,再变回来即可。

如果需要再次回到之前所在页面,再添加相应的跳转VC的方法

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

25

26

27

28

29

    func chooseLanguage() {

        DispatchQueue.global().async {

            let sheet = UIAlertController.init(title: String.localized_Choose_Language, message: nil, preferredStyle: .actionSheet)

             

            sheet.addAction(UIAlertAction.init(title: String.localized_English, style: .default, handler: { (action) in

                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.english.rawValue

                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()

                UIApplication.shared.keyWindow?.alpha = 0

                AlertHelper.showHudWithMessage(message: "Setting Language...")

                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {

                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})

                    AlertHelper.hideHudMessage()

                })

                 

            }))

            sheet.addAction(UIAlertAction.init(title: String.localized_Chinese, style: .default, handler: { (action) in

                UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.chinese.rawValue

                UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()

                UIApplication.shared.keyWindow?.alpha = 0

                AlertHelper.showHudWithMessage(message: "Setting Language...")

                DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {

                    UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})

                    AlertHelper.hideHudMessage()

                })

            }))

            sheet.addAction(UIAlertAction.init(title: String.localized_Cancel, style: .cancel, handler: nil))

            self.present(sheet, animated: true, completion: nil)

        }

    }

至此App的语言切换与本地化就都讲完了,是不是很简单呢~~

后记

对于普通的小项目本地化的内容其实远没有那么复杂,需要替换的内容也很少,只要添加过一次语言,再添加新语言就非常简单了。

2018-04-27 09:10:00 weixin_33851429 阅读数 35

前言

语言本地化 大家肯定都多少都听过,今天我要分享的是快速实现语言本地化,与App内语言切换

核心内容主要是三个部分

storyboard/xib本地化

纯代码本地化

语言切换

准备工作

项目中添加语言

7877747-11f0e693477323a9.jpg

storyboard/xib本地化

storyboard/xib做本地化Xcode基本上是一键搞定了。

很简单

只要勾勾选选就可以了

这边只涉及到一个更新的问题

通过 ibtools命令 可以使storyboard/xib生成新的代码

首先cd 到stroyboard/xib 目录

执行ibtool xxx.storyboard --generate-strings-file new.strings

打开new.strings 将新内容手动复制到原来的string上。

纯代码本地化

创建string文件

7877747-33ed745c1b7842f0.jpg
7877747-eae8fe7280124329.jpg

勾选语言,把几种全部勾上,包括Base (为下文使用脚本生成代码做准备)

7877747-af9d18aed9367e9c.jpg

添加脚本

将脚本执行移动到编译上方

7877747-a6ce7baec3b8439c.jpg

移动位置

添加脚本

# Localizable.strings文件路径

localizableFile="${SRCROOT}/${PROJECT_NAME}/Support/en.lproj/Localizable.strings"

# 生成的swift文件路径(根据个人习惯修改)

localizedFile="${SRCROOT}/${PROJECT_NAME}/Source/Utils/LocalizedUtils.swift"

# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件

sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" > "${localizedFile}.tmp"

# 先将localized作为计算属性输出到目标文件

echo -e "import Foundation\n\nextension String {\n  var localized: String { return NSLocalizedString(self, comment: self) }" > "${localizedFile}"

# 再将临时文件中的常量增量输出到目标文件

cat "${localizedFile}.tmp" >> "${localizedFile}"

# 最后增量输出一个"}"到目标文件,完成输出

echo -e "\n}" >> "${localizedFile}"

# 删除临时文件

rm "${localizedFile}.tmp"

这里需要注意的是几个目录需要对应好,否则会报错

build一下就能自动生成相关代码 就可以直接用了

语言切换

语言切换的基本原理是使用Userdefault存储当前选择的语言,在设置的时候改变其内容即可

主要涉及到两个问题

storyboard/xib如何切换语言

如何刷新界面

对于上面都算是正常的本地化的内容,基本上介绍本地化的教程都会有。

对于自动化脚本这块算是比较新颖。

但是,脚本对于带空格的字符串生成的内容还是有问题,由于是使用sed命令,本人还不是很熟,只能想其他办法,这时候Base.lproj就派上用场了

我们将空格都替换成下划线,或者驼峰命名,在Base中一一对应,

在具体的en和zh中写具体内容,这时Base的作用就是为了方便自动生成代码而已了。(如果不想搞乱Base,新建一个即可)

关于storyboard/xib切换语言

替换Bundle即可

自定义一个Bundle,重写localizedString方法,每次都从Userdefault中获取当前选择语言,再使用方法替换将Bundle.main替换成自定义的Bundle

enum Language : String {

case english = "en"

case chinese = "zh-Hans"

}

/**

*  当调用onLanguage后替换掉mainBundle为当前语言的bundle

*/

class BundleEx: Bundle {

override func localizedString(forKey key: String, value: String?, table tableName: String?) -> String {

if let bundle = Bundle.getLanguageBundel() {

return bundle.localizedString(forKey: key, value: value, table: tableName)

}else {

return super.localizedString(forKey: key, value: value, table: tableName)

}

}

}

extension Bundle {

private static var onLanguageDispatchOnce: ()->Void = {

//替换Bundle.main为自定义的BundleEx

object_setClass(Bundle.main, BundleEx.self)

}

func onLanguage(){

Bundle.onLanguageDispatchOnce()

}

class func getLanguageBundel() -> Bundle? {

let languageBundlePath = Bundle.main.path(forResource: UserDefaults.standard[AppStatic.kCurrentLanguage] as? String, ofType: "lproj")

//        print("path = \(languageBundlePath)")

guard languageBundlePath != nil else {

return nil

}

let languageBundle = Bundle.init(path: languageBundlePath!)

guard languageBundle != nil else {

return nil

}

return languageBundle!

}

}

其中为Userdefault自定义了下标

public subscript(key: String) -> Any? {

get {

return object(forKey: key)

}

set {

set(newValue, forKey: key)

}

}

在读取字符串时执行一次

Bundle.main.onLanguage()

我就直接写到了脚本里,将脚本修改如下(几个文件的路径自己注意一下)

# Localizable.strings文件路径

localizableFile="${SRCROOT}/Base.lproj/Localizable.strings"

# 生成的swift文件路径(根据个人习惯修改)

localizedFile="${SRCROOT}/Public/LocalizedUtils.swift"

# 将localizable.strings中的文本转为swift格式的常量,存入一个临时文件

sed "s/^\"/  static var localized_/g" "${localizableFile}" | sed "s/\" = \"/: String { return \"/g" | sed "s/;$/.localized }/g" > "${localizedFile}.tmp"

# 先将localized作为计算属性输出到目标文件

echo -e "import Foundation\n\nextension String {\n  var localized: String { Bundle.main.onLanguage() \n return NSLocalizedString(self, comment: self) }" > "${localizedFile}"

# 再将临时文件中的常量增量输出到目标文件

cat "${localizedFile}.tmp" >> "${localizedFile}"

# 最后增量输出一个"}"到目标文件,完成输出

echo -e "\n}" >> "${localizedFile}"

# 删除临时文件

rm "${localizedFile}.tmp"

关于刷新界面

对于所有界面的刷新最方便的就是重新设置rootViewController

将keyWindow先变黑,假装loading个几秒,再变回来即可。

如果需要再次回到之前所在页面,再添加相应的跳转VC的方法

func chooseLanguage() {

DispatchQueue.global().async {

let sheet = UIAlertController.init(title: String.localized_Choose_Language, message: nil, preferredStyle: .actionSheet)

sheet.addAction(UIAlertAction.init(title: String.localized_English, style: .default, handler: { (action) in

UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.english.rawValue

UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()

UIApplication.shared.keyWindow?.alpha = 0

AlertHelper.showHudWithMessage(message: "Setting Language...")

DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {

UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})

AlertHelper.hideHudMessage()

})

}))

sheet.addAction(UIAlertAction.init(title: String.localized_Chinese, style: .default, handler: { (action) in

UserDefaults.standard[AppStatic.kCurrentLanguage] = Language.chinese.rawValue

UIApplication.shared.keyWindow?.rootViewController = RedbotTabBar()

UIApplication.shared.keyWindow?.alpha = 0

AlertHelper.showHudWithMessage(message: "Setting Language...")

DispatchQueue.main.asyncAfter(deadline: DispatchTime.now()+1, execute: {

UIView.animate(withDuration: 1, animations: {UIApplication.shared.keyWindow?.alpha = 1})

AlertHelper.hideHudMessage()

})

}))

sheet.addAction(UIAlertAction.init(title: String.localized_Cancel, style: .cancel, handler: nil))

self.present(sheet, animated: true, completion: nil)

}

}

至此App的语言切换与本地化就都讲完了,是不是很简单呢~~

后记

对于普通的小项目本地化的内容其实远没有那么复杂,需要替换的内容也很少,只要添加过一次语言,再添加新语言就非常简单了。

iOS 多国语言本地化与App内语言切换(Swift)

阅读数 13

写在前面本文同步个人博客简书慕课使用Xcode9.3Swift4.1前言语言本地化大家肯定都多少都听过,今天我要分享的是快速实现语言本地化,与App内语言切换核心内容主要是三个部分storyboard/xib本地化纯代码本地化语言切换准备工作项目中添加语言storyboard/xib本地化storyboard/xib做本地化Xcode基本上是一键搞...

博文 来自: weixin_34082695

swift应用内切换语言实现本地化

阅读数 348

1:在projectinfo中的locations添加需要的语言2:创建Localizable.strings文件点击右边的localization勾选需要的语言3:创建InfoPlist.strings翻译Info.plist中的提示文字和应用名称4:创建LanguageHelper.swiftimportUIKitletUserLanguage="UserLa...

博文 来自: wahaha13168

App内实现语言国际化(OC和Swift)

阅读数 18

1、OC实现语言国际化(NSBundle扩展).h文件#import<Foundation/Foundation.h>@interfaceNSBundle(Language)+(void)setLanguage:(NSString*)language;@end```.m文件```#import<ob...

博文 来自: weixin_34074740

swift 国际化,app内部中英语言切换

阅读数 1640

重点是app内部切换1.配置languege.strings.infoPist.Strings2.LocalizationTool.swift,实现内部切换////LocalizationTool.swift//IFXY////CreatedbyLiuXingon

博文 来自: aa654403231

swift - app国际化, 优雅切换

阅读数 25

在开发的过程中,可能你们的app会有这样的需求,需要坚持多种语言.怎么做到不重启app就更换了语言,下面我们来讨论下.首先我们知道,如果你的app中需要支持多种语言,往往我们会选择一个Localizable.strings,它会一种语言生成一个文件夹,相当于一个包吧.所以我们只要做到切换时,选择对应的语言包就可以了.废话不多说,...

博文 来自: weixin_33836874
没有更多推荐了,返回首页