在 H5 项目中,咱们会常常遇到页面中存在单个甚至多个 input/textarea 输入框与底部固定元素的布局状况。在 input/textarea 输入框获取焦点时,会自动触发键盘弹起,而键盘弹出在 ios 与 android 的 webview 中表现并不是一致,同时当咱们主动触发键盘收起时也一样存在差别化。而不管如何,咱们但愿功能流畅的同时,尽可能保持用户体验的一致性,所以有了下面一系列兼容性问题的研究。android
键盘高度 + 页面高度 = 原页面高度;
是错误的误导,只有在某种很巧合的布局状况下才可套用此公式。在 h5 中目前没有接口能够直接监听键盘事件,但咱们能够经过分析键盘弹出、收起的触发过程及表现形式,来判断键盘是弹出仍是收起的状态。ios
下面举例说明,其中页面中含有一个输入框:web
<div class="txd"> Welcome to TXD! </div> <div class="input"> <input id="input" type="tel" /> </div> 复制代码
ios & android 键盘弹出:浏览器
const $input = document.getElementById('input'); $input.addEventListener('focus', () => { // 处理键盘弹出后所需的页面逻辑 }, false); 复制代码
ios 键盘收起:bash
const $input = document.getElementById('input'); $input.addEventListener('blur', () => { // 处理键盘收起后所需的页面逻辑 }, false); 复制代码
android 键盘弹出与收起:iphone
/*键盘弹起后页面高度变小*/
const originHeight = document.documentElement.clientHeight || document.body.clientHeight;
window.addEventListener('resize', () => { const resizeHeight = document.documentElement.clientHeight || document.body.clientHeight; if (resizeHeight < originHeight) { // 键盘弹起所后所需的页面逻辑 } else { // 键盘弹起所后所需的页面逻辑 } }, false); 复制代码
在实践中经过判断 userAgent 来决定使用哪一种方法:布局
const ua = window.navigator.userAgent.toLocaleLowerCase();
const isIOS = /iphone|ipad|ipod/.test(ua);
const isAndroid = /android/.test(ua);
复制代码
接下来根据上面的讨论说明几种常见场景:post
(1)头部及中间输入框处于正常的文档流,底部元素 fixed优化
ios 键盘遮挡在页面上,页面高度始终不变,页面能够滚动,底部元素被遮挡;ui
android 页面高度减小,页面不可滚动,fixed 元素的 bottom 属性的基线为键盘;
(2)头部及输入框处于正常文档流,且所占可视区域变大,底部元素 fixed
ios 的 height 没有发生变化,页面能够滚动;
android 页面高度变小,但为了使正常文档流的元素能够正常显示,页面能够上下滚动,fixed 元素的 bottom 属性的基线为键盘;
(3)头部处于正常文档流,输入框脱离正常文档流 fixed bottom 定位
ios 的 height 没有发生变化,且始终保证输入框处于可视区域中;
android 页面高度变小,页面不可滚动,fixed 输入框 bottom 属性的基线为键盘;
在了解清楚 h5 中键盘的弹出收起的性质后,在处理兼容性问题会容易不少。同时也可以使用Element.scrollIntoViewIfNeeded() 方法辅助解决问题(好比在切换不一样的输入法时,可能致使有用信息被遮挡的状况)优化体验。
做者:阿里巴巴TXD连接:https://juejin.im/post/5c6d1c8b6fb9a049de6df441来源:掘金著做权归做者全部。商业转载请联系做者得到受权,非商业转载请注明出处。