Swift实现iOS类似微信输入框跟随键盘弹出的效果
2015-10-20 11:20:10 | 来源:玩转帮会 | 投稿:佚名 | 编辑:小柯

原标题:Swift实现iOS类似微信输入框跟随键盘弹出的效果

Swift 实现 iOS 类似微信输入框跟随键盘弹出的效果 - 技术文摘 | 玩赚乐 1

作者:@TimeRanger 授权本站转载

Swift 实现 iOS 类似微信输入框跟随键盘弹出的效果 - 技术文摘 | 玩赚乐 2

封面(图文无关)

为什么要做这个效果

在聊天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.实现两个监控方法

实现键盘弹出的方法:

funckeyBoardWillShow(note:NSNotification)
{
//1
letuserInfo=note.userInfoas!NSDictionary
//2
varkeyBoardBounds=(userInfo[UIKeyboardFrameEndUserInfoKey]as!NSValue).CGRectValue()
letduration=(userInfo[UIKeyboardAnimationDurationUserInfoKey]as!NSNumber).doubleValue
//3
varkeyBoardBoundsRect=self.view.convertRect(keyBoardBounds,toView:nil)
//4
varkeyBaoardViewFrame=keyBaordView.frame
vardeltaY=keyBoardBounds.size.height
//5
letanimations:(()->Void)={
self.keyBaordView.transform=CGAffineTransformMakeTranslation(0,-deltaY)
ifduration>0{
letoptions=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

varkeyBoardBoundsRect=self.view.convertRect(keyBoardBounds,toView:nil)

由于取出的位置信息是绝对的,所以要将其转换为对应于当前view的位置,否则位置信息会出错!

//4

varkeyBaoardViewFrame=keyBaordView.frame
vardeltaY=keyBoardBounds.size.height

保存下输入框的位置信息和y坐标需要变换的量以便后面调用

//5

letanimations:(()->Void)={
self.keyBaordView.transform=CGAffineTransformMakeTranslation(0,-deltaY)
ifduration>0{
letoptions=UIViewAnimationOptions(UInt((userInfo[UIKeyboardAnimationCurveUserInfoKey]as!NSNumber).integerValue<<16))
UIView.animateWithDuration(duration,delay:0,options:options,animations:animations,completion:nil)
}else{
animations()
}
}

首先使用仿射变换CGAffineTransformMakeTranslation,使输入框的高度减少deltaY也就是跟随键盘的位置向上移动;

此处难点在这里

letoptions=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

这样就可以还原所有变换~

下面是该方法的实现:

funckeyBoardWillHide(note:NSNotification)
{
letuserInfo=note.userInfoas!NSDictionary
letduration=(userInfo[UIKeyboardAnimationDurationUserInfoKey]as!NSNumber).doubleValue
letanimations:(()->Void)={
self.keyBaordView.transform=CGAffineTransformIdentity
}
ifduration>0{
letoptions=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的效果:

Swift 实现 iOS 类似微信输入框跟随键盘弹出的效果 - 技术文摘 | 玩赚乐 3

键盘弹出demo.gif

tags:

上一篇  下一篇

相关:

这啤酒爱打复古风广告,就比着这个风格出了男装系列

世界排名第二的啤酒集团 SABMiller,前几天已经被排名第一的百威花了大约 1045 亿美元收购,这个今年年内规

今日应用:让你找回打红白机快感的独立游戏

这是一款像素风的纵向卷轴游戏,一听就超级复古呢; 游戏的作者是一位 22 岁名叫 Ojiro Fumoto 的少年,他

什么时候应该避免注释代码?

本文由玩赚乐(www.banghui.org)– 小峰原创翻译,转载请看清文末的转载要求,欢迎加入技术翻译小组!看到标

不要被塑料杯底的编号骗了,它并不能决定质量

PP 虽然是塑料中最适合用于食品行业的,但几乎是最便宜的一种塑料,这可能出乎很多人的意料,至于解答题主的

我们到底应该吃什么油?

有肠粉问我平时要吃什么油,什么油好,什么油坏。我本想偷懒找篇科普来回答,但到网上搜索后大失所望:一些

有没有“一闻就倒”的迷魂药?

最近,一则让人心惊肉跳的“通知”在朋友圈里被疯狂转发,内容是有人被上门推销的“迷魂香皂”迷晕,导致家

精神变态者VS反社会者,谁更危险?

精神变态者(Psychopath)和反社会者(Sociopath),两个表面上看似性感的心理学概念在社会和好莱坞的共谋下

并非是我选择禁欲,而是性好像在选择避开我

上个月,当我给诊所打电话,想补点 10 年来一直用的避孕药,我接通了一位医生——不是我平常那个妇科医生—

每个城市的夜晚都需要一些亮点来保持活力,包括巴黎

10 月 15 日,Google 法国在巴黎开启了一系列名为“ Les Heures Magiques”(“The Magic Hours”)的活动

用这个星球大战主题喷头洗澡的话,别洗太久就好

作为一个星球大战迷,你可以收集相关主题的书籍、笔、马克杯、影碟、游戏、沙发靠垫……但这都是八十年代的

站长推荐: