教你寫能被舒服舒服又舒服地調用的iOS庫


目錄

前言

腦洞開一開

分析

整容

結語

 

前言

 

2014年過的那么快,過年又那么塊,2015年又是飛快地節奏,真尼瑪感覺上帝是不是無聊使用了變速外掛開啟了加速模式~到現在博主都無法接受已經上班的事實……在地鐵臉被擠在玻璃上的時候只能用眼神寫滿傻X射這個世界一臉!!原諒博主那么雞糞~因為哥最近生病了,不嗨心~~,我想來去想不明白,博主每周健身4天,胸肌壓女友、拳頭比沙煲、吃喝健又康、體硬似野馬,怎么會生病呢?我苦思多日,終於有一天早上起床看了鏡子半個小時后我才頓悟,我靠~原來長得帥真的是錯!上帝在嫉妒我才讓我生病的~哈哈哈 咳咳~這是一個嚴肅的技術博客~是絕對拒絕惡俗低下的內容的,大丈夫應該心系天下才對,最近柴靜和霧霾真的觸動了我們每個人的心;某極端組織像嗨大了一樣,真的是屠龍寶刀在手,裝備任爆的節奏;近下的港島撐完雨傘又要趕人了……我真的是非常痛心疾首,維護世界和平一直是鄙人的願望,現在的世界這么混亂,本人真的很慚愧,都怪我咯~~沒有因為什么,所以今天博主要教大家怎么寫一個自己用着安逸,修改巴適,他人用起來又無比舒服的iOS庫!!

 

腦洞開一開


本人不才,接觸代碼的世界時間真心不是很久,對於系統底層優化、框架級別設計、設計模式的效率優劣等方面的學習和研究比不深刻,所以如果你是大神,你狠溜的話,我還是建議您路過呵呵一下然后就關了這個網頁吧,雕蟲小技無法入爾法眼啊~我不是怕被噴很菜(反正都被老師噴了二十年了……^_^)而是我對大神都是有敬畏之心的,不能說讓你看你就看,你可以試一下是吧,你也不喜歡看完后是Duang 那么多特效 特技……咳咳 反正這篇文章真的很菜~~~適合比我菜的菜鳥,慎讀。

不過嘛我們要寫一個庫,能夠讓別人用,用起來要很舒服的,那肯定得要設計得當,肯定要熟悉平台,設計模式非常溜才行,這個沒錯~但是平台、架構、設計這個東西這么復雜,我解釋大家也不懂,那我就不解釋了(你強迫症發作死都要懂可自行找一個大學,里面有老師~)

我發誓不扯BB了 我們一起來寫一個能被舒服舒服又舒服地調用的iOS庫,最近衛生紙漲價了,好煩,其實也是有原因的,因為畢竟生產費用什么的也都…我頭上的磚頭誰扔的!!??信不信我不打死你丫的……

假設我們要開發一個像微信朋友圈或者微博這玩意兒的東西,並且要有能隱藏和展開正文、正文中能夠識別富文本(網址、電話、@姓名……)、還要有圖片縮略圖而且點擊查看、能夠有回復、回復文中也要支持富文本……這些功能。像這樣
圖1


所謂君子生非異也 善假於物也。上面的功能這么多這么復雜自己寫,你確定你不是吃飽了撐着?經理也說要敏捷開發,所以二話不說閃現到code4app,code.cocoachina祭出最強殺器——搜索引擎,找到了這個庫WFCoretext https://github.com/TigerWf/WFCoretext 發現它完美符合我們的需求呀,棒棒噠

 

分析


首先我們感謝WFCoretext 的作者的開源貢獻,請收下我的膝蓋,我們馬上來用一用~
圖2


槽點不多
我們來分析分析它怎么搞得把
工程結構 圖3


嗯額 還是比較簡單的 View文件夾里面的是控件啦
圖4


具體實現 大家可以自己看啦,這位哥哥代碼風格還是比較規范的,看起來不費勁
Manager文件夾中是一些富文本匹配規則,其中YMTextData很重要 下面說
我們直接來看看怎么使用的把
點開WXViewController

導包並且聲明變量,變量在實現接下來的實現中會進行初始化

#import "WXViewController.h"
#import "YMTableViewCell.h"
#import "ContantHead.h"
#import "YMShowImageView.h"
#import "YMTextData.h"
#import "YMReplyInputView.h"

#define dataCount 10
#define kLocationToBottom 20
#define kAdmin @"小虎-tiger"


@interface WXViewController ()<UITableViewDataSource,UITableViewDelegate,cellDelegate,InputDelegate>
{
NSMutableArray *_imageDataSource;

NSMutableArray *_contentDataSource;//模擬接口給的數據

NSMutableArray *_tableDataSource;//tableview數據源

NSMutableArray *_shuoshuoDatasSource;//說說數據源

UITableView *mainTable;

UIButton *replyBtn;

YMReplyInputView *replyView ;
}
@end  

這三個方法分別構建初始化了一個tableview,初始化並賦值圖片數據,初始化並賦值其他數據

- (void) initTableview;
- (void)configImageData;
- (void)loadTextData;

 

在loadTextData中會將數據包裝成YMTextData的數組,這樣一個tableCell里面的數據就使用一個YMTextData的數據
然后在

- (void)calculateHeight:(NSMutableArray *)dataArray

中會計算出數據所占用view的高度 這里面也就實現了 我們需求里面可以擴展可以收縮的功能

計算完高度然后就重新加載tableview了 然后tableview的各種delegate方法 各種datasource方法就呼呼的運行了

其中 以下方法中又再次使用了我們在

 

- (void)calculateHeight:(NSMutableArray *)dataArray

方法中計算出來的高度來設置tablecell的高度

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
YMTextData *ym = [_tableDataSource objectAtIndex:indexPath.row];
BOOL unfold = ym.foldOrNot;
return TableHeader + kLocationToBottom + ym.replyHeight + ym.showImageHeight + kDistance + (ym.islessLimit?0:30) + (unfold?ym.shuoshuoHeight:ym.unFoldShuoHeight) + kReplyBtnDistance;
}

以下這個方法又把YMTextData 賦值給了YMTableViewCell

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
static NSString *CellIdentifier = @"ILTableViewCell";
YMTableViewCell *cell = (YMTableViewCell *)[tableView dequeueReusableCellWithIdentifier:CellIdentifier];
if (cell == nil) {
cell = [[YMTableViewCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:CellIdentifier];
}
cell.stamp = indexPath.row;
cell.replyBtn.tag = indexPath.row;
cell.replyBtn.appendIndexPath = indexPath;
[cell.replyBtn addTarget:self action:@selector(replyAction:) forControlEvents:UIControlEventTouchUpInside];
cell.delegate = self;
[cell setYMViewWith:[_tableDataSource objectAtIndex:indexPath.row]];
return cell;
}

阿拉巴拉…… 呼呼
我就想問你一句 累不累?
現在我分析了一遍這個庫,你知道該怎么用了么,不要懷疑自己的智商,我也要再看一遍才知道怎么用。
當然我們不能懷疑作者對於開源技術做出的貢獻!我們也不能懷疑作者的開發技術,畢竟這個庫的bug還是比較少的,作者是偉大崇高的!減少了世界的碳排量方便了你我他,為很多代碼工作者提供了方便,讓他們可以不加班,早早的回家陪老婆陪基友陪孩子陪寵物……所以我要再次感謝作者

但是作為一個有完美強迫症的博主,問自己一句 為什么這個庫那么難用?因為它難用
怎么才算是好用?系統自帶的組件,使用和學習起來那么容易?這應該算是好用吧。

所以接下來,我們要開刀WXViewController 讓他DUANG的一下,變得使用起來 舒服舒服又舒服!!

整容


我們不妨WXViewController的實現細節都封裝起來!它是一個tableview、它怎么計算高度,它的回復按鈕怎么生成……巴拉巴拉我都不想管!!我的願望是,我只要扔進去數據,就自動生成一個朋友圈出來!!!!

有了願望,急着召喚神龍也沒用!規則是要集齊龍珠呀!!!好我們先干起來!整容WXViewController!!!

我們要開發一個東西就叫做 “朋友圈模板”吧 在“朋友圈模板.h”中要有一個“朋友圈模板Delegate” 然后里面要有一個方法

-(返回的數據*)每行朋友圈的數據:index;

我們使用的時候就這樣:

真朋友圈.h

@interface 真朋友圈 : 朋友圈模板<朋友圈模板Delegate> 
@end

真朋友圈.m

@implementation 真朋友圈
- (void)viewDidLoad {
self.delegate = self;
}

-(返回的數據*)每行朋友圈的數據:index{
return [朋友圈數據數組 objectAtIndex:index];
}
@end

這樣就好了!你就再也不用關心朋友圈怎么實現了 你只要關系你的數據部分!!就問你Nice不Nice???
所以這里只要你開發好了“朋友圈模板.m”那么以后“朋友圈模板.h”和“朋友圈模板.m”就是你寫好的能被人舒服舒服又舒服調用的庫了!酷不酷??

博主就手把手教你怎么寫這個“朋友圈模板.m”吧 嘻嘻嘻 手把手哦 呵呵呵 手 把 手喲 博主是很有愛的哦~~~~

我們要開發一個DDRichTextViewController來代替WXViewController <——這個太難用了

我們先來寫寫DDRichTextViewController.h嘛

 

//學學系統組件 我們也來弄一個delegate和datasource~ 其實都是delegate為了更好地區分功能,datasource主要用來設置數據有關

@protocol DDRichTextViewDelegate 
@required
-(NSString*)senderName;//必須要實現!不然評論別人的時候沒名字 最恨匿名渣渣,自己叫的名字都不敢直接說!!“有誰知道我買充氣娃娃都匿名呢 呵呵,啊?為什么我心里想的會變成文字顯示出來!!!納尼!!”
@optional
-(BOOL)hideReplyButtonForIndex:(NSInteger)index;//是否隱藏回復按鈕,有時候我們不讓人回復 就把回復按鈕隱藏起來了
-(void)didPromulgatorPressForIndex:(NSInteger)index name:(NSString*)name;//發布者的頭像或者名字被點擊
-(void)didRichTextPressedFromText:(NSString*)text index:(NSInteger)index;//正文的富文本被點擊的回調
-(void)didRichTextPressedFromText:(NSString*)text index:(NSInteger)index replyIndex:(NSInteger)replyIndex;//評論的富文本被點擊的回調
-(void)replyForIndex:(NSInteger)index replyText:(NSString*)text;//回復文字的內容的回調
@end


@protocol DDRichTextViewDataSource 
@required
-(YMTextData*)dataForRowAtIndex:(NSInteger)index;//這個就是每行需要的數據了!
-(NSInteger)numberOfRowsInDDRichText;//需要返回多少行
@end

@interface DDRichTextViewController : UIViewController<UITableViewDataSource,UITableViewDelegate,cellDelegate,InputDelegate>
@property (weak, nonatomic) id delegate;
@property (weak, nonatomic) id dataSource;

@end

然后就是DDRichTextViewController.m了
基本上就是對WXViewController.m的封裝了! 讓其內部實現的細節都對使用者透明化
比如在DDRichTextViewController中實現了uitableView的datasource和delegate
在這個方法中 tableview需要顯示的行數就由繼承DDRichTextViewController的子類的datasource中的

-(NSInteger)numberOfRowsInDDRichText;


這個方法返回的數據作為參數!

比如這樣:

-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
return [[self dataSource] numberOfRowsInDDRichText];
}

所以當我們使用我們自己寫的庫的時候根本不在乎這個方法

-(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView;
  

因為我們封裝到了

  -(NSInteger)numberOfRowsInDDRichText;

類似其他實現以及方法都進行了封裝

這當中處理@required很簡單,用戶必須已經實現了 所以直接調用就好 。 但是 @optional的方法用戶不一定會去實現,所以當中最重要的就是要去判斷這個方法存不存在:方法如下

respondsToSelector:NSSelectorFromString(@“方法名:”) //這個凡是繼承NSObject的類都擁有這個方法 這個是基礎了,是運行時判斷方法存不存在的

詳細如下

if ([self.delegate respondsToSelector:NSSelectorFromString(@"hideReplyButtonForIndex:")]) { //判斷hideReplyButtonForIndex方法存不存在 存在才會執行如下的代碼
if ([[self delegate] hideReplyButtonForIndex:indexPath.section]) {
cell.hideReply = YES;
}
}

下面是詳細的代碼
(本來想貼詳細代碼的,博主一思忖!最好下載我的Demo進行研究 這樣可以在方法之間跳轉 更能看得懂!!地址在文后!而且你們在博主的Demo項目中star一下 我就愛死你Y的了)

最后我們來看看怎么使用寫好的DDRichTextViewController

我們新建一個TestViewController

TestViewController.h

#import "DDRichTextViewController.h"

@interface TestViewController : DDRichTextViewController<DDRichTextViewDataSource,DDRichTextViewDelegate>

@end
 

TestViewController.m

//
// TestViewController.m
// WFCoretext
//
// Created by David on 15/2/7.
// Copyright (c) 2015年 tigerwf. All rights reserved.
//

#import "TestViewController.h"

@interface TestViewController ()

@end

@implementation TestViewController
NSMutableArray * ymDataArray;

- (void)viewDidLoad {
[super viewDidLoad];
NSMutableArray MyDataArr = [[NSMutableArray alloc]init];//!!!!這里應該自己初始化數據
self.delegate = self;
self.dataSource = self;


}


//下面兩個是datasource方法
-(NSInteger)numberOfRowsInDDRichText{
return 5;
}

-(YMTextData *)dataForRowAtIndex:(NSInteger)index{
return [MyDataArr objectAtIndex:0];//!!!!!!!! MyDataArr 是一個YMTextData的數組!!所以你的朋友圈數據的每一項都必須是YMTextData或者繼承YMTextData的子類!!
}

//下面所有都是delegate的方法 朋友圈所有的特性都使用以下的delegate方法進行控制 方法有可選和必選的 可自行實現 接口調用簡單
-(NSString *)senderName{
return @"David";
}

-(BOOL)hideReplyButtonForIndex:(NSInteger)index{
return NO;
}

-(void)didPromulgatorNameOrHeadPicPressedForIndex:(NSInteger)index name:(NSString *)name{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"發布者回調" message:[NSString stringWithFormat:@"姓名:%@\n index:%d",name,index] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
}


-(void)didRichTextPressedFromText:(NSString*)text index:(NSInteger)index{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"正文富文本點擊回調" message:[NSString stringWithFormat:@"點擊的內容:%@\n index:%d",text,index] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
}

-(void)didRichTextPressedFromText:(NSString *)text index:(NSInteger)index replyIndex:(NSInteger)replyIndex{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"評論的富文本點擊回調" message:[NSString stringWithFormat:@"點擊的內容:%@\n index:%d \n replyIndex:%d",text,index,replyIndex] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
}

-(void)replyForIndex:(NSInteger)index replyText:(NSString*)text{
UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"回復的回調" message:[NSString stringWithFormat:@"回復的內容:%@\n index:%d",text,index] delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil, nil];
[alert show];
}

@end

 

效果如下
圖5

項目及Demo地址:https://github.com/daiweilai/DDRichText

結語


博主對WXViewController的改動還是頗多的,不單單是簡單地封裝,我還做了表情和姓名的正則判斷,還對YMTableViewCell做了大量的邏輯修改,回調的接口做更改和增添也甚多!需要你親自去發現博主隱藏的愛,但無論如何這都是潦草的項目,想要正式的使用在企業開發中這還遠遠不夠的!沒有進行模塊和單元的測試,其中圖片的處理方式也不好,這里我是直接要求用戶添加Image文件的 這個應該改成 添加圖片地址,然后讓這個庫異步去請求顯示的……所以還是需要大家的開源精神和力量去貢獻自己的,燃燒自己的,騷年文章結束了

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM