iOS之簡(jiǎn)單瀑布流的實(shí)現(xiàn)

前言

超簡(jiǎn)單的瀑布流實(shí)現(xiàn)填具,這里說(shuō)一下筆者的思路,詳細(xì)代碼在這里

效果演示

實(shí)現(xiàn)思路

collectionView能實(shí)現(xiàn)各中吊炸天的布局陶耍,其精髓就在于UICollectionViewLayout犯戏,因此我們要自定義一個(gè)layout來(lái)繼承系統(tǒng)的UICollectionViewLayout,所有工作都在這個(gè)類中進(jìn)行

1.定義所需屬性

瀑布流的思路就是娘汞,從上往下歹茶,那一列最短,就把下一個(gè)item放在哪一列你弦,因此我們需要定義一個(gè)字典來(lái)記錄每一列的最大y值

每一個(gè)item都有一個(gè)attributes惊豺,因此定義一個(gè)數(shù)組來(lái)保存每一個(gè)item的attributes

我們還必須知道有多少列以及列間距、行間距禽作、section到collectionView的邊距


//總列數(shù)
@property (nonatomic, assign) NSInteger columnCount;
//列間距
@property (nonatomic, assign) NSInteger columnSpacing;
//行間距
@property (nonatomic, assign) NSInteger rowSpacing;
//section到collectionView的邊距
@property (nonatomic, assign) UIEdgeInsets sectionInset;
//保存每一列最大y值的數(shù)組
@property (nonatomic, strong) NSMutableDictionary *maxYDic;
//保存每一個(gè)item的attributes的數(shù)組
@property (nonatomic, strong) NSMutableArray *attributesArray;

2.重寫(xiě)系統(tǒng)方法

我們一共需要重寫(xiě)4個(gè)方法
a.- (void)prepareLayout
b.- (CGSize)collectionViewContentSize
c.- (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath
d.- (NSArray *)layoutAttributesForElementsInRect:(CGRect)rect

- (void)prepareLayout 方法

布局前的一些準(zhǔn)備工作都在這里進(jìn)行
初始化字典尸昧,有幾列就有幾個(gè)鍵值對(duì),key為第幾列旷偿,value為列的最大y值烹俗,初始值為上內(nèi)邊距

for (int i = 0; i < self.columnCount; i++) {
    self.maxYDic[@(i)] = @(self.sectionInset.top);
}

創(chuàng)建每個(gè)item的attributes,并存入數(shù)組

//根據(jù)collectionView獲取總共有多少個(gè)item
NSInteger itemCount = [self.collectionView numberOfItemsInSection:0];

//為每一個(gè)item創(chuàng)建一個(gè)attributes并存入數(shù)組
for (int i = 0; i < itemCount; i++) {
    UICollectionViewLayoutAttributes *attributes = [self layoutAttributesForItemAtIndexPath:[NSIndexPath indexPathForItem:i inSection:0]];
    [self.attributesArray addObject:attributes];
}
- (CGSize)collectionViewContentSize 方法

用來(lái)計(jì)算collectionView的contentSize

一般瀑布流只能垂直滾動(dòng)萍程,不能水平滾動(dòng)幢妄,因此contentSize.width = 0,我們只需要計(jì)算contentSize.height即可

從字典中找出最長(zhǎng)列的最大y值茫负,再加上下面的內(nèi)邊距蕉鸳,即為contentSize.height

- (CGSize)collectionViewContentSize {
    //假設(shè)第0列是最長(zhǎng)的那列
    __block NSNumber *maxIndex = @0;
    //遍歷字典,找出最長(zhǎng)的那一列
    [self.maxYDic enumerateKeysAndObjectsUsingBlock:^(NSNumber *key, NSNumber *obj, BOOL *stop) {
        //如果maxColumn列的最大y值小于obj忍法,則讓maxColumn等于obj所屬的列
        if ([self.maxYDic[maxIndex] floatValue] < obj.floatValue) {
            maxIndex = key;
        }
    }];
    //collectionView的contentSize.height就等于最長(zhǎng)列的最大y值+下內(nèi)邊距
    return CGSizeMake(0, [self.maxYDic[maxIndex] floatValue] + self.sectionInset.bottom);
}
- (UICollectionViewLayoutAttributes *)layoutAttributesForItemAtIndexPath:(NSIndexPath *)indexPath 方法

該方法則用來(lái)設(shè)置每個(gè)item的attributes潮尝,在這里,我們只需要簡(jiǎn)單的設(shè)置每個(gè)item的attributes.frame即可

首先我們必須得知collectionView的尺寸饿序,然后我們根據(jù)collectionView的寬度勉失,以及列數(shù)、各個(gè)間距來(lái)計(jì)算每個(gè)item的寬度

item的寬度 = (collectionView的寬度 - 內(nèi)邊距及列邊距) / 列數(shù)


CGFloat collectionViewWidth = self.collectionView.frame.size.width;

//self.sectionInset.left:左邊距    self.sectionInset.right:右邊距
//(self.columnCount - 1) * columnSpacing:一行中所有的列邊距
CGFloat itemWidth = (collectionViewWidth - self.sectionInset.left - self.sectionInset.right - (self.columnCount - 1) * self.columnSpacing) / self.columnCount;

接下來(lái)計(jì)算item的坐標(biāo)嗤堰,要想計(jì)算坐標(biāo)戴质,那就必須知道最短的那一列,先遍歷字典踢匣,找出最短列是哪一列(minColumn)以及其最大y值

item的y值就等于最短列的最大y值再加上行間距告匠,x值就等于左邊距 + (item寬度 + 列間距) * minColumn


//找出最短的那一列
__block NSNumber *minIndex = @0;
[self.maxYDic enumerateKeysAndObjectsUsingBlock:^(NSNumber *key, NSNumber *obj, BOOL *stop) {
    if ([self.maxYDic[minIndex] floatValue] > obj.floatValue) {
        minIndex = key;
    }
}];

//根據(jù)最短列的列數(shù)計(jì)算item的x值
CGFloat itemX = self.sectionInset.left + (self.columnSpacing + itemWidth) * minIndex.integerValue;

//item的y值 = 最短列的最大y值 + 行間距
CGFloat itemY = [self.maxYDic[minIndex] floatValue] + self.rowSpacing;

接下來(lái)便是item的高度,我們應(yīng)該根據(jù)圖片的原始尺寸以及計(jì)算出來(lái)的寬度离唬,等比例縮放來(lái)計(jì)算高度后专,但是在layout類中,我們是拿不到圖片的输莺,因此我們可以定義一個(gè)block屬性戚哎,或者代理裸诽,讓外界來(lái)計(jì)算并返回給我們,我們需要將item的寬度以及indexPath傳遞給外界

@property (nonatomic, strong) CGFloat(^itemHeightBlock)(CGFloat itemHeight,NSIndexPath *indexPath);

根據(jù)返回值來(lái)設(shè)置item的高度

if (self.itemHeightBlock) itemHeight = self.itemHeightBlock(itemWidth, indexPath);

最后設(shè)置attributes的frame并更新字典

//設(shè)置attributes的frame
attributes.frame = CGRectMake(itemX, itemY, itemWidth, itemHeight);

//更新字典中的最短列的最大y值
self.maxYDic[minIndex] = @(CGRectGetMaxY(attributes.frame));
- (NSArray *)layoutAttributesForElementsInRect:(CGRect)rect 方法

該方法用來(lái)返回rect范圍內(nèi)型凳,item的attributes
直接返回attributesArray即可

- (NSArray *)layoutAttributesForElementsInRect:(CGRect)rect {
    return self.attributesArray;
}

使用

布局類寫(xiě)完了丈冬,接下來(lái)就可以直接使用了

//創(chuàng)建布局對(duì)象
XRWaterfallLayout *waterfall = [[XRWaterfallLayout alloc] init];
//設(shè)置相關(guān)屬性
waterfall.columnCount = 3;//共多少列
waterfall.columnSpacing = 10;//列間距
waterfall.rowSpacing = 10;//行間距
waterfall.sectionInset = UIEdgeInsetsMake(10, 10 , 10, 10);//內(nèi)邊距
[waterfall setItemHeightBlock:^CGFloat(CGFloat itemWidth, NSIndexPath *indexPath) {
    //根據(jù)圖片的原始尺寸,及顯示寬度甘畅,等比例縮放來(lái)計(jì)算顯示高度
    XRImage *image = self.images[indexPath.item];
    return image.imageH / image.imageW * itemWidth;
}];
collectionView.collectionViewLayout = waterfall;
......
......

具體代碼請(qǐng)到這里下載https://github.com/codingZero/XRWaterfallLayout埂蕊,覺(jué)得不錯(cuò)的,請(qǐng)獻(xiàn)上你的star

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末疏唾,一起剝皮案震驚了整個(gè)濱河市蓄氧,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌槐脏,老刑警劉巖喉童,帶你破解...
    沈念sama閱讀 218,858評(píng)論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異顿天,居然都是意外死亡堂氯,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,372評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門(mén)露氮,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)祖灰,“玉大人,你說(shuō)我怎么就攤上這事畔规【址觯” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 165,282評(píng)論 0 356
  • 文/不壞的土叔 我叫張陵叁扫,是天一觀的道長(zhǎng)三妈。 經(jīng)常有香客問(wèn)我,道長(zhǎng)莫绣,這世上最難降的妖魔是什么畴蒲? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,842評(píng)論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮对室,結(jié)果婚禮上模燥,老公的妹妹穿的比我還像新娘。我一直安慰自己掩宜,他們只是感情好蔫骂,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,857評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著牺汤,像睡著了一般辽旋。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,679評(píng)論 1 305
  • 那天补胚,我揣著相機(jī)與錄音码耐,去河邊找鬼。 笑死溶其,一個(gè)胖子當(dāng)著我的面吹牛骚腥,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播握联,決...
    沈念sama閱讀 40,406評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼桦沉,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了金闽?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,311評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤剿骨,失蹤者是張志新(化名)和其女友劉穎代芜,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體浓利,經(jīng)...
    沈念sama閱讀 45,767評(píng)論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡挤庇,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了贷掖。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片嫡秕。...
    茶點(diǎn)故事閱讀 40,090評(píng)論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖苹威,靈堂內(nèi)的尸體忽然破棺而出昆咽,到底是詐尸還是另有隱情,我是刑警寧澤牙甫,帶...
    沈念sama閱讀 35,785評(píng)論 5 346
  • 正文 年R本政府宣布掷酗,位于F島的核電站,受9級(jí)特大地震影響窟哺,放射性物質(zhì)發(fā)生泄漏泻轰。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,420評(píng)論 3 331
  • 文/蒙蒙 一且轨、第九天 我趴在偏房一處隱蔽的房頂上張望浮声。 院中可真熱鬧,春花似錦旋奢、人聲如沸泳挥。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,988評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)羡洁。三九已至,卻和暖如春爽丹,著一層夾襖步出監(jiān)牢的瞬間筑煮,已是汗流浹背辛蚊。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,101評(píng)論 1 271
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留真仲,地道東北人袋马。 一個(gè)月前我還...
    沈念sama閱讀 48,298評(píng)論 3 372
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像秸应,于是被迫代替她去往敵國(guó)和親虑凛。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,033評(píng)論 2 355

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

  • 前言 iOS里的UI控件其實(shí)沒(méi)有幾個(gè)软啼,界面基本就是圍繞那么幾個(gè)控件靈活展開(kāi)桑谍,最難的應(yīng)屬UICollectionVi...
    alenpaulkevin閱讀 31,686評(píng)論 9 175
  • 實(shí)現(xiàn)瀑布流簡(jiǎn)單,實(shí)現(xiàn)分區(qū)瀑布流祸挪,并且每個(gè)區(qū)的瀑布流的列數(shù)不一樣且有區(qū)頭和區(qū)尾锣披,就不是太容易了。我嫌麻煩不愿意自己寫(xiě)...
    ac986bb0e59a閱讀 793評(píng)論 2 3
  • 序言 前段時(shí)間開(kāi)發(fā)的時(shí)候贿条,需要在tableView上拉的時(shí)候?qū)崿F(xiàn)最底下的cell隨著滑動(dòng)從左邊移動(dòng)出來(lái)的效果(淘寶...
    sindri的小巢閱讀 10,585評(píng)論 18 38
  • 朝夕午陽(yáng)折射影雹仿, 道別心路是非論。 不恐瓊樓不懼海整以, 他生彼岸似人潮胧辽。 生非生,死非死公黑。 烏江春水漸稀薄邑商, 唯心向...
    雨墨春秋閱讀 156評(píng)論 0 1
  • 這個(gè)和當(dāng)時(shí)整個(gè)社會(huì)背景有關(guān),在佛教創(chuàng)立以前帆调,古印度當(dāng)時(shí)的人大部分信奉的是婆羅門(mén)主義奠骄,經(jīng)過(guò)了幾百年的發(fā)展,人們開(kāi)始對(duì)...
    Gatsby_anan閱讀 1,025評(píng)論 0 0