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


需求:在SDK中使用資源文件,例如xib, image, 字符串國(guó)際化等等.


閱讀前提:

  • 了解如何創(chuàng)建SDK
  • 了解如何在一個(gè)項(xiàng)目中導(dǎo)入并使用SDK

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

簡(jiǎn)書(shū)地址 : 如何在SDK中使用資源文件

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

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


原理

SDK中使用xib垛叨,圖片或者其他資源文件時(shí)不同于直接在一個(gè)項(xiàng)目中使用的方法肝箱,因?yàn)槲覀儽仨毭鞔_一點(diǎn)渴频,SDK中只有.h,.m,.mm,.swift等文件可以直接導(dǎo)入使用,像圖片荚坞,xib文件,做國(guó)際化的.string文件等資源文件無(wú)法直接通過(guò)加載Framework的路徑找到,我們必須要在項(xiàng)目Build Phases的Copy Bundle Resources添加我們需要使用的資源文件(nib, image, .string...)酬诀,因此,我們最好在SDK中新建一個(gè)Bundle文件存放所有的資源文件骆撇,然后在主工程中Copy Bundle Resources添加該bundle文件即可加載所有SDK中的資源文件.

創(chuàng)建Bundle

為了統(tǒng)一存放SDK中所有資源文件

注意:我們需要在每次編譯完成后手動(dòng)或?qū)懩_本自動(dòng)將資源文件拷貝到Bundle中以至于才能在主工程中生效瞒御。

此后我們?cè)赟DK項(xiàng)目所用到的所有資源文件都應(yīng)該放到Bundle之中。

0-createBundle

情景分類

一.使用Xib文件

1. Build SDK工程

如果項(xiàng)目中包含xib文件神郊,則項(xiàng)目每次編譯后會(huì)在Framework中將自動(dòng)生成nib文件肴裙,如下圖所示

1-nib文件

2.將新編譯好的Framework放入工程中

  • 首先拷貝編譯好的framework到我們工程目錄中(即工程中進(jìn)入Framework,不會(huì)自行百度)
  • 使用Nib文件分為兩種方式
    • 直接將Nib文件導(dǎo)入主項(xiàng)目資源文件中
    • (推薦)將Framework中編譯好的Nib文件放入Bundle中,再將Bundle導(dǎo)入項(xiàng)目的資源文件中,好處是可以統(tǒng)一管理一個(gè)Framework中所有的資源文件
2-nib文件導(dǎo)入項(xiàng)目

如果只使用nib則如下圖涌乳,否則可按相同方式將bundle導(dǎo)入項(xiàng)目


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文件蜻懦。

因?yàn)槲覀冊(cè)诘诙街幸呀?jīng)將bundle放入項(xiàng)目的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];

如果未進(jìn)行第2步中將nib文件導(dǎo)入項(xiàng)目則程序會(huì)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中創(chuàng)建好的bundle中

4-SDK_Picture

2. 使用圖片夕晓,加載bundle中圖片的路徑(Note:必須與真實(shí)路徑相同阻肩,可參考Demo)

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

三.使用國(guó)際化語(yǔ)言適配(中英文適配)

1.創(chuàng)建.string類型文件以支持中英文

5-create_strings

2. 設(shè)置strings文件的Localize使其支持多種語(yǔ)言

6-set_strings

3. 設(shè)置項(xiàng)目支持多種語(yǔ)音

7-set_project

8-set_project_file

4. 在多語(yǔ)言文件中添加對(duì)應(yīng)的字符

9-set_string

5. 創(chuàng)建NSBundle分類提供類方法以支持在SDK中使用多語(yǔ)言

原理同上,即在budle中找到對(duì)應(yīng)的中英文sting文件运授,然后利用- (NSString *)localizedStringForKey:(NSString *)key value:(nullable NSString *)value table:(nullable NSString *)tableName NS_FORMAT_ARGUMENT(1);方法來(lái)檢索bundle中對(duì)應(yīng)語(yǔ)言的字符串烤惊。

代碼中涉及APP手動(dòng)選擇中英文模式,而我們這里主要實(shí)現(xiàn)根據(jù)系統(tǒng)當(dāng)前設(shè)置的語(yǔ)言環(huán)境吁朦,所以忽略人為設(shè)置語(yǔ)言的情況

// 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];
}

//這個(gè)設(shè)置語(yǔ)言是通過(guò)讀取當(dāng)前系統(tǒng)使用語(yǔ)言
+ (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"; // 簡(jiǎn)體中文
        } else {
            language = @"zh-Hant"; // 繁體中文
        }
    } else {
        language = @"en";
    }
    return language;
}

//這個(gè)是設(shè)置語(yǔ)言通過(guò)Plist文件來(lái)讀取
+ (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";     //語(yǔ)言為英語(yǔ):en
                break;
            case 2:
                return @"zh-Hans";//語(yǔ)言為簡(jiǎn)中:zh-Hans
                break;
            case 3:
                return @"zh-Hant";//語(yǔ)言為繁中:zh-Hanz
                break;
            default:
                return @"en";
                break;
        }
    }
    return @"en";
}

//這個(gè)是設(shè)置語(yǔ)言通過(guò)開(kāi)發(fā)者手動(dòng)調(diào)用柒室,從NSUserDefaults里面去讀kDSADLanguageStyle這個(gè)字段是哪一種語(yǔ)言
+ (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.在代碼中使用多語(yǔ)言字符串

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

附:在Framework每次編譯后添加腳本來(lái)拷貝資源文件到Bundle中

原因

每當(dāng)資源文件有所改動(dòng),例如圖片的增加減少逗宜,strings文件中添加新字段等等雄右,我們都需要手動(dòng)將新的文件放入Bundle對(duì)應(yīng)的文件夾下,如果我們想利用腳本自動(dòng)放所有資源文件纺讲,可利用Xcode特性擂仍。

操作

  • 首先,在項(xiàng)目中新建一個(gè)腳本


    10-createScript
  • 其次熬甚,在腳本中將我們項(xiàng)目的所有資源文件拷貝到bundle中對(duì)應(yīng)的目錄下(注意:文件路徑必須正確逢渔,如遇到Framework中的nib文件,項(xiàng)目的位置在個(gè)人電腦中是不用的乡括,需要查詢并修改)


    11-writeScript
  • 最后肃廓,我們?cè)贐uild完項(xiàng)目后智厌,所有資源文件會(huì)更新到最新


    12-getFile
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市盲赊,隨后出現(xiàn)的幾起案子铣鹏,更是在濱河造成了極大的恐慌,老刑警劉巖哀蘑,帶你破解...
    沈念sama閱讀 217,185評(píng)論 6 503
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件诚卸,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡绘迁,警方通過(guò)查閱死者的電腦和手機(jī)惨险,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,652評(píng)論 3 393
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)脊髓,“玉大人,你說(shuō)我怎么就攤上這事栅受〗酰” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,524評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵屏镊,是天一觀的道長(zhǎng)依疼。 經(jīng)常有香客問(wèn)我,道長(zhǎng)而芥,這世上最難降的妖魔是什么律罢? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,339評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮棍丐,結(jié)果婚禮上误辑,老公的妹妹穿的比我還像新娘。我一直安慰自己歌逢,他們只是感情好巾钉,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,387評(píng)論 6 391
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著秘案,像睡著了一般砰苍。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上阱高,一...
    開(kāi)封第一講書(shū)人閱讀 51,287評(píng)論 1 301
  • 那天赚导,我揣著相機(jī)與錄音,去河邊找鬼赤惊。 笑死吼旧,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的未舟。 我是一名探鬼主播黍少,決...
    沈念sama閱讀 40,130評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼寡夹,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了厂置?” 一聲冷哼從身側(cè)響起菩掏,我...
    開(kāi)封第一講書(shū)人閱讀 38,985評(píng)論 0 275
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎昵济,沒(méi)想到半個(gè)月后智绸,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,420評(píng)論 1 313
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡访忿,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,617評(píng)論 3 334
  • 正文 我和宋清朗相戀三年瞧栗,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片海铆。...
    茶點(diǎn)故事閱讀 39,779評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡迹恐,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出卧斟,到底是詐尸還是另有隱情殴边,我是刑警寧澤,帶...
    沈念sama閱讀 35,477評(píng)論 5 345
  • 正文 年R本政府宣布珍语,位于F島的核電站锤岸,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏板乙。R本人自食惡果不足惜是偷,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,088評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望募逞。 院中可真熱鬧蛋铆,春花似錦、人聲如沸放接。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,716評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)透乾。三九已至洪燥,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間乳乌,已是汗流浹背捧韵。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,857評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留汉操,地道東北人再来。 一個(gè)月前我還...
    沈念sama閱讀 47,876評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親芒篷。 傳聞我的和親對(duì)象是個(gè)殘疾皇子搜变,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,700評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容