移动端兼容性问题解决方案

Chason
2021-03-19 / 0 评论 / 0 点赞 / 803 阅读 / 7,887 字
温馨提示:
本文最后更新于 2021-03-19,若内容或图片失效,请留言反馈。部分素材来自网络,若不小心影响到您的利益,请联系我们删除。
1. IOS 移动端 click 事件 300ms 的延迟响应

移动设备上的 web 网页是有 300ms 延迟的,往往会造成按钮点击延迟甚至是点击失效。这是由于区分双击事件和双击屏幕缩放的历史原因造成的,

2007 年苹果发布首款 iphone 上 IOS 系统搭载的 safari 为了将适用于 PC 端上大屏幕的网页能比较好的展示在手机端上,使用了双击缩放(double tap to zoom)的方案,比如你在手机上用浏览器打开一个 PC 上的网页,你可能在看到页面内容虽然可以撑满整个屏幕,但是字体、图片都很小看不清,此时可以快速双击屏幕上的某一部分,你就能看清该部分放大后的内容,再次双击后能回到原始状态。

双击缩放是指用手指在屏幕上快速点击两次,iOS 自带的 Safari 浏览器会将网页缩放至原始比例。

原因就出在浏览器需要如何判断快速点击上,当用户在屏幕上单击某一个元素时候,例如跳转链接<a href="#"></a>,此处浏览器会先捕获该次单击,但浏览器不能决定用户是单纯要点击链接还是要双击该部分区域进行缩放操作,所以,捕获第一次单击后,浏览器会先 Hold 一段时间 t,如果在 t 时间区间里用户未进行下一次点击,则浏览器会做单击跳转链接的处理,如果 t 时间里用户进行了第二次单击操作,则浏览器会禁止跳转,转而进行对该部分区域页面的缩放操作。那么这个时间区间 t 有多少呢?在 IOS safari 下,大概为 300 毫秒。这就是延迟的由来。造成的后果用户纯粹单击页面,页面需要过一段时间才响应,给用户慢体验感觉,对于 web 开发者来说是,页面 js 捕获 click 事件的回调函数处理,需要 300ms 后才生效,也就间接导致影响其他业务逻辑的处理。

解决方案:

  • fastclick 可以解决在手机上点击事件的 300ms 延迟
  • zepto 的 touch 模块,tap 事件也是为了解决在 click 的延迟问题

触摸事件的响应顺序为 touchstart --> touchmove --> touchend --> click,也可以通过绑定 ontouchstart 事件,加快对事件的响应,解决 300ms 延迟问题

2. 一些情况下对非可点击元素如(label,span)监听 click 事件,ios 下不会触发,css 增加 cursor:pointer 就搞定了。
3. 三星手机遮罩层下的 input、select、a 等元素可以被点击和 focus(点击穿透)

问题发现于三星手机,这个在特定需求下才会有,因此如果没有类似问题的可以不看。首先需求是浮层操作,在三星上被遮罩的元素依然可以获取 focus、click、change),有两种解决方案,

  1. 是通过层显示以后加入对应的 class 名控制,截断显示层下方可获取焦点元素的事件获取
  2. 是通过将可获取焦点元素加入的 disabled 属性,也可以利用属性加 dom 锁定的方式(disabled 的一种变换方式)
4. h5 底部输入框被键盘遮挡问题

h5 页面有个很蛋疼的问题就是,当输入框在最底部,点击软键盘后输入框会被遮挡。可采用如下方式解决

var oHeight = $(document).height(); //浏览器当前的高度

   $(window).resize(function(){

        if($(document).height() < oHeight){

        $("#footer").css("position","static");
    }else{

        $("#footer").css("position","absolute");
    }

   });

关于 Web 移动端 Fixed 布局的解决方案,这篇文章也不错

http://efe.baidu.com/blog/mobile-fixed-layout/

5. 不让 Android 手机识别邮箱
<meta content="email=no" name="format-detection" />

6. 禁止 iOS 识别长串数字为电话
<meta content="telephone=no" name="format-detection" />

7. 禁止 iOS 弹出各种操作窗口
-webkit-touch-callout: none;

8. 消除 transition 闪屏
-webkit-transform-style: preserve-3d;     /*设置内嵌的元素在 3D 空间如何呈现:保留 3D*/
-webkit-backface-visibility: hidden;      /*(设置进行转换的元素的背面在面对用户时是否可见:隐藏)*/

9. iOS 系统中文输入法输入英文时,字母之间可能会出现一个六分之一空格

可以通过正则去掉 this.value = this.value.replace(/\u2006/g, &#39;&#39;);

10. 禁止 ios 和 android 用户选中文字
-webkit-user-select:none

11. 在 ios 和 andriod 中,audio 元素和 video 元素在无法自动播放

应对方案:触屏即播

$('html').one('touchstart',function(){
    audio.play()
})

12. ios 下取消 input 在输入的时候英文首字母的默认大写
<input autocapitalize="off" autocorrect="off" />

13. android 下取消输入语音按钮
input::-webkit-input-speech-button {display: none}

14. CSS 动画页面闪白,动画卡顿
解决方法:
1.尽可能地使用合成属性transform和opacity来设计CSS3动画,不使用position的left和top来定位
2.开启硬件加速

  -webkit-transform: translate3d(0, 0, 0);
     -moz-transform: translate3d(0, 0, 0);
      -ms-transform: translate3d(0, 0, 0);
          transform: translate3d(0, 0, 0);

15. fixed 定位缺陷

ios 下 fixed 元素容易定位出错,软键盘弹出时,影响 fixed 元素定位

android 下 fixed 表现要比 iOS 更好,软键盘弹出时,不会影响 fixed 元素定位

ios4 下不支持 position:fixed

解决方案: 可用 iScroll 插件解决这个问题

16. 阻止旋转屏幕时自动调整字体大小
html,
body,
form,
fieldset,
p,
div,
h1,
h2,
h3,
h4,
h5,
h6 {
  -webkit-text-size-adjust: none;
}

17. Input 的 placeholder 会出现文本位置偏上的情况

input 的 placeholder 会出现文本位置偏上的情况:PC 端设置 line-height 等于 height 能够对齐,而移动端仍然是偏上,解决是设置 line-height:normal

18. 往返缓存问题

点击浏览器的回退,有时候不会自动执行 js,特别是在 mobilesafari 中。这与往返缓存(bfcache) 有关系。

解决方法 :window.onunload = function(){};

19. calc 的兼容性处理

CSS3 中的 calc 变量在 iOS6 浏览器中必须加-webkit-前缀,目前的 FF 浏览器已经无需-moz-前缀。
Android 浏览器目前仍然不支持 calc,所以要在之前增加一个保守尺寸:

div {
    width: 95%;
    width: -webkit-calc(100% - 50px);
    width: calc(100% - 50px);
}

20. iOS6 下伪类:hover

除了<a>之外的元素无效;在 Android 下则有效。类似

div#topFloatBar_l:hover #topFloatBar_menu { display:block; }

这样的导航显示在 iOS6 点击没有点击效果,只能通过增加点击侦听器给元素增减 class 来控制子元素。

21. 在移动端修改难看的点击的高亮效果,iOS 和安卓下都有效:
* {-webkit-tap-highlight-color:rgba(0,0,0,0);}

不过这个方法在现在的安卓浏览器下,只能去掉那个橙色的背景色,点击产生的高亮边框还是没有去掉,有待解决!

一个 CSS3 的属性,加上后,所关联的元素的事件监听都会失效,等于让元素变得“看得见,点不着”。IE 到 11 才开始支持,其他浏览器的当前版本基本都支持。详细介绍见这里:https://developer.mozilla.org/zh-CN/docs/Web/CSS/pointer-events

pointer-events: none;

22. Zepto 点透的解决方案

zepto 的 tap 是通过兼听绑定在 document 上的 touch 事件来完成 tap 事件的模拟的,及 tap 事件是冒泡到 document 上触发的,在点击完成时的 tap 事件(touchstart\touchend)需要冒泡到 document 上才会触发,而在冒泡到 document 之前,用户手的接触屏幕(touchstart)和离开屏幕(touchend)是会触发 click 事件的,因为 click 事件有延迟触发(这就是为什么移动端不用 click 而用 tap 的原因)(大概是 300ms,为了实现 safari 的双击事件的设计),所以在执行完 tap 事件之后,弹出来的选择组件马上就隐藏了,此时 click 事件还在延迟的 300ms 之中,当 300ms 到来的时候,click 到的其实不是完成而是隐藏之后的下方的元素,如果正下方的元素绑定的有 click 事件此时便会触发,如果没有绑定 click 事件的话就当没 click,但是正下方的是 input 输入框(或者 select 选择框或者单选复选框),点击默认聚焦而弹出输入键盘,也就出现了上面的点透现象。

引入 fastclick.js,在页面中加入如下 js 代码

 window.addEventListener( "load", function() {
     FastClick.attach( document.body );
 }, false );

或者有 zepto 或者 jQuery 的 js 里面加上

 $(function() {
     FastClick.attach(document.body);
 });

当然 require 的话就这样:

 var FastClick = require('fastclick');
 FastClick.attach(document.body, options);

方案二:用 touchend 代替 tap 事件并阻止掉 touchend 的默认行为 preventDefault()

 $("#cbFinish").on("touchend", function (event) {
     //很多处理比如隐藏什么的
     event.preventDefault();
 });

方案三:延迟一定的时间(300ms+)来处理事件

 $("#cbFinish").on("tap", function (event) {
     setTimeout(function(){
     //很多处理比如隐藏什么的
     },320);
 });

23. user-select:none;造成 iPhone5 上输入框的光标不显示,而光标不显示造成无法连续输入。

1、外观

A、页面高度渲染错误

在各移动端浏览器中经常会出现这种页面高度 100%的渲染错误,页面低端和系统自带的导航条重合了,高度的不正确我们需要重置修正它,通过 javascript 代码重置掉:

document.documentElement.style.height = window.innerHeight + 'px';

B、叠加区高亮

    在部分 android 机型中点击页面某一块区域可能会出现如图所示的黄色框秒闪,这是部分机型系统自身的默认定制样式,给该元素一个 CSS 样式重置掉:

-webkit-tap-highlight-color:rgba(0,0,0,0);

2、行为

A、事件无法被触发

    在部分 android 机型的微信环境中会出现事件无法触发、表单无法输入的情况,我们针对需要输入或者触发事件的元素设置样式:-webkit-transform: translate3d(0,0,0) ,不过新版本的微信已经直接修复了该问题。

B、:active  效果不兼容

    在 android 4.0 版本以下 CSS :active 伪状态效果无法兼容,我们给该元素的 touch 系列的事件(touchstart/touchend/touchmove)绑定一个空匿名方法:

var element=document.getElementsById(”btnShare”);
element.addEventListener(‘touchstart’,function(){},false);

3、应用

A、浏览器崩溃

var act = function(){
	window.removeEventListener('devicemotion',act);
};
window.addEventListener('devicemotion',act,false);

解绑函数写在了事件处理中导致小米手机中的微信崩溃,那么我们不要将解绑时间写在事件处理中即可。

B、预加载、自动播放无效

如上表所示,经过简单的测试发现预加载、自动播放的有效性受操作系统、浏览器(webview)、版本等的影响,苹果官方规定必须由用户手动触发才会载入音频,那么我们捕捉一次用户输入后,让音频加载实现预加载:

//play and pause it once
 document.addEventListener('touchstart', function () {
 document.getElementsByTagName('audio')[0].play();
 document.getElementsByTagName('audio')[0].pause();
});

C、无法同时播放多音频

    在 android 设备中,播放后一音频会打断前一音频,而不会同步播放,这个是目前系统资深决定的,我们只有采取优雅降权的方法让 android 选择不一样风格的音频前后切换播放而不是同时播放,达到与预期接近的音频效果。

D、不支持局部滚动

####  在 android 4.0 版本以下在 body(html)元素之外的元素  overflow:scroll  样式设置滚动条无效,这里有两种解决方案:
    1. 巧用布局直接设置样式滚动条在 body(html)上,其他元素“错觉滚动”。
    2. 利用 iscroll、自写 js 控制 translate、scrollTop 模拟

4、系统/硬件

A、怪异悬浮的表单

    在部分 android  机型中的输入框可能会出现如图怪异的多余的浮出表单,经过观察与测试发现只有 input:password 类型的输入框存在,那么我们只要使用 input:text 类型的输入框并通过样式-webkit-text-security: disc;  隐藏输入密码从而解决。

B、错误出现滚动条

    在游戏内嵌页中出现了不应该出现的滚动条,而且内容并没有超出内容区宽度,经过测试 overflow:hidden  无效,通过一系列尝试使用古老的    写法解决,多尝试一下不同的写法和属性会有不一样的惊喜哦!

C、链接打开系统浏览器

    在游戏内 webview 的部分 android 机型中可能会出现点击链接调用系统浏览器的情况,这是一个非常不好的体验。那么我们尝试给这个元素添加  target="_blank"'  属性有可能解决,如果还不能解决那么需要修改 IOS 或 android 原生系统函数了。

D、Flex box  不兼容

    在游戏内嵌 webview 中碰到 Flex box 布局不兼容的情况,图中所示下面部分的导航错位了,虽然之前有仔细查看过 Flex box 的兼容性,但是在游戏内嵌页中无法确定其调用的系统浏览器版本及兼容,所以导致错误,所以我们写完整历史版本呢的 3 种 Flex box  解决。那么我们思考在写页面过程中还是本着保守稳定的方式书写样式可以减少不不要的麻烦。

原文:https://www.cnblogs.com/hellofangfang/p/9834731.html

0

评论区