亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 系統 > iOS > 正文

iOS自定義UICollectionViewLayout實現瀑布流布局

2020-07-26 03:01:14
字體:
來源:轉載
供稿:網友

移動端訪問不佳,請訪問我的個人博客

最近項目中需要用到瀑布流的效果,但是用UICollectionViewFlowLayout又達不到效果,自己動手寫了一個瀑布流的layout,下面是我的心路路程
先上效果圖與demo地址:

因為是用UICollectionView來實現瀑布流的,決定繼承UICollectionViewLayout來自定義一個layout來實現一個簡單瀑布流的布局,下面是需要重寫的方法:

重寫這個屬性得出UICollectionView的ContentSize:collectionViewContentSize
重寫這個方法來得到每個item的布局:layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes?
重寫這個方法給UICollectionView所有item的布局:layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]?
重寫這個方法來實現UICollectionView前的操作:prepare()

實現思路

通過代理模式獲得到需要的列數和每一item的高度,用過列數與列之間的間隔和UICollectionView的寬度來得出每一列的寬度,item從左邊到右布局,下一列的item放到高度最小的列下面,防止每列的高度不均勻,下面貼上代碼和注釋:

import UIKit@objc protocol WCLWaterFallLayoutDelegate { //waterFall的列數 func columnOfWaterFall(_ collectionView: UICollectionView) -> Int //每個item的高度 func waterFall(_ collectionView: UICollectionView, layout waterFallLayout: WCLWaterFallLayout, heightForItemAt indexPath: IndexPath) -> CGFloat}class WCLWaterFallLayout: UICollectionViewLayout { //代理 weak var delegate: WCLWaterFallLayoutDelegate? //行間距 @IBInspectable var lineSpacing: CGFloat = 0 //列間距 @IBInspectable var columnSpacing: CGFloat = 0 //section的top @IBInspectable var sectionTop: CGFloat = 0 { willSet {  sectionInsets.top = newValue } } //section的Bottom @IBInspectable var sectionBottom: CGFloat = 0 { willSet {  sectionInsets.bottom = newValue } } //section的left @IBInspectable var sectionLeft: CGFloat = 0 { willSet {  sectionInsets.left = newValue } } //section的right @IBInspectable var sectionRight: CGFloat = 0 { willSet {  sectionInsets.right = newValue } } //section的Insets @IBInspectable var sectionInsets: UIEdgeInsets = UIEdgeInsets.zero //每行對應的高度 private var columnHeights: [Int: CGFloat]   = [Int: CGFloat]() private var attributes: [UICollectionViewLayoutAttributes] = [UICollectionViewLayoutAttributes]() //MARK: Initial Methods init(lineSpacing: CGFloat, columnSpacing: CGFloat, sectionInsets: UIEdgeInsets) { super.init() self.lineSpacing = lineSpacing self.columnSpacing = columnSpacing self.sectionInsets = sectionInsets } required init?(coder aDecoder: NSCoder) { super.init(coder: aDecoder) } //MARK: Public Methods //MARK: Override override var collectionViewContentSize: CGSize { var maxHeight: CGFloat = 0 for height in columnHeights.values {  if height > maxHeight {  maxHeight = height  } } return CGSize.init(width: collectionView?.frame.width ?? 0, height: maxHeight + sectionInsets.bottom) } override func prepare() { super.prepare() guard collectionView != nil else {  return } if let columnCount = delegate?.columnOfWaterFall(collectionView!) {  for i in 0..<columnCount {  columnHeights[i] = sectionInsets.top  } } let itemCount = collectionView!.numberOfItems(inSection: 0) attributes.removeAll() for i in 0..<itemCount {  if let att = layoutAttributesForItem(at: IndexPath.init(row: i, section: 0)) {  attributes.append(att)  } } } override func layoutAttributesForItem(at indexPath: IndexPath) -> UICollectionViewLayoutAttributes? { if let collectionView = collectionView {  //根據indexPath獲取item的attributes  let att = UICollectionViewLayoutAttributes.init(forCellWith: indexPath)  //獲取collectionView的寬度  let width = collectionView.frame.width  if let columnCount = delegate?.columnOfWaterFall(collectionView) {  guard columnCount > 0 else {   return nil  }  //item的寬度 = (collectionView的寬度 - 內邊距與列間距) / 列數  let totalWidth = (width - sectionInsets.left - sectionInsets.right - (CGFloat(columnCount) - 1) * columnSpacing)  let itemWidth = totalWidth / CGFloat(columnCount)  //獲取item的高度,由外界計算得到  let itemHeight = delegate?.waterFall(collectionView, layout: self, heightForItemAt: indexPath) ?? 0  //找出最短的那一列  var minIndex = 0  for column in columnHeights {   if column.value < columnHeights[minIndex] ?? 0 {   minIndex = column.key   }  }  //根據最短列的列數計算item的x值  let itemX = sectionInsets.left + (columnSpacing + itemWidth) * CGFloat(minIndex)  //item的y值 = 最短列的最大y值 + 行間距  let itemY = (columnHeights[minIndex] ?? 0) + lineSpacing  //設置attributes的frame  att.frame = CGRect.init(x: itemX, y: itemY, width: itemWidth, height: itemHeight)  //更新字典中的最大y值  columnHeights[minIndex] = att.frame.maxY  }  return att } return nil } override func layoutAttributesForElements(in rect: CGRect) -> [UICollectionViewLayoutAttributes]? { return attributes }}

最后附帶demo地址,大家喜歡的話可以star一下

上面是簡單的瀑布流的實現過程,希望大家能學到東西,有很多地方考慮的不足,歡迎大家交流學習,謝謝大家的閱讀。

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
亚洲香蕉在线观看| 精品国产乱码久久久久酒店| 在线日韩第一页| 国产做受69高潮| 久久久久久伊人| 亚洲色图校园春色| 黄色精品一区二区| 亚洲区免费影片| 亚洲天堂av高清| 欧美日韩国产精品一区| 人妖精品videosex性欧美| 国产精品欧美一区二区三区奶水| 亚洲毛片在线免费观看| 国产精品91免费在线| 中文日韩电影网站| 久久久精品亚洲| 国产午夜精品全部视频播放| 成人动漫网站在线观看| 日韩av免费在线| 成人免费网站在线看| 日韩免费看的电影电视剧大全| 欧美电影免费观看高清| 国产91免费看片| 热久久免费视频精品| 亚洲欧洲在线免费| 亚洲成人精品视频| www.日韩av.com| 在线观看日韩视频| 亚洲国产成人在线播放| 久久91精品国产91久久跳| 91免费国产网站| 亚洲欧洲一区二区三区在线观看| 国色天香2019中文字幕在线观看| 亚洲欧美一区二区三区情侣bbw| 国产欧美精品久久久| 国产精品aaa| 欧美三级免费观看| 久久视频精品在线| 26uuu另类亚洲欧美日本一| 久久久久久91香蕉国产| 精品美女国产在线| 欧美午夜精品久久久久久浪潮| 国产精品视频不卡| 日韩中文字幕视频在线观看| 中文字幕亚洲综合久久筱田步美| 中文字幕精品在线| 一区二区三区久久精品| 成人国产精品久久久久久亚洲| 国产精品久久久久久久av电影| 亚洲综合社区网| 亚洲一区二区三区香蕉| 精品免费在线观看| 日韩av在线免费观看| 久久久久久久97| 国产精品揄拍500视频| 欧美日韩视频在线| 久久久久久久久网站| 欧美黑人一区二区三区| 久久精品一偷一偷国产| 亚洲国产中文字幕久久网| 欧美成在线视频| 国产成人精品日本亚洲| 久久躁日日躁aaaaxxxx| 久久精品成人一区二区三区| 国产精品男女猛烈高潮激情| 亚洲国产精品一区二区久| 91手机视频在线观看| 亚洲第一男人av| 在线看国产精品| 国产精品亚洲视频在线观看| 欧美日韩国产成人高清视频| 欧美黄网免费在线观看| 亚洲xxxx3d| 日本免费一区二区三区视频观看| 亚洲美女在线观看| 欧美精品手机在线| 欧美精品免费在线| 国产欧美亚洲精品| 国产精品美女呻吟| 亚洲第一视频在线观看| 亚洲欧美激情精品一区二区| 国产日韩换脸av一区在线观看| 久久久国产91| 高清日韩电视剧大全免费播放在线观看| 欧美亚州一区二区三区| 亚洲国产精品久久久久秋霞不卡| 久久精品91久久香蕉加勒比| 欧美日韩性视频| 亚洲加勒比久久88色综合| 97视频国产在线| 91久久久在线| 亚洲国产毛片完整版| 国产成人精品电影久久久| 亚洲日韩中文字幕在线播放| 欧美日韩国内自拍| 欧美在线免费视频| 国产精品久久久久久搜索| 亚洲视频在线播放| 欧美日韩一区二区在线| 亚洲天天在线日亚洲洲精| 亚洲激情自拍图| 欧美又大又硬又粗bbbbb| 成人福利网站在线观看11| 国产成人久久久精品一区| 高清一区二区三区四区五区| 国产精品吹潮在线观看| 日韩在线视频观看| 欧美大肥婆大肥bbbbb| 美女999久久久精品视频| 九九热最新视频//这里只有精品| 欧美视频一二三| 91久久久久久| 国产中文日韩欧美| 欧美猛交ⅹxxx乱大交视频| 久久精品国产精品亚洲| 大荫蒂欧美视频另类xxxx| 午夜精品一区二区三区在线视频| 正在播放国产一区| 欧美性猛交xxxx乱大交| 国产手机视频精品| 成人xxxxx| 亚洲电影成人av99爱色| 国产日韩欧美电影在线观看| 91美女片黄在线观看游戏| 欧美在线观看一区二区三区| 91成人性视频| 日本精品一区二区三区在线| 午夜精品久久久久久久久久久久| 亚洲成色777777女色窝| 久久久国产影院| 免费97视频在线精品国自产拍| 国产精自产拍久久久久久蜜| 91精品国产高清久久久久久| 人妖精品videosex性欧美| 亚洲精品久久久久中文字幕二区| 久久中文字幕一区| www.xxxx精品| 欧美富婆性猛交| 久久久女女女女999久久| 日韩av免费看网站| 午夜精品久久久久久久久久久久久| 精品欧美aⅴ在线网站| 91精品久久久久久| 91精品成人久久| 久久久噜噜噜久久久| 亚洲成人av片| 亚洲欧美中文日韩在线v日本| 久久精品99无色码中文字幕| 亚洲国产成人久久综合一区| 中文字幕日韩欧美在线视频| 日韩大陆欧美高清视频区| 久久精品国产96久久久香蕉| 日韩欧美999| 日韩在线播放一区| 在线播放亚洲激情| 久久久久久这里只有精品| 亚洲激情视频网| 中文字幕欧美日韩在线| 日韩精品视频免费在线观看| 欧美精品免费在线| 91美女高潮出水| 日韩av一卡二卡| 欧美亚洲成人精品| 91伊人影院在线播放|