swift实现ios类似微信输入框跟随键盘弹出的效果
2017-06-27 19:18
1606 查看
封面(图文无关)
为什么要做这个效果
在聊天app,例如微信中,你会注意到一个效果,就是在你点击输入框时输入框会跟随键盘一起向上弹出,当你点击其他地方时,输入框又会跟随键盘一起向下收回,二者完全无缝连接,那么这是怎么实现的呢,也许你会说直接在键盘弹出的时候把输入框也向上移动不就行了?但是我使用这种方法的时候,发现效果十分不理想,会有明显的滞后现象,原因有以下几点:1.键盘弹出动画并不是匀速,键盘和输入框的时间曲线不完全一致,运动不同步
2.各种键盘的高度不一样(比如搜狗输入法就比系统自带键盘要高)
3.无法确定键盘动画的时间,会导致延迟
解决方案
使用本地通知,对键盘的状态(弹出、收回)进行监控,当键盘状态发生改变时,在相应的方法中对输入框的位置进行操作。这里应用了两种在ios编程中很重要的思想:
Key-value coding(KVC) 和
key-value observing(KVO)
1.使用
NSNotificationCenter.defaultCenter().addObserver()添加对
UIKeyboardWillShowNotification和
UIKeyboardWillHideNotification键的监控,当这些值发生改变时发送通知
NSNotificationCenter.defaultCenter().addObserver(self, selector:"keyBoardWillShow:", name:UIKeyboardWillShowNotification, object: nil) NSNotificationCenter.defaultCenter().addObserver(self, selector:"keyBoardWillHide:", name:UIKeyboardWillHideNotification, object: nil)
2.实现两个监控方法
实现键盘弹出的方法:
func keyBoardWillShow(note:NSNotification) { //1 let userInfo = note.userInfo as! NSDictionary //2 var keyBoardBounds = (userInfo[UIKeyboardFrameEndUserInfoKey] as! NSValue).CGRectValue() let duration = (userInfo[UIKeyboardAnimationDurationUserInfoKey] as! NSNumber).doubleValue //3 var keyBoardBoundsRect = self.view.convertRect(keyBoardBounds, toView:nil) //4 var keyBaoardViewFrame = keyBaordView.frame var deltaY = keyBoardBounds.size.height //5 let animations:(() -> Void) = { self.keyBaordView.transform = CGAffineTransformMakeTranslation(0,-deltaY) if duration > 0 { let options = UIViewAnimationOptions(UInt((userInfo[UIKeyboardAnimationCurveUserInfoKey] as! NSNumber).integerValue << 16)) UIView.animateWithDuration(duration, delay: 0, options:options, animations: animations, completion: nil) }else{ animations() } }
代码分析
//1
let userInfo = note.userInfo as! NSDictionary
将通知的用户信息取出,转化为字典类型,里面所存的就是我们所需的信息:键盘动画的时长、时间曲线;键盘的位置、高度信息。有了这些信息我们就可以do some magic了~
//2
通过对应的键
UIKeyboardFrameEndUserInfoKey,取出键盘位置信息
通过
UIKeyboardAnimationDurationUserInfoKey,取出动画时长信息
//3
var keyBoardBoundsRect = self.view.convertRect(keyBoardBounds, toView:nil)
由于取出的位置信息是绝对的,所以要将其转换为对应于当前view的位置,否则位置信息会出错!
//4
var keyBaoardViewFrame = keyBaordView.frame var deltaY = keyBoardBounds.size.height
保存下输入框的位置信息和y坐标需要变换的量以便后面调用
//5
let animations:(() -> Void) = { self.keyBaordView.transform = CGAffineTransformMakeTranslation(0,-deltaY) if duration > 0 { let options = UIViewAnimationOptions(UInt((userInfo[UIKeyboardAnimationCurveUserInfoKey] as! NSNumber).integerValue << 16)) UIView.animateWithDuration(duration, delay: 0, options:options, animations: animations, completion: nil) }else{ animations() } }
首先使用仿射变换
CGAffineTransformMakeTranslation,使输入框的高度减少deltaY也就是跟随键盘的位置向上移动;
此处难点在这里
let options = UIViewAnimationOptions(UInt((userInfo[UIKeyboardAnimationCurveUserInfoKey] as! NSNumber).integerValue << 16))
这里是将时间曲线信息(一个64为的无符号整型)转换为
UIViewAnimationOptions类型,要通过左移16来完成类型转换。
这个方法是在一个比较著名的解决bug的网站stackoverflow里找到的。
自我感觉这是比较坑的地方,它居然没有用来进行类型转换的方法,竟然还得要位!运!算!不过相信今后这个坑会被apple填上吧。。
然后呢就是把这些东西全部装进UIView的动画函数中,执行动画。
UIView.animateWithDuration(duration, delay: 0, options:options, animations: animations, completion: nil)
这样键盘弹出的方法就完全实现了!
接下来就是收回键盘的部分了:
这部分呢就比较简单了,收回键盘时只需要动画时长
duration和时间曲线信息
options所以只要留下他们就行了,然后再将输入框的位置还原即可,这里有一个很巧妙的办法
self.keyBaordView.transform = CGAffineTransformIdentity
这样就可以还原所有变换~
下面是该方法的实现:
func keyBoardWillHide(note:NSNotification) { let userInfo = note.userInfo as! NSDictionary let duration = (userInfo[UIKeyboardAnimationDurationUserInfoKey] as! NSNumber).doubleValue let animations:(() -> Void) = { self.keyBaordView.transform = CGAffineTransformIdentity } if duration > 0 { let options = UIViewAnimationOptions(UInt((userInfo[UIKeyboardAnimationCurveUserInfoKey] as! NSNumber).integerValue << 16)) UIView.animateWithDuration(duration, delay: 0, options:options, animations: animations, completion: nil) }else{ animations() } }
实际上这个方法不会运行,因为并没有判断是否应该收回键盘,我的解决方法是当手指点击输入框之上的任何地方就会收回键盘,这个在我的完整demo会看到。
demo源代码github
demo的效果:
键盘弹出demo.gif
相关文章推荐
- swift实现ios类似微信输入框跟随键盘弹出的效果
- swift实现ios类似微信输入框跟随键盘弹出的效果
- swift实现ios类似微信输入框跟随键盘弹出的效果
- swift实现ios类似微信输入框跟随键盘弹出的效果
- Swift 实现 iOS 类似微信输入框跟随键盘弹出的效果
- 【转】swift实现ios类似微信输入框跟随键盘弹出的效果
- 实现类似微信朋友圈,点击评论按钮,弹出键盘并且带有输入框
- iOS自定义提示弹出框实现类似UIAlertView的效果
- 实现类似微信朋友圈或者QQ空间,评论回复,九宫格布局。处理键盘弹出后定位到当前点击的被评论人处。另:滑动时候FPS在57-60之间,纵享丝滑
- iOS自定义提示弹出框实现类似UIAlertView的效果
- iOS完美实现输入框随键盘无缝滑动的效果_KLCPopKeyBoard
- iOS TextField 弹出键盘时实现view整体上移下移:防止弹出键盘遮挡输入框
- iOS 实现输入框被编辑以及取消编辑跟随键盘移动
- IOS简单实现输入框随着键盘的弹出动态上升(很简单的)
- ionic在开发ios系统微信时键盘挡住输入框的解决方法(键盘弹出问题)
- 简单利用Dialog实现Ios从底部弹出的效果,合QQ空间里面的发表说说弹出拍照的效果类似
- 1. 实现微信形式的键盘与操作框的弹出收入效果
- iOS实现输入框跟随键盘自动上移的实例代码
- iOS 聊天输入框跟随键盘运动动画实现
- 实现类似QQ、微信聊天界面,标题栏固定,键盘不遮挡底部输入框