如何在SDK中使用资源文件(xib,image,localizable string...)


需求:在SDK中使用资源文件,例如xib, image, 字符串国际化等等.


阅读前提:

  • 了解如何创建SDK
  • 了解如何在一个项目中导入并使用SDK

GitHub地址(附代码) : 如何在SDK中使用资源文件

简书地址 : 如何在SDK中使用资源文件

博客地址 : 如何在SDK中使用资源文件

掘金地址 : 如何在SDK中使用资源文件


原理

SDK中使用xib,图片或者其他资源文件时不同于直接在一个项目中使用的方法,因为我们必须明确一点,SDK中只有.h,.m,.mm,.swift等文件可以直接导入使用,像图片,xib文件,做国际化的.string文件等资源文件无法直接通过加载Framework的路径找到,我们必须要在项目Build Phases的Copy Bundle Resources添加我们需要使用的资源文件(nib, image, .string...),因此,我们最好在SDK中新建一个Bundle文件存放所有的资源文件,然后在主工程中Copy Bundle Resources添加该bundle文件即可加载所有SDK中的资源文件.

创建Bundle

为了统一存放SDK中所有资源文件

注意:我们需要在每次编译完成后手动或写脚本自动将资源文件拷贝到Bundle中以至于才能在主工程中生效。

此后我们在SDK项目所用到的所有资源文件都应该放到Bundle之中。

0-createBundle

情景分类

一.使用Xib文件

1. Build SDK工程

如果项目中包含xib文件,则项目每次编译后会在Framework中将自动生成nib文件,如下图所示

1-nib文件

2.将新编译好的Framework放入工程中

  • 首先拷贝编译好的framework到我们工程目录中(即工程中进入Framework,不会自行百度)
  • 使用Nib文件分为两种方式
    • 直接将Nib文件导入主项目资源文件中
    • (推荐)将Framework中编译好的Nib文件放入Bundle中,再将Bundle导入项目的资源文件中,好处是可以统一管理一个Framework中所有的资源文件
2-nib文件导入项目

如果只使用nib则如下图,否则可按相同方式将bundle导入项目


3-niblocation

3. 使用SDK中的xib文件

  • 直接使用framework中的nib文件
    NSString *path = [[NSBundle mainBundle] pathForResource:@"XDXTestFramework" ofType:@"framework"];
    TestJumpVC *vc = [[TestJumpVC alloc] initWithNibName:@"TestJumpVC" bundle:[NSBundle bundleWithPath:path]];
    
    [self presentViewController:vc animated:YES completion:nil];
  • (推荐)使用framework中bundle中的nib文件。

因为我们在第二步中已经将bundle放入项目的copy bundle resources中,所以下面代码中使用[NSBundle mainBundle]

    NSString *path = [[NSBundle mainBundle] pathForResource:@"XDXAllResources" ofType:@"bundle"];
    TestJumpVC *vc = [[TestJumpVC alloc] initWithNibName:@"TestJumpVC" bundle:[NSBundle bundleWithPath:[path stringByAppendingString:@"/xibs"]]];

    [self presentViewController:vc animated:YES completion:nil];

如果未进行第2步中将nib文件导入项目则程序会crash,并提示“Terminating app due to uncaught exception 'NSInternalInconsistencyException', reason: 'Could not load NIB in bundle: 'NSBundle </var/containers/Bundle/Application/D46C1A83-E6F4-4DB2-8F02-EC0ED05C6C99/XDXSDKResourceFileDemo.app> (loaded)' with name 'TestJumpVC''”

二.使用图片

1. 将图片拷贝到Framework中创建好的bundle中

4-SDK_Picture

2. 使用图片,加载bundle中图片的路径(Note:必须与真实路径相同,可参考Demo)

    // Test image
    NSString *path = [[[NSBundle mainBundle] pathForResource:@"XDXAllResources" ofType:@"bundle"] stringByAppendingString:@"/images/1.png"];
    self.testImage.image = [UIImage imageWithContentsOfFile:path];

三.使用国际化语言适配(中英文适配)

1.创建.string类型文件以支持中英文

5-create_strings

2. 设置strings文件的Localize使其支持多种语言

6-set_strings

3. 设置项目支持多种语音

7-set_project

8-set_project_file

4. 在多语言文件中添加对应的字符

9-set_string

5. 创建NSBundle分类提供类方法以支持在SDK中使用多语言

原理同上,即在budle中找到对应的中英文sting文件,然后利用- (NSString *)localizedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1);方法来检索bundle中对应语言的字符串。

代码中涉及APP手动选择中英文模式,而我们这里主要实现根据系统当前设置的语言环境,所以忽略人为设置语言的情况

// Note : Be consistent with you create bundle‘s name.
#define XDXRouterResBundle @"XDXAllResources"

@implementation NSBundle (XDXLocalizable)

+ (instancetype)XDX_localizableBundleWithBundleName:(NSString *)bundleName {
    static NSBundle *localizableBundle = nil;
    if (localizableBundle == nil) {
        if (!bundleName) {
            bundleName = XDXRouterResBundle;
        }
        NSString *bundleType = nil;
        if (bundleName && ![bundleName hasSuffix:@"bundle"]) {
            bundleType = @"bundle";
        }
        NSString *bundlePath = [[NSBundle mainBundle] pathForResource:bundleName ofType:bundleType];
        
        localizableBundle = [NSBundle bundleWithPath:bundlePath];
    }
    return localizableBundle;
}

+ (NSString *)XDX_localizedStringForKey:(NSString *)key {
    return [self XDX_localizedStringForKey:key value:nil];
}

+ (NSString *)XDX_localizedStringForKey:(NSString *)key value:(NSString *)value{
    NSBundle *bundle = nil;
    //NSString *language = [self getLanguageFromSystem];
    //NSString *language = [self getLanguageFromPlist];
    NSString * language = [self getLanguageFromDevelopersSetup];
    if (!language) {
        language = [self getLanguageFromSystem];
        NSLog(@"Current language is %@",language);
    }
    //从FrameworkTestBundle.bundle中查找资源
    NSString *bundlePath = [[NSBundle XDX_localizableBundleWithBundleName:nil] pathForResource:language ofType:@"lproj"];
    bundle = [NSBundle bundleWithPath:bundlePath];
    value = [bundle localizedStringForKey:key value:value table:nil];
    return [[NSBundle mainBundle] localizedStringForKey:key value:value table:nil];
}

//这个设置语言是通过读取当前系统使用语言
+ (NSString *)getLanguageFromSystem{
    NSString *language = [NSLocale preferredLanguages].firstObject;
    if ([language hasPrefix:@"en"]) {
        language = @"en";
    } else if ([language hasPrefix:@"zh"]) {
        if ([language rangeOfString:@"Hans"].location != NSNotFound) {
            language = @"zh-Hans"; // 简体中文
        } else {
            language = @"zh-Hant"; // 繁體中文
        }
    } else {
        language = @"en";
    }
    return language;
}

//这个是设置语言通过Plist文件来读取
+ (NSString *)getLanguageFromPlist{
    NSString *bundlePath = [[NSBundle mainBundle] pathForResource:@"SDKInternationalizationDemoPlist.plist" ofType:nil];
    if (!bundlePath) {
        return nil;
    }
    NSDictionary *dict = [NSDictionary dictionaryWithContentsOfFile:bundlePath];
    if (dict) {
        NSInteger languageNum = [[dict valueForKey:@"language"] integerValue];
        switch (languageNum) {
            case 1:
                return @"en";     //语言为英语:en
                break;
            case 2:
                return @"zh-Hans";//语言为简中:zh-Hans
                break;
            case 3:
                return @"zh-Hant";//语言为繁中:zh-Hanz
                break;
            default:
                return @"en";
                break;
        }
    }
    return @"en";
}

//这个是设置语言通过开发者手动调用,从NSUserDefaults里面去读kDSADLanguageStyle这个字段是哪一种语言
+ (NSString *)getLanguageFromDevelopersSetup{
    NSUserDefaults *userDefaults = [NSUserDefaults standardUserDefaults];
    NSInteger languageStyle = [[userDefaults valueForKey:@"TODO"] integerValue];
    if (!languageStyle) {
        return nil;
    }
    switch (languageStyle) {
        case 1:
            return @"en";
            break;
        case 2:
            return @"zh-Hans";
            break;
        case 3:
            return @"zh-Hant";
            break;
        default:
            return @"en";
            break;
    }
}

@end

6.在代码中使用多语言字符串

self.testLabel.text = [NSBundle XDX_localizedStringForKey:@"Test"];

附:在Framework每次编译后添加脚本来拷贝资源文件到Bundle中

原因

每当资源文件有所改动,例如图片的增加减少,strings文件中添加新字段等等,我们都需要手动将新的文件放入Bundle对应的文件夹下,如果我们想利用脚本自动放所有资源文件,可利用Xcode特性。

操作

  • 首先,在项目中新建一个脚本


    10-createScript
  • 其次,在脚本中将我们项目的所有资源文件拷贝到bundle中对应的目录下(注意:文件路径必须正确,如遇到Framework中的nib文件,项目的位置在个人电脑中是不用的,需要查询并修改)


    11-writeScript
  • 最后,我们在Build完项目后,所有资源文件会更新到最新


    12-getFile
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容