50天iOS挑戰(zhàn)(Swift) - 第10天:制作應(yīng)用啟動引導(dǎo)頁面

50天iOS挑戰(zhàn)(Swift) - 第10天:制作應(yīng)用啟動引導(dǎo)頁面

50天放椰,每天一個(gè)Swift語言的iOS練手項(xiàng)目愉粤,覆蓋iOS開發(fā)的主要知識衣厘。貴在堅(jiān)持,重在思考


文章列表:http://www.reibang.com/nb/13566182
Github項(xiàng)目:https://github.com/Minecodecraft/50DaysOfSwift


簡介

很多應(yīng)用在用戶初次啟動時(shí)影暴,會展示一個(gè)What's new頁面型宙,如果打造一個(gè)多屏幕適配的啟動界面,同時(shí)又保證低耦合性呢妆兑?Let's do it!
本節(jié)將介紹啟動界面的制作芯勘,下一節(jié)介紹登錄界面的多屏幕適配。
由于代碼較為復(fù)雜衡怀,文章中只講制作過程安疗,建議結(jié)合代碼同步理解。

主要知識點(diǎn): Xib設(shè)計(jì)模式蝶桶、控制器解耦掉冶、代碼重構(gòu)、數(shù)據(jù)模型恢共、KVC璧亚、KVO、CollectionView透硝、PageControl疯搅、控件約束(屏幕適配)

豎屏模式:


GIF

橫屏模式:


GIF2



設(shè)計(jì)思路

1幔欧、設(shè)計(jì)架構(gòu)
由于引導(dǎo)界面只是龐大項(xiàng)目的一個(gè)小模塊,所以要盡可能減少耦合性觉义。
本著各司其職的思路浴井,我們將引導(dǎo)界面和登錄界面分別放到兩個(gè)故事版中。
引導(dǎo)界面需要用到的cell厉碟,采用Xib方式構(gòu)建。(也可代碼構(gòu)建)
應(yīng)用啟動時(shí)崭参,判斷啟動某個(gè)界面的工作款咖,交給NavigationController。對于控制器間通訊海洼,采用代碼塊內(nèi)指針方式富腊,以避免強(qiáng)引用循環(huán)(也可使用代理、閉方式避免)是整。

導(dǎo)航頁面邏輯架構(gòu)

2民假、界面設(shè)計(jì)
由于要兼容橫豎屏羊异,所以界面約束的設(shè)計(jì)格外重要。
根據(jù)自己需求添加界面約束即可野舶,在此不再贅述。

3赴蝇、配置CollectionView
我們采用Xib構(gòu)造巢掺,所以要在CollectionView注冊Cell

collectionView.register(UINib.init(nibName: "PageViewCell", bundle: nil), forCellWithReuseIdentifier: guideCell)
collectionView.register(UINib.init(nibName: "LoginViewCell", bundle: nil), forCellWithReuseIdentifier: loginCell)

3陆淀、建立數(shù)據(jù)模型
開發(fā)中我們與服務(wù)器通信時(shí)經(jīng)常采用json先嬉,數(shù)據(jù)常以模型的形式存在。為了減少耦合性含懊,我們采用數(shù)據(jù)模型來表示每一個(gè)引導(dǎo)頁的內(nèi)容。
蘋果為我們提供了更簡單的對屬性賦值方式:KVC

var imageName: String?
var title: String?
var detail: String?

init(dict: [String: Any]) {
    super.init()
    setValuesForKeys(dict)
}

override func setValue(_ value: Any?, forUndefinedKey key: String) {
    // Do something
}

記著重寫setValue:forUndefinedKey:酥筝,因?yàn)槟J(rèn)實(shí)現(xiàn)會拋出異常雏门,程序crash

完成模型建立后,我們就可以通過鍵值對的方式對數(shù)據(jù)模型賦值宙帝。

4募闲、最后一頁隱藏按鈕
從GIF中可以看到,翻到最后一頁時(shí)我們要隱藏PageControl和兩個(gè)按鈕控件沪编。
我們采用更改控件約束的方式

        if off {
            pageControlBottom.constant = -20
            skipButtonTop.constant = -50
            continueButtonTop.constant = -50
            // disable button to avoid crash
            skipButton.isEnabled = false
            continueButton.isEnabled = false
        } else {
            pageControlBottom.constant = 10
            skipButtonTop.constant = 0
            continueButtonTop.constant = 0
            skipButton.isEnabled = true
            continueButton.isEnabled = true
        }

當(dāng)然年扩,這只是改變了其frame厨幻,系統(tǒng)并不一定立即重繪,我們調(diào)用layoutIfNeed方法進(jìn)行重繪况脆。

        // layout subviews
        UIView.animate(withDuration: 0.5, delay: 0, options: .curveEaseOut, animations: {
            self.view.layoutIfNeeded()
        })

5格了、復(fù)用頁面跳轉(zhuǎn)方法
由于劃屏和點(diǎn)擊按鈕都需要調(diào)用翻頁方法,將翻頁/頁面跳轉(zhuǎn)/按鈕隱藏方法封裝弹惦,提供接口供其他方法調(diào)用悄但。

優(yōu)化/Bug解決

1、橫屏旋轉(zhuǎn)時(shí)頁面位置錯(cuò)誤
因?yàn)锳uto Layout Guide調(diào)整layout時(shí)助泽,并不會調(diào)整對應(yīng)的index,所以需要在willTransition中手動調(diào)用

override func willTransition(to newCollection: UITraitCollection, with coordinator: UIViewControllerTransitionCoordinator) {
        let indexPath = IndexPath(item: pageControl.currentPage, section: 0)
        
        self.collectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: true)
    }

但是這樣還是沒有解決隐解,原因在于collectionView在旋轉(zhuǎn)之前完成了scrollToItem方法诫睬。
解決方法很簡單,提交一個(gè)延時(shí)0.01秒任務(wù)溜嗜。

    override func willTransition(to newCollection: UITraitCollection, with coordinator: UIViewControllerTransitionCoordinator) {
        let indexPath = IndexPath(item: pageControl.currentPage, section: 0)
        
        DispatchQueue.main.asyncAfter(deadline: .now()+0.01) {
            self.collectionView.scrollToItem(at: indexPath, at: .centeredHorizontally, animated: true)
        }
    }

2架谎、解決導(dǎo)航控制器無rootViewController的問題
我們在APPDelegate中設(shè)置UIWindow為導(dǎo)航控制器谷扣,如下面代碼

    func application(_ application: UIApplication, didFinishLaunchingWithOptions launchOptions: [UIApplicationLaunchOptionsKey: Any]?) -> Bool {
        // Override point for customization after application launch.
        
        window = UIWindow(frame: UIScreen.main.bounds)
        window?.makeKeyAndVisible()
        
        let nvc = MainNavigationController()
        window?.rootViewController = nvc
        
        return true
    }

但是,當(dāng)我們在導(dǎo)航控制器中初始化引導(dǎo)頁面控制器/登錄界面控制器/登陸成功界面控制器時(shí)裹匙,可能出現(xiàn)沒有view沒有成功present的情況末秃。原因和步驟1很類似,將present任務(wù)延時(shí)提交即可惰匙。
如下面代碼铃将,導(dǎo)航控制器的初始化過程:

class MainNavigationController: UINavigationController {
    
    override func viewDidLoad() {
        super.viewDidLoad()

        view.backgroundColor = UIColor.cyan
        
        // 添加delay時(shí)間,放置在window初始化之前顯示
        if isFirstRun() {
            perform(#selector(showGuideController), with: nil, afterDelay: 0.01)
        }
        else if isLoggedIn() {
            // show
        }
        else {
            perform(#selector(showLoginController), with: nil, afterDelay: 0.01)
        }
    }
    
    func isFirstRun() -> Bool {
        return true
    }
    
    func isLoggedIn() -> Bool {
        return false
    }
    
    @objc func showGuideController() {
        let storyboard = UIStoryboard(name: "Guide", bundle: nil)
        let guideVC = storyboard.instantiateInitialViewController() as! GuideViewController
        
        present(guideVC, animated: false) {
            // do something
        }
    }
    
    func finishShowGuideController() {
        dismiss(animated: false, completion: nil)
        showLoginController()
    }
    
    @objc func showLoginController() {
        let storyboard = UIStoryboard(name: "Login", bundle: nil)
        let loginVC = storyboard.instantiateInitialViewController() as! LoginViewController
        
        present(loginVC, animated: false) {
            // do something
        }
    }
    
}

詳細(xì)過程請見代碼,下一節(jié)將介紹登錄界面的屏幕適配問題悯仙,包括輸入時(shí)屏幕上移、重繪輸入框等功能實(shí)現(xiàn)稚虎。



項(xiàng)目源碼

項(xiàng)目源碼已傳至Github:(https://github.com/Minecodecraft/50DaysOfSwift)
后續(xù)教程偎捎、代碼茴她、庫持續(xù)更新,歡迎Star關(guān)注丈牢。希望可以對大家有所幫助

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末己沛,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子垮卓,更是在濱河造成了極大的恐慌师幕,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,607評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件灭将,死亡現(xiàn)場離奇詭異后控,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)浩淘,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,239評論 3 395
  • 文/潘曉璐 我一進(jìn)店門馋袜,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人察皇,你說我怎么就攤上這事泽台。” “怎么了稻爬?”我有些...
    開封第一講書人閱讀 164,960評論 0 355
  • 文/不壞的土叔 我叫張陵蜕依,是天一觀的道長。 經(jīng)常有香客問我友瘤,道長,這世上最難降的妖魔是什么束倍? 我笑而不...
    開封第一講書人閱讀 58,750評論 1 294
  • 正文 為了忘掉前任绪妹,我火速辦了婚禮柿究,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘笛求。我一直安慰自己探入,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,764評論 6 392
  • 文/花漫 我一把揭開白布苗膝。 她就那樣靜靜地躺著植旧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪问窃。 梳的紋絲不亂的頭發(fā)上完沪,一...
    開封第一講書人閱讀 51,604評論 1 305
  • 那天,我揣著相機(jī)與錄音听皿,去河邊找鬼宽档。 笑死,一個(gè)胖子當(dāng)著我的面吹牛又厉,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播昔逗,決...
    沈念sama閱讀 40,347評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼篷朵,長吁一口氣:“原來是場噩夢啊……” “哼婆排!你這毒婦竟也來了段只?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,253評論 0 276
  • 序言:老撾萬榮一對情侶失蹤澈缺,失蹤者是張志新(化名)和其女友劉穎炕婶,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體项滑,經(jīng)...
    沈念sama閱讀 45,702評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡涯贞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,893評論 3 336
  • 正文 我和宋清朗相戀三年宋渔,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片严蓖。...
    茶點(diǎn)故事閱讀 40,015評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡谈飒,死狀恐怖态蒂,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情钾恢,我是刑警寧澤,帶...
    沈念sama閱讀 35,734評論 5 346
  • 正文 年R本政府宣布稿黍,位于F島的核電站崩哩,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏邓嘹。R本人自食惡果不足惜汹押,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,352評論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望窖维。 院中可真熱鬧妙痹,春花似錦、人聲如沸沛贪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,934評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽媚送。三九已至寇甸,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間吟秩,已是汗流浹背绽淘。 一陣腳步聲響...
    開封第一講書人閱讀 33,052評論 1 270
  • 我被黑心中介騙來泰國打工沪铭, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留壮池,地道東北人偏瓤。 一個(gè)月前我還...
    沈念sama閱讀 48,216評論 3 371
  • 正文 我出身青樓厅克,卻偏偏與公主長得像橙依,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子褪储,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,969評論 2 355

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

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫、插件浪读、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,105評論 4 62
  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,152評論 25 707
  • 工作后再與曾經(jīng)的老同學(xué)們見面聊天很有意思互订,基本上聊天半小時(shí)內(nèi)會涉及房價(jià)問題痘拆,以及,“你怎么胖了纺蛆?!” 肥胖得算工傷...
    陳跡智鵬飛閱讀 1,897評論 0 3
  • 沫沫快一歲了桥氏,已經(jīng)成功和兩次感冒作斗爭温峭。第一次是風(fēng)寒感冒。第二次是病毒性感冒字支。 在護(hù)理上我實(shí)在忍不住分享我入手的這...
    Vinc姐閱讀 389評論 0 2