項(xiàng)目結(jié)構(gòu)更加清晰贺归,方便以后調(diào)試bug
怎么讓項(xiàng)目結(jié)構(gòu)更加清晰,誰(shuí)的事情誰(shuí)管理
-
分析項(xiàng)目架構(gòu)
- 方便處理多人開發(fā)
- 讓更多的功能復(fù)用
- 讓代碼的結(jié)構(gòu)更加清晰
-
自定義類的思路
- 當(dāng)系統(tǒng)的某些類不能滿足需求的時(shí)候,需要給系統(tǒng)的類添加某些功能驯用,但還要保持原有類的功能驼鞭,采用自定義類,繼承系統(tǒng)的類(自定義控件轨蛤,自定義模型)
- 讓項(xiàng)目的結(jié)構(gòu)更加清晰蜜宪,誰(shuí)的事情誰(shuí)管理,采取自定義類祥山,以后這個(gè)類的問題圃验,可以馬上定位到這個(gè)類做的哪些方法(自定義控制器)
-
復(fù)習(xí)程序的運(yùn)行
- main -> UIApplicationMain
- 創(chuàng)建UIApplication對(duì)象
- 創(chuàng)建UIApplication對(duì)象的代理
- 開啟主運(yùn)行循環(huán),保持程序一直運(yùn)行
- 加載info.plist文件,判斷下是否指定main
-
關(guān)于窗口的建立
- 創(chuàng)建窗口
- 創(chuàng)建窗口的根控制器
- 添加子控制器
- 讓窗口顯示
- 代碼實(shí)現(xiàn) (去消info中Main)
// 1.創(chuàng)建窗口
self.window = [[UIWindow alloc] initWithFrame:[UIScreen mainScreen].bounds];
// 2.創(chuàng)建窗口的跟控制器
// 創(chuàng)建tabBarVc
UITabBarController *tabBarVc = [[UITabBarController alloc] init];
// 設(shè)置窗口的根控制器
self.window.rootViewController = tabBarVc;
// 添加子控制器
UIViewController *vc = [[UIViewController alloc] init];
vc.view.backgroundColor = [UIColor redColor];
[tabBarVc addChildViewController:vc];
...
// 3.讓窗口顯示
[self.window makeKeyAndVisible];
-
搭建主架構(gòu)(讓程序更清晰化)
- 不同的頁(yè)面交給不同的控制器管理缝呕,并放在不同的文件夾中澳窑。
結(jié)構(gòu)清晰化,誰(shuí)的控件誰(shuí)管理供常。
-
自定義UITabBarController
- 內(nèi)部添加子控制器 可以進(jìn)行自定義方法摊聋,來抽取相同代碼,簡(jiǎn)化程序
- 自定義 子控制器栈暇,誰(shuí)的控件麻裁,誰(shuí)管理,方便管理子控制器- 關(guān)于主流框架 (UITabBarController 和 UINavigationController) 主要是上面一個(gè)條,下面一個(gè)條煎源。
- 一般是在 UITabBarController上添加UINavigationController 這樣的話色迂, 在UINavigationController上push子控制器的話 (這樣的話上面的條可以叫給push進(jìn)去的子控制器進(jìn)行管理)
-
注意, 每運(yùn)行一步手销,都都要看看所運(yùn)行的效果歇僧,滿不滿足自己的需求。
- 此處由于美工的的圖片(tabBar的圖標(biāo)圖片锋拖,大于tabBar的寬度诈悍,所以沒有顯示出想要的效果)【tabBar的寬度44 < 圖片高度】
所以需要自定義tabBar,因?yàn)橄到y(tǒng)的tabBar不滿足需求
- 此處由于美工的的圖片(tabBar的圖標(biāo)圖片锋拖,大于tabBar的寬度诈悍,所以沒有顯示出想要的效果)【tabBar的寬度44 < 圖片高度】
- 自定義TabBar兽埃,取代原有tabBar侥钳,并實(shí)現(xiàn)原有tabBar的功能
- 有多少按鈕控件,控件的樣式有什么決定柄错。 (對(duì)于tabBar慕趴,它的樣式是有對(duì)應(yīng)控制器決定的,可以這么說鄙陡,對(duì)應(yīng)控制器冕房,將所設(shè)置的樣式素材,存放在
tabBarItem 屬性中
tabBarItem就是一個(gè)模型數(shù)據(jù)趁矾,tabBar從模型數(shù)據(jù)中讀取了素材耙册,再賦值到它對(duì)應(yīng)的位置上) - 所以底層實(shí)現(xiàn)我認(rèn)為是這么幾步。
- 從外界讀取模型數(shù)據(jù)毫捣,有多少控制器详拙,應(yīng)該就有多少組模型數(shù)據(jù)。
- 提取模型數(shù)據(jù)蔓同,并將其賦值給子控件button上饶辙。
- 給子控件設(shè)置位置
- 點(diǎn)擊按鈕,跳轉(zhuǎn)到對(duì)應(yīng)的控制器斑粱。
- 當(dāng)然弃揽,還有一些細(xì)節(jié)需要處理,比如按鈕不需要高亮指示则北,只需要選中和正常兩種狀態(tài)矿微, (所以需要自定義按鈕,重寫
- (void)setHighlighted:(BOOL)highlighted
方法尚揣,為空即可涌矢,讓其在高亮?xí)r不做任何操作);按鈕默認(rèn)選中第一個(gè)快骗;選中一個(gè)那么前一個(gè)就要取消選中娜庇。 - 具體代碼如下塔次。
- 有多少按鈕控件,控件的樣式有什么決定柄错。 (對(duì)于tabBar慕趴,它的樣式是有對(duì)應(yīng)控制器決定的,可以這么說鄙陡,對(duì)應(yīng)控制器冕房,將所設(shè)置的樣式素材,存放在
// 由于不確定什么時(shí)候有數(shù)據(jù)加入,所以采用懶加載方式名秀,進(jìn)行items的加載俺叭,加載完成后才進(jìn)行按鈕的創(chuàng)建以及賦值操作,不需要在視圖 一創(chuàng)建的時(shí)候泰偿,就取創(chuàng)建按鈕, 節(jié)約功耗
- (void)setItems:(NSArray *)items
{
_items = items;
for (UITabBarItem *item in items) {
// 從items中取出模型數(shù)據(jù)蜈垮,并將其賦值給對(duì)應(yīng)按鈕
UIButton *btn = [LXLTabBarButton buttonWithType:UIButtonTypeCustom];
// 設(shè)置內(nèi)容
[btn setBackgroundImage:item.image forState:UIControlStateNormal];
[btn setBackgroundImage:item.selectedImage forState:UIControlStateSelected];
[btn addTarget:self action:@selector(btnClick:) forControlEvents:UIControlEventTouchDown];
btn.tag = self.subviews.count;
if (self.subviews.count == 0) {
// 為了程序之處耗跛,沒有子控件的時(shí)候,按鈕就已經(jīng)點(diǎn)擊攒发。
[self btnClick:btn];
}
[self addSubview:btn];
}
}
// 進(jìn)行子控件的布局调塌, 根據(jù)子控件的個(gè)數(shù),遍歷對(duì)每個(gè)子控件進(jìn)行布局
- (void)layoutSubviews
{
[super layoutSubviews];
int count = (int)self.subviews.count;
CGFloat btnW = self.bounds.size.width / count;
CGFloat btnH = self.bounds.size.height;
CGFloat btnY = 0;
CGFloat btnX = 0;
// 布局按鈕的位置
for (int i = 0; i < count; i++) {
UIButton *btn = self.subviews[i];
btnX = i * btnW;
btn.frame = CGRectMake(btnX, btnY, btnW, btnH);
}
}
// 監(jiān)聽點(diǎn)擊事件惠猿, 由于是逆向傳值羔砾,所以需要代理,來完成點(diǎn)擊事件處理偶妖,并且傳入?yún)?shù)(點(diǎn)擊的是哪個(gè)按鈕)tag 姜凄,來完成讓控制器的跳轉(zhuǎn)
- (void)btnClick:(UIButton *)btn
{
_selBtn.selected = NO;
btn.selected = YES;
_selBtn = btn;
// 通知代理點(diǎn)擊了哪個(gè)角標(biāo)的按鈕
if ([_delegate respondsToSelector:@selector(tabBar:didClickBtn:)]) {
[_delegate tabBar:self didClickBtn:btn.tag];
}
}
- 對(duì)于父控件,tabBarController
- 創(chuàng)建自定義tabBar控件趾访,并將其作為子控件放置在态秧,原系統(tǒng)tabBar之上,蓋住系統(tǒng)tabBar扼鞋。(需要注意的是申鱼,要移除系統(tǒng)原有的tabBar,但是系統(tǒng)底部處理的時(shí)候并不是馬上移除云头,會(huì)在過一段事件才進(jìn)行移除操作)捐友。
- 需要將對(duì)應(yīng)的模型數(shù)組數(shù)據(jù)(懶加載),傳入溃槐,屬于順傳
- 作為代理匣砖,完成對(duì)應(yīng)的方法,進(jìn)行控制器的跳轉(zhuǎn)
- 具體代碼如下
// 取得模型數(shù)據(jù)昏滴,并存放在模型數(shù)組中
- (void)setUpOneChildViewController:(UIViewController *)vc image:(UIImage *)image selImage:(UIImage *)selImage
{
// 設(shè)置tabBarButton的圖片脆粥,tabBarButton的內(nèi)容由對(duì)應(yīng)的子控制器的tabBarItem
vc.tabBarItem.image = image;
vc.tabBarItem.selectedImage = selImage;
// 保存對(duì)應(yīng)子控制器的UITabBarItem
[self.items addObject:vc.tabBarItem];
[self addChildViewController:vc];
}
- (void)setUpTabBar
{
// 1.移除系統(tǒng)的tabBar,移除系統(tǒng)自帶的tabBarButton
[self.tabBar removeFromSuperview];
NSLog(@"%@",self.tabBar);
// 2.添加自己的tabBar
LXLTabBar *tabBar = [[LXLTabBar alloc] init];
tabBar.delegate = self;
// tabBar按鈕的個(gè)數(shù),由tabBar子控制器個(gè)數(shù)決定
// tabBar.count = (int)self.childViewControllers.count;
// 傳對(duì)應(yīng)子控制器的tabBarItem數(shù)組
tabBar.items = self.items;
tabBar.backgroundColor = [UIColor greenColor];
tabBar.frame = self.tabBar.frame;
[self.view addSubview:tabBar];
}
// 遵守協(xié)議實(shí)現(xiàn)代理方法影涉,進(jìn)行界面的跳轉(zhuǎn)变隔。
#pragma mark -LXLTabBarDelgate方法
// 當(dāng)點(diǎn)擊tabBar上的條的時(shí)候調(diào)用
- (void)tabBar:(XMGTabBar *)tabBar didClickBtn:(NSInteger)index
{
// 切換界面
self.selectedIndex = index;
}
- 對(duì)于跳轉(zhuǎn)按鈕的添加
- 我覺得有兩種方案
- 添加到最后一個(gè)cell上去
- 直接添加到collectionView上去。
- 第一種方案的實(shí)現(xiàn)
- 我覺得有兩種方案
// 判斷是否是最后一個(gè)cell,可以根據(jù)indexPath進(jìn)行判斷
- (UICollectionViewCell *)collectionView:(UICollectionView *)collectionView cellForItemAtIndexPath:(NSIndexPath *)indexPath
{
...
// 給最后一個(gè)cell添加一個(gè)立即體驗(yàn)按鈕
// 告訴cell是否是最后一個(gè)
// 宏定義數(shù)目為4
[cell setIndexPath:indexPath count:LXLPage];
// [cell setIndexPath:indexPath count:4]
// if (indexPath.item == XMGPage - 1) {
// // 添加立即體驗(yàn)按鈕
//
// }
return cell;
}
- cell內(nèi)部拿到
indexPath
和count
進(jìn)行判斷
// 用來判斷下當(dāng)前cell對(duì)象是否是最后一個(gè)cell
- (void)setIndexPath:(NSIndexPath *)indexPath count:(int)count
{
if (indexPath.item == count - 1) {
// 最后一個(gè)cell
// 添加一個(gè)立即體驗(yàn)按鈕,首先保存整個(gè)cell只有一個(gè)體驗(yàn)按鈕
// 顯示這個(gè)按鈕
self.startBtn.hidden = NO;
}else{ // 不是最后一個(gè)cell
// 隱藏這個(gè)按鈕,涉及到循環(huán)引用蟹倾,否則就會(huì)被別的cell所引用匣缘。
self.startBtn.hidden = YES;
}
}
- 由于cell的按鈕是用的時(shí)候才進(jìn)行顯示猖闪,(所以可以進(jìn)行懶加載
)
- (UIButton *)startBtn
{
if (_startBtn == nil) {
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
_startBtn= btn;
[btn setBackgroundImage:[UIImage imageNamed:@"guideStart"] forState:UIControlStateNormal];
// 尺寸可以進(jìn)行自適應(yīng)'自己根據(jù)圖片來安排大小'
[btn sizeToFit];
// 位置確定
btn.center = CGPointMake(self.width * 0.5 , self.height * 0.9);
[self.contentView addSubview:btn];
}
return _startBtn;
}
- 對(duì)于控制器之間的跳轉(zhuǎn),點(diǎn)擊按鈕跳轉(zhuǎn)到新控制器肌厨。
- 沒有任何聯(lián)系的控制器跳轉(zhuǎn)培慌,直接跳轉(zhuǎn)即可。 創(chuàng)建到主窗口根控制器顯示柑爸,代替原有主窗口根控制器
// 點(diǎn)擊立即體驗(yàn)按鈕的時(shí)候調(diào)用
- (void)start
{
// 跳轉(zhuǎn)到主框架界面吵护。界面之間跳轉(zhuǎn),導(dǎo)航控制器表鳍,tabBarVc,modal
// 不能使用modal原因:新特性界面一直存在馅而,被窗口的根控制器一直強(qiáng)引用
LXLTabBarController *vc = [[LXLTabBarController alloc] init];
// 設(shè)置窗口的根控制器為主框架控制器
[UIApplication shareApplication].keyWindow.rootViewController = vc;
}
- 第二種方法(代碼實(shí)現(xiàn))
- (viod)viewDidLoad{
UIButton *btn = [UIButton buttonWithType:UIButtonTypeCustom];
[btn setImage:[UIImage imageNamed:@"guideStart"] forState:UIControlStateNormal];
[btn sizeToFit];
// 初始化位置直接設(shè)置在 最終位置,而且只創(chuàng)建一次譬圣,不必思考情況
btn.center = CGPointMake(self.collectionView.width * (LXLPageCount - 0.5), self.collectionView.height * 0.9);
[self.collectionView addSubview:btn];
[btn addTarget:self action:@selector(btnClick) forControlEvents:UIControlEventTouchUpInside];
}
// 跳轉(zhuǎn)控制器
- (void)btnClick
{
LXLTabBarController *vc =[[LXLTabBarController alloc]init];
[UIApplication sharedApplication].keyWindow.rootViewController =vc;
}
- 總結(jié)一下瓮恭,我覺得第二個(gè)方法好,因?yàn)槭强刂破鬟M(jìn)行跳轉(zhuǎn)厘熟,我覺得完全是控制器自己的事屯蹦,交給它完全是可以處理的。如果交給cell的話绳姨,還需要額外的設(shè)置登澜,以及判斷清空。從代碼量來看飘庄,還是直接設(shè)置好一點(diǎn)帖渠。