swift语言IOS8开发战记16 Protocol and Map

简介:     接着之前的内容来讲,我们在每个店铺的location后面增加一个地图定位功能,需要把label的尺寸也调节一下,调整后的DetailViewCell代码如下:import UIKit...
    接着之前的内容来讲,我们在每个店铺的location后面增加一个地图定位功能,需要把label的尺寸也调节一下,调整后的DetailViewCell代码如下:
import UIKit

class DetailTableViewCell: UITableViewCell {
    
    var keyLabel: UILabel!
    var valueLabel: UILabel!
    var mapButton: UIButton!
    var buttonWidth:CGFloat = 40
    var buttonHeight:CGFloat = 30
    var viewWidth:CGFloat = 320
    override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        keyLabel = UILabel(frame: CGRectMake(14, 7, 100, 40)
        self.contentView.addSubview(keyLabel)
        let labelFont:UIFont = UIFont(name: "Avenir-Medium",size:12)!
        valueLabel = UILabel(frame: CGRectMake(100, 4, 170, 40))
        valueLabel.font = labelFont
        valueLabel.numberOfLines = 0 //行数不限
        valueLabel.adjustsFontSizeToFitWidth = true
        self.contentView.addSubview(valueLabel)
//mapButton
        mapButton = UIButton(frame: CGRectMake(viewWidth - buttonWidth - 5, 7, buttonWidth, buttonHeight))
        mapButton.backgroundColor = UIColor.orangeColor()
        mapButton.setTitle("Map", forState: .Normal)
        mapButton.hidden = true
        self.contentView.addSubview(mapButton)
    }

    required init(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

可以看到在UIViewController中我们要在当前页面中增加一部分内容需要用self.view.addSubview,但是在UITableViewCell中想要在当前页面增加内容,需要用self.contentView.addSubview。另外我们可以看到mapButton.hidden的属性设为true,这是因为我们现在设置的是针对每一个cell的,而只有location项后面的才需要显示Map按钮,所以设为隐藏,然后在DetailViewController中初始化每一行的时候把location对应那一行的Map按钮hidden属性设为false,显示出来,代码如下:

func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let indentfier = "detailCell"
        var cell = DetailTableViewCell (style: .Default, reuseIdentifier: indentfier)
        switch indexPath.row {
        case 0:
            cell.keyLabel.text = "Name"
            cell.valueLabel.text = rest?.name
        case 1:            cell.keyLabel.text = "Type"
            cell.valueLabel.text = rest?.type
        case 2:
            cell.keyLabel.text = "Location"
            cell.valueLabel.text = rest?.location
            cell.mapButton.hidden = false
        case 3:
            cell.keyLabel.text = "Be here"
            let here = rest!.isVisit ? "Yes,I'm here" : "No"
            cell.valueLabel.text = here
        default:
            break
        }
        return cell
    }

效果如图:


我们想要在这个页面的导航中显示当前餐馆的名字,需要在DetailViewCellController中的计算属性restSingle中的set方法中增加代码:


self.navigationItem.title = rest?.name

效果如图:


我们来试一下写一个代理,在DetailTableViewCell.swift的代码现在是这样:

import UIKit

protocol DetailTableViewDelegate:NSObjectProtocol {
    func detailTableView(cell:DetailTableViewCell ) ->Void
}

class DetailTableViewCell: UITableViewCell {
    
    var keyLabel: UILabel!
    var valueLabel: UILabel!
    var mapButton: UIButton!
    var detailDelegate:DetailTableViewDelegate!
    var buttonWidth:CGFloat = 40
    var buttonHeight:CGFloat = 30
    var viewWidth:CGFloat = 320
    
    
    required init(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    override init(style: UITableViewCellStyle, reuseIdentifier: String?) {
        super.init(style: style, reuseIdentifier: reuseIdentifier)
        keyLabel = UILabel(frame: CGRectMake(14, 7, 100, 40))
        self.contentView.addSubview(keyLabel)
        let labelFont:UIFont = UIFont(name: "Avenir-Medium",size:12)!
        valueLabel = UILabel(frame: CGRectMake(100, 4, 170, 40))
        valueLabel.font = labelFont
        valueLabel.numberOfLines = 0 //行数不限
        valueLabel.adjustsFontSizeToFitWidth = true
        self.contentView.addSubview(valueLabel)
        mapButton = UIButton(frame: CGRectMake(viewWidth - buttonWidth - 5, 7, buttonWidth, buttonHeight))
        mapButton.backgroundColor = UIColor.orangeColor()
        mapButton.setTitle("Map", forState: .Normal)
        mapButton.hidden = true
        mapButton.addTarget(self, action: "clickMapBtn", forControlEvents: .TouchUpInside)
        self.contentView.addSubview(mapButton)
    }

  func  clickMapBtn(){
    self.detailDelegate.detailTableView(self)
    }
    
}

我们定义了代理DetailTableViewProtocol,我们让DetailViewController实现这个代理,然后在它的返回cell的代码方法中在return cell之前赋值

cell.detailDelegate = self

并且因为实现了代理,所以需要实现代理的方法:

func detailTableView(cell: DetailTableViewCell ) {
        let mapViewController = MapViewController()
        self.presentViewController(mapViewController, animated: true, completion: nil)
    }

我们新定义了一个mapViewController用来实现点击map按钮的事件,mapViewController代码如下:

import UIKit

class MapViewController: UIViewController {

    override func viewDidLoad() {
        super.viewDidLoad()

     self.view.backgroundColor = UIColor.orangeColor()
    }

    



}

运行点击Map按钮来试一下,结果如下:



证明我们的代理起作用了,现在把这个界面加上地图的功能,我们在MapViewController中增加一个包MapKit,代码如下:

import UIKit
import MapKit

class MapViewController: UIViewController {

    var mapView: MKMapView!
    override func viewDidLoad() {
        super.viewDidLoad()
        mapView = MKMapView(frame: self.view.bounds)
        self.view.addSubview(mapView)

     self.view.backgroundColor = UIColor.orangeColor()
    }

}
运行后的效果如下:


地图是空的,我们先不纠结这个问题,之前我们做的有个问题是在share和review的页面中背景图片是写死的,现在让上一个页面给新页面传值。两个按钮的实现方法是一样的,以share按钮为例,首先在ShareViewController中增加一个存储属性和一个计算属性,这里只有一个存储属性就能起作用了,计算属性留到后面:

var imageString: String!
    var iamge:String{
        set(newString){
            imageString = newString
        }
        get{
            return imageString
        }
        
    }

把下面我们之前写成固定值的地方改写成imageString:

imageView.image = UIImage(named: imageString)

之后回到上一个控制器也就是DetailViewController中,在share按钮调用的action方法clickShare中增加传值的代码,修改后的方法如下:

func clickShare(sender: UIButton){
        let share = ShareViewController()
        share.imageString = rest?.image
        self.presentViewController(share, animated: true, completion: nil )//临时展示一个控制器
    }

review如法炮制就好。

目录
相关文章
|
1月前
|
Go
Go 语言为什么不支持并发读写 map?
Go 语言为什么不支持并发读写 map?
|
2月前
|
Unix 调度 Swift
苹果iOS新手开发之Swift 中获取时间戳有哪些方式?
在Swift中获取时间戳有四种常见方式:1) 使用`Date`对象获取秒级或毫秒级时间戳;2) 通过`CFAbsoluteTimeGetCurrent`获取Core Foundation的秒数,需转换为Unix时间戳;3) 使用`DispatchTime.now()`获取纳秒级精度的调度时间点;4) `ProcessInfo`提供设备启动后的秒数,不表示绝对时间。不同方法适用于不同的精度和场景需求。
54 3
|
17天前
|
Go
Golang语言之映射(map)快速入门篇
这篇文章是关于Go语言中映射(map)的快速入门教程,涵盖了map的定义、创建方式、基本操作如增删改查、遍历、嵌套map的使用以及相关练习题。
22 5
|
1月前
|
存储 移动开发 Swift
使用Swift进行iOS应用开发:探索现代移动开发的魅力
【8月更文挑战第12天】使用Swift进行iOS应用开发,不仅能够享受到Swift语言带来的简洁、快速、安全的编程体验,还能够充分利用iOS平台提供的丰富资源和强大功能。然而,iOS应用开发并非易事,需要开发者具备扎实的编程基础、丰富的实践经验和不断学习的精神。希望本文能够为您的iOS应用开发之旅提供一些有益的参考和帮助。
|
1月前
|
算法 安全 Go
go 语言中 map 的相关知识
go 语言中 map 的相关知识
|
2月前
|
Swift iOS开发 Kotlin
苹果iOS新手开发之Swift中实现类似Kotlin的作用域函数
Swift可通过扩展实现类似Kotlin作用域函数效果。如自定义`let`, `run`, `with`, `apply`, `also`,增强代码可读性和简洁性。虽无直接内置支持,但利用Swift特性可达成相似功能。
48 7
|
2月前
|
调度 Swift Android开发
苹果iOS新手开发之Swift中的并发任务和消息机制
Swift的消息机制类似Android的Handler,实现任务调度有三种方式: 1. **Grand Central Dispatch (GCD)**:使用`DispatchQueue`在主线程或后台线程执行任务。 2. **OperationQueue**:提供高级接口管理`Operation`对象。 3. **RunLoop**:处理事件如输入源、计时器,类似Android的`Looper`和`Handler`。 **示例**: - GCD:在不同线程执行代码块。 - OperationQueue:创建操作并执行。 - RunLoop:用Timer添加到RunLoop中。
78 2
|
1月前
|
Java Serverless Go
Golang 开发函数计算问题之在 Golang 中避免 "concurrent map writes" 异常如何解决
Golang 开发函数计算问题之在 Golang 中避免 "concurrent map writes" 异常如何解决
|
2月前
|
安全 编译器 Swift
探索iOS开发:Swift语言的现代魔法
【7月更文挑战第11天】本文深入探讨了Swift编程语言,它如何革新iOS开发领域,以及它为开发者带来的独特优势。我们将从Swift的基础语法出发,通过实际案例分析其性能优化技巧,最后讨论Swift在跨平台开发中的潜力。文章旨在为读者提供一个全面而深入的视角,了解Swift不仅仅是一门语言,更是一种推动创新的力量。
|
2月前
|
安全 Go
Go语言map并发安全,互斥锁和读写锁谁更优?
Go并发编程中,`sync.Mutex`提供独占访问,适合读写操作均衡或写操作频繁的场景;`sync.RWMutex`允许多个读取者并行,适用于读多写少的情况。明智选择锁可提升程序性能和稳定性。示例展示了如何在操作map时使用这两种锁。
46 0