美文网首页开发swift是未来PonySwift
Swift编程(二):UIViewController之间的传值

Swift编程(二):UIViewController之间的传值

作者: zZ爱吃菜 | 来源:发表于2015-11-05 10:50 被阅读6134次

现在前面:

欢迎大家关注我的个人博客:<a href="http://blog.noasis.cn" >博客地址</a>,这里主要是我在个人开发时候遇到的坑和挖完的坑,包括 PHP CentOS 以及 Swift 等相关只是

知识背景:

  1. 在日程开放中页面传值是最为常见的,(具体动画特效见我的<a href="https://github.com/liu1013269528/Collection">我的GitHub</a> )
效果图效果图

学习目标:

  1. 学习闭包的使用
  2. 学习页面传值FirstViewController --> SecondViewController
  3. 通过闭包回调将值传回:SecondViewController --> FirstViewController

步骤

1. 创建两个UIViewController:

LWRootViewController.swift 和 LWSecondViewController.swift
因为是Demo,我的布局就比较随意.一般两种方式(纯代码创建和storyboard)本次使用纯代码创建

2. LWRootViewController的具体代码

1. 创建组件,初始化,代码如下:

<pre>```swift

var textFieldWithTagOne: UITextField?
var textFieldWithTagTwo: UITextField?

override func viewDidLoad() {
    super.viewDidLoad()
    self.view.backgroundColor = UIColor.whiteColor()
    
    
    //通过tag获取控件
    textFieldWithTagOne = UITextField(frame: CGRectMake(0, 0, 150,30))
    textFieldWithTagOne!.center = CGPoint(x: self.view.center.x, y: 100)
    textFieldWithTagOne!.backgroundColor = UIColor.grayColor()
    textFieldWithTagOne!.tag = 1
    textFieldWithTagOne!.addTarget(self, action: "getValue:", forControlEvents: UIControlEvents.EditingDidBegin)
    self.view.addSubview(textFieldWithTagOne!)
    
    textFieldWithTagTwo = UITextField(frame: CGRectMake(0, 0, 150,30))
    textFieldWithTagTwo!.center = CGPoint(x: self.view.center.x, y: 200)
    textFieldWithTagTwo!.backgroundColor = UIColor.grayColor()
    textFieldWithTagTwo!.tag = 2
    textFieldWithTagTwo!.addTarget(self, action: "getValue:", forControlEvents: UIControlEvents.EditingDidBegin)
    self.view.addSubview(textFieldWithTagTwo!)
    
}

   - 这里涉及了代码创建界面时的为何放在ViewDidLoad()中,其实这就要扯到     UIViewController的生命周期了。(我会在文章末尾出简单讲一下^__^)

#### 2. 创建两个事件:
 
<pre> ```swift
//给两个textField添加的点击事件,通过tag
 func getValue(sender: UITextField) {
        let tag = sender.tag
        let second = LWSecondViewController()
        second.initWithClosure(tag, dataList: ["我是选项1", "我是选项2", "我是选项3", "我是选项4", "我是选项5", "我是选项6"], closuer: getValueClosure)
        //页面跳转,从下往上
        self.presentViewController(second, animated: true, completion: nil)
    }
    //闭包,在SecondViewController中回调传值的事件
    func getValueClosure(tag: Int,string: String) {
        let tf = self.view.viewWithTag(tag) as! UITextField
        tf.text = string
    }
```</pre>

### 3. LWSecondViewController.swift的具体代码:

在上一节,我已经详细讲解了如何代码创建UITableView以及如何调用自定义UITableViewCell和自定义UITableCell的制作,这里我就不再一一赘述。

  #### 1. 变量声明跟初始化

<pre>```swift

private let cellId = "DemoListID"
    
    //声明一个闭包
    private var myClosure: sendValueClosure?
    private var myTag: Int?
    private var dataList = [String]()
    //下面的方法需要传入上个界面的someFunctionThatAClosure函数指针
    func initWithClosure(tag: Int,dataList: [String],closuer: sendValueClosure?) {
        //讲函数指针赋值给myClosure闭包,该闭包中覆盖了someFunctionThatTakesAClosure函数中的局部变量等引用
        self.myTag = tag
        self.dataList = dataList
        self.myClosure = closuer
    }
    
    override init(nibName nibNameOrNil: String?, bundle nibBundleOrNil: NSBundle?) {
        super.init(nibName: nibNameOrNil, bundle: nibBundleOrNil)
    }

    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
    
    
    override func viewDidLoad() {
        super.viewDidLoad()
        self.view.backgroundColor = UIColor.whiteColor()
        let tableList = UITableView(frame: CGRectMake(0, 30, self.view.frame.width, self.view.frame.height))
        let nib = UINib(nibName: "DemoListCell", bundle: nil) //Cell文件名
        tableList.registerNib(nib, forCellReuseIdentifier: cellId)
        tableList.delegate = self
        tableList.dataSource = self
        tableList.reloadData()
        self.view.addSubview(tableList)
    }
```</pre>


#### 2. 有UITableView直接要写UITableViewDelegate和UITableViewDataSource的代理
 - 今天告诉大家另外一种方式,使用扩展在LWSecondViewController的类的外面添加
 - 在UITableView的点击事件中调用闭包,将当前的Cell的值传递给回LWRootViewController


<pre>```swift

extension LWSecondViewController: UITableViewDataSource {
    func numberOfSectionsInTableView(tableView: UITableView) -> Int {
        return 1
    }
    
    func tableView(tableView: UITableView, numberOfRowsInSection section: Int) -> Int {
        return self.dataList.count
    }
    
    func tableView(tableView: UITableView, cellForRowAtIndexPath indexPath: NSIndexPath) -> UITableViewCell {
        let cell = tableView.dequeueReusableCellWithIdentifier(self.cellId, forIndexPath: indexPath) as! DemoListCell
        //cell.cellImg.image = UIImage(named: powerData[indexPath.row][2])
        cell.cellLabel.text = self.dataList[indexPath.row]
        return cell
    }
    
}

extension LWSecondViewController: UITableViewDelegate {
    func tableView(tableView: UITableView, didSelectRowAtIndexPath indexPath: NSIndexPath)
    {
        let index = indexPath.row
        let string = self.dataList[index]
        if let closure = self.myClosure {
            closure(tag: self.myTag!,string: string)
        }
        self.dismissViewControllerAnimated(true, completion: nil)
    }
}
```</pre>

ok!就是这么简单,回复让我知道 (*^__^*)


# TIPS

## 添加事件小贴士
1. 代码添加事件
 <pre>```swift
    item.addTarget(self, action: "click:", forControlEvents: UIControlEvents.TouchUpInside)
```</pre>
2. action: "funcName:" 和 action: Selector("funcName")的区别
 - 第一种可以讲当前的item当做对象传递给事件:funcName(sender: AnyObject)
 - 第二种不可以:funcName()


## 这里跟大家讲一下生命周期:

1. viewDidLoad 页面已经载入完毕,可以显示了
2. viewWillAppear 页面界面将要显示
3. viewDidAppear 页面界面已经展示出来
4. viewWillDisappear 界面即将消失(可以做一下小动画在界面消失前)
5. viewDidDisAppear 界面消失不见,但还是在内存中
6. viewDidUnload 界面已经从内存中释放出来

相关文章

  • Swift编程(二):UIViewController之间的传值

    现在前面: 欢迎大家关注我的个人博客: 博客地址 ,这里主要是我在个人开发时候遇到的坑和挖完的坑,包括 PHP C...

  • OC与swift的数据传输

    简介 该项目主要介绍了oc与swift之间、swift内部几种常见的传值方式(属性传值、代码块传值、代理传值、通知...

  • swift传值

    本文将介绍swift中的传值方式:属性传值、代理传值、闭包传值、通知传值本文将在两个VC之间进行传值:HomeVC...

  • Swift 和OC的blcok属性不一致问题

    组件之间进行属性传值的时候,如果通过CTMediator来进行block传值,swift和OC间就会出现block...

  • acitvity之间传值与返回传值

    title: acitvity之间传值与返回传值 精明的编程者听说了此道,并遵循它;平庸的编程者听说了此道,并寻...

  • Swift:界面之间传值

    1、通知传值 首先我们来看看通知传值,通知可实现任意界面之间的数据传递,但必须满足一个条件,就是保证在发送通知的时...

  • swift 界面之间传值

    1、闭包 从A页面进入B页面,B页面有操作需要更新或传值到A页面 首先你需要在B页面定义一个闭包类型的变量 在A页...

  • iOS开发-通知传值

    通知的使用 通知传值 通知监听 1.通知传值 (UIViewController) 我碰到的情况:在scrollv...

  • Vue3.2组件间通信

    组件之间的传值及通信 一、第一种场景 两层组件之间传值和监听值改变 二、第二种场景 多层组件间传值和监听值改变父组...

  • 11.4页面传值

    //一、AppDelegate.swift // //AppDelegate.swift //页面传值 // //...

网友评论

  • 英俊神武:写个demo更直观
  • 当时就吓死我了:必须过来点个赞。。。
  • 青菜白玉堂: @IBOutlet weak var ceshiBtn: UIButton!

    var fanhuiString = "ww"

    let ceshi2Btn = UIButton()

    override func viewDidLoad() {
    super.viewDidLoad(
    ceshiBtn.addTarget(self, action: #selector(dianji), for: .touchUpInside)
    view.addSubview(ceshi2Btn)

    }

    //闭包,回调传值的事件
    func getValueClosure(tag: Int,string: String) {


    print("ceshi2Btn:\(self.ceshiBtn) fanhuiString:\(fanhuiString) ")
    // self.ceshiBtn.titleLabel?.text = "\(tag)\(string)"
    fanhuiString = "\(tag)\(string)"
    print("\(fanhuiString)")


    print("\(tag)\(string)")

    }

    func dianji() -> () {
    print("ces\(fanhuiString)")
    }
    zZ爱吃菜:@青菜白玉堂 看一下操作顺序,我在你的代码中看不到点击按钮事件以及回调事件的顺序,如果点击事件再回调事件之前,那么fanhuiString 值就是初始值。
    青菜白玉堂:fanhuiString在getValueClosure打印是正确的返回值,在当前按钮点击事件的打印中还是原来的值,无法把最新的返回值拿出来,求指导
  • 青菜白玉堂:你好,我用了你上面写的两个控制器之间回调,后面触发返回前一页面,但是在回调方法里不能给控件更新返回的值,我声明一个字符串用来保存返回值,但是在这个回调方法里打印是更新为返回的值了,但是在外面其它地方打印都是原来的值,这是什么问题啊?

本文标题:Swift编程(二):UIViewController之间的传值

本文链接:https://www.haomeiwen.com/subject/boxchttx.html