利用RAC制作一個登錄界面小Demo

本人有若干成套學(xué)習(xí)視頻, 可試看! 可試看! 可試看, 重要的事情說三遍 包含Java, 數(shù)據(jù)結(jié)構(gòu)與算法, iOS, 安卓, python, flutter等等, 如有需要, 聯(lián)系微信tsaievan.

這里有一篇關(guān)于MVVM, 我覺得最好的解釋說明:

MVVM 介紹

對于MVC架構(gòu), 即我們熟知的Model-View-Controller架構(gòu), 如下圖所示:

MVC架構(gòu)

M層請求數(shù)據(jù), V層展示數(shù)據(jù), C層處理邏輯. 但在實際開發(fā)過程中, C層和V層是偶聯(lián)在一起的, 就形成了這種格局:

MVC架構(gòu)

C層與M層交互, 這就往往會造成C層的邏輯過多, 代碼臃腫.

這個時候, 就產(chǎn)生了MVVM架構(gòu), 即Model-View-ViewModel架構(gòu).即在C層和M層中間加了一個ViewModel層, ViewModel就是用來為C層瘦身的. 它的出現(xiàn), 大大減少了控制器中的邏輯處理. 是C層變得輕巧.

MVVM架構(gòu)

RAC在信號傳遞方面有著得天獨厚的優(yōu)勢, 所以RAC+MVVM架構(gòu)可以說是天作之合. 今天利用RAC寫了一個登錄界面, 把C層的邏輯處理抽出來放到了ViewModel層 :

登錄界面的架構(gòu)

C層擁有V層和VM層:

C層擁有V層和VM層

V層和VM層的數(shù)據(jù)交互放在C層處理:

V層和VM層的數(shù)據(jù)交互放在C層處理

這個小Demo沒有Model, Model現(xiàn)在不直接與C層交互, 而是通過ViewModel.這樣一來, 各個類之間各行其是:

  • 需求1: 賬號和密碼都填時, 登錄按鈕才可以點擊:

步驟1: View層的數(shù)據(jù)通過C層給到VM層:

- (void)loginButtonEnable {
    RAC(self.loginViewModel, username) = _loginView.usernameTextField.rac_textSignal;
    RAC(self.loginViewModel, password) = _loginView.passwordTextField.rac_textSignal;
}

步驟2: VM層進(jìn)行邏輯處理:

self.loginButtonEnableSignal = [RACSignal combineLatest:@[RACObserve(self, username), RACObserve(self, password)] reduce:^id(NSString *username, NSString *password){
        return @(username.length && password.length);
    }];

以上代碼會返回一個包裝了的BOOL值

步驟3: View層顯示數(shù)據(jù):

 RAC(self.loginView.loginButton, enabled) = _loginViewModel.loginButtonEnableSignal;

給button的enabled屬性賦值, 告知按鈕是否可以點擊.

  • 需求2: 響應(yīng)按鈕的點擊事件, 當(dāng)賬號密碼正確時提示正確, 并跳轉(zhuǎn)控制器, 當(dāng)賬號和密碼錯誤時, 提示錯誤.

步驟1: View層的按鈕響應(yīng)點擊事件, 并執(zhí)行VM層的命令:

    [[_loginView.loginButton rac_signalForControlEvents:UIControlEventTouchUpInside] subscribeNext:^(__kindof UIControl * _Nullable x) {
        [self.loginViewModel.loginCommad execute:nil];
    }];

步驟2: VM層創(chuàng)建命令:

self.loginCommad = [[RACCommand alloc] initWithSignalBlock:^RACSignal * _Nonnull(id input) {
        return [RACSignal createSignal:^RACDisposable * _Nullable(id<RACSubscriber>  _Nonnull subscriber) {
            NSMutableURLRequest *request = [NSMutableURLRequest requestWithURL:[NSURL URLWithString:@"http://120.25.226.186:32812/login"]];
            request.HTTPMethod = @"POST";
            NSString *paramString = [NSString stringWithFormat:@"username=%@&pwd=%@&type=JSON", self.username, self.password];
            NSData *paramData = [paramString dataUsingEncoding:NSUTF8StringEncoding];
            request.HTTPBody = paramData;
            [[[NSURLSession sharedSession] dataTaskWithRequest:request completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
                NSDictionary *resultDictionary = [NSJSONSerialization JSONObjectWithData:data options:kNilOptions error:NULL];
                [subscriber sendNext:resultDictionary];
                /****************** -------- 發(fā)送完成這一步很重要, 不然后面的無法信號無法執(zhí)行 -------- ******************/
                [subscriber sendCompleted];
            }] resume];
            return nil;
        }];
    }];
  • 我們會將處理網(wǎng)絡(luò)請求的邏輯寫在命令中.
  • 創(chuàng)建命令會填入一個block參數(shù), 這個block參數(shù)會返回一個信號,在這個信號中, 我們會將請求的網(wǎng)絡(luò)數(shù)據(jù)發(fā)送出去.
  • 我們需要訂閱這個信號才能拿到這個網(wǎng)絡(luò)數(shù)據(jù), 如何拿到呢?
    • 在命令中有一個信號源屬性executionSignals
    • 信號源還有一個屬性switchToLatest讓我們可以拿到block中返回的信號,然后再訂閱這個信號, 我們就可以拿到服務(wù)器返回的數(shù)據(jù), 然后進(jìn)行登錄邏輯的處理:
    [self.loginCommad.executionSignals.switchToLatest subscribeNext:^(NSDictionary *x) {
        if ([x.allKeys.lastObject isEqualToString:@"success"]) {
            [SVProgressHUD showSuccessWithStatus:@"登錄成功"];
            [SVProgressHUD dismissWithDelay:1 completion:^{
                YFFirstPageViewController *firstPageVC = [[YFFirstPageViewController alloc] init];
                [UIView animateWithDuration:1 animations:^{
                    [UIApplication sharedApplication].keyWindow.rootViewController = firstPageVC;
                }];
            }];
        }else {
            [SVProgressHUD showErrorWithStatus:@"登錄失敗"];
            [SVProgressHUD dismissWithDelay:1];
        }
    }];
  • 還可以監(jiān)控命令的執(zhí)行過程:
    [[self.loginCommad.executing skip:1] subscribeNext:^(NSNumber * _Nullable x) {
        if ([x boolValue]) {
            [SVProgressHUD showWithStatus:@"正在登錄中"];
        }
    }];

以上這些邏輯都是在VM層處理的.

在V層中, 我們只需要處理控件的布局即可, 而不需要處理任何邏輯事件:
控件的布局

這就是整個Demo的架構(gòu), 實現(xiàn)邏輯和代碼, 效果如下:

Demo運行效果

+++++++++++++++Demo下載鏈接+++++++++++++++密碼: nfx3

PS. 本人有若干成套學(xué)習(xí)視頻, 包含Java, 數(shù)據(jù)結(jié)構(gòu)與算法, iOS, 安卓, python, flutter等等, 如有需要, 聯(lián)系微信tsaievan.

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末精钮,一起剝皮案震驚了整個濱河市曲掰,隨后出現(xiàn)的幾起案子庆聘,更是在濱河造成了極大的恐慌棕孙,老刑警劉巖屋彪,帶你破解...
    沈念sama閱讀 222,729評論 6 517
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異均牢,居然都是意外死亡涩禀,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 95,226評論 3 399
  • 文/潘曉璐 我一進(jìn)店門咆繁,熙熙樓的掌柜王于貴愁眉苦臉地迎上來讳推,“玉大人,你說我怎么就攤上這事么介∧茸瘢” “怎么了?”我有些...
    開封第一講書人閱讀 169,461評論 0 362
  • 文/不壞的土叔 我叫張陵壤短,是天一觀的道長设拟。 經(jīng)常有香客問我,道長久脯,這世上最難降的妖魔是什么纳胧? 我笑而不...
    開封第一講書人閱讀 60,135評論 1 300
  • 正文 為了忘掉前任,我火速辦了婚禮帘撰,結(jié)果婚禮上跑慕,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好核行,可當(dāng)我...
    茶點故事閱讀 69,130評論 6 398
  • 文/花漫 我一把揭開白布牢硅。 她就那樣靜靜地躺著,像睡著了一般芝雪。 火紅的嫁衣襯著肌膚如雪减余。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,736評論 1 312
  • 那天惩系,我揣著相機與錄音位岔,去河邊找鬼。 笑死堡牡,一個胖子當(dāng)著我的面吹牛抒抬,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播晤柄,決...
    沈念sama閱讀 41,179評論 3 422
  • 文/蒼蘭香墨 我猛地睜開眼擦剑,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了可免?” 一聲冷哼從身側(cè)響起抓于,我...
    開封第一講書人閱讀 40,124評論 0 277
  • 序言:老撾萬榮一對情侶失蹤做粤,失蹤者是張志新(化名)和其女友劉穎浇借,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體怕品,經(jīng)...
    沈念sama閱讀 46,657評論 1 320
  • 正文 獨居荒郊野嶺守林人離奇死亡妇垢,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,723評論 3 342
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了肉康。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闯估。...
    茶點故事閱讀 40,872評論 1 353
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖吼和,靈堂內(nèi)的尸體忽然破棺而出涨薪,到底是詐尸還是另有隱情,我是刑警寧澤炫乓,帶...
    沈念sama閱讀 36,533評論 5 351
  • 正文 年R本政府宣布刚夺,位于F島的核電站,受9級特大地震影響末捣,放射性物質(zhì)發(fā)生泄漏侠姑。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 42,213評論 3 336
  • 文/蒙蒙 一箩做、第九天 我趴在偏房一處隱蔽的房頂上張望莽红。 院中可真熱鬧,春花似錦邦邦、人聲如沸安吁。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,700評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽鬼店。三九已至馍管,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間薪韩,已是汗流浹背确沸。 一陣腳步聲響...
    開封第一講書人閱讀 33,819評論 1 274
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留俘陷,地道東北人罗捎。 一個月前我還...
    沈念sama閱讀 49,304評論 3 379
  • 正文 我出身青樓,卻偏偏與公主長得像拉盾,于是被迫代替她去往敵國和親桨菜。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,876評論 2 361

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