近期在做项目发现,如果在键盘显示的情况下,使用系统自带的拖拽手势拖动页面,不成功pop, controller中的键盘会被自动显示出来。
通常显示键盘,是因为textView,textFiled等控件becomeFirstResponder了。那么问题就的节点就是可以现定位到是否是在viewWillAppear方法里有手动设置FirstResponder。
然而找了一圈,是在viewWillAppear方法里并没发现有手动设置FirstResponder。并且在viewWillDisappear方法里,还做了resignFirstResponder操作。
嗯,线索断了。。。
那就重写一下textView的becomeFirstResponder方法,看看是谁调用的。
从下面的堆栈信息里可以看出, becomeFirstResponder方法是由系统自己调用的。
textView 调用了_restoreFirstResponder方法。恢复了textView 的第一响应的身份。
我们看一下_restoreFirstResponder方法做哪些事情。
- interactionAssistant (我将其解释为交互助手)
- checkEditabilityAndSetFirstResponderIfNecessary (通过交互助手检测textView 是否有编辑能力并且是否有必要将其设置为第一响应者)
- isFirstResponder (是否是第一响应者,如果是则恢复第一响应者的身份)
有restore就应该有store,那么问题来了,是什么时候进行store的呢?
写了一个简单的demo。发现在执行viewWillDisappear方法前,会先调用textView的resignFirstRespoder方法。
那就能确定store的操作一定是在这次resignFirstRespoder前被调用的。
可以这么说是在系统判断当前手势为pop手势时就会先storeFirstResponder,然后再取消了当前的页面的第一响应者。
但是,我没有找到具体的storeFirstResponder的地方。哪位同学如果知道具体store的时机,烦请在在评论中指出。