开始用Swift开发iOS 10 - 6 创建简单的Table Based App

开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第1张图片

table view应该是iOS应用中最常用的UI element。最好的例子就是iPhone自带的一些应用,如电话,邮件,设置等。TED,Google+,Airbnb,微信等等都是很好例子。

创建一个项目

  • 项目名称为SimpleTable,模板为"Single View application"

设计UI

  • 选中Main.storyboard,从Object library中拖动Table View进入视图
  • 改变Table View的大小至整个view,修改属性Prototype Cells为1
  • 选中Table View Cell,修改StyleBasicIdentifierCell。table view cell的标准类型有 basic、right detail、left detail 和 subtitle,当然还有定制类型custom。
  • 选中Table View,设置四个spacing约束,上下左右的距离都设置为0
开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第2张图片

为UITableView添加两个协议

  • Object library中的每一UI component都是对应一个class,如 Table View就是对应UITableView。可以通过点击并悬停在UI component上查看对应的class和介绍。
  • ViewController.swift文件的UIViewController后,添加代码, UITableViewDataSource, UITableViewDelegate,表示ViewController类实现了UITableViewDataSourceUITableViewDelegate两个协议。
    出现红色感叹号,这是xcode的问题提示,点击参看问题描述:

Type 'ViewController' does not conform to protocol
'UITableViewDataSource'

开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第3张图片

问题描述为ViewController不符合协议UITableViewDataSource。通过command+点击 (最新的xcode9变成了command+option+点击)到UITableViewDataSource中看看你:

public protocol UITableViewDataSource : NSObjectProtocol {

    
    @available(iOS 2.0, *)
    public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int

    
    // Row display. Implementers should *always* try to reuse cells by setting each cell's reuseIdentifier and querying for available reusable cells with dequeueReusableCellWithIdentifier:
    // Cell gets various attributes set automatically based on table (separators) and data source (accessory views, editing controls)
    
    @available(iOS 2.0, *)
    public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell

    
    @available(iOS 2.0, *)
    optional public func numberOfSections(in tableView: UITableView) -> Int // Default is 1 if not implemented

    
    @available(iOS 2.0, *)
    optional public func tableView(_ tableView: UITableView, titleForHeaderInSection section: Int) -> String? // fixed font style. use custom view (UILabel) if you want something different

    @available(iOS 2.0, *)
    optional public func tableView(_ tableView: UITableView, titleForFooterInSection section: Int) -> String?

    
    // Editing
    
    // Individual rows can opt out of having the -editing property set for them. If not implemented, all rows are assumed to be editable.
    @available(iOS 2.0, *)
    optional public func tableView(_ tableView: UITableView, canEditRowAt indexPath: IndexPath) -> Bool

 ...
  • UITableViewDataSource协议中定义了很多方法,除了前两个方法没有optional其它都有,有的表示这个方法不一定要实现,没有的就一定要实现,把这个两个方法实现了,问题提示就会消失。这两个方法从名字和返回值类型也大概能知道做了什么:

    • public func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int 一个section有几行,也就是一个section有几个UITableViewCell, section就是一组UITableViewCell的意思,Table View可以定义多个section,默认是一个。
    • public func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) -> UITableViewCell 返回每一行的 UITableViewCell
  • ViewController.swift中定义一个变量restaurantNames,类型是数组,表示一系列餐馆的名字。

var restaurantNames = ["Cafe Deadend", "Homei", "Teakha", "Cafe Loisl", "PetiteOyster", "For Kee Restaurant", "Po's Atelier", "Bourke Street Bakery", "Haigh'sChocolate", "Palomino Espresso", "Upstate", "Traif", "Graham Avenue Meats AndDeli", "Waffle & Wolf", "Five Leaves", "Cafe Lore", "Confessional","Barrafina", "Donostia", "Royal Oak", "CASK Pub and Kitchen"]
  • 定义UITableViewDataSource的两个方法:
func tableView(_ tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        // 1
        return restaurantNames.count
    }
    
    func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) ->
        UITableViewCell {
        // 2
        let cellIdentifier = "Cell"
        let cell = tableView.dequeueReusableCell(withIdentifier: cellIdentifier,
                                                     for: indexPath)
        // 3
        cell.textLabel?.text = restaurantNames[indexPath.row]
        return cell
            
    }
  • 1 餐馆的数目就是section的行数
  • 2 "Cell"与之前定义的UITableViewCellIdentifier属性是对应的。dequeueReusableCell方法是产生一个UITableViewCell
  • 3 UITableViewCell中有可算属性textLabel,其实就是一个UILabel,由于是可选属性,调用时也用可选链式调用cell.textLabel?.text

连接 DataSource 和 Delegate

运行app,没有数据显示。尽管上面已经在代码中让ViewController继承了UITableViewDataSource, UITableViewDelegate,但storyboard并不知道。

  • 在document outline中选择Table View,使用control-dragView Controller,在弹出框中选择dataSource。同样的方法选择delegate
开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第4张图片
  • 确认连接是否成功。选中Table View,在connetion检查器中查看;或者直接在document outline中右击Table View
    开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第5张图片
  • 运行app


    开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第6张图片

添加图片到Table View

  • 从simpletable-image1.zip下载图片,解压后一起拖到asset catalog
  • ViewController.swifttableView(_:cellForRowAtIndexPath:)方法的return cell前添加代码cell.imageView?.image = UIImage(named: "restaurant")。使每个UITableViewCell的image都是一样的。

隐藏状态栏

顶部状态来和table view 数据重叠了,只要在ViewController加一段代码就可以:

override var prefersStatusBarHidden: Bool {
    return true
}

prefersStatusBarHidden是父类UIViewController中的属性,所以要加 override,表示重写了。

不同的Cell对应不同的图片

  • 从simpletable-image2.zip下载图片,解压后一起拖到asset catalog
  • 修改ViewController.swifttableView(_:cellForRowAtIndexPath:)为:
func tableView(_ tableView: UITableView, cellForRowAt indexPath: IndexPath) ->
        UITableViewCell {
        let cellIdentifier = "Cell"
        let cell = tableView.dequeueReusableCell(withIdentifier: cellIdentifier,
                                                     for: indexPath)
        // Configure the cell...
        let restaurantName = restaurantNames[indexPath.row]
        cell.textLabel?.text = restaurantName
        // 1
        let imageName = restaurantName.lowercased().replacingOccurrences(of: " ", with: "")
        if let image = UIImage(named: imageName) {
            cell.imageView?.image = image
        } else {
            cell.imageView?.image = UIImage(named: "restaurant")
        }
        
        return cell
            
    }
  • 1 把菜馆名字字符串先修改成小写,然后去空格
  • 最终结果
开始用Swift开发iOS 10 - 6 创建简单的Table Based App_第7张图片

代码

Beginning-iOS-Programming-with-Swift

说明

此文是学习appcode网站出的一本书 《Beginning iOS 10 Programming with Swift》 的一篇记录

你可能感兴趣的:(开始用Swift开发iOS 10 - 6 创建简单的Table Based App)