iOS實(shí)現(xiàn)自定義表單實(shí)例代碼
前言
最近在開(kāi)發(fā)一個(gè)APP,需要讓用戶填寫數(shù)據(jù),然后上傳到服務(wù)端進(jìn)行計(jì)算并返回結(jié)果在客戶端中展示。其中需要填寫的數(shù)據(jù)項(xiàng)多達(dá)十幾項(xiàng),大部分是必填。所有表單數(shù)據(jù)在一個(gè)頁(yè)面中實(shí)現(xiàn),在APP中這樣的設(shè)計(jì)其實(shí)挺逆天的,但產(chǎn)品經(jīng)理堅(jiān)持要這么弄,也只能硬著頭皮寫。頁(yè)面的表單數(shù)據(jù)樣式五花八門,下圖是其中幾行截圖
第一、二行的 textfield 其實(shí)是一個(gè)選擇框,只能從下拉選項(xiàng)中選擇一個(gè)。第三個(gè)只允許輸入數(shù)字。
頁(yè)面由另一個(gè)同學(xué)實(shí)現(xiàn),表單的數(shù)據(jù)基本都在 cellForRowAtIndexPath 實(shí)現(xiàn),結(jié)果是這樣的:
看著這么多的if...else...一下子就凌亂了。讓我怎么接手實(shí)現(xiàn)網(wǎng)絡(luò)接口,上傳表單數(shù)據(jù),難道也寫這么多的if...else...?這么實(shí)現(xiàn)之后要改其中某行數(shù)據(jù)的話,比如增加或刪除一行表單,就得改N個(gè)地方。這樣不僅浪費(fèi)時(shí)間,而且容易出錯(cuò),要么改錯(cuò)了要么沒(méi)改全。這樣的代碼后期維護(hù)成本太高,只能重寫了。那么問(wèn)題來(lái)了,怎么改?從何開(kāi)始?
XLForm
XLForm is the most flexible and powerful iOS library to create dynamic table-view forms. Fully compatible with Swift & Obj-C.
XLForm 是最靈活且最強(qiáng)大的創(chuàng)建動(dòng)態(tài)表單的iOS庫(kù)。更多的使用方法可以參考這篇文章:http://www.dbjr.com.cn/article/138943.htm
以下是這個(gè)庫(kù)一個(gè)簡(jiǎn)單的結(jié)構(gòu)圖:
最主要的是紅色方框的三個(gè)類:XLFormRowDescriptor, XLFormSectionDescriptor,XLFormDescriptor。XLFormDescriptor結(jié)構(gòu)和UITablView一樣,有Section,有Row,它就是為成為UITableView的數(shù)據(jù)源而設(shè)計(jì)的。XLFormRowDescriptor定義了每行表單的數(shù)據(jù)內(nèi)容,包括行樣式,標(biāo)題,行類型,選擇項(xiàng)內(nèi)容,標(biāo)簽,合法性驗(yàn)證等。XLFormSectionDescriptor是由XLFormRowDescriptor組合而成的,而XLFormSectionDescriptor最終又組成了XLFormDescriptor。
由于我們要實(shí)現(xiàn)的APP表單行樣式更復(fù)雜,有的一行要提交兩項(xiàng)數(shù)據(jù),所以需要對(duì)XLFormRowDescriptor做些改動(dòng)。代碼如下:
@property (strong, nonatomic) NSMutableDictionary *cellConfig; @property (strong, nonatomic) NSDictionary *textFieldConfig; @property (strong, readonly, nonatomic) NSString *rowType; @property (strong, readonly, nonatomic) NSArray *leftOptions; @property (strong, readonly, nonatomic) WWEFormRightSelectorOption *rightOptions; @property (strong, nonatomic) NSString *title; @property (strong, nonatomic) id value; @property (strong, nonatomic) NSString *tag; @property (nonatomic) BOOL required; @property (strong, nonatomic) WWEBaseTableViewCell *tableViewCell; -(id)initWithRowType:(NSString *)rowType title:(NSString *)title leftOptions:(NSArray *)leftOptions rightOptions:(WWEFormRightSelectorOption *)rightOptions; -(WWEFormValidation *)doValidation; @end @interface WWEFormRightSelectorOption : NSObject @property (readonly, nonatomic) NSArray *rightOptions; @property (readonly, nonatomic) NSString *httpParameterKey; @property (readonly, nonatomic) NSString *selectorTitle; @property (nonatomic) NSInteger selectedIndex; +(WWEFormRightSelectorOption *)formRightSelectorOptionWithTitle:(NSString *)title httpParameterKey:(NSString *)httpParameterKey rightOptions:(NSArray *)rightOptions;
這樣,表單數(shù)據(jù)就獨(dú)立于UI,TableViewCell是可配置的。通過(guò)XLFormRowDescriptor 來(lái)配置TableViewCell。只要指定行類型,就給出相應(yīng)的類型的Cell。TableViewController中的Cell是由XLFormRowDescriptor提供的。
- (WWEBaseTableViewCell *)tableViewCell { if (!_tableViewCell) { id cellClass = [self cellClassesForRowDescriptorTypes][self.rowType]; NSAssert(cellClass, @"Not defined XLFormRowDescriptorType: %@", self.rowType ?: @""); _tableViewCell = [[cellClass alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:nil]; NSAssert([_tableViewCell isKindOfClass:[WWEBaseTableViewCell class]], @"UITableViewCell must extend from WWEBaseTableViewCell"); _tableViewCell.rowDescriptor = self; } return _tableViewCell; }
那么 XLFormRowDescriptor 怎么根據(jù)不同的行給出不同的cell?cell需要繼承同一個(gè)父類,這個(gè)父類足夠簡(jiǎn)單,只有一個(gè)WWEFormRowDescriptor屬性,其它需要實(shí)現(xiàn)的方法則由WWECellProtocal負(fù)責(zé)。
@class WWEFormRowDescriptor; @interface WWEBaseTableViewCell : UITableViewCell<WWECellProtocal> @property (nonatomic, weak) WWEFormRowDescriptor *rowDescriptor; @end
@class WWEFormRowDescriptor; @protocol WWECellProtocal <NSObject> @required @property (nonatomic, weak) WWEFormRowDescriptor *rowDescriptor; -(void)configure; -(void)update; @end
另外,將表單配置數(shù)據(jù)獨(dú)立出來(lái),而不是寫死在代碼中。使用plist文件,這樣初始化form時(shí)只需讀取這個(gè)文件,就完成了cell的配置。后續(xù)如果要改動(dòng)表單的內(nèi)容,不改動(dòng)樣式,就只需修改plist文件,而無(wú)需改動(dòng)代碼。
最后TableViewController的代碼就格外簡(jiǎn)單了:
這樣上傳表單數(shù)據(jù)就無(wú)比輕松了, [self.form localValidationErrors]
驗(yàn)證數(shù)據(jù)合法性,全部合法的話再調(diào)用 [self.form httpParameters]
就獲取了所有的表單數(shù)據(jù),傳給網(wǎng)絡(luò)接口就大功告成了。
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
iOS實(shí)現(xiàn)app間跳轉(zhuǎn)功能
這篇文章主要為大家詳細(xì)介紹了iOS實(shí)現(xiàn)app間跳轉(zhuǎn)功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05iOS Swift控制器轉(zhuǎn)場(chǎng)動(dòng)畫(huà)示例代碼
這篇文章主要給大家介紹了關(guān)于iOS Swift控制器轉(zhuǎn)場(chǎng)動(dòng)畫(huà)的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)各位iOS開(kāi)發(fā)者們具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2018-01-01iOS中長(zhǎng)條藍(lán)色按鈕(button)實(shí)現(xiàn)代碼
本文通過(guò)實(shí)例代碼給大家介紹了iOS中長(zhǎng)條藍(lán)色按鈕(button)實(shí)現(xiàn)方法,代碼簡(jiǎn)單易懂,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下吧2017-08-08iOS項(xiàng)目的開(kāi)發(fā)命名規(guī)范教程
為了團(tuán)隊(duì)各成員之間代碼的互通、可讀、易維護(hù)性,特制訂此開(kāi)發(fā)規(guī)范。下面這篇文章主要給大家介紹了關(guān)于iOS項(xiàng)目的開(kāi)發(fā)命名規(guī)范的相關(guān)資料,需要的朋友可以參考借鑒,下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2017-11-11iOS實(shí)現(xiàn)自定義購(gòu)物車角標(biāo)顯示購(gòu)物數(shù)量(添加商品時(shí)角標(biāo)抖動(dòng) Vie)
本文主要介紹了iOS實(shí)現(xiàn)自定義購(gòu)物車及角標(biāo)顯示購(gòu)物數(shù)量(添加商品時(shí)角標(biāo)抖動(dòng) Vie)的相關(guān)知識(shí)。具有很好的參考價(jià)值。下面跟著小編一起來(lái)看下吧2017-04-04兼容iOS 10 升級(jí)xcode8出現(xiàn)的問(wèn)題及一些適配問(wèn)題的解決方案
這篇文章主要介紹了兼容iOS 10 升級(jí)xcode8出現(xiàn)的問(wèn)題及一些適配問(wèn)題的解決方案,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-09-09Swift 開(kāi)發(fā)之懶加載的實(shí)例詳解
這篇文章主要介紹了Swift 開(kāi)發(fā)之懶加載的實(shí)例詳解的相關(guān)資料,希望通過(guò)本文能幫助到大家,實(shí)現(xiàn)這樣的功能,需要的朋友可以參考下2017-09-09IOS 頭文件導(dǎo)入-@class注意事項(xiàng)總結(jié)
這篇文章主要介紹了IOS 頭文件導(dǎo)入-@class注意事項(xiàng)總結(jié)的相關(guān)資料,需要的朋友可以參考下2017-05-05