react在安卓中输入框被手机键盘遮挡问题的解决方法
前言
react 起源于 facebook 的内部项目,因为该公司对市场上所有 javascript mvc 框架,都不满意,就决定自己写一套,用来架设 instagram 的网站。做出来以后,发现这套东西很好用,就在2013年5月开源了。
本文主要介绍了关于react在安卓输入框被键盘遮挡的相关内容,分享出来动大家参考学习,下面话不多说了,来一起看看详细的介绍吧
问题概述
今天遇到了一个问题,在安卓手机上,当我要点击输入“店铺名称”时,手机软键盘弹出来刚好把输入框挡住了;挡住就算了,关键是页面还不能向上滑动,整个手机窗口被压为原来的二分之一左右;
然后
然后找了一些方案,不过不大适用,或者是有点麻烦;所以需要整合一下,
首先,我想一下我要实现的效果(2018/9/3补充:评论区有更加简单的实现方法)
想要实现的效果
如图,当手机键盘出现时,页面是可以*滚动的,而且当前聚焦的输入框往红线处靠齐,这样就刚好在剩下的窗口的垂直正中间,这样就不会出现输入框被挡住,看不到自己输入的内容了 ;
第一步,使屏幕压小时,页面内容可以滚动查看
如下图所示,黑色框代表屏幕,蓝色框代表页面大小,当屏幕被压小时,页面内容必须保持原来的高度:
实现原理,页面一进来时,我就获取窗口的高度,给最外层的div设置一个最小高度,这样就算窗口压小了,页面还能维持原来的高度,可以滚动浏览:
let initwindowheight=window.innerheight let wrapdiv=document.getelementsbyclassname('animated-router-forward-enter-done')[0] wrapdiv.style.minheight =initwindowheight+'px'
第二步,滚到红线处
由于我们不能直接知道软键盘什么时候出来,不过软键盘出来的时候窗口高度会缩小,所以我们可以通过监听窗口大小变化事件来判断软键盘是否弹出,比如浏览器窗口高度突然缩小25%以上,那么我们就认为是软键盘出来了,然后我们获取聚焦input距离页面顶部的距离,计算距离红线有多少距离,假设距离是60,那么我们就让页面向上滚动60,这时input就刚刚好到了红线处;
window.onresize=function(){ if(initwindowheight-window.innerheight>initwindowheight/4&&document.queryselectorall(':focus').length>0){ //offset是封装好的一个获取元素距离页面顶部滚动距离的方法 if(offset(document.queryselectorall(':focus')[0]).top>initwindowheight/4){ document.body.scrolltop=offset(document.queryselectorall(':focus')[0]).top-initwindowheight/4 } }else if(window.innerheight-initwindowheight<20){ document.body.scrolltop=0 } };
完整代码
因为可能有多个页面要调用,所以我把代码放到一个单独的js文件中:
function pageinputscroll() { let initwindowheight=window.innerheight settimeout(() => { let wrapdiv=document.getelementsbyclassname('animated-router-forward-enter-done')[0] //console.log(wrapdiv.style) wrapdiv.style.minheight =initwindowheight+'px' }, 500); //由于我们不能直接知道软键盘什么时候出来,不过软键盘出来的时候窗口高度会缩小,所以我们可以通过监听窗口大小变化事件来判断软键盘是否弹出 window.onresize=function(){ //如果浏览器窗口高度缩小25%以上,就认为是软键盘出来了 if(initwindowheight-window.innerheight>initwindowheight/4&&document.queryselectorall(':focus').length>0){ if(offset(document.queryselectorall(':focus')[0]).top>initwindowheight/4){ document.body.scrolltop=offset(document.queryselectorall(':focus')[0]).top-initwindowheight/4 } }else if(window.innerheight-initwindowheight<20){ document.body.scrolltop=0 } }; } function offset(element) { var offest = { top: 0, left: 0 }; var _position; getoffset(element, true); return offest; // 递归获取 offset, 可以考虑使用 getboundingclientrect function getoffset(node, init) { // 非element 终止递归 if (node.nodetype !== 1) { return; } _position = window.getcomputedstyle(node)['position']; // position=static: 继续递归父节点 if (typeof(init) === 'undefined' && _position === 'static') { getoffset(node.parentnode); return; } offest.top = node.offsettop + offest.top - node.scrolltop; offest.left = node.offsetleft + offest.left - node.scrollleft; // position = fixed: 获取值后退出递归 if (_position === 'fixed') { return; } getoffset(node.parentnode); } } export {pageinputscroll};
在react页面中引入js并调用:
import {pageinputscroll} from '../../util/pageinputscroll' ...... componentdidmount(){ pageinputscroll() }
如果只是想在安卓下使用,可以加一个判断:
if(/android/i.test(navigator.useragent)){ pageinputscroll() }
效果动图
我在pc端的谷歌浏览器模拟一下实现的效果:
备注
offset()方法是使用js实现类似jquery的offset()的一个方法,参考自:
总结:
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对的支持。