我们经常需要填写表单,键盘的行为存在各种差异。

在 H5 项目中,我们会经常遇到页面中存在单个甚至多个 input/textarea 输入框与底部固定元素的布局情况。在 input/textarea 输入框获取焦点时,会自动触发键盘弹起,而键盘弹出在 ios 与 android 的 webview 中表现并非一致,同时当我们主动触发键盘收起时也同样存在差异化。而无论如何,我们希望功能流畅的同时,尽量保持用户体验的一致性,因此有了下面一系列兼容性问题的研究。

01

键盘弹出的不同表现

  • IOS:IOS 的键盘处在窗口的最上层,当键盘弹起时,webview 的高度 height 并没有改变,只是 scrollTop 发生变化,页面可以滚动。且页面可以滚动的最大限度为弹出的键盘的高度,而只有键盘弹出时页面恰好也滚动到最底部时,scrollTop 的变化值为键盘的高度,其他情况下则无法获取。这就导致在 IOS 情况下难以获取键盘的真实高度。

  • Android: webview 中留出空间,该空间小于等于的键盘空间,变化的高度差会随着布局而不同,有的认为 键盘高度 + 页面高度 = 原页面高度; 是错误的误导,只有在某种很巧合的布局情况下才可套用此公式。

02

键盘收起的不同表现

  • IOS:触发键盘上的按钮收起键盘或者输入框以外的页面区域时,输入框会失去焦点,因此会触发输入框的 blur 事件。

  • Android: 触发键盘上的按钮收起键盘时,输入框并不会失去焦点,因此不会触发页面的 blur 事件;触发输入框以外的区域时,输入框会失去焦点,触发输入框的 blur 事件。

03

监听键盘的弹出与收起

在 h5 中目前没有接口可以直接监听键盘事件,但我们可以通过分析键盘弹出、收起的触发过程及表现形式,来判断键盘是弹出还是收起的状态。

  • 键盘弹出:输入框获取焦点时会自动触发键盘的弹起动作,因此,我们可以监听输入框的 focus 事件,在里面实现键盘弹出后所需的页面逻辑。这在 ios 及 android 中表现一致。

  • 键盘收起:从第 2 部分可知,触发键盘收起的不同形式会存在差异化表现,当触发其他页面区域收起键盘时,我们可以监听输入框的 blur 事件,在里面实现键盘收起后所需的页面逻辑。而在通过键盘按钮收起键盘时在 ios 与 android 端存在差异化表现,下面具体分析:

  • IOS:触发了输入框 blur 事件,仍然通过该办法监听。

  • Android:没有触发输入框的 blur 事件。但通过第 1、2 部分我们可以知道,在 android 中,键盘的状态切换(弹出、收起)不仅和输入框关联,同时还会影响到 webview 高度的变化,那我们不妨通过监听 webview height 的变化来判断键盘是否收起。

下面举例说明,其中页面中含有一个输入框:

class="txd">

   Welcome to TXD!  

class="input">

    id="input" type="tel" />

ios & android 键盘弹出:

const $input = document.getElementById('input');

$input.addEventListener('focus', () => {

   // 处理键盘弹出后所需的页面逻辑

}, false);

ios 键盘收起:

const $input = document.getElementById('input');

$input.addEventListener('blur', () => {

   // 处理键盘收起后所需的页面逻辑

}, false);

android 键盘弹出与收起:

/*键盘弹起后页面高度变小*/

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);

04

使用场景

接下来根据上面的讨论说明几种常见场景:

(1)头部及中间输入框处于正常的文档流,底部元素 fixed

ios 键盘遮挡在页面上,页面高度始终不变,页面可以滚动,底部元素被遮挡;

android 页面高度减少,页面不可滚动,fixed 元素的 bottom 属性的基线为键盘;

(2)头部及输入框处于正常文档流,且所占可视区域变大,底部元素 fixed

ios 的 height 没有发生变化,页面可以滚动;

android 页面高度变小,但为了使正常文档流的元素可以正常显示,页面可以上下滚动,fixed 元素的 bottom 属性的基线为键盘;

(3)头部处于正常文档流,输入框脱离正常文档流 fixed bottom 定位
(上面为了便于观察,将输入框包裹元素的背景设置了透明度)

ios 的 height 没有发生变化,且始终保证输入框处于可视区域中;

android 页面高度变小,页面不可滚动,fixed 输入框 bottom 属性的基线为键盘;

05

小结

  1. 在 ios 中,无论何种布局,为了使输入框展示在可视区域中,键盘弹出时,页面会向上滚动,该过程与 Element.scrollIntoViewIfNeeded() 方法(将不在浏览器窗口的可见区域内的元素滚动到浏览器窗口的可见区域)产生的效果一致;且高度始终不变,页面可滚动。

  2. 在 android 中,键盘唤起后,页面可滚动与否由其处在正常文档流中的元素决定:如果正常文档流中的元素可全量展示,页面不可滚动,否则页面支持滚动;

  3. 在 android 中,键盘唤起后,fixed 元素的基准会发生变化:根据 bottom 定位的元素,其基线变为键盘上部;根据 top 定位的元素,仍然根据页面顶部,因此为照顾正常文档流及 fixed 元素的用户体验,有的元素可根据顶部定位,有的可以根据底部定位。

在了解清楚 h5 中键盘的弹出收起的性质后,在处理兼容性问题会容易很多。同时也可使用Element.scrollIntoViewIfNeeded() 方法辅助解决问题(比如在切换不同的输入法时,可能导致有用信息被遮挡的情况)优化体验。

android 如何判断h5页面是否加载完成_H5 键盘兼容性小结相关推荐

  1. h5页面文件加载不出来被浏览器当成广告屏蔽

    问题:h5.shumensy.com/ad的js未加载 定位分析:上周重构了相关网页的目录结构 问题解决:手机浏览器对url请求设置关键字ad拦截,该网页的js目录结构为:ad/adindex,所以被 ...

  2. H5页面初次加载swiper导航切换按钮可以点击,刷新后就失效不可点击问题@令狐张豪

    原因:加载顺序的问题,数据还没有完全加载的时候就已经渲染swiper了 解决方案:(在swiper加上这两行代码) observer:true,//修改swiper自己或子元素时,自动初始化swipe ...

  3. android accessibility 模拟键盘事件_H5 键盘兼容性小结

    在 H5 项目中,我们会经常遇到页面中存在单个甚至多个 input/textarea 输入框与底部固定元素的布局情况.在 input/textarea 输入框获取焦点时,会自动触发键盘弹起,而键盘弹出 ...

  4. h5如何动态获取键盘高度_H5 键盘兼容性小结

    在 H5 项目中,我们会经常遇到页面中存在单个甚至多个 input/textarea 输入框与底部固定元素的布局情况.在 input/textarea 输入框获取焦点时,会自动触发键盘弹起,而键盘弹出 ...

  5. h5底部输入框被键盘遮挡_H5 键盘兼容性小结

    " 我们经常需要填写表单,键盘的行为存在各种差异." 在 H5 项目中,我们会经常遇到页面中存在单个甚至多个 input/textarea 输入框与底部固定元素的布局情况.在 in ...

  6. JS判断访问设备(userAgent)加载不同页面 JS判断客户端操作系统类型(platform)

    //平台.设备和操作系统var system ={win : false,mac : false,xll : false};//检测平台var p = navigator.platform;syste ...

  7. android 屏幕旋转不重新加载,Android webview旋转屏幕导致页面重新加载问题解决办法...

    Android webview旋转屏幕导致页面重新加载问题解决办法 1. 在create时候加个状态判断 protected void onCreate(Bundle savedInstanceSta ...

  8. 如何判断C# webbrower页面是否加载完毕方法

    如何判断C# webbrower页面是否加载完毕方法 ? | 浏览:116 | 更新:2014-08-21 14:48 webbrower里面,每次打开一个页面,里面有多个iframe,那么navig ...

  9. html 判断页面加载完成,Javascript判断页面是否加载完成

    很多时候我们在使用document.getElementById的时候直接在script标签中获取对象,然后使用,此时程序会出现该对象为undefined. var dom=document.getE ...

最新文章

  1. 在OpenCV中图像边界扩展 copyMakeBorder 的实现
  2. ubuntu用不了root用户:~$ su - root Password: su: Authentication failure怎么办?
  3. Metasploit从文件中读取目标地址
  4. ARM中 __IO的作用解析
  5. HDU 4616 Game 树形DP
  6. P2633 Count on a tree
  7. el-table列宽设置百分比无效;el-table使用min-width设置百分比;el-table百分比设置无效;
  8. Python 数据分析三剑客之 Matplotlib(十):3D 图的绘制
  9. vue全局引入openlayers_vue项目中openlayers绘制行政区划
  10. 实验4.2 实现客户机(CLIENT)类
  11. Python 爬虫 —— BeautifulSoup
  12. 【图像分割】基于matlab随机游走算法图像分割【含Matlab源码 149期】
  13. NHibernate 的 ID 标识选择器
  14. idea设置修改字体大小与样式【亲测好用】
  15. 【视频编码】1080P、720P、4CIF、CIF所需要的理论带宽
  16. 一些网站后台模板源码分析
  17. Java语言HttpClient使用代理IP
  18. win10开启移动热点,手机无法获取ip地址
  19. 力扣-两数之和 (梦开始的地方)
  20. 10.数据中台 --- 数据中台运营机制

热门文章

  1. 【LeetCode-面试算法经典-Java实现】【054-Spiral Matrix(螺旋矩阵)】
  2. 《信息安全系统设计基础》实验四 外设驱动程序设计
  3. NOIP2008提高组(前三题) -SilverN
  4. XAMPP 1.8.2-2 Apache Web Server won't start, always stops immediately
  5. 列表导航栏实例(04)——精美模板赏析
  6. __format__
  7. python实现计算器
  8. npm ERR! { Error: EPERM: operation not permitted, mkdir 'C:\Program Files\nodejs\node_cache\_locks'
  9. 'weblogic.kernel.Default (self-tuning) 问题weblogic层面解决办法
  10. (Mysql)连接问题之1130