基于MVC拆分TableView的Cell动态创建模型

本文主要介绍在MVC模式下,拆分TableView的代理,实现不同数据源动态创建不同的Cell。

文件结构如下:

文件结构图示.png

文件结构解析

Controllers

这里放的是页面的Controller,主要负责逻辑加载和设置TableView的代理

Models
  • Items
    一个item对应一个Cell,对数据的单元封装
  • DataSource
    自定义的DataSource子类,用于实现TableView的UITableViewDataSource代理,并且分发item与Cell的对应关系

model主要接收数据,封装成item,并加入数组

Views
  • Cells
    自定义与item一一对应的Cell

重点解析

TableView有两个代理,分别是是UITableViewDelegate和UITableViewDataSource,这里需要重点针对UITableViewDataSource进行一些定制。

LittleHTableViewDataSource

首先我们自己定义一个协议LittleHTableViewDataSourcePotocol具体如下:

public protocol LittleHTableViewDataSourcePotocol:NSObjectProtocol,UITableViewDataSource
{
    func tableView(_ tableView:UITableView, objectForRowAt indexPath:IndexPath) -> AnyObject?
    
    func tableView(_ tableView: UITableView, cellClassForObject object: AnyObject) -> LittleHTableViewBaseCell.Type
    
    func tableView(_ tableView: UITableView, indexPathForObject object: AnyObject) -> IndexPath?
    
    func tableView(_ tableView: UITableView, cell:UITableViewCell, willAppearAtIndexPath indexPath: IndexPath) -> Void
}

然后我们声明一个自定义的DataSource类,实现LittleHTableViewDataSourcePotocol协议

public class LittleHTableViewDataSource: NSObject,LittleHTableViewDataSourcePotocol {
    public var items : NSArray
    
    public init(items : NSArray) {
        self.items = items
    }
    
    // MARK:
    // MARK: -LittleHTableViewDataSourcePotocol 方法
    public func tableView(_ tableView: UITableView, objectForRowAt indexPath: IndexPath) -> AnyObject? {
        if indexPath.row < self.items.count {
            return self.items.object(at: indexPath.row) as AnyObject?
        } else {
            return nil
        }
    }
    
    public func tableView(_ tableView: UITableView, cellClassForObject object: AnyObject) -> LittleHTableViewBaseCell.Type {
        return LittleHTableViewBaseCell.self
    }
    
    public func tableView(_ tableView: UITableView, indexPathForObject object: AnyObject) -> IndexPath? {
        return nil
    }
    
    public func tableView(_ tableView: UITableView, cell:UITableViewCell, willAppearAtIndexPath indexPath: IndexPath) -> Void {
        
    }
    
    // MARK:
    // MARK: -UITableViewDataSource方法
    
    public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return self.items.count
    }
    
    public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell {
        let object:AnyObject? = self.tableView(tableView, objectForRowAt: indexPath)
        
        var cell1:UITableViewCell
        
        if let obj1 = object {
            let cellClass:LittleHTableViewBaseCell.Type = self.tableView(tableView, cellClassForObject: obj1)
            let identifier:String = cellClass.LittleHIdentifier()
            var cell:UITableViewCell? = tableView.dequeueReusableCell(withIdentifier: identifier)
            if cell == nil {
                cell = cellClass.init(style: UITableViewCellStyle.default, reuseIdentifier: identifier)
            }
            cell1 = cell!
        } else {
            cell1 = LittleHTableViewBaseCell(style: UITableViewCellStyle.default, reuseIdentifier: nil)
        }
        
        if cell1 is LittleHTableViewBaseCell {
            let cell2 = cell1 as! LittleHTableViewBaseCell
            cell2.setObject(object)
        }
        
        return cell1
    }
    
    public func numberOfSections(in tableView: UITableView) -> Int {
        return 1
    }
    

    open func tableView(_ tableView: UITableView, sectionForSectionIndexTitle title: String, at index: Int) -> Int {
        if let view1 = tableView.tableHeaderView {
            if index == 0 {
                tableView.scrollRectToVisible(view1.bounds, animated: false)
                return -1
            }
        }
        
        //        let index2 = advance(title.startIndex, 1)
        var index2:String.Index = title.startIndex
        index2 = title.index(index2, offsetBy: 1)
        
        let letter:String = title.substring(to: index2)
        let sectionCount:Int = tableView.numberOfSections
        
        var i:Int = 0
        for _ in 1...sectionCount {
            let section :String = (tableView.dataSource?.tableView!(tableView, titleForHeaderInSection: i))!
            if section.hasPrefix(letter) {
                return i
            }
            i += 1
        }
        
        if index > sectionCount {
            return sectionCount - 1
        } else {
            return index
        }
    }
}

在我们自己定义的这个DataSource里面可以看到已经实现了UITableViewDataSource,我们通过LittleHTableViewDataSourcePotocol协议的

func tableView(_ tableView: UITableView, cellClassForObject object: AnyObject) -> LittleHTableViewBaseCell.Type

方法获取到具体Cell的classType,然后创建这个类型的Cell,并且通过LittleHTableViewDataSourcePotocol协议的

func tableView(_ tableView:UITableView, objectForRowAt indexPath:IndexPath) -> AnyObject?

方法获取Cell对应的数据,在Cell的setObject方法中进行数据赋值。

Controller关键代码

在我们使用到tableView的地方,将tableView的UITableViewDataSource代理设置为这个类的对象。如:

public var dataSource : LittleHTableViewDataSource?{
        get{
            return _dataSource
        }
        
        set{
            if newValue != _dataSource {
                _dataSource = newValue
            }
            
            if let tableView1 = self.tableView {
                tableView1.dataSource = _dataSource
                tableView1.reloadData()
            }
        }
    }

我们希望Cell的高度能在Cell内部返回,最好在item数据封装的时候就直接计算好Cell的高度,然后直接返回,实现UITableViewDelegate的下面方法将返回的高度的任务转移到Cell

open func tableView(_ tableView: UITableView, heightForRowAt indexPath: IndexPath) -> CGFloat {
        if (tableView.dataSource is LittleHTableViewDataSource) {
            let dataSource:LittleHTableViewDataSource? = tableView.dataSource as? LittleHTableViewDataSource
            if let d1 = dataSource {
                let object = d1.tableView(tableView, objectForRowAt: indexPath)
                if let obj1 = object {
                    let t1:LittleHTableViewBaseCell.Type =  d1.tableView(tableView, cellClassForObject: obj1)
                    return t1.tableView(tableView, rowHeightForObject: obj1)
                }
            }
        }
        return 44.0
    }

Cell

cell上需要实现

public func setObject (_ obj:AnyObject?) -> Void

来获取数据
实现

public class func tableView(_ tableView: UITableView, rowHeightForObject object: AnyObject) -> CGFloat

获取高度

实现

public class func LittleHIdentifier() -> String

设置reuseIdentifier

主要思路,将TableView的UITableViewDataSource的Cell构建能够根据classType去动态构建,把数据类型item和Cell一一对应。

demo地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 211,948评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,371评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,490评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,521评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,627评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,842评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,997评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,741评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,203评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,534评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,673评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,339评论 4 330
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,955评论 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,770评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,000评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,394评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,562评论 2 349

推荐阅读更多精彩内容