商务服务
js实现手机横竖屏事件
2025-01-03 03:39

在做H5项目时,需要在横竖屏变化时,做一些处理。毫无疑问,需要使用orientationchange来监听横竖屏的变化。

js实现手机横竖屏事件

https://github.com/zuopf769/notebook/tree/master/fe/js%E5%AE%9E%E7%8E%B0%E6%89%8B%E6%9C%BA%E6%A8%AA%E7%AB%96%E5%B1%8F%E4%BA%8B%E4%BB%B6

方案1 orientationchange事件

 

orientationchange事件在低端的adroid机器上存在兼容性问题,可能不会触发

方案2 resize配合(window.inner/outerWidth, window.inner/outerHeight)

 
缺点

只要的变化,就会不断触发触发事件。可以使用来优化一下 如果有多个地方需要监听横竖屏,就需要注册多个。

能不能通过订阅与发布模式来改进一下,只注册一个resize负责监听横竖屏变化,只要横竖发生变化就发布通知订阅的对象。其他需要监听横竖屏的地方只需订阅一下即可。

关键代码如下

 
代码

完整代码

改进方案1

基于CSS3@media媒体查询检测来实现

通过window.innerWidth > window.innerHeight来实现的是一种伪检测,有点不可靠。 可不可以通过浏览器来实现检测?如基于CSS3@media媒体查询来实现。

实现思路
  • 创建包含标识横竖屏状态的特定css样式
  • 通过JS向页面中注入CSS代码
  • resize回调函数中获取横竖屏的状态

这里选择的节点font-family作为检测样式属性。 理由如下

  1. 选择主要为了避免reflow和repaint
  2. 选择font-family样式,主要是因为font-family有如下特性
        优先使用排在前面的字体。
        如果找不到该种字体,或者该种字体不包括所要渲染的文字,则使用下一种字体。
        如果所列出的字体,都无法满足需要,则让操作系统自行决定使用哪种字体。 这样我们就可以指定特定标识来标识横竖屏的状态,不过需要将指定的标识放置在其他字体的前面,这样就不会引起hmtl字体的变化。
 
代码

完整代码

改进方案2

可以再改进一下,在支持orientationchange时,就使用原生的orientationchange,不支持则使用上面方案

 
完整代码

代码

改进方案3

目前,上述几种方案都是通过自定制的订阅与发布事件模式来实现的。这里可以基于浏览器的事件机制,来模拟orientationchange。即对orientationchange的不兼容进行修复。

 
代码

完整代码

方案3 matchMedia

css3为我们提供了很强大的media query,而我们时常需要在js中动态的知道什么时候某个状态满足了。CSS Object Model(CSSOM)Views规范增加了对Javascript操作CSS media query的原生支持,它在window对象下增加了matchMedia()方法。

MediaQueryList对象

你可以传入一个CSS media query然后返回一个MediaQueryList对象。这个对象包括两个属性:matches,布尔值数据,表示CSS media query是否与当前的显示状态匹配;media对应传入的参数字符串。如下

 
MediaQueryList对象监听器
 

当视图状态发生改变时,监听器对应的函数就会执行,而对应的MediaQueryList对象也会传入。用这个方式吗,你可以让你的Javascript可以很快地响应布局变化,并且不需要用轮询的方式。另外关于media query的实现原理一直不太清楚,什么时候media query就生效的,比如说转屏,是否是只要当前屏幕width或height发生改变时就去查询media query估计就待看看webkit源码才能清楚了。

原生的matchMedia有些问题; GMU在遵照CSS Object Model(CSSOM)Views规范,在zepto基础上扩展实现了$.matchMedia方法,该方法返回一个对象,该对象包含matches(是否match query,当前查询query, addListener和removeListener,调用方式与原生window.matchMedia一致。

具体思路

(1)为页面添加检测元素

css media query主要还是在style上起作用,故在页面创建一个div,作为media query作用的对象。

 
(2)为检测元素添加transition样式及media query样式

当query条件满足时,去动态修改transition作用的属性,如(width),则可触发transitionEnd事件,这样则相当于可以监测到media query。

 

注意下这里的query:一开始加载页面成功后为

 

一开始检测元素的宽度为0,mediaQuery加载后就为1了,触发了transition动画,当有竖屏转为横屏时,mediaQuery条件就不符合了,检测元素的宽度又变回到了0,所以又会触发transition动画

注册transitionEnd事件

 

封装addListener及removeListener接口

主要记录在闭包中的listeners数组件,添加和删除回调函数即可

 

用$.mediaQuery实现转屏ortchange事件

实现转屏,只需将query传入检测转屏的query即可。这里在实现时,最开始遇了一点问题。最开始query值为"screen and (orientation: portrait)",可在某个三星的机器上测试,居然键盘出来会改变视口大小,即认为是orientation改变了。后来经测试后query换成了"screen and (width: " + window.innerWidth + "px)",即检测设备第一次打开时的width,若转屏后width必然不满足,使得页面上的检测元素width未受media query的css影响而改变而触发transitionEnd。具体代码如下

 

使用方法

 

总结

最新文章
苹果手机充不进电是什么原因手机充不上电是什么原因「苹果手机充不进电是什么原因」
随着智能手机的普及,苹果手机凭借其卓越的性能和独特的生态系统,赢得了众多用户的青睐。然而,在使用过程中,有时会遇到手机无
高考考场分布图来了!还要注意这些……手机带「高考考场分布图来了!还要注意这些……」
山东省2023年夏季高考和高中学业水平等级考试(以下统称“夏季高考”)将于6月7—10日举行,考试临近,有哪些事需要注意?让我们
荣耀X50i+和荣耀X50i配置对比详解,哪款更值得入手?手机配置对比「荣耀X50i+和荣耀X50i配置对比详解,哪款更值得入手?」
今年各大国产千元手机都卷起来了,有越来越多的越级配置来到了千元手机市场,吸引了许多网友的关注。继荣耀X50i之后,荣耀官方在
云电脑教程:怎么用云电脑手机玩游戏?手机云电脑「云电脑教程:怎么用云电脑手机玩游戏?」
游戏已经成为刚需,不少游戏都开始在手机上玩,即使没出手游也是可以同步进行,和小极来了解手机玩游戏教程。首先,在手机上下载
至尊贵族奢华 诺基亚8800A奢华6300元诺基亚8800手机「至尊贵族奢华 诺基亚8800A奢华6300元」
2011年4月12日, 诺基亚 8800A (行货)在给出了6300元的报价,这个 价格 相比前一段时间大降200元。该机为行货版本,其配件包括
手机卡丢了怎么办电信卡不用了呢怎么注销电信手机卡怎么注销「手机卡丢了怎么办电信卡不用了呢怎么注销」
大家好,关于手机卡丢了怎么办电信卡不用了呢怎么注销很多朋友都还不太明白,今天小编就来为大家分享关于电信卡丢了怎么销号的知
华为matex使用的是什么系统?华为手机是什么系统「华为matex使用的是什么系统?」
大家都知道,在华为matex未发布之前,不少人都以为它会采用华为自研的鸿蒙系统,但这种说法被否定了,首款鸿蒙手机可能是明年3月
小米6手机通话录音在哪里?小米手机通话录音在哪里找「小米6手机通话录音在哪里?」
小米手机录音功能具体操作步骤:1、按电话图标进入电话界面2、按菜单键打开菜单,找到右下角设置图标3、点击设置图标,进入设置
华为手机adb驱动工具华为手机adb驱动工具v1.3 官方华为手机驱动「华为手机adb驱动工具华为手机adb驱动工具v1.3 官方」
华为手机adb驱动工具专门用于华为安卓手机,安装本驱动后才能连接电脑与手机进行各种操作,可以解锁手机。软件操作简单,打开后