H5开发:横屏适配
平常我们做过的需求里,主要是以竖屏式为主,而横屏式较少。对于竖屏式场景来说,大家的经验会比较丰富,因此,此次主要式探讨下横屏式场景下的一些需要注意的点,特别是怎样去做横屏适配。
对于 H5 横屏页面来说,要实现横屏的话,主要是解决两点:
1.无论用户手持方向如何,都需要保证屏幕横向显示。
2.由于屏幕分辨率的多样化,因此就算是横屏下也是需要进行横屏适配,保证页面在所有分辨率下都能够合理适配。
强制横屏显示
在横屏式 H5横屏页面中可以采取简单的措施进行处理,在页面内容按竖排方向显示时,开发者进行对用户提示其保持横屏体验。
但是,这对用户体验并不友好,因为这对于那些习惯于打开锁定为竖排方向功能的 iOS 平台用户,或者是关闭屏幕旋转功能的 Android 平台用户来说,他们需要多一个处理步骤——先关闭竖排方向锁定或是开启屏幕旋转,然后再横向手持设备。
因此,更好的做法是强制横屏显示,对屏幕 resize 事件进行监听,当判断为竖屏时将整个根容器进行逆时针 CSS3 旋转 90 度即可,代码如下所示。
// 利用 CSS3 旋转 对根容器逆时针旋转 90 度
var detectOrient = function() {
var width = document.documentElement.clientWidth,
height = document.documentElement.clientHeight,
$wrapper = document.getElementById("J_wrapper"),
style = ""; if( width >= height ){ // 横屏
style += "width:" + width + "px;"; // 注意旋转后的宽高切换
style += "height:" + height + "px;";
style += "-webkit-transform: rotate(0); transform: rotate(0);";
style += "-webkit-transform-origin: 0 0;";
style += "transform-origin: 0 0;";
}
else{ // 竖屏
style += "width:" + height + "px;";
style += "height:" + width + "px;";
style += "-webkit-transform: rotate(90deg); transform: rotate(90deg);";
// 注意旋转中点的处理
style += "-webkit-transform-origin: " + width / 2 + "px " + width / 2 + "px;";
style += "transform-origin: " + width / 2 + "px " + width / 2 + "px;";
}
$wrapper.style.cssText = style;
}
window.onresize = detectOrient;
detectOrient();
但是!这里有坑:如果你是采用 CreateJS 框架进行开发,那么就不能通过 CSS3 途径对包含 Canvas 的根容器进行旋转处理,因为旋转后会导致 Canvas 内的舞台元素的事件响应位置错乱。
解决办法是,换成利用 CreateJS 框架内的 Stage 的 rotation
属性对整个舞台旋转处理,代码如下:
if(self.isPortrait) { // 竖屏
// 舞台旋转
self.stage.x = self.canvasHeight; // 注意:x偏移相当于旋转中点处理,更简单
self.stage.rotation = 90;
// more...
}else { // 横屏
self.stage.x = 0;
self.stage.rotation = 0;
// more...
}
横屏适配处理
面对移动端多分辨率繁复冗杂的情况,我们对于一般情况下(也就是常见的竖屏式)页面适配处理可以说是烂熟于心,但是切换到横屏式场景下,同样的页面适配方法可以直接应用吗?会不会有什么问题呢?
下面分别从 DOM 和 Canvas 两方面去着手阐述如何做横屏适配处理。
解决 DOM 的横屏适配问题
在移动端,常见的移动端适配方案是 REM 方案,而为了减少 JS 与 CSS 的耦合,采用的是 VW + REM 方法。
因为页面适配的场景往往是竖屏式的,因此 VW + REM 方案表现得十分完美。但是遇上横屏式,它的缺点就暴露了出来。
由于响应断点的限制最大宽度处理,会导致页面两侧留白,当然这可以通过去掉最大宽度限制来解决。而真正的缺点在于,由于 vw 单位的特性,适配换算大小是根据屏幕宽度而言的,因此屏幕宽度越大导致容器、文字会越大,还可能导致 DOM 元素超出屏幕外,且文字过大并不是我们所想要的用户体验。
那么,换成 px 单位的固定布局如何?
但 px 单位的固定布局只适合于部分场景,对于需要内容全屏覆盖的场景,就可能存在这样的不理想的用户体验:绝对定位的元素之间空隙过大,导致布局不美观,又或者空隙过小,导致元素叠放被遮挡。
了解到,vw 单位的特点是适配换算大小时是根据屏幕宽度而定的,那么在强制横屏显示时,我们就可以同理转换为屏幕高度来而定,也就是 vw 单位替换成 vh 单位。这样进一步改良之后就会得到满意的适配效果。
具体实现可参考如下 SCSS 代码:
$vw_base: 375;
$vw_fontsize: 20;
html {
font-size: 20px; //不支持vw单位时,回退到px单位
font-size: ($vw_fontsize / $vw_base) * 100vw;
}
@media screen and (orientation: landscape) {
html {
font-size: 20px;
font-size: ($vw_fontsize / $vw_base) * 100vh;
}
}
解决 Canvas 的横屏适配问题
解决 Canvas 的横屏适配问题,目前在实际应用中有两种主流的方案:
- 通过做两套Canvas的方案。
- 采用缩放的手段进行适配的方案。
两套 Canvas 的方案的做法是,页面包含两个 Canvas 分别用于横竖屏时的相应显示,但是它们的数据是打通的。但是,该方案难免会有局限性,比较适合逻辑数据处理简单、且舞台元素少且居中的场景;
而缩放适配方案做法是,采用的最为常见的缩放手段——利用 CSS3 Transform 的 scale
属性,达到“一种设计尺寸适配多种分辨率屏幕”的目的。
对于常用的 CreateJS、PixiJS 框架来说,它们并没有配套的现成的横屏适配解决方案可以被采用的,尤其是我们如果采用原生 Javascript 去开发一个横屏项目的时候。
因此,下面我们来研究下如何解决 Canvas 横屏适配问题。
注意:下面文中示例代码都是在 CreateJS 框架的基础上进行编写的。
选用合适的缩放模式
横屏适配的核心是缩放,通过 scale
属性等手法将Canvas缩放至适合屏幕窗口大小。类似于 background-size
属性的表现,缩放适配也可以有很多种模式,或有裁剪或无裁剪,或根据长边缩放或根据短边缩放等等。根据一些常见的实际应用场景,有比较常用的五种缩放模式:Contain、Cover、Fill、Fixed-Width、Fixed-Height。根据游戏的不同的实际场景需求,我们可以选其中一种缩放模式进行适配。
下面,我们逐一解释以上五种缩放模式的定义、实现与其适用的场景。
a. Contain模式
Canvas可以类比为一张图,而图片的适配,我们可以联想到经常用以适配背景图片的属性 background-size
,其属性值包括 contain
、cover
。
借助 contain
的概念,我们把缩放的其中一种模式称为 Contain 模式。因为在这种模式下,舞台内容(gameArea)会保持宽高比进行缩放适配浏览器可视窗口(window),缩放至其能显示完整的舞台内容。
根据推导结论,简单代码实现如下:
// Contain模式核心原理函数
CONTAIN: function(){
var self = this;
self.radioX = self.radioY = Math.min((self.winWidth / self.designWidth) , (self.winHeight / self.designHeight));
self.canvasWidth = self.designWidth;
self.canvasHeight = self.designHeight;
}
可以看出,在 Contain 模式下,如果舞台内容宽高比与浏览器可视窗口的宽高比不相等时,舞台内容并没有填满整个浏览器可视窗口,此时就会出现上下或左右两侧会存在留空部分。
对于这种 Contain 模式,会比较适合舞台背景为纯色或者是渐变类型的H5轻互动,舞台内容与窗口的紧邻处得以自然过渡衔接,不会突兀。
b. Cover模式
同样地,借助 cover
的概念把其中一种模式称为 Cover 模式。在这种模式下,舞台内容(gameArea)会保持宽高比进行缩放适配浏览器可视窗口(window),缩放至舞台内容填满窗口。
根据推导结论,简单代码实现如下:
// Cover模式核心原理函数
COVER: function(){
var self = this;
self.radioX = self.radioY = Math.max((self.winWidth / self.designWidth) , (self.winHeight / self.designHeight));
self.canvasWidth = self.designWidth;
self.canvasHeight = self.designHeight;
}
在 Cover 模式下,如果舞台内容宽高比与浏览器可视窗口的宽高比不相等时,由于舞台内容需要填满整个浏览器可视窗口,此时就会出现上下或者左右两侧被裁剪的情况。
那么,如果能保证游戏场景内的重点显示内容全部显示,被裁剪内容无关紧要时,那么这种 H5 轻互动类型就可以考虑采用 Cover 模式。
怎么做到保证想要重点显示的内容可以不被裁剪呢?这时要谈到一个“安全区域”的概念,指的是绝对不会被裁剪的内容区域,它应该是由最小的屏幕可视窗口(目前应该是 iPhone 4 )与最大的屏幕可视窗口(目前应该是 iPhone 7 Plus)叠加后得出的重叠区域.
c. Fill模式
Fill 模式,可以类比为 backgrouns-size: 100% 100%
的表现,在这种模式下,不会保持宽高比,舞台内容(gameArea)的宽高分别按照舞台内容与浏览器可视窗口(window)的宽度比与高度比进行缩放,缩放至舞台内容拉伸铺满窗口。
根据推导结论,简单代码实现如下:
// Fill模式核心原理函数
FILL: function(){
var self = this;
self.radioX = (self.winWidth / self.stageWidth);
self.radioY = (self.winHeight / self.stageHeight);
self.canvasWidth = self.designWidth;
self.canvasHeight = self.designHeight;
}
这种模式下既不会留空,也不会被裁剪,但是在舞台内容宽高比与浏览器可视窗口的宽高比不相等时,显示的内容会有一定程度的拉伸形变。
这种暴力的处理方式虽然免去了留空和裁剪的烦恼,但是会存在拉伸形变,这就得看是否能够被接受了。
d. Fixed-Width模式
区别于图像,Canvas 是可以进行动态绘制大小的。所以,我们可以考虑根据屏幕窗口大小变化来动态绘制 Canvas。
从保持舞台横向内容不变的角度考虑,我们提出这样的模式:舞台内容(gameArea)等比进行缩放至与浏览器可视窗口的一致的宽度大小,而舞台的高度(Canvas高度)进行重新绘制其高度为浏览器可视窗口的高度,称之为 Fixed-Width 模式。
根据推导结论,简单代码实现如下:
// Fixed-Width模式核心原理函数
FIXED_WIDTH: function(){
var self = this;
self.radioX = self.radioY = self.winWidth / self.designWidth;
self.canvasWidth = self.designWidth;
self.canvasHeight = self.winHeight / self.radioY;
}
在 Fixed-Width 模式下,无论在什么分辨率下,舞台横向内容保持不变,而纵向高度则会动态裁补,这就会比较适用于那些场戏场景可以纵向拓展的 H5 轻互动类型。
e. Fixed-Height模式
说完 Fixed-Width 模式,换个角度考虑便得出 Fixed-Height 模式,舞台内容(gameArea)等比进行缩放至与浏览器可视窗口的一致的高度大小,而舞台的宽度(Canvas宽度)进行重新绘制其宽度为浏览器可视窗口的宽度。
根据推导结论,简单代码实现如下:
// Fixed-Height模式核心原理函数
FIXED_HEIGHT: function(){
var self = this;
self.radioX = self.radioY= self.winHeight / self.designHeight;
self.canvasWidth = self.winWidth / self.radioX;
self.canvasHeight = self.designHeight;
}
与 Fixed-Width 模式相反,Fixed-Height 模式下,舞台纵向内容保持不变,而横向宽度则会动态裁补。对于这种模式的应用场景应该会比较广泛,譬如常见的跑酷游戏类型H5轻互动。
加入重定位和重绘制策略
综合以上五种缩放模式,我们可以看到对于 Cover、Fixed-Width、Fixed-Height 模式而言,有存在被裁剪的可能性。特别是 Fixed-Height 模式,对于横屏游戏来说这是比较常用的模式,但是在屏幕较小的时候难免会被裁剪,而且我们是不希望贴边元素被裁剪掉的,譬如位于右上角的音乐图标。而对于 Fixed-Width、Fixed—Height 模式,它们还存在舞台区域需要补充绘制的情况,因此对某些舞台元素来说需要重新设定其渲染大小。
所以,除了基本的缩放适配模式实现之外,为了解决贴边元素不被裁剪以及对一些舞台元素重绘制的需求,我们还需要加入两个策略:重定位和重绘制。
a. 重定位
贴边元素重定位策略的实现原理很简单,对需要重新定位的元素对象额外设置 top
、left
、right
、bottom
的自定义属性(当然你可以命名为其他属性名),这样我们就可以在适配的时候根据这些自定义属性以及实际显示的 Canvas 大小进行重新计算位置。
为了保证性能,下面是策略里需要注意的地方:
- 在舞台里,并不是所有游戏元素都是需要被重定位的,因此我们只需要创建一个数组记录需要被重定位的元素。
- 适当控制重定位次数,我们不需要在每一帧 tick 绘制的时候都进行重定位,只需要在 Canvas 大小改变的时候进行处理。
以下是重定位策略相关的代码:
// halfCutHeight、halfCutWidth是根据适配后的实际Canvas大小计算出来的相对距离
_setSize: function(){
// ...
if(self.isPortrait) {
// ...
self.halfCutWidth = (self.canvasWidth * self.radioY - this.winWidth ) / 2 / self.radioY;
self.halfCutHeight = (self.canvasHeight * self.radioX - this.winHeight) / 2 / self.radioX;
}else {
// ...
self.halfCutWidth = (self.canvasWidth * self.radioX - this.winWidth ) / 2 / self.radioX;
self.halfCutHeight = (self.canvasHeight * self.radioY - this.winHeight) / 2 / self.radioY;
}
// ...
},
// 贴边元素重定位核心处理函数
_adjustPosition: function(item){
var self = this;
item && self.adjustPositionArr.push(item);
self.adjustPositionArr.map(function(item, index, arr){
(typeof item.top == "number") && (item.y = item.top + self.halfCutHeight >= 0 ? self.halfCutHeight : 0);
(typeof item.left == "number") && (item.x = item.left + self.halfCutWidth >= 0 ? self.halfCutWidth : 0);
(typeof item.bottom == "number") && (item.y = self.canvasHeight - item.getBounds().height - item.bottom + self.halfCutHeight >= 0 ? self.halfCutHeight : 0);
(typeof item.right == "number") && (item.x = self.canvasWidth - item.getBounds().width - item.right - self.halfCutWidth);
});
},
// 暴露方法:提供给开发者记录需要重定位的贴边元素
adjustPosition: function(item){
var self = this;
self._adjustPosition(item);
}
b. 重绘制
对于一些以舞台区域(gameArea)作为其大小设置的参考标准的元素,在适配时遇到需要补全绘制区域时,舞台区域大小发生变化,相应地,该元素就需要进行重新绘制,这就是重绘制策略的存在意义。
同样地,为了保证性能,重绘制策略也是同样需要保证:
- 创建对应的数组记录全显图形对象。
- 不在每一帧 tick 时进行重绘制,只在适配的时候重绘制。
以下是重绘制策略的相关代码:
// 全显图形重绘制核心处理函数
_adjustFullSize: function(item){
var self = this;
item && self.adjustFullSizeArr.push(item);
self.adjustFullSizeArr.map(function(item, index, arr){
item.drawRect(0, 0, self.canvasWidth, self.canvasHeight);
});
},
// 暴露方法:提供给开发者记录需要重绘制的全显图形
adjustPosition: function(item){
var self = this;
self._adjustPosition(item);
}
至此,Canvas 横屏适配问题才得以完全解决。
内容篇幅较长,简单总结下,一个简单的解决 Canvas 横屏适配问题的方案至少需要包括两点实现:
选用合适的缩放模式。
方案内置五种缩放模式,在实际应用中根据场景不同而采用不同的缩放进行适配。加入重定位和重绘制策略。
为了保证贴边元素不被裁剪以及舞台元素动态渲染大小以适应舞台区域的动态变化。
移动端判断手机横竖屏状态
在做移动端开发的时候,在HTML页面head中加入如下代码
<meta name="viewport" content="width=device-width, user-scalable=no, initial-scale=1"/>
设置屏幕宽度为设备宽度,禁止用户手动调整缩放,页面初始缩放程度为1
CSS判断横屏还是竖屏
1.写在同一个css文件中
@media screen and (orientation: portrait) {
/*竖屏 css*/
}
@media screen and (orientation: landscape) {
/*横屏 css*/
}
2.分开写在2个CSS中
<!--竖屏-->
<link rel="stylesheet" media="all and (orientation:portrait)" href="portrait.css"> <!--横屏-->
<link rel="stylesheet" media="all and (orientation:landscape)" href="landscape.css">
js判断横屏还是竖屏
//判断手机横竖屏状态:
window.addEventListener("onorientationchange" in window ? "orientationchange" : "resize", function() {
if (window.orientation === 180 || window.orientation === 0) {
//alert('竖屏状态!');
}
if (window.orientation === 90 || window.orientation === -90 ){
//alert('横屏状态!');
}
}, false);
//移动端的浏览器一般都支持window.orientation这个参数,通过这个参数可以判断出手机是处在横屏还是竖屏状态。
H5开发:横屏适配的更多相关文章
- H5开发中的问题总结
最近公司做了一个出行日记的项目,里面的页面十多页,天天加班,做到吐血.总体来说,写页面的时候虽然是十多个页面,其实难度还是在每个页面的特效上.公司是易到用车,出行日记的页面在APP里有生成入口,有兴趣 ...
- 移动h5开发资源整理
这2年来,移动h5开发逐渐成为一种主流,也不断趋向于成熟.硬件和浏览器的不断更新,曾经的浏览器兼容也不再是开发者的噩梦. 接触h5开发一年多,从最初的新手到现在,陆陆续续遇到过很多坑.这里把想到的一些 ...
- 用H5开发微信还是开发APP?
用H5开发微信还是开发APP? 随着技术的飞速发展,HTML第五版技术标准的更新,在移动端,由于其相对较低的开发成本及强大的跨平台运行能力,越来越多的信息型产品也开始选择这样轻量级的H5页面进行快速迭 ...
- 移动H5开发入门教程:12点webAPP前端开发经验
如果你是一名移动H5前端开发人员,25学堂的小编认为下面的分享的12点webAPP前端开发经验是你必须掌握的基础知识点.算是一篇移动H5开发入门教程吧! 1. viewport:也就是可视区域.对于桌 ...
- 浅谈无线h5开发
最近一直在做h5的项目,对h5开发有了自己的理解.首先h5开发并不是指的html5的开发,而是指无线端的web开发,至于为什么叫h5开发,我觉得一方面是因为html5近几年还是挺受关注,另一方面h5在 ...
- 带你使用h5开发移动端小游戏
带你使用h5开发移动端小游戏 在JY1.x版本中,你要做一个pc端的小游戏,会非常的简单,包括说,你要在低版本的浏览器IE8中,也不会出现明显的卡顿现象,你只需要关心游戏的逻辑就行了,比较适合逻辑较为 ...
- 关于APP,原生和H5开发技术的争论
App的开发技术,目前流行的两种方式,原生和Html5.原生分了安卓平台和ios平台(还有小众的黑莓.死去的塞班就不说了),H5就是Html5. 目前争论不休的问题,在早先前争论CS,BS架构的软件系 ...
- 移动H5开发入门知识,CSS的单位汇总与用法
说到css的单位,大家应该首先想到的是px,也就是像素,我们在网页布局中一般都是用px,但是近年来自适应网页布局越来越多,em和百分比也经常用到了.然后随着手机的流行,web app和hybrid a ...
- H5开发中的故障
本篇博文会不断的收录我在做H5页面时遇到的问题以及解决方案,当然有的问题,我也没有遇到好的解决方案,所以如果你有解决的办法,请务必不吝赐教! H5开发中的故障 微信APP返回按钮不刷新页面 ...
随机推荐
- 必问的Java集合框架面试题
Arraylist 与 LinkedList 异同 是否保证线程安全: ArrayList 和 LinkedList 都是不同步的,也就是不保证线程安全: 底层数据结构: Arraylist 底层使用 ...
- Writing custom protocol for nanomsg
http://vitiy.info/writing-custom-protocol-for-nanomsg/ nanomsg is next version of ZeroMQ lib, provid ...
- [原]Django-issue(1)---postgresql数据库连接密码错误
环境: Django==1.9.13 psycopg2==2.7.5 Python 3.6.5 postgresql 1.18.1 配置django的时候出现问题 检查setting,问题点:由于安装 ...
- vmare 往 virtualbox迁移
vmare实在太卡了.抓狂. 于是想迁移到virtualbox观察下. 谷歌了下方案,发现众说纷纭. 有操作超级复杂的,比如:http://stackoverflow.com/questions/69 ...
- java中的动态代理Proxy
动态代理是java语言的一个神奇的地方,不是很好理解,下面来看看关键的地方. InvocationHandler 是一个接口,官方文档解释说,每个代理的实例都有一个与之关联的 InvocationHa ...
- 【C++ 实验5 类和对象】
1. #include <iostream> #include <vector> #include <string> using namespace std; // ...
- Java编程基础篇第二章
关键字 概述:被Java语言赋予特定含义的单词. 特点:组成关键字的字母全部为小写字母. 标识符 概述:给类,接口,包,方法,常量起名字时的字符序列 组成规则:英文大小写字母,数字,$和— 命名规则. ...
- xilink 烧写flash
no 右键
- MySQL 连接数相关参数设置
当我们的数据库服务器遇到如下的错误:ERROR: Too many connections 一种可能性就是我们的压力确实很大,需要增加服务器硬件资源或者增加数据库服务器.但是大多数情况下是我们的连接数 ...
- ArcGIS API for JavaScript经典例子
地址为本地 1.绘制图形: http://localhost/arcgis_js_api/sdk/sandbox/sandbox.html?sample=toolbar_draw 2.双击编辑图形 h ...