分析WyhPageControl遍蟋,談?wù)刄I組件的封裝思想

WyhPageControl

A customizable pageControl携龟,support many styles including custom image蒸健、tintcolor 头朱,etc

Github

https://github.com/XiaoWuTongZhi/WyhPageControl

CocoaPods Support

pod search WyhPageControl
pod 'WyhPageControl', '~> 1.0.0'

前段時間項目整體UI及交互大改运悲,需要準備封裝很多UI組件,借著這個機會封裝了不少實用組件项钮,這里給大家介紹一款支持自定義的pageControl吧(其實我就是來騙star的班眯,github上很少有人去封裝pageControl希停,有一個上千star的pageControl也已經(jīng)好多年沒維護了,因為太簡單了署隘,但是今天介紹的不僅僅是一個第三方組件脖苏,更多的教大家如何去封裝一個自定義UI的方式方法)。

傳統(tǒng)意義的UIPageControl想必根本滿足不了廣大開發(fā)同胞的使用了定踱,項目越來越遠離Native頁面棍潘,也著實讓我們很頭疼!

PageControl封裝起來很簡單崖媚,因為它的功能也很簡單亦歉,無非是多一些系統(tǒng)沒有的自定義小點點的功能罷了,不過封裝思想很重要畅哑。從工作至今肴楷,封裝過很多組件,其實思想都大同小異荠呐,下面我們來分析一波代碼H琛(敲黑板了奧)

封裝思想

這里僅談?wù)?code>UI組件的封裝,日后我還會出一些針對業(yè)務(wù)模塊的封裝思想泥张。

UI組件封裝都大同小異呵恢,像Native原生tableView就是一個很好的例子,支持自定義最大化的組件往往并不是暴露很多的自定義屬性媚创,而是直接用代理回調(diào)的方式渗钉,讓使用者去自定義這個組件的樣式,而不是已定的樣式钞钙。這一點是很重要的鳄橘,你要清楚你所暴露的自定義屬性永遠沒有辦法滿足所有人的需求,因此代理回調(diào)很重要芒炼。讓我們通過分析WyhPageControl來理解如何通過代理回調(diào)來自定義UI組件:

WyhPageControl代碼分析

WyhPageControl設(shè)計之初就是想自定義pageControl的小圓點樣式瘫怜、間距、圓點尺寸等本刽,那么完全可以仿照tableView的代理模式鲸湃,將小圓點作為cell,通過代理回調(diào)的方式盅安,讓用戶去自定義唤锉,當然還是要暴露一些自定義屬性的世囊,最好維持UIPageControl的屬性不變别瞭,起碼使用起來更舒服。

WyhPageControl作為主體View株憾,WyhPageControlDot作為cell蝙寨,內(nèi)部實現(xiàn)一點要分清楚哪些是支持reload的方法:

初始化方法晒衩,采用block回調(diào)自定義配置使代碼塊更聚合,block內(nèi)無需考慮循環(huán)引用墙歪,dataSourcedelegate一定要分清听系,參考UITableView

- (instancetype)initWithDataSource:(id<WyhPageControlDataSource>)dataSource
                          Delegate:(id<WyhPageControlDelegate>)delegate
                 WithConfiguration:(void (^)(WyhPageControl *))configuration {
    if (self = [self init]) {
        if(configuration) configuration(self);
        _dataSource = dataSource;
        _delegate = delegate;
        
        [self reloadUI];
    }
    return self;
}

創(chuàng)建協(xié)議代理虹菲,來高度自定義你的dot靠胜,dataSourcedelegate一定要分清并分開,結(jié)構(gòu)一定要嚴格規(guī)范毕源,為了可拓展性和便于維護:

@class WyhPageControl;

@protocol WyhPageControlDataSource <NSObject>

@optional

- (WyhPageControlDot *)pageControl:(WyhPageControl *)pageControl dotForIndex:(NSInteger)index;

@end

@protocol WyhPageControlDelegate <NSObject>

@optional
- (void)pageControl:(WyhPageControl *)pageControl didClickForIndex:(NSInteger)index;

@end

初始化一些屬性浪漠,確保所有屬性都有默認值。

- (void)initializeConfig {
    self.clipsToBounds = YES;
    
    _numberOfPages = 0;
    _currentPage = 0;
    _hidesForSinglePage = NO;
    _pageIndicatorTintColor = [UIColor lightGrayColor];
    _currentPageIndicatorTintColor = [UIColor darkGrayColor];
    _backgroundColor = [UIColor clearColor];
    _borderWidth = 0.f;
    _cornerRadius = 0.f;
    _borderColor = [UIColor darkGrayColor];
    _backgroundImage = nil;
    _showReloadActivityIndicator = YES;
    
    // const
    _dotLeftMargin = 15.f;
    _dotTopMargin = 8.f;
    _dotSpace = 8.f;
    
    // ui
    _coverView = [[UIView alloc]init];
    _coverImageView = [[UIImageView alloc]init];
    _indicator = [[UIActivityIndicatorView alloc]initWithActivityIndicatorStyle:(UIActivityIndicatorViewStyleGray)];
    [_indicator sizeToFit];
    [self addSubview:_coverView];
    [self addSubview:_coverImageView];
    [self addSubview:_indicator];
}

并重寫這些屬性的setter方法霎褐,使其重新set的時候會發(fā)生樣式的變化址愿。

#pragma mark - Setter

- (void)setNumberOfPages:(NSUInteger)numberOfPages {
    _numberOfPages = numberOfPages;
}

- (void)setHidesForSinglePage:(BOOL)hidesForSinglePage {
    _hidesForSinglePage = hidesForSinglePage;
}

- (void)setBackgroundColor:(UIColor *)backgroundColor {
    _backgroundColor = backgroundColor;
    _coverView.backgroundColor = backgroundColor;
}

- (void)setPageIndicatorTintColor:(UIColor *)pageIndicatorTintColor {
    _pageIndicatorTintColor = pageIndicatorTintColor;
}

- (void)setCurrentPageIndicatorTintColor:(UIColor *)currentPageIndicatorTintColor {
    _currentPageIndicatorTintColor = currentPageIndicatorTintColor;
}

- (void)setDotLeftMargin:(CGFloat)dotLeftMargin {
    _dotLeftMargin = dotLeftMargin;
}

- (void)setDotTopMargin:(CGFloat)dotTopMargin {
    _dotTopMargin = dotTopMargin;
}

- (void)setDotSpace:(CGFloat)dotSpace {
    _dotSpace = dotSpace;
}

- (void)setBorderWidth:(CGFloat)borderWidth {
    _borderWidth = borderWidth;
    self.layer.borderWidth = borderWidth;
}

- (void)setBorderColor:(UIColor *)borderColor {
    _borderColor = borderColor;
    self.layer.borderColor = borderColor.CGColor;
}

- (void)setBackgroundImage:(UIImage *)backgroundImage {
    _backgroundImage = backgroundImage;
    _coverImageView.image = backgroundImage;
}

- (void)setCornerRadius:(CGFloat)cornerRadius {
    _cornerRadius = cornerRadius;
    self.layer.cornerRadius = cornerRadius;
}

通過dataSource指定的樣式來創(chuàng)建dotvisibleDots是用來存放所有dot的數(shù)組冻璃,這里一定要判斷代理人是否給回調(diào)了dot响谓,如果沒有自定創(chuàng)建一個默認的dot,并通過用戶設(shè)置的間距等屬性省艳,設(shè)置dot的位置娘纷,并添加點擊手勢。(這里要注意的是跋炕,這個initDots方法一定是一個支持reload的失驶,使用者可能會根據(jù)不同情況返回不同的dot,這點必須清楚)

- (void)initDots {
    
    [self.visibleDots makeObjectsPerformSelector:@selector(removeFromSuperview)];
    self.visibleDots = [NSMutableArray new];
    
    WyhPageControlDot *lastDot ;
    
    for (int i = 0; i < _numberOfPages; i++) {
        
        WyhPageControlDot *dot = nil;
        
        if (![self.dataSource respondsToSelector:@selector(pageControl:dotForIndex:)]) {
            dot = [[WyhPageControlDot alloc]init];
            dot.unSelectTintColor = _pageIndicatorTintColor;
            dot.selectTintColor = _currentPageIndicatorTintColor;
        }else {
            dot = [self.dataSource pageControl:self dotForIndex:i];
        }
        dot.hidden = _isReloading;
        // frame
        CGFloat dotX = (!lastDot)?_dotLeftMargin:CGRectGetMaxX(lastDot.frame)+_dotSpace;
        dot.frame = CGRectMake(dotX, 0, dot.size.width, dot.size.height);
        // gesture
        UITapGestureRecognizer *tapges = [[UITapGestureRecognizer alloc]initWithTarget:self action:@selector(pageControlDotTapAction:)];
        [dot addGestureRecognizer:tapges];
        [self addSubview:dot];
        [self.visibleDots addObject:dot];
        lastDot = dot;
    }
    [self bringSubviewToFront:self.indicator];
    
    [self configDotsUI];
    [self autoConfigBounds];
    [self configAllDotsCenterY]; // must after autoConfigBounds.
}

增加與之對應(yīng)的reload方法拋給用戶枣购,同tableView一樣嬉探,這個reload方法執(zhí)行后,會重新調(diào)用所有代理回調(diào)及自定義屬性棉圈,保證更新機制涩堤,這也是整個UI自定義組件封裝最重要的環(huán)節(jié)!(WyhPageControl增加了一個轉(zhuǎn)圈圈的菊花分瘾,用戶可以定義顯示與隱藏當在reload時)

- (void)reloadData {
    
    [self reloadUI];
}

- (void)reloadUI {
    
    [self showActivity:YES];
    [self checkHiddenIfNeeded];
    [self configPageControlUI];
    [self initDots];
    [self showActivity:NO];
}

最后就是當點擊dot時胎围,回調(diào)代理方法:

- (void)pageControlDotTapAction:(UITapGestureRecognizer *)tapGes {
    WyhPageControlDot *dot = (WyhPageControlDot *)tapGes.view;
    NSInteger index = [self.visibleDots indexOfObject:dot];
    if (index == NSNotFound) {
        NSAssert(NO, @"Can't found this tap dot !");
        return;
    }

    [self moveToIndex:index];
    // call back
    if ([self.delegate respondsToSelector:@selector(pageControl:didClickForIndex:)]) {
        [self.delegate pageControl:self didClickForIndex:index];
    }
}

每一個dot有兩種狀態(tài)對應(yīng)兩種UI樣式,選中和未選中德召,目前僅支持自定義 選中/未選中 顏色白魂、背景圖片。

@interface WyhPageControlDot : UIView

@property(nonatomic, strong) UIColor *unSelectTintColor;

@property(nonatomic, strong) UIColor *selectTintColor;

@property (nonatomic, strong) UIImage *unselectImage;

@property (nonatomic, strong) UIImage *selectImage;

@property (nonatomic, assign) CGSize size; // default is (20,20)

@property (nonatomic, strong) UIColor *borderColor; //defult is nil;

@property (nonatomic, assign) CGFloat borderWidth; // default is 0.f;

@property (nonatomic, assign) CGFloat conerRadius; //default is 10.f

- (void)setSelected:(BOOL)selected;


@end

dot如果不滿足你的需求上岗,同cell一樣福荸,你也可以自定義繼承這個dot的Base類,來自定義你的圓點肴掷,這里就不舉例子了敬锐。

使用方法請大家去demo中自行查看背传,很簡單,同tableView類似台夺。

總結(jié)

通過分析這個簡單的組件径玖,希望朋友們對于UI組件封裝思想能更加理解,最后希望喜歡的朋友們到GitHub幫點個star颤介,歡迎各種好朋友梳星,一起來探討、研究滚朵,接下來我會出一些其他方面的丰泊,不只是UI層次的,簡書這個平臺挺好始绍,(但就是有時太懶)瞳购,大家共勉吧。

開啟傳送門:WyhPageControl

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末亏推,一起剝皮案震驚了整個濱河市学赛,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌吞杭,老刑警劉巖盏浇,帶你破解...
    沈念sama閱讀 211,561評論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異芽狗,居然都是意外死亡绢掰,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,218評論 3 385
  • 文/潘曉璐 我一進店門童擎,熙熙樓的掌柜王于貴愁眉苦臉地迎上來滴劲,“玉大人,你說我怎么就攤上這事顾复“嗤冢” “怎么了?”我有些...
    開封第一講書人閱讀 157,162評論 0 348
  • 文/不壞的土叔 我叫張陵芯砸,是天一觀的道長萧芙。 經(jīng)常有香客問我,道長假丧,這世上最難降的妖魔是什么双揪? 我笑而不...
    開封第一講書人閱讀 56,470評論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮包帚,結(jié)果婚禮上渔期,老公的妹妹穿的比我還像新娘。我一直安慰自己婴噩,他們只是感情好擎场,可當我...
    茶點故事閱讀 65,550評論 6 385
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著几莽,像睡著了一般迅办。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上章蚣,一...
    開封第一講書人閱讀 49,806評論 1 290
  • 那天站欺,我揣著相機與錄音,去河邊找鬼纤垂。 笑死矾策,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的峭沦。 我是一名探鬼主播贾虽,決...
    沈念sama閱讀 38,951評論 3 407
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼吼鱼!你這毒婦竟也來了蓬豁?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,712評論 0 266
  • 序言:老撾萬榮一對情侶失蹤菇肃,失蹤者是張志新(化名)和其女友劉穎地粪,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體琐谤,經(jīng)...
    沈念sama閱讀 44,166評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡蟆技,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,510評論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了斗忌。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片质礼。...
    茶點故事閱讀 38,643評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖织阳,靈堂內(nèi)的尸體忽然破棺而出几苍,到底是詐尸還是另有隱情,我是刑警寧澤陈哑,帶...
    沈念sama閱讀 34,306評論 4 330
  • 正文 年R本政府宣布妻坝,位于F島的核電站,受9級特大地震影響惊窖,放射性物質(zhì)發(fā)生泄漏刽宪。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 39,930評論 3 313
  • 文/蒙蒙 一界酒、第九天 我趴在偏房一處隱蔽的房頂上張望圣拄。 院中可真熱鬧,春花似錦毁欣、人聲如沸庇谆。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,745評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽饭耳。三九已至串述,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間寞肖,已是汗流浹背纲酗。 一陣腳步聲響...
    開封第一講書人閱讀 31,983評論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留新蟆,地道東北人觅赊。 一個月前我還...
    沈念sama閱讀 46,351評論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像琼稻,于是被迫代替她去往敵國和親吮螺。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 43,509評論 2 348

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