何谓 viewport
根据 w3c 对于 CSS Device Adaptation 的解释,他大致上是在做这些事情,
1.根据装置的显示区域来展示文件
2.放大或缩小文件,来符合或设定上给予装置的可视区域
3.允许设定或初始化缩放的级别,或是其他规则
实际上,他是跟着显示装置在运作的,所以他的所有设定都跟显示装置有关,而这里的显示装置跟 Media Query 里面的 screen, projection, print, tv, tty, aural, handheld, embossed, braille 这几样东西是不太一样的(话说,我上次的文章似乎没提到这一块)。 虽然,Media Query 上述那几项规则设定,很明显都与其显示的装置有关,但是,他们的目的不同,操作的方式也不一样。
欧,你说 print 为什么也算显示装置,因为对打印机而言,印出来也是显示的一种(但是印出来有 @page 可以用喔!
@viewport vs. @media
- 举个例子来说,
- @viewport {
- width: device-width;
- initial-scale: 1;
- }
- @media screen and (min-width: 768px) {
- h1 {
- font-size: 16px;
- }
- }
- @media screen and (max-width: 768px) {
- h1 {
- font-size: 26px;
- }
- }
上面那个 @viewport 与 @media 初始化设定中是无关的。这样理解吗?什么?不行?那先看一下步骤,
1.viewport 规则套用,宽度设定为 device-width,且初始缩放数值为 1
2.@media 套用于 screen 与其条件 (min-width: 768px)
3.@media 套用于 screen 与其条件 (max-width: 768px)
对于我们的视角来说,device-width 可以解释为我们所看见的装置的宽度,而 768px 这件事情,则是告诉该显示的本文遇到这个条件时,应该要显示的结果。所以,条件会显示的结果是看 @media 来决定。举个例子来解释这个例子,
1.假设我们的装置 device-width 是 800px
2.那么第一个条件会成立
疑?无法理解?
- @viewport {
- width: 768px;
- }
- @media only screen and (width: 768px) {
- h1 {
- font-size: 16px;
- }
- }
- @media only screen and (width: 1200px) {
- h1 {
- font-size: 26px;
- }
- }
那这样应该相当好理解了吧!这就是 @viewport 在做的事情,另外,由于 @viewport 有绝对优先权,所以他会比 @media 还要早执行,所以就算这样写,也是会成立的,
- @media only screen and (width: 768px) {
- h1 {
- font-size: 16px;
- }
- }
- @media only screen and (width: 1200px) {
- h1 {
- font-size: 26px;
- }
- }
- @viewport {
- width: 768px;
- }
实例
首先举个例子,iphone6s的像素分辨率为1334x750,326ppi,4.7英寸。如图
下面我们一个一个来解释。
像素密度
A.定义
326ppi代表的就是像素密度(pixels per inch),指的是每英寸的长度上排列的像素点数量。
a.屏幕分辨率
首先1334x750,代表的是屏幕分辨率,也就是说iPhone6s的屏幕是由纵向像素1334*横向像素750个像素点组成。单位pixel。
b.物理尺寸
如图可以知道,4.7英寸也就指的是iPhone6s屏幕对角线的物理长度。1英寸是一个固定长度,等于2.54厘米。
B.像素密度的计算
像素密度=对角线分辨率/屏幕尺寸。如下
计算:勾股定理算出对角线的分辨率:√(1334²+750²)
对角线分辨率除以屏幕尺寸:√(1334²+750²)/4.7≈440dpi。
C.直观感受理解像素密度
像素密度越高,代表屏幕显示效果越精细。Retina屏比普通屏清晰很多,就是因为Retina屏幕像素密度高。最明显的就是iPhone 3GS和iPhone 4的区别。屏幕尺寸一样因为屏幕分辨率相差一倍所以屏幕像素密度也相差一倍。显示效果如下
如图看出iPhone4显示要比iPhone 3gs清晰的多。根本原因是因为iPhone3gs的1个像素点的大小相对于iPhone4来说等于它屏幕的4个(2x2)像素大小。如下图。
注意注意注意:这里的1像素指的是物理设备的1个像素点。也就是说不同像素密度的物理设备的一个像素点的大小是不一样的。也就意味着对于物理设备屏幕的单个像素点的大小是不是一个固定的大小。用稍微专业一点的话来说就是,单个的设备像素的大小是不定的。
重要概念理解
上面我们讨论的是关于物理设备的像素密度(再次强调:这里的像素就是物理像素或者说是设备像素,对应于像素密度不同的设备具体像素点的大小都是不一样大的。像素密度越大,像素点越小,显示的细节就越多越丰富。反之则情况刚好相反)。那么接下来我们要说的几个重要概念是什么呢?我们先从css像素与设备像素开始讨论。
A.css像素和设备像素
首先我们需要明确一点:在我们编写CSS的时候的px和设备自身的px是没有任何关系的。
css pixel:浏览器使用的抽象单位,主要用来在网页上绘制内容。
device pixel:显示屏幕的最小物理单位,每个dp包含自己的颜色、亮度。
既然css pixel与device pixel没有任何关系,那么我们在编写页面时怎么设置多宽(用多少的css像素)的网页才能适配不同的设备(才能覆盖设备的像素)呢???为了解决这个问题,我们再引入一个重要的概念viewport。
B.viewport
viewport翻译过来就是视口的意思。viewport有两种,一种是layout viewport,还有一种是visual viewport。这两种viewport的大小都是以css像素进行绘制的。
1.layout viewport:把layout viewport想象成为不会变更大小和形状的大图。现在想像你有一个小一些的框架,你通过它来看这张大图。(可以理解为「管中窥豹」)这个小框架的周围被不透明的材料所环绕,这掩盖了你所有的视线,只留这张大图的一部分给你。你通过这个框架所能看到的大图的部分就是visual viewport。当你保持框架(缩小)来看整个图片的时候,你可以不用管大图,或者你可以靠近一些(放大)只看局部。你也可以改变框架的方向,但是大图(layout viewport)的大小和形状永远不会变。
计算:document.documentElement.clientWidth/Height
2.visual viewport:是页面当前显示在屏幕上的部分。用户可以通过滚动来改变他所看到的页面的部分,或者通过缩放来改变visual viewport的大小。
计算:window.innerWidth/Height
当你缩小或者放大页面时,visual viewport的大小会发生变化。当你放大页面,能看到页面的内容也就越少,当你缩小页面能看到的页面内容也就越多。根本上是因为当你缩小或者放大网页时,网页上的css像素就变小或者变大了,但是layout viewport的大小是不变的。也就意味着,当css像素变大时,layout viewport中能容纳的单位css像素也就变少了,因此visual viewport的大小就变小了。反之也是一个道理。这个过程中,layout viewport并不会发生任何变化(当然,在窗口的大小发生变化的时候,layout viewport的大小也会随之变化,窗口越大能够容纳的CSS像素越多,layout viewport也就越大,反之亦然)。如图
1:1.3
因此可以把layout viewport想象成为一个虚拟的"窗口"。窗口可大于或小于设备的可视区域(也就是visual viewport),一般设备默认layout viewport大于visual viewport。这样不会破坏没有针对设备的浏览器优化的网页的布局,用户可以通过平移和缩放来看网页的其他部分。
部分机型默认layout viewport:
你也可以通过上面给出的document.documentElement.clientWidth/Height在chrome控制台中进行测试layout viewport的大小。
C.小总结1
根据以上的讨论我们知道了:
我们写页面时是在layout viewport上进行的布局,用的是css像素,可以由document.documentElement.clientWidth/Height计算。(与程序员相关)
各种设备的屏幕分辨率表示的像素是设备像素可以由screen.width/height计算。(与硬件产品相关)
我们观看页面时是通过visual viewport进行观看的,用的也是css像素,可以由window.innerWidth/Height来计算。(与用户行为相关)
设置layout viewport大小
通过上面的讨论,我们知道,屏幕的分辨率的设备像素与我们编写页面时使用的CSS像素并没有什么直接关系。而且
问题1.从大的方面来看:首先,我们编写的CSS是在layout viewport中绘制。其次,从下图可以看出,设备分辨率和layout viewport也没有什么直接关系
问题2.从小的方面来看:首先, CSS像素在不放大和缩小的情况下,在任何设备中的1个CSS像素都是一样大的。其次,不同像素密度的设备,设备屏幕(与像素密度有关)中的1个设备像素的大小是各不相同的。可以知道,在没有进行任何设置的情况下,单位设备像素与单位css像素也扯不上什么关系。
设置viewport
如果我们想要自己写出来的页面在各个设备下都工作良好,就会出现困难?因此为了让他们之间扯上关系,我们便需要做一些设置。像下面这样
- "viewport" content="width=device-width,initial-scale=1,maximum-scale=1,user-scalable=no">
这样设置了之后,我们的layout viewport的大小也就与设备大小相等了(高度上也是相等的)。如图。既然它们相等那么可以推理得出此时的单位设备像素和单位CSS像素在缩放比例为1的情况下,也就是相等的了。
这时候我们显示的页面的屏幕和我们布局页面的大小是一样的,那么在进行布局的时候,也就容易了很多。
小总结2:
知道了这样设置后,对于我们布局来说问题也就不大了。
1.我们在layout viewport上布局,这样设置后,layout viewport的大小等于设备屏幕的大小
2.不同像素密度的设备的单位设备像素大小的变化也就会同步引起单位CSS像素大小的变化。也就是说,我们在用CSS编写网页的时候,在不同像素密度的设备中的css像素大小是不一样的是变化的。(这里一定要理解)。
但是,这里还有一个问题,iphone6 plus 414这个宽度是从哪里来的呢?
像素密度与转换系数
Android和IOS都会通过转换系数让控件适应屏幕的尺寸。而这个转换系数就和最开始讲的像素密度PPI有关。不同的PPI转化系数不同。如图
由图可以知道iphone6 plus的屏幕分辨率为1242*2208,转化系数为3。
1242/3=414。
414便由此而得,这是设备自身进行的转化,而我们设置的layout viewport的大小也就等于设备自身转换后的大小。