RxSwift 实践(二)基本常用控件(UILabel、UIButton、UITextField)

RxCocoa 是让 Cocoa APIs 更容易使用响应式编程的一个框架。RxCocoa 能够让我们方便地进行响应式网络请求、响应式的用户交互、绑定数据模型到 UI 控件等等。而且大多数的 UIKit 控件都有响应式扩展,它们都是通过 rx 属性进行使用。
demo下载

UILabel

用计时器的时间绑定到label上

    let timer = Observable<Int>.interval(0.1, scheduler: MainScheduler.instance)
    timer.map {  String(format: "%0.2d:%0.2d:%0.1d", arguments: [($0 / 600) % 600, ($0 % 600 ) / 10, $0 % 10])       }
        .bind(to: label.rx.text)
        .disposed(by: disposeBag)

UIButton

  • 按钮的点击事件
    button.rx.tap
        .subscribe(onNext:{ [weak self] in
            self?.showMessage(text: "按钮被点击啦")
        })
        .disposed(by: disposeBag)
  • 按钮的title和image进行绑定
    //绑定title
    let timer = Observable<Int>.interval(1, scheduler: MainScheduler.instance)
    timer.map { "\($0)"     }
        .bind(to: button.rx.title(for: .normal))
        .disposed(by: disposeBag)
        
    //绑定image(绑定backgroundImage也是一样的)
    let timer2 = Observable<Int>.interval(1, scheduler: MainScheduler.instance)
    timer2.map{
        let imageName = ( $0 % 2 == 0 ? "navBack1" : "navBack3" )
        return UIImage.init(named: imageName)!
    }
        .bind(to: button.rx.image())
        .disposed(by:disposeBag)
  • 按钮的选中事件.
    例子:当其中1个按钮选中的时候,将另外2个选中取消
   //将按钮放入数组中,并进行强制解包
   let buttons = [button1,button2,button3].map({$0!})
   button1.isSelected = true
   
   //创建一个可观察序列,它可以发送最后一次点击的按钮(也就是我们需要选中的按钮)
   let selectButton = Observable.from(buttons.map({button in button.rx.tap.map{return button}}))
                                .merge()

   //遍历按钮对selectedButton进行订阅,根据它是否是当前选中的按钮绑定isSelected属性
   for btn in buttons {
       selectButton.map {$0 == btn}
       .bind(to: btn.rx.isSelected)
       .disposed(by: disposeBag)
   }
  • 按钮是否可用
    将UISwitch开关事件绑定和UIButton是否可使用进行绑定
switchs.rx.isOn.bind(to: button1.rx.isEnabled)
                .disposed(by: disposeBag)

UISwitch

开关状态

    switchs.rx.isOn.asObservable()
        .subscribe(onNext:{[weak self]
            print("开关状态:\($0)")
            self.button2.isEnabled = $0
        })
        .disposed(by: disposeBag)

UITextField

  • textField的值改变
    textField1.rx.text.orEmpty.asObservable()
        .subscribe(onNext:{
            print($0)
        })
        .disposed(by: disposeBag)
  • 将textField的值绑定到其他UI控件上
    //文本框的变化序列
    let tfInput = textField1.rx.text.orEmpty.asDriver()
                            .throttle(0.5)
    
    //将内容绑定到另外一个输入框
    tfInput.drive(textField2.rx.text)
        .disposed(by: disposeBag)
    
    //将内容绑定到label
    tfInput.map({String(format: "当前输入了%ld个字", $0.count)})
        .drive(label.rx.text)
        .disposed(by: disposeBag)
    
    //将内容绑定到button 但输入超过5个才可以点击
    tfInput.map({ $0.count > 5 })
        .drive(button.rx.isEnabled)
        .disposed(by: disposeBag)
  • 同时监听两个textField
    //同事监听两个textField
    Observable.combineLatest(textField1.rx.text.orEmpty,textField2.rx.text.orEmpty){
        text1,text2 -> String in
        return String(format: "前1个值:%@, 后1个值:%@",text1,text2)
    }
        .map({$0})
        .bind(to: totalLabel.rx.text)
        .disposed(by: disposeBag)

事件监听
通过 rx.controlEvent 可以监听输入框的各种事件,且多个事件状态可以自由组合。

  • editingDidBegin:开始编辑(开始输入内容)
  • editingChanged:输入内容发生改变
  • editingDidEnd:结束编辑
  • editingDidEndOnExit:按下 return 键结束编辑
  • allEditingEvents:包含前面的所有编辑相关事件
    //监听textField的回车事件
    textField1.rx.controlEvent(.editingDidEndOnExit)
        .subscribe(onNext:{ [weak self] in
            self?.textField2.becomeFirstResponder()
        })
        .disposed(by: disposeBag)
    
    //监听textField的所有事件
    textField2.rx.controlEvent(.allEditingEvents)
        .subscribe(onNext:{
            print("正再监听所以输入事件")
        })
        .disposed(by: disposeBag)

UITextView

UITextView 还封装了如下几个委托回调方法:

  • didBeginEditing:开始编辑
  • didEndEditing:结束编辑
  • didChange:编辑内容发生改变
  • didChangeSelection:选中部分发生变化
    textView.rx.didBeginEditing
        .subscribe(onNext:{
            print("开始编辑")
        })
        .disposed(by: disposeBag)
    
    textView.rx.didEndEditing
        .subscribe(onNext:{
            print("结束编辑")
        })
        .disposed(by: disposeBag)

    textView.rx.didChange
        .subscribe(onNext:{
            print("内容变了")
        })
        .disposed(by: disposeBag)

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

推荐阅读更多精彩内容